Android开发中,如何将SQLite和APK一起打包发布

package com.test.db;   
   
import java.io.File;   
import java.io.FileOutputStream;   
import java.io.InputStream;   
import java.io.OutputStream;   
import java.io.UnsupportedEncodingException;   
   
import android.app.Activity;   
import android.database.Cursor;   
import android.database.sqlite.SQLiteDatabase;   
import android.os.Bundle;   
   
public class DbtestActivity extends Activity {   
    /** Called when the activity is first created. */   
    @Override   
    public void onCreate(Bundle savedInstanceState) {   
        super.onCreate(savedInstanceState);   
        setContentView(R.layout.main);   
   
        // com.test.db 是程序的包名,请根据自己的程序调整    
        // /data/data/com.test.db/    
        // databases 目录是准备放 SQLite 数据库的地方,也是 Android 程序默认的数据库存储目录    
        // 数据库名为 test.db    
        String DB_PATH = "/data/data/com.test.db/databases/";   
        String DB_NAME = "test.db";   
   
        // 检查 SQLite 数据库文件是否存在    
        if ((new File(DB_PATH + DB_NAME)).exists() == false) {   
            // 如 SQLite 数据库文件不存在,再检查一下 database 目录是否存在    
            File f = new File(DB_PATH);   
            // 如 database 目录不存在,新建该目录    
            if (!f.exists()) {   
                f.mkdir();   
            }   
   
            try {   
                // 得到 assets 目录下我们实现准备好的 SQLite 数据库作为输入流    
                InputStream is = getBaseContext().getAssets().open(DB_NAME);   
                // 输出流    
                OutputStream os = new FileOutputStream(DB_PATH + DB_NAME);   
   
                // 文件写入    
                byte[] buffer = new byte[1024];   
                int length;   
                while ((length = is.read(buffer)) > 0) {   
                    os.write(buffer, 0, length);   
                }   
   
                // 关闭文件流    
                os.flush();   
                os.close();   
                is.close();   
            } catch (Exception e) {   
                e.printStackTrace();   
            }   
        }   
   
        // 下面测试 /data/data/com.test.db/databases/ 下的数据库是否能正常工作    
        SQLiteDatabase database = SQLiteDatabase.openOrCreateDatabase(DB_PATH + DB_NAME, null);   
        Cursor cursor = database.rawQuery("select * from test", null);   
   
        if (cursor.getCount() > 0) {   
            cursor.moveToFirst();   
            try {   
                // 解决中文乱码问题    
                byte test[] = cursor.getBlob(0);   
                String strtest = new String(test, "utf-8").trim();   
   
                // 看输出的信息是否正确    
                System.out.println(strtest);   
            } catch (UnsupportedEncodingException e) {   
                // TODO Auto-generated catch block    
                e.printStackTrace();   
            }   
        }   
        cursor.close();   
    }   
}   

package com.test.db; 

import java.io.File; 
import java.io.FileOutputStream; 
import java.io.InputStream; 
import java.io.OutputStream; 
import java.io.UnsupportedEncodingException; 

import android.app.Activity; 
import android.database.Cursor; 
import android.database.sqlite.SQLiteDatabase; 
import android.os.Bundle; 

public class DbtestActivity extends Activity { 
    /** Called when the activity is first created. */ 
    @Override 
    public void onCreate(Bundle savedInstanceState) { 
        super.onCreate(savedInstanceState); 
        setContentView(R.layout.main); 

        // com.test.db 是程序的包名,请根据自己的程序调整 
        // /data/data/com.test.db/ 
        // databases 目录是准备放 SQLite 数据库的地方,也是 Android 程序默认的数据库存储目录 
        // 数据库名为 test.db 
        String DB_PATH = "/data/data/com.test.db/databases/"; 
        String DB_NAME = "test.db"; 

        // 检查 SQLite 数据库文件是否存在 
        if ((new File(DB_PATH + DB_NAME)).exists() == false) { 
            // 如 SQLite 数据库文件不存在,再检查一下 database 目录是否存在 
            File f = new File(DB_PATH); 
            // 如 database 目录不存在,新建该目录 
            if (!f.exists()) { 
                f.mkdir(); 
            } 

            try { 
                // 得到 assets 目录下我们实现准备好的 SQLite 数据库作为输入流 
                InputStream is = getBaseContext().getAssets().open(DB_NAME); 
                // 输出流 
                OutputStream os = new FileOutputStream(DB_PATH + DB_NAME); 

                // 文件写入 
                byte[] buffer = new byte[1024]; 
                int length; 
                while ((length = is.read(buffer)) > 0) { 
                    os.write(buffer, 0, length); 
                } 

                // 关闭文件流 
                os.flush(); 
                os.close(); 
                is.close(); 
            } catch (Exception e) { 
                e.printStackTrace(); 
            } 
        } 

        // 下面测试 /data/data/com.test.db/databases/ 下的数据库是否能正常工作 
        SQLiteDatabase database = SQLiteDatabase.openOrCreateDatabase(DB_PATH + DB_NAME, null); 
        Cursor cursor = database.rawQuery("select * from test", null); 

        if (cursor.getCount() > 0) { 
            cursor.moveToFirst(); 
            try { 
                // 解决中文乱码问题 
                byte test[] = cursor.getBlob(0); 
                String strtest = new String(test, "utf-8").trim(); 

                // 看输出的信息是否正确 
                System.out.println(strtest); 
            } catch (UnsupportedEncodingException e) { 
                // TODO Auto-generated catch block 
                e.printStackTrace(); 
            } 
        } 
        cursor.close(); 
    } 


最近一时心血来潮,尝试开发 Android 程序,练习的项目是个简单的天气预报程序。其中天气预报自然要涉及全国各地各个地区,我把这些地区名以及地区代码放在一个 SQLite 数据库里,当然 Android 对 SQLite 也支持的很好。问题是,学习过程中发现很多教程或者示例,都是程序运行后即时创建一个数据库,然后建表、插入数据什么的,而我自己的想法是在电脑上把数据库事先准备好,然后随 APK 一起打包发布,否则在 Java 代码里面 hard code 几千条 insert 语句,显然是很傻瓜的行为。经过多方网络搜寻,终于找到比较完善的解决方法。 

在 Eclipse 里新建好工程后,默认会有一个 assets 目录,在 Eclipse 中直接将准备好的 SQLite 数据库复制到该目录中,然后在主 Activity 里面编码: 

程序启动时候回去检查数据库文件在不在,如果不存在,就会把我们准备好的数据库复制到哪个 databases 目录下,而且如果用户卸载了这个程序,那么这个目录和数据库也将随之卸载。 

参考文章: 
Using your own SQLite database in Android applications 

Embed a database in the .apk of a distributed application [Android] 

解决这个问题后,花了几天找了一下天气预报的实现方式,其实最主要的是寻找数据源,靠谱的数据源很少很少。 

最终决定用 weather.com.cn 这个比较权威的数据源,其提供的数据格式是 JSON,并参考了 通过 weather.com.cn 获取全国天气数据 这篇文章,最终简单实现了天气预报功能 

该程序可以提供中国大陆及香港、澳门、台湾各城市实时天气,所有数据来源于中国国家气象局,准确率相当高
posted @ 2012-09-05 16:22  子福当自强  阅读(750)  评论(0编辑  收藏  举报
悟道