文档章节

android中使用sqlite、复制assets下的数据库到SD卡、支持大于1M的文件

simpower
 simpower
发布于 2016/04/20 19:38
字数 994
阅读 109
收藏 3

初学android,达人忽略,欢迎扔石头.

android中使用sqlite、复制assets下的数据库到SD卡、支持大于1M的文件

如果使用SD卡,需要在AndroidManifest.xml中设置权限

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"></uses-permission>
package cn.arthur.common;
 import java.io.File;
 import java.io.FileOutputStream;
 import java.io.IOException;
 import java.io.InputStream;
 import java.io.OutputStream;
 import android.content.Context;
 import android.database.sqlite.SQLiteDatabase;
 import android.database.sqlite.SQLiteDatabase.CursorFactory;
import android.database.sqlite.SQLiteException;
import android.database.sqlite.SQLiteOpenHelper;
/**
 * @author Joshua
 * 用法:
 * DBHelper dbHelper = new DBHelper(this);
 * dbHelper.createDataBase();
 * SQLiteDatabase db = dbHelper.getWritableDatabase();
 * Cursor cursor = db.query()
 * db.execSQL(sqlString);
 * 注意:execSQL不支持带;的多条SQL语句,只能一条一条的执行,晕了很久才明白
 * 见execSQL的源码注释 (Multiple statements separated by ;s are not supported.)
 * 将把assets下的数据库文件直接复制到DB_PATH,但数据库文件大小限制在1M以下
 * 如果有超过1M的大文件,则需要先分割为N个小文件,然后使用copyBigDatabase()替换copyDatabase()
 */
public class DBHelper extends SQLiteOpenHelper {
    //用户数据库文件的版本
    private static final int DB_VERSION    = 1;
    //数据库文件目标存放路径为系统默认位置,cn.arthur.examples 是你的包名
    private static String DB_PATH        = "/data/data/cn.arthur.examples/databases/";
/*
    //如果你想把数据库文件存放在SD卡的话
    private static String DB_PATH        = android.os.Environment.getExternalStorageDirectory().getAbsolutePath()
                                        + "/arthurcn/drivertest/packfiles/";
*/
    private static String DB_NAME         = "hello.db";
    private static String ASSETS_NAME     = "hello.db";
    private SQLiteDatabase myDataBase    = null;
    private final Context myContext;
     /** 
      * 如果数据库文件较大,使用FileSplit分割为小于1M的小文件
      * 此例中分割为 hello.db.101    hello.db.102    hello.db.103
      */
    //第一个文件名后缀
    private static final int ASSETS_SUFFIX_BEGIN    = 101;
    //最后一个文件名后缀
    private static final int ASSETS_SUFFIX_END        = 103;
    
    /**
     * 在SQLiteOpenHelper的子类当中,必须有该构造函数
     * @param context    上下文对象
     * @param name        数据库名称
     * @param factory    一般都是null
     * @param version    当前数据库的版本,值必须是整数并且是递增的状态
     */
    public DBHelper(Context context, String name, CursorFactory factory, int version) {
        //必须通过super调用父类当中的构造函数
        super(context, name, null, version);
        this.myContext = context;
    }
    
    public DBHelper(Context context, String name, int version){
        this(context,name,null,version);
    }
    public DBHelper(Context context, String name){
        this(context,name,DB_VERSION);
    }
    
    public DBHelper (Context context) {
        this(context, DB_PATH + DB_NAME);
    }
    
    public void createDataBase() throws IOException{
        boolean dbExist = checkDataBase();
        if(dbExist){
            //数据库已存在,do nothing.
        }else{
            //创建数据库
            try {
                File dir = new File(DB_PATH);
                if(!dir.exists()){
                    dir.mkdirs();
                }
                File dbf = new File(DB_PATH + DB_NAME);
                if(dbf.exists()){
                    dbf.delete();
                }
                SQLiteDatabase.openOrCreateDatabase(dbf, null);
                // 复制asseets中的db文件到DB_PATH下
                copyDataBase();
            } catch (IOException e) {
                throw new Error("数据库创建失败");
            }
        }
    }
    
    //检查数据库是否有效
    private boolean checkDataBase(){
        SQLiteDatabase checkDB = null;
        String myPath = DB_PATH + DB_NAME;
        try{            
            checkDB = SQLiteDatabase.openDatabase(myPath, null, SQLiteDatabase.OPEN_READONLY);
        }catch(SQLiteException e){
            //database does't exist yet.
        }
        if(checkDB != null){
            checkDB.close();
        }
         return checkDB != null ? true : false;
    }
 
    /**
     * Copies your database from your local assets-folder to the just created empty database in the
     * system folder, from where it can be accessed and handled.
     * This is done by transfering bytestream.
     * */
    private void copyDataBase() throws IOException{
        //Open your local db as the input stream
        InputStream myInput = myContext.getAssets().open(ASSETS_NAME);
        // Path to the just created empty db
        String outFileName = DB_PATH + DB_NAME;
        //Open the empty db as the output stream
        OutputStream myOutput = new FileOutputStream(outFileName);
        //transfer bytes from the inputfile to the outputfile
        byte[] buffer = new byte[1024];
        int length;
        while ((length = myInput.read(buffer))>0){
            myOutput.write(buffer, 0, length);
        }
        //Close the streams
        myOutput.flush();
        myOutput.close();
        myInput.close();
    }
    
    //复制assets下的大数据库文件时用这个
    private void copyBigDataBase() throws IOException{
        InputStream myInput;
        String outFileName = DB_PATH + DB_NAME;
        OutputStream myOutput = new FileOutputStream(outFileName);
        for (int i = ASSETS_SUFFIX_BEGIN; i < ASSETS_SUFFIX_END+1; i++) {
            myInput = myContext.getAssets().open(ASSETS_NAME + "." + i);
            byte[] buffer = new byte[1024];
            int length;
            while ((length = myInput.read(buffer))>0){
                myOutput.write(buffer, 0, length);
            }
            myOutput.flush();
            myInput.close();
        }
        myOutput.close();
    }
    
    @Override
    public synchronized void close() {
        if(myDataBase != null){
            myDataBase.close();
        }
        super.close();
    }
    
    /**
     * 该函数是在第一次创建的时候执行,
     * 实际上是第一次得到SQLiteDatabase对象的时候才会调用这个方法
     */
    @Override
    public void onCreate(SQLiteDatabase db) {
    }
    
    /**
     * 数据库表结构有变化时采用
     */
    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
    }
}

how copy database from assets to ExternalStorage?

public class MainActivity extends Activity {

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    File f1=new File(Environment.getExternalStorageDirectory(),"aaa");
    f1.mkdir();
    InputStream in = null;
    try {
        Toast.makeText(getBaseContext(), "error", Toast.LENGTH_LONG).show();
        in = getBaseContext().getAssets().open("home");
    } catch (IOException e) {
        // TODO Auto-generated catch block
        Toast.makeText(getBaseContext(), "error", Toast.LENGTH_LONG).show();
        e.printStackTrace();
    }
    OutputStream ou = null;
    try {
        ou = new FileOutputStream(Environment.getExternalStorageDirectory()+"aaa"+ "/"+"home" );
    } catch (FileNotFoundException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    try {
        CopyDB(in, ou);
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}
public void CopyDB(InputStream in, OutputStream ou) throws IOException
{
    byte []buffer=new byte[1024];
    int length;
    while((length=in.read(buffer))>0)
    {
        try {
            ou.write(buffer, 0, length);
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        }
    in.close();
    ou.close();

}

https://stackoverflow.com/questions/26870624/how-copy-database-from-assets-to-externalstorage

本文转载自:http://www.cnblogs.com/som5/archive/2011/01/23/android_sqlite_assets_sdcard.html

共有 人打赏支持
simpower
粉丝 26
博文 601
码字总数 45471
作品 0
海淀
程序员
私信 提问
浅入浅出Android(010):如何将已有的sqlite数据库放入程序中

建数据库 首先,建立sqlite数据库appdata.db,在其中建表填数据: CREATE TABLE user (name TEXT); INSERT INTO user VALUES ('letian');INSERT INTO user VALUES ('xiaoming'); 建立Android......

樂天
2014/05/27
0
0
【Android游戏开发十三】(保存游戏数据 [下文])详解SQLite存储方式,并把SQLite的数据库文件存储在SD卡中!!!

李华明Himi 原创,转载务必在明显处注明: 转载自【黑米GameDev街区】 原文链接: http://www.himigame.com/android-game/329.html 很多童鞋说我的代码运行后,点击home或者back后会程序异常,...

迷途d书童
2012/03/19
0
0
android是用外部sqlite文件问题求解.

有一个10m多大小的sqlite文件,要在android应用中使用它。 尝试过的方法如下: 1、把sqlite文件放入raw目录下,通过文件读写操作,将sqlite文件存入到sd卡中。这时候会出现错误:android.dat...

苏谷子
2013/06/21
1K
2
安卓应用安全指南 4.5.1 使用 SQLite 示例代码

安卓应用安全指南 4.5.1 使用 SQLite 示例代码 原书:Android Application Secure Design/Secure Coding Guidebook 译者:飞龙 协议:CC BY-NC-SA 4.0 4.5.1.1 创建/操作数据库 在 Android ...

apachecn_飞龙
2018/03/22
0
0
安卓应用安全指南 4.5.3 使用 SQLite 高级话题

安卓应用安全指南 4.5.3 使用 SQLite 高级话题 原书:Android Application Secure Design/Secure Coding Guidebook 译者:飞龙 协议:CC BY-NC-SA 4.0 4.5.3.1 在 SQL 语句的断言中使用通配符...

apachecn_飞龙
2018/03/22
0
0

没有更多内容

加载失败,请刷新页面

加载更多

指针数组和数组指针的区别

这两个名字不同当然所代表的意思也就不同。我刚开始看到这就吓到了,主要是中文太博大精深了,整这样的简称太专业了,把人都绕晕了。从英文解释或中文全称看就比较容易理解。 指针数组:arr...

天王盖地虎626
21分钟前
0
0
Qt那些事0.0.18

今天要记一下Qt中的Resource。自我感觉理解的不错,但是还会难免有谬误,在日后有可能会更新,也有可能不会。 小声的念叨一句,女人心,海底针。 今天就直接跳过了关于QML在qrc文件中的介绍,...

Ev4n
29分钟前
0
0
深入解析js的作用域、预解析机制

虽然,ES6在我们工作中应用得越来越广泛,但是还是很多项目保留着ES5的写法,所以,今天,带着大家重新巩固下ES5下的作用域及预解析机制。 概念: 作用域:域,指的是一个空间、范围、区域,...

前端攻城老湿
32分钟前
0
0
Spring Cloud Feign - 声明式 REST Client

1、Feign是什么 声明式REST client,来自NetFlix。 允许你编写无实现代码调用REST services 替换RestTemplate(甚至更简单) Spring Cloud 为使用Feign提供了包装器 2、怎样使用Feign 对比:...

Benz001
37分钟前
2
0
前端、后端和全栈到底不该学什么

1、前言 在职业规划咨询过程中经常会被问到这样的问题: 老师,我是该深入钻研专精一门,走技术大牛路线,还是所有都要精通,做一个全栈工程师? 类似问题的变种还有,老师我是不是该30岁最迟...

前端攻城小牛
39分钟前
3
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部