Compare commits

..

No commits in common. 'main' and 'dev' have entirely different histories.
main ... dev

@ -4,7 +4,9 @@
<component name="GradleSettings">
<option name="linkedExternalProjectsSettings">
<GradleProjectSettings>
<option name="testRunner" value="CHOOSE_PER_TEST" />
<option name="externalProjectPath" value="$PROJECT_DIR$" />
<option name="gradleJvm" value="#GRADLE_LOCAL_JAVA_HOME" />
<option name="modules">
<set>
<option value="$PROJECT_DIR$" />
@ -12,7 +14,7 @@
<option value="$PROJECT_DIR$/basemvplib" />
</set>
</option>
<option name="resolveModulePerSourceSet" value="false" />
<option name="resolveExternalAnnotations" value="false" />
</GradleProjectSettings>
</option>
</component>

@ -1,6 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="VcsDirectoryMappings">
<mapping directory="$PROJECT_DIR$" vcs="Git" />
<mapping directory="" vcs="Git" />
</component>
</project>

@ -1,15 +1,12 @@
// 声明包名
package com.monke.monkeybook;
import android.app.Application;// 导入Application类代表整个应用程序
import android.test.ApplicationTestCase;// 导入ApplicationTestCase类它提供了应用测试的基础
import android.app.Application;
import android.test.ApplicationTestCase;
/**
* <a href="http://d.android.com/tools/testing/testing_android.html">Testing Fundamentals</a>
* Android
*/
public class ApplicationTest extends ApplicationTestCase<Application> {
// 构造函数,参数为应用程序类的.class文件对象
public ApplicationTest() {
super(Application.class);
}

@ -1,7 +1,7 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
package="com.monke.monkeybook"> <!-- 声明包名 -->
package="com.monke.monkeybook">
<application
android:name=".MApplication"
@ -12,60 +12,39 @@
android:theme="@style/CAppTheme"
tools:replace="android:theme"
android:networkSecurityConfig="@xml/network_security_config">
<!-- 自定义应用程序类 -->
<!-- 允许备份 -->
<!-- 应用图标 -->
<!-- 应用名称 -->
<!-- 支持从右向左的布局 -->
<!-- 应用主题 -->
<!-- 指示工具覆盖主题 -->
<!-- 网络安全配置 -->
<!-- 欢迎界面 -->
<activity
android:name=".view.impl.WelcomeActivity"
android:screenOrientation="portrait">
<!-- 欢迎活动类 -->
<!-- 竖屏模式 -->
<intent-filter> <!-- 指定启动活动的意图过滤器 -->
<action android:name="android.intent.action.MAIN" /><!-- 主活动 -->
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" /><!-- 启动器 -->
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<!-- 主活动 -->
<activity
android:name=".view.impl.MainActivity"
android:launchMode="singleTask" <!-- 单一任务模式 -->
android:screenOrientation="portrait" /><!-- 竖屏模式 -->
<!-- 搜索活动 -->
android:launchMode="singleTask"
android:screenOrientation="portrait" />
<activity
android:name=".view.impl.SearchActivity"
android:configChanges="locale|keyboardHidden|orientation|screenSize"
android:screenOrientation="portrait"
android:theme="@style/CAppTransparentTheme"
android:windowSoftInputMode="stateHidden|adjustPan" /><!-- 软件输入法隐藏时更改视图 -->
android:windowSoftInputMode="stateHidden|adjustPan" />
<service android:name=".service.DownloadService" /><!-- 下载服务 -->
<service android:name=".service.DownloadService" />
<!-- 库活动 -->
<activity
android:name=".view.impl.LibraryActivity"
android:screenOrientation="portrait"
android:theme="@style/CAppTransparentTheme" />
<!-- 选择书籍活动 -->
<activity
android:name=".view.impl.ChoiceBookActivity"
android:screenOrientation="portrait" />
<!-- 书籍详情活动 -->
<activity
android:name=".view.impl.BookDetailActivity"
android:screenOrientation="portrait"
android:theme="@style/CAppTransparentTheme" />
<!-- 阅读书籍活动 -->
<activity
android:name=".view.impl.ReadBookActivity"
android:launchMode="singleTask"
@ -73,28 +52,24 @@
<intent-filter>
<action android:name="android.intent.action.VIEW"></action>
<category android:name="android.intent.category.DEFAULT"></category>
<data android:mimeType="text/plain"></data><!-- 处理文本类型数据 -->
<data android:mimeType="text/plain"></data>
</intent-filter>
</activity>
<!-- 导入书籍活动 -->
<activity
android:name=".view.impl.ImportBookActivity"
android:screenOrientation="portrait"
android:theme="@style/CAppTransparentTheme" />
<!-- 元数据 -->
<meta-data
android:name="UMENG_CHANNEL_VALUE"
android:value="${UMENG_CHANNEL_VALUE}" /><!-- 用于第三方渠道的元数据 -->
android:value="${UMENG_CHANNEL_VALUE}" />
</application>
<!-- 权限要求 -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /><!-- 读取外部存储权限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /><!-- 写入外部存储权限 -->
<uses-permission android:name="android.permission.INTERNET" /><!-- 访问互联网权限 -->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /><!-- 访问网络状态权限 -->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" /><!-- 访问Wi-Fi状态权限 -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" /><!-- 读取电话状态权限 -->
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
</manifest>

@ -1,13 +1,13 @@
//指定了当前类BookContentBeanDao所在的包路径即com.monke.monkeybook.dao
package com.monke.monkeybook.dao;
//导入必要的类包含GreenDAO、SQLite和Android数据库操作的相关类
import android.database.Cursor;// 导入Cursor类用于数据库查询结果
import android.database.sqlite.SQLiteStatement;// 导入SQLiteStatement类用于预编译SQL语句
import org.greenrobot.greendao.AbstractDao;// 导入AbstractDao类这是GreenDAO的基类用于实现DAO功能
import org.greenrobot.greendao.Property;// 导入Property类用于属性映射
import org.greenrobot.greendao.internal.DaoConfig;// 导入DaoConfig类包含DAO的配置
import org.greenrobot.greendao.database.Database;// 导入Database类代表数据库操作
import org.greenrobot.greendao.database.DatabaseStatement;// 导入DatabaseStatement类用于执行数据库语句
import android.database.Cursor;
import android.database.sqlite.SQLiteStatement;
import org.greenrobot.greendao.AbstractDao;
import org.greenrobot.greendao.Property;
import org.greenrobot.greendao.internal.DaoConfig;
import org.greenrobot.greendao.database.Database;
import org.greenrobot.greendao.database.DatabaseStatement;
// 导入实体类 BookContentBean
import com.monke.monkeybook.bean.BookContentBean;
@ -18,7 +18,7 @@ import com.monke.monkeybook.bean.BookContentBean;
//继承自GreenDAO的AbstractDao类BookContentBean为实体类String为主键类型
public class BookContentBeanDao extends AbstractDao<BookContentBean, String> {
//定义常量TABLENAME表示该表的名称
public static final String TABLENAME = "BOOK_CONTENT_BEAN";// 表名常量
public static final String TABLENAME = "BOOK_CONTENT_BEAN";
/**
* Properties of entity BookContentBean.<br/>
* Can be used for QueryBuilder and for referencing column names.
@ -26,10 +26,10 @@ public class BookContentBeanDao extends AbstractDao<BookContentBean, String> {
*/
// 定义实体类属性映射:对应数据库表字段名的属性
public static class Properties {
public final static Property DurChapterUrl = new Property(0, String.class, "durChapterUrl", true, "DUR_CHAPTER_URL");// 章节URL属性
public final static Property DurChapterIndex = new Property(1, int.class, "durChapterIndex", false, "DUR_CHAPTER_INDEX");// 章节索引属性
public final static Property DurCapterContent = new Property(2, String.class, "durCapterContent", false, "DUR_CAPTER_CONTENT");// 章节内容属性
public final static Property Tag = new Property(3, String.class, "tag", false, "TAG");// 标签属性
public final static Property DurChapterUrl = new Property(0, String.class, "durChapterUrl", true, "DUR_CHAPTER_URL");
public final static Property DurChapterIndex = new Property(1, int.class, "durChapterIndex", false, "DUR_CHAPTER_INDEX");
public final static Property DurCapterContent = new Property(2, String.class, "durCapterContent", false, "DUR_CAPTER_CONTENT");
public final static Property Tag = new Property(3, String.class, "tag", false, "TAG");
};
//构造函数传入DaoConfig用于初始化DAO
@ -51,10 +51,10 @@ public class BookContentBeanDao extends AbstractDao<BookContentBean, String> {
String constraint = ifNotExists? "IF NOT EXISTS ": "";
//使用SQL执行表创建语句
db.execSQL("CREATE TABLE " + constraint + "\"BOOK_CONTENT_BEAN\" (" + //
"\"DUR_CHAPTER_URL\" TEXT PRIMARY KEY NOT NULL ," + // 0: durChapterUrl, 主键列
"\"DUR_CHAPTER_INDEX\" INTEGER NOT NULL ," + // 1: durChapterIndex, 整数列
"\"DUR_CAPTER_CONTENT\" TEXT," + // 2: durCapterContent,章节内容列
"\"TAG\" TEXT);"); // 3: tag, 标签列
"\"DUR_CHAPTER_URL\" TEXT PRIMARY KEY NOT NULL ," + // 0: durChapterUrl
"\"DUR_CHAPTER_INDEX\" INTEGER NOT NULL ," + // 1: durChapterIndex
"\"DUR_CAPTER_CONTENT\" TEXT," + // 2: durCapterContent
"\"TAG\" TEXT);"); // 3: tag
}
/** Drops the underlying database table.
@ -62,73 +62,73 @@ public class BookContentBeanDao extends AbstractDao<BookContentBean, String> {
*/
public static void dropTable(Database db, boolean ifExists) {
//使用SQL删除表确保表存在时删除
String sql = "DROP TABLE " + (ifExists ? "IF EXISTS " : "") + "\"BOOK_CONTENT_BEAN\"";// 生成删除表的SQL语句
db.execSQL(sql);// 执行删除表操作
String sql = "DROP TABLE " + (ifExists ? "IF EXISTS " : "") + "\"BOOK_CONTENT_BEAN\"";
db.execSQL(sql);
}
@Override
protected final void bindValues(DatabaseStatement stmt, BookContentBean entity) {
//清除绑定的值,确保每次绑定时都是干净的
stmt.clearBindings();// 清除先前绑定的参数
stmt.clearBindings();
//绑定durChapterUrl字段到SQL语句的第一个位置
String durChapterUrl = entity.getDurChapterUrl();// 获取durChapterUrl属性值
String durChapterUrl = entity.getDurChapterUrl();
if (durChapterUrl != null) {
//将章节URL字符串绑定到SQL语句的第一个位置
stmt.bindString(1, durChapterUrl);// 将durChapterUrl绑定到SQL语句的第一个参数
stmt.bindString(1, durChapterUrl);
}
//绑定durChapterIndex字段到SQL语句的第二个位置
stmt.bindLong(2, entity.getDurChapterIndex());//将章节索引绑定到SQL语句的第二个位置
//绑定durCapterContent字段到SQL语句的第三个位置
String durCapterContent = entity.getDurCapterContent();// 获取durCapterContent属性值
String durCapterContent = entity.getDurCapterContent();
if (durCapterContent != null) {
//将章节内容绑定到SQL语句的第三个位置
stmt.bindString(3, durCapterContent);// 将durCapterContent绑定到SQL语句的第三个参数
stmt.bindString(3, durCapterContent);
}
//绑定tag字段到SQL语句的第四个位置
String tag = entity.getTag();// 获取tag属性值
String tag = entity.getTag();
if (tag != null) {
//将标签绑定到SQL语句的第四个位置
stmt.bindString(4, tag);// 将tag绑定到SQL语句的第四个参数
stmt.bindString(4, tag);
}
}
@Override
protected final void bindValues(SQLiteStatement stmt, BookContentBean entity) {
//清除绑定的值,确保每次绑定时都是干净的
stmt.clearBindings();// 清除先前绑定的参数
stmt.clearBindings();
//绑定durChapterUrl字段到SQL语句的第一个位置
String durChapterUrl = entity.getDurChapterUrl();// 获取durChapterUrl属性值
String durChapterUrl = entity.getDurChapterUrl();
if (durChapterUrl != null) {
//将章节URL字符串绑定到SQL语句的第一个位置
stmt.bindString(1, durChapterUrl);// 将durChapterUrl绑定到SQL语句的第一个参数
stmt.bindString(1, durChapterUrl);
}
//绑定durChapterIndex字段到SQL语句的第二个位置
stmt.bindLong(2, entity.getDurChapterIndex()); //将章节索引绑定到SQL语句的第二个位置
//绑定durCapterContent字段到SQL语句的第三个位置
String durCapterContent = entity.getDurCapterContent();// 获取durCapterContent属性值
String durCapterContent = entity.getDurCapterContent();
if (durCapterContent != null) {
//将章节内容绑定到SQL语句的第三个位置
stmt.bindString(3, durCapterContent);// 将durCapterContent绑定到SQL语句的第三个参数
stmt.bindString(3, durCapterContent);
}
//绑定tag字段到SQL语句的第四个位置
String tag = entity.getTag();// 获取tag属性值
String tag = entity.getTag();
if (tag != null) {
//将标签绑定到SQL语句的第四个位置
stmt.bindString(4, tag);// 将tag绑定到SQL语句的第四个参数
stmt.bindString(4, tag);
}
}
@Override
public String readKey(Cursor cursor, int offset) {
//从Cursor中读取主键(durChapterUrl)若为null则返回null
return cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0);// 根据游标读取主键值
return cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0);
}
@Override
@ -140,38 +140,38 @@ public class BookContentBeanDao extends AbstractDao<BookContentBean, String> {
cursor.isNull(offset + 2) ? null : cursor.getString(offset + 2), // durCapterContent
cursor.isNull(offset + 3) ? null : cursor.getString(offset + 3) // tag
);
return entity;// 返回创建的实体对象
return entity;
}
@Override
public void readEntity(Cursor cursor, BookContentBean entity, int offset) {
//从Cursor中读取值并设置到BookContentBean对象的相应属性中
entity.setDurChapterUrl(cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0));// 设置durChapterUrl属性
entity.setDurChapterIndex(cursor.getInt(offset + 1));// 设置durChapterIndex属性
entity.setDurCapterContent(cursor.isNull(offset + 2) ? null : cursor.getString(offset + 2));// 设置durCapterContent属性
entity.setTag(cursor.isNull(offset + 3) ? null : cursor.getString(offset + 3));// 设置tag属性
entity.setDurChapterUrl(cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0));
entity.setDurChapterIndex(cursor.getInt(offset + 1));
entity.setDurCapterContent(cursor.isNull(offset + 2) ? null : cursor.getString(offset + 2));
entity.setTag(cursor.isNull(offset + 3) ? null : cursor.getString(offset + 3));
}
@Override
protected final String updateKeyAfterInsert(BookContentBean entity, long rowId) {
//返回更新时需要的主键字段这里是durChapterUrl
return entity.getDurChapterUrl();// 返回用于更新的主键
return entity.getDurChapterUrl();
}
@Override
public String getKey(BookContentBean entity) {
//获取实体对象的主键值返回durChapterUrl
if(entity != null) {
return entity.getDurChapterUrl();// 返回实体的主键值
return entity.getDurChapterUrl();
} else {
return null;// 如果实体为空返回null
return null;
}
}
@Override
protected final boolean isEntityUpdateable() {
//指示实体是否可以被更新这里返回true表示支持更新
return true;// 表示该实体可被更新
return true;
}
}

@ -44,12 +44,12 @@ public class BookInfoBeanDao extends AbstractDao<BookInfoBean, String> {
//构造函数使用DaoConfig配置创建DAO对象
public BookInfoBeanDao(DaoConfig config) {
super(config);// 调用父类的构造函数传入DaoConfig配置
super(config);
}
//构造函数使用DaoConfig和DaoSession配置创建DAO对象
public BookInfoBeanDao(DaoConfig config, DaoSession daoSession) {
super(config, daoSession);// 调用父类的构造函数传入DaoConfig和DaoSession配置
super(config, daoSession);
}
/** Creates the underlying database table.
@ -85,64 +85,64 @@ public class BookInfoBeanDao extends AbstractDao<BookInfoBean, String> {
//将BookInfoBean实体对象的值绑定到DatabaseStatement对象中
protected final void bindValues(DatabaseStatement stmt, BookInfoBean entity) {
//清除绑定的值,确保每次绑定时都是干净的
stmt.clearBindings();// 清除之前的绑定值
stmt.clearBindings();
//绑定name字段到SQL语句的第一个位置
String name = entity.getName();// 获取name属性值
String name = entity.getName();
if (name != null) {
//将书名绑定到SQL语句的第一个位置
stmt.bindString(1, name);// 将name绑定到SQL语句的第一个位置
stmt.bindString(1, name);
}
//绑定tag字段到SQL语句的第二个位置
String tag = entity.getTag();// 获取tag属性值
String tag = entity.getTag();
if (tag != null) {
//将标签绑定到SQL语句的第二个位置
stmt.bindString(2, tag);// 将tag绑定到SQL语句的第二个位置
stmt.bindString(2, tag);
}
//绑定noteUrl字段到SQL语句的第三个位置
String noteUrl = entity.getNoteUrl(); // 获取noteUrl属性值
String noteUrl = entity.getNoteUrl();
if (noteUrl != null) {
//将书籍URL绑定到SQL语句的第三个位置
stmt.bindString(3, noteUrl);// 将noteUrl绑定到SQL语句的第三个位置
stmt.bindString(3, noteUrl);
}
//绑定chapterUrl字段到SQL语句的第四个位置
String chapterUrl = entity.getChapterUrl();// 获取chapterUrl属性值
String chapterUrl = entity.getChapterUrl();
if (chapterUrl != null) {
//将章节URL绑定到SQL语句的第四个位置
stmt.bindString(4, chapterUrl);// 将chapterUrl绑定到SQL语句的第四个位置
stmt.bindString(4, chapterUrl);
}
//将最后刷新时间long 类型绑定到SQL语句的第五个位置
stmt.bindLong(5, entity.getFinalRefreshData()); // 将finalRefreshData绑定到SQL语句的第五个位置
stmt.bindLong(5, entity.getFinalRefreshData());
//绑定coverUrl字段到SQL语句的第六个位置
String coverUrl = entity.getCoverUrl();// 获取coverUrl属性值
String coverUrl = entity.getCoverUrl();
if (coverUrl != null) {
//将封面图URL绑定到SQL语句的第六个位置
stmt.bindString(6, coverUrl);// 将coverUrl绑定到SQL语句的第六个位置
stmt.bindString(6, coverUrl);
}
//绑定author字段到SQL语句的第七个位置
String author = entity.getAuthor();// 获取author属性值
String author = entity.getAuthor();
if (author != null) {
//将作者绑定到SQL语句的第七个位置
stmt.bindString(7, author);// 将author绑定到SQL语句的第七个位置
stmt.bindString(7, author);
}
//绑定introduce字段到SQL语句的第八个位置
String introduce = entity.getIntroduce();// 获取introduce属性值
String introduce = entity.getIntroduce();
if (introduce != null) {
//将简介绑定到SQL语句的第八个位置
stmt.bindString(8, introduce);// 将introduce绑定到SQL语句的第八个位置
stmt.bindString(8, introduce);
}
//绑定origin字段到SQL语句的第九个位置
String origin = entity.getOrigin();// 获取origin属性值
String origin = entity.getOrigin();
if (origin != null) {
//将来源绑定到SQL语句的第九个位置
stmt.bindString(9, origin);// 将origin绑定到SQL语句的第九个位置
stmt.bindString(9, origin);
}
}
@ -153,53 +153,53 @@ public class BookInfoBeanDao extends AbstractDao<BookInfoBean, String> {
stmt.clearBindings();
//绑定name字段值
String name = entity.getName();// 获取name属性值
String name = entity.getName();
if (name != null) {
stmt.bindString(1, name);// 将name绑定到SQL语句的第一个位置
stmt.bindString(1, name);
}
//绑定tag字段值
String tag = entity.getTag();// 获取tag属性值
String tag = entity.getTag();
if (tag != null) {
stmt.bindString(2, tag);// 将tag绑定到SQL语句的第二个位置
stmt.bindString(2, tag);
}
//绑定noteUrl字段值
String noteUrl = entity.getNoteUrl();// 获取noteUrl属性值
String noteUrl = entity.getNoteUrl();
if (noteUrl != null) {
stmt.bindString(3, noteUrl);// 将noteUrl绑定到SQL语句的第三个位置
stmt.bindString(3, noteUrl);
}
//绑定chapterUrl字段值
String chapterUrl = entity.getChapterUrl();// 获取chapterUrl属性值
String chapterUrl = entity.getChapterUrl();
if (chapterUrl != null) {
stmt.bindString(4, chapterUrl);// 将chapterUrl绑定到SQL语句的第四个位置
stmt.bindString(4, chapterUrl);
}
//绑定finalRefreshData字段值
stmt.bindLong(5, entity.getFinalRefreshData());// 将finalRefreshData绑定到SQL语句的第五个位置
stmt.bindLong(5, entity.getFinalRefreshData());
//绑定coverUrl字段值
String coverUrl = entity.getCoverUrl();// 获取coverUrl属性值
String coverUrl = entity.getCoverUrl();
if (coverUrl != null) {
stmt.bindString(6, coverUrl);// 将coverUrl绑定到SQL语句的第六个位置
stmt.bindString(6, coverUrl);
}
//绑定author字段值
String author = entity.getAuthor();// 获取author属性值
String author = entity.getAuthor();
if (author != null) {
stmt.bindString(7, author);// 将author绑定到SQL语句的第七个位置
stmt.bindString(7, author);
}
//绑定introduce字段值
String introduce = entity.getIntroduce();// 获取introduce属性值
String introduce = entity.getIntroduce();
if (introduce != null) {
stmt.bindString(8, introduce);// 将introduce绑定到SQL语句的第八个参数
stmt.bindString(8, introduce);
}
//绑定origin字段值
String origin = entity.getOrigin();// 获取来源属性值
String origin = entity.getOrigin();
if (origin != null) {
stmt.bindString(9, origin);// 将来源绑定到SQL语句的第九个参数
stmt.bindString(9, origin);
}
}
@ -207,7 +207,7 @@ public class BookInfoBeanDao extends AbstractDao<BookInfoBean, String> {
//从Cursor中读取主键
public String readKey(Cursor cursor, int offset) {
//主键在第三列(offset + 2)如果为空返回null否则返回字符串
return cursor.isNull(offset + 2) ? null : cursor.getString(offset + 2);// 读取主键noteUrl字段
return cursor.isNull(offset + 2) ? null : cursor.getString(offset + 2);
}
@Override
@ -226,28 +226,28 @@ public class BookInfoBeanDao extends AbstractDao<BookInfoBean, String> {
cursor.isNull(offset + 8) ? null : cursor.getString(offset + 8) // origin
);
//返回创建好的BookInfoBean对象
return entity;// 返回读取到的实体对象
return entity;
}
@Override
//将Cursor中的数据读取到已存在的BookInfoBean实体中
public void readEntity(Cursor cursor, BookInfoBean entity, int offset) {
entity.setName(cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0));//设置name
entity.setTag(cursor.isNull(offset + 1) ? null : cursor.getString(offset + 1));//设置tag
entity.setNoteUrl(cursor.isNull(offset + 2) ? null : cursor.getString(offset + 2));//设置noteUrl
entity.setChapterUrl(cursor.isNull(offset + 3) ? null : cursor.getString(offset + 3));//设置chapterUrl
entity.setFinalRefreshData(cursor.getLong(offset + 4));//设置finalRefreshData
entity.setCoverUrl(cursor.isNull(offset + 5) ? null : cursor.getString(offset + 5));//设置coverUrl
entity.setAuthor(cursor.isNull(offset + 6) ? null : cursor.getString(offset + 6));//设置author
entity.setIntroduce(cursor.isNull(offset + 7) ? null : cursor.getString(offset + 7));//设置introduce
entity.setOrigin(cursor.isNull(offset + 8) ? null : cursor.getString(offset + 8));//设置origin
entity.setName(cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0));//name
entity.setTag(cursor.isNull(offset + 1) ? null : cursor.getString(offset + 1));//tag
entity.setNoteUrl(cursor.isNull(offset + 2) ? null : cursor.getString(offset + 2));//noteUrl
entity.setChapterUrl(cursor.isNull(offset + 3) ? null : cursor.getString(offset + 3));//chapterUrl
entity.setFinalRefreshData(cursor.getLong(offset + 4));//finalRefreshData
entity.setCoverUrl(cursor.isNull(offset + 5) ? null : cursor.getString(offset + 5));//coverUrl
entity.setAuthor(cursor.isNull(offset + 6) ? null : cursor.getString(offset + 6));//author
entity.setIntroduce(cursor.isNull(offset + 7) ? null : cursor.getString(offset + 7));//introduce
entity.setOrigin(cursor.isNull(offset + 8) ? null : cursor.getString(offset + 8));//origin
}
@Override
//插入数据后更新主键
protected final String updateKeyAfterInsert(BookInfoBean entity, long rowId) {
//使用noteUrl作为主键
return entity.getNoteUrl();// 返回用于更新的主键
return entity.getNoteUrl();
}
@Override
@ -255,9 +255,9 @@ public class BookInfoBeanDao extends AbstractDao<BookInfoBean, String> {
public String getKey(BookInfoBean entity) {
if(entity != null) {
//返回noteUrl作为主键
return entity.getNoteUrl();// 返回实体的主键noteUrl
return entity.getNoteUrl();
} else {
return null;// 如果实体为空返回null
return null;
}
}
@ -265,7 +265,7 @@ public class BookInfoBeanDao extends AbstractDao<BookInfoBean, String> {
//是否可更新实体
protected final boolean isEntityUpdateable() {
//可更新
return true;// 表示该实体可更新
return true;
}
}

@ -36,12 +36,12 @@ public class BookShelfBeanDao extends AbstractDao<BookShelfBean, String> {
// 构造函数使用DaoConfig配置初始化Dao
public BookShelfBeanDao(DaoConfig config) {
super(config);// 调用父类构造函数进行初始化
super(config);
}
// 构造函数使用DaoConfig和DaoSession配置初始化Dao
public BookShelfBeanDao(DaoConfig config, DaoSession daoSession) {
super(config, daoSession);// 调用父类构造函数进行初始化
super(config, daoSession);
}
/** Creates the underlying database table.
@ -66,27 +66,27 @@ public class BookShelfBeanDao extends AbstractDao<BookShelfBean, String> {
// 删除BOOK_SHELF_BEAN表的方法
public static void dropTable(Database db, boolean ifExists) {
// 构造删除表的SQL语句
String sql = "DROP TABLE " + (ifExists ? "IF EXISTS " : "") + "\"BOOK_SHELF_BEAN\"";// 如果指定ifExists则添加条件
String sql = "DROP TABLE " + (ifExists ? "IF EXISTS " : "") + "\"BOOK_SHELF_BEAN\"";
// 执行SQL语句删除表
db.execSQL(sql);// 执行删除表的SQL语句
db.execSQL(sql);
}
@Override
// 将BookShelfBean实体对象的值绑定到DatabaseStatement语句中
protected final void bindValues(DatabaseStatement stmt, BookShelfBean entity) {
// 清除之前的绑定
stmt.clearBindings();// 清除绑定,以保证每次都是干净的
stmt.clearBindings();
String noteUrl = entity.getNoteUrl(); // 获取noteUrl属性值
String noteUrl = entity.getNoteUrl();
if (noteUrl != null) {
// 绑定noteUrl值到第一个参数位置
stmt.bindString(1, noteUrl);// 将noteUrl绑定到SQL语句的第一个位置
stmt.bindString(1, noteUrl);
}
stmt.bindLong(2, entity.getDurChapter()); // 绑定durChapter值到第二个参数位置
stmt.bindLong(3, entity.getDurChapterPage()); // 绑定durChapterPage值到第三个参数位置
stmt.bindLong(4, entity.getFinalDate()); // 绑定finalDate值到第四个参数位置
String tag = entity.getTag();// 获取tag属性值
String tag = entity.getTag();
if (tag != null) {
stmt.bindString(5, tag); // 绑定tag值到第五个参数位置
}
@ -96,21 +96,21 @@ public class BookShelfBeanDao extends AbstractDao<BookShelfBean, String> {
// 将BookShelfBean实体对象的值绑定到SQLiteStatement语句中与上一个方法功能类似只是对象不同
protected final void bindValues(SQLiteStatement stmt, BookShelfBean entity) {
// 清除之前的绑定
stmt.clearBindings();// 清除绑定,以保证每次都是干净的
stmt.clearBindings();
String noteUrl = entity.getNoteUrl();// 获取noteUrl属性值
String noteUrl = entity.getNoteUrl();
if (noteUrl != null) {
// 绑定noteUrl值到第一个参数位置
stmt.bindString(1, noteUrl);// 将noteUrl绑定到SQL语句的第一个位置
stmt.bindString(1, noteUrl);
}
stmt.bindLong(2, entity.getDurChapter()); // 绑定durChapter值到第二个参数位置
stmt.bindLong(3, entity.getDurChapterPage()); // 绑定durChapterPage值到第三个参数位置
stmt.bindLong(4, entity.getFinalDate()); // 绑定finalDate值到第四个参数位置
String tag = entity.getTag();// 获取tag属性值
String tag = entity.getTag();
if (tag != null) {
// 绑定tag值到第五个参数位置
stmt.bindString(5, tag);// 将tag绑定到SQL语句的第五个位置
stmt.bindString(5, tag);
}
}
@ -118,7 +118,7 @@ public class BookShelfBeanDao extends AbstractDao<BookShelfBean, String> {
// 从游标中读取主键
public String readKey(Cursor cursor, int offset) {
// 从偏移量offset+0的位置读取主键值如果为空返回null否则返回字符串
return cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0);// noteUrl是主键
return cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0);
}
@Override
@ -133,24 +133,24 @@ public class BookShelfBeanDao extends AbstractDao<BookShelfBean, String> {
cursor.isNull(offset + 4) ? null : cursor.getString(offset + 4) // tag
);
// 返回创建好的BookShelfBean对象
return entity;// 返回读取到的实体对象
return entity;
}
@Override
// 将游标中的数据读取到已存在的BookShelfBean实体中
public void readEntity(Cursor cursor, BookShelfBean entity, int offset) {
entity.setNoteUrl(cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0)); // 设置noteUrl
entity.setDurChapter(cursor.getInt(offset + 1)); // 设置durChapter
entity.setDurChapterPage(cursor.getInt(offset + 2)); // 设置durChapterPage
entity.setFinalDate(cursor.getLong(offset + 3)); // 设置finalDate
entity.setTag(cursor.isNull(offset + 4) ? null : cursor.getString(offset + 4)); // 设置tag
entity.setNoteUrl(cursor.isNull(offset + 0) ? null : cursor.getString(offset + 0)); // noteUrl
entity.setDurChapter(cursor.getInt(offset + 1)); // durChapter
entity.setDurChapterPage(cursor.getInt(offset + 2)); // durChapterPage
entity.setFinalDate(cursor.getLong(offset + 3)); // finalDate
entity.setTag(cursor.isNull(offset + 4) ? null : cursor.getString(offset + 4)); // tag
}
@Override
// 插入数据后更新主键
protected final String updateKeyAfterInsert(BookShelfBean entity, long rowId) {
// 返回noteUrl作为主键
return entity.getNoteUrl();// 用于更新的主键值
return entity.getNoteUrl();
}
@Override
@ -158,9 +158,9 @@ public class BookShelfBeanDao extends AbstractDao<BookShelfBean, String> {
public String getKey(BookShelfBean entity) {
if(entity != null) {
// 返回noteUrl作为主键
return entity.getNoteUrl();// 返回实体的主键值
return entity.getNoteUrl();
} else {
return null;// 如果实体为空则返回null
return null;
}
}
@ -168,7 +168,7 @@ public class BookShelfBeanDao extends AbstractDao<BookShelfBean, String> {
// 判断实体是否可更新
protected final boolean isEntityUpdateable() {
// 返回true表示实体可更新
return true;// 该实体类型支持更新
return true;
}
}

@ -11,24 +11,22 @@ import org.greenrobot.greendao.database.Database; // 导入Database类GreenDa
import org.greenrobot.greendao.database.DatabaseStatement; // 导入DatabaseStatement类用于执行SQL语句
// 导入ChapterListBean类这是数据库操作的对象
import com.monke.monkeybook.bean.ChapterListBean;// 导入ChapterListBean实体类
import com.monke.monkeybook.bean.ChapterListBean;
// THIS CODE IS GENERATED BY greenDAO, DO NOT EDIT.
/**
* DAO for table "CHAPTER_LIST_BEAN".
* "CHAPTER_LIST_BEAN"DAO
*/
// ChapterListBeanDao类继承AbstractDao操作ChapterListBean对象主键类型为String
public class ChapterListBeanDao extends AbstractDao<ChapterListBean, String> {
// 数据库表名常量
public static final String TABLENAME = "CHAPTER_LIST_BEAN";// 定义常量,表示数据库表的名称
public static final String TABLENAME = "CHAPTER_LIST_BEAN";
/**
* Properties of entity ChapterListBean.<br/>
* Can be used for QueryBuilder and for referencing column names.
*/
public static class Properties {
// 定义实体类属性映射,指向数据库表中的字段
public final static Property NoteUrl = new Property(0, String.class, "noteUrl", false, "NOTE_URL"); // 属性0noteUrlString类型非主键数据库列名NOTE_URL
public final static Property DurChapterIndex = new Property(1, int.class, "durChapterIndex", false, "DUR_CHAPTER_INDEX"); // 属性1durChapterIndexint类型非主键数据库列名DUR_CHAPTER_INDEX
public final static Property DurChapterUrl = new Property(2, String.class, "durChapterUrl", true, "DUR_CHAPTER_URL"); // 属性2durChapterUrlString类型主键数据库列名DUR_CHAPTER_URL
@ -39,19 +37,19 @@ public class ChapterListBeanDao extends AbstractDao<ChapterListBean, String> {
// 构造函数使用DaoConfig配置初始化Dao
public ChapterListBeanDao(DaoConfig config) {
super(config);// 调用父类构造函数初始化DAO对象
super(config);
}
// 构造函数使用DaoConfig和DaoSession配置初始化Dao
public ChapterListBeanDao(DaoConfig config, DaoSession daoSession) {
super(config, daoSession);// 调用父类构造函数初始化DAO对象
super(config, daoSession);
}
/** Creates the underlying database table. */
// 创建CHAPTER_LIST_BEAN表的方法
public static void createTable(Database db, boolean ifNotExists) {
// 根据ifNotExists判断是否需要添加"IF NOT EXISTS"条件
String constraint = ifNotExists? "IF NOT EXISTS ": "";// 如果ifNotExists为真则添加条件
String constraint = ifNotExists? "IF NOT EXISTS ": "";
// 执行SQL语句创建表
db.execSQL("CREATE TABLE " + constraint + "\"CHAPTER_LIST_BEAN\" (" + //
"\"NOTE_URL\" TEXT," + // NOTE_URL字段TEXT类型
@ -68,47 +66,47 @@ public class ChapterListBeanDao extends AbstractDao<ChapterListBean, String> {
// 删除CHAPTER_LIST_BEAN表的方法
public static void dropTable(Database db, boolean ifExists) {
// 构造删除表的SQL语句
String sql = "DROP TABLE " + (ifExists ? "IF EXISTS " : "") + "\"CHAPTER_LIST_BEAN\"";// 如果ifExists为真则加上条件
String sql = "DROP TABLE " + (ifExists ? "IF EXISTS " : "") + "\"CHAPTER_LIST_BEAN\"";
// 执行SQL语句删除表
db.execSQL(sql);// 执行删除表的SQL语句
db.execSQL(sql);
}
@Override
// 将ChapterListBean实体对象的值绑定到DatabaseStatement语句中
protected final void bindValues(DatabaseStatement stmt, ChapterListBean entity) {
// 清除之前的绑定
stmt.clearBindings();// 确保每次绑定之前清除旧的绑定
stmt.clearBindings();
String noteUrl = entity.getNoteUrl();// 获取noteUrl属性值
String noteUrl = entity.getNoteUrl();
if (noteUrl != null) {
// 绑定noteUrl值到第一个参数位置
stmt.bindString(1, noteUrl);// 将noteUrl绑定到SQL语句的第一个位置
stmt.bindString(1, noteUrl);
}
// 绑定durChapterIndex值到第二个参数位置
stmt.bindLong(2, entity.getDurChapterIndex());// 将durChapterIndex绑定到SQL语句的第二个位置
stmt.bindLong(2, entity.getDurChapterIndex());
String durChapterUrl = entity.getDurChapterUrl();// 获取durChapterUrl属性值
String durChapterUrl = entity.getDurChapterUrl();
if (durChapterUrl != null) {
// 绑定durChapterUrl值到第三个参数位置
stmt.bindString(3, durChapterUrl);// 将durChapterUrl绑定到SQL语句的第三个位置
stmt.bindString(3, durChapterUrl);
}
String durChapterName = entity.getDurChapterName();// 获取durChapterName属性值
String durChapterName = entity.getDurChapterName();
if (durChapterName != null) {
// 绑定durChapterName值到第四个参数位置
stmt.bindString(4, durChapterName);// 将durChapterName绑定到SQL语句的第四个位置
stmt.bindString(4, durChapterName);
}
String tag = entity.getTag();// 获取tag属性值
String tag = entity.getTag();
if (tag != null) {
// 绑定tag值到第五个参数位置
stmt.bindString(5, tag);// 将tag绑定到SQL语句的第五个位置
stmt.bindString(5, tag);
}
Boolean hasCache = entity.getHasCache();// 获取hasCache属性值
Boolean hasCache = entity.getHasCache();
if (hasCache != null) {
// 绑定hasCache值到第六个参数位置true为1false为0
stmt.bindLong(6, hasCache ? 1L: 0L);// 将hasCache绑定到SQL语句的第六个位置布尔值转为整数
stmt.bindLong(6, hasCache ? 1L: 0L);
}
}
@ -116,38 +114,38 @@ public class ChapterListBeanDao extends AbstractDao<ChapterListBean, String> {
// 将ChapterListBean实体对象的值绑定到SQLiteStatement语句中与上一个方法功能类似只是对象不同
protected final void bindValues(SQLiteStatement stmt, ChapterListBean entity) {
// 清除之前的绑定
stmt.clearBindings();// 确保每次绑定之前清除旧的绑定
stmt.clearBindings();
String noteUrl = entity.getNoteUrl();// 获取noteUrl属性值
String noteUrl = entity.getNoteUrl();
if (noteUrl != null) {
// 绑定noteUrl值到第一个参数位置
stmt.bindString(1, noteUrl);// 将noteUrl绑定到SQL语句的第一个位置
stmt.bindString(1, noteUrl);
}
// 绑定durChapterIndex值到第二个参数位置
stmt.bindLong(2, entity.getDurChapterIndex());// 将durChapterIndex绑定到SQL语句的第二个位置
stmt.bindLong(2, entity.getDurChapterIndex());
String durChapterUrl = entity.getDurChapterUrl();// 获取durChapterUrl属性值
String durChapterUrl = entity.getDurChapterUrl();
if (durChapterUrl != null) {
// 绑定durChapterUrl值到第三个参数位置
stmt.bindString(3, durChapterUrl);// 将durChapterUrl绑定到SQL语句的第三个位置
stmt.bindString(3, durChapterUrl);
}
String durChapterName = entity.getDurChapterName();// 获取durChapterName属性值
String durChapterName = entity.getDurChapterName();
if (durChapterName != null) {
// 绑定durChapterName值到第四个参数位置
stmt.bindString(4, durChapterName);// 将durChapterName绑定到SQL语句的第四个位置
stmt.bindString(4, durChapterName);
}
String tag = entity.getTag();// 获取tag属性值
String tag = entity.getTag();
if (tag != null) {
// 绑定tag值到第五个参数位置
stmt.bindString(5, tag);// 将tag绑定到SQL语句的第五个位置
stmt.bindString(5, tag);
}
Boolean hasCache = entity.getHasCache();// 获取hasCache属性值
Boolean hasCache = entity.getHasCache();
if (hasCache != null) {
// 绑定hasCache值到第六个参数位置true为1false为0
stmt.bindLong(6, hasCache ? 1L: 0L);// 将hasCache绑定到SQL语句的第六个位置布尔值转为整数
stmt.bindLong(6, hasCache ? 1L: 0L);
}
}
@ -155,7 +153,7 @@ public class ChapterListBeanDao extends AbstractDao<ChapterListBean, String> {
// 从游标中读取主键
public String readKey(Cursor cursor, int offset) {
// 从偏移量offset+2的位置读取主键值如果为空返回null否则返回字符串
return cursor.isNull(offset + 2) ? null : cursor.getString(offset + 2);// 返回durChapterUrl作为主键
return cursor.isNull(offset + 2) ? null : cursor.getString(offset + 2);
}
@Override
@ -171,7 +169,7 @@ public class ChapterListBeanDao extends AbstractDao<ChapterListBean, String> {
cursor.isNull(offset + 5) ? null : cursor.getShort(offset + 5) != 0 // hasCache 将short转换为boolean
);
// 返回创建好的ChapterListBean对象
return entity;// 返回读取到的实体对象
return entity;
}
@Override
@ -189,7 +187,7 @@ public class ChapterListBeanDao extends AbstractDao<ChapterListBean, String> {
// 插入数据后更新主键
protected final String updateKeyAfterInsert(ChapterListBean entity, long rowId) {
// 返回durChapterUrl作为主键
return entity.getDurChapterUrl();// 用于更新的主键值
return entity.getDurChapterUrl();
}
@Override
@ -197,9 +195,9 @@ public class ChapterListBeanDao extends AbstractDao<ChapterListBean, String> {
public String getKey(ChapterListBean entity) {
if(entity != null) {
// 返回durChapterUrl作为主键
return entity.getDurChapterUrl();// 返回实体的主键值
return entity.getDurChapterUrl();
} else {
return null; // 实体为空返回null
return null;
}
}
@ -207,7 +205,7 @@ public class ChapterListBeanDao extends AbstractDao<ChapterListBean, String> {
// 判断实体是否可更新
protected final boolean isEntityUpdateable() {
// 返回true表示实体可更新
return true;// 该实体类型支持更新
return true;
}
}

@ -20,7 +20,7 @@ import org.greenrobot.greendao.identityscope.IdentityScopeType; // 导入Identit
// DaoMaster类继承AbstractDaoMaster是所有DAO的管理者
public class DaoMaster extends AbstractDaoMaster {
// 数据库模式版本号
public static final int SCHEMA_VERSION = 1;// 定义常量,表示数据库的版本号
public static final int SCHEMA_VERSION = 1;
/** Creates underlying database table using DAOs. */
// 创建所有数据库表的方法
@ -55,20 +55,19 @@ public class DaoMaster extends AbstractDaoMaster {
// 创建DaoMaster实例
DaoMaster daoMaster = new DaoMaster(db);
// 创建DaoSession实例并返回
return daoMaster.newSession();// 创建并返回DaoSession
return daoMaster.newSession();
}
// 构造函数使用SQLiteDatabase对象初始化DaoMaster
public DaoMaster(SQLiteDatabase db) {
// 使用StandardDatabase包装SQLiteDatabase
this(new StandardDatabase(db));// 创建StandardDatabase对象并调用另一个构造函数
this(new StandardDatabase(db));
}
// 构造函数使用Database对象初始化DaoMaster
public DaoMaster(Database db) {
// 调用父类的构造函数
super(db, SCHEMA_VERSION);
// 注册所有DAO类供后续使用
registerDaoClass(BookContentBeanDao.class); // 注册BookContentBeanDao
registerDaoClass(BookInfoBeanDao.class); // 注册BookInfoBeanDao
registerDaoClass(BookShelfBeanDao.class); // 注册BookShelfBeanDao
@ -79,14 +78,14 @@ public class DaoMaster extends AbstractDaoMaster {
// 创建一个新的DaoSession使用默认的IdentityScopeType.Session
public DaoSession newSession() {
// 创建DaoSession实例使用Session作用域
return new DaoSession(db, IdentityScopeType.Session, daoConfigMap);// 返回新的DaoSession实例
// 创建DaoSession实例
return new DaoSession(db, IdentityScopeType.Session, daoConfigMap);
}
// 创建一个新的DaoSession使用指定的IdentityScopeType
public DaoSession newSession(IdentityScopeType type) {
// 创建DaoSession实例,使用指定的作用域类型
return new DaoSession(db, type, daoConfigMap);// 返回新的DaoSession实例
// 创建DaoSession实例
return new DaoSession(db, type, daoConfigMap);
}
/**
@ -97,21 +96,21 @@ public class DaoMaster extends AbstractDaoMaster {
// 构造函数,指定上下文和数据库名称
public OpenHelper(Context context, String name) {
// 调用父类的构造函数
super(context, name, SCHEMA_VERSION);/
super(context, name, SCHEMA_VERSION);
}
// 构造函数,指定上下文、数据库名称和游标工厂
public OpenHelper(Context context, String name, CursorFactory factory) {
// 调用父类的构造函数
super(context, name, factory, SCHEMA_VERSION);// 设置上下文、数据库名称、游标工厂和版本号
super(context, name, factory, SCHEMA_VERSION);
}
@Override
// 创建数据库时调用
public void onCreate(Database db) {
// 输出日志信息,记录数据库创建事件
// 输出日志信息
Log.i("greenDAO", "Creating tables for schema version " + SCHEMA_VERSION);
// 创建所有数据库表
createAllTables(db, false);// 调用创建所有表的方法
createAllTables(db, false);
}
}
@ -121,13 +120,13 @@ public class DaoMaster extends AbstractDaoMaster {
// 构造函数,指定上下文和数据库名称
public DevOpenHelper(Context context, String name) {
// 调用父类的构造函数
super(context, name); // 设置上下文和数据库名称
super(context, name);
}
// 构造函数,指定上下文、数据库名称和游标工厂
public DevOpenHelper(Context context, String name, CursorFactory factory) {
// 调用父类的构造函数
super(context, name, factory);// 设置上下文、数据库名称和游标工厂
super(context, name, factory);
}
@Override
@ -138,7 +137,7 @@ public class DaoMaster extends AbstractDaoMaster {
// 删除所有数据库表
dropAllTables(db, true);
// 重新创建数据库表
onCreate(db);// 调用创建表的方法
onCreate(db);
}
}

@ -8,47 +8,24 @@ import com.monke.monkeybook.listener.OnGetChapterListListener;
import java.util.List;
import io.reactivex.Observable;
// 定义了一个名为IEasouBookModel的接口用于规范与书籍相关的各种数据获取操作的方法签名
// 该接口可能会有不同的实现类来具体处理对应的业务逻辑,例如与不同数据源交互获取书籍信息等
public interface IEasouBookModel {
/**
*
*
* @param content
* @param page
* @param rankKind
* @return ObservableList<SearchBookBean>
* Observable
*
*/
Observable<List<SearchBookBean>> searchBook(String content, int page, int rankKind);
/**
*
*
* @param bookShelfBean BookShelfBean
*
* @return ObservableBookShelfBean
* Observable
*
*/
Observable<BookShelfBean> getBookInfo(final BookShelfBean bookShelfBean);
/**
*
*
* @param bookShelfBean BookShelfBean
* @param getChapterListListener
*
*
*/
void getChapterList(final BookShelfBean bookShelfBean, OnGetChapterListListener getChapterListListener);
/**
*
*
* @param durChapterUrl URLURL
* @param durChapterIndex 便
* @return ObservableBookContentBeanBean
* Observable
*
*/
Observable<BookContentBean> getBookContent(final String durChapterUrl, final int durChapterIndex);
}
}

@ -3,42 +3,21 @@ package com.monke.monkeybook.model;
import com.monke.monkeybook.bean.LibraryBean;
import com.monke.monkeybook.bean.SearchBookBean;
import //
com.monke.monkeybook.cache.ACache;
import com.monke.monkeybook.cache.ACache;
import java.util.List;
import io.reactivex.Observable;
// IGxwztvBookModel接口继承自IStationBookModel接口意味着它会继承IStationBookModel中定义的方法
// 同时在此接口中又额外定义了一些与特定业务可能和gxwztv相关相关的书籍数据获取等操作的方法规范
public interface IGxwztvBookModel extends IStationBookModel {
/**
*
*
* @param url URL
* @param page 便
* @return ObservableList<SearchBookBean>
* Observable
*/
Observable<List<SearchBookBean>> getKindBook(String url, int page);
/**
*
*
* @param aCache ACache
* 便
* @return ObservableLibraryBeanLibraryBean
* Observable
*
*/
Observable<LibraryBean> getLibraryData(ACache aCache);
/**
*
*
* @param data
* LibraryBean
* @return ObservableLibraryBeanLibraryBean
* Observable便
*
*/
Observable<LibraryBean> analyLibraryData(String data);
}
}

@ -2,21 +2,10 @@
package com.monke.monkeybook.model;
import com.monke.monkeybook.bean.LocBookShelfBean;
// 导入Java中用于操作文件的类意味着该接口相关操作可能涉及对本地文件的处理比如导入本地书籍文件等
import java.io.File;
import io.reactivex.Observable;
// 定义了IImportBookModel接口该接口主要规范了导入书籍相关操作的方法签名
// 具体的导入书籍逻辑可由实现该接口的类来完成
public interface IImportBookModel {
/**
*
*
* @param book Filetxtepub
*
* @return ObservableLocBookShelfBean
* Observable
*/
Observable<LocBookShelfBean> importBook(File book);
}
}

@ -8,53 +8,25 @@ import com.monke.monkeybook.listener.OnGetChapterListListener;
import java.util.List;
import io.reactivex.Observable;
// IStationBookModel接口定义了一系列与书籍相关操作的方法规范这些操作可能是围绕某个站点Station展开的
// 比如从特定的网络站点获取书籍数据等,具体实现类会按照这些规范来实现对应的业务逻辑
public interface IStationBookModel {
/**
*
*
* @param content
*
* @param page
* 便
* @return ObservableList<SearchBookBean>
* Observable
*
*/
Observable<List<SearchBookBean>> searchBook(String content, int page);
/**
*
*
* @param bookShelfBean BookShelfBean
*
* BookShelfBean
* @return ObservableBookShelfBean
* Observable
*
*/
Observable<BookShelfBean> getBookInfo(final BookShelfBean bookShelfBean);
/**
*
*
* @param bookShelfBean BookShelfBean
*
* OnGetChapterListListener
* @param getChapterListListener
*
*
*/
void getChapterList(final BookShelfBean bookShelfBean, OnGetChapterListListener getChapterListListener);
/**
*
*
* @param durChapterUrl URLURL
*
* @param durChapterIndex 便使
*
* @return ObservableBookContentBeanBean
* Observable便
*
*/
Observable<BookContentBean> getBookContent(final String durChapterUrl, final int durChapterIndex);
}
}

@ -8,70 +8,28 @@ import com.monke.monkeybook.listener.OnGetChapterListListener;
import java.util.List;
import io.reactivex.Observable;
// IWebBookModel接口定义了一系列与网络书籍相关操作的方法规范
// 主要涉及从网络获取书籍各种信息(如书籍详情、目录、章节内容等)以及搜索不同类型书籍的功能,
// 具体的实现类会按照这些方法签名去实现对应的网络交互逻辑
public interface IWebBookModel {
/**
*
*
* @param bookShelfBean BookShelfBean
*
* BookShelfBean
* @return ObservableBookShelfBean
* Observable
*
*/
Observable<BookShelfBean> getBookInfo(final BookShelfBean bookShelfBean);
/**
*
*
* @param bookShelfBean BookShelfBean
*
* OnGetChapterListListener
*
* @param getChapterListListener
*
*
*
*/
void getChapterList(final BookShelfBean bookShelfBean,OnGetChapterListListener getChapterListListener);
/**
*
*
* @param durChapterUrl URLURL
*
* @param durChapterIndex 便使
*
* @param tag
* @return ObservableBookContentBeanBean
* Observable便
*
*/
Observable<BookContentBean> getBookContent(final String durChapterUrl, final int durChapterIndex, String tag);
/**
*
*
* @param url URL
*
* @param page 便
* 2
* @return ObservableList<SearchBookBean>
* Observable
*
*/
Observable<List<SearchBookBean>> getKindBook(String url,int page);
/**
*
*
* @param content
*
* @param page
* 便3
* @param tag
* @return ObservableList<SearchBookBean>
* Observable
*
*/
Observable<List<SearchBookBean>> searchOtherBook(String content,int page,String tag);
}
}

@ -1,22 +1,7 @@
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model;
// IWebContentModel接口定义了一个用于分析书籍内容的方法规范
// 具体的实现类需要按照此规范来实现对书籍内容相关的解析操作,
// 可能是从网络获取到的书籍文本内容等进行特定的分析处理。
public interface IWebContentModel {
/**
*
*
* @param s
*
* @param realUrl URL
* URL
* @return
*
* @throws Exception
*
*/
String analyBookcontent(String s,String realUrl) throws Exception;
}

@ -7,48 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// 实现IWebContentModel接口的类用于处理特定网站的网页内容解析
public class Content17duxsModelImpl implements IWebContentModel {
// 定义一个表示特定网站的标签这里的值是网站的网址
public class Content17duxsModelImpl implements IWebContentModel{
public static final String TAG = "http://www.17duxs.com";
// 获取Content17duxsModelImpl类的单例实例
public static Content17duxsModelImpl getInstance() {
return new Content17duxsModelImpl();
}
// 私有构造方法,按照单例模式的常见思路,防止外部直接实例化该类,不过此处并没有严格限制单例唯一性相关逻辑
private Content17duxsModelImpl() {
}
// 实现IWebContentModel接口中的方法用于解析书籍内容
// 参数s是待解析的网页内容字符串通常是通过网络请求获取到的网页源码等realUrl参数可能是对应的真实网址也许用于一些后续验证或者相关逻辑
// 该方法可能会抛出异常,调用者需要处理相关异常情况
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup解析传入的网页内容字符串得到一个Document对象方便后续对网页DOM结构进行操作
Document doc = Jsoup.parse(s);
// 通过id为"content"获取对应的文本节点列表
List<TextNode> contentEs = doc.getElementById("content").textNodes();
StringBuilder content = new StringBuilder();
// 遍历获取到的文本节点列表
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容并去除前后空白字符trim方法的作用
String temp = contentEs.get(i).text().trim();
// 将文本中的全角空格(&nbsp;对应的字符)和半角空格都替换为空字符串,进行文本内容的格式化清理
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 如果处理后的文本长度大于0说明有有效内容
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 给有效内容添加两个全角空格("\u3000\u3000" 表示两个全角空格)作为缩进格式
content.append("\u3000\u3000" + temp);
// 如果不是最后一个文本节点,添加换行符,用于区分不同的文本节点内容,使解析后的内容格式更合理
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将处理好的内容以字符串形式返回,最终返回的就是解析并格式化后的书籍内容字符串
return content.toString();
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
@ -8,48 +8,33 @@ import org.jsoup.nodes.Element;
import org.jsoup.nodes.TextNode;
import java.util.List;
// Content17kModelImpl类实现了IWebContentModel接口用于处理来自17k网站相关内容的解析逻辑
public class Content17kModelImpl implements IWebContentModel {
// 定义一个表示17k网站的标签字符串可能用于标识来源等用途
public static final String TAG = "http://www.17k.com";
// 采用单例模式获取Content17kModelImpl类的唯一实例
public static Content17kModelImpl getInstance() {
return new Content17kModelImpl();
}
// 将构造函数私有化保证外部不能通过new关键字随意创建实例符合单例模式的要求
private Content17kModelImpl() {
}
// 实现IWebContentModel接口中的analyBookcontent方法用于解析书籍内容
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup解析传入的字符串通常是网页内容的HTML字符串构建成Document对象方便后续对HTML文档结构进行操作
Document doc = Jsoup.parse(s);
// 通过id获取HTML文档中id为"chapterContentWapper"的元素,这个元素大概率包含了书籍章节的具体内容
Element all = doc.getElementById("chapterContentWapper");
// 获取该元素下所有子元素中的第一个元素包含的文本节点列表此处假设这个结构符合17k网站内容的布局特点
List<TextNode> contentEs = all.getAllElements().get(0).textNodes();
StringBuilder content = new StringBuilder();
// 遍历文本节点列表
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容,并去除首尾空格
String temp = contentEs.get(i).text().trim();
// 替换掉文本中的不间断空格(" ")以及普通空格,进一步清理文本内容
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 如果清理后的文本长度大于0说明有实际内容
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 给文本添加两个全角空格作为缩进格式("\u3000\u3000"),使其在展示时有一定的排版效果
content.append("\u3000\u3000" + temp);
// 如果不是最后一个文本节点,添加换行符,使不同部分的文本内容分行显示
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将处理好的文本内容以字符串形式返回
return content.toString();
}
}
}

@ -1,21 +1,15 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
import org.jsoup.Jsoup;
// 导入Jsoup库中的Document类用于表示解析后的HTML文档结构
import org.jsoup.nodes.Document;
// 导入Jsoup库中的TextNode类用于表示文本节点方便对HTML中的文本内容进行操作
import org.jsoup.nodes.TextNode;
import java.util.List;
// Content3dllcModelImpl类实现了IWebContentModel接口意味着它需要按照该接口定义的规范来实现相应的方法
public class Content3dllcModelImpl implements IWebContentModel {
// 定义一个静态的常量字符串TAG用于标识相关的网址或者某种特定的标记
// 可能在后续的代码逻辑中用于判断来源或者做相关的分类等用途
public class Content3dllcModelImpl implements IWebContentModel{
public static final String TAG = "http://www.3dllc.cc";
// 它用于获取Content3dllcModelImpl类的实例方便外部代码调用获取该类对象进行相关操作
public static Content3dllcModelImpl getInstance() {
return new Content3dllcModelImpl();
}
@ -23,41 +17,21 @@ public class Content3dllcModelImpl implements IWebContentModel {
private Content3dllcModelImpl() {
}
// 实现IWebContentModel接口中定义的analyBookcontent方法用于分析书籍内容。
// 参数s表示要解析的包含书籍内容的HTML字符串
// realUrl参数表示该内容对应的真实网址
// 该方法声明抛出Exception异常意味着在解析过程中如果出现了比如HTML格式不正确、找不到指定元素等问题时会将异常抛给调用者处理
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup的parse方法将传入的HTML字符串解析成一个Document对象
// 这样后续就可以方便地通过Document对象提供的方法来查找、操作HTML文档中的各种元素和内容
Document doc = Jsoup.parse(s);
// 通过Document对象的.getElementById方法根据元素的id属性值这里是"content")来获取对应的元素下的所有文本节点,
// 推测这里的"content"元素可能就是包含了书籍正文内容的HTML元素获取到的是一个TextNode类型的列表方便后续逐个处理这些文本节点内容
List<TextNode> contentEs = doc.getElementById("content").textNodes();
// 创建一个可变的字符串构建器对象,用于逐步构建最终的书籍内容字符串,相比直接使用普通字符串拼接,这样效率更高,尤其在大量字符串拼接操作时
StringBuilder content = new StringBuilder();
// 遍历获取到的文本节点列表
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容,并去除两端的空白字符(空格、制表符等),得到一个相对比较“纯净”的文本内容字符串
String temp = contentEs.get(i).text().trim();
// 将文本内容中的不间断空格(&nbsp;对应的字符)替换为空字符串,同时把普通空格也替换为空字符串,进一步清理文本内容,
// 可能是为了保证最终书籍内容格式的一致性或者符合特定的解析要求
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 如果清理后的文本内容长度大于0说明该文本节点包含有效的文本信息需要将其添加到最终的书籍内容字符串构建器中
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在文本内容前添加两个全角空格(\u3000表示全角空格可能是为了符合某种排版或者显示格式要求比如缩进等
// 然后添加到content字符串构建器中
content.append("\u3000\u3000" + temp);
// 如果当前不是最后一个文本节点,就在后面添加一个换行符(\r\n在Windows系统下常见的换行表示方式
// 使得最终的书籍内容文本在显示时不同文本节点的内容能分行显示,更符合阅读和排版习惯
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将构建好的包含书籍内容的字符串构建器对象转换为普通字符串并返回,这个字符串就是经过解析、清理和格式化后的书籍正文内容
return content.toString();
}
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
@ -7,48 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// Content44pqModelImpl类实现了IWebContentModel接口主要用于处理来自http://www.44pq.net网站相关书籍内容的解析操作
public class Content44pqModelImpl implements IWebContentModel {
// 定义一个表示网站来源的标签字符串这里代表的是http://www.44pq.net可能用于后续区分不同来源等情况
public class Content44pqModelImpl implements IWebContentModel{
public static final String TAG = "http://www.44pq.net";
// 使用单例模式获取Content44pqModelImpl类的唯一实例确保在程序中只有一个该类的实例存在
public static Content44pqModelImpl getInstance() {
return new Content44pqModelImpl();
}
// 将构造函数私有化防止外部代码通过new关键字随意创建该类的多个实例符合单例模式的设计要求
private Content44pqModelImpl() {
}
// 重写IWebContentModel接口中的analyBookcontent方法用于解析书籍内容
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库的parse方法将传入的字符串通常是对应网页的HTML内容字符串解析为Document对象
// 这样就可以方便地按照HTML文档结构来操作和提取信息了
Document doc = Jsoup.parse(s);
// 通过id获取Document对象中id为"content"的元素包含的所有文本节点,这里假设该"content"元素包含了书籍章节的实际文本内容,
// 具体依赖于http://www.44pq.net网站页面的HTML结构布局
List<TextNode> contentEs = doc.getElementById("content").textNodes();
StringBuilder content = new StringBuilder();
// 遍历获取到的文本节点列表
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容,并去除其首尾的空白字符,使文本更加规整
String temp = contentEs.get(i).text().trim();
// 替换掉文本中的不间断空格(" ")以及普通空格,进一步清理文本,去除不必要的空白格式
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 如果经过清理后文本的长度大于0说明该文本节点包含有实际有效的内容
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 给有效的文本内容添加两个全角空格("\u3000\u3000")作为缩进,以便在最终展示文本时具有更好的排版效果
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是列表中的最后一个节点,就在后面添加换行符("\r\n"),使不同部分的文本内容分行显示,增强可读性
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将处理好的包含书籍章节内容的文本以字符串形式返回,以便在其他地方使用该解析后的内容
return content.toString();
}
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
@ -7,59 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// Content630bookCCModelImpl类实现了IWebContentModel接口主要负责对来自网址为http://www.630book.cc的书籍内容进行解析处理
public class Content630bookCCModelImpl implements IWebContentModel {
// 定义一个公共静态的常量字符串TAG其值为"http://www.630book.cc",用于标识该类所针对处理内容的来源网站。
// 在整个程序架构中,这个标识可能会用于区分不同来源网站的相关逻辑,例如根据不同来源做特定的格式调整或者数据存储等操作
public class Content630bookCCModelImpl implements IWebContentModel{
public static final String TAG = "http://www.630book.cc";
// 这是一个静态方法用于获取Content630bookCCModelImpl类的实例采用的是单例模式。
// 通过这种方式确保在整个程序运行过程中,该类只有一个实例存在,有助于节省内存资源,避免因多次创建实例可能导致的资源浪费和数据不一致等问题
public static Content630bookCCModelImpl getInstance() {
return new Content630bookCCModelImpl();
}
// 将构造函数私有化这是实现单例模式的关键步骤之一。私有化构造函数后外部类无法直接通过new关键字来创建该类的实例
// 只能通过上面定义的getInstance方法来获取唯一的实例以此保证单例模式的正确实现
private Content630bookCCModelImpl() {
}
// 重写了IWebContentModel接口中定义的analyBookcontent方法此方法是整个类的核心功能所在用于解析书籍内容。
// 它接收两个参数一个是字符串s通常代表从网页获取到的包含书籍章节内容的HTML格式的字符串另一个是realUrl可能表示该内容对应的真实网址
// 虽然在当前代码中暂时未看到对realUrl参数的进一步使用但在更复杂的场景下可以基于此参数做更多与网址相关的逻辑处理
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的参数s即包含书籍内容的HTML字符串解析为Document对象。
// Document对象是Jsoup对HTML文档结构的一种表示形式通过它可以方便地按照HTML的文档结构如标签、元素、属性等来查找、提取和操作相应的内容
Document doc = Jsoup.parse(s);
// 通过Document对象的getElementById方法根据元素的id属性查找id为"content"的元素,并获取该元素所包含的所有文本节点。
// 这里假设在http://www.630book.cc网站的页面结构中id为"content"的元素存放着需要解析的书籍章节的实际文本内容,
// 如果网站页面结构发生变化,这个假设可能就不成立了,相应的代码逻辑可能需要调整
List<TextNode> contentEs = doc.getElementById("content").textNodes();
StringBuilder content = new StringBuilder();
// 开始遍历获取到的文本节点列表contentEs对其中的每一个文本节点进行处理
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容并调用trim方法去除文本前后的空白字符包括空格、制表符、换行符等使得获取到的文本更加干净、规范
// 避免多余的空白字符影响后续对文本内容的处理以及最终的展示效果
String temp = contentEs.get(i).text().trim();
// 使用replaceAll方法替换文本中的不间断空格" ")以及普通空格,进一步清理文本内容,确保文本格式的一致性和简洁性,
// 这样在后续拼接和展示文本时不会出现因空格格式不一致而导致的排版问题
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过清理后的文本长度是否大于0如果大于0说明该文本节点包含有实际有意义的文本内容需要将其添加到最终的解析结果中进行拼接处理
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 给有效的文本内容添加两个全角空格("\u3000\u3000")作为缩进,使解析后的文本在展示时(比如在阅读软件或者文本编辑工具中)具有更好的排版效果,
// 更符合一般阅读的视觉习惯,方便读者阅读和区分不同段落的内容
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是列表中的最后一个节点,也就是后面还有其他文本节点需要处理,就在当前文本后面添加换行符("\r\n"
// 这样可以让不同部分的文本内容分行显示,清晰地分隔开各段文本,使整个书籍内容的展示更加有条理,易于阅读和理解
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将经过上述一系列处理、拼接后的文本内容以字符串形式返回,这个返回的字符串就是最终解析好的书籍章节内容,
// 可以被其他模块(比如显示模块、存储模块等)获取并进一步使用,以实现完整的书籍阅读或者数据处理等功能
return content.toString();
}
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
@ -7,54 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// Content92zwModelImpl类实现了IWebContentModel接口其主要目的是处理来自网址为http://www.92zw.la相关书籍内容的解析工作
public class Content92zwModelImpl implements IWebContentModel {
// 定义一个静态的常量字符串TAG用于标识该类所针对处理内容的来源网站这里是http://www.92zw.la
// 或许在后续的代码中可用于区分不同网站来源的相关逻辑处理等情况
public class Content92zwModelImpl implements IWebContentModel{
public static final String TAG = "http://www.92zw.la";
// 采用单例模式通过此静态方法获取Content92zwModelImpl类的唯一实例。单例模式可以保证在整个应用程序运行期间
// 该类只有一个实例存在,避免重复创建实例造成资源浪费等问题
public static Content92zwModelImpl getInstance() {
return new Content92zwModelImpl();
}
// 将构造函数私有化这是实现单例模式的常见做法外部代码无法直接通过new关键字来创建该类的多个实例
// 只能通过上面的getInstance方法获取唯一实例
private Content92zwModelImpl() {
}
// 重写IWebContentModel接口中的analyBookcontent方法该方法用于对书籍内容进行具体的解析操作
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库的parse方法将传入的参数s通常是从网页获取到的包含书籍内容的HTML字符串解析为Document对象。
// Document对象可以看作是对整个HTML文档结构的一种抽象表示方便后续基于文档结构去查找和提取相关元素及内容
Document doc = Jsoup.parse(s);
// 通过id属性查找Document对象中id为"contents"的元素并获取该元素包含的所有文本节点。这里假设id为"contents"的元素
// 存放着需要解析的书籍章节的实际文本内容这是基于http://www.92zw.la网站页面HTML结构的特定设计若网站结构改变此处可能需调整
List<TextNode> contentEs = doc.getElementById("contents").textNodes();
StringBuilder content = new StringBuilder();
// 遍历获取到的文本节点列表contentEs对每个文本节点进行相应处理
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容并调用trim方法去除文本前后的空白字符空格、制表符、换行符等使得获取到的文本更干净纯粹
String temp = contentEs.get(i).text().trim();
// 替换文本中的不间断空格(" ")以及普通空格,进一步清理文本内容,去除多余的空白,避免影响后续文本展示的格式和可读性
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过清理后的文本长度是否大于0如果大于0说明该文本节点包含有实际有意义的内容需要进行后续处理和拼接
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 给有效的文本内容添加两个全角空格("\u3000\u3000")作为缩进,使解析后的文本在展示时(比如在文本编辑器或者阅读软件中)
// 具有更好的排版效果,看起来更加整齐美观,符合一般阅读的格式习惯
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是列表中的最后一个节点,也就是后面还有其他文本节点需要处理,就在当前文本后面添加换行符("\r\n"
// 这样可以让不同部分的文本内容分行显示,更清晰地分隔开各段文本内容,方便阅读和查看
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将经过上述一系列处理、拼接后的文本内容以字符串形式返回,这个返回的字符串就是解析好的书籍章节内容,
// 可以供其他模块进一步使用,比如显示在阅读界面上等
return content.toString();
}
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
@ -7,63 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// ContentAszwModelImpl类实现了IWebContentModel接口其主要职责是对来自网址为http://www.aszw.org的书籍内容进行解析处理操作
public class ContentAszwModelImpl implements IWebContentModel {
// 定义一个公共静态的常量字符串TAG用于标识该类所处理内容对应的网站来源此处为"http://www.aszw.org"。
// 在整个程序的更宏观层面,这个标识可能会在诸如区分不同网站内容、针对不同来源做差异化处理(比如不同的存储策略、格式转换规则等)等场景中发挥作用
public class ContentAszwModelImpl implements IWebContentModel{
public static final String TAG = "http://www.aszw.org";
// 这是一个静态方法遵循单例模式的设计思路用于获取ContentAszwModelImpl类的唯一实例。
// 单例模式确保在程序运行期间,该类只会存在一个实例对象,这样做有助于避免因频繁创建实例而导致的内存资源浪费,同时保证了数据的一致性等,
// 适用于像这种对特定网站内容解析的功能模块,因为通常不需要多个实例来完成同样的解析任务
public static ContentAszwModelImpl getInstance() {
return new ContentAszwModelImpl();
}
// 将构造函数私有化这是实现单例模式的重要手段之一。通过私有化构造函数外部代码就无法直接通过new关键字来创建该类的多个实例了
// 只能按照既定的单例模式规则通过上面定义的getInstance方法获取唯一可用的实例对象
private ContentAszwModelImpl() {
}
// 重写IWebContentModel接口中定义的analyBookcontent方法此方法是整个类实现书籍内容解析功能的核心所在。
// 它接收两个参数:
// 参数s通常是一个包含了从网页上获取到的书籍章节内容的HTML格式字符串也就是要进行解析处理的原始文本数据
// 参数realUrl表示该内容对应的真实网址不过在当前代码中暂时没有看到针对realUrl的具体使用逻辑在更复杂的应用场景下它可以用于例如验证网址合法性、
// 根据网址不同进行额外的处理或者记录等操作
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的功能强大的parse方法将传入的参数s也就是前面提到的包含书籍内容的HTML字符串解析成一个Document对象。
// Document对象在Jsoup框架里代表着对整个HTML文档结构的一种抽象表示借助它可以方便地按照HTML文档固有的结构像通过标签、元素的id或class属性等方式
// 去查找、提取以及操作相应的内容,为后续解析书籍内容奠定基础
Document doc = Jsoup.parse(s);
// 通过Document对象提供的getElementById方法依据元素的id属性去查找id为"contents"的元素,并获取这个元素所包含的所有文本节点。
// 这里基于对http://www.aszw.org网站页面结构的一种假设即认为id为"contents"的元素里面存放着需要解析提取的书籍章节的实际文本内容。
// 倘若该网站后续对页面结构进行了更改,导致这个元素的位置、作用或者内容存储方式发生变化,那么此处的代码逻辑可能就需要相应地进行调整了
List<TextNode> contentEs = doc.getElementById("contents").textNodes();
StringBuilder content = new StringBuilder();
// 开启一个循环对获取到的文本节点列表contentEs进行遍历依次处理其中的每一个文本节点目的是将它们整理、拼接成最终期望的书籍内容文本格式
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前正在处理的文本节点的文本内容并调用trim方法去除该文本前后可能存在的空白字符例如空格、制表符、换行符等
// 这样做是为了让获取到的文本更加纯粹、规范,避免那些多余的空白字符在后续的文本处理和展示过程中造成不必要的干扰,比如影响排版效果等
String temp = contentEs.get(i).text().trim();
// 利用replaceAll方法来替换文本中的不间断空格" ")以及普通空格,进一步对文本内容进行清理,使得文本的格式更加统一、简洁,
// 确保在后续将各个文本节点的内容拼接起来后,文本整体的展示效果更加整齐、美观,不会因为空格格式不一致等问题而显得杂乱无章
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述清理操作后的文本长度是否大于0如果大于0就说明这个文本节点包含了实际有意义的内容值得添加到最终解析好的书籍内容当中去
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 给有效的文本内容添加两个全角空格("\u3000\u3000")作为缩进,这么做是为了让解析后的书籍内容在展示时(比如在阅读类应用程序或者文本编辑工具里呈现时)
// 具备更好的排版效果,更符合人们日常阅读书籍时对于段落格式、缩进等方面的视觉习惯,方便读者清晰地阅读和区分不同部分的内容
content.append("\u3000\u3000" + temp);
// 如果当前正在处理的文本节点不是列表中的最后一个节点,也就是后面还有其他文本节点需要继续处理和拼接,那么就在当前文本内容后面添加一个换行符("\r\n"
// 这样可以让不同部分的文本内容分行显示,使得整个书籍内容在展示上更加有条理、层次分明,更易于阅读者理解和浏览
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将经过前面一系列处理、拼接操作后得到的文本内容以字符串的形式返回,这个返回的字符串就是最终解析完成的、符合预期格式的书籍章节内容,
// 它可以被程序中的其他模块(例如用于展示书籍内容的界面模块、负责将内容存储起来的存储模块等)获取并进一步加以利用,以实现完整的书籍阅读或者相关的数据处理等功能
return content.toString();
}
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
@ -7,63 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// ContentBaishukuModelImpl类实现了IWebContentModel接口其主要功能是针对来自网址为http://baishuku.com的书籍内容进行解析处理。
public class ContentBaishukuModelImpl implements IWebContentModel {
// 定义一个公共静态的常量字符串TAG其值为"http://baishuku.com",用于标识该类所负责处理内容对应的网站来源。
// 在整个程序架构中,这个标识有可能会用于区分不同网站的内容,例如根据不同来源网站做差异化的逻辑处理,像内容格式调整、存储方式的不同选择等操作。
public class ContentBaishukuModelImpl implements IWebContentModel{
public static final String TAG = "http://baishuku.com";
// 这是一个静态方法采用单例模式来获取ContentBaishukuModelImpl类的唯一实例。
// 单例模式确保在程序运行期间,该类只会存在一个实例对象,避免了因多次创建实例而可能造成的内存资源浪费,同时保证了在处理相关内容时数据的一致性等情况。
// 对于这种针对特定网站内容解析的功能模块来说,通常不需要多个实例来完成同样的解析工作,所以使用单例模式较为合适。
public static ContentBaishukuModelImpl getInstance() {
return new ContentBaishukuModelImpl();
}
// 将构造函数私有化这是实现单例模式的关键步骤之一。通过把构造函数设置为私有外部代码就无法直接通过new关键字来创建该类的多个实例了
// 只能按照单例模式规定的方式也就是通过上面定义的getInstance方法来获取唯一可用的实例对象以此保证单例模式的正确实现。
private ContentBaishukuModelImpl() {
}
// 重写了IWebContentModel接口中定义的analyBookcontent方法此方法是整个类实现解析书籍内容这一核心功能的关键所在。
// 它接收两个参数:
// 参数s是一个字符串通常代表从http://baishuku.com网站获取到的包含书籍章节内容的HTML格式的字符串也就是要进行解析处理的原始文本数据
// 参数realUrl表示该内容对应的真实网址不过在当前代码中暂时未看到针对realUrl参数有进一步的使用逻辑在更复杂的应用场景下
// 可以基于这个参数做一些额外的操作,比如验证网址的合法性、根据网址不同做不同的处理策略等。
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的参数s即包含书籍内容的HTML字符串解析为一个Document对象。
// Document对象在Jsoup框架里是对HTML文档结构的一种抽象表示通过它可以方便地按照HTML文档的结构特点例如通过元素的id、class属性等方式
// 去查找、提取和操作相应的内容,这为后续解析书籍内容打下了基础。
Document doc = Jsoup.parse(s);
// 通过Document对象提供的getElementById方法按照元素的id属性查找id为"content"的元素,并获取这个元素包含的所有文本节点。
// 这里基于对http://baishuku.com网站页面结构的一种假设即认为id为"content"的元素存放着需要解析的书籍章节的实际文本内容。
// 如果该网站后续对页面结构进行了改变,导致这个元素的相关情况(如位置、内容存储方式等)发生变化,那么此处的代码逻辑可能就需要相应地调整了。
List<TextNode> contentEs = doc.getElementById("content").textNodes();
StringBuilder content = new StringBuilder();
// 开始循环遍历获取到的文本节点列表contentEs目的是对其中的每一个文本节点进行处理最终将它们整理、拼接成符合要求的书籍内容文本格式。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前正在处理的文本节点的文本内容并调用trim方法去除文本前后可能存在的空白字符比如空格、制表符、换行符等
// 这样做可以让获取到的文本更加纯净、规范,避免多余的空白字符对后续文本处理以及最终展示效果产生不良影响,例如造成排版混乱等问题。
String temp = contentEs.get(i).text().trim();
// 使用replaceAll方法替换文本中的不间断空格" ")以及普通空格,进一步对文本内容进行清理,使得文本格式更加统一、简洁,
// 保证在后续拼接各个文本节点内容后,整个书籍内容的展示效果更加整齐、美观,不会出现因空格格式不一致而显得杂乱的情况。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述清理操作后的文本长度是否大于0如果大于0则说明这个文本节点包含了实际有意义的内容需要将其添加到最终解析好的书籍内容当中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 给有效的文本内容添加两个全角空格("\u3000\u3000")作为缩进,这样做可以让解析后的书籍内容在展示时(比如在阅读应用程序或者文本编辑工具中呈现时)
// 具有更好的排版效果,更符合人们日常阅读书籍时对段落格式、缩进等方面的视觉习惯,便于读者清晰地阅读和区分不同部分的内容。
content.append("\u3000\u3000" + temp);
// 如果当前正在处理的文本节点不是列表中的最后一个节点,也就是后面还有其他文本节点需要继续处理和拼接,
// 那么就在当前文本内容后面添加一个换行符("\r\n"),使得不同部分的文本内容能够分行显示,让整个书籍内容在展示上更有条理、层次分明,更易于阅读者理解和浏览。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将经过前面一系列处理、拼接操作后得到的文本内容以字符串的形式返回,这个返回的字符串就是最终解析完成的、符合预期格式的书籍章节内容,
// 它可以被程序中的其他模块(例如用于展示书籍内容的界面模块、负责存储书籍内容的存储模块等)获取并进一步加以利用,以实现完整的书籍阅读或者相关的数据处理等功能。
return content.toString();
}
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
@ -7,70 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// ContentBxwx9ModelImpl类实现了IWebContentModel接口其主要用途是对来自网址为http://www.bxwx9.org的书籍内容进行解析处理操作。
public class ContentBxwx9ModelImpl implements IWebContentModel {
// 定义一个公共静态的常量字符串TAG其值被设定为"http://www.bxwx9.org"该TAG用于标识此解析类所对应的内容来源网站。
// 在整个程序的更大架构中,这个标识可以用于区分不同来源网站的内容,比如针对不同网站的内容采取不同的后续处理逻辑,像存储格式的调整、展示样式的适配等情况。
public class ContentBxwx9ModelImpl implements IWebContentModel{
public static final String TAG = "http://www.bxwx9.org";
// 这是一个遵循单例模式设计的静态方法用于获取ContentBxwx9ModelImpl类的唯一实例。
// 单例模式确保在程序运行期间,此类只会存在一个实例对象,这样做能够避免因重复创建实例而可能导致的内存资源浪费,同时保障在处理相关书籍内容时数据的一致性等方面的要求。
// 对于专门针对特定网站书籍内容解析的功能模块而言,通常一个实例就足以完成相应的解析任务,所以采用单例模式是比较合适的选择。
public static ContentBxwx9ModelImpl getInstance() {
return new ContentBxwx9ModelImpl();
}
// 将构造函数私有化这是实现单例模式的重要手段之一。通过把构造函数设置为私有属性外部代码就无法直接通过使用new关键字来创建该类的多个实例了。
// 外部只能按照单例模式所规定的方式也就是通过调用上面定义的getInstance方法来获取唯一可用的实例对象以此保证单例模式能够正确地得以实现。
private ContentBxwx9ModelImpl() {
}
// 重写了IWebContentModel接口中定义的analyBookcontent方法该方法是整个ContentBxwx9ModelImpl类实现解析书籍内容这一核心功能的关键所在。
// 它接收两个参数:
// 参数s是一个字符串类型的变量通常表示从http://www.bxwx9.org网站获取到的包含书籍章节内容的HTML格式的字符串也就是后续要进行解析处理的原始文本数据
// 参数realUrl同样是字符串类型它代表该内容对应的真实网址不过在当前代码的现有逻辑中暂时没有看到针对这个realUrl参数有进一步的具体使用情况
// 但在更为复杂的应用场景下,这个参数可以被用于诸如验证网址的合法性、根据不同网址执行不同的解析策略或者记录相关网址信息等操作。
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的功能强大的parse方法将传入的参数s即包含书籍内容的HTML字符串解析成为一个Document对象。
// 在Jsoup框架里Document对象是对HTML文档结构进行抽象表示的一种重要形式借助它能够方便地依据HTML文档自身的结构特点例如通过元素的id、class属性等方式
// 去查找、提取以及操作对应的内容,这为后续深入解析书籍内容搭建了良好的基础。
Document doc = Jsoup.parse(s);
// 通过Document对象提供的便捷方法getElementById按照元素的id属性去查找id为"content"的元素,并获取这个元素所包含的所有文本节点。
// 此处基于对http://www.bxwx9.org网站页面结构的一种既有假设即认为id为"content"的这个元素存放着需要解析的书籍章节的实际文本内容。
// 然而,如果该网站后续对自身的页面结构进行了调整修改,导致这个元素在页面中的位置、其所存储内容的方式等方面发生了变化,
// 那么此处的代码逻辑就很可能需要相应地做出适应性的调整改变了。
List<TextNode> contentEs = doc.getElementById("content").textNodes();
StringBuilder content = new StringBuilder();
// 开启一个循环对获取到的文本节点列表contentEs进行逐一的遍历操作目的在于对其中每一个文本节点都按照既定的规则进行处理
// 最终将它们合理地整理、拼接成为符合预期格式的书籍内容文本。
for (int i = 0; i < contentEs.size(); i++) {
// 首先获取当前正在处理的文本节点所包含的文本内容然后调用trim方法去除该文本前后可能存在的空白字符这些空白字符可能包含空格、制表符、换行符等等
// 通过这样的操作可以使得获取到的文本更加纯净、规范,从而避免多余的空白字符对后续文本的处理流程以及最终的展示效果产生不良的影响,
// 例如可能会造成文本排版出现混乱无序的情况等。
String temp = contentEs.get(i).text().trim();
// 接着使用Java中的replaceAll方法来对文本中的不间断空格" ")以及普通空格进行替换操作,进一步对文本内容进行清理优化,
// 以此让文本的格式变得更加统一、简洁,进而保证在后续把各个文本节点的内容拼接起来之后,整个书籍内容在展示的时候能够更加整齐、美观,
// 不会由于空格格式不一致等问题而显得杂乱无章、影响阅读体验。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 对经过上述一系列清理操作后的文本进行长度判断如果其长度大于0那就意味着这个文本节点中包含了实际有意义的内容
// 这样的文本内容是需要被添加到最终解析好的书籍内容当中去的。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 针对有实际意义的文本内容,给它添加两个全角空格("\u3000\u3000")作为缩进,这么做的好处是可以让解析后的书籍内容在展示时(比如在阅读类的应用程序或者文本编辑工具中呈现时),
// 具备更好的排版效果,更加贴合人们日常阅读书籍时对于段落格式、缩进等方面的视觉习惯,方便读者能够更加清晰地阅读和区分不同部分的文本内容。
content.append("\u3000\u3000" + temp);
// 如果当前正在处理的文本节点并非是列表中的最后一个节点,也就是说后面还有其他的文本节点需要继续进行处理和拼接操作,
// 那么就在当前文本内容的后面添加一个换行符("\r\n"),通过这样的方式可以使得不同部分的文本内容能够分行显示,
// 让整个书籍内容在展示方面更加有条理、层次分明,更有助于阅读者去理解和浏览其中的内容。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 经过前面循环中对各个文本节点的处理、拼接等一系列操作之后,将最终得到的文本内容以字符串的形式返回。
// 这个返回的字符串就是经过完整解析、并且符合预期格式要求的书籍章节内容了,它可以被程序中的其他相关模块(例如用于展示书籍内容的界面模块、
// 负责将书籍内容进行存储的存储模块等)获取并进一步加以利用,以此来实现完整的书籍阅读或者其他相关的数据处理等功能。
return content.toString();
}
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.ErrorAnalyContentManager;
@ -8,75 +8,37 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// ContentCommendModelImpl类从命名来看可能是用于通用的内容推荐相关模型实现但从代码功能上更像是内容解析相关可能命名不太准确需结合更完整的业务场景判断
public class ContentCommendModelImpl {
// 采用单例模式通过此静态方法获取ContentCommendModelImpl类的唯一实例。
// 单例模式可以保证在整个应用程序运行期间,该类只有一个实例存在,避免重复创建实例造成资源浪费等问题,适用于一些全局唯一的功能模块场景。
public static ContentCommendModelImpl getInstance() {
return new ContentCommendModelImpl();
}
// 将构造函数私有化这是实现单例模式的常见做法外部代码无法直接通过new关键字来创建该类的多个实例
// 只能通过上面的getInstance方法获取唯一实例以此确保类实例的唯一性和对其创建过程的控制。
private ContentCommendModelImpl() {
}
// analyBookcontent方法用于解析书籍内容并将解析结果设置到BookContentBean对象中返回同时处理可能出现的解析异常情况。
// 它接收三个参数:
// - bookContentBean一个BookContentBean类型的对象可能用于承载解析后的书籍内容以及相关状态信息等最终也会将解析结果设置到这个对象中返回。
// - s一个字符串通常代表从网页获取到的包含书籍章节内容的HTML格式的字符串也就是要进行解析处理的原始文本数据。
// - realUrl表示该内容对应的真实网址可能在记录错误信息、判断网站支持情况等方面会用到这个参数。
public BookContentBean analyBookcontent(BookContentBean bookContentBean, String s, String realUrl) throws Exception {
// 调用ErrorAnalyContentManager的单例实例的writeNewErrorUrl方法将当前尝试解析的真实网址记录下来
// 可能用于后续统计哪些网址出现过解析问题或者进行错误分析等相关操作具体功能取决于ErrorAnalyContentManager类的实现逻辑。
ErrorAnalyContentManager.getInstance().writeNewErrorUrl(realUrl);
try {
// 使用Jsoup库的parse方法将传入的参数s即包含书籍内容的HTML字符串解析为Document对象。
// Document对象可以看作是对整个HTML文档结构的一种抽象表示方便后续基于文档结构去查找和提取相关元素及内容
// 例如通过元素的id、class等属性查找特定元素。
try{
Document doc = Jsoup.parse(s);
// 通过id属性查找Document对象中id为"content"的元素,并获取该元素包含的所有文本节点。
// 这里假设id为"content"的元素存放着需要解析的书籍章节的实际文本内容不过这依赖于对应网页的HTML结构设计
// 若网页结构改变,此处获取内容的逻辑可能会失效,需要相应调整。
List<TextNode> contentEs = doc.getElementById("content").textNodes();
StringBuilder content = new StringBuilder();
// 遍历获取到的文本节点列表contentEs对每个文本节点进行相应处理目的是整理并拼接出最终的书籍内容文本。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容并调用trim方法去除文本前后的空白字符空格、制表符、换行符等使得获取到的文本更干净纯粹
// 避免多余空白字符影响后续文本处理和展示效果,比如排版等方面。
String temp = contentEs.get(i).text().trim();
// 替换文本中的不间断空格(" ")以及普通空格,进一步清理文本内容,去除不必要的空白,使文本格式更统一,便于后续拼接展示。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过清理后的文本长度是否大于0如果大于0说明该文本节点包含有实际有意义的内容需要进行后续处理和拼接。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 给有效的文本内容添加两个全角空格("\u3000\u3000")作为缩进,使解析后的文本在展示时(比如在文本编辑器或者阅读软件中)
// 具有更好的排版效果,看起来更加整齐美观,符合一般阅读的格式习惯。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是列表中的最后一个节点,也就是后面还有其他文本节点需要处理,就在当前文本后面添加换行符("\r\n"
// 这样可以让不同部分的文本内容分行显示,更清晰地分隔开各段文本内容,方便阅读和查看。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将处理好的包含书籍章节内容的文本设置到bookContentBean对象的durCapterContent属性中
// 这样通过这个对象就可以将解析后的内容传递到其他需要使用的地方了。
bookContentBean.setDurCapterContent(content.toString());
} catch (Exception e) {
// 如果在解析过程中出现异常,先打印异常的堆栈信息,方便进行调试和排查问题,了解异常发生的具体位置和原因。
}catch (Exception e){
e.printStackTrace();
// 设置bookContentBean对象的durCapterContent属性为一段提示信息告知用户当前站点暂时不支持解析
// 并提供了反馈的QQ联系方式以及承诺解决问题的时效等信息用于向用户说明情况。
bookContentBean.setDurCapterContent(realUrl.substring(0, realUrl.indexOf('/', 8)) + "站点暂时不支持解析请反馈给Monke QQ:1105075896,半小时内解决,超级效率的程序员");
// 同时将bookContentBean对象的right属性设置为false表示解析出现了问题该内容可能是不正确或者不完全的
// 其他使用这个对象的地方可以根据这个属性来判断解析结果的有效性。
bookContentBean.setRight(false);
}
// 最后将处理后的bookContentBean对象返回这个对象中包含了解析后的书籍内容以及相关状态信息
// 可供其他模块进一步使用,比如展示给用户或者进行后续的业务逻辑处理等。
return bookContentBean;
}
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
@ -7,73 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// ContentDhzwModelImpl类实现了IWebContentModel接口其主要功能是针对来自网址为http://www.dhzw.org的书籍内容进行解析处理以提取出可用的文本内容。
public class ContentDhzwModelImpl implements IWebContentModel {
// 定义一个公共静态的常量字符串TAG其值设定为"http://www.dhzw.org",用于标识该类所处理内容对应的网站来源。
// 在整个程序架构中,这个标识可用于区分不同网站的内容,例如后续根据不同来源网站采取不同的处理逻辑,像针对特定网站内容进行格式转换、存储策略调整等操作。
public class ContentDhzwModelImpl implements IWebContentModel{
public static final String TAG = "http://www.dhzw.org";
// 这是一个静态方法采用单例模式来获取ContentDhzwModelImpl类的唯一实例。
// 单例模式能确保在程序运行期间,该类只会存在一个实例对象,避免了因多次创建实例而可能引发的内存资源浪费问题,同时保证了在解析相关书籍内容时数据的一致性等情况。
// 对于这种专门针对特定网站内容解析的功能模块而言,通常只需一个实例就能完成相应的解析任务,所以单例模式是比较合适的设计选择。
public static ContentDhzwModelImpl getInstance() {
return new ContentDhzwModelImpl();
}
// 将构造函数私有化这是实现单例模式的关键步骤之一。通过把构造函数设置为私有外部代码就无法直接通过new关键字来创建该类的多个实例了
// 只能按照单例模式规定的方式也就是通过上面定义的getInstance方法来获取唯一可用的实例对象以此保障单例模式的正确实现。
private ContentDhzwModelImpl() {
}
// 重写IWebContentModel接口中定义的analyBookcontent方法此方法是整个类实现解析书籍内容这一核心功能的关键所在。
// 它接收两个参数:
// 参数s是一个字符串通常代表从http://www.dhzw.org网站获取到的包含书籍章节内容的HTML格式的字符串也就是要进行解析处理的原始文本数据
// 参数realUrl表示该内容对应的真实网址不过在当前代码的现有逻辑中暂时未看到针对realUrl参数有进一步的具体使用情况
// 但在更为复杂的应用场景下,该参数可用于验证网址合法性、根据网址不同执行不同的解析策略或者记录相关网址信息等操作。
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的参数s即包含书籍内容的HTML字符串解析为一个Document对象。
// 在Jsoup框架里Document对象是对HTML文档结构进行抽象表示的一种重要形式借助它能够方便地依据HTML文档自身的结构特点例如通过元素的id、class属性等方式
// 去查找、提取以及操作对应的内容,这为后续深入解析书籍内容搭建了良好的基础。
Document doc = Jsoup.parse(s);
// 通过Document对象提供的便捷方法getElementById按照元素的id属性去查找id为"BookText"的元素,并获取这个元素所包含的所有文本节点。
// 此处基于对http://www.dhzw.org网站页面结构的一种既有假设即认为id为"BookText"的这个元素存放着需要解析的书籍章节的实际文本内容。
// 倘若该网站后续对自身的页面结构进行了调整修改,导致这个元素在页面中的位置、其所存储内容的方式等方面发生了变化,
// 那么此处的代码逻辑就很可能需要相应地做出适应性的调整改变了。
List<TextNode> contentEs = doc.getElementById("BookText").textNodes();
StringBuilder content = new StringBuilder();
// 开启一个循环对获取到的文本节点列表contentEs进行逐一的遍历操作目的在于对其中每一个文本节点都按照既定的规则进行处理
// 最终将它们合理地整理、拼接成为符合预期格式的书籍内容文本。
for (int i = 0; i < contentEs.size(); i++) {
// 首先获取当前正在处理的文本节点所包含的文本内容然后调用trim方法去除该文本前后可能存在的空白字符这些空白字符可能包含空格、制表符、换行符等等
// 通过这样的操作可以使得获取到的文本更加纯净、规范,从而避免多余的空白字符对后续文本的处理流程以及最终的展示效果产生不良的影响,
// 例如可能会造成文本排版出现混乱无序的情况等。
String temp = contentEs.get(i).text().trim();
// 接着使用Java中的replaceAll方法来对文本中的不间断空格" ")以及普通空格进行替换操作,进一步对文本内容进行清理优化,
// 以此让文本的格式变得更加统一、简洁,进而保证在后续把各个文本节点的内容拼接起来之后,整个书籍内容在展示的时候能够更加整齐、美观,
// 不会由于空格格式不一致等问题而显得杂乱无章、影响阅读体验。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 额外的处理:将文本中的全角空格(“ ”)也进行替换去除,进一步清理文本内容,确保文本格式更加规范,此处同样是基于对网站内容格式特点的一种处理需求假设。
temp = temp.replaceAll(" ", "");
// 对经过上述一系列清理操作后的文本进行长度判断如果其长度大于0那就意味着这个文本节点中包含了实际有意义的内容
// 这样的文本内容是需要被添加到最终解析好的书籍内容当中去的。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 针对有实际意义的文本内容,给它添加两个全角空格("\u3000\u3000")作为缩进,这么做的好处是可以让解析后的书籍内容在展示时(比如在阅读类的应用程序或者文本编辑工具中呈现时),
// 具备更好的排版效果,更加贴合人们日常阅读书籍时对于段落格式、缩进等方面的视觉习惯,方便读者能够更加清晰地阅读和区分不同部分的文本内容。
// 这里注意在添加缩进时再次获取了当前文本节点处理后的文本内容调用了一次text().trim().replaceAll(" ","")),可能代码存在一些重复逻辑,可考虑优化。
content.append("\u3000\u3000" + contentEs.get(i).text().trim().replaceAll(" ", ""));
// 如果当前正在处理的文本节点并非是列表中的最后一个节点,也就是说后面还有其他的文本节点需要继续进行处理和拼接操作,
// 那么就在当前文本内容的后面添加一个换行符("\r\n"),通过这样的方式可以使得不同部分的文本内容能够分行显示,
// 让整个书籍内容在展示方面更加有条理、层次分明,更有助于阅读者去理解和浏览其中的内容。
content.append("\u3000\u3000" + contentEs.get(i).text().trim().replaceAll(" ",""));
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 经过前面循环中对各个文本节点的处理、拼接等一系列操作之后,将最终得到的文本内容以字符串的形式返回。
// 这个返回的字符串就是经过完整解析、并且符合预期格式要求的书籍章节内容了,它可以被程序中的其他相关模块(例如用于展示书籍内容的界面模块、
// 负责将书籍内容进行存储的存储模块等)获取并进一步加以利用,以此来实现完整的书籍阅读或者其他相关的数据处理等功能。
return content.toString();
}
}
}

@ -1,4 +1,4 @@
// Copyright (c) 2017. 章钦豪. All rights reserved.
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.model.impl;
import com.monke.monkeybook.model.IWebContentModel;
@ -7,70 +7,32 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
// ContentEasouModelImpl类实现了IWebContentModel接口其主要职责是对来自网址为http://book.easou.com的书籍内容进行解析处理操作。
public class ContentEasouModelImpl implements IWebContentModel {
// 定义一个公共静态的常量字符串TAG其值被设定为"http://book.easou.com"该TAG用于标识此解析类所对应的内容来源网站。
// 在整个程序的更大架构中,这个标识可以用于区分不同来源网站的内容,比如针对不同网站的内容采取不同的后续处理逻辑,像存储格式的调整、展示样式的适配等情况。
public static final String TAG = "http://book.easou.com";
// 这是一个遵循单例模式设计的静态方法用于获取ContentEasouModelImpl类的唯一实例。
// 单例模式确保在程序运行期间,此类只会存在一个实例对象,这样做能够避免因重复创建实例而可能导致的内存资源浪费,同时保障在处理相关书籍内容时数据的一致性等方面的要求。
// 对于专门针对特定网站书籍内容解析的功能模块而言,通常一个实例就足以完成相应的解析任务,所以采用单例模式是比较合适的选择。
public static ContentEasouModelImpl getInstance() {
public static ContentEasouModelImpl getInstance(){
return new ContentEasouModelImpl();
}
// 将构造函数私有化这是实现单例模式的重要手段之一。通过把构造函数设置为私有属性外部代码就无法直接通过使用new关键字来创建该类的多个实例了。
// 外部只能按照单例模式所规定的方式也就是通过调用上面定义的getInstance方法来获取唯一可用的实例对象以此保证单例模式能够正确地得以实现。
private ContentEasouModelImpl() {
private ContentEasouModelImpl(){
}
// 重写了IWebContentModel接口中定义的analyBookcontent方法该方法是整个ContentEasouModelImpl类实现解析书籍内容这一核心功能的关键所在。
// 它接收两个参数:
// 参数s是一个字符串类型的变量通常表示从http://book.easou.com网站获取到的包含书籍章节内容的HTML格式的字符串也就是后续要进行解析处理的原始文本数据
// 参数realUrl同样是字符串类型它代表该内容对应的真实网址不过在当前代码的现有逻辑中暂时没有看到针对这个realUrl参数有进一步的具体使用情况
// 但在更为复杂的应用场景下,这个参数可以被用于诸如验证网址的合法性、根据不同网址执行不同的解析策略或者记录相关网址信息等操作。
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的功能强大的parse方法将传入的参数s即包含书籍内容的HTML字符串解析成为一个Document对象。
// 在Jsoup框架里Document对象是对HTML文档结构进行抽象表示的一种重要形式借助它能够方便地依据HTML文档自身的结构特点例如通过元素的id、class属性等方式
// 去查找、提取以及操作对应的内容,这为后续深入解析书籍内容搭建了良好的基础。
public String analyBookcontent(String s, String realUrl) throws Exception{
Document doc = Jsoup.parse(s);
// 通过Document对象提供的便捷方法getElementsByClass按照元素的class属性去查找class为"show"的元素并获取查找到的第一个该类元素通过调用get(0)
// 然后获取这个元素所包含的所有文本节点。这里基于对http://book.easou.com网站页面结构的一种既有假设即认为class为"show"的这个元素(第一个匹配的)存放着需要解析的书籍章节的实际文本内容。
// 然而,如果该网站后续对自身的页面结构进行了调整修改,导致这个元素在页面中的位置、其所存储内容的方式等方面发生了变化,
// 那么此处的代码逻辑就很可能需要相应地做出适应性的调整改变了。
List<TextNode> contentEs = doc.getElementsByClass("show").get(0).textNodes();
StringBuilder content = new StringBuilder();
// 开启一个循环对获取到的文本节点列表contentEs进行逐一的遍历操作目的在于对其中每一个文本节点都按照既定的规则进行处理
// 最终将它们合理地整理、拼接成为符合预期格式的书籍内容文本。
for (int i = 0; i < contentEs.size(); i++) {
// 首先获取当前正在处理的文本节点所包含的文本内容然后调用trim方法去除该文本前后可能存在的空白字符这些空白字符可能包含空格、制表符、换行符等等
// 通过这样的操作可以使得获取到的文本更加纯净、规范,从而避免多余的空白字符对后续文本的处理流程以及最终的展示效果产生不良的影响,
// 例如可能会造成文本排版出现混乱无序的情况等。
String temp = contentEs.get(i).text().trim();
// 接着使用Java中的replaceAll方法来对文本中的不间断空格" ")以及普通空格进行替换操作,进一步对文本内容进行清理优化,
// 以此让文本的格式变得更加统一、简洁,进而保证在后续把各个文本节点的内容拼接起来之后,整个书籍内容在展示的时候能够更加整齐、美观,
// 不会由于空格格式不一致等问题而显得杂乱无章、影响阅读体验。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 对经过上述一系列清理操作后的文本进行长度判断如果其长度大于0那就意味着这个文本节点中包含了实际有意义的内容
// 这样的文本内容是需要被添加到最终解析好的书籍内容当中去的。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 针对有实际意义的文本内容,给它添加两个全角空格("\u3000\u3000")作为缩进,这么做的好处是可以让解析后的书籍内容在展示时(比如在阅读类的应用程序或者文本编辑工具中呈现时),
// 具备更好的排版效果,更加贴合人们日常阅读书籍时对于段落格式、缩进等方面的视觉习惯,方便读者能够更加清晰地阅读和区分不同部分的文本内容。
content.append("\u3000\u3000" + temp);
// 如果当前正在处理的文本节点并非是列表中的最后一个节点,也就是说后面还有其他的文本节点需要继续进行处理和拼接操作,
// 那么就在当前文本内容的后面添加一个换行符("\r\n"),通过这样的方式可以使得不同部分的文本内容能够分行显示,
// 让整个书籍内容在展示方面更加有条理、层次分明,更有助于阅读者去理解和浏览其中的内容。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 经过前面循环中对各个文本节点的处理、拼接等一系列操作之后,将最终得到的文本内容以字符串的形式返回。
// 这个返回的字符串就是经过完整解析、并且符合预期格式要求的书籍章节内容了,它可以被程序中的其他相关模块(例如用于展示书籍内容的界面模块、
// 负责将书籍内容进行存储的存储模块等)获取并进一步加以利用,以此来实现完整的书籍阅读或者其他相关的数据处理等功能。
return content.toString();
}
}

@ -8,81 +8,34 @@ import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentFuheishuModelImplIWebContentModelTAGhttp://fuheishu.com
*
*
*/
public class ContentFuheishuModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址此处为http://fuheishu.com
public class ContentFuheishuModelImpl implements IWebContentModel{
public static final String TAG = "http://fuheishu.com";
/**
* ContentFuheishuModelImpl
* 便使
*
* @return ContentFuheishuModelImpl
*/
public static ContentFuheishuModelImpl getInstance() {
public static ContentFuheishuModelImpl getInstance(){
return new ContentFuheishuModelImpl();
}
/**
*
* getInstance
*/
private ContentFuheishuModelImpl() {
private ContentFuheishuModelImpl(){
}
/**
* IWebContentModelanalyBookcontent
* HTML
*
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库的parse方法将传入的原始字符串通常是HTML内容解析为一个Document对象
// 方便后续通过DOM操作来提取需要的书籍内容相关节点等信息。
public String analyBookcontent(String s, String realUrl) throws Exception{
Document doc = Jsoup.parse(s);
// 从realUrl中提取书籍的唯一标识bookId这里通过截取网址字符串的方式获取。
// 先找到从第21个字符之后此处假设网址的特定结构可能需要根据实际网址情况调整出现的第一个'/'的位置,
// 然后从该位置的下一个字符开始,截取到'.html'之前的字符串作为bookId用于后续定位书籍内容所在的HTML元素。
String bookId = realUrl.substring(realUrl.indexOf("/", 21) + 1, realUrl.indexOf(".html"));
// 通过获取到的bookId在解析后的Document对象中查找id为"content" + bookId的元素并获取其包含的所有文本节点TextNode列表。
// 这些文本节点预计包含了实际的书籍内容信息,后续将对这些节点进行处理以提取出完整的书籍内容。
List<TextNode> contentTNs = doc.getElementById("content" + bookId).textNodes();
// 创建一个StringBuilder对象用于高效地拼接处理后的书籍内容字符串避免频繁创建新的字符串对象带来的性能开销。
String bookId = realUrl.substring(realUrl.indexOf("/",21)+1,realUrl.indexOf(".html"));
List<TextNode> contentTNs = doc.getElementById("content"+bookId).textNodes();
StringBuilder stringBuilder = new StringBuilder();
// 遍历获取到的所有文本节点contentTNs对每个节点进行处理。
for (int i = 0; i < contentTNs.size(); i++) {
// 获取当前文本节点的文本内容,并去除两端的空白字符(空格、制表符、换行等),得到较为纯净的文本内容。
for(int i=0;i<contentTNs.size();i++){
String temp = contentTNs.get(i).text().trim();
// 进一步去除文本内容中的不间断空格(&nbsp;对应的Unicode字符这里替换为空字符串以及普通空格
// 使文本内容的格式更加规整,去除多余的空白间隔。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 如果处理后的文本内容长度大于0说明该文本节点包含有效的内容信息将其添加到stringBuilder中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效文本内容前添加两个全角空格(\u3000作为缩进格式使书籍内容排版更美观符合一般的阅读格式要求。
stringBuilder.append("\u3000\u3000" + temp);
// 如果当前不是最后一个文本节点,就在后面添加一个换行符(\r\n实现不同文本节点内容之间的换行
// 让书籍内容在显示时更有条理,便于阅读。
if (i < contentTNs.size() - 1) {
stringBuilder.append("\r\n");
}
}
}
// 将处理好的书籍内容字符串通过StringBuilder的toString方法转换为普通字符串并返回该结果
// 该字符串即为最终解析并格式化后的书籍内容,可以用于后续的展示、存储等操作。
return stringBuilder.toString();
}
}

@ -7,79 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentKewaishuModelImplIWebContentModelTAGhttp://www.kewaishu.org
*
*/
public class ContentKewaishuModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址此处为http://www.kewaishu.org
public class ContentKewaishuModelImpl implements IWebContentModel{
public static final String TAG = "http://www.kewaishu.org";
/**
* ContentKewaishuModelImpl
* 便使
*
*
* @return ContentKewaishuModelImpl
*/
public static ContentKewaishuModelImpl getInstance() {
return new ContentKewaishuModelImpl();
}
/**
*
* getInstance
*/
private ContentKewaishuModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
*
*
* @param s HTML
* @param realUrl 便
* @return 使
* @throws Exception 使JsoupHTMLHTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库的parse方法将传入的原始字符串通常为HTML格式的网页内容解析为一个Document对象
// 这样后续就能基于DOM文档对象模型结构方便地查找和操作网页中的各种元素提取我们需要的书籍内容相关部分。
Document doc = Jsoup.parse(s);
// 首先通过id查找网页中id为"content"的元素,然后在这个元素内查找所有的<p>标签元素,
// 这里取第一个<p>标签元素通过get(0)获取并获取其包含的所有文本节点TextNode列表。
// 假设这里认为书籍内容是包含在这个特定的<p>标签元素中的文本节点集合里,通过这样的方式定位到需要解析的文本内容所在位置。
List<TextNode> contentEs = doc.getElementById("content").getElementsByTag("p").get(0).textNodes();
// 创建一个StringBuilder对象用于高效地拼接处理后的书籍内容字符串相比于直接使用字符串相加操作
// StringBuilder可以减少创建多个临时字符串对象带来的性能开销更适合用于频繁的字符串拼接场景。
StringBuilder content = new StringBuilder();
// 遍历获取到的所有文本节点contentEs对每个节点进行处理和拼接操作。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容,并去除两端的空白字符(如空格、制表符、换行等),得到相对纯净、无多余空白的文本内容。
String temp = contentEs.get(i).text().trim();
// 进一步替换文本内容中的不间断空格Unicode编码为“&nbsp;”对应的字符,这里替换为空字符串)以及普通空格,
// 以此进一步清理文本,让文本格式更加规整,避免出现多余的空白间隔影响阅读体验。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 如果处理后的文本内容长度大于0说明该文本节点包含有效的信息将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效文本内容前添加两个全角空格(\u3000作为缩进格式使最终呈现的书籍内容排版更美观符合一般阅读时的段落格式要求。
content.append("\u3000\u3000" + temp);
// 如果当前不是最后一个文本节点,就在添加的文本内容后面添加一个换行符(\r\n
// 实现不同文本节点内容之间的换行,使书籍内容在展示时更有条理,便于阅读和区分不同的段落内容。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将经过处理和拼接后存储在StringBuilder中的书籍内容字符串转换为普通字符串通过调用toString方法来实现
// 并将最终整理好的书籍内容字符串返回,该字符串可用于后续的展示、存储等操作,比如在阅读软件中显示书籍文本内容等。
return content.toString();
}
}
}

@ -7,78 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentLeduwoModelImplIWebContentModelTAGhttp://leduwo.com
* 便使
*/
public class ContentLeduwoModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址这里对应的是http://leduwo.com方便知晓该解析逻辑针对的具体站点。
public class ContentLeduwoModelImpl implements IWebContentModel{
public static final String TAG = "http://leduwo.com";
/**
* ContentLeduwoModelImpl
* 便
*
* @return ContentLeduwoModelImpl
*/
public static ContentLeduwoModelImpl getInstance() {
return new ContentLeduwoModelImpl();
}
/**
*
* getInstance
*/
private ContentLeduwoModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'便
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库的parse方法将传入的原始字符串一般是HTML格式的网页内容解析为一个Document对象。
// 通过创建Document对象后续就能基于DOM文档对象模型的结构方便地在网页内容中查找、定位和操作各种元素从而提取出我们需要的书籍内容相关部分。
Document doc = Jsoup.parse(s);
// 通过id查找网页中id为"content"的元素并获取该元素包含的所有文本节点TextNode列表。
// 这里假设网页中书籍的具体内容是存放在id为"content"的这个元素内的文本节点当中,以此方式来定位要解析的文本内容所在位置。
List<TextNode> contentEs = doc.getElementById("content").textNodes();
// 创建一个StringBuilder对象用于高效地拼接处理后的书籍内容字符串。
// 相比于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销特别适合在需要多次进行字符串拼接的场景下使用。
StringBuilder content = new StringBuilder();
// 遍历获取到的所有文本节点contentEs对每个文本节点进行相应的处理和拼接操作。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容,并去除其两端的空白字符(例如空格、制表符、换行等),得到相对纯净、没有多余空白的文本内容。
String temp = contentEs.get(i).text().trim();
// 进一步替换文本内容中的不间断空格Unicode编码对应的“&nbsp;”字符,这里将其替换为空字符串)以及普通空格,
// 以此进一步清理文本内容,使得文本的格式更加规整,避免出现多余的空白间隔影响阅读体验和内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 如果处理后的文本内容长度大于0说明该文本节点包含有效的信息此时将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效文本内容前添加两个全角空格(\u3000作为缩进格式这样能让最终呈现的书籍内容排版更加美观符合一般阅读时对于段落格式的要求。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是最后一个文本节点,就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,使得书籍内容在展示时更有条理,方便阅读者区分不同的段落内容。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 将经过处理和拼接后存储在StringBuilder中的书籍内容字符串转换为普通字符串通过调用toString方法来实现这一转换操作
// 最后将整理好的书籍内容字符串返回,这个返回的字符串可以用于后续的各种操作,比如在阅读软件中展示书籍的文本内容、将内容存储到本地文件等。
return content.toString();
}
}
}

@ -7,86 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentLewen8ModelImplIWebContentModelTAGhttp://www.lewen8.com
*
*/
public class ContentLewen8ModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于明确标识该类所对应的网站地址这里设定为http://www.lewen8.com
// 方便在整个程序架构中知晓此解析逻辑具体是针对哪个站点的书籍内容进行操作的。
public class ContentLewen8ModelImpl implements IWebContentModel{
public static final String TAG = "http://www.lewen8.com";
/**
* ContentLewen8ModelImpl
* 便
*
*
* @return ContentLewen8ModelImpl
*/
public static ContentLewen8ModelImpl getInstance() {
return new ContentLewen8ModelImpl();
}
/**
*
* getInstance
*/
private ContentLewen8ModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'便
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般为HTML格式的网页内容解析成一个Document对象。
// Document对象基于DOM文档对象模型的结构来表示HTML文档通过它可以方便地按照HTML元素的层级关系、标签名称、属性等特征来查找、定位和操作网页中的各种元素
// 进而帮助我们提取出期望的书籍内容相关部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"content"的元素,
// 然后获取该元素所包含的所有文本节点TextNode列表。这里隐含的假设是网页中书籍的实际内容是存放在id为"content"这个元素内部的文本节点当中,
// 通过这种方式来确定要解析的文本内容在整个HTML文档中的具体位置。
List<TextNode> contentEs = doc.getElementById("content").textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相较于直接使用字符串相加操作使用StringBuilder可以显著减少创建多个临时字符串对象所带来的性能开销
// 尤其适用于像这种需要频繁进行字符串拼接的场景,能提升程序的运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有文本节点contentEs针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 首先获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包括空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,以便后续更清晰地处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符在HTML中常用于表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的常规要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件系统等。
return content.toString();
}
}
}

@ -7,90 +7,32 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentPbtxtModelImplIWebContentModelTAGhttp://www.pbtxt.com
* 便
*/
public class ContentPbtxtModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址此处明确为http://www.pbtxt.com
// 便于在整个程序体系里知晓此解析逻辑是专门针对该站点的书籍内容而设计的。
public class ContentPbtxtModelImpl implements IWebContentModel{
public static final String TAG = "http://www.pbtxt.com";
/**
* ContentPbtxtModelImpl
*
*
*
* @return ContentPbtxtModelImpl
*/
public static ContentPbtxtModelImpl getInstance() {
public static ContentPbtxtModelImpl getInstance(){
return new ContentPbtxtModelImpl();
}
/**
*
* getInstance
*/
private ContentPbtxtModelImpl() {
private ContentPbtxtModelImpl(){
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'HTML
*
* @param s HTML
* @param realUrl HTML
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般是HTML格式的网页内容解析为一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它能够按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
public String analyBookcontent(String s, String realUrl) throws Exception{
Document doc = Jsoup.parse(s);
// 从'realUrl'中提取书籍的唯一标识bookId此处是通过截取网址字符串的方式来获取。
// 先找到从第21个字符之后这里假设网址具有特定的结构实际应用中可能需要根据具体网址的格式特点来调整出现的第一个'/'的位置,
// 然后从该位置的下一个字符开始,截取到'.html'之前的字符串作为bookId后续将利用这个bookId来定位书籍内容所在的具体HTML元素。
String bookId = realUrl.substring(realUrl.indexOf("/", 21) + 1, realUrl.indexOf(".html"));
// 通过获取到的bookId在解析后的Document对象中查找id为"content" + bookId的元素并获取该元素包含的所有文本节点TextNode列表。
// 这里的假设是书籍内容存放在这个特定id的元素内的文本节点当中通过这样的操作来精准定位到需要解析的书籍内容所在位置。
List<TextNode> contentTNs = doc.getElementById("content" + bookId).textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相比于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
String bookId = realUrl.substring(realUrl.indexOf("/",21)+1,realUrl.indexOf(".html"));
List<TextNode> contentTNs = doc.getElementById("content"+bookId).textNodes();
StringBuilder stringBuilder = new StringBuilder();
// 开始遍历获取到的所有文本节点contentTNs针对每个文本节点进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentTNs.size(); i++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,方便后续进一步处理和展示。
for(int i=0;i<contentTNs.size();i++){
String temp = contentTNs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到stringBuilder这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
stringBuilder.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentTNs.size() - 1) {
stringBuilder.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return stringBuilder.toString();
}
}

@ -7,86 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentPpxsModelImplIWebContentModelTAGhttp://www.ppxs.net
* 便
*/
public class ContentPpxsModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址这里明确为http://www.ppxs.net
// 使得在整个程序架构中能够清晰知晓此解析逻辑是专门针对该站点的书籍内容而设置的。
public class ContentPpxsModelImpl implements IWebContentModel{
public static final String TAG = "http://www.ppxs.net";
/**
* ContentPpxsModelImpl
* 便
*
*
* @return ContentPpxsModelImpl
*/
public static ContentPpxsModelImpl getInstance() {
return new ContentPpxsModelImpl();
}
/**
*
* getInstance
*/
private ContentPpxsModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串通常为HTML格式的网页内容解析成一个Document对象。
// Document对象基于DOM文档对象模型来表示HTML文档通过它可以依据HTML元素的各种属性如id、class等、标签名称以及层级关系等
// 方便地查找、定位并操作网页中的各种元素,进而帮助我们准确地提取出与书籍内容相关的文本部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"booktext"的元素,
// 然后获取该元素包含的所有文本节点TextNode列表。这里假设书籍内容存放在id为"booktext"的这个元素内的文本节点当中,
// 通过这种方式来定位要解析的书籍内容在整个HTML文档中的具体位置。
List<TextNode> contentEs = doc.getElementById("booktext").textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 与直接使用字符串相加操作相比使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 尤其适用于像这样需要多次进行字符串拼接的场景,有助于提高程序的运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有文本节点contentEs针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,以便后续更好地进行处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使最终呈现的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -7,86 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentQulaModelImplIWebContentModelTAGhttp://www.qu.la
* 便
*/
public class ContentQulaModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该类所对应的网站地址这里明确指定为http://www.qu.la
// 这样在整个程序架构中就能清楚地知道这个解析类是针对该网站的书籍内容进行处理的。
public class ContentQulaModelImpl implements IWebContentModel{
public static final String TAG = "http://www.qu.la";
/**
* ContentQulaModelImpl
* 便
*
*
* @return ContentQulaModelImpl
*/
public static ContentQulaModelImpl getInstance() {
return new ContentQulaModelImpl();
}
/**
*
* getInstance
*/
private ContentQulaModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般为HTML格式的网页内容解析成一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它可以按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"content"的元素,
// 然后获取该元素包含的所有文本节点TextNode列表。这里假设网页中书籍的实际内容是存放在id为"content"这个元素内的文本节点当中,
// 通过这种方式来定位要解析的文本内容在整个HTML文档中的具体位置。
List<TextNode> contentEs = doc.getElementById("content").textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相较于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在像这样需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有文本节点contentEs针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,以便后续更好地进行处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -7,88 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentQzreadModelImplIWebContentModelTAGhttp://www.qzread.com
* HTML
*/
public class ContentQzreadModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于明确标识该类所对应的网站地址此处设定为http://www.qzread.com
// 这样在整个程序架构中就能清晰知晓此解析逻辑是专门针对该站点的书籍内容进行操作的。
public class ContentQzreadModelImpl implements IWebContentModel{
public static final String TAG = "http://www.qzread.com";
/**
* ContentQzreadModelImpl
* 便
*
*
* @return ContentQzreadModelImpl
*/
public static ContentQzreadModelImpl getInstance() {
return new ContentQzreadModelImpl();
}
/**
*
* getInstance
*/
private ContentQzreadModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'便
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般为HTML格式的网页内容解析成一个Document对象。
// Document对象基于DOM文档对象模型的结构来表示HTML文档通过它可以按照HTML元素的层级关系、标签名称、属性等特征来查找、定位和操作网页中的各种元素
// 进而帮助我们提取出期望的书籍内容相关部分。
Document doc = Jsoup.parse(s);
// 首先通过getElementsByClass方法依据元素的class属性查找网页中class为"txt"的所有元素,
// 然后取其中的第一个元素通过get(0)获取这是因为假设符合该class的元素中第一个包含了我们要解析的书籍内容相关结构。
// 在这个获取到的元素基础上再通过getElementsByTag方法依据标签名称查找其中的第一个<p>标签元素同样通过get(0)获取),
// 这里又假设书籍内容在这个特定的<p>标签元素内部。最后获取该<p>标签元素所包含的所有文本节点TextNode列表
// 通过这样层层查找的方式来确定要解析的文本内容在整个HTML文档中的具体位置。
List<TextNode> contentEs = doc.getElementsByClass("txt").get(0).getElementsByTag("p").get(0).textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相较于直接使用字符串相加操作使用StringBuilder可以显著减少创建多个临时字符串对象所带来的性能开销
// 尤其适用于像这种需要频繁进行字符串拼接的场景,能提升程序的运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有文本节点contentEs针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 首先获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包括空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,以便后续更清晰地处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符在HTML中常用于表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -7,86 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentRanwenaModelImplIWebContentModelTAGhttp://www.ranwena.com
* 便
*/
public class ContentRanwenaModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于明确标识该内容解析类对应的网站地址这里设定为http://www.ranwena.com
// 使得在整个程序架构中能够清楚知晓此解析逻辑是专门针对该站点的书籍内容而设计的。
public class ContentRanwenaModelImpl implements IWebContentModel{
public static final String TAG = "http://www.ranwena.com";
/**
* ContentRanwenaModelImpl
* 便
*
*
* @return ContentRanwenaModelImpl
*/
public static ContentRanwenaModelImpl getInstance() {
return new ContentRanwenaModelImpl();
}
/**
*
* getInstance
*/
private ContentRanwenaModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* idHTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般为HTML格式的网页内容解析为一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档借助它能够按照HTML元素的各种属性如id、class等、标签名称以及元素间的层级关系等
// 方便地查找、定位并操作网页中的各种元素,从而准确提取出与书籍内容相关的文本部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"content"的元素,
// 然后获取该元素所包含的所有文本节点TextNode列表。这里假设书籍内容存放在id为"content"这个元素内的文本节点当中,
// 通过这种方式来定位要解析的书籍内容在整个HTML文档中的具体位置。
List<TextNode> contentEs = doc.getElementById("content").textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相比于直接使用字符串相加操作使用StringBuilder能够避免频繁创建新的字符串对象所带来的性能开销
// 特别适用于像这样需要多次进行字符串拼接的场景,有助于提高程序的运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有文本节点contentEs针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,方便后续更好地进行处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使最终呈现的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -6,83 +6,23 @@ import org.jsoup.Jsoup;
import org.jsoup.nodes.Document;
import org.jsoup.nodes.Element;
/**
* ContentShulouModelImplIWebContentModelTAGhttp://www.shulou.cc
*
*/
public class ContentShulouModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址此处明确为http://www.shulou.cc
// 便于在整个程序体系里知晓此解析逻辑是专门针对该站点的书籍内容而设计的。
public class ContentShulouModelImpl implements IWebContentModel{
public static final String TAG = "http://www.shulou.cc";
/**
* ContentShulouModelImpl
*
*
*
* @return ContentShulouModelImpl
*/
public static ContentShulouModelImpl getInstance() {
public static ContentShulouModelImpl getInstance(){
return new ContentShulouModelImpl();
}
/**
*
* getInstance
*/
private ContentShulouModelImpl() {
private ContentShulouModelImpl(){
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return HTML使
* @throws Exception 使JsoupHTMLHTML
*
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般是HTML格式的网页内容解析为一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它能够按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
public String analyBookcontent(String s, String realUrl) throws Exception{
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"content"的元素。
// 这里假设网页中书籍的实际内容主要存放在这个id为"content"的元素内通过这种方式来定位要解析的书籍内容在整个HTML文档中的大致位置。
Element contentE = doc.getElementById("content");
// 将获取到的包含书籍内容相关信息的Element对象转换为字符串形式方便后续进行一系列的文本替换操作来清理和格式化内容。
String contentString = contentE.toString();
// 以下是一系列的字符串替换操作目的是去除内容字符串中不需要的空格、换行符、HTML中的特殊空格占位符&nbsp;以及一些不必要的HTML标签等
// 让文本内容更加纯净、格式更加规整,更符合阅读展示的要求。
// 去除所有普通空格
contentString = contentString.replaceAll(" ", "");
// 去除所有换行符
contentString = contentString.replaceAll("\n", "");
// 去除HTML中的不间断空格&nbsp;
contentString = contentString.replaceAll("&nbsp;", "");
// 去除开头的 <div id="content"> 标签,因为已经提取出了其内部的内容,这个标签在最终展示的文本中不需要出现。
contentString = contentString.replaceAll("<divid=\"content\">", "");
// 去除结尾的 </div> 标签同理这也是不需要在最终文本内容里呈现的HTML标签部分。
contentString = contentString.replaceAll("</div>", "");
// 去除空的 <p> 标签(<p></p>),这类空标签对书籍内容展示没有实际意义,去除可使内容更简洁。
contentString = contentString.replaceAll("<p></p>", "");
// 将连续的两个 <br> 标签通常用于HTML中表示换行替换为 \r\n\u3000\u3000
// 其中 \r\n 表示换行,\u3000\u3000 表示添加两个全角空格作为缩进,使文本排版更美观,符合阅读习惯。
contentString = contentString.replaceAll("<br><br>", "\r\n\u3000\u3000");
// 在整个处理后的内容字符串开头添加两个全角空格(\u3000\u3000作为缩进进一步优化排版效果使书籍内容看起来更整齐、美观。
contentString = contentString.replaceAll(" ", "").replaceAll("\n", "").replaceAll("&nbsp;", "").replaceAll("<divid=\"content\">", "").replaceAll("</div>", "").replaceAll("<p></p>","").replaceAll("<br><br>", "\r\n\u3000\u3000");
contentString = "\u3000\u3000" + contentString;
// 返回经过上述一系列处理后的书籍内容字符串,该字符串可用于后续在阅读应用等场景中的展示、存储等操作。
return contentString;
}
}
}

@ -7,86 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentSnwx8ModelImplIWebContentModelTAGhttp://www.snwx8.com
*
*/
public class ContentSnwx8ModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址这里明确为http://www.snwx8.com
// 便于在整个程序架构中知晓此解析逻辑是专门针对该站点的书籍内容而设计的。
public class ContentSnwx8ModelImpl implements IWebContentModel{
public static final String TAG = "http://www.snwx8.com";
/**
* ContentSnwx8ModelImpl
* 便
*
*
* @return ContentSnwx8ModelImpl
*/
public static ContentSnwx8ModelImpl getInstance() {
return new ContentSnwx8ModelImpl();
}
/**
*
* getInstance
*/
private ContentSnwx8ModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般为HTML格式的网页内容解析成一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它可以按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"BookText"的元素,
// 然后获取该元素包含的所有文本节点TextNode列表。这里假设网页中书籍的实际内容是存放在id为"BookText"这个元素内的文本节点当中,
// 通过这种方式来定位要解析的文本内容在整个HTML文档中的具体位置。
List<TextNode> contentEs = doc.getElementById("BookText").textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相较于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在像这样需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有文本节点contentEs针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,以便后续更好地进行处理和展示。
for(int i=0;i<contentEs.size();i++){
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if(temp.length()>0){
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -7,86 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentSuimengModelImplIWebContentModelTAGhttp://www.suimeng.la
* 便
*/
public class ContentSuimengModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该类所对应的网站地址这里明确指定为http://www.suimeng.la
// 这样在整个程序架构中就能清楚地知道这个解析类是针对该网站的书籍内容进行处理的。
public class ContentSuimengModelImpl implements IWebContentModel{
public static final String TAG = "http://www.suimeng.la";
/**
* ContentSuimengModelImpl
* 便
*
*
* @return ContentSuimengModelImpl
*/
public static ContentSuimengModelImpl getInstance() {
return new ContentSuimengModelImpl();
}
/**
*
* getInstance
*/
private ContentSuimengModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般为HTML格式的网页内容解析成一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它可以按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"ccontent"的元素,
// 然后获取该元素包含的所有文本节点TextNode列表。这里假设网页中书籍的实际内容是存放在id为"ccontent"这个元素内的文本节点当中,
// 通过这种方式来定位要解析的文本内容在整个HTML文档中的具体位置。
List<TextNode> contentEs = doc.getElementById("ccontent").textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相较于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在像这样需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有文本节点contentEs针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,以便后续更好地进行处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -8,85 +8,33 @@ import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentSyzwwModelImplIWebContentModelTAGhttp://www.syzww.net
*
*
*/
public class ContentSyzwwModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址此处明确为http://www.syzww.net
// 便于在整个程序体系中知晓此解析逻辑是针对该站点的书籍内容而设计的。
public class ContentSyzwwModelImpl implements IWebContentModel{
public static final String TAG = "http://www.syzww.net";
/**
* ContentSyzwwModelImpl
* 便
*
*
* @return ContentSyzwwModelImpl
*/
public static ContentSyzwwModelImpl getInstance() {
return new ContentSyzwwModelImpl();
}
/**
*
* getInstance
*/
private ContentSyzwwModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般是HTML格式的网页内容解析为一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它能够按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"content"的元素,
// 然后获取该元素包含的所有文本节点TextNode列表。这里假设网页中书籍的实际内容是存放在id为"content"这个元素内的文本节点当中,
// 通过这种方式来定位要解析的文本内容在整个HTML文档中的具体位置。
List<TextNode> contentEs = doc.getElementById("content").textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相比于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有文本节点contentEs针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,以便后续更好地进行处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -7,89 +7,34 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.Element;
import org.jsoup.select.Elements;
/**
* ContentVodtwModelImplIWebContentModelTAGhttp://www.vodtw.com
* 便
*/
public class ContentVodtwModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址此处明确为http://www.vodtw.com
// 便于在整个程序体系里知晓此解析逻辑是专门针对该站点的书籍内容而设计的。
public static final String TAG = "http://www.vodtw.com";
/**
* ContentVodtwModelImpl
*
*
*
* @return ContentVodtwModelImpl
*/
public static ContentVodtwModelImpl getInstance() {
return new ContentVodtwModelImpl();
}
/**
*
* getInstance
*/
private ContentVodtwModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般是HTML格式的网页内容解析为一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它能够按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"BookText"的元素。
// 这里假设网页中书籍的实际内容主要存放在这个id为"BookText"的元素内通过这种方式来定位要解析的书籍内容在整个HTML文档中的大致位置。
Element contentE = doc.getElementById("BookText");
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相比于直接使用字符串相加操作使用StringBuilder能够避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在像这样需要多次进行字符串拼接的场景,有助于提高程序的运行效率。
StringBuilder content = new StringBuilder();
// 通过获取到的contentE元素使用getElementsByTag方法依据标签名称"p"来查找其内部所有的<p>标签元素,
// 并将这些元素存储在Elements集合中这里假设书籍内容是分布在这些<p>标签内的文本信息,后续将从这些<p>标签元素中提取具体文本内容。
Elements contentEs = contentE.getElementsByTag("p");
// 开始遍历获取到的所有<p>标签元素contentEs针对每个元素依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前<p>标签元素的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,方便后续更好地进行处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该<p>标签元素包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前<p>标签元素不是遍历过程中的最后一个元素,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同<p>标签元素内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -7,86 +7,31 @@ import org.jsoup.nodes.Document;
import org.jsoup.nodes.TextNode;
import java.util.List;
/**
* ContentWxguanModelImplIWebContentModelTAGhttp://www.wxguan.com
* 便
*/
public class ContentWxguanModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于明确标识该内容解析类对应的网站地址这里设定为http://www.wxguan.com
// 便于在整个程序架构中知晓此解析逻辑是专门针对该站点的书籍内容而设计的。
public class ContentWxguanModelImpl implements IWebContentModel{
public static final String TAG = "http://www.wxguan.com";
/**
* ContentWxguanModelImpl
* 便
*
*
* @return ContentWxguanModelImpl
*/
public static ContentWxguanModelImpl getInstance() {
return new ContentWxguanModelImpl();
}
/**
*
* getInstance
*/
private ContentWxguanModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'便
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般为HTML格式的网页内容解析成一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它可以按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 进而帮助我们准确地提取出与书籍内容相关的部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"content"的元素,
// 然后获取该元素包含的所有文本节点TextNode列表。这里假设网页中书籍的实际内容是存放在id为"content"这个元素内的文本节点当中,
// 通过这种方式来定位要解析的文本内容在整个HTML文档中的具体位置。
List<TextNode> contentEs = doc.getElementById("content").textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相较于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在像这样需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有文本节点contentEs针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,以便后续更好地进行处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -6,87 +6,32 @@ import org.jsoup.Jsoup;
import org.jsoup.nodes.Document;
import org.jsoup.select.Elements;
/**
* ContentXqingdouCCModelImplIWebContentModelTAGhttp://www.xqingdou.cc
* 便
*/
public class ContentXqingdouCCModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址此处明确为http://www.xqingdou.cc
// 便于在整个程序体系里知晓此解析逻辑是专门针对该站点的书籍内容而设计的。
public static final String TAG = "http://www.xqingdou.cc";
/**
* ContentXqingdouCCModelImpl
*
*
*
* @return ContentXqingdouCCModelImpl
*/
public static ContentXqingdouCCModelImpl getInstance() {
return new ContentXqingdouCCModelImpl();
}
/**
*
* getInstance
*/
private ContentXqingdouCCModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般是HTML格式的网页内容解析为一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它能够按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
public String analyBookcontent(String s, String realUrl) throws Exception{
Document doc = Jsoup.parse(s);
// 首先通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"chapter_content"的元素,
// 这里假设书籍内容所在的区域是在这个id为"chapter_content"的元素内部。
// 然后在此元素基础上再使用getElementsByTag方法依据标签名称"p"来查找其内部所有的<p>标签元素,
// 并将这些<p>标签元素存储在Elements集合中通常会假设书籍的具体文本内容是分布在这些<p>标签当中,后续将从这些元素中提取具体的文本内容。
Elements contentEs = doc.getElementById("chapter_content").getElementsByTag("p");
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相比于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在像这样需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有<p>标签元素contentEs针对每个元素依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前<p>标签元素的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,方便后续更好地进行处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该<p>标签元素包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前<p>标签元素不是遍历过程中的最后一个元素,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同<p>标签元素内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -6,87 +6,32 @@ import org.jsoup.Jsoup;
import org.jsoup.nodes.Document;
import org.jsoup.select.Elements;
/**
* ContentXqingdouModelImplIWebContentModelTAGhttp://www.xqingdou.com
* HTML
*/
public class ContentXqingdouModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型对应的网站地址这里明确指定为http://www.xqingdou.com
// 便于在整个程序架构中知晓此解析逻辑是专门针对该站点书籍内容而设计的。
public static final String TAG = "http://www.xqingdou.com";
/**
* ContentXqingdouModelImpl
* 便
*
*
* @return ContentXqingdouModelImpl
*/
public static ContentXqingdouModelImpl getInstance() {
return new ContentXqingdouModelImpl();
}
/**
*
* getInstance
*/
private ContentXqingdouModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* ID
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串通常为HTML格式的网页内容解析成一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档借助它可以依据HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
public String analyBookcontent(String s, String realUrl) throws Exception{
Document doc = Jsoup.parse(s);
// 首先通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"chapter_content"的元素,
// 这里假设书籍内容所在的主要区域是在这个id为"chapter_content"的元素内部,这是定位书籍文本内容大致位置的第一步。
// 接着,在找到的"chapter_content"元素基础上调用getElementsByTag方法依据标签名称"p"来查找该元素内部所有的<p>标签元素,
// 并将这些<p>标签元素存储在Elements集合中通常认为书籍的具体文本内容是分布在这些<p>标签里面的,后续将从这些元素中提取具体文本内容进行整理。
Elements contentEs = doc.getElementById("chapter_content").getElementsByTag("p");
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相较于直接使用字符串相加操作使用StringBuilder能避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在像这样需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的所有<p>标签元素contentEs针对每个元素依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前<p>标签元素的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,以便后续更清晰地处理和展示。
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该<p>标签元素包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前<p>标签元素不是遍历过程中的最后一个元素,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同<p>标签元素内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -9,125 +9,52 @@ import org.jsoup.nodes.TextNode;
import org.jsoup.select.Elements;
import java.util.List;
/**
* ContentYb3ModelImplIWebContentModelTAGhttp://www.yb3.cc
*
* 便
*/
public class ContentYb3ModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址此处明确为http://www.yb3.cc
// 便于在整个程序体系里知晓此解析逻辑是专门针对该站点的书籍内容而设计的。
public class ContentYb3ModelImpl implements IWebContentModel{
public static final String TAG = "http://www.yb3.cc";
/**
* ContentYb3ModelImpl
*
*
*
* @return ContentYb3ModelImpl
*/
public static ContentYb3ModelImpl getInstance() {
return new ContentYb3ModelImpl();
}
/**
*
* getInstance
*/
private ContentYb3ModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
*
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般是HTML格式的网页内容解析为一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它能够按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"content"的元素,
// 这里假设书籍内容所在的主要区域是在这个id为"content"的元素内部,这是定位书籍文本内容大致位置的第一步。
Element contentE = doc.getElementById("content");
// 获取id为"content"的元素下包含的所有文本节点TextNode列表尝试第一种提取书籍内容的方式
// 即直接从该元素下的文本节点中获取文本内容,后续会判断这些文本节点是否符合要求来进行相应处理。
List<TextNode> contentTextNodes = contentE.textNodes();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相比于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在像这样需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
StringBuilder content = new StringBuilder();
// 首先判断获取到的文本节点列表是否不为空且包含至少一个文本节点,说明可以尝试从这些文本节点中提取内容。
if (contentTextNodes!= null && contentTextNodes.size() > 0) {
// 开始遍历获取到的所有文本节点contentTextNodes针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
if(contentTextNodes!=null && contentTextNodes.size()>0){
for (int i = 0; i < contentTextNodes.size(); i++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 再判断处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
if (contentTextNodes.get(i).text().trim().length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + contentTextNodes.get(i).text().trim());
// 如果当前文本节点不是遍历过程中的最后一个文本节点,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同文本节点内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentTextNodes.size() - 1) {
content.append("\r\n");
}
}
}
} else {
// 如果通过第一种方式(直接从文本节点获取内容)没有获取到有效内容,说明网页结构可能不太一样,
// 则尝试另一种方式获取id为"content"的元素的所有子元素children通过遍历这些子元素来提取可能存在的书籍内容文本。
}else{
Elements contentEs = contentE.children();
// 开始遍历这些子元素contentEs尝试从每个子元素中提取和整理书籍内容文本。
for (int i = 0; i < contentEs.size(); i++) {
// 获取当前子元素的文本内容转换为字符串后调用trim方法去除两端空白字符再将其中的全角空格Unicode编码对应的“ ”字符替换为空字符串
// 然后判断处理后的文本内容长度是否大于0如果大于0则说明该子元素可能包含有实际有效的信息需要进一步处理来提取其中的书籍内容文本。
if (contentEs.get(i).text().toString().trim().replaceAll(" ", "").length() > 0) {
// 获取当前子元素下包含的所有文本节点TextNode列表尝试从这些文本节点中提取更细化的书籍内容文本信息。
if (contentEs.get(i).text().toString().trim().replaceAll(" ","").length() > 0) {
List<TextNode> tempTextNodes = contentEs.get(i).textNodes();
// 判断获取到的文本节点列表是否不为空且包含至少一个文本节点,如果是,则进行相应的文本提取和整理操作。
if (tempTextNodes!= null && tempTextNodes.size() > 0) {
// 开始遍历当前子元素下的所有文本节点tempTextNodes针对每个文本节点依次进行相应的处理和拼接操作以构建最终的书籍内容字符串。
if(tempTextNodes!=null && tempTextNodes.size()>0){
for (int j = 0; j < tempTextNodes.size(); j++) {
// 获取当前文本节点的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
String temp = tempTextNodes.get(j).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该文本节点包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 添加一个换行符(\r\n实现不同文本节点内容之间的换行让书籍内容在展示时更有条理。
content.append("\r\n");
}
}
} else {
// 如果当前子元素下没有文本节点(或者文本节点列表为空),但该子元素本身的文本内容经过前面处理后还有效,
// 则直接将其文本内容去除全角空格后添加到content这个StringBuilder对象中并添加一个换行符\r\n实现换行效果。
content.append("\u3000\u3000" + contentEs.get(i).text().trim().replaceAll(" ", ""));
}else{
content.append("\u3000\u3000" + contentEs.get(i).text().trim().replaceAll(" ",""));
content.append("\r\n");
}
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -6,88 +6,31 @@ import org.jsoup.Jsoup;
import org.jsoup.nodes.Document;
import org.jsoup.select.Elements;
/**
* ContentZhulangModelImplIWebContentModelTAGhttp://book.zhulang.com
*
*/
public class ContentZhulangModelImpl implements IWebContentModel {
// 定义一个静态常量TAG用于标识该内容解析模型所对应的网站地址此处明确为http://book.zhulang.com
// 便于在整个程序体系里知晓此解析逻辑是专门针对该站点的书籍内容而设计的。
public class ContentZhulangModelImpl implements IWebContentModel{
public static final String TAG = "http://book.zhulang.com";
/**
* ContentZhulangModelImpl
*
*
*
* @return ContentZhulangModelImpl
*/
public static ContentZhulangModelImpl getInstance() {
return new ContentZhulangModelImpl();
}
/**
*
* getInstance
*/
private ContentZhulangModelImpl() {
}
/**
* IWebContentModelanalyBookcontent
* 's'HTML
* 'realUrl'
*
* @param s HTML
* @param realUrl
* @return 使
* @throws Exception 使JsoupHTMLHTML
* HTML
*/
@Override
public String analyBookcontent(String s, String realUrl) throws Exception {
// 使用Jsoup库提供的parse方法将传入的原始字符串一般是HTML格式的网页内容解析为一个Document对象。
// Document对象基于DOM文档对象模型结构来表示HTML文档通过它能够按照HTML元素的标签、属性、层级关系等特征方便地查找、定位和操作网页中的各种元素
// 从而帮助我们准确地提取出与书籍内容相关的部分。
Document doc = Jsoup.parse(s);
// 通过调用Document对象的getElementById方法依据元素的id属性查找网页中id为"read-content"的元素,
// 然后获取该元素的所有子元素children并将这些子元素存储在Elements集合中这里假设书籍内容是分布在这些子元素当中
// 通过后续遍历这些子元素来提取具体的文本内容。
Elements contentEs = doc.getElementById("read-content").children();
// 创建一个StringBuilder对象用于高效地拼接整理后的书籍内容字符串。
// 相比于直接使用字符串相加操作使用StringBuilder可以避免频繁创建新的字符串对象所带来的性能开销
// 特别适合在像这样需要多次进行字符串拼接的场景下使用,有助于提升程序运行效率。
StringBuilder content = new StringBuilder();
// 开始遍历获取到的子元素contentEs不过这里注意起始索引为3结束索引为contentEs.size() - 1
// 说明可能跳过了前面几个特定的元素(可能是网页中一些非书籍正文内容的元素,比如标题、导航栏之类的相关元素),
// 同时也排除了最后一个元素(可能有一些特殊格式或者不属于正文内容的元素),仅对中间认为是书籍正文内容的子元素进行处理。
for (int i = 3; i < contentEs.size() - 1; i++) {
// 获取当前子元素的文本内容然后调用trim方法去除文本内容两端的空白字符包含空格、制表符、换行等
// 得到相对纯净、没有多余空白干扰的文本内容,方便后续更好地进行处理和展示。
for (int i = 3; i < contentEs.size()-1; i++) {
String temp = contentEs.get(i).text().trim();
// 进一步对处理后的文本内容进行清理将其中的不间断空格Unicode编码对应的“&nbsp;”字符常用于HTML中表示空格占位等情况以及普通空格都替换为空字符串
// 这样可以让文本的格式更加规整,避免因多余的空白间隔而影响阅读体验以及内容展示的美观性。
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
// 判断经过上述处理后的文本内容长度是否大于0如果大于0则说明该子元素包含有实际有效的信息需要将其添加到content这个StringBuilder对象中。
temp = temp.replaceAll(" ","").replaceAll(" ","");
if (temp.length() > 0) {
// 在每个有效的文本内容前面添加两个全角空格(\u3000作为缩进格式使得最终呈现出来的书籍内容在排版上更加美观
// 更符合一般阅读时对于段落格式的要求,便于读者区分不同的内容段落。
content.append("\u3000\u3000" + temp);
// 如果当前子元素不是遍历过程中的最后一个子元素,那么就在添加的文本内容后面添加一个换行符(\r\n
// 以此实现不同子元素内容之间的换行,让书籍内容在展示时更有条理,方便读者清晰地阅读和理解。
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
// 最后通过调用StringBuilder对象的toString方法将经过一系列处理和拼接后存储在其中的书籍内容字符串转换为普通字符串格式
// 并将这个整理好的书籍内容字符串返回,该返回值可以用于后续的各种操作,比如在阅读应用程序中展示给用户查看、保存到本地文件等。
return content.toString();
}
}
}

@ -33,61 +33,32 @@ import io.reactivex.android.schedulers.AndroidSchedulers;
import io.reactivex.functions.Function;
import io.reactivex.schedulers.Schedulers;
/**
* GxwztvBookModelImplMBaseModelImplIGxwztvBookModelTAGhttp://www.gxwztv.com
* HTML
* RxJava
*/
public class GxwztvBookModelImpl extends MBaseModelImpl implements IGxwztvBookModel {
// 定义一个静态常量TAG用于标识该模型所对应的网站地址此处明确为http://www.gxwztv.com
// 便于在整个程序体系里知晓此模型相关的网络操作、数据解析等逻辑是针对该站点而设计的。
public static final String TAG = "http://www.gxwztv.com";
/**
* GxwztvBookModelImpl
*
*
*
* @return GxwztvBookModelImpl
*/
public static GxwztvBookModelImpl getInstance() {
return new GxwztvBookModelImpl();
}
/**
*
*
* Observable
* 便RxJava
*
* @param aCache ACache便访
* @return ObservableLibraryBean
*/
@Override
public Observable<LibraryBean> getLibraryData(final ACache aCache) {
return getRetrofitObject(TAG).create(IGxwztvApi.class).getLibraryData("")
// 使用flatMap操作符将网络请求获取到的字符串数据进行进一步处理先尝试缓存数据再进行解析操作
.flatMap(new Function<String, ObservableSource<LibraryBean>>() {
@Override
public ObservableSource<LibraryBean> apply(String s) throws Exception {
// 如果获取到的数据不为空且长度大于0并且缓存对象不为空则将数据存入缓存缓存的键由LibraryPresenterImpl.LIBRARY_CACHE_KEY指定
if (s!= null && s.length() > 0 && aCache!= null) {
aCache.put(LibraryPresenterImpl.LIBRARY_CACHE_KEY, s);
}
// 调用解析主页数据的方法将获取到的数据传入进行解析并返回解析后的Observable对象
return analyLibraryData(s);
}
});
return getRetrofitObject(TAG).create(IGxwztvApi.class).getLibraryData("").flatMap(new Function<String, ObservableSource<LibraryBean>>() {
@Override
public ObservableSource<LibraryBean> apply(String s) throws Exception {
if (s != null && s.length() > 0 && aCache != null) {
aCache.put(LibraryPresenterImpl.LIBRARY_CACHE_KEY, s);
}
return analyLibraryData(s);
}
});
}
/**
*
* ObservableHTML
* LibraryBeanObservableEmitter
*
* @param data HTML
* @return ObservableLibraryBean
*/
@Override
public Observable<LibraryBean> analyLibraryData(final String data) {
@ -95,35 +66,24 @@ public class GxwztvBookModelImpl extends MBaseModelImpl implements IGxwztvBookMo
@Override
public void subscribe(ObservableEmitter<LibraryBean> e) throws Exception {
LibraryBean result = new LibraryBean();
// 使用Jsoup库将传入的HTML格式字符串数据解析为Document对象以便后续按照DOM结构查找和提取元素及内容
Document doc = Jsoup.parse(data);
// 通过元素的类名获取名为"container"的元素,这里假设主页的主要内容都在这个元素内部,是后续提取各种信息的基础元素
Element contentE = doc.getElementsByClass("container").get(0);
// 解析最新书籍信息
// 获取所有类名为"list-group-item text-nowrap modal-open"的元素,这些元素被认为包含了最新书籍的相关信息
//解析最新书籍
Elements newBookEs = contentE.getElementsByClass("list-group-item text-nowrap modal-open");
List<LibraryNewBookBean> libraryNewBooks = new ArrayList<LibraryNewBookBean>();
for (int i = 0; i < newBookEs.size(); i++) {
// 从每个包含最新书籍信息的元素中获取<a>标签元素,通常<a>标签包含了书籍的链接、书名等关键信息
Element itemE = newBookEs.get(i).getElementsByTag("a").get(0);
// 创建一个LibraryNewBookBean对象将书名、书籍链接等信息封装进去其中链接是在TAG基础上拼接获取到的相对链接
LibraryNewBookBean item = new LibraryNewBookBean(itemE.text(), TAG + itemE.attr("href"), TAG, "gxwztv.com");
libraryNewBooks.add(item);
}
result.setLibraryNewBooks(libraryNewBooks);
// 解析不同分类书籍信息,先初始化一个空的列表用于存放不同分类的书籍列表信息
//////////////////////////////////////////////////////////////////////
List<LibraryKindBookListBean> kindBooks = new ArrayList<LibraryKindBookListBean>();
// 解析男频女频分类书籍信息
// 获取所有类名为"col-xs-12"的元素,这里假设这些元素包含了男频女频相关的分类信息
//解析男频女频
Elements hotEs = contentE.getElementsByClass("col-xs-12");
for (int i = 1; i < hotEs.size(); i++) {
LibraryKindBookListBean kindItem = new LibraryKindBookListBean();
// 获取分类名称,即通过类名为"panel-title"的元素获取其文本内容作为分类名
kindItem.setKindName(hotEs.get(i).getElementsByClass("panel-title").get(0).text());
// 获取该分类下的书籍元素列表,通过先获取类名为"panel-body"的元素,再获取其内部的<li>标签元素来定位书籍信息所在元素
Elements bookEs = hotEs.get(i).getElementsByClass("panel-body").get(0).getElementsByTag("li");
List<SearchBookBean> books = new ArrayList<SearchBookBean>();
@ -131,59 +91,47 @@ public class GxwztvBookModelImpl extends MBaseModelImpl implements IGxwztvBookMo
SearchBookBean searchBookBean = new SearchBookBean();
searchBookBean.setOrigin("gxwztv.com");
searchBookBean.setTag(TAG);
// 获取书籍名称,通过获取<span>标签的文本内容作为书名
searchBookBean.setName(bookEs.get(j).getElementsByTag("span").get(0).text());
// 获取书籍详情链接同样是在TAG基础上拼接相对链接
searchBookBean.setNoteUrl(TAG + bookEs.get(j).getElementsByTag("a").get(0).attr("href"));
// 获取书籍封面链接,通过获取<img>标签的src属性值作为封面链接
searchBookBean.setCoverUrl(bookEs.get(j).getElementsByTag("img").get(0).attr("src"));
books.add(searchBookBean);
}
kindItem.setBooks(books);
kindBooks.add(kindItem);
}
// 解析部分分类推荐书籍信息
// 获取所有类名为"panel panel-info index-category-qk"的元素,这些元素包含了部分分类推荐的相关信息
//解析部分分类推荐
Elements kindEs = contentE.getElementsByClass("panel panel-info index-category-qk");
for (int i = 0; i < kindEs.size(); i++) {
LibraryKindBookListBean kindItem = new LibraryKindBookListBean();
kindItem.setKindName(kindEs.get(i).getElementsByClass("panel-title").get(0).text());
// 获取分类链接同样是在TAG基础上拼接相对链接
kindItem.setKindUrl(TAG + kindEs.get(i).getElementsByClass("listMore").get(0).getElementsByTag("a").get(0).attr("href"));
List<SearchBookBean> books = new ArrayList<SearchBookBean>();
// 获取第一个推荐书籍的元素,这里通过<dl>标签来定位
Element firstBookE = kindEs.get(i).getElementsByTag("dl").get(0);
SearchBookBean firstBook = new SearchBookBean();
firstBook.setTag(TAG);
firstBook.setOrigin("gxwztv.com");
// 获取书籍名称,通过获取<a>标签的文本内容作为书名(这里取第二个<a>标签的文本,可能是根据网页结构来确定的具体书名所在位置)
firstBook.setName(firstBookE.getElementsByTag("a").get(1).text());
// 获取书籍详情链接在TAG基础上拼接相对链接
firstBook.setNoteUrl(TAG + firstBookE.getElementsByTag("a").get(0).attr("href"));
// 获取书籍封面链接,通过获取<a>标签下的<img>标签的src属性值作为封面链接
firstBook.setCoverUrl(firstBookE.getElementsByTag("a").get(0).getElementsByTag("img").get(0).attr("src"));
firstBook.setKind(kindItem.getKindName());
books.add(firstBook);
// 获取其他推荐书籍的元素列表,通过类名为"book_textList"的元素下的<li>标签来定位
Elements otherBookEs = kindEs.get(i).getElementsByClass("book_textList").get(0).getElementsByTag("li");
for (int j = 0; j < otherBookEs.size(); j++) {
SearchBookBean item = new SearchBookBean();
item.setTag(TAG);
item.setOrigin("gxwztv.com");
item.setKind(kindItem.getKindName());
item.setNoteUrl(TAG + otherBookEs.get(j).getElementsByTag("a").get(0).attr("href"));
item.setNoteUrl(TAG+otherBookEs.get(j).getElementsByTag("a").get(0).attr("href"));
item.setName(otherBookEs.get(j).getElementsByTag("a").get(0).text());
books.add(item);
}
kindItem.setBooks(books);
kindBooks.add(kindItem);
}
//////////////
result.setKindBooks(kindBooks);
// 通过ObservableEmitter发射解析好的LibraryBean对象包含了主页上的各种书籍相关信息
e.onNext(result);
e.onComplete();
}
@ -191,58 +139,33 @@ public class GxwztvBookModelImpl extends MBaseModelImpl implements IGxwztvBookMo
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
/**
*
*
* SearchBookBeanObservable
*
* @param content
* @param page
* @return ObservableList<SearchBookBean>
*/
@Override
public Observable<List<SearchBookBean>> searchBook(String content, int page) {
return getRetrofitObject(TAG).create(IGxwztvApi.class).searchBook(content, page)
.flatMap(new Function<String, ObservableSource<List<SearchBookBean>>>() {
@Override
public ObservableSource<List<SearchBookBean>> apply(String s) throws Exception {
return analySearchBook(s);
}
});
return getRetrofitObject(TAG).create(IGxwztvApi.class).searchBook(content, page).flatMap(new Function<String, ObservableSource<List<SearchBookBean>>>() {
@Override
public ObservableSource<List<SearchBookBean>> apply(String s) throws Exception {
return analySearchBook(s);
}
});
}
/**
*
* HTML
* SearchBookBeanObservableEmitter
*
* @param s HTML
* @return ObservableList<SearchBookBean>
*/
public Observable<List<SearchBookBean>> analySearchBook(final String s) {
return Observable.create(new ObservableOnSubscribe<List<SearchBookBean>>() {
@Override
public void subscribe(ObservableEmitter<List<SearchBookBean>> e) throws Exception {
try {
Document doc = Jsoup.parse(s);
// 获取id为"novel-list"的元素下类名为"list-group-item clearfix"的元素列表,这里假设这些元素包含了搜索到的每本书籍的相关信息
Elements booksE = doc.getElementById("novel-list").getElementsByClass("list-group-item clearfix");
if (null!= booksE && booksE.size() >= 2) {
if (null != booksE && booksE.size() >= 2) {
List<SearchBookBean> books = new ArrayList<SearchBookBean>();
for (int i = 1; i < booksE.size(); i++) {
SearchBookBean item = new SearchBookBean();
item.setTag(TAG);
// 获取书籍作者信息,通过类名为"col-xs-2"的元素获取其文本内容作为作者名
item.setAuthor(booksE.get(i).getElementsByClass("col-xs-2").get(0).text());
// 获取书籍分类信息,通过类名为"col-xs-1"的元素获取其文本内容作为分类名
item.setKind(booksE.get(i).getElementsByClass("col-xs-1").get(0).text());
// 获取书籍最后章节信息,通过类名为"col-xs-4"的元素下的<a>标签获取其文本内容作为最后章节名
item.setLastChapter(booksE.get(i).getElementsByClass("col-xs-4").get(0).getElementsByTag("a").get(0).text());
item.setOrigin("gxwztv.com");
// 获取书籍名称,通过类名为"col-xs-3"的元素下的<a>标签获取其文本内容作为书名
item.setName(booksE.get(i).getElementsByClass("col-xs-3").get(0).getElementsByTag("a").get(0).text());
// 获取书籍详情链接在TAG基础上拼接相对链接
item.setNoteUrl(TAG + booksE.get(i).getElementsByClass("col-xs-3").get(0).getElementsByTag("a").get(0).attr("href"));
item.setCoverUrl("noimage");
books.add(item);
@ -260,8 +183,171 @@ public class GxwztvBookModelImpl extends MBaseModelImpl implements IGxwztvBookMo
});
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
@Override
public Observable<BookShelfBean> getBookInfo(final BookShelfBean bookShelfBean) {
return getRetrofitObject(TAG).create(IGxwztvApi.class).getBookInfo(bookShelfBean.getNoteUrl().replace(TAG, "")).flatMap(new Function<String, ObservableSource<BookShelfBean>>() {
@Override
public ObservableSource<BookShelfBean> apply(String s) throws Exception {
return analyBookInfo(s, bookShelfBean);
}
});
}
private Observable<BookShelfBean> analyBookInfo(final String s, final BookShelfBean bookShelfBean) {
return Observable.create(new ObservableOnSubscribe<BookShelfBean>() {
@Override
public void subscribe(ObservableEmitter<BookShelfBean> e) throws Exception {
bookShelfBean.setTag(TAG);
bookShelfBean.setBookInfoBean(analyBookinfo(s, bookShelfBean.getNoteUrl()));
e.onNext(bookShelfBean);
e.onComplete();
}
});
}
private BookInfoBean analyBookinfo(String s, String novelUrl) {
BookInfoBean bookInfoBean = new BookInfoBean();
bookInfoBean.setNoteUrl(novelUrl); //id
bookInfoBean.setTag(TAG);
Document doc = Jsoup.parse(s);
Element resultE = doc.getElementsByClass("panel panel-warning").get(0);
bookInfoBean.setCoverUrl(resultE.getElementsByClass("panel-body").get(0).getElementsByClass("img-thumbnail").get(0).attr("src"));
bookInfoBean.setName(resultE.getElementsByClass("active").get(0).text());
bookInfoBean.setAuthor(resultE.getElementsByClass("col-xs-12 list-group-item no-border").get(0).getElementsByTag("small").get(0).text());
Element introduceE = resultE.getElementsByClass("panel panel-default mt20").get(0);
String introduce = "";
if (introduceE.getElementById("all") != null) {
introduce = introduceE.getElementById("all").text().replace("[收起]", "");
} else {
introduce = introduceE.getElementById("shot").text();
}
bookInfoBean.setIntroduce("\u3000\u3000" + introduce);
bookInfoBean.setChapterUrl(TAG + resultE.getElementsByClass("list-group-item tac").get(0).getElementsByTag("a").get(0).attr("href"));
bookInfoBean.setOrigin("gxwztv.com");
return bookInfoBean;
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
@Override
public void getChapterList(final BookShelfBean bookShelfBean, final OnGetChapterListListener getChapterListListener) {
getRetrofitObject(TAG).create(IGxwztvApi.class).getChapterList(bookShelfBean.getBookInfoBean().getChapterUrl().replace(TAG, "")).flatMap(new Function<String, ObservableSource<WebChapterBean<BookShelfBean>>>() {
@Override
public ObservableSource<WebChapterBean<BookShelfBean>> apply(String s) throws Exception {
return analyChapterList(s, bookShelfBean);
}
})
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new SimpleObserver<WebChapterBean<BookShelfBean>>() {
@Override
public void onNext(WebChapterBean<BookShelfBean> value) {
if (getChapterListListener != null) {
getChapterListListener.success(value.getData());
}
}
@Override
public void onError(Throwable e) {
e.printStackTrace();
if (getChapterListListener != null) {
getChapterListListener.error();
}
}
});
}
private Observable<WebChapterBean<BookShelfBean>> analyChapterList(final String s, final BookShelfBean bookShelfBean) {
return Observable.create(new ObservableOnSubscribe<WebChapterBean<BookShelfBean>>() {
@Override
public void subscribe(ObservableEmitter<WebChapterBean<BookShelfBean>> e) throws Exception {
bookShelfBean.setTag(TAG);
WebChapterBean<List<ChapterListBean>> temp = analyChapterlist(s, bookShelfBean.getNoteUrl());
bookShelfBean.getBookInfoBean().setChapterlist(temp.getData());
e.onNext(new WebChapterBean<BookShelfBean>(bookShelfBean, temp.getNext()));
e.onComplete();
}
});
}
private WebChapterBean<List<ChapterListBean>> analyChapterlist(String s, String novelUrl) {
Document doc = Jsoup.parse(s);
Elements chapterlist = doc.getElementById("chapters-list").getElementsByTag("a");
List<ChapterListBean> chapterBeans = new ArrayList<ChapterListBean>();
for (int i = 0; i < chapterlist.size(); i++) {
ChapterListBean temp = new ChapterListBean();
temp.setDurChapterUrl(TAG + chapterlist.get(i).attr("href")); //id
temp.setDurChapterIndex(i);
temp.setDurChapterName(chapterlist.get(i).text());
temp.setNoteUrl(novelUrl);
temp.setTag(TAG);
chapterBeans.add(temp);
}
Boolean next = false;
return new WebChapterBean<List<ChapterListBean>>(chapterBeans, next);
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
@Override
public Observable<BookContentBean> getBookContent(final String durChapterUrl, final int durChapterIndex) {
return getRetrofitObject(TAG).create(IGxwztvApi.class).getBookContent(durChapterUrl.replace(TAG, "")).flatMap(new Function<String, ObservableSource<BookContentBean>>() {
@Override
public ObservableSource<BookContentBean> apply(String s) throws Exception {
return analyBookContent(s, durChapterUrl, durChapterIndex);
}
});
}
private Observable<BookContentBean> analyBookContent(final String s, final String durChapterUrl, final int durChapterIndex) {
return Observable.create(new ObservableOnSubscribe<BookContentBean>() {
@Override
public void subscribe(ObservableEmitter<BookContentBean> e) throws Exception {
BookContentBean bookContentBean = new BookContentBean();
bookContentBean.setDurChapterIndex(durChapterIndex);
bookContentBean.setDurChapterUrl(durChapterUrl);
bookContentBean.setTag(TAG);
try {
Document doc = Jsoup.parse(s);
List<TextNode> contentEs = doc.getElementById("txtContent").textNodes();
StringBuilder content = new StringBuilder();
for (int i = 0; i < contentEs.size(); i++) {
String temp = contentEs.get(i).text().trim();
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
if (temp.length() > 0) {
content.append("\u3000\u3000" + temp);
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
bookContentBean.setDurCapterContent(content.toString());
bookContentBean.setRight(true);
} catch (Exception ex) {
ex.printStackTrace();
ErrorAnalyContentManager.getInstance().writeNewErrorUrl(durChapterUrl);
bookContentBean.setDurCapterContent(durChapterUrl.substring(0, durChapterUrl.indexOf('/', 8)) + "站点暂时不支持解析请反馈给Monke QQ:1105075896,半小时内解决,超级效率的程序员");
bookContentBean.setRight(false);
}
e.onNext(bookContentBean);
e.onComplete();
}
});
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
/**
*
* BookShelfBean
/**
*
*/
@Override
public Observable<List<SearchBookBean>> getKindBook(String url, int page) {
url = url + page + ".htm";
return getRetrofitObject(GxwztvBookModelImpl.TAG).create(IGxwztvApi.class).getKindBooks(url.replace(GxwztvBookModelImpl.TAG, "")).flatMap(new Function<String, ObservableSource<List<SearchBookBean>>>() {
@Override
public ObservableSource<List<SearchBookBean>> apply(String s) throws Exception {
return analySearchBook(s);
}
});
}
}

@ -25,68 +25,36 @@ import io.reactivex.Observable;
import io.reactivex.ObservableEmitter;
import io.reactivex.ObservableOnSubscribe;
/**
* ImportBookModelImplMBaseModelImplIImportBookModel
* MD5
* RxJavaObservable便
*/
public class ImportBookModelImpl extends MBaseModelImpl implements IImportBookModel {
/**
* ImportBookModelImpl
* 便
*
*
* @return ImportBookModelImpl
*/
public static ImportBookModelImpl getInstance() {
return new ImportBookModelImpl();
}
/**
*
* MD5
* LocBookShelfBeanObservable
*
* @param book File
* @return ObservableLocBookShelfBean
*/
@Override
public Observable<LocBookShelfBean> importBook(final File book) {
return Observable.create(new ObservableOnSubscribe<LocBookShelfBean>() {
@Override
public void subscribe(ObservableEmitter<LocBookShelfBean> e) throws Exception {
// 创建MessageDigest实例用于计算文件的MD5值MD5值可作为书籍文件的唯一标识来判断是否重复导入等情况
MessageDigest md = MessageDigest.getInstance("MD5");
// 创建文件输入流,用于读取书籍文件内容
FileInputStream in = new FileInputStream(book);
byte[] buffer = new byte[2048];
int len;
// 循环读取文件内容每次读取指定长度的数据并将其更新到MessageDigest中用于计算MD5值
while ((len = in.read(buffer, 0, 2048))!= -1) {
while ((len = in.read(buffer, 0, 2048)) != -1) {
md.update(buffer, 0, len);
}
// 关闭文件输入流,释放资源
in.close();
in = null;
// 将计算得到的MD5值转换为十六进制字符串形式方便后续作为标识使用
String md5 = new BigInteger(1, md.digest()).toString(16);
BookShelfBean bookShelfBean = null;
// 从数据库中查询是否已存在具有相同MD5值即相同书籍文件的记录通过构建查询条件并执行查询获取结果列表
List<BookShelfBean> temp = DbHelper.getInstance().getmDaoSession().getBookShelfBeanDao().queryBuilder()
.where(BookShelfBeanDao.Properties.NoteUrl.eq(md5)).build().list();
// 标记书籍是否为新导入的初始化为true表示默认是新书
List<BookShelfBean> temp = DbHelper.getInstance().getmDaoSession().getBookShelfBeanDao().queryBuilder().where(BookShelfBeanDao.Properties.NoteUrl.eq(md5)).build().list();
Boolean isNew = true;
// 如果查询到的结果列表不为空且包含至少一个元素,说明书籍已存在,不是新书
if (temp!= null && temp.size() > 0) {
if (temp!=null && temp.size()>0) {
isNew = false;
bookShelfBean = temp.get(0);
// 从数据库中获取对应的书籍详情信息并设置到bookShelfBean对象中确保书籍相关信息的完整性
bookShelfBean.setBookInfoBean(DbHelper.getInstance().getmDaoSession().getBookInfoBeanDao().queryBuilder()
.where(BookInfoBeanDao.Properties.NoteUrl.eq(bookShelfBean.getNoteUrl())).build().list().get(0));
bookShelfBean.setBookInfoBean(DbHelper.getInstance().getmDaoSession().getBookInfoBeanDao().queryBuilder().where(BookInfoBeanDao.Properties.NoteUrl.eq(bookShelfBean.getNoteUrl())).build().list().get(0));
} else {
// 如果是新书则创建一个新的BookShelfBean对象用于存储书籍的基本信息
bookShelfBean = new BookShelfBean();
bookShelfBean.setFinalDate(System.currentTimeMillis());
bookShelfBean.setDurChapter(0);
@ -94,7 +62,6 @@ public class ImportBookModelImpl extends MBaseModelImpl implements IImportBookMo
bookShelfBean.setTag(BookShelfBean.LOCAL_TAG);
bookShelfBean.setNoteUrl(md5);
// 设置书籍详情信息中的作者、书名等基本信息,书名通过去除文件扩展名来获取,作者暂设为“佚名”
bookShelfBean.getBookInfoBean().setAuthor("佚名");
bookShelfBean.getBookInfoBean().setName(book.getName().replace(".txt", "").replace(".TXT", ""));
bookShelfBean.getBookInfoBean().setFinalRefreshData(System.currentTimeMillis());
@ -102,31 +69,17 @@ public class ImportBookModelImpl extends MBaseModelImpl implements IImportBookMo
bookShelfBean.getBookInfoBean().setNoteUrl(md5);
bookShelfBean.getBookInfoBean().setTag(BookShelfBean.LOCAL_TAG);
// 调用方法保存书籍的章节信息到数据库中
saveChapter(book, md5);
// 将书籍详情信息插入或替换到数据库中(如果已存在相同标识的记录则替换)
DbHelper.getInstance().getmDaoSession().getBookInfoBeanDao().insertOrReplace(bookShelfBean.getBookInfoBean());
// 将书籍基本信息插入或替换到数据库中
DbHelper.getInstance().getmDaoSession().getBookShelfBeanDao().insertOrReplace(bookShelfBean);
}
// 从数据库中获取该书籍的章节列表信息并设置到bookShelfBean对象的对应属性中确保书籍章节信息的完整性
bookShelfBean.getBookInfoBean().setChapterlist(DbHelper.getInstance().getmDaoSession().getChapterListBeanDao().queryBuilder()
.where(ChapterListBeanDao.Properties.NoteUrl.eq(bookShelfBean.getNoteUrl())).orderAsc(ChapterListBeanDao.Properties.DurChapterIndex).build().list());
// 通过ObservableEmitter发射包含是否为新书以及完整书籍信息的LocBookShelfBean对象供订阅者接收处理
e.onNext(new LocBookShelfBean(isNew, bookShelfBean));
bookShelfBean.getBookInfoBean().setChapterlist(DbHelper.getInstance().getmDaoSession().getChapterListBeanDao().queryBuilder().where(ChapterListBeanDao.Properties.NoteUrl.eq(bookShelfBean.getNoteUrl())).orderAsc(ChapterListBeanDao.Properties.DurChapterIndex).build().list());
e.onNext(new LocBookShelfBean(isNew,bookShelfBean));
e.onComplete();
}
});
}
/**
* BookShelfBeanshelfs
* NoteUrl
*
* @param temp BookShelfBean
* @param shelfs List<BookShelfBean>
* @return truefalse
*/
private Boolean isAdded(BookShelfBean temp, List<BookShelfBean> shelfs) {
if (shelfs == null || shelfs.size() == 0) {
return false;
@ -146,68 +99,47 @@ public class ImportBookModelImpl extends MBaseModelImpl implements IImportBookMo
}
}
/**
*
*
* ChapterListBean
*
* @param book File
* @param md5 MD5
* @throws IOException I/O
*/
private void saveChapter(File book, String md5) throws IOException {
// 定义一个正则表达式用于匹配章节标题的模式这里假设章节标题一般以“第X章”开头X表示数字及一些其他可能的字符长度限制在1到7之间后面可跟一些其他内容
String regex = "第.{1,7}章.{0,}";
String encoding;
// 创建文件输入流,用于读取书籍文件内容,为后续检测编码格式和解析内容做准备
FileInputStream fis = new FileInputStream(book);
byte[] buf = new byte[4096];
// 创建UniversalDetector实例用于检测文件的编码格式它可以自动识别多种常见的编码类型
UniversalDetector detector = new UniversalDetector(null);
int nread;
// 循环读取文件内容每次读取一部分数据交给UniversalDetector进行分析直到检测完成或读取完整个文件
while ((nread = fis.read(buf)) > 0 &&!detector.isDone()) {
while ((nread = fis.read(buf)) > 0 && !detector.isDone()) {
detector.handleData(buf, 0, nread);
}
detector.dataEnd();
// 获取检测到的文件编码格式,如果未检测到则默认为"utf-8"编码
encoding = detector.getDetectedCharset();
if (encoding == null || encoding.length() == 0)
encoding = "utf-8";
fis.close();
fis = null;
// 章节页码索引用于记录章节的顺序初始化为0表示从第一章开始
int chapterPageIndex = 0;
String title = null;
StringBuilder contentBuilder = new StringBuilder();
fis = new FileInputStream(book);
// 根据检测到的编码格式创建InputStreamReader用于以正确的编码方式读取文件内容
InputStreamReader inputreader = new InputStreamReader(fis, encoding);
// 创建BufferedReader方便逐行读取文件内容进行解析
BufferedReader buffreader = new BufferedReader(inputreader);
String line;
// 逐行读取书籍文件内容,进行章节信息的解析和提取
while ((line = buffreader.readLine())!= null) {
while ((line = buffreader.readLine()) != null) {
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(line);
if (m.find()) {
// 如果匹配到章节标题先提取标题前可能存在的内容如果有并添加到章节内容的临时存储字符串中contentBuilder
String temp = line.trim().substring(0, line.trim().indexOf("第"));
if (temp!= null && temp.trim().length() > 0) {
String temp = line.trim().substring(0,line.trim().indexOf("第"));
if(temp!= null && temp.trim().length()>0){
contentBuilder.append(temp);
}
// 如果临时存储的章节内容字符串不为空且去除全角空格等空白字符后仍有内容,则将当前章节内容保存到数据库中,并更新相关索引和清空临时存储内容
if (contentBuilder.toString().length() > 0) {
if (contentBuilder.toString().replaceAll(" ", "").trim().length() > 0) {
if(contentBuilder.toString().replaceAll(" ","").trim().length()>0){
saveDurChapterContent(md5, chapterPageIndex, title, contentBuilder.toString());
chapterPageIndex++;
}
contentBuilder.delete(0, contentBuilder.length());
}
// 将匹配到的章节标题提取出来,作为当前章节的标题
title = line.trim().substring(line.trim().indexOf("第"));
} else {
if (line.trim().length() == 0) {
@ -224,7 +156,6 @@ public class ImportBookModelImpl extends MBaseModelImpl implements IImportBookMo
}
}
}
// 处理文件末尾剩余的章节内容(如果有),同样将其保存到数据库中
if (contentBuilder.length() > 0) {
saveDurChapterContent(md5, chapterPageIndex, title, contentBuilder.toString());
contentBuilder.delete(0, contentBuilder.length());
@ -236,16 +167,6 @@ public class ImportBookModelImpl extends MBaseModelImpl implements IImportBookMo
fis = null;
}
/**
*
* ChapterListBean
*
*
* @param md5 MD5
* @param chapterPageIndex
* @param name
* @param content
*/
private void saveDurChapterContent(String md5, int chapterPageIndex, String name, String content) {
ChapterListBean chapterListBean = new ChapterListBean();
chapterListBean.setNoteUrl(md5);
@ -261,4 +182,4 @@ public class ImportBookModelImpl extends MBaseModelImpl implements IImportBookMo
DbHelper.getInstance().getmDaoSession().getBookContentBeanDao().insertOrReplace(chapterListBean.getBookContentBean());
DbHelper.getInstance().getmDaoSession().getChapterListBeanDao().insertOrReplace(chapterListBean);
}
}
}

@ -28,84 +28,42 @@ import io.reactivex.android.schedulers.AndroidSchedulers;
import io.reactivex.functions.Function;
import io.reactivex.schedulers.Schedulers;
/**
* LingdiankanshuStationBookModelImplMBaseModelImplIStationBookModel
* TAGhttp://www.lingdiankanshu.co进行交互实现如搜索书籍、获取书籍详情、章节列表以及章节内容等功能。
* HTMLRxJava
*/
public class LingdiankanshuStationBookModelImpl extends MBaseModelImpl implements IStationBookModel {
// 定义一个静态常量TAG用于标识该模型所对应的网站地址此处明确为http://www.lingdiankanshu.co
// 便于在整个程序体系里知晓此模型相关的网络操作、数据解析等逻辑是针对该站点而设计的。
public static final String TAG = "http://www.lingdiankanshu.co";
/**
* LingdiankanshuStationBookModelImpl
*
*
*
* @return LingdiankanshuStationBookModelImpl
*/
public static LingdiankanshuStationBookModelImpl getInstance() {
return new LingdiankanshuStationBookModelImpl();
}
/**
*
* http://zhannei.baidu.com发起搜索书籍的网络请求调用对应的API
* SearchBookBeanObservable
*
*
* @param content
* @param page 1
* @return ObservableList<SearchBookBean>
*/
@Override
public Observable<List<SearchBookBean>> searchBook(String content, int page) {
return getRetrofitObject("http://zhannei.baidu.com").create(ILingdiankanshuApi.class).searchBook(content, page - 1, "16865089933227718744")
.flatMap(new Function<String, ObservableSource<List<SearchBookBean>>>() {
@Override
public ObservableSource<List<SearchBookBean>> apply(String s) throws Exception {
return analySearchBook(s);
}
});
return getRetrofitObject("http://zhannei.baidu.com").create(ILingdiankanshuApi.class).searchBook(content, page - 1, "16865089933227718744").flatMap(new Function<String, ObservableSource<List<SearchBookBean>>>() {
@Override
public ObservableSource<List<SearchBookBean>> apply(String s) throws Exception {
return analySearchBook(s);
}
});
}
/**
*
* HTML
* SearchBookBeanObservableEmitter
*
* @param s HTML
* @return ObservableList<SearchBookBean>
*/
public Observable<List<SearchBookBean>> analySearchBook(final String s) {
return Observable.create(new ObservableOnSubscribe<List<SearchBookBean>>() {
@Override
public void subscribe(ObservableEmitter<List<SearchBookBean>> e) throws Exception {
try {
Document doc = Jsoup.parse(s);
// 获取类名为"result-list"的元素下,类名为"result-item result-game-item"的元素列表,这里假设这些元素包含了每本搜索到的书籍的相关信息
Elements booksE = doc.getElementsByClass("result-list").get(0).getElementsByClass("result-item result-game-item");
if (null!= booksE && booksE.size() > 1) {
if (null != booksE && booksE.size() > 1) {
List<SearchBookBean> books = new ArrayList<SearchBookBean>();
for (int i = 0; i < booksE.size(); i++) {
SearchBookBean item = new SearchBookBean();
item.setTag(TAG);
// 获取书籍作者信息,通过层层定位元素,先找到类名为"result-game-item-info"的元素,再找到其内部类名为"result-game-item-info-tag"的元素,
// 然后获取第二个<span>标签的文本内容作为作者名(这里的元素定位方式是根据网页结构来确定的具体作者名所在位置)
item.setAuthor(booksE.get(i).getElementsByClass("result-game-item-info").get(0).getElementsByClass("result-game-item-info-tag").get(0).getElementsByTag("span").get(1).text());
// 获取书籍分类信息,同样通过层层定位元素,获取对应位置的<span>标签文本内容作为分类名
item.setKind(booksE.get(i).getElementsByClass("result-game-item-info").get(0).getElementsByClass("result-game-item-info-tag").get(1).getElementsByTag("span").get(1).text());
// item.setState(); // 此处代码似乎未完成对书籍状态的设置相关逻辑,可能需要后续补充
// 获取书籍最后章节信息,通过对应元素下的<a>标签获取其文本内容作为最后章节名
// item.setState();
item.setLastChapter(booksE.get(i).getElementsByClass("result-game-item-info").get(0).getElementsByClass("result-game-item-info-tag").get(3).getElementsByTag("a").get(0).text());
item.setOrigin("lingdiankanshu.co");
// 获取书籍名称,通过类名为"result-item-title result-game-item-title"的元素下的<a>标签获取其文本内容作为书名
item.setName(booksE.get(i).getElementsByClass("result-item-title result-game-item-title").get(0).getElementsByTag("a").get(0).text());
// 获取书籍详情链接,通过对应的<a>标签获取其href属性值作为详情链接
item.setNoteUrl(booksE.get(i).getElementsByClass("result-item-title result-game-item-title").get(0).getElementsByTag("a").get(0).attr("href"));
// 获取书籍封面链接,通过<img>标签的src属性值作为封面链接
item.setCoverUrl(booksE.get(i).getElementsByTag("img").get(0).attr("src"));
books.add(item);
}
@ -123,35 +81,16 @@ public class LingdiankanshuStationBookModelImpl extends MBaseModelImpl implement
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
/**
*
* BookShelfBean
* BookShelfBeanObservable
*
* @param bookShelfBean BookShelfBean
* @return ObservableBookShelfBean
*/
@Override
public Observable<BookShelfBean> getBookInfo(final BookShelfBean bookShelfBean) {
return getRetrofitObject(TAG).create(ILingdiankanshuApi.class).getBookInfo(bookShelfBean.getNoteUrl().replace(TAG, ""))
.flatMap(new Function<String, ObservableSource<BookShelfBean>>() {
@Override
public ObservableSource<BookShelfBean> apply(String s) throws Exception {
return analyBookInfo(s, bookShelfBean);
}
});
return getRetrofitObject(TAG).create(ILingdiankanshuApi.class).getBookInfo(bookShelfBean.getNoteUrl().replace(TAG, "")).flatMap(new Function<String, ObservableSource<BookShelfBean>>() {
@Override
public ObservableSource<BookShelfBean> apply(String s) throws Exception {
return analyBookInfo(s, bookShelfBean);
}
});
}
/**
*
* HTML
* BookShelfBeananalyBookinfoObservableEmitter
*
* @param s HTML
* @param bookShelfBean BookShelfBean
* @return ObservableBookShelfBean
*/
private Observable<BookShelfBean> analyBookInfo(final String s, final BookShelfBean bookShelfBean) {
return Observable.create(new ObservableOnSubscribe<BookShelfBean>() {
@Override
@ -164,34 +103,23 @@ public class LingdiankanshuStationBookModelImpl extends MBaseModelImpl implement
});
}
/**
* HTMLBookInfoBean
*
* @param s HTML
* @param novelUrl BookInfoBean
* @return BookInfoBean
*/
private BookInfoBean analyBookinfo(String s, String novelUrl) {
BookInfoBean bookInfoBean = new BookInfoBean();
bookInfoBean.setNoteUrl(novelUrl); //id
bookInfoBean.setTag(TAG);
Document doc = Jsoup.parse(s);
// 获取类名为"box_con"的元素,这里假设书籍详情的主要信息都在这个元素内部,是后续提取各种信息的基础元素
Element resultE = doc.getElementsByClass("box_con").get(0);
// 获取书籍封面链接,通过定位元素找到对应的<img>标签并获取其src属性值作为封面链接
bookInfoBean.setCoverUrl(resultE.getElementById("fmimg").getElementsByTag("img").get(0).attr("src"));
// 获取书籍名称,通过定位元素找到对应的<h1>标签,并获取其文本内容作为书名
bookInfoBean.setName(resultE.getElementById("info").getElementsByTag("h1").get(0).text());
String author = resultE.getElementById("info").getElementsByTag("p").get(0).text().toString().trim();
// 对获取到的作者信息字符串进行处理,去除一些多余的空格、特殊空格以及"作者:"等前缀内容,得到纯净的作者名
author = author.replace(" ", "").replace(" ", "").replace("作者:", "");
author = author.replace(" ", "").replace("  ", "").replace("作者:", "");
bookInfoBean.setAuthor(author);
List<TextNode> contentEs = resultE.getElementById("intro").textNodes();
StringBuilder content = new StringBuilder();
for (int i = 0; i < contentEs.size(); i++) {
String temp = contentEs.get(i).text().trim();
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
if (temp.length() > 0) {
content.append("\u3000\u3000" + temp);
if (i < contentEs.size() - 1) {
@ -207,31 +135,20 @@ public class LingdiankanshuStationBookModelImpl extends MBaseModelImpl implement
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
/**
*
* BookShelfBean
* BookShelfBean
* RxJava线IO线线OnGetChapterListListener
*
* @param bookShelfBean BookShelfBean
* @param getChapterListListener successerror
*/
@Override
public void getChapterList(final BookShelfBean bookShelfBean, final OnGetChapterListListener getChapterListListener) {
getRetrofitObject(TAG).create(ILingdiankanshuApi.class).getChapterList(bookShelfBean.getBookInfoBean().getChapterUrl().replace(TAG, ""))
.flatMap(new Function<String, ObservableSource<WebChapterBean<BookShelfBean>>>() {
@Override
public ObservableSource<WebChapterBean<BookShelfBean>> apply(String s) throws Exception {
return analyChapterList(s, bookShelfBean);
}
})
getRetrofitObject(TAG).create(ILingdiankanshuApi.class).getChapterList(bookShelfBean.getBookInfoBean().getChapterUrl().replace(TAG, "")).flatMap(new Function<String, ObservableSource<WebChapterBean<BookShelfBean>>>() {
@Override
public ObservableSource<WebChapterBean<BookShelfBean>> apply(String s) throws Exception {
return analyChapterList(s, bookShelfBean);
}
})
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new SimpleObserver<WebChapterBean<BookShelfBean>>() {
@Override
public void onNext(WebChapterBean<BookShelfBean> value) {
if (getChapterListListener!= null) {
if (getChapterListListener != null) {
getChapterListListener.success(value.getData());
}
}
@ -239,17 +156,90 @@ public class LingdiankanshuStationBookModelImpl extends MBaseModelImpl implement
@Override
public void onError(Throwable e) {
e.printStackTrace();
if (getChapterListListener!= null) {
if (getChapterListListener != null) {
getChapterListListener.error();
}
}
});
}
/**
*
* HTML
* ChapterListBeanBookShelfBeanObservableEmitter
*
* @param s HTML
* @param bookShelfBean
private Observable<WebChapterBean<BookShelfBean>> analyChapterList(final String s, final BookShelfBean bookShelfBean) {
return Observable.create(new ObservableOnSubscribe<WebChapterBean<BookShelfBean>>() {
@Override
public void subscribe(ObservableEmitter<WebChapterBean<BookShelfBean>> e) throws Exception {
bookShelfBean.setTag(TAG);
WebChapterBean<List<ChapterListBean>> temp = analyChapterlist(s, bookShelfBean.getNoteUrl());
bookShelfBean.getBookInfoBean().setChapterlist(temp.getData());
e.onNext(new WebChapterBean<BookShelfBean>(bookShelfBean, temp.getNext()));
e.onComplete();
}
});
}
private WebChapterBean<List<ChapterListBean>> analyChapterlist(String s, String novelUrl) {
Document doc = Jsoup.parse(s);
Elements chapterlist = doc.getElementById("list").getElementsByTag("dd");
List<ChapterListBean> chapterBeans = new ArrayList<ChapterListBean>();
for (int i = 0; i < chapterlist.size(); i++) {
ChapterListBean temp = new ChapterListBean();
temp.setDurChapterUrl(novelUrl + chapterlist.get(i).getElementsByTag("a").get(0).attr("href")); //id
temp.setDurChapterIndex(i);
temp.setDurChapterName(chapterlist.get(i).getElementsByTag("a").get(0).text());
temp.setNoteUrl(novelUrl);
temp.setTag(TAG);
chapterBeans.add(temp);
}
Boolean next = false;
return new WebChapterBean<List<ChapterListBean>>(chapterBeans, next);
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
@Override
public Observable<BookContentBean> getBookContent(final String durChapterUrl, final int durChapterIndex) {
return getRetrofitObject(TAG).create(ILingdiankanshuApi.class).getBookContent(durChapterUrl.replace(TAG, "")).flatMap(new Function<String, ObservableSource<BookContentBean>>() {
@Override
public ObservableSource<BookContentBean> apply(String s) throws Exception {
return analyBookContent(s, durChapterUrl, durChapterIndex);
}
});
}
private Observable<BookContentBean> analyBookContent(final String s, final String durChapterUrl, final int durChapterIndex) {
return Observable.create(new ObservableOnSubscribe<BookContentBean>() {
@Override
public void subscribe(ObservableEmitter<BookContentBean> e) throws Exception {
BookContentBean bookContentBean = new BookContentBean();
bookContentBean.setDurChapterIndex(durChapterIndex);
bookContentBean.setDurChapterUrl(durChapterUrl);
bookContentBean.setTag(TAG);
try {
Document doc = Jsoup.parse(s);
List<TextNode> contentEs = doc.getElementById("content").textNodes();
StringBuilder content = new StringBuilder();
for (int i = 0; i < contentEs.size(); i++) {
String temp = contentEs.get(i).text().trim();
temp = temp.replaceAll(" ", "").replaceAll(" ", "");
if (temp.length() > 0) {
content.append("\u3000\u3000" + temp);
if (i < contentEs.size() - 1) {
content.append("\r\n");
}
}
}
bookContentBean.setDurCapterContent(content.toString());
bookContentBean.setRight(true);
} catch (Exception ex) {
ex.printStackTrace();
ErrorAnalyContentManager.getInstance().writeNewErrorUrl(durChapterUrl);
bookContentBean.setDurCapterContent(durChapterUrl.substring(0, durChapterUrl.indexOf('/', 8)) + "站点暂时不支持解析请反馈给Monke QQ:1105075896,半小时内解决,超级效率的程序员");
bookContentBean.setRight(false);
}
e.onNext(bookContentBean);
e.onComplete();
}
});
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
}

@ -12,20 +12,8 @@ import io.reactivex.Observable;
import io.reactivex.ObservableEmitter;
import io.reactivex.ObservableOnSubscribe;
/**
* WebBookModelImplIWebBookModel
* tag
* 便
*/
public class WebBookModelImpl implements IWebBookModel {
/**
* WebBookModelImpl
* 便
*
*
* @return WebBookModelImpl
*/
public static WebBookModelImpl getInstance() {
return new WebBookModelImpl();
}
@ -34,22 +22,17 @@ public class WebBookModelImpl implements IWebBookModel {
/**
*
* BookShelfBeantag
* GxwztvBookModelImpl.TAG
* LingdiankanshuStationBookModelImpl.TAG
* null
* BookShelfBeanObservable
*
* @param bookShelfBean BookShelfBeantag
* @return ObservableBookShelfBeannull
* return BookShelfBean
*/
@Override
public Observable<BookShelfBean> getBookInfo(BookShelfBean bookShelfBean) {
if (bookShelfBean.getTag().equals(GxwztvBookModelImpl.TAG)) {
if(bookShelfBean.getTag().equals(GxwztvBookModelImpl.TAG)){
return GxwztvBookModelImpl.getInstance().getBookInfo(bookShelfBean);
} else if (bookShelfBean.getTag().equals(LingdiankanshuStationBookModelImpl.TAG)) {
}
else if(bookShelfBean.getTag().equals(LingdiankanshuStationBookModelImpl.TAG)){
return LingdiankanshuStationBookModelImpl.getInstance().getBookInfo(bookShelfBean);
} else {
}
else {
return null;
}
}
@ -58,23 +41,18 @@ public class WebBookModelImpl implements IWebBookModel {
/**
*
* BookShelfBeantag
* GxwztvBookModelImpl.TAG
* LingdiankanshuStationBookModelImpl.TAG
* OnGetChapterListListenerBookShelfBean
*
*
* @param bookShelfBean BookShelfBeantag
* @param getChapterListListener successerror
* return BookShelfBean
*/
@Override
public void getChapterList(final BookShelfBean bookShelfBean, OnGetChapterListListener getChapterListListener) {
if (bookShelfBean.getTag().equals(GxwztvBookModelImpl.TAG)) {
if(bookShelfBean.getTag().equals(GxwztvBookModelImpl.TAG)){
GxwztvBookModelImpl.getInstance().getChapterList(bookShelfBean, getChapterListListener);
} else if (bookShelfBean.getTag().equals(LingdiankanshuStationBookModelImpl.TAG)) {
}
else if(bookShelfBean.getTag().equals(LingdiankanshuStationBookModelImpl.TAG)){
LingdiankanshuStationBookModelImpl.getInstance().getChapterList(bookShelfBean, getChapterListListener);
} else {
if (getChapterListListener!= null)
}
else{
if(getChapterListListener!=null)
getChapterListListener.success(bookShelfBean);
}
}
@ -83,24 +61,16 @@ public class WebBookModelImpl implements IWebBookModel {
/**
*
* durChapterUrltag
* GxwztvBookModelImpl.TAG
* LingdiankanshuStationBookModelImpl.TAG
* ObservableBookContentBean
* BookContentBeanObservable
*
* @param durChapterUrl 便
* @param durChapterIndex
* @param tag
* @return ObservableBookContentBeanObservableBookContentBean
*/
@Override
public Observable<BookContentBean> getBookContent(String durChapterUrl, int durChapterIndex, String tag) {
if (tag.equals(GxwztvBookModelImpl.TAG)) {
if(tag.equals(GxwztvBookModelImpl.TAG)){
return GxwztvBookModelImpl.getInstance().getBookContent(durChapterUrl, durChapterIndex);
} else if (tag.equals(LingdiankanshuStationBookModelImpl.TAG)) {
}
else if(tag.equals(LingdiankanshuStationBookModelImpl.TAG)){
return LingdiankanshuStationBookModelImpl.getInstance().getBookContent(durChapterUrl, durChapterIndex);
} else
}
else
return Observable.create(new ObservableOnSubscribe<BookContentBean>() {
@Override
public void subscribe(ObservableEmitter<BookContentBean> e) throws Exception {
@ -112,24 +82,16 @@ public class WebBookModelImpl implements IWebBookModel {
/**
*
* tag
* GxwztvBookModelImpl.TAG
* LingdiankanshuStationBookModelImpl.TAG
* ObservableSearchBookBean
* SearchBookBeanObservable
*
* @param content
* @param page
* @param tag
* @return ObservableList<SearchBookBean>Observable
*/
@Override
public Observable<List<SearchBookBean>> searchOtherBook(String content, int page, String tag) {
if (tag.equals(GxwztvBookModelImpl.TAG)) {
public Observable<List<SearchBookBean>> searchOtherBook(String content,int page,String tag){
if(tag.equals(GxwztvBookModelImpl.TAG)){
return GxwztvBookModelImpl.getInstance().searchBook(content, page);
} else if (tag.equals(LingdiankanshuStationBookModelImpl.TAG)) {
}
else if(tag.equals(LingdiankanshuStationBookModelImpl.TAG)){
return LingdiankanshuStationBookModelImpl.getInstance().searchBook(content, page);
} else {
}
else{
return Observable.create(new ObservableOnSubscribe<List<SearchBookBean>>() {
@Override
public void subscribe(ObservableEmitter<List<SearchBookBean>> e) throws Exception {
@ -139,19 +101,11 @@ public class WebBookModelImpl implements IWebBookModel {
});
}
}
/**
*
* GxwztvBookModelImpl
* GxwztvBookModelImplSearchBookBeanObservable
*
*
* @param url GxwztvBookModelImpl
* @param page
* @return ObservableList<SearchBookBean>
*/
@Override
public Observable<List<SearchBookBean>> getKindBook(String url, int page) {
return GxwztvBookModelImpl.getInstance().getKindBook(url, page);
public Observable<List<SearchBookBean>> getKindBook(String url,int page) {
return GxwztvBookModelImpl.getInstance().getKindBook(url,page);
}
}
}

@ -1,3 +1,4 @@
//Copyright (c) 2017. 章钦豪. All rights reserved.
package com.monke.monkeybook.service;
import android.app.NotificationManager;
@ -11,7 +12,6 @@ import android.os.Looper;
import android.support.annotation.Nullable;
import android.support.v4.app.NotificationCompat;
import android.widget.Toast;
import com.hwangjr.rxbus.RxBus;
import com.hwangjr.rxbus.annotation.Subscribe;
import com.hwangjr.rxbus.annotation.Tag;
@ -30,9 +30,7 @@ import com.monke.monkeybook.dao.DbHelper;
import com.monke.monkeybook.dao.DownloadChapterBeanDao;
import com.monke.monkeybook.model.impl.WebBookModelImpl;
import com.monke.monkeybook.view.impl.MainActivity;
import java.util.List;
import io.reactivex.Observable;
import io.reactivex.ObservableEmitter;
import io.reactivex.ObservableOnSubscribe;
@ -43,24 +41,18 @@ import io.reactivex.disposables.Disposable;
import io.reactivex.functions.Function;
import io.reactivex.schedulers.Schedulers;
// DownloadService类继承自Service用于处理书籍下载相关的业务逻辑如启动、暂停、取消下载等操作
public class DownloadService extends Service {
// 用于管理通知的显示和取消等操作
private NotificationManager notifyManager;
// 通知的唯一标识符
private int notifiId = 19931118;
// 标记是否开始下载
private Boolean isStartDownload = false;
// 标记是否已经初始化
private Boolean isInit = false;
// 服务创建时调用的方法
@Override
public void onCreate() {
super.onCreate();
}
// 服务销毁时调用的方法用于注销RxBus注册以及设置初始化标记为true
@Override
public void onDestroy() {
super.onDestroy();
@ -68,31 +60,25 @@ public class DownloadService extends Service {
isInit = true;
}
// 服务启动时调用的方法进行一些初始化操作如获取通知管理器、注册RxBus等
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
if (!isInit) {
isInit = true;
// 获取系统的通知服务
notifyManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
// 注册RxBus用于接收相关事件
RxBus.get().register(this);
}
return super.onStartCommand(intent, flags, startId);
}
// 用于绑定服务这里返回null表示不支持绑定
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
// 添加新的下载任务,将新的下载章节数据插入数据库,并根据下载状态决定是否开始下载
private void addNewTask(final List<DownloadChapterBean> newData) {
isStartDownload = true;
Observable.create(new ObservableOnSubscribe<Boolean>() {
// 在这个方法中执行将下载章节数据插入数据库的操作
@Override
public void subscribe(ObservableEmitter<Boolean> e) throws Exception {
DbHelper.getInstance().getmDaoSession().getDownloadChapterBeanDao().insertOrReplaceInTx(newData);
@ -100,12 +86,9 @@ public class DownloadService extends Service {
e.onComplete();
}
})
// 指定在主线程观察结果
.observeOn(AndroidSchedulers.mainThread())
// 指定在IO线程执行插入数据库操作
.subscribeOn(Schedulers.io())
.subscribe(new SimpleObserver<Boolean>() {
// 插入成功后如果当前没有正在下载则调用toDownload方法开始下载
@Override
public void onNext(Boolean value) {
if (!isDownloading) {
@ -120,38 +103,30 @@ public class DownloadService extends Service {
});
}
// 标记是否正在下载
private Boolean isDownloading = false;
// 下载失败时的重试次数
public static final int reTryTimes = 1;
// 开始下载的主要逻辑方法,查找待下载的章节并进行下载操作
private void toDownload() {
isDownloading = true;
if (isStartDownload) {
Observable.create(new ObservableOnSubscribe<DownloadChapterBean>() {
@Override
public void subscribe(ObservableEmitter<DownloadChapterBean> e) throws Exception {
// 获取书架上的书籍列表,按最后更新日期降序排列
List<BookShelfBean> bookShelfBeanList = DbHelper.getInstance().getmDaoSession().getBookShelfBeanDao().queryBuilder().orderDesc(BookShelfBeanDao.Properties.FinalDate).list();
if (bookShelfBeanList!= null && bookShelfBeanList.size() > 0) {
if (bookShelfBeanList != null && bookShelfBeanList.size() > 0) {
for (BookShelfBean bookItem : bookShelfBeanList) {
// 排除本地标签的书籍(可能表示已经下载好的本地书籍)
if (!bookItem.getTag().equals(BookShelfBean.LOCAL_TAG)) {
// 查询该书籍下待下载的章节列表,按章节索引升序排列,取第一个章节(即下一个要下载的章节)
List<DownloadChapterBean> downloadChapterList = DbHelper.getInstance().getmDaoSession().getDownloadChapterBeanDao().queryBuilder().where(DownloadChapterBeanDao.Properties.NoteUrl.eq(bookItem.getNoteUrl())).orderAsc(DownloadChapterBeanDao.Properties.DurChapterIndex).limit(1).list();
if (downloadChapterList!= null && downloadChapterList.size() > 0) {
if (downloadChapterList != null && downloadChapterList.size() > 0) {
e.onNext(downloadChapterList.get(0));
e.onComplete();
return;
}
}
}
// 如果没有待下载章节,删除所有下载章节记录
DbHelper.getInstance().getmDaoSession().getDownloadChapterBeanDao().deleteAll();
e.onNext(new DownloadChapterBean());
} else {
// 如果书架列表为空,同样删除所有下载章节记录
DbHelper.getInstance().getmDaoSession().getDownloadChapterBeanDao().deleteAll();
e.onNext(new DownloadChapterBean());
}
@ -163,14 +138,12 @@ public class DownloadService extends Service {
.subscribe(new SimpleObserver<DownloadChapterBean>() {
@Override
public void onNext(DownloadChapterBean value) {
if (value.getNoteUrl()!= null && value.getNoteUrl().length() > 0) {
// 如果章节有有效的URL开始下载该章节
if (value.getNoteUrl() != null && value.getNoteUrl().length() > 0) {
downloading(value, 0);
} else {
Observable.create(new ObservableOnSubscribe<Object>() {
@Override
public void subscribe(ObservableEmitter<Object> e) throws Exception {
// 如果章节URL为空删除所有下载章节记录
DbHelper.getInstance().getmDaoSession().getDownloadChapterBeanDao().deleteAll();
e.onNext(new Object());
e.onComplete();
@ -205,17 +178,14 @@ public class DownloadService extends Service {
}
}
// 具体下载章节的逻辑方法,处理下载过程中的重试、保存数据等操作
private void downloading(final DownloadChapterBean data, final int durTime) {
if (durTime < reTryTimes && isStartDownload) {
// 发送下载进度相关的事件,并显示通知
isProgress(data);
Observable.create(new ObservableOnSubscribe<BookContentBean>() {
@Override
public void subscribe(ObservableEmitter<BookContentBean> e) throws Exception {
// 查询数据库中是否已经存在该章节的内容
List<BookContentBean> result = DbHelper.getInstance().getmDaoSession().getBookContentBeanDao().queryBuilder().where(BookContentBeanDao.Properties.DurChapterUrl.eq(data.getDurChapterUrl())).list();
if (result!= null && result.size() > 0) {
if (result != null && result.size() > 0) {
e.onNext(result.get(0));
} else {
e.onNext(new BookContentBean());
@ -226,7 +196,6 @@ public class DownloadService extends Service {
@Override
public ObservableSource<BookContentBean> apply(final BookContentBean bookContentBean) throws Exception {
if (bookContentBean.getDurChapterUrl() == null || bookContentBean.getDurChapterUrl().length() <= 0) {
// 如果章节内容不存在通过WebBookModelImpl获取章节内容然后进行相关数据库操作插入内容、更新章节列表等
return WebBookModelImpl.getInstance().getBookContent(data.getDurChapterUrl(), data.getDurChapterIndex(), data.getTag()).map(new Function<BookContentBean, BookContentBean>() {
@Override
public BookContentBean apply(BookContentBean bookContentBean) throws Exception {
@ -239,7 +208,6 @@ public class DownloadService extends Service {
}
});
} else {
// 如果章节内容已经存在,直接删除下载章节记录并返回该章节内容
return Observable.create(new ObservableOnSubscribe<BookContentBean>() {
@Override
public void subscribe(ObservableEmitter<BookContentBean> e) throws Exception {
@ -256,18 +224,18 @@ public class DownloadService extends Service {
.subscribe(new SimpleObserver<BookContentBean>() {
@Override
public void onNext(BookContentBean value) {
if (isStartDownload) {
if(isStartDownload){
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
if (isStartDownload) {
if(isStartDownload){
toDownload();
} else {
}else{
isPause();
}
}
}, 800);
} else {
},800);
}else{
isPause();
}
}
@ -284,7 +252,6 @@ public class DownloadService extends Service {
Observable.create(new ObservableOnSubscribe<Boolean>() {
@Override
public void subscribe(ObservableEmitter<Boolean> e) throws Exception {
// 如果重试次数超过限制或者下载失败,删除该下载章节记录
DbHelper.getInstance().getmDaoSession().getDownloadChapterBeanDao().delete(data);
e.onNext(true);
e.onComplete();
@ -295,18 +262,18 @@ public class DownloadService extends Service {
.subscribe(new SimpleObserver<Boolean>() {
@Override
public void onNext(Boolean value) {
if (isStartDownload) {
if(isStartDownload){
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
if (isStartDownload) {
if(isStartDownload){
toDownload();
} else {
}else{
isPause();
}
}
}, 800);
} else {
},800);
}else{
isPause();
}
}
@ -314,7 +281,7 @@ public class DownloadService extends Service {
@Override
public void onError(Throwable e) {
e.printStackTrace();
if (!isStartDownload)
if(!isStartDownload)
isPause();
}
});
@ -323,19 +290,16 @@ public class DownloadService extends Service {
}
}
// 对外提供的启动下载的方法设置下载开始标记并调用toDownload方法
public void startDownload() {
isStartDownload = true;
toDownload();
}
// 对外提供的暂停下载的方法,设置下载暂停标记并取消所有通知
public void pauseDownload() {
isStartDownload = false;
notifyManager.cancelAll();
}
// 对外提供的取消下载的方法,先删除所有下载章节记录,然后暂停下载
public void cancelDownload() {
Observable.create(new ObservableOnSubscribe<Object>() {
@Override
@ -370,18 +334,17 @@ public class DownloadService extends Service {
});
}
// 处理下载暂停相关逻辑判断是否还有待下载章节根据情况发送相应的RxBus事件
private void isPause() {
isDownloading = false;
Observable.create(new ObservableOnSubscribe<DownloadChapterBean>() {
@Override
public void subscribe(ObservableEmitter<DownloadChapterBean> e) throws Exception {
List<BookShelfBean> bookShelfBeanList = DbHelper.getInstance().getmDaoSession().getBookShelfBeanDao().queryBuilder().orderDesc(BookShelfBeanDao.Properties.FinalDate).list();
if (bookShelfBeanList!= null && bookShelfBeanList.size() > 0) {
if (bookShelfBeanList != null && bookShelfBeanList.size() > 0) {
for (BookShelfBean bookItem : bookShelfBeanList) {
if (!bookItem.getTag().equals(BookShelfBean.LOCAL_TAG)) {
List<DownloadChapterBean> downloadChapterList = DbHelper.getInstance().getmDaoSession().getDownloadChapterBeanDao().queryBuilder().where(DownloadChapterBeanDao.Properties.NoteUrl.eq(bookItem.getNoteUrl())).orderAsc(DownloadChapterBeanDao.Properties.DurChapterIndex).limit(1).list();
if (downloadChapterList!= null && downloadChapterList.size() > 0) {
if (downloadChapterList != null && downloadChapterList.size() > 0) {
e.onNext(downloadChapterList.get(0));
e.onComplete();
return;
@ -401,11 +364,9 @@ public class DownloadService extends Service {
.subscribe(new SimpleObserver<DownloadChapterBean>() {
@Override
public void onNext(DownloadChapterBean value) {
if (value.getNoteUrl()!= null && value.getNoteUrl().length() > 0) {
// 如果还有待下载章节,发送暂停下载的监听事件
if (value.getNoteUrl() != null && value.getNoteUrl().length() > 0){
RxBus.get().post(RxBusTag.PAUSE_DOWNLOAD_LISTENER, new Object());
} else {
// 如果没有待下载章节,发送下载完成的监听事件
}else{
RxBus.get().post(RxBusTag.FINISH_DOWNLOAD_LISTENER, new Object());
}
}
@ -416,43 +377,27 @@ public class DownloadService extends Service {
}
});
}
// 方法作用处理下载进度相关操作发送下载进度相关信息到RxBus并创建和发送通知展示下载进度情况
// 参数downloadChapterBean表示下载章节相关的数据对象包含了如书籍名称、当前章节名称等信息
private void isProgress(DownloadChapterBean downloadChapterBean) {
// 通过RxBus发送下载进度监听器相关的事件传递下载章节数据对象以便其他地方能接收到并处理进度信息
RxBus.get().post(RxBusTag.PROGRESS_DOWNLOAD_LISTENER, downloadChapterBean);
// 创建一个意图用于启动主活动MainActivity这里的上下文是当前类所在的上下文环境
Intent mainIntent = new Intent(this, MainActivity.class);
// 根据创建的意图创建一个PendingIntent对象用于后续设置到通知中使得点击通知可以启动对应的活动
// 这里的请求码设置为0并且设置当有相同请求码的PendingIntent更新时更新当前的PendingIntent
PendingIntent mainPendingIntent = PendingIntent.getActivity(this, 0, mainIntent, PendingIntent.FLAG_UPDATE_CURRENT);
// 创建 Notification.Builder 对象,用于构建通知
//创建 Notification.Builder 对象
NotificationCompat.Builder builder = new NotificationCompat.Builder(this)
// 设置通知的小图标这里使用的是资源文件中定义的ic_launcher图标
.setSmallIcon(R.mipmap.ic_launcher)
// 设置点击通知后自动清除该通知
//点击通知后自动清除
.setAutoCancel(true)
// 设置通知的标题,展示正在下载的书籍名称
.setContentTitle("正在下载:" + downloadChapterBean.getBookName())
// 根据当前章节名称是否为空来设置通知的内容文本,如果为空则设置为空格,否则设置为章节名称
.setContentText(downloadChapterBean.getDurChapterName() == null? " " : downloadChapterBean.getDurChapterName())
// 将前面创建的PendingIntent设置到通知中使得点击通知能启动对应的活动
.setContentTitle("正在下载:"+downloadChapterBean.getBookName())
.setContentText(downloadChapterBean.getDurChapterName()==null?" ":downloadChapterBean.getDurChapterName())
.setContentIntent(mainPendingIntent);
// 使用通知管理器发送通知notifiId应该是预先定义好的通知的唯一标识用于区分不同的通知
//发送通知
notifyManager.notify(notifiId, builder.build());
}
// 方法作用完成下载相关的后续操作发送下载完成的事件到RxBus取消所有通知并弹出提示告知用户全部离线章节下载完成
private void finishDownload() {
// 通过RxBus发送下载完成监听器相关的事件传递一个空的Object对象作为完成下载的一种通知机制方便其他地方监听并处理完成下载的情况
RxBus.get().post(RxBusTag.FINISH_DOWNLOAD_LISTENER, new Object());
// 取消所有已发送的通知
notifyManager.cancelAll();
// 在主线程中执行以下代码块用于弹出一个短暂显示的Toast提示信息告知用户全部离线章节下载完成
new Handler(Looper.getMainLooper()).post(new Runnable() {
@Override
public void run() {
@ -461,8 +406,6 @@ public class DownloadService extends Service {
});
}
// 方法作用响应RxBus中PAUSE_DOWNLOAD事件调用pauseDownload方法此处假设pauseDownload方法在别处定义用于暂停下载任务相关操作
// 该方法被标注为订阅RxBus中特定标签的事件并且在主线程中执行
@Subscribe(
thread = EventThread.MAIN_THREAD,
tags = {
@ -473,8 +416,6 @@ public class DownloadService extends Service {
pauseDownload();
}
// 方法作用响应RxBus中START_DOWNLOAD事件调用startDownload方法此处假设startDownload方法在别处定义用于启动下载任务相关操作
// 该方法被标注为订阅RxBus中特定标签的事件并且在主线程中执行
@Subscribe(
thread = EventThread.MAIN_THREAD,
tags = {
@ -485,8 +426,6 @@ public class DownloadService extends Service {
startDownload();
}
// 方法作用响应RxBus中CANCEL_DOWNLOAD事件调用cancelDownload方法此处假设cancelDownload方法在别处定义用于取消下载任务相关操作
// 该方法被标注为订阅RxBus中特定标签的事件并且在主线程中执行
@Subscribe(
thread = EventThread.MAIN_THREAD,
tags = {
@ -497,9 +436,6 @@ public class DownloadService extends Service {
cancelDownload();
}
// 方法作用响应RxBus中ADD_DOWNLOAD_TASK事件调用addNewTask方法此处假设addNewTask方法在别处定义用于添加新的下载任务相关操作
// 该方法接收一个DownloadChapterListBean类型的对象newData从其中获取具体的数据假设是下载任务列表数据传递给addNewTask方法进行处理
// 该方法被标注为订阅RxBus中特定标签的事件并且在主线程中执行
@Subscribe(
thread = EventThread.MAIN_THREAD,
tags = {
@ -508,4 +444,5 @@ public class DownloadService extends Service {
)
public void addTask(DownloadChapterListBean newData) {
addNewTask(newData.getData());
}
}
}

@ -1,181 +1,164 @@
// 定义包名
package com.monke.monkeybook.widget;
import android.annotation.TargetApi;// 导入 TargetApi 注解,标明特定的 API 级别
import android.content.Context;// 导入 Context 类,提供应用的上下文
import android.os.Build;// 导入 Build 类,用于获取当前 Android 版本信息
import android.support.annotation.AttrRes; // 导入 AttrRes 注解,表示属性资源的类型。
import android.support.annotation.NonNull;// 导入 NonNull 注解,表示不允许为 null。
import android.support.annotation.Nullable;// 导入 Nullable 注解,表示可以为 null。
import android.support.annotation.StyleRes;// 导入 StyleRes 注解,表示样式资源的类型。
import android.support.v7.widget.LinearLayoutManager;// 导入 LinearLayoutManager用于 RecyclerView 的线性布局。
import android.support.v7.widget.RecyclerView;// 导入 RecyclerView 类,适用于高效的列表展示。
import android.util.AttributeSet;// 导入 AttributeSet 类,用于访问 XML 布局中的自定义属性。
import android.view.LayoutInflater;// 导入 LayoutInflater 类,用于将布局文件转换为 View 对象。
import android.view.View;// 导入 View 类,所有 UI 控件的基类。
import android.view.animation.Animation;// 导入 Animation 类,用于创建动画效果。
import android.view.animation.AnimationUtils;// 导入 AnimationUtils 类,提供加载动画资源的方法。
import android.widget.FrameLayout;// 导入 FrameLayout 类,允许堆叠子视图的布局。
import android.widget.LinearLayout; // 导入 LinearLayout 类,允许在垂直或水平方向上排列子视图。
import android.widget.TextView;// 导入 TextView 类,用于显示文本。
import com.monke.monkeybook.R;// 导入应用资源类,访问资源文件。
import com.monke.monkeybook.bean.BookShelfBean;// 导入 BookShelfBean 类,书架数据模型。
import com.monke.monkeybook.view.adapter.ChapterListAdapter;// 导入 ChapterListAdapter 类,章节列表适配器。
public class ChapterListView extends FrameLayout{// 创建 ChapterListView 类,继承自 FrameLayout
private TextView tvName;// 显示书名的 TextView
private TextView tvListCount;// 显示章节数量的 TextView
private RecyclerView rvList;// 显示章节列表的 RecyclerView
private RecyclerViewBar rvbSlider;// 自定义的 RecyclerView 导航条
private FrameLayout flBg;// 背景 FrameLayout
private LinearLayout llContent;// 内容 LinearLayout
private ChapterListAdapter chapterListAdapter;// 章节列表适配器
private Animation animIn; // 动画进入效果
private Animation animOut; // 动画退出效果
// 默认构造函数
import android.annotation.TargetApi;
import android.content.Context;
import android.os.Build;
import android.support.annotation.AttrRes;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.annotation.StyleRes;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.View;
import android.view.animation.Animation;
import android.view.animation.AnimationUtils;
import android.widget.FrameLayout;
import android.widget.LinearLayout;
import android.widget.TextView;
import com.monke.monkeybook.R;
import com.monke.monkeybook.bean.BookShelfBean;
import com.monke.monkeybook.view.adapter.ChapterListAdapter;
public class ChapterListView extends FrameLayout{
private TextView tvName;
private TextView tvListCount;
private RecyclerView rvList;
private RecyclerViewBar rvbSlider;
private FrameLayout flBg;
private LinearLayout llContent;
private ChapterListAdapter chapterListAdapter;
private Animation animIn;
private Animation animOut;
public ChapterListView(@NonNull Context context) {
this(context,null);// 调用带参构造函数,传递 null
this(context,null);
}
// 带 AttributeSet 的构造函数
public ChapterListView(@NonNull Context context, @Nullable AttributeSet attrs) {
this(context, attrs,0);// 调用带风格属性的构造函数
this(context, attrs,0);
}
// 带属性和风格的构造函数
public ChapterListView(@NonNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr) {
super(context, attrs, defStyleAttr);// 调用父类构造函数
init();// 调用初始化方法
super(context, attrs, defStyleAttr);
init();
}
// 具有更多参数的构造函数
@TargetApi(Build.VERSION_CODES.LOLLIPOP)// 指示支持 API 21 及以上
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public ChapterListView(@NonNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr, @StyleRes int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);// 调用父类构造函数
init();// 调用初始化方法
super(context, attrs, defStyleAttr, defStyleRes);
init();
}
// 初始化方法
private void init() {
setVisibility(INVISIBLE);// 默认隐藏视图
LayoutInflater.from(getContext()).inflate(R.layout.view_chapterlist, this, true); // 加载布局
initData(); // 初始化数据
initView(); // 初始化视图组件
setVisibility(INVISIBLE);
LayoutInflater.from(getContext()).inflate(R.layout.view_chapterlist,this,true);
initData();
initView();
}
// 初始化数据和动画
private void initData() {
// 加载进入动画
animIn = AnimationUtils.loadAnimation(getContext(),R.anim.anim_pop_chapterlist_in);
/
animIn.setAnimationListener(new Animation.AnimationListener() {
@Override
public void onAnimationStart(Animation animation) {
flBg.setOnClickListener(null);// 动画开始时禁止点击
flBg.setOnClickListener(null);
}
@Override
public void onAnimationEnd(Animation animation) {
flBg.setOnClickListener(new OnClickListener() {// 动画结束后允许点击
flBg.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
dimissChapterList();// 点击背景时关闭章节列表
dimissChapterList();
}
});
}
@Override
public void onAnimationRepeat(Animation animation) {
// 动画重复时的反应(可根据需要实现
}
});
// 加载退出动画
animOut = AnimationUtils.loadAnimation(getContext(),R.anim.anim_pop_chapterlist_out);// 退出动画
// 设置退出动画监听器
animOut = AnimationUtils.loadAnimation(getContext(),R.anim.anim_pop_chapterlist_out);
animOut.setAnimationListener(new Animation.AnimationListener() {
@Override
public void onAnimationStart(Animation animation) {
flBg.setOnClickListener(null);// 动画开始时禁止点击
flBg.setOnClickListener(null);
}
@Override
public void onAnimationEnd(Animation animation) {
llContent.setVisibility(INVISIBLE); // 隐藏内容
setVisibility(INVISIBLE); // 隐藏视图
llContent.setVisibility(INVISIBLE);
setVisibility(INVISIBLE);
}
@Override
public void onAnimationRepeat(Animation animation) {
// 动画重复时的反应(可根据需要实现)
}
});
}
// 显示章节列表
public void show(int durChapter) {
chapterListAdapter.setIndex(durChapter); // 设置当前章节索引
((LinearLayoutManager) rvList.getLayoutManager()).scrollToPositionWithOffset(durChapter, 0); // 滚动到特定章节
if(getVisibility() != VISIBLE) {
setVisibility(VISIBLE); // 显示章节列表
animOut.cancel(); // 取消退出动画
animIn.cancel(); // 取消进入动画
llContent.setVisibility(VISIBLE); // 显示内容
llContent.startAnimation(animIn); // 开始进入动画
chapterListAdapter.setIndex(durChapter);
((LinearLayoutManager) rvList.getLayoutManager()).scrollToPositionWithOffset(durChapter,0);
if(getVisibility()!=VISIBLE){
setVisibility(VISIBLE);
animOut.cancel();
animIn.cancel();
llContent.setVisibility(VISIBLE);
llContent.startAnimation(animIn);
}
}
// 点击监听接口
public interface OnItemClickListener{// 点击监听接口
public void itemClick(int index);// 点击事件方法
public interface OnItemClickListener{
public void itemClick(int index);
}
private OnItemClickListener itemClickListener; // 用于监听点击事件的实例
private BookShelfBean bookShelfBean; // 书架数据类实例
// 初始化视图组件
private void initView() { // 初始化视图组件
flBg = (FrameLayout) findViewById(R.id.fl_bg); // 获取背景 FrameLayout
llContent = (LinearLayout) findViewById(R.id.ll_content); // 获取内容 LinearLayout
tvName = (TextView) findViewById(R.id.tv_name); // 获取书名 TextView
tvListCount = (TextView) findViewById(R.id.tv_listcount); // 获取章节数量 TextView
rvList = (RecyclerView) findViewById(R.id.rv_list); // 获取章节列表 RecyclerView
rvList.setLayoutManager(new LinearLayoutManager(getContext())); // 设置布局管理器
rvList.setItemAnimator(null); // 关闭动画效果
rvbSlider = (RecyclerViewBar) findViewById(R.id.rvb_slider); // 获取自定义滑动条
private OnItemClickListener itemClickListener;
private BookShelfBean bookShelfBean;
private void initView() {
flBg = (FrameLayout) findViewById(R.id.fl_bg);
llContent = (LinearLayout) findViewById(R.id.ll_content);
tvName = (TextView) findViewById(R.id.tv_name);
tvListCount = (TextView) findViewById(R.id.tv_listcount);
rvList = (RecyclerView) findViewById(R.id.rv_list);
rvList.setLayoutManager(new LinearLayoutManager(getContext()));
rvList.setItemAnimator(null);
rvbSlider = (RecyclerViewBar) findViewById(R.id.rvb_slider);
}
public void setData(BookShelfBean bookShelfBean, OnItemClickListener clickListener) { // 设置数据
this.itemClickListener = clickListener; // 设置点击监听器
this.bookShelfBean = bookShelfBean; // 设置书架数据
tvName.setText(bookShelfBean.getBookInfoBean().getName()); // 设置书名
tvListCount.setText("共" + bookShelfBean.getBookInfoBean().getChapterlist().size() + "章"); // 设置章节数量文本
chapterListAdapter = new ChapterListAdapter(bookShelfBean, new OnItemClickListener() { // 创建章节列表适配器
public void setData(BookShelfBean bookShelfBean,OnItemClickListener clickListener) {
this.itemClickListener = clickListener;
this.bookShelfBean = bookShelfBean;
tvName.setText(bookShelfBean.getBookInfoBean().getName());
tvListCount.setText("共"+bookShelfBean.getBookInfoBean().getChapterlist().size()+"章");
chapterListAdapter = new ChapterListAdapter(bookShelfBean, new OnItemClickListener() {
@Override
// 处理章节单击事件
public void itemClick(int index) {
if(itemClickListener != null) {
itemClickListener.itemClick(index); // 回调点击事件
rvbSlider.scrollToPositionWithOffset(index); // 滚动到对应位置
if(itemClickListener!=null){
itemClickListener.itemClick(index);
rvbSlider.scrollToPositionWithOffset(index);
}
}
});
rvList.setAdapter(chapterListAdapter); // 设置适配器到 RecyclerView
rvbSlider.setRecyclerView(rvList); // 设置滑动条的 RecyclerView
rvList.setAdapter(chapterListAdapter);
rvbSlider.setRecyclerView(rvList);
}
// 关闭章节列表
public Boolean dimissChapterList() {
if(getVisibility() != VISIBLE) {
return false; // 如果已经隐藏,返回 false
} else {
animOut.cancel(); // 取消退出动画
animIn.cancel(); // 取消进入动画
llContent.startAnimation(animOut); // 开始退出动画
return true; // 返回 true 表示正在关闭
public Boolean dimissChapterList(){
if(getVisibility()!=VISIBLE){
return false;
}else{
animOut.cancel();
animIn.cancel();
llContent.startAnimation(animOut);
return true;
}
}
}

@ -1,43 +1,33 @@
package com.monke.monkeybook.widget;
// 导入必要的 Android 类
import android.content.Context;// 导入 Context 类,用于访问应用的环境信息。
import android.graphics.Canvas;// 导入 Canvas 类,用于进行自定义绘制和图形操作。
import android.text.Layout;// 导入 Layout 类,用于处理文本的布局,控制文本的排列和格式。
import android.text.StaticLayout;// 导入 StaticLayout 类,帮助绘制多行静态文本。
import android.text.TextPaint;// 导入 TextPaint 类,扩展了 Paint 类用于绘制文本并包含字体相关的信息。
import android.util.AttributeSet;// 导入 AttributeSet 类,处理 XML 布局中定义的自定义属性。
import android.content.Context;
import android.graphics.Canvas;
import android.text.Layout;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.util.AttributeSet;
/**
* Created by ZQH on 2017/4/10.
*/
public class MTextView extends android.support.v7.widget.AppCompatTextView {
// 构造函数,接收上下文和属性集
public MTextView(Context context, AttributeSet attrs) {
super(context, attrs);// 调用父类构造函数
super(context, attrs);
}
// 布局方法,处理视图的布局改变
@Override
protected void onLayout(boolean changed, int left, int top, int right,
int bottom) {
super.onLayout(changed, left, top, right, bottom);// 调用父类的方法
super.onLayout(changed, left, top, right, bottom);
}
// 绘制方法,用于在画布上绘制文本
@Override
protected void onDraw(Canvas canvas) {
TextPaint paint = getPaint();// 获取 TextPaint 对象
paint.setColor(getTextColors().getDefaultColor());// 设置文本颜色为默认颜色
// 创建 StaticLayout 用于处理多行文本的绘制
Layout layout = new StaticLayout(getText(),
paint,// 使用的 Paint 对象
canvas.getWidth(),// 输入的宽度
Layout.Alignment.ALIGN_NORMAL,// 文本对齐方式
getLineSpacingMultiplier(),// 行间距倍数
getLineSpacingExtra(),// 行间距额外值
false);// 是否需要添加额外的文本行
// 在画布上绘制文本
TextPaint paint = getPaint();
paint.setColor(getTextColors().getDefaultColor());
Layout layout = new StaticLayout(getText(), paint, canvas.getWidth(), Layout.Alignment.ALIGN_NORMAL, getLineSpacingMultiplier(), getLineSpacingExtra(), false);
layout.draw(canvas);
}
}

@ -1,270 +1,238 @@
//定义类所在的包路径,表示该文件属于com.monke.monkeybook.widget包
package com.monke.monkeybook.widget;
// 导入必要的类
import android.animation.Animator; // 导入 Animator 类,用于实现动画效果的基础类。
import android.animation.ObjectAnimator; // 导入 ObjectAnimator 类,提供对对象属性的动画效果。
import android.annotation.TargetApi; // 导入 TargetApi 注解,用于指示特定的 API 级别。
import android.content.Context; // 导入 Context 类,用于访问应用的环境信息,如资源、数据库等。
import android.content.res.TypedArray; // 导入 TypedArray 类,帮助从 XML 文件中解析自定义属性。
import android.graphics.Rect; // 导入 Rect 类,用于表示矩形区域,通常用于图形绘制和布局。
import android.os.Build; // 导入 Build 类,提供有关当前 Android 版本的静态信息。
import android.os.CountDownTimer; // 导入 CountDownTimer 类,提供一个倒计时的计时器。
import android.support.annotation.Nullable; // 导入 Nullable 注解,表示某个元素可以为 null。
import android.support.v7.widget.LinearLayoutManager; // 导入 LinearLayoutManager用于 RecyclerView 的线性布局管理。
import android.support.v7.widget.RecyclerView; // 导入 RecyclerView 类,提供高效的视图列表。
import android.util.AttributeSet; // 导入 AttributeSet 类,处理 XML 布局中定义的自定义属性。
import android.view.MotionEvent; // 导入 MotionEvent 类,用于处理触摸事件。
import android.view.View; // 导入 View 类Android 中所有视图控件的基类。
import android.view.ViewGroup; // 导入 ViewGroup 类,所有视图组的基类,用于管理子视图。
import android.view.ViewTreeObserver; // 导入 ViewTreeObserver 类,提供对视图树的观察功能
import android.widget.FrameLayout; // 导入 FrameLayout 类,一个可堆叠子视图的布局。
import android.widget.ImageView; // 导入 ImageView 类,显示图片的控件。
import android.widget.LinearLayout; // 导入 LinearLayout 类,垂直或水平排列子视图的布局。
import com.monke.monkeybook.R; // 导入应用资源类,访问资源(如布局、字符串、颜色等)。
import com.monke.monkeybook.utils.DensityUtil; // 导入 DensityUtil 类,通常用于处理与屏幕密度相关的工具函数。
import android.animation.Animator;
import android.animation.ObjectAnimator;
import android.annotation.TargetApi;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Rect;
import android.os.Build;
import android.os.CountDownTimer;
import android.support.annotation.Nullable;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.view.ViewTreeObserver;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.LinearLayout;
import com.monke.monkeybook.R;
import com.monke.monkeybook.utils.DensityUtil;
public class RecyclerViewBar extends LinearLayout {
// 定义滑动动画的时间常量
public static long SLIDE_ANIM_TIME = 800;
// 定义滑块的 ImageView
private ImageView ivSlider;
// 定义滑块的高度,默认为 35dp
private int sliderHeight = DensityUtil.dp2px(getContext(), 35f);
// 定义 RecyclerView
private RecyclerView recyclerView;
// 定义滑块显示和隐藏的动画对象
private Animator slideIn;
private Animator slideOut;
// 构造方法,初始化
public RecyclerViewBar(Context context) {
this(context, null);
}
// 带有属性集的构造方法
public RecyclerViewBar(Context context, @Nullable AttributeSet attrs) {
this(context, attrs, 0);
}
// 带有样式属性的构造方法
public RecyclerViewBar(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init(attrs);// 调用初始化方法
init(attrs);
}
// 针对 Lollipop 及以上版本的构造方法
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public RecyclerViewBar(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);
init(attrs);// 调用初始化方法
init(attrs);
}
// 初始化方法
private void init(AttributeSet attrs) {
setOrientation(VERTICAL);// 设置垂直布局
// 获取自定义属性
setOrientation(VERTICAL);
TypedArray a = getContext().obtainStyledAttributes(attrs, R.styleable.RecyclerViewBar);
// 设置滑块高度
sliderHeight = a.getDimensionPixelSize(R.styleable.RecyclerViewBar_slider_height, sliderHeight);
// 设置滑块的左右内边距
int paddingLeft = a.getDimensionPixelSize(R.styleable.RecyclerViewBar_slider_paddingLeft, 0);
int paddingRight = a.getDimensionPixelSize(R.styleable.RecyclerViewBar_slider_paddingRight, 0);
// 创建滑块 ImageView
ivSlider = new ImageView(getContext());
ivSlider.setPadding(paddingLeft, 0, paddingRight, 0);// 设置滑块内边距
ivSlider.setAlpha(0f);// 初始透明度为 0
ivSlider.setClickable(true);// 设置为可点击
addView(ivSlider);// 将 ImageView 添加到布局中
// 设置滑块的布局参数
ivSlider.setPadding(paddingLeft, 0, paddingRight, 0);
ivSlider.setAlpha(0f);
ivSlider.setClickable(true);
addView(ivSlider);
LayoutParams layoutParams = new LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, sliderHeight);
ivSlider.setLayoutParams(layoutParams);// 应用布局参数
ivSlider.setImageResource(R.drawable.icon_slider);// 设置滑块图标
ivSlider.setScaleType(ImageView.ScaleType.FIT_XY);// 设置图像缩放模式
ivSlider.setLayoutParams(layoutParams);
ivSlider.setImageResource(R.drawable.icon_slider);
ivSlider.setScaleType(ImageView.ScaleType.FIT_XY);
initIvSlider();// 初始化滑块的触摸事件
initIvSlider();
// 添加全局布局监听
RecyclerViewBar.this.getViewTreeObserver().addOnGlobalLayoutListener(layoutInitListener);
}
// 用于记录触摸的 final Y 坐标
private float finalY = -10000;
// 初始化滑块的触摸事件监听
private void initIvSlider() {
// 设置触摸事件监听器
ivSlider.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
int action = event.getAction();// 获取触摸事件类型
int action = event.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
finalY = event.getY();// 记录初始 Y 坐标
return true;// 消耗事件
finalY = event.getY();
return true;
case MotionEvent.ACTION_MOVE:
if (finalY >= 0) {// 检查是否为合法的 Y 坐标
float tempY = event.getY();// 获取当前 Y 坐标
float durY = tempY - finalY;// 计算移动的距离
updateSlider(durY);// 更新滑块位置
if (finalY >= 0) {
float tempY = event.getY();
float durY = tempY - finalY;
updateSlider(durY);
showSlide(); // 显示滑块
showSlide();
} else {
finalY = event.getY();// 更新 finalY
finalY = event.getY();
}
return true;// 消耗事件
return true;
case MotionEvent.ACTION_UP:
if (finalY >= 0) {// 检查是否为合法的 Y 坐标
finalY = -10000;// 重置 finalY
timeCountDown.cancel();// 取消计时器
timeCountDown.start();// 启动计时器
return true;// 消耗事件
if (finalY >= 0) {
finalY = -10000;
timeCountDown.cancel();
timeCountDown.start();
return true;
}
break;
default:
if (finalY >= 0) {
finalY = -10000;// 重置 finalY
return true;// 消耗事件
finalY = -10000;
return true;
}
break;
}
return false;// 未消费事件
return false;
}
});
}
// 更新滑块位置
private void updateSlider(float durY) {
LayoutParams l = (LayoutParams) ivSlider.getLayoutParams();// 获取滑块的布局参数
float finalMarginTop = l.topMargin + durY;// 计算新的顶部边距
// 限制滑块的位置
LayoutParams l = (LayoutParams) ivSlider.getLayoutParams();
float finalMarginTop = l.topMargin + durY;
if (finalMarginTop < 0) {
finalMarginTop = 0;
} else if (finalMarginTop > getHeight() - sliderHeight) {
finalMarginTop = getHeight() - sliderHeight;
}
// 如果 RecyclerView 不为空
if (recyclerView != null) {
// 计算 RecyclerView 的滚动位置
int position = Math.round(finalMarginTop / (getHeight() - sliderHeight) * (recyclerView.getAdapter().getItemCount() - 1));
((LinearLayoutManager) recyclerView.getLayoutManager()).scrollToPositionWithOffset(position, 0);// 滚动到指定位置
((LinearLayoutManager) recyclerView.getLayoutManager()).scrollToPositionWithOffset(position, 0);
}
l.topMargin = Math.round(finalMarginTop);// 更新滑块的顶部边距
ivSlider.setLayoutParams(l);// 应用新的布局参数
l.topMargin = Math.round(finalMarginTop);
ivSlider.setLayoutParams(l);
}
// 设置 RecyclerView
public void setRecyclerView(RecyclerView recyclerView) {
this.recyclerView = recyclerView;// 保存 RecyclerView 的引用
this.recyclerView = recyclerView;
if (this.recyclerView != null) {
// 添加滚动监听
this.recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
// 如果滚动状态不为停止,则显示滑块
super.onScrollStateChanged(recyclerView, newState);
if (newState != 0) {
showSlide();
} else {
timeCountDown.cancel();// 停止计时器
timeCountDown.start();// 启动计时器
timeCountDown.cancel();
timeCountDown.start();
}
}
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
// 将滑块滚动到当前可见项目的位置
scrollToPositionWithOffset(((LinearLayoutManager) recyclerView.getLayoutManager()).findFirstVisibleItemPosition());
}
});
}
}
// 根据位置滚动并更新滑块位置
public void scrollToPositionWithOffset(int position) {
// 如果 RecyclerView 存在且位置合法
if (recyclerView != null && position < recyclerView.getAdapter().getItemCount()) {
float temp = position * 1.0f / recyclerView.getAdapter().getItemCount();
LayoutParams l = (LayoutParams) ivSlider.getLayoutParams();// 获取滑块的布局参数
l.topMargin = Math.round(((getHeight() - sliderHeight) * temp));// 计算新的顶部边距
ivSlider.setLayoutParams(l);// 应用新的布局参数
LayoutParams l = (LayoutParams) ivSlider.getLayoutParams();
l.topMargin = Math.round(((getHeight() - sliderHeight) * temp));
ivSlider.setLayoutParams(l);
}
}
// 显示滑块
private void showSlide() {
if (ivSlider.getAlpha() < 1) {// 如果透明度小于1
if (ivSlider.getAlpha() < 1) {
if (slideOut != null && slideOut.isRunning()) {
slideOut.cancel();// 如果滑出动画正在执行,取消它
slideOut.cancel();
}
if (slideIn == null) {
slideIn = ObjectAnimator.ofFloat(ivSlider, "alpha", ivSlider.getAlpha(), 1f);// 创建滑入动画
slideIn.setDuration((long) (SLIDE_ANIM_TIME * (1f - ivSlider.getAlpha())));// 设置动画持续时间
slideIn = ObjectAnimator.ofFloat(ivSlider, "alpha", ivSlider.getAlpha(), 1f);
slideIn.setDuration((long) (SLIDE_ANIM_TIME * (1f - ivSlider.getAlpha())));
}
if (!slideIn.isRunning()) {
slideIn.start();// 启动滑入动画
slideIn.start();
}
}
}
// 隐藏滑块
private void hideSlide() {
if (ivSlider.getAlpha() > 0) {// 如果透明度大于0
if (ivSlider.getAlpha() > 0) {
if (slideIn != null && slideIn.isRunning()) {
slideIn.cancel();// 如果滑入动画正在执行,取消它
slideIn.cancel();
}
if (slideOut == null) {
slideOut = ObjectAnimator.ofFloat(ivSlider, "alpha", ivSlider.getAlpha(), 0f);// 创建滑出动画
slideOut.setDuration((long) (SLIDE_ANIM_TIME * ivSlider.getAlpha()));// 设置动画持续时间
slideOut = ObjectAnimator.ofFloat(ivSlider, "alpha", ivSlider.getAlpha(), 0f);
slideOut.setDuration((long) (SLIDE_ANIM_TIME * ivSlider.getAlpha()));
}
if (!slideOut.isRunning()) {
slideOut.start();// 启动滑出动画
slideOut.start();
}
}
}
// 创建计时器,用于滑块的自动隐藏
private TimeCountDown timeCountDown = new TimeCountDown();
// 定义倒计时类
class TimeCountDown extends CountDownTimer {
// 构造方法,设置计时器
public TimeCountDown() {
this(1000, 1000);// 1000 毫秒后完成,间隔 1000 毫秒
this(1000, 1000);
}
public TimeCountDown(long millisInFuture, long countDownInterval) {
super(millisInFuture, countDownInterval);// 调用父类构造方法
super(millisInFuture, countDownInterval);
}
@Override
public void onTick(long millisUntilFinished) {
// 每次计时器滴答时调用(可选实现,暂时为空)
}
@Override
public void onFinish() {
hideSlide();// 计时结束时隐藏滑块
hideSlide();
}
}
// 用于存储视图的高度
private int height = 0;
// 全局布局监听器
private ViewTreeObserver.OnGlobalLayoutListener layoutInitListener = new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
if(getHeight()>0){// 确保当前高度大于0
if(getHeight()>0){
if (height == 0) {
height = getHeight();// 保存初始高度
height = getHeight();
} else {
int diff = height - getHeight();// 计算高度差
int diff = height - getHeight();
if (diff != 0) {
LayoutParams l = (LayoutParams) ivSlider.getLayoutParams();// 获取滑块的布局参数
LayoutParams l = (LayoutParams) ivSlider.getLayoutParams();
l.topMargin = (int) ((l.topMargin*1.0f/(height-sliderHeight))*(getHeight()-sliderHeight));
ivSlider.setLayoutParams(l);// 更新滑块的顶部边距
height = getHeight();// 更新高度
ivSlider.setLayoutParams(l);
height = getHeight();
}
}
}

@ -1,45 +1,40 @@
package com.monke.monkeybook.widget.refreshview; // 定义包名
package com.monke.monkeybook.widget.refreshview;
import android.content.Context; // 导入 Context 类,提供应用的上下文
import android.content.res.TypedArray; // 导入 TypedArray 类,用于处理自定义属性
import android.support.v7.widget.LinearLayoutManager; // 导入 LinearLayoutManager 类,用于线性布局管理器
import android.support.v7.widget.RecyclerView; // 导入 RecyclerView 类,支持高效的列表展示
import android.util.AttributeSet; // 导入 AttributeSet 类,用于访问 XML 布局中的自定义属性
import android.view.LayoutInflater; // 导入 LayoutInflater 类,用于将布局文件转换为 View 对象
import android.view.MotionEvent; // 导入 MotionEvent 类,处理触摸事件
import android.view.View; // 导入 View 类,所有 UI 控件的基类
import android.widget.FrameLayout; // 导入 FrameLayout 类,按层次排列子视图
import android.content.Context;
import android.content.res.TypedArray;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.widget.FrameLayout;
import com.monke.monkeybook.R; // 导入应用资源类,访问资源文件
import com.monke.monkeybook.R;
// 创建 RefreshRecyclerView 类,继承自 FrameLayout
public class RefreshRecyclerView extends FrameLayout {
private View view; // 根视图
private RefreshProgressBar rpb; // 自定义进度条
private RecyclerView recyclerView; // RecyclerView 控件
private View view;
private RefreshProgressBar rpb;
private RecyclerView recyclerView;
private View noDataView; // 无数据视图
private View refreshErrorView; // 刷新错误视图
private View noDataView;
private View refreshErrorView;
// 构造函数
public RefreshRecyclerView(Context context) {
this(context, null); // 调用带有 AttributeSet 的构造函数
this(context, null);
}
public RefreshRecyclerView(Context context, AttributeSet attrs) {
this(context, attrs, 0); // 调用带有默认样式的构造函数
this(context, attrs, 0);
}
public RefreshRecyclerView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
// 从 XML 布局中加载视图
view = LayoutInflater.from(context).inflate(R.layout.view_refresh_recyclerview, this, false);
// 初始化 UI 组件
rpb = (RefreshProgressBar) view.findViewById(R.id.rpb);
recyclerView = (RecyclerView) view.findViewById(R.id.rv);
// 获取自定义属性并设置给 RefreshProgressBar
TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.RefreshProgressBar);
rpb.setSpeed(a.getDimensionPixelSize(R.styleable.RefreshProgressBar_speed, rpb.getSpeed()));
rpb.setMaxProgress(a.getInt(R.styleable.RefreshProgressBar_max_progress, rpb.getMaxProgress()));
@ -47,253 +42,235 @@ public class RefreshRecyclerView extends FrameLayout {
rpb.setBgColor(a.getColor(R.styleable.RefreshProgressBar_bg_color, rpb.getBgColor()));
rpb.setSecondColor(a.getColor(R.styleable.RefreshProgressBar_second_color, rpb.getSecondColor()));
rpb.setFontColor(a.getColor(R.styleable.RefreshProgressBar_font_color, rpb.getFontColor()));
a.recycle(); // 释放 TypedArray
a.recycle();
bindEvent(); // 绑定事件
addView(view); // 将根视图添加到 FrameLayout 中
bindEvent();
addView(view);
}
private float durTouchY = -1000000; // 记录触摸 Y 坐标
private BaseRefreshListener baseRefreshListener; // 刷新监听器
private float durTouchY = -1000000;
private BaseRefreshListener baseRefreshListener;
// 设置刷新监听器
public void setBaseRefreshListener(BaseRefreshListener baseRefreshListener) {
this.baseRefreshListener = baseRefreshListener;
}
private OnLoadMoreListener loadMoreListener; // 加载更多监听器
private OnLoadMoreListener loadMoreListener;
// 设置加载更多监听器
public void setLoadMoreListener(OnLoadMoreListener loadMoreListener) {
this.loadMoreListener = loadMoreListener;
}
// 绑定事件
private void bindEvent() {
recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState); // 调用父类方法
super.onScrollStateChanged(recyclerView, newState);
}
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy); // 调用父类方法
// 判断是否可以加载更多
if (((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).canLoadMore() &&
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getItemCount() - 1
== ((LinearLayoutManager) recyclerView.getLayoutManager()).findLastVisibleItemPosition()) {
// 如果没有加载错误,开始加载更多
if (!((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getLoadMoreError()) {
super.onScrolled(recyclerView, dx, dy);
if (((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).canLoadMore() && ((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getItemCount() - 1 == ((LinearLayoutManager) recyclerView.getLayoutManager()).findLastVisibleItemPosition()) {
if(!((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getLoadMoreError()){
if (null != loadMoreListener) {
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(2, false); // 设置为正在加载更多状态
loadMoreListener.startLoadmore(); // 调用加载更多方法
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(2, false);
loadMoreListener.startLoadmore();
}
}
}
}
});
recyclerView.setOnTouchListener(refreshTouchListener); // 设置触摸事件监听器
recyclerView.setOnTouchListener(refreshTouchListener);
}
// 获取自定义进度条
public RefreshProgressBar getRpb() {
return rpb;
}
// 获取 RecyclerView
public RecyclerView getRecyclerView() {
return recyclerView;
}
// 处理刷新错误
public void refreshError() {
rpb.setIsAutoLoading(false); // 设置为非自动加载状态
rpb.clean(); // 清空进度
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, true); // 设置请求状态为未请求
rpb.setIsAutoLoading(false);
rpb.clean();
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, true);
if (noDataView != null) {
noDataView.setVisibility(GONE); // 隐藏无数据视图
noDataView.setVisibility(GONE);
}
if (refreshErrorView != null) {
refreshErrorView.setVisibility(VISIBLE); // 显示刷新错误视图
refreshErrorView.setVisibility(VISIBLE);
}
}
// 开始刷新
public void startRefresh() {
if (baseRefreshListener != null && baseRefreshListener instanceof OnRefreshWithProgressListener) {
// 带有进度的刷新
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsAll(false, false); // 设置还有更多数据
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(1, false); // 设置为正在刷新状态
rpb.setSecondDurProgress(rpb.getSecondMaxProgress()); // 设置当前进度为最大进度
rpb.setMaxProgress(((OnRefreshWithProgressListener) baseRefreshListener).getMaxProgress()); // 获取最大进度
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsAll(false, false);
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(1, false);
rpb.setSecondDurProgress(rpb.getSecondMaxProgress());
rpb.setMaxProgress(((OnRefreshWithProgressListener) baseRefreshListener).getMaxProgress());
} else {
// 非带进度的刷新
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(1, true); // 设置为正在刷新状态
rpb.setIsAutoLoading(true); // 设置为自动加载
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(1, true);
rpb.setIsAutoLoading(true);
if (noDataView != null) {
noDataView.setVisibility(GONE); // 隐藏无数据视图
noDataView.setVisibility(GONE);
}
if (refreshErrorView != null) {
refreshErrorView.setVisibility(GONE); // 隐藏刷新错误视图
refreshErrorView.setVisibility(GONE);
}
}
}
// 完成刷新
public void finishRefresh(Boolean needNoti) {
finishRefresh(((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getItemcount() == 0, needNoti); // 判断是否有数据并完成刷新
finishRefresh(((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getItemcount() == 0, needNoti);
}
// 完成刷新,判断是否还有更多数据
public void finishRefresh(Boolean isAll, Boolean needNoti) {
rpb.setDurProgress(0); // 重置进度条
rpb.setDurProgress(0);
if (isAll) {
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, false); // 设置请求状态为未请求
rpb.setIsAutoLoading(false); // 设置为非自动加载状态
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsAll(isAll, needNoti); // 设置没有更多数据
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, false);
rpb.setIsAutoLoading(false);
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsAll(isAll, needNoti);
} else {
rpb.setIsAutoLoading(false); // 设置为非自动加载状态
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, needNoti); // 设置请求状态为未请求
rpb.setIsAutoLoading(false);
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, needNoti);
}
// 根据数据情况显示无数据视图
if (isAll) {
if (noDataView != null) {
if (((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getItemcount() == 0)
noDataView.setVisibility(VISIBLE); // 如果没有数据,则显示无数据视图
noDataView.setVisibility(VISIBLE);
else
noDataView.setVisibility(GONE); // 否则隐藏无数据视图
noDataView.setVisibility(GONE);
}
if (refreshErrorView != null) {
refreshErrorView.setVisibility(GONE); // 隐藏刷新错误视图
refreshErrorView.setVisibility(GONE);
}
}
}
// 完成加载更多
public void finishLoadMore(Boolean isAll, Boolean needNoti) {
if (isAll) {
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, false); // 设置请求状态为未请求
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsAll(isAll, needNoti); // 设置没有更多数据
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, false);
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsAll(isAll, needNoti);
} else {
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, needNoti); // 设置请求状态为未请求
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(0, needNoti);
}
if (noDataView != null) {
noDataView.setVisibility(GONE); // 隐藏无数据视图
noDataView.setVisibility(GONE);
}
if (refreshErrorView != null) {
refreshErrorView.setVisibility(GONE); // 隐藏刷新错误视图
refreshErrorView.setVisibility(GONE);
}
}
// 设置适配器
public void setRefreshRecyclerViewAdapter(RefreshRecyclerViewAdapter refreshRecyclerViewAdapter, RecyclerView.LayoutManager layoutManager) {
refreshRecyclerViewAdapter.setClickTryAgainListener(new RefreshRecyclerViewAdapter.OnClickTryAgainListener() {
@Override
public void loadMoreErrorTryAgain() {
if (loadMoreListener != null)
loadMoreListener.loadMoreErrorTryAgain(); // 处理加载更多错误重试
loadMoreListener.loadMoreErrorTryAgain();
}
});
recyclerView.setLayoutManager(layoutManager); // 设置布局管理器
recyclerView.setAdapter(refreshRecyclerViewAdapter); // 设置适配器
recyclerView.setLayoutManager(layoutManager);
recyclerView.setAdapter(refreshRecyclerViewAdapter);
}
// 加载更多错误
public void loadMoreError() {
rpb.setIsAutoLoading(false); // 设置为非自动加载状态
rpb.clean(); // 清空进度
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setLoadMoreError(true, true); // 设置加载错误状态
rpb.setIsAutoLoading(false);
rpb.clean();
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setLoadMoreError(true, true);
}
// 设置无数据视图和刷新错误视图
public void setNoDataAndrRefreshErrorView(View noData, View refreshError) {
if (noData != null) {
noDataView = noData; // 设置无数据视图
noDataView.setVisibility(GONE); // 隐藏无数据视图
addView(noDataView, getChildCount() - 1); // 将视图添加到 FrameLayout 中
noDataView = noData;
// noDataView.setOnTouchListener(refreshTouchListener);
noDataView.setVisibility(GONE);
addView(noDataView, getChildCount() - 1);
}
if (refreshError != null) {
refreshErrorView = refreshError; // 设置刷新错误视图
addView(refreshErrorView, 2); // 将视图添加到 FrameLayout 中
refreshErrorView.setVisibility(GONE); // 隐藏刷新错误视图
refreshErrorView = refreshError;
// refreshErrorView.setOnTouchListener(refreshTouchListener);
addView(refreshErrorView, 2);
refreshErrorView.setVisibility(GONE);
}
}
private OnTouchListener refreshTouchListener = new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
int action = event.getAction(); // 获取触摸事件动作
int action = event.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN: // 触摸开始时
durTouchY = event.getY(); // 记录 Y 坐标
case MotionEvent.ACTION_DOWN:
durTouchY = event.getY();
break;
case MotionEvent.ACTION_MOVE: // 触摸移动时
case MotionEvent.ACTION_MOVE:
if (durTouchY == -1000000)
durTouchY = event.getY(); // 初始化 Y 坐标
float dY = event.getY() - durTouchY; // 计算下拉距离
durTouchY = event.getY(); // 更新 Y 坐标
// 判断是否可以刷新
durTouchY = event.getY();
float dY = event.getY() - durTouchY; //>0下拉
durTouchY = event.getY();
if (baseRefreshListener != null && ((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getIsRequesting() == 0 && rpb.getSecondDurProgress() == rpb.getSecondFinalProgress()) {
if (rpb.getVisibility() != View.VISIBLE) {
rpb.setVisibility(View.VISIBLE); // 显示进度条
rpb.setVisibility(View.VISIBLE);
}
if (recyclerView.getAdapter().getItemCount() > 0) {
if (0 == ((LinearLayoutManager) recyclerView.getLayoutManager()).findFirstCompletelyVisibleItemPosition()) {
rpb.setSecondDurProgress((int) (rpb.getSecondDurProgress() + dY)); // 更新进度
rpb.setSecondDurProgress((int) (rpb.getSecondDurProgress() + dY));
}
} else {
rpb.setSecondDurProgress((int) (rpb.getSecondDurProgress() + dY)); // 更新进度
rpb.setSecondDurProgress((int) (rpb.getSecondDurProgress() + dY));
}
if (rpb.getSecondDurProgress() <= 0) {
return false; // 返回 false 继续处理
return false;
} else {
return true; // 返回 true 终止触摸事件继续处理
return true;
}
}
break;
case MotionEvent.ACTION_UP: // 触摸抬起时
// 判断是否能开始刷新
case MotionEvent.ACTION_UP:
if (baseRefreshListener != null && rpb.getSecondMaxProgress() > 0 && rpb.getSecondDurProgress() > 0) {
if (rpb.getSecondDurProgress() >= rpb.getSecondMaxProgress() && ((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getIsRequesting() == 0) {
if (baseRefreshListener instanceof OnRefreshWithProgressListener) {
// 带有进度的刷新
//带有进度的
//执行刷新响应
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsAll(false, false);
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(1, true); // 设置为正在刷新状态
rpb.setMaxProgress(((OnRefreshWithProgressListener) baseRefreshListener).getMaxProgress()); // 获取最大进度
baseRefreshListener.startRefresh(); // 执行刷新
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(1, true);
rpb.setMaxProgress(((OnRefreshWithProgressListener) baseRefreshListener).getMaxProgress());
baseRefreshListener.startRefresh();
if (noDataView != null) {
noDataView.setVisibility(GONE); // 隐藏无数据视图
noDataView.setVisibility(GONE);
}
if (refreshErrorView != null) {
refreshErrorView.setVisibility(GONE); // 隐藏刷新错误视图
refreshErrorView.setVisibility(GONE);
}
} else {
// 不带进度的刷新
//不带进度的
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsAll(false, false);
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(1, true); // 设置为正在刷新状态
baseRefreshListener.startRefresh(); // 执行刷新
((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).setIsRequesting(1, true);
baseRefreshListener.startRefresh();
if (noDataView != null) {
noDataView.setVisibility(GONE); // 隐藏无数据视图
noDataView.setVisibility(GONE);
}
if (refreshErrorView != null) {
refreshErrorView.setVisibility(GONE); // 隐藏刷新错误视图
refreshErrorView.setVisibility(GONE);
}
rpb.setIsAutoLoading(true); // 设置为自动加载状态
rpb.setIsAutoLoading(true);
}
} else {
if (((RefreshRecyclerViewAdapter) recyclerView.getAdapter()).getIsRequesting() != 1)
rpb.setSecondDurProgressWithAnim(0); // 动画归零
rpb.setSecondDurProgressWithAnim(0);
}
}
durTouchY = -1000000; // 重置 Y 坐标
durTouchY = -1000000;
break;
}
return false; // 返回 false 继续处理触摸事件
return false;
}
};
}

@ -1,204 +1,177 @@
// 定义包名
package com.monke.monkeybook.widget.refreshview;
import android.os.Handler; // 导入 Handler 类,用于处理线程间通信和执行任务
import android.os.Looper; // 导入 Looper 类,用于控制线程的消息循环
import android.support.v7.widget.RecyclerView; // 导入 RecyclerView 类,支持高效的列表展示
import android.util.Log; // 导入 Log 类,用于调试日志
import android.view.LayoutInflater; // 导入 LayoutInflater 类,用于将布局文件转换为 View 对象
import android.view.View; // 导入 View 类,所有 UI 控件的基类
import android.view.ViewGroup; // 导入 ViewGroup 类,视图的容器类
import android.widget.FrameLayout; // 导入 FrameLayout 类,允许堆叠子视图的布局
import android.widget.TextView; // 导入 TextView 类,用于显示文本
import android.os.Handler;
import android.os.Looper;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.FrameLayout;
import android.widget.TextView;
import com.monke.monkeybook.R; // 导入应用资源类,访问资源文件
import com.monke.monkeybook.R;
// 创建抽象类 RefreshRecyclerViewAdapter继承自 RecyclerView.Adapter
public abstract class RefreshRecyclerViewAdapter extends RecyclerView.Adapter {
private final int LOADMORETYPE = 2001; // 加载更多的视图类型
private final int LOADMORETYPE = 2001;
private Handler handler; // 处理线程间通信的 Handler
private int isRequesting = 0; // 当前请求状态: 0-未请求1-正在下拉刷新2-正在加载更多
private Boolean needLoadMore = false; // 是否需要加载更多
private Boolean isAll = false; // 判断是否还有更多数据
private Boolean loadMoreError = false; // 判断是否加载更多时发生错误
private Handler handler;
private int isRequesting = 0; //0是未执行网络请求 1是正在下拉刷新 2是正在加载更多
private Boolean needLoadMore = false;
private Boolean isAll = false; //判断是否还有更多
private Boolean loadMoreError = false;
private OnClickTryAgainListener clickTryAgainListener; // 加载失败重试监听器接口
private OnClickTryAgainListener clickTryAgainListener;
// 点击重试的监听器接口
public interface OnClickTryAgainListener {
public void loadMoreErrorTryAgain();// 加载失败时的重试方法
public void loadMoreErrorTryAgain();
}
// 构造函数
public RefreshRecyclerViewAdapter(Boolean needLoadMore) {
this.needLoadMore = needLoadMore; // 设置是否需要加载更多
handler = new Handler(); // 初始化 Handler
this.needLoadMore = needLoadMore;
handler = new Handler();
}
// 获取请求状态
public int getIsRequesting() {
return isRequesting; // 返回当前请求状态
return isRequesting;
}
// 设置请求状态
public void setIsRequesting(int isRequesting, Boolean needNoti) {
this.isRequesting = isRequesting; // 更新请求状态
if (this.isRequesting == 1) { // 如果正在请求刷新
isAll = false; // 设置为没有更多数据
this.isRequesting = isRequesting;
if (this.isRequesting == 1) {
isAll = false;
}
// 根据需要通知数据变化
if (needNoti) {
if (Looper.myLooper() == Looper.getMainLooper()) {
notifyItemRangeChanged(getItemCount(), getItemCount() - getItemcount()); // 通知数据范围变化
notifyItemRangeChanged(getItemCount(), getItemCount() - getItemcount());
} else {
handler.post(new Runnable() {
@Override
public void run() {
notifyDataSetChanged(); // 在主线程中通知数据已更新
notifyDataSetChanged();
}
});
}
}
}
// 创建视图持有者
@Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
if (viewType == LOADMORETYPE) { // 如果是加载更多类型
if (viewType == LOADMORETYPE) {
return new LoadMoreViewHolder(LayoutInflater.from(parent.getContext()).inflate(R.layout.view_refresh_loadmore, parent, false));
} else {
return onCreateViewholder(parent, viewType); // 否则调用抽象方法创建其他类型的视图持有者
}
} else
return onCreateViewholder(parent, viewType);
}
// 抽象方法,创建其他类型的视图持有者
public abstract RecyclerView.ViewHolder onCreateViewholder(ViewGroup parent, int viewType);
// 绑定视图数据
@Override
public void onBindViewHolder(final RecyclerView.ViewHolder holder, int position) {
if (holder.getItemViewType() == LOADMORETYPE) { // 如果是加载更多视图
if (holder.getItemViewType() == LOADMORETYPE) {
if (!loadMoreError) {
((LoadMoreViewHolder) holder).tvLoadMore.setText("正在加载..."); // 显示加载中
((LoadMoreViewHolder) holder).tvLoadMore.setText("正在加载...");
} else {
((LoadMoreViewHolder) holder).tvLoadMore.setText("加载失败,点击重试"); // 显示加载失败信息
((LoadMoreViewHolder) holder).tvLoadMore.setText("加载失败,点击重试");
}
// 设置点击事件
((LoadMoreViewHolder) holder).tvLoadMore.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (null != clickTryAgainListener && loadMoreError) {
clickTryAgainListener.loadMoreErrorTryAgain(); // 调用重试方法
loadMoreError = false; // 重置加载错误状态
((LoadMoreViewHolder) holder).tvLoadMore.setText("正在加载..."); // 更新加载状态
clickTryAgainListener.loadMoreErrorTryAgain();
loadMoreError = false;
((LoadMoreViewHolder) holder).tvLoadMore.setText("正在加载...");
}
}
});
} else {
onBindViewholder(holder, position); // 绑定其他类型视图的数据
}
} else
onBindViewholder(holder, position);
}
// 抽象方法,绑定其他类型视图的数据
public abstract void onBindViewholder(RecyclerView.ViewHolder holder, int position);
// 获取视图类型
@Override
public int getItemViewType(int position) {
if (needLoadMore && isRequesting != 1 && !isAll && position == getItemCount() - 1 && getItemcount() > 0) {
return LOADMORETYPE; // 如果满足条件,返回加载更多视图类型
return LOADMORETYPE;
} else {
return getItemViewtype(position); // 否则返回其他视图类型
return getItemViewtype(position);
}
}
// 抽象方法,获取其他类型视图的类型
public abstract int getItemViewtype(int position);
// 获取项计数
@Override
public int getItemCount() {
if (needLoadMore && isRequesting != 1 && !isAll && getItemcount() > 0) {
return getItemcount() + 1; // 如果需要加载更多,返回总数 + 1加载更多视图
} else {
return getItemcount(); // 返回正常的项计数
}
return getItemcount() + 1;
} else
return getItemcount();
}
public abstract int getItemcount(); // 抽象方法,获取正常的项计数
public abstract int getItemcount();
// 设置是否有更多数据
public void setIsAll(Boolean isAll, Boolean needNoti) {
this.isAll = isAll; // 更新是否有更多数据
// 根据需要通知数据变化
this.isAll = isAll;
if (needNoti) {
if (Looper.myLooper() == Looper.getMainLooper()) {
// notifyItemRangeChanged(getItemCount(),getItemCount()-getItemcount());
if (getItemCount() > getItemcount()) {
notifyItemRangeChanged(getItemCount(), getItemCount() - getItemcount()); // 通知数据范围变化
} else {
notifyItemRemoved(getItemCount() + 1); // 通知加载更多视图移除
}
notifyItemRangeChanged(getItemCount(), getItemCount() - getItemcount());
} else
notifyItemRemoved(getItemCount() + 1);
} else {
handler.post(new Runnable() {
@Override
public void run() {
notifyDataSetChanged(); // 在主线程中通知数据已更新
notifyDataSetChanged();
}
});
}
}
}
// 加载更多视图持有者类
class LoadMoreViewHolder extends RecyclerView.ViewHolder {
FrameLayout llLoadMore; // 加载更多的容器
TextView tvLoadMore; // 加载更多的文本视图
FrameLayout llLoadMore;
TextView tvLoadMore;
// 构造函数
public LoadMoreViewHolder(View itemView) {
super(itemView);
llLoadMore = (FrameLayout) itemView.findViewById(R.id.ll_loadmore); // 获取加载更多的容器
tvLoadMore = (TextView) itemView.findViewById(R.id.tv_loadmore); // 获取加载更多文本视图
llLoadMore = (FrameLayout) itemView.findViewById(R.id.ll_loadmore);
tvLoadMore = (TextView) itemView.findViewById(R.id.tv_loadmore);
}
}
// 判断是否可以加载更多
public Boolean canLoadMore() {
return needLoadMore && isRequesting == 0 && !isAll && getItemcount() > 0; // 返回是否满足加载更多条件
return needLoadMore && isRequesting == 0 && !isAll && getItemcount() > 0;
}
// 获取重试监听器
public OnClickTryAgainListener getClickTryAgainListener() {
return clickTryAgainListener; // 返回重试监听器
return clickTryAgainListener;
}
// 设置重试监听器
public void setClickTryAgainListener(OnClickTryAgainListener clickTryAgainListener) {
this.clickTryAgainListener = clickTryAgainListener; // 赋值给重试监听器
this.clickTryAgainListener = clickTryAgainListener;
}
// 获取加载更多错误状态
public Boolean getLoadMoreError() {
return loadMoreError; // 返回加载更多错误状态
return loadMoreError;
}
// 设置加载更多错误
public void setLoadMoreError(Boolean loadMoreError, Boolean needNoti) {
this.isRequesting = 0; // 重置请求状态
this.loadMoreError = loadMoreError; // 设置加载错误状态
// 根据需要通知数据变化
this.isRequesting = 0;
this.loadMoreError = loadMoreError;
if (needNoti) {
if (Looper.myLooper() == Looper.getMainLooper()) {
notifyDataSetChanged(); // 在主线程中通知数据已更新
notifyDataSetChanged();
} else {
handler.post(new Runnable() {
@Override
public void run() {
notifyDataSetChanged(); // 在主线程中通知数据已更新
notifyDataSetChanged();
}
});
}
}
}
}
}

@ -1,117 +1,101 @@
// 定义包名
package com.monke.monkeybook.widget.refreshview;
import android.annotation.TargetApi; // 导入 TargetApi 注解,标明特定的 API 级别
import android.content.Context; // 导入 Context 类,提供应用的上下文
import android.os.Build; // 导入 Build 类,用于获取当前 Android 版本信息
import android.support.annotation.NonNull; // 导入 NonNull 注解,表示不允许为 null
import android.util.AttributeSet; // 导入 AttributeSet 类,用于访问 XML 布局中的自定义属性
import android.view.MotionEvent; // 导入 MotionEvent 类,处理触摸事件
import android.view.View; // 导入 View 类,所有 UI 控件的基类
import android.widget.ScrollView; // 导入 ScrollView 类,支持滚动的布局
import android.annotation.TargetApi;
import android.content.Context;
import android.os.Build;
import android.support.annotation.NonNull;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.widget.ScrollView;
// 创建 RefreshScrollView 类,继承自 ScrollView
public class RefreshScrollView extends ScrollView{
private RefreshProgressBar rpb; // 自定义的刷新进度条
private float durTouchY = -1000000; // 记录触摸Y坐标
private BaseRefreshListener baseRefreshListener; // 刷新监听器实例
private Boolean isRefreshing = false; // 是否正在刷新
private OnTouchListener touchListener; // 触摸事件监听器
private RefreshProgressBar rpb;
private float durTouchY = -1000000;
private BaseRefreshListener baseRefreshListener;
private Boolean isRefreshing = false;
private OnTouchListener touchListener;
// 构造函数
public RefreshScrollView(Context context) {
this(context,null);// 调用带有 AttributeSet 的构造函数
this(context,null);
}
public RefreshScrollView(Context context, AttributeSet attrs) {
this(context, attrs,0);// 调用带有风格属性的构造函数
this(context, attrs,0);
}
public RefreshScrollView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);// 调用父类构造函数
super(context, attrs, defStyleAttr);
}
@TargetApi(Build.VERSION_CODES.LOLLIPOP)// 指示支持 API 21 及以上
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public RefreshScrollView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);// 调用父类构造函数
super(context, attrs, defStyleAttr, defStyleRes);
}
// 设置刷新进度条
public void setRpb(@NonNull RefreshProgressBar rpb) {
this.rpb = rpb; // 赋值给 rpb
init(); // 初始化
this.rpb = rpb;
init();
}
// 初始化方法
private void init() {
// 设置触摸事件监听器
touchListener = new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
int action = event.getAction(); // 获取当前触摸事件的动作
int action = event.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN: // 触摸开始时
durTouchY = event.getY(); // 记录初始触摸Y坐标
case MotionEvent.ACTION_DOWN:
durTouchY = event.getY();
break;
case MotionEvent.ACTION_MOVE: // 触摸移动时
if (durTouchY == -1000000) // 确保初始触摸坐标已记录
durTouchY = event.getY(); // 更新触摸Y坐标
float dY = event.getY() - durTouchY; // 计算移动的Y距离向下则为正值
durTouchY = event.getY(); // 更新初始触摸坐标
// 如果满足刷新条件
if (baseRefreshListener != null && !isRefreshing && rpb.getSecondDurProgress() == rpb.getSecondFinalProgress() && getScrollY() <= 0) {
// 设置刷新进度条可见
case MotionEvent.ACTION_MOVE:
if (durTouchY == -1000000)
durTouchY = event.getY();
float dY = event.getY() - durTouchY; //>0下拉
durTouchY = event.getY();
if (baseRefreshListener != null && !isRefreshing && rpb.getSecondDurProgress() == rpb.getSecondFinalProgress() && getScrollY()<=0) {
if (rpb.getVisibility() != View.VISIBLE) {
rpb.setVisibility(View.VISIBLE);
}
// 更新进度条的当前进度
rpb.setSecondDurProgress((int) (rpb.getSecondDurProgress() + dY));
// 判断当前进度是否小于等于0
if (rpb.getSecondDurProgress() <= 0) {
return false; // 返回 false 以允许 ScrollView 处理事件
return false;
} else {
return true; // 返回 true 表示事件被处理
return true;
}
}
break;
case MotionEvent.ACTION_UP: // 触摸抬起时
// 如果满足刷新条件且当前进度大于0
case MotionEvent.ACTION_UP:
if (baseRefreshListener != null && rpb.getSecondMaxProgress() > 0 && rpb.getSecondDurProgress() > 0) {
// 如果当前进度达到最大值且不在刷新状态,则开始刷新
if (rpb.getSecondDurProgress() >= rpb.getSecondMaxProgress() && !isRefreshing) {
startRefresh();
} else {
// 否则,将进度条动画归零
rpb.setSecondDurProgressWithAnim(0);
}
}
durTouchY = -1000000; // 重置触摸坐标
durTouchY = -1000000;
break;
}
return false; // 事件未完全处理,允许 ScrollView 继续处理
return false;
}
};
this.setOnTouchListener(touchListener); // 设置触摸监听器
this.setOnTouchListener(touchListener);
}
// 设置刷新监听器
public void setBaseRefreshListener(BaseRefreshListener baseRefreshListener) {
this.baseRefreshListener = baseRefreshListener; // 赋值给监听器
this.baseRefreshListener = baseRefreshListener;
}
// 开始刷新
public void startRefresh() {
if (baseRefreshListener != null) {
isRefreshing = true; // 设置为正在刷新状态
rpb.setIsAutoLoading(true); // 设置进度条为自动加载状态
baseRefreshListener.startRefresh(); // 调用监听器的刷新方法
public void startRefresh(){
if(baseRefreshListener!=null){
isRefreshing = true;
rpb.setIsAutoLoading(true);
baseRefreshListener.startRefresh();
}
}
// 完成刷新
public void finishRefresh() {
isRefreshing = false; // 设置为未刷新状态
rpb.setDurProgress(0); // 重置进度条当前进度
rpb.setIsAutoLoading(false); // 退出自动加载状态
public void finishRefresh(){
isRefreshing = false;
rpb.setDurProgress(0);
rpb.setIsAutoLoading(false);
}
}

@ -1,9 +1,7 @@
<?xml version="1.0" encoding="utf-8"?> <!-- 声明此 XML 文档的版本和编码格式 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="500"> <!-- 定义动画的持续时间为500毫秒 -->
android:duration="500">
<translate
android:fromYDelta="100%"
android:toYDelta="0" />
<!-- 动画开始时Y轴位置为100%(即视图完全在屏幕下方) -->
<!-- 动画结束时Y轴位置为0即视图返回到其原始位置 -->
</set>

@ -1,9 +1,7 @@
<?xml version="1.0" encoding="utf-8"?> <!-- 声明此 XML 文档的版本和编码格式 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="500"> <!-- 动画持续时间为500毫秒 -->
android:duration="500">
<translate
android:fromYDelta="0"
android:toYDelta="100%" />
<!-- 动画开始时视图在 Y 轴的位置为 0其原始位置 -->
<!-- 动画结束时视图在 Y 轴上的位置为其自身高度的 100%(完全下移到视图下方) -->
</set>

@ -1,7 +1,7 @@
<?xml version="1.0" encoding="utf-8"?> <!-- 声明此 XML 文档的版本和编码格式 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="600"> <!-- 定义整个动画的持续时间为600毫秒 -->
android:duration="600">
<translate
android:fromYDelta="100%" <!-- 动画开始时,视图在 Y 轴的起始位置为其高度的 100%(即在屏幕底部之外) -->
android:toYDelta="0"/> <!-- 动画结束时,视图在 Y 轴的目标位置为 0即回到其原始位置 -->
android:fromYDelta="100%"
android:toYDelta="0"/>
</set>

@ -1,6 +1,6 @@
<?xml version="1.0" encoding="utf-8"?> <!-- 声明此 XML 文档的版本和编码格式 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="200"> <!-- 定义整个动画的持续时间为200毫秒 -->
<translate android:fromXDelta="-100%" <!-- X -100% -->
android:toXDelta="0"/> <!-- 动画结束时,视图在 X 轴的目标位置为0即回到其原始位置 -->
android:duration="200">
<translate android:fromXDelta="-100%"
android:toXDelta="0"/>
</set>

@ -1,6 +1,6 @@
<?xml version="1.0" encoding="utf-8"?><!-- 声明 XML 文档的版本和编码格式 -->
<set xmlns:android="http://schemas.android.com/apk/res/android"<!-- 使 Android -->
android:duration="200"> <!-- 设置动画的持续时间为 200 毫秒 -->
<translate android:fromXDelta="0" <!-- X 0 -->
android:toXDelta="-100%"/> <!-- 设置动画的结束 X 坐标变化量为 -100%,表示向左移动至视图的左侧 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="200">
<translate android:fromXDelta="0"
android:toXDelta="-100%"/>
</set>

@ -1,21 +1,12 @@
<?xml version="1.0" encoding="utf-8"?> <!-- 声明此 XML 文档的版本和编码格式 -->
<!-- 定义动画集合,使用 Android 的命名空间 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="300"> <!-- 设置动画的持续时间为 300 毫秒 -->
<!-- 定义缩放动画的起始 X 轴缩放比为 0.6 (60%) -->
android:duration="300">
<scale android:fromXScale="0.6"
<!-- 定义缩放动画的起始 Y 轴缩放比为 0.6 (60%) -->
android:fromYScale="0.6"
<!-- 定义缩放动画的结束 X 轴缩放比为 1.0 (100%) -->
android:toXScale="1.0"
<!-- 定义缩放动画的结束 Y 轴缩放比为 1.0 (100%) -->
android:toYScale="1.0"
<!-- 设置缩放的 X 轴支点位置为视图宽度的 50% (中心点) -->
android:pivotX="50%"
<!-- 设置缩放的 Y 轴支点位置为视图高度的 50% (中心点) -->
android:pivotY="50%"/>
<!-- 定义透明度动画的起始透明度为 0完全透明 -->
<alpha android:fromAlpha="0"
<!-- 定义透明度动画的结束透明度为 1不透明 -->
android:toAlpha="1"/>
</set>

@ -1,15 +1,12 @@
<?xml version="1.0" encoding="utf-8"?> <!-- 声明此 XML 文档的版本和编码格式 -->
<!-- 定义动画集合,使用 Android 的命名空间 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="300"> <!-- 设置整个动画的持续时间为 300 毫秒 -->
<scale android:fromXScale="1.0" <!-- X 1.0 (100%) -->
android:fromYScale="1.0" <!-- 定义缩放动画的起始 Y 轴缩放比为 1.0 (100%) -->
android:toXScale="0.6" <!-- 定义缩放动画的结束 X 轴缩放比为 0.6 (60%) -->
android:toYScale="0.6" <!-- 定义缩放动画的结束 Y 轴缩放比为 0.6 (60%) -->
android:pivotX="50%" <!-- 设置缩放的 X 轴支点位置为视图宽度的 50% (中心点) -->
android:pivotY="50%"/> <!-- 设置缩放的 Y 轴支点位置为视图高度的 50% (中心点) -->
<alpha android:fromAlpha="1" <!-- 1 -->
android:toAlpha="0"/> <!-- 定义透明度动画的结束透明度为 0完全透明 -->
android:duration="300">
<scale android:fromXScale="1.0"
android:fromYScale="1.0"
android:toXScale="0.6"
android:toYScale="0.6"
android:pivotX="50%"
android:pivotY="50%"/>
<alpha android:fromAlpha="1"
android:toAlpha="0"/>
</set>

@ -1,7 +1,6 @@
<?xml version="1.0" encoding="utf-8"?> <!-- 声明此 XML 文档的版本和编码格式 -->
<!-- 定义动画集合,使用 Android 的命名空间 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="200"> <!-- 设置整个动画的持续时间为 200 毫秒 -->
<translate android:fromXDelta="100%" <!-- 100% -->
android:toXDelta="0"/><!-- 定义平移动画的结束位置为 0视图移动到它的原始位置 -->
android:duration="200">
<translate android:fromXDelta="100%"
android:toXDelta="0"/>
</set>

@ -1,9 +1,6 @@
<?xml version="1.0" encoding="utf-8"?> <!-- 声明 XML 版本和编码方式 -->
<!-- 定义 XML 的命名空间指定Android属性 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="200"> <!-- 设置整体动画的持续时间为200毫秒 -->
android:duration="200">
<translate android:fromXDelta="0"
android:toXDelta="100%"/>
<!-- 动画开始时的水平偏移量从0开始 -->
<!-- 动画结束时的水平偏移量移动到视图宽度的100% -->
</set>

@ -1,7 +1,6 @@
<?xml version="1.0" encoding="utf-8"?> <!-- 声明 XML 版本和编码方式 -->
<!-- 定义 XML 的命名空间 -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:duration="200"><!-- 设置整体动画的持续时间为200毫秒 -->
android:duration="200">
<!--<translate-->
<!--android:fromYDelta="100%"-->
<!--android:toYDelta="0" />-->
@ -10,9 +9,4 @@
android:fromXScale="1"
android:toXScale="1"
android:pivotY="100%"/>
<!-- 动画开始时在Y轴上的缩放比例初始为0不可见 -->
<!-- 动画结束时在Y轴上的缩放比例设置为1正常大小 -->
<!-- X轴上的缩放比例保持为1不缩放 -->
<!-- X轴缩放结束时仍然为1不缩放 -->
<!-- 设置缩放的中心点在Y轴的100%位置(底部) -->
</set>

@ -1,18 +1,14 @@
// 声明当前类的包名
package com.monke.monkeybook;
import org.junit.Test; // 导入JUnit的@Test注解用于标识测试方法
import static org.junit.Assert.*;// 导入JUnit的断言静态方法用于验证测试结果
import org.junit.Test;
import static org.junit.Assert.*;
/**
* To work on unit tests, switch the Test Artifact in the Build Variants view.
*
*/
public class ExampleUnitTest {
// 标识这是一个JUnit测试方法
@Test
public void addition_isCorrect() throws Exception {
// 使用assertEquals方法检查两个值是否相等
assertEquals(4, 2 + 2);// 断言2 + 2的结果应该等于4
assertEquals(4, 2 + 2);
}
}

@ -1,14 +1,13 @@
// 声明包名,指明该类属于 com.monke.basemvplib 包
package com.monke.basemvplib;
// 导入需要的类
import android.app.Application;// 导入 Android 的 Application 类,使得可以对应用程序进行测试
import android.test.ApplicationTestCase;// 导入 ApplicationTestCase 类,这是用于测试 Application 的基类
import android.app.Application;
import android.test.ApplicationTestCase;
/**
* <a href="http://d.android.com/tools/testing/testing_android.html">Testing Fundamentals</a>
*/
public class ApplicationTest extends ApplicationTestCase<Application> {// 定义一个公共类 ApplicationTest继承自 ApplicationTestCase指定测试的类为 Application
public ApplicationTest() {// 定义构造函数
super(Application.class);// 调用父类的构造函数,传入 Application.class告知需要测试的应用类是 Application
public class ApplicationTest extends ApplicationTestCase<Application> {
public ApplicationTest() {
super(Application.class);
}
}

@ -1,6 +1,5 @@
// 定义包名
package com.monke.basemvplib;
// 导入所需的类
import android.app.Activity;
import com.monke.basemvplib.impl.BaseActivity;
@ -13,25 +12,18 @@ import java.util.List;
* Activity,Activity
*/
public class AppActivityManager {
// 使用静态变量存储Activity的弱引用列表避免内存泄漏
private static List<WeakReference<Activity>> activities;
private AppActivityManager(){// 私有构造函数,确保外部无法直接实例化
activities = new ArrayList<>();// 初始化Activity列表
private AppActivityManager(){
activities = new ArrayList<>();
}
// 声明一个volatile变量用于保存AppActivityManager的实例确保线程安全
private static volatile AppActivityManager instance;
/**
* AppActivityManager
* 使线
*/
public static AppActivityManager getInstance(){
// 如果instance为空则进入同步代码块
if(null == instance){
synchronized (AppActivityManager.class){
// 再次检查instance是否为空避免多线程同时创建多个实例
if(null == instance){
instance = new AppActivityManager();
}
@ -40,46 +32,36 @@ public class AppActivityManager {
return instance;
}
/**
* Activity
* @return Activity
*/
public List<WeakReference<Activity>> getActivities() {
return activities;// 返回存储Activity弱引用的列表
return activities;
}
/*
Activity
@param activity Activity
Activity
*/
public void add(Activity activity){
activities.add(new WeakReference<Activity>(activity));// 将Activity的弱引用添加到列表中
activities.add(new WeakReference<Activity>(activity));
}
/*
Activity
@param activity Activity
*/
public void remove(Activity activity){
for(WeakReference<Activity> temp :activities){// 遍历活动列表
if(null != temp.get() && temp.get() == activity){// 如果找到了对应的Activity
activities.remove(temp);// 从列表中移除
break;// 结束循环
for(WeakReference<Activity> temp :activities){
if(null != temp.get() && temp.get() == activity){
activities.remove(temp);
break;
}
}
}
/*
Activity
@param activityClass Activity
*/
public void remove(Class<?> activityClass){
for(Iterator<WeakReference<Activity>> iterator = activities.iterator();iterator.hasNext();){
// 获取下一个活动的弱引用
WeakReference<Activity> item = iterator.next();
// 判断是否匹配
if(null != item && null != item.get() && item.get().getClass() == activityClass){
// 移除匹配的Activity
iterator.remove();
}
}
@ -87,53 +69,47 @@ public class AppActivityManager {
/*
activity
@param activities Activity
*/
public void finishActivity(BaseActivity... activities){
for(int i=0;i<activities.length;i++){// 遍历传入的Activity数组
if(null != activities[i]){// 如果Activity不为null
activities[i].finish();// 关闭Activity
for(int i=0;i<activities.length;i++){
if(null != activities[i]){
activities[i].finish();
}
}
}
/*
activity(class)
@param activityClasses Activity
*/
public void finishActivity(Class<?>... activityClasses){
// 存储待关闭的Activity
ArrayList<WeakReference<Activity>> waitfinish = new ArrayList<>();
for(WeakReference<Activity> temp :activities){// 遍历当前管理的Activity
for(int i=0;i<activityClasses.length;i++){// 遍历需要关闭的Activity类数组
if(null != temp.get() && temp.get().getClass() == activityClasses[i]){// 判断是否匹配
waitfinish.add(temp);// 添加到待关闭列表
break;// 结束内层循环
for(WeakReference<Activity> temp :activities){
for(int i=0;i<activityClasses.length;i++){
if(null != temp.get() && temp.get().getClass() == activityClasses[i]){
waitfinish.add(temp);
break;
}
}
}
// 遍历待关闭的Activity
for(WeakReference<Activity> activityWeakReference:waitfinish){
if(null != activityWeakReference.get()){// 如果弱引用有效
activityWeakReference.get().finish();// 关闭Activity
if(null != activityWeakReference.get()){
activityWeakReference.get().finish();
}
}
}
/*
Activity
param activityClass Activity
@return
*/
public Boolean isExist(Class<?> activityClass){
Boolean result = false;// 默认不存在
for(Iterator<WeakReference<Activity>> iterator = activities.iterator();iterator.hasNext();){// 遍历活动列表
WeakReference<Activity> item = iterator.next();// 获取下一个活动的弱引用
if(null != item && null != item.get() && item.get().getClass() == activityClass){// 如果找到目标Activity
result = true;// 标记为存在
break;// 结束循环
Boolean result = false;
for(Iterator<WeakReference<Activity>> iterator = activities.iterator();iterator.hasNext();){
WeakReference<Activity> item = iterator.next();
if(null != item && null != item.get() && item.get().getClass() == activityClass){
result = true;
break;
}
}
return result;// 返回是否存在的结果
return result;
}
}

@ -1,12 +1,10 @@
// 定义包名
package com.monke.basemvplib;
// 导入所需的类
import android.app.Application;// 导入Android框架中的Application类
public class BaseApplication extends Application{//定义一个扩展自Application的基类
import android.app.Application;
public class BaseApplication extends Application{
@Override
public void onCreate() {
// 调用父类的方法
super.onCreate();
}
}

@ -1,52 +1,43 @@
// 定义包名
package com.monke.basemvplib;
// 导入所需的类
import java.io.IOException;// 导入 IOException 类,处理输入输出异常
import java.lang.annotation.Annotation;// 导入 Annotation 类的定义,使用 Java 反射时需要这个类
import java.lang.reflect.Type;// 导入 Type 接口,表示 Java 类型的反射信息
import java.nio.charset.Charset;// 导入 Charset 类,处理字符集
import okhttp3.ResponseBody;// 导入 OkHttp 库中的 ResponseBody 类,用于处理网络响应体
import okio.BufferedSource;// 导入 Okio 库中的 BufferedSource 类,用于高效处理流数据
import okio.Okio;// 导入 Okio 库中的 Okio 类,提供工具方法
import retrofit2.Converter;// 导入 Retrofit 库中的 Converter 类,为响应转换器提供基类
import retrofit2.Retrofit;// 导入 Retrofit 库中的 Retrofit 类,表示 Retrofit 的核心类
/**
* Retrofit ResponseBody
*/
import java.io.IOException;
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;
import java.nio.charset.Charset;
import okhttp3.ResponseBody;
import okio.BufferedSource;
import okio.Okio;
import retrofit2.Converter;
import retrofit2.Retrofit;
public class EncodoConverter extends Converter.Factory {
// 默认的编码方式
private String encode = "utf-8";
private EncodoConverter(){// 私有构造函数,禁止无参实例化
private EncodoConverter(){
}
private EncodoConverter(String encode){// 带编码方式的构造函数
private EncodoConverter(String encode){
this.encode = encode;
}
public static EncodoConverter create(){// 静态方法,创建默认编码的 EncodoConverter 实例
public static EncodoConverter create(){
return new EncodoConverter();
}
public static EncodoConverter create(String en){// 静态方法,创建指定编码的 EncodoConverter 实例
public static EncodoConverter create(String en){
return new EncodoConverter(en);
}
// 重写 responseBodyConverter 方法,返回自定义的转换器
@Override
public Converter<ResponseBody, String> responseBodyConverter(Type type, Annotation[] annotations,
Retrofit retrofit) {
// 返回一个新的字符串转换器
return new Converter<ResponseBody, String>() {
// 实现转换逻辑
@Override
public String convert(ResponseBody value) throws IOException {
// 使用 Okio 读取 ResponseBody 数据
BufferedSource bufferedSource = Okio.buffer(value.source());
// 以指定编码读取字符串
String responseData = bufferedSource.readString(Charset.forName(encode));
return responseData;// 返回读取的字符串
return responseData;
}
};
}

@ -1,11 +1,7 @@
// 定义包名
package com.monke.basemvplib;
// 导入支持库中的 NonNull 注解,用于标记非空参数
import android.support.annotation.NonNull;
/**
* IPresenter Presenter
*/
public interface IPresenter {
/**
* View使View

@ -1,16 +1,7 @@
// 定义包名
package com.monke.basemvplib;
// 导入 Context 类,这是 Android 中的一个基础类,表示应用程序的环境信息
import android.content.Context;
/**
* IView View
*/
public interface IView {
/**
* Context
* @return View
*/
public Context getContext();
}

@ -1,96 +1,79 @@
// 定义包名
package com.monke.basemvplib.impl;
// 导入需要的类和接口
import android.app.ActivityOptions;// 导入android.app.ActivityOptions类用于在Android 5.0Lollipop及以后的版本中创建Activity之间的转场动画。
import android.content.Context;// 导入android.content.Context类这是一个抽象类提供了一个接口来访问应用程序的全局信息。
import android.content.Intent;// 导入android.content.Intent类它是一种携带请求的消息对象用于在组件之间传递指令
import android.os.Build;// 导入android.os.Build类提供了一个接口来访问关于Android构建版本的信息。
import android.os.Bundle;// 导入android.os.Bundle类它是一个存储键值对的数据结构通常用于在Activity、Fragment等组件间传递数据。
import android.support.annotation.NonNull;// 导入android.support.annotation.NonNull注解用于标记方法参数或返回值不允许为null。
import android.view.View;// 导入android.view.View类它是所有UI组件的基类代表屏幕上的一个矩形区域。
import com.monke.basemvplib.AppActivityManager;// 导入com.monke.basemvplib.AppActivityManager类这是一个管理所有Activity的类可能用于管理Activity的生命周期等。
import com.monke.basemvplib.IPresenter;// 导入com.monke.basemvplib.IPresenter接口它定义了MVP模式中的Presenter层的行为。
import com.monke.basemvplib.IView;// 导入com.monke.basemvplib.IView接口它定义了MVP模式中的View层的行为。
// 导入com.trello.rxlifecycle2.components.support.RxAppCompatActivity类这是一个支持RxJava生命周期处理的Activity基类。
import android.app.ActivityOptions;
import android.content.Context;
import android.content.Intent;
import android.os.Build;
import android.os.Bundle;
import android.support.annotation.NonNull;
import android.view.View;
import com.monke.basemvplib.AppActivityManager;
import com.monke.basemvplib.IPresenter;
import com.monke.basemvplib.IView;
import com.trello.rxlifecycle2.components.support.RxAppCompatActivity;
// 定义 BaseActivity 类,它是一个抽象类,用于继承 RxAppCompatActivity 并实现 IView 接口
public abstract class BaseActivity<T extends IPresenter> extends RxAppCompatActivity implements IView {
// 定义一个常量字符串用于在Intent中传递是否需要启动共享元素动画的标志
public final static String start_share_ele= "start_with_share_ele";
protected Bundle savedInstanceState;// 定义一个Bundle类型的变量用于存储保存的实例状态
protected T mPresenter;// 定义一个泛型变量mPresenter用于存储与当前Activity相关联的Presenter
private Boolean startShareAnim = false;// 定义一个布尔变量,用于标记是否启动共享元素动画
protected Bundle savedInstanceState;
protected T mPresenter;
private Boolean startShareAnim = false;
@Override
// onCreate 方法是 Activity 生命周期中的第一个回调方法,用于初始化 Activity
protected void onCreate(Bundle savedInstanceState) {
// 调用超类的 onCreate 方法
super.onCreate(savedInstanceState);
// 将传入的 savedInstanceState 赋值给成员变量
this.savedInstanceState = savedInstanceState;
// 如果 getIntent() 不为空,则从 Intent 中获取是否需要启动共享元素动画的标志
if(getIntent()!=null){
startShareAnim = getIntent().getBooleanExtra(start_share_ele,false);
}
// 将当前 Activity 添加到 AppActivityManager 的管理列表中
AppActivityManager.getInstance().add(this);
// 初始化 SDK
initSDK();
// 调用抽象方法 onCreateActivity用于设置布局内容
onCreateActivity();
// 调用抽象方法 initInjector用于初始化 Presenter
mPresenter = initInjector();
// 将 View 绑定到 Presenter
attachView();
initData();// 调用抽象方法 initData用于初始化数据
bindView();// 调用方法 bindView用于绑定视图
bindEvent();// 调用方法 bindEvent用于绑定事件
firstRequest();// 调用方法 firstRequest用于执行首次逻辑操作
initData();
bindView();
bindEvent();
firstRequest();
}
/**
*
*/
protected void firstRequest() { // 首次逻辑操作的方法,子类需要根据需要实现具体的逻辑
protected void firstRequest() {
}
/**
*
*/
protected void bindEvent() {// 事件触发绑定的方法,子类需要根据需要实现具体的事件绑定逻辑
protected void bindEvent() {
}
/**
*
*/
protected void bindView() {// 控件绑定的方法,子类需要根据需要实现具体的控件绑定逻辑
protected void bindView() {
}
/**
* PV
*/
private void attachView() {// P层绑定V层的方法将 Presenter 和 View 绑定在一起
// 如果 mPresenter 不为空,则调用其 attachView 方法,将当前 View 绑定到 Presenter
private void attachView() {
if (null != mPresenter) {
mPresenter.attachView(this);
}
}
// onPause 方法是 Activity 生命周期中的回调方法,当 Activity 不可见时调用
@Override
protected void onPause() {
// 调用超类的 onPause 方法
super.onPause();
}
/**
* PV
*/
private void detachView() {// P层解绑V层的方法将 Presenter 和 View 解绑
// 如果 mPresenter 不为空,则调用其 detachView 方法,将当前 View 从 Presenter 解绑
private void detachView() {
if (null != mPresenter) {
mPresenter.detachView();
}
@ -99,7 +82,7 @@ public abstract class BaseActivity<T extends IPresenter> extends RxAppCompatActi
/**
* SDK
*/
protected void initSDK() {// SDK初始化的方法子类需要根据需要实现具体的 SDK 初始化逻辑
protected void initSDK() {
}
@ -108,73 +91,64 @@ public abstract class BaseActivity<T extends IPresenter> extends RxAppCompatActi
*
* @return
*/
protected abstract T initInjector();// P层绑定的方法子类需要根据需要实现具体的 Presenter 初始化逻辑,并返回对应的 Presenter 实例
protected abstract T initInjector();
/**
* setContentView()
*/
protected abstract void onCreateActivity(); // 布局载入的方法,子类需要根据需要实现具体的布局载入逻辑
protected abstract void onCreateActivity();
/**
*
*/
protected abstract void initData();// 数据初始化的方法,子类需要根据需要实现具体的数据初始化逻辑
protected abstract void initData();
@Override
protected void onResume() {// onResume 方法是 Activity 生命周期中的回调方法,当 Activity 可见时调用
super.onResume();// 调用超类的 onResume 方法
protected void onResume() {
super.onResume();
}
@Override
protected void onDestroy() {// onDestroy 方法是 Activity 生命周期中的回调方法,当 Activity 销毁时调用
super.onDestroy();// 调用超类的 onDestroy 方法
detachView();// 调用 detachView 方法,将 View 从 Presenter 解绑
AppActivityManager.getInstance().remove(this);// 将当前 Activity 从 AppActivityManager 的管理列表中移除
protected void onDestroy() {
super.onDestroy();
detachView();
AppActivityManager.getInstance().remove(this);
}
////////////////////////////////启动Activity转场动画/////////////////////////////////////////////
protected void startActivityForResultByAnim(Intent intent, int requestCode, int animIn, int animExit) {// 通过动画启动一个新的 Activity并指定进入和退出的动画资源 ID
startActivityForResult(intent, requestCode);// 调用 startActivityForResult 方法启动新的 Activity并指定请求码
overridePendingTransition(animIn, animExit);// 调用 overridePendingTransition 方法设置进入和退出的动画
protected void startActivityForResultByAnim(Intent intent, int requestCode, int animIn, int animExit) {
startActivityForResult(intent, requestCode);
overridePendingTransition(animIn, animExit);
}
protected void startActivityByAnim(Intent intent, int animIn, int animExit) {// 通过动画启动一个新的 Activity并指定进入和退出的动画资源 ID
startActivity(intent);// 调用 startActivity 方法启动新的 Activity
overridePendingTransition(animIn, animExit);// 调用 overridePendingTransition 方法设置进入和退出的动画
protected void startActivityByAnim(Intent intent, int animIn, int animExit) {
startActivity(intent);
overridePendingTransition(animIn, animExit);
}
// 通过共享元素动画启动一个新的 Activity并指定进入和退出的动画资源 ID
protected void startActivityForResultByAnim(Intent intent, int requestCode, @NonNull View view, @NonNull String transitionName, int animIn, int animExit) {
// 如果当前系统版本大于等于 LOLLIPOP则使用共享元素动画
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
// 调用 startActivityForResult 方法启动新的 Activity并指定请求码和共享元素动画的参数
startActivityForResult(intent, requestCode, ActivityOptions.makeSceneTransitionAnimation(this, view, transitionName).toBundle());
} else {
// 如果当前系统版本小于 LOLLIPOP则使用普通动画启动 Activity
startActivityForResultByAnim(intent, requestCode, animIn, animExit);
}
}
// 通过共享元素动画启动一个新的 Activity并指定进入和退出的动画资源 ID
protected void startActivityByAnim(Intent intent, @NonNull View view, @NonNull String transitionName, int animIn, int animExit) {
// 如果当前系统版本大于等于 LOLLIPOP则使用共享元素动画
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
// 将是否需要启动共享元素动画的标志放入 Intent
intent.putExtra(start_share_ele,true);
// 调用 startActivity 方法启动新的 Activity并指定共享元素动画的参数
startActivity(intent, ActivityOptions.makeSceneTransitionAnimation(this, view, transitionName).toBundle());
} else {
// 如果当前系统版本小于 LOLLIPOP则使用普通动画启动 Activity
startActivityByAnim(intent, animIn, animExit);
}
}
public Context getContext(){// 获取当前 Context 对象的方法
public Context getContext(){
return this;
}
public Boolean getStart_share_ele() {// 获取是否需要启动共享元素动画的标志的方法
public Boolean getStart_share_ele() {
return startShareAnim;
}
}

@ -1,82 +1,67 @@
// 定义包名
package com.monke.basemvplib.impl;
// 导入需要的类和接口
import android.os.Bundle;// 导入android.os.Bundle类它是一个存储键值对的数据结构通常用于在Activity、Fragment等组件间传递数据。
import android.support.annotation.Nullable;// 导入android.support.annotation.Nullable注解用于标记方法参数、返回值或字段可以为null。
import android.view.LayoutInflater;// 导入android.view.LayoutInflater类它用于在Activity或Fragment中实例化视图。
import android.view.View;// 导入android.view.View类它是所有UI组件的基类代表屏幕上的一个矩形区域。
import android.view.ViewGroup;// 导入android.view.ViewGroup类它是一个视图的容器可以包含其他的View或ViewGroup。
import com.monke.basemvplib.IPresenter;// 导入com.monke.basemvplib.IPresenter接口它定义了MVP模式中的Presenter层的行为。
import com.monke.basemvplib.IView;// 导入com.monke.basemvplib.IView接口它定义了MVP模式中的View层的行为。
// 导入com.trello.rxlifecycle2.components.RxFragment类这是一个支持RxJava生命周期处理的Fragment基类。
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import com.monke.basemvplib.IPresenter;
import com.monke.basemvplib.IView;
import com.trello.rxlifecycle2.components.RxFragment;
// 定义 BaseFragment 类,它是一个抽象类,用于继承 RxFragment 并实现 IView 接口
public abstract class BaseFragment<T extends IPresenter> extends RxFragment implements IView{
// 定义一个 View 类型的变量,用于存储 Fragment 的视图
protected View view;
// 定义一个 Bundle 类型的变量,用于存储保存的实例状态
protected Bundle savedInstanceState;
// onCreateView 方法是 Fragment 生命周期中的第一个回调方法,用于初始化 Fragment
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
// 将传入的 savedInstanceState 赋值给成员变量
this.savedInstanceState = savedInstanceState;
// 初始化第三方 SDK
initSDK();
// 调用抽象方法 createView用于创建视图
view = createView(inflater, container);
// 调用方法 initData用于初始化数据
initData();
// 调用方法 bindView用于绑定视图
bindView();
// 调用方法 bindEvent用于绑定事件
bindEvent();
// 调用方法 firstRequest用于执行首次逻辑操作
firstRequest();
// 返回创建的视图
return view;
}
/**
*
*/
protected void bindEvent() {// 事件触发绑定的方法,子类需要根据需要实现具体的事件绑定逻辑
protected void bindEvent() {
}
/**
*
*/
protected void bindView() {// 控件绑定的方法,子类需要根据需要实现具体的控件绑定逻辑
protected void bindView() {
}
/**
*
*/
protected void initData() {// 数据初始化的方法,子类需要根据需要实现具体的数据初始化逻辑
protected void initData() {
}
/**
*
*/
protected void firstRequest() { // 首次逻辑操作的方法,子类需要根据需要实现具体的首次逻辑操作
protected void firstRequest() {
}
/**
*
*/
// 加载布局的抽象方法,子类需要根据需要实现具体的布局加载逻辑
protected abstract View createView(LayoutInflater inflater, ViewGroup container);
/**
* SDK
*/
protected void initSDK() {// 第三方SDK初始化的方法子类需要根据需要实现具体的第三方 SDK 初始化逻辑
protected void initSDK() {
}
}

@ -1,6 +1,5 @@
// 定义包名
package com.monke.basemvplib.impl;
// 导入所需的类
import com.monke.basemvplib.EncodoConverter;
import java.util.concurrent.TimeUnit;
@ -10,33 +9,31 @@ import retrofit2.Retrofit;
import retrofit2.adapter.rxjava2.RxJava2CallAdapterFactory;
import retrofit2.converter.scalars.ScalarsConverterFactory;
public class BaseModelImpl {// 定义 BaseModelImpl 类
// 定义一个 OkHttpClient.Builder 类型的保护变量,用于构建 OkHttpClient 实例
public class BaseModelImpl {
protected OkHttpClient.Builder clientBuilder = new OkHttpClient.Builder()
.connectTimeout(10, TimeUnit.SECONDS)// 设置连接超时时间为10秒
.writeTimeout(10, TimeUnit.SECONDS)// 设置写超时时间为10秒
.readTimeout(10, TimeUnit.SECONDS)// 设置读超时时间为10秒
.addNetworkInterceptor(new RetryIntercepter(1));// 添加一个网络拦截器,用于重试机制
.connectTimeout(10, TimeUnit.SECONDS)
.writeTimeout(10, TimeUnit.SECONDS)
.readTimeout(10, TimeUnit.SECONDS)
.addNetworkInterceptor(new RetryIntercepter(1));
protected Retrofit getRetrofitObject(String url) {
// 创建 Retrofit 实例,设置 baseUrl、转换工厂、调用适配器和客户端
return new Retrofit.Builder().baseUrl(url)
//增加返回值为字符串的支持(以实体类返回)
.addConverterFactory(ScalarsConverterFactory.create())
//增加返回值为Oservable<T>的支持
.addCallAdapterFactory(RxJava2CallAdapterFactory.create())
.client(clientBuilder.build())// 设置客户端
.build();// 构建 Retrofit 实例
.client(clientBuilder.build())
.build();
}
protected Retrofit getRetrofitString(String url, String encode) {
// 创建 Retrofit 实例,设置 baseUrl、转换工厂、调用适配器和客户端
return new Retrofit.Builder().baseUrl(url)
//增加返回值为字符串的支持(以实体类返回),使用自定义的编码转换器
//增加返回值为字符串的支持(以实体类返回)
.addConverterFactory(EncodoConverter.create(encode))
//增加返回值为Oservable<T>的支持
.addCallAdapterFactory(RxJava2CallAdapterFactory.create())
.client(clientBuilder.build()) // 设置客户端
.build();// 构建 Retrofit 实例
.client(clientBuilder.build())
.build();
}
}

@ -1,17 +1,14 @@
// 定义包名
package com.monke.basemvplib.impl;
// 导入所需的类和接口
import android.support.annotation.NonNull;// 导入android.support.annotation.NonNull注解用于标记方法参数或返回值不允许为null。
import com.monke.basemvplib.IPresenter;// 导入com.monke.basemvplib.IPresenter接口它定义了MVP模式中的Presenter层的行为和职责。
import com.monke.basemvplib.IView;// 导入com.monke.basemvplib.IView接口它定义了MVP模式中的View层的行为和职责。
// 定义 BasePresenterImpl 类,它是一个抽象类,用于实现 IPresenter 接口
public abstract class BasePresenterImpl<T extends IView> implements IPresenter{// 定义一个泛型变量 mView它是 IView 接口的子类,用于持有 View 层的引用
import android.support.annotation.NonNull;
import com.monke.basemvplib.IPresenter;
import com.monke.basemvplib.IView;
public abstract class BasePresenterImpl<T extends IView> implements IPresenter{
protected T mView;
@Override
public void attachView(@NonNull IView iView) {
// 将传入的 IView 类型参数强制转换为泛型 T 类型,并赋值给 mView
mView = (T) iView;
}
}

@ -1,35 +1,27 @@
// 定义包名
package com.monke.basemvplib.impl;
// 导入所需的类
import java.io.IOException;// 导入java.io.IOException类它是所有I/O错误的基类用于在网络请求过程中捕获和处理I/O异常。
import okhttp3.Interceptor;// 导入okhttp3.Interceptor类它是OkHttp库中用于拦截和处理网络请求和响应的接口。
import okhttp3.Request;// 导入okhttp3.Request类它表示一个HTTP请求包括请求方法、URL、头部和正文。
import okhttp3.Response;// 导入okhttp3.Response类它表示一个HTTP响应包括状态码、头部、正文和一个可以关闭连接的方法。
import java.io.IOException;
import okhttp3.Interceptor;
import okhttp3.Request;
import okhttp3.Response;
// 定义 RetryIntercepter 类,实现 Interceptor 接口
public class RetryIntercepter implements Interceptor {
public int maxRetry;//最大重试次数
private int retryNum = 0;//假如设置为3次重试的话则最大可能请求4次默认1次+3次重试
public RetryIntercepter(int maxRetry) {// 构造函数,用于初始化最大重试次数
public RetryIntercepter(int maxRetry) {
this.maxRetry = maxRetry;
}
@Override
public Response intercept(Chain chain) throws IOException {
// 获取请求对象
Request request = chain.request();
// 执行请求并获取响应
Response response = chain.proceed(request);
// 当响应不成功且重试次数小于最大重试次数时,进行重试
while (!response.isSuccessful() && retryNum < maxRetry) {
// 重试次数加1
retryNum++;
// 再次执行请求
response = chain.proceed(request);
}
// 返回最终的响应对象
return response;
}
}

@ -1,18 +1,15 @@
//导入包名
package com.monke.basemvplib;
import org.junit.Test;// 导入JUnit的@Test注解用于标记测试方法
import org.junit.Test;
import static org.junit.Assert.*;// 导入JUnit的断言静态方法方便在测试中使用各种断言
import static org.junit.Assert.*;
/**
* To work on unit tests, switch the Test Artifact in the Build Variants view.
*
*/
public class ExampleUnitTest {
@Test // 使用 @Test 注解标识这个方法是一个测试方法
@Test
public void addition_isCorrect() throws Exception {
// 使用assertEquals方法检查两个值是否相等
assertEquals(4, 2 + 2);// 断言2 + 2的结果应该等于4
assertEquals(4, 2 + 2);
}
}
Loading…
Cancel
Save