上节我们为DrySister编写了一个异步图片加载缓存框架——SisterLoader(妹子加载器) 成功的从网络加载的图片缓存到了磁盘和内存中,当我们断开网络后,仍然能够查看这些图片, 但是,细心的你可能发现了一个很尴尬的地方,我们在有网的情况下进入APP,获取到图片相关 的信息,比如URL,如果退出了,断网,然后进来,图片就加载不出来了,图片已经缓存了,但是 我们没有图片对应的URL,就显得有些鸡肋了。所以我们需要对后台接口返回的数据进行存储, 当每次加载的时候,把返回的数据信息存储到本地,然后当无网的时候,可以让他加载本地 数据!所以本节我们为DrySister添加SQLite用于存储相关数据。任务都说清楚了,搞起~

在develop上开辟一个新的db分支,来完成本节相关代码的编写。
首先,我们添加一个用来判断网络状态的工具类:NetworkUtils.java, 就一个简单的判断网络是否可用的方法~这里我们还需要在AndroidManifest.xml 里加一个权限:android.permission.ACCESSNETWORKSTATE"的权限!
public class NetworkUtils {
/** 获取网络信息 */
private static NetworkInfo getActiveNetworkInfo(Context context) {
ConnectivityManager cm = (ConnectivityManager) context
.getSystemService(Context.CONNECTIVITY_SERVICE);
return cm.getActiveNetworkInfo();
}
/** 判断网络是否可用 */
public static boolean isAvailable(Context context) {
NetworkInfo info = getActiveNetworkInfo(context);
return info != null && info.isAvailable();
}
}
接着定义一个数据库字段常量的类:TableDefine.java, 在里面把数据库名,字段写上:
public class TableDefine {
public static final String TABLE_FULI = "fuli";
public static final String COLUMN_ID = "id";
public static final String COLUMN_FULI_ID = "_id";
public static final String COLUMN_FULI_CREATEAT = "createAt";
public static final String COLUMN_FULI_DESC = "desc";
public static final String COLUMN_FULI_PUBLISHEDAT = "publishedAt";
public static final String COLUMN_FULI_SOURCE = "source";
public static final String COLUMN_FULI_TYPE = "type";
public static final String COLUMN_FULI_URL = "url";
public static final String COLUMN_FULI_USED = "used";
public static final String COLUMN_FULI_WHO = "who";
}
再接着编写数据库的创建类:SisterOpenHelper.java,在里面完成数据库的创建
public class SisterOpenHelper extends SQLiteOpenHelper{
private static final String DB_NAME = "sister.db"; //数据库名
private static final int DB_VERSION = 1; //数据库版本号
public SisterOpenHelper(Context context) {
super(context, DB_NAME, null, DB_VERSION);
}
@Override
public void onCreate(SQLiteDatabase db) {
String createTableSql = "CREATE TABLE IF NOT EXISTS " + TableDefine.TABLE_FULI + " ("
+ TableDefine.COLUMN_ID + " INTEGER PRIMARY KEY AUTOINCREMENT, "
+ TableDefine.COLUMN_FULI_ID + " TEXT, "
+ TableDefine.COLUMN_FULI_CREATEAT + " TEXT, "
+ TableDefine.COLUMN_FULI_DESC + " TEXT, "
+ TableDefine.COLUMN_FULI_PUBLISHEDAT + " TEXT, "
+ TableDefine.COLUMN_FULI_SOURCE + " TEXT, "
+ TableDefine.COLUMN_FULI_TYPE + " TEXT, "
+ TableDefine.COLUMN_FULI_URL + " TEXT, "
+ TableDefine.COLUMN_FULI_USED + " BOOLEAN, "
+ TableDefine.COLUMN_FULI_WHO + " TEXT"
+ ")";
db.execSQL(createTableSql);
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { }
}
再接着,我们需要编写一个数据库的操作类,就是做增删改查,分页查询等操作的类, 这里我们把这个类写成单例的:SisterDBHelper.java,里面的编写的方法有这些:

具体代码:
public class SisterDBHelper {
private static final String TAG = "SisterDBHelper";
private static SisterDBHelper dbHelper;
private SisterOpenHelper sqlHelper;
private SQLiteDatabase db;
private SisterDBHelper() {
sqlHelper = new SisterOpenHelper(DrySisterApp.getContext());
}
/** 单例 */
public static SisterDBHelper getInstance() {
if(dbHelper == null) {
synchronized (SisterDBHelper.class) {
if(dbHelper == null) {
dbHelper = new SisterDBHelper();
}
}
}
return dbHelper;
}
/** 插入一个妹子 */
public void insertSister(Sister sister) {
db = getWritableDB();
ContentValues contentValues = new ContentValues();
contentValues.put(TableDefine.COLUMN_FULI_ID,sister.get_id());
contentValues.put(TableDefine.COLUMN_FULI_CREATEAT,sister.getCreateAt());
contentValues.put(TableDefine.COLUMN_FULI_DESC,sister.getDesc());
contentValues.put(TableDefine.COLUMN_FULI_PUBLISHEDAT,sister.getPublishedAt());
contentValues.put(TableDefine.COLUMN_FULI_SOURCE,sister.getSource());
contentValues.put(TableDefine.COLUMN_FULI_TYPE,sister.getType());
contentValues.put(TableDefine.COLUMN_FULI_URL,sister.getUrl());
contentValues.put(TableDefine.COLUMN_FULI_USED,sister.getUsed());
contentValues.put(TableDefine.COLUMN_FULI_WHO,sister.getWho());
db.insert(TableDefine.TABLE_FULI,null,contentValues);
closeIO(null);
}
/** 插入一堆妹子(使用事务) */
public void insertSisters(ArrayList<Sister> sisters) {
db = getWritableDB();
db.beginTransaction();
try{
for (Sister sister: sisters) {
ContentValues contentValues = new ContentValues();
contentValues.put(TableDefine.COLUMN_FULI_ID,sister.get_id());
contentValues.put(TableDefine.COLUMN_FULI_CREATEAT,sister.getCreateAt());
contentValues.put(TableDefine.COLUMN_FULI_DESC,sister.getDesc());
contentValues.put(TableDefine.COLUMN_FULI_PUBLISHEDAT,sister.getPublishedAt());
contentValues.put(TableDefine.COLUMN_FULI_SOURCE,sister.getSource());
contentValues.put(TableDefine.COLUMN_FULI_TYPE,sister.getType());
contentValues.put(TableDefine.COLUMN_FULI_URL,sister.getUrl());
contentValues.put(TableDefine.COLUMN_FULI_USED,sister.getUsed());
contentValues.put(TableDefine.COLUMN_FULI_WHO,sister.getWho());
db.insert(TableDefine.TABLE_FULI,null,contentValues);
}
db.setTransactionSuccessful();
} finally {
if(db != null && db.isOpen()) {
db.endTransaction();
closeIO(null);
}
}
}
/** 删除妹子(根据_id) */
public void deleteSister(String _id) {
db = getWritableDB();
db.delete(TableDefine.TABLE_FULI,"_id =?",new String[]{_id});
closeIO(null);
}
/** 删除所有妹子 */
public void deleteAllSisters() {
db = getWritableDB();
db.delete(TableDefine.TABLE_FULI,null,null);
closeIO(null);
}
/** 更新妹子信息(根据_id) */
public void deleteSister(String _id,Sister sister) {
db = getWritableDB();
ContentValues contentValues = new ContentValues();
contentValues.put(TableDefine.COLUMN_FULI_ID,sister.get_id());
contentValues.put(TableDefine.COLUMN_FULI_CREATEAT,sister.getCreateAt());
contentValues.put(TableDefine.COLUMN_FULI_DESC,sister.getDesc());
contentValues.put(TableDefine.COLUMN_FULI_PUBLISHEDAT,sister.getPublishedAt());
contentValues.put(TableDefine.COLUMN_FULI_SOURCE,sister.getSource());
contentValues.put(TableDefine.COLUMN_FULI_TYPE,sister.getType());
contentValues.put(TableDefine.COLUMN_FULI_URL,sister.getUrl());
contentValues.put(TableDefine.COLUMN_FULI_USED,sister.getUsed());
contentValues.put(TableDefine.COLUMN_FULI_WHO,sister.getWho());
db.update(TableDefine.TABLE_FULI,contentValues,"_id =?",new String[]{_id});
closeIO(null);
}
/** 查询当前表中有多少个妹子 */
public int getSistersCount() {
db = getReadableDB();
Cursor cursor = db.rawQuery("SELECT COUNT (*) FROM " + TableDefine.TABLE_FULI,null);
cursor.moveToFirst();
int count = cursor.getInt(0);
Log.v(TAG,"count:" + count);
closeIO(cursor);
return count;
}
/** 分页查询妹子,参数为当前页和每一个的数量,页数从0开始算 */
public List<Sister> getSistersLimit(int curPage,int limit) {
db = getReadableDB();
List<Sister> sisters = new ArrayList<>();
String startPos = String.valueOf(curPage * limit); //数据开始位置
if(db != null) {
Cursor cursor = db.query(TableDefine.TABLE_FULI,new String[] {
TableDefine.COLUMN_FULI_ID, TableDefine.COLUMN_FULI_CREATEAT,
TableDefine.COLUMN_FULI_DESC, TableDefine.COLUMN_FULI_PUBLISHEDAT,
TableDefine.COLUMN_FULI_SOURCE, TableDefine.COLUMN_FULI_TYPE,
TableDefine.COLUMN_FULI_URL, TableDefine.COLUMN_FULI_USED,
TableDefine.COLUMN_FULI_WHO,
},null,null,null,null,TableDefine.COLUMN_ID,startPos + "," + limit);
while (cursor.moveToNext()) {
Sister sister = new Sister();
sister.set_id(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_ID)));
sister.setCreateAt(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_CREATEAT)));
sister.setDesc(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_DESC)));
sister.setPublishedAt(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_PUBLISHEDAT)));
sister.setSource(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_SOURCE)));
sister.setType(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_TYPE)));
sister.setUrl(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_URL)));
sister.setUsed(cursor.getInt(cursor.getColumnIndex(TableDefine.COLUMN_FULI_USED)));
sisters.add(sister);
}
closeIO(cursor);
}
return sisters;
}
/** 查询所有妹子 */
public List<Sister> getAllSisters() {
db = getReadableDB();
List<Sister> sisters = new ArrayList<>();
Cursor cursor = db.rawQuery("SELECT * FROM "+TableDefine.TABLE_FULI,null);
cursor.moveToFirst();
while (cursor.moveToNext()) {
Sister sister = new Sister();
sister.set_id(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_ID)));
sister.setCreateAt(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_CREATEAT)));
sister.setDesc(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_DESC)));
sister.setPublishedAt(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_PUBLISHEDAT)));
sister.setSource(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_SOURCE)));
sister.setType(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_TYPE)));
sister.setUrl(cursor.getString(cursor.getColumnIndex(TableDefine.COLUMN_FULI_URL)));
sister.setUsed(cursor.getInt(cursor.getColumnIndex(TableDefine.COLUMN_FULI_USED)));
sisters.add(sister);
}
closeIO(cursor);
return sisters;
}
/** 获得可写数据库的方法 */
private SQLiteDatabase getWritableDB() {
return sqlHelper.getWritableDatabase();
}
/** 获得可读数据库的方法 */
private SQLiteDatabase getReadableDB() {
return sqlHelper.getReadableDatabase();
}
/** 关闭cursor和数据库的方法 */
private void closeIO(Cursor cursor) {
if(cursor != null) {
cursor.close();
}
if(db != null) {
db.close();
}
}
}
嗯,还是蛮简单的,都是一些SQLite的方法调用而已,接着我们对布局做一下更改, 把activity_main.xml改成下面的代码,就是换成上一个和下一个这样:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<Button
android:id="@+id/btn_previous"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:visibility="invisible"
android:text="上一个"/>
<Button
android:id="@+id/btn_next"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_toRightOf="@id/btn_previous"
android:text="下一个"/>
<ImageView
android:id="@+id/img_show"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_above="@+id/btn_previous"/>
</RelativeLayout>
最后就是对MainActivity.java的一些逻辑更改了,核心的要点如下:
代码实现:
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
private Button previousBtn;
private Button nextBtn;
private ImageView showImg;
private ArrayList<Sister> data;
private int curPos = 0; //当前显示的是哪一张
private int page = 1; //当前页数
private PictureLoader loader;
private SisterApi sisterApi;
private SisterTask sisterTask;
private SisterLoader mLoader;
private SisterDBHelper mDbHelper;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
sisterApi = new SisterApi();
loader = new PictureLoader();
mLoader = SisterLoader.getInstance(MainActivity.this);
mDbHelper = SisterDBHelper.getInstance();
initData();
initUI();
}
private void initData() {
data = new ArrayList<>();
sisterTask = new SisterTask();
sisterTask.execute();
}
private void initUI() {
previousBtn = (Button) findViewById(R.id.btn_previous);
nextBtn = (Button) findViewById(R.id.btn_next);
showImg = (ImageView) findViewById(R.id.img_show);
previousBtn.setOnClickListener(this);
nextBtn.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.btn_previous:
--curPos;
if (curPos == 0) {
previousBtn.setVisibility(View.INVISIBLE);
}
if (curPos == data.size() - 1) {
sisterTask = new SisterTask();
sisterTask.execute();
} else if(curPos < data.size()) {
mLoader.bindBitmap(data.get(curPos).getUrl(), showImg, 400, 400);
}
break;
case R.id.btn_next:
previousBtn.setVisibility(View.VISIBLE);
if(curPos < data.size()) {
++curPos;
}
if (curPos > data.size() - 1) {
sisterTask = new SisterTask();
sisterTask.execute();
} else if(curPos < data.size()){
mLoader.bindBitmap(data.get(curPos).getUrl(), showImg, 400, 400);
}
break;
}
}
private class SisterTask extends AsyncTask<Void, Void, ArrayList<Sister>> {
public SisterTask() {
}
@Override
protected ArrayList<Sister> doInBackground(Void... params) {
ArrayList<Sister> result = new ArrayList<>();
if (page < (curPos + 1) / 10 + 1) {
++page;
}
//判断是否有网络
if (NetworkUtils.isAvailable(getApplicationContext())) {
result = sisterApi.fetchSister(10, page);
//查询数据库里有多少个妹子,避免重复插入
if(mDbHelper.getSistersCount() / 10 < page) {
mDbHelper.insertSisters(result);
}
} else {
result.clear();
result.addAll(mDbHelper.getSistersLimit(page - 1, 10));
}
return result;
}
@Override
protected void onPostExecute(ArrayList<Sister> sisters) {
super.onPostExecute(sisters);
data.addAll(sisters);
if (data.size() > 0 && curPos + 1 < data.size()) {
mLoader.bindBitmap(data.get(curPos).getUrl(), showImg, 400, 400);
}
}
@Override
protected void onCancelled() {
super.onCancelled();
sisterTask = null;
}
}
@Override
protected void onDestroy() {
super.onDestroy();
if (sisterTask != null) {
sisterTask.cancel(true);
}
}
}
操作步骤:
把项目跑起来后,一直按下一页下一页缓存一堆图片,接着断开网络,进入DrySister, Duang~,神奇的发现,有妹子出现了,按下一页或者上一页也可以切换图片,妈妈再也 不用担心我在没网的时候不可以看DrySister了!
运行截图:

好的,本节在上节的基础上行添加了SQlite保存后台数据,当无网的时候也可以查看 妹子图片,代码的大的改动如上述,还有一些小改,具体看代码。 最后把db分支的代码合并到develop分支上,然后删除db分支,把更新后的 develop分支推送到Github!
代码下载:
https://github.com/coder-pig/DrySister/tree/develop 欢迎follow,star,觉得有什么想加进来的可以提下issues!
当我使用Bundler时,是否需要在我的Gemfile中将其列为依赖项?毕竟,我的代码中有些地方需要它。例如,当我进行Bundler设置时:require"bundler/setup" 最佳答案 没有。您可以尝试,但首先您必须用鞋带将自己抬离地面。 关于ruby-我需要将Bundler本身添加到Gemfile中吗?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/4758609/
对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
我有一个ModularSinatra应用程序,我正在尝试将Bootstrap添加到应用程序中。get'/bootstrap/application.css'doless:"bootstrap/bootstrap"end我在views/bootstrap中有所有less文件,包括bootstrap.less。我收到这个错误:Less::ParseErrorat/bootstrap/application.css'reset.less'wasn'tfound.Bootstrap.less的第一行是://CSSReset@import"reset.less";我尝试了所有不同的路径格式,但它
我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此
我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r
Arel3.0.2提供了两个类来指定连接类型:Arel::Nodes::InnerJoin和Arel::Nodes::OuterJoin并使用InnerJoin默认。foo=Arel::Table.new('foo')bar=Arel::Table.new('bar')foo.join(bar,Arel::Nodes::InnerJoin)#innerfoo.join(bar,Arel::Nodes::OuterJoin)#outerfoo.join(bar,???)#left如果要生成左连接,如何连接两个表? 最佳答案 你可以使用
刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr
我正在使用Sequel构建一个愿望list系统。我有一个wishlists和itemstable和一个items_wishlists连接表(该名称是续集选择的名称)。items_wishlists表还有一个用于facebookid的额外列(因此我可以存储opengraph操作),这是一个NOTNULL列。我还有Wishlist和Item具有续集many_to_many关联的模型已建立。Wishlist类也有:selectmany_to_many关联的选项设置为select:[:items.*,:items_wishlists__facebook_action_id].有没有一种方法可以
我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R