android sqlitejian监听,tencent/sqlite.md · zhoujian/AndroidInterView - Gitee.com

#### 腾讯-数据库版本如何单独升级,并且将原有数据迁移过去

> 在我们开发的应用中,一般都会涉及到数据库,使用数据的时候会涉及到数据库的升级、数据的迁移、增加行的字段等。比如,用户定制数据的保存,文件的端点续传信息的保存等都会涉及到数据库。

>

​我们应用第一个版本是V1.0,在迭代版本V1.1 时,我们在数据库中增加了一个字段。因此V1.0的数据库在V1.1版本需要升级,V1.0版本升级到V1.1时原来数据库中的数据不能丢失,

​那么在V1.1中就要有地方能够检测出来版本的差异,并且把V1.0软件的数据库升级到V1.1软件能够使用的数据库。也就是说,要在V1.0软件的数据库的那个表中增加那个字段,并赋予这个字段默认值。

应用中怎么检测数据库需要升级呢? SQLiteOpenHelper 类中有一个方法

```

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {}

```

​当我们创建对象的时候如果传入的版本号大于之前的版本号,该方法就会被调用,通过判断oldVersion 和 newVersion 就可以决定如何升级数据库。在这个函数中把老版本数据库的相应表中增加字段,并给每条记录增加默认值即可。新版本号和老版本号都会作为onUpgrade函数的参数传进来,便于开发者知道数据库应该从哪个版本升级到哪个版本。升级完成后,数据库会自动存储最新的版本号为当前数据库版本号。

##### 数据库升级

SQLite提供了ALTER TABLE命令,允许用户重命名或添加新的字段到已有表中,但是不能从表中删除字段。并且只能在表的末尾添加字段,比如,为Orders 表中添加一个字段:”ALTER TABLE Order ADDCOLUMN Country”

代码如下:

public class OrderDBHelper extends SQLiteOpenHelper {

private static final int DB_VERSION = 1;

private static final String DB_NAME = "Test.db";

public static final String TABLE_NAME = "Orders";

public OrderDBHelper(Context context, int version) {

super(context, DB_NAME, null, version);

}

@Override

public void onCreate(SQLiteDatabase db) {

String sql = "create table if not exists " + TABLE_NAME + " (Id integer primary key, " +

"CustomName text, OrderPrice integer)";

db.execSQL(sql);

}

@Override

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

Log.e("owen", "DB onUpgrade");

if (newVersion == 2) {

db.execSQL("ALTER TABLE " + TABLE_NAME + " ADD COLUMN Country");

Cursor cr = db.rawQuery("select * from " + TABLE_NAME, null);

while (cr.moveToNext()) {

String name = cr.getString(cr.getColumnIndex("CustomName"));

ContentValues values = new ContentValues();

values.put("CustomName", name);

values.put("Country", "China");

db.update(TABLE_NAME, values, "CustomName=?", new String[] {name});

}

cr.close();

}

}

```

OrderDBHelper orderDBHelper = new OrderDBHelper(this, 2);

SQLiteDatabase db = orderDBHelper.getWritableDatabase();

ContentValues contentValues = new ContentValues();

contentValues.put("OrderPrice", 100);

contentValues.put("CustomName", "OwenChan");

db.insert(OrderDBHelper.TABLE_NAME, null, contentValues);

Log.e("owen", "create finish");

Cursor cr = db.rawQuery("select * from " + OrderDBHelper.TABLE_NAME , null);

while (cr.moveToNext()) {

String name = cr.getString(cr.getColumnIndex("CustomName"));

Log.e("owen", "name:" + name);

String country = cr.getString(cr.getColumnIndex("Country"));

Log.e("owen", "country:" + country);

}

cr.close();

db.close();

```

##### 数据库的迁移

可以分一下几个步骤迁移数据库

1、 将表名改成临时表

ALTER TABLE Order RENAME TO _Order;

2、创建新表

> CREATETABLE Test(Id VARCHAR(32) PRIMARY KEY ,CustomName VARCHAR(32) NOTNULL , Country VARCHAR(16) NOTNULL);

>

3、导入数据

> INSERTINTO Order SELECT id, “”, Age FROM _Order;

>

4、删除临时表

> DROPTABLE _Order;

>

通过以上四个步骤,就可以完成旧数据库结构向新数据库结构的迁移,并且其中还可以保证数据不会因为升级而流失。

当然,如果遇到减少字段的情况,也可以通过创建临时表的方式来实现。

实现代码如下

```

@Override

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

if (newVersion == 2) {

char str = '"';

db.beginTransaction();

db.execSQL("ALTER TABLE Order RENAME TO _Order");

db.execSQL("CREATE TABLE Order(Id integer primary key autoincrement , CustomName VARCHAR(20) NOT NULL,"

+ " Country VARCHAR(32) NOT NULL , OrderPrice VARCHAR(16) NOT NULL)");

db.execSQL("INSERT INTO Order SELECT Id, " + str + str

+ ", CustomName, OrderPrice FROM _Order");

db.setTransactionSuccessful();

db.endTransaction();

}

}

```

##### 多个数据库版本的升级

假如我们开发的程序已经发布了两个版本:V1.0,V2.0,我们正在开发V3.0。版本号分别是1,2,3。对于这种情况,我们应该如何实现升级?

用户的选择有:

> 1) V1.0 -> V3.0 DB 1 -> 2

> 2) V2.0 -> V3.0 DB 2 -> 3

数据库的每一个版本所代表的数据库必须是定义好的,比如说V1.0的数据库,它可能只有两张表TableA和TableB,如果V2.0要添加一张表TableC,如果V3.0要修改TableC,数据库结构如下:

> V1.0 —> TableA, TableB

> V1.2 —> TableA, TableB, TableC

> V1.3 —> TableA, TableB, TableC (Modify)

代码如下:

```

@Override

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

if (1 == oldVersion) {

String sql = "Create table C....";

db.execSQL(sql);

oldVersion = 2;

}

if (2 == oldVersion) {

//modify C

oldVersion = 3;

}

}

```

##### 导入已有数据库

```

/**

* Created by Owen Chan

* On 2017-09-26.

*/

public class DbManager {

public static final String PACKAGE_NAME = "com.example.sql";

public static final String DB_NAME = "table.db";

public static final String DB_PATH = "/data/data/" + PACKAGE_NAME;

private Context mContext;

public DbManager(Context mContext) {

this.mContext = mContext;

}

public SQLiteDatabase openDataBase() {

return SQLiteDatabase.openOrCreateDatabase(DB_PATH + "/" + DB_NAME, null);

}

public void importDB() {

File file = new File(DB_PATH + "/" + DB_NAME);

if (!file.exists()) {

try {

FileOutputStream out = new FileOutputStream(file);

int buffer = 1024;

InputStream in = mContext.getResources().openRawResource(R.raw.xxxx);

byte[] bts = new byte[buffer];

int lenght;

while ((lenght = in.read(bts)) > 0) {

out.write(bts, 0, bts.length);

}

out.close();

in.close();

} catch (Exception e) {

e.printStackTrace();

}

}

}

}

```

##### 调用方式

```

@Override

protected void onResume() {

super.onResume();

DbManager dbManager = new DbManager(this);

dbManager.importDB();

SQLiteDatabase db = dbManager.openDataBase();

db.execSQL("do what you want");

}

```

一键复制

编辑

Web IDE

原始数据

按行查看

历史

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ls /usr/local/lib/libsqlite3.* /usr/local/lib/libsqlite3.a /usr/local/lib/libsqlite3.so.0 /usr/local/lib/libsqlite3.la /usr/local/lib/libsqlite3.so.0.8.6 /usr/local/lib/libsqlite3.so ls /usr/local/include/sqlite3.h /usr/local/include/sqlite3.h arm-linux-gcc -L./lib-lsqlite3 -o sqlite sqlite.c /tmp/ccKtSnwq.o: In function `main': sqlite.c:(.text+0xf4): undefined reference to `sqlite3_open' sqlite.c:(.text+0x120): undefined reference to `sqlite3_errmsg' sqlite.c:(.text+0x140): undefined reference to `sqlite3_close' sqlite.c:(.text+0x170): undefined reference to `sqlite3_exec' sqlite.c:(.text+0x1b0): undefined reference to `sqlite3_free' sqlite.c:(.text+0x1d8): undefined reference to `sqlite3_exec' sqlite.c:(.text+0x218): undefined reference to `sqlite3_free' sqlite.c:(.text+0x240): undefined reference to `sqlite3_exec' sqlite.c:(.text+0x280): undefined reference to `sqlite3_free' sqlite.c:(.text+0x28c): undefined reference to `sqlite3_close' collect2: ld returned 1 exit status [root@localhost install]# arm-linux-gcc -o sqlite sqlite.c -L./usr/local/lib -lsqlite3 /opt/FriendlyARM/toolschain/4.5.1/lib/gcc/arm-none-linux-gnueabi/4.5.1/../../../../arm-none-linux-gnueabi/bin/ld: cannot find -lsqlite3 collect2: ld returned 1 exit status [root@localhost install]# arm-linux-gcc -o sqlite sqlite.c -L./usr/local/lib -lsqlite3 -static /opt/FriendlyARM/toolschain/4.5.1/lib/gcc/arm-none-linux-gnueabi/4.5.1/../../../../arm-none-linux-gnueabi/bin/ld: cannot find -lsqlite3 collect2: ld returned 1 exit status [root@localhost install]# chmod +x /usr/local/libsqlite3.so chmod: cannot access `/usr/local/libsqlite3.so': No such file or directory [root@localhost install]# chmod +x /usr/local/libsqlite3.so.0.8.6 chmod: cannot access `/usr/local/libsqlite3.so.0.8.6': No such file or directory [root@localhost install]# arm-linux-gcc -o sqlite sqlite.c -L./usr/local/lib -lsqlite3.so -static /opt/FriendlyARM/toolschain/4.5.1/lib/gcc/arm-none-linux-gnueabi/4.5.1/../../../../arm-none-linux-gnueabi/bin/ld: cannot find -lsqlite3.so collect2: ld returned 1 exit status [root@localhost install]# arm-linux-gcc -o sqlite sqlite.c -L./usr/local/lib -lsqlite3.so /opt/FriendlyARM/toolschain/4.5.1/lib/gcc/arm-none-linux-gnueabi/4.5.1/../../../../arm-none-linux-gnueabi/bin/ld: cannot find -lsqlite3.so collect2: ld returned 1 exit status
06-06

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值