菜单

Android SQLite详解

2018年11月18日 - sqlite

又不得不在表的末梢添加字段,比如,为 Subscription添加鲜独字段:

于运SQLite时,我建议先下充斥一个当地SQLite客户端来说明操作,在当地写的SQL语句运行是后,再更换到Android中。我之所以之是SQLite
Expert
Personal。
先是创建一个持续在SQLiteOpenHelper的切近,并重写onCreate()onUpgrade()方法。

经过上述四个步骤,就得好旧数据库结构于新数据库结构的动迁,并且其中还可保证数据不会见应为升级而化为乌有。

当列开发被,我们一点都见面为此到数据库。在Android中,我们一般采用SQLite,因为Android在android.database.sqlite保证封装了好多SQLite操作的API。我要好写了一个Demo来总结SQLite的用,托管在Github上,大家可点击下载APK,也可以点击下载源码。Demo截图如下:

复制代码 代码如下:

public OrderDao(Context context) {
    this.context = context;
    ordersDBHelper = new OrderDBHelper(context);
}

复制代码 代码如下:

public class OrderDBHelper extends SQLiteOpenHelper{
    private static final int DB_VERSION = 1;
    private static final String DB_NAME = "myTest.db";
    public static final String TABLE_NAME = "Orders";

    public OrderDBHelper(Context context) {
        super(context, DB_NAME, null, DB_VERSION);
    }

    @Override
    public void onCreate(SQLiteDatabase sqLiteDatabase) {
        // create table Orders(Id integer primary key, CustomName text, OrderPrice integer, Country text);
        String sql = "create table if not exists " + TABLE_NAME + " (Id integer primary key, CustomName text, OrderPrice integer, Country text)";
        sqLiteDatabase.execSQL(sql);
    }

    @Override
    public void onUpgrade(SQLiteDatabase sqLiteDatabase, int oldVersion, int newVersion) {
        String sql = "DROP TABLE IF EXISTS " + TABLE_NAME;
        sqLiteDatabase.execSQL(sql);
        onCreate(sqLiteDatabase);
    }
}

ALTER TABLE Subscription RENAME TO __temp__Subscription;

删除数据

剔除数据的措施除了execSQL还有delete(String table,String whereClause,String[] whereArgs),whereClause是抹条件,whereArgs是去条件值数组。

db = ordersDBHelper.getWritableDatabase();
db.beginTransaction();

// delete from Orders where Id = 7
db.delete(OrderDBHelper.TABLE_NAME, "Id = ?", new String[]{String.valueOf(7)});
db.setTransactionSuccessful();

更拘留去的源码,里面会拼装删除条件与去条件值数组:

public int delete(String table, String whereClause, String[] whereArgs) {
    acquireReference();
    try {
        SQLiteStatement statement =  new SQLiteStatement(this, "DELETE FROM " + table +
                (!TextUtils.isEmpty(whereClause) ? " WHERE " + whereClause : ""), whereArgs);
        try {
            return statement.executeUpdateDelete();
        } finally {
            statement.close();
        }
    } finally {
        releaseReference();
    }
}

做Android应用,不可避免的会暨SQLite打交道。随着以的络绎不绝升迁,原有的数据库结构或已经不再适应新的效能,这时候,就需针对SQLite数据库的布局进行提升了。

修改数据

改数据与插数据大相似,调用的法子除了execSQL还得是update(String table,ContentValues values,String whereClause, String[] whereArgs)

db = ordersDBHelper.getWritableDatabase();
db.beginTransaction();

// update Orders set OrderPrice = 800 where Id = 6
ContentValues cv = new ContentValues();
cv.put("OrderPrice", 800);
db.update(OrderDBHelper.TABLE_NAME,
        cv,
        "Id = ?",
        new String[]{String.valueOf(6)});
db.setTransactionSuccessful();

另外,如果赶上复杂的改操作,比如在修改的又,需要开展多少的更换,那么好采用以一个事务中执如下语句来实现修改表的急需。

日增数量

以我之Demo中,有三三两两栽增加数据操作:
初始化数据
当登Demo程序时,先判断表中是否生多少,如果表中没多少,我将先行上加有数量。在初始化数据常常,因为一次性要添加的多寡比多,所以我一直行使的是execSQL方法:

db = ordersDBHelper.getWritableDatabase();
db.beginTransaction();

db.execSQL("insert into " + OrderDBHelper.TABLE_NAME + " (Id, CustomName, OrderPrice, Country) values (1, 'Arc', 100, 'China')");
db.execSQL("insert into " + OrderDBHelper.TABLE_NAME + " (Id, CustomName, OrderPrice, Country) values (2, 'Bor', 200, 'USA')");
db.execSQL("insert into " + OrderDBHelper.TABLE_NAME + " (Id, CustomName, OrderPrice, Country) values (3, 'Cut', 500, 'Japan')");
db.execSQL("insert into " + OrderDBHelper.TABLE_NAME + " (Id, CustomName, OrderPrice, Country) values (4, 'Bor', 300, 'USA')");
db.execSQL("insert into " + OrderDBHelper.TABLE_NAME + " (Id, CustomName, OrderPrice, Country) values (5, 'Arc', 600, 'China')");
db.execSQL("insert into " + OrderDBHelper.TABLE_NAME + " (Id, CustomName, OrderPrice, Country) values (6, 'Doom', 200, 'China')");

db.setTransactionSuccessful();

插入一长达新数据
俺们还好下insert(String table,String nullColumnHack,ContentValues values)方法来插入,ContentValues其中贯彻即是HashMap,但是彼此还是有异样的,ContenValues
Key只能是String类型,Value只能存储中心型的数目,像string,int之类的,不克积存对象这种事物:

public ContentValues() {
    // Choosing a default size of 8 based on analysis of typical
    // consumption by applications.
    mValues = new HashMap<String, Object>(8);
}

运用insert()方法我们插入一漫漫新数据(7, “Jne”, 700,
“China”),对于修改数据的操作我们一般作为事务(Transaction)处理:

db = ordersDBHelper.getWritableDatabase();
db.beginTransaction();

// insert into Orders(Id, CustomName, OrderPrice, Country) values (7, "Jne", 700, "China");
ContentValues contentValues = new ContentValues();
contentValues.put("Id", 7);
contentValues.put("CustomName", "Jne");
contentValues.put("OrderPrice", 700);
contentValues.put("Country", "China");
db.insertOrThrow(OrderDBHelper.TABLE_NAME, null, contentValues);

db.setTransactionSuccessful();

复制代码 代码如下:

数据库操作无外乎:“增删查改”。对于“增删改”这仿佛对发明内容变换的操作,我们需先调用getWritableDatabase(),在履行之早晚可调用通用的execSQL(String sql)方或者相应之操作API:insert()delete()update()。而针对性“查”,需要调用getReadableDatabase(),这时就不能够使用execSQL方法了,得使用query()rawQuery()计。下面开始逐项介绍。

汝或许感兴趣之稿子:

招来数据

探寻数据来少数单主意,一是public Cursor query(String table,String[] columns,String selection,String[] selectionArgs,String groupBy,String having,String orderBy,String limit);,另外一个凡是public Cursor rawQuery(String sql, String[] selectionArgs)rawQuery的写法类似上面的execSQL,在斯不举行牵线,query道被之参数如下:

  • table:表名称

咱们好看看返回的类且是CursorCursor凡是一个游标接口,提供了遍历查询结果的办法,如运动指针方法move(),获得列值方法。Cursor游标常用方法如下:

咱先来查阅同一翻看用户称也”Bor”的消息:

db = ordersDBHelper.getReadableDatabase();

// select * from Orders where CustomName = 'Bor'
cursor = db.query(OrderDBHelper.TABLE_NAME,
        ORDER_COLUMNS,
        "CustomName = ?",
        new String[] {"Bor"},
        null, null, null);

if (cursor.getCount() > 0) {
    List<Order> orderList = new ArrayList<Order>(cursor.getCount());
    while (cursor.moveToNext()) {
        Order order = parseOrder(cursor);
        orderList.add(order);
    }
    return orderList;
}

自然我们呢堪查询总数、最要命价值最小值之类的,以询问Country为China的用户总数也例:

db = ordersDBHelper.getReadableDatabase();
// select count(Id) from Orders where Country = 'China'
cursor = db.query(OrderDBHelper.TABLE_NAME,
        new String[]{"COUNT(Id)"},
        "Country = ?",
        new String[] {"China"},
        null, null, null);

if (cursor.moveToFirst()) {
    count = cursor.getInt(0);
}

迄今为止SQLite就介绍了了,大家好下载Demo详见查看。Demo中还有有别样比较干货的东西,大家可打挖掘。当然Demo中呢略微不足,我还会见更新,尽量做到为用户通过Demo就能够学会如何用SQLite。

复制代码 代码如下:

以此近乎主要用来建数据库与建表用,我们再度创一个OrderDao用于拍卖所有的数据操作方法。在OrderDao钟实例化OrderDBHelper:

1. 以表名改呢临时表

复制代码 代码如下:

或者

CREATE TABLE Subscription (OrderId VARCHAR(32) PRIMARY KEY ,UserName
VARCHAR(32) NOT NULL ,ProductId VARCHAR(16) NOT NULL);

INSERT INTO Subscription SELECT OrderId, “”, ProductId FROM
__temp__Subscription;

复制代码 代码如下:

INSERT INTO Subscription() SELECT OrderId, “”, ProductId FROM
__temp__Subscription;

SQLite提供了ALTER
TABLE命令,允许用户重命名或续加新的字段到曾经发表中,但是不能够起表中删除字段。

本来,如果遇减少字段的景,也得以透过创办临时表的方式来促成。

2. 创造新表

3. 导入数据

DROP TABLE __temp__Subscription;

* 注意 双引号”” 是因此来补偿原来不设有的数的
4. 删减临时表

ALTER TABLE Subscription ADD COLUMN Activation BLOB;
ALTER TABLE Subscription ADD COLUMN Key BLOB;

相关文章

标签:,

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图