1.背景

所谓外部数据库文件此处指的就是一个在外部单独创建的db文件,假设有这么一个场景,我们项目中有一些本地数据,不需要接口去获取的(不需要进行网络操作),写死的数据,比如全国各个省各个市的一些基本信息,每个市的信息可以作为表里的一条记录存放,在项目中使用,此时如何我们已经有了包含这些信息的db文件,我们就可以通过greendao来操作这个db文件,更具方便进行开发工作,当然这只是个模拟情况,至于合不合理,有没有更好的方式,此处不过多讨论,重点讲这么一种方式,这种方式可以用于一些不经常变化的数据。

2.项目配置

  • 首先看一下项目结构:

res/raw目录存放的就是外部的db文件的压缩文件,我们可以打开看一下数据库结构,db文件包含两张表,student和teacher里面简单的插入了几条测试数据:

 

  • 引入greendao库文件或者引用库工程:

2.代码实现

实现之前先说一下具体的思路,程序运行,首先把raw目录下的db文件拷贝到数据库存储的默认目录,然后通过greendao的api对这个文件进行操作即可;

    • 我们需要获取应用db存储的路径,通过如下方式:
private void getAppInfo()
{
// 获取packageManager的实例
PackageManager packageManager = getPackageManager();
// getPackageName()是你当前类的包名,0代表是获取版本信息
try
{
packInfo = packageManager.getPackageInfo(getPackageName(), );
}
catch (NameNotFoundException e)
{
e.printStackTrace();
}
}
    • 通过以上方式即可获取到数据库的路径: 

    • 拷贝操作:

public static boolean copyRawDBToApkDb(Context context, int copyRawDbResId, String apkDbPath, String dbName,boolean refresh)
throws IOException
{
boolean b = false; File f = new File(apkDbPath);
if (!f.exists())
{
f.mkdirs();
} File dbFile = new File(apkDbPath + dbName);
b = isDbFileExists(dbFile,refresh);
if (!b)
{
InputStream is = context.getResources().openRawResource(copyRawDbResId); ZipInputStream zis = new ZipInputStream(new BufferedInputStream(is));
ZipEntry entry; while ((entry = zis.getNextEntry()) != null)
{
int size;
byte[] buffer = new byte[ * ]; OutputStream fos = new FileOutputStream(apkDbPath + entry.getName());
BufferedOutputStream bos = new BufferedOutputStream(fos, buffer.length); while ((size = zis.read(buffer, , buffer.length)) != -)
{
bos.write(buffer, , size);
}
bos.flush();
bos.close();
}
zis.close();
is.close();
}
return !b;
}

此处是拷贝操作的关键代码,需要我们传入raw资源ID,数据库的拷贝路径,数据库文件名,是否覆盖已经存在的db文件,@return 拷贝是否成功,关于refresh参数,我们一般希望只拷贝一次,当我们在某些情况下更新了这个db文件的话就可以设置为true进行覆盖操作;

  • 使用greenDao java工程,生成外部db文件所对应表的实体类Dao类等代码:

上图中的DBController类是我封装的数据库操作类,更方便我们去进行操作,DBController的关键代码如下:

/**
* 外部数据库控制类
*/
public class DBController
{
private static DaoMaster daoMasterEcmc; private static DaoMaster daoMasterSchool; // 默认DB
private static DaoSession daoSessionDefault; // 拷贝的db
private static DaoSession daoSchoolSession; /**
* 默认数据库名称:localdata
*/
public static final String DATABASE_NAME = "localdata.db"; /**
* 拷贝数据库名称:school
*/
public static final String DATABASE_SCHOOL_NAME = "school.db"; private static DaoMaster obtainMaster(Context context, String dbName)
{
return new DaoMaster(new DaoMaster.DevOpenHelper(context, dbName, null).getWritableDatabase());
} private static DaoMaster getDaoMaster(Context context, String dbName)
{
if (dbName == null)
return null;
if (daoMasterEcmc == null)
{
daoMasterEcmc = obtainMaster(context, dbName);
}
return daoMasterEcmc;
} private static DaoMaster getSchoolDaoMaster(Context context, String dbName)
{
if (dbName == null)
return null;
if (daoMasterSchool == null)
{
daoMasterSchool = obtainMaster(context, dbName);
}
return daoMasterSchool;
} /**
* 取得DaoSession
*
* @return
*/
public static DaoSession getDaoSession(String dbName)
{ if (daoSchoolSession == null)
{
daoSchoolSession = getSchoolDaoMaster(MainApplication.getIns(), dbName).newSession();
}
return daoSchoolSession;
} /**
* 默认操作localdata数据库
*/
public static DaoSession getDaoSession()
{ if (daoSessionDefault == null)
{
daoSessionDefault = getDaoMaster(MainApplication.getIns(), DATABASE_NAME).newSession();
}
return daoSessionDefault;
}
}
  • 可能我们还需要默认的greendao数据库进行其他的操作,至于默认的操作此处不再详细介绍,不了解的可以看greendao基本使用,此处我们演示的外部DB文件命名为school.db,默认的greenDao数据库命名为history.db,下面我们在MainActivity进行测试操作:
public class MainActivity extends Activity
{
private StringBuilder builder; @Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//创建默认的数据,并插入一条数据
HistoryDao historyDao = DBLocalController.getDaoSession().getHistoryDao();
History entity = new History();
entity.setName("科罗拉多");
entity.setImageUrl("http://www.baidu.com");
historyDao.insert(entity);
//拷贝外部DB文件到指定目录
copyRawDB();
//通过greendao查询外部db文件数据
selDBData();
} private void selDBData()
{
StudentDao student = DBController.getDaoSession(DBController.DATABASE_SCHOOL_NAME).getStudentDao();
List<Student> students = student.queryBuilder().list();
builder = new StringBuilder();
for (int i = ; i < students.size(); i++)
{
builder.append(students.get(i).getName() + "---");
}
Toast.makeText(MainActivity.this, builder.toString(), Toast.LENGTH_SHORT).show();
} private void copyRawDB()
{
try
{
// 拷贝res/raw/xxxxdb.zip 到
// /data/data/com.xinhang.mobileclient/databases/ 目录下面
boolean isSuccess = DBUtils.copyRawDBToApkDb(MainActivity.this, R.raw.schooldb, DBUtils.APK_DB_PATH, DBUtils.ECMC_DB_NAME, false);
}
catch (IOException e)
{
e.printStackTrace();
}
} }

运行结果:

可以看到我们的外部db文件已经拷贝到数据库默认路径下,还有我们的默认数据库也创建成功;区分两个数据库的方式是通过DBLocalController.getDaoSession(name)方法,想操作哪个数据库传入对应的数据库名称即可,gif操作图如下;


源码下载:源码下载,有问题欢迎交流讨论!

Android GreenDao操作外部DB数据库文件的更多相关文章

  1. Android虚拟机中的sqlite数据库文件

    Android虚拟机中的sqlite数据库文件 ①

  2. 查看Android下生成的.db数据库

    1.在cmd中找到sdk中的platform-tools文件夹. 2.输入adb shell命令. 3.再输入sqlite3 /data/data/com.svs.db/databases/svs.d ...

  3. android greendao的外部封装不太友好。

    https://github.com/greenrobot/greenDAO 下载下官网的示例,有完整的封装版本,但自已封装是碰到很多问题. 因greenDao的Master和Session中很多方法 ...

  4. Android 使用greenDAO 3.2.2 操作外部数据库

    项目开发中有时需要用到一些写死的数据,如公司的产品信息之类的.这就需要我们先把数据库文件保存在资源文件夹下,然后当应用创建时将数据库文件拷到应用安装目录的/databases/文件夹下,然后再对数据进 ...

  5. 通过adb shell操作android真机的SQLite数据库

    要通过命令行直接操作android真机上的SQLite数据库,可以直接通过adb shell来完成,不过,前提是必须获得root权限. 另外,android系统其实就是linux的shell,这个应该 ...

  6. Android下创建一个SQLite数据库

    数据库:SQLite(轻量级,嵌入式的数据库) 大量的相似结构的数据的储存,快速的查询.特殊的文件(按照一定的格式生成) 数据库的创建 创建文件 1.声明文件对象,文件是不会被创建出来的. File ...

  7. 【转】Phonegap离线调用SQLite数据库文件

    按:不可多得的好文章,转过来以免源丢失 文章来源:http://liuwei.co/index.php/default/The-quickest-way-to-execute-many-sql-for ...

  8. Android -- com.android.providers.media,external.db

    external.db android是管理多媒体文件(音频.视频.图片)的信息是在/data/data/com.android.providers.media下的数据库文件external.db. ...

  9. Android 查看和管理sqlite数据库

    在Android中可以使用Eclipse插件DDMS来查看,也可以使用Android工具包中的adb工具来查看.android项目中的sqlite数据库位于/data/data/项目包/databas ...

随机推荐

  1. web.xml中的<jsp-config>的用法详解

    <jsp-config> 包括<taglib> 和<jsp-property-group> 两个子元素. 其中<taglib>元素在JSP 1.2时就已 ...

  2. JavaScript高级程序设计学习笔记第五章--引用类型

    一.object类型 1.创建object类型的两种方式: 第一种,使用构造函数 var person = new Object();或者是var person={};/与new Object()等价 ...

  3. nginx中有关命令和日志切割,配置文件加载的详细阐述

    一.Nginx简介 Nginx (“engine x”) 是俄罗斯人Igor Sysoev(塞索耶夫)编写的一款高性能的 HTTP 和反向代理服务器.Nginx 已经在俄罗斯最大的门户网站── Ram ...

  4. day1 java基础回顾- 文件路径

    绝对路径 以根目录或某盘符开头的路径(或者说完整的路径) 例如: l  c:/a.txt (Windows操作系统中) l  c:/xxx/a.txt (Windows操作系统中) l  /var/x ...

  5. 【机器学习】迭代决策树GBRT(渐进梯度回归树)

    一.决策树模型组合 单决策树C4.5由于功能太简单,并且非常容易出现过拟合的现象,于是引申出了许多变种决策树,就是将单决策树进行模型组合,形成多决策树,比较典型的就是迭代决策树GBRT和随机森林RF. ...

  6. GridView ,后台修改 跟新完毕,前端 未跟新处理

    //Response.Redirect(Request.Url.ToString());//重新定位 GridView_dept.DataBind(); //重新绑定都可以

  7. Kth Largest Element in a Stream

    Design a class to find the kth largest element in a stream. Note that it is the kth largest element ...

  8. UOJ #32. 【UR #2】跳蚤公路【Floydbellman-ford】

    首先看这个范围很夸张但是其实有限制的也就在1e18*n范围里(走完一圈的边权),然后限制一定是有负环 用Floyd传递闭包,然后设f[i][j][k]为从1走了i步到j并且有k个x的最短路,用B-F处 ...

  9. Bzoj3315 [Usaco2013 Nov]Pogo-Cow(luogu3089)

    3315: [Usaco2013 Nov]Pogo-Cow Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 352  Solved: 181[Submit ...

  10. 一文搞定 Redis 复制(全会的举个手看看)

    阅读本文大概需要 5 分钟. 本文大纲 复制过程 数据间的同步 全量复制 部分复制 心跳 异步复制 总结 一.复制过程 Step 1:从节点执行 slaveof 命令. Step 2:从节点只是保存了 ...