將數據庫保存在數據庫對於重複或者結構化數據(好比契約信息)而言是理想之選。SQL數據庫的主要原則之一是架構:數據庫如何組織正式聲明。架構體現於用於建立數據庫的SQL語句。它有助於建立伴隨類,即契約類,其以一種系統性、自記錄的方式明確指定架構佈局。java
契約類是用於定義URL、表格和列名稱的常數的容器。契約類容許跨同一軟件包中的全部其餘類使用相同的常數。能夠在一個位置更改列名稱並使其在整個代碼中傳播。組織契約類的一種良好方法是將對於整個數據庫而言是全局性的定義放入類的根級別。而後爲枚舉其列的每一個表格建立內部類。android
SQLiteOpenHelper類中有一組有用的API。當使用此類獲取對數據庫的引用時,系統將只在須要之時而不是應用啓動過程當中執行可能長期運行的操做:建立和更新數據庫。因爲它們可能長期運行,所以須要確保在後臺線程中調用getWritableDatabase()或getReadableDatabase(),好比使用AsyncTask或IntentService。git
SQLite是輕量級嵌入式數據庫引擎,它支持SQL語言,而且只利用不多的內存就有很好的性能。此外它仍是開源的,任何人均可以使用它。許多開源項目(Mozilla,PHP,Python)都使用了SQLite。SQLite由如下幾個組件組成:SQL編譯器、內核、後端以及附件。SQLite經過利用虛擬數據庫引擎(VDBE)。使調試、修改和擴展SQLite的內核變得更加方便。github
特色:面向資源有限的設備,沒有服務器進程,全部數據存放在同一文件中跨平臺,可自由複製。優勢是高效,Android運行時環境包含了完整的SQLite。sql
SQLite和其餘數據庫最大的不一樣就是對數據類型的支持,建立一個表時,能夠在CREATE TABLE語句中指定某列的數據類型,可是你能夠把任何數據類型放入任何列中。當某個值插入數據庫時,SQLite將檢查它的類型,若是該類型與關聯的列不匹配,則SQLite會嘗試將該值轉換成該列的類型。若是不能轉換,則該值將做爲其自己具備的類型存儲。好比能夠把一個字符串(String)放入INTEGER列。SQLite稱這爲「弱類型」(manifest typing.)。此外,SQLite不支持一些標準的SQL功能,特別是外鍵約束(FOREIGN KEY constrains),嵌套 transcaction 和 RIGHT OUTER JOIN 和 FULL OUTER JOIN, 還有一些 ALTER TABLE 功能。除了上述功能外,SQLite 是一個完整的 SQL 系統,擁有完整的觸發器,交易等等。Android在運行時集成了SQLite,全部每一個Android應用程序均可以使用SQLite數據庫。數據庫
因爲JDBC(Java數據庫鏈接)會消耗太多的系統資源,因此JDBC對於手機這種內存設備來講並不合適。所以,Android提供了一些新的API來使用SQLite數據庫。數據庫存儲在data/<項目文件夾>/database/下。Android開發中使用SQLite數據庫,Activity能夠經過Content Provider或者Service訪問一個數據庫。在Android應用程序中使用SQLite,必須本身建立數據庫,而後建立表,索引,填充數據。後端
Android提供了SQLiteOpenHelper幫助建立一個數據庫,只要繼承SQLiteOpenHelper類,就能夠輕鬆的建立數據庫。SQLiteOpenHelper類根據開發應用程序的須要,封裝了建立和更新數據庫使用的邏輯。SQLiteOpenHelper的子類,至少須要實現三個方法。服務器
SQLiteOpenHelper的子類,至少須要實現三個方法。架構
(1)構造函數,調用父類的SQLiteOpenHelper的構造函數。這個方法須要四個參數:上下文環境(例如,一個Activity),數據庫名稱,一個可選的遊標工程(一般爲null),一個表明正在使用的數據庫模型版本的整數。app
(2)onCreate()方法,它須要一個SQLiteDatabase對象做爲參數,根據須要對這個對象填充表和初始化數據。
(3)onUpgrage()方法,它須要三個參數,一個SQLiteDatabase,一箇舊的版本號和一個新的版本號,這樣就能夠清楚如何將一個數據庫從舊的模型轉變到新的模型。
要從數據庫執行寫入和讀取的操做,請分別調用getWriteableDatabase()和getReadableDatabase(),兩者都會返回一個表示數據庫的SQLiteDatabase對象,並提供用於SQLite操做的方法。當完成了對數據庫的操做(加入Activity已經關閉),須要調用SQLiteDatabase的close()方法來釋放掉數據庫鏈接。爲了建立表和索引,須要調用SQLiteDatabase的execSQL()方法來執行DDL語句。若是沒有異常,這個方法沒有返回值。SQLite會自動爲主鍵列建立索引。一般狀況下,第一次建立數據庫時建立了表和索引。可使用execSQL()方法執行INSERT、UPDATE、DELETE等語句來更新表的數據,還可使用SQLiteDatabase對象的insert()、update()和delete()方法。
update()方法有四個參數,分別是表名,表示列名和值的ContentValues對象,可選的WHERE條件和可選的填充WHERE語句的字符串,這些字符串會替換WHERE條件中的」?」標記。
有兩個方法使用SELECT從SQLite數據庫檢索數據。
(1)使用rawQuery()直接調用SELECT語句,使用query()方法建立一個查詢。返回值是一個cursor對象。
(2)使用遊標
無論如何執行查詢,都會返回一個Cursor(數據庫遊標),經過requery()方法從新執行查詢獲得遊標。
public class SQDBLiteHelper extends SQLiteOpenHelper { private static final String DATABASE_NAME="test.db"; private static final int DATABASE_VERSION = 1; public SQDBLiteHelper(Context context){ super(context,DATABASE_NAME,null,DATABASE_VERSION); } //數據庫第一次被建立時onCreate會被調用 @Override public void onCreate(SQLiteDatabase db) { } //數據庫升級 @Override public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { } //數據庫降級 @Override public void onDowngrade(SQLiteDatabase db, int oldVersion, int newVersion) { } }
db.insert("person",null,values);
Cursor c = db.rawQuery("SELECT * FROM person",null); while (c.moveToNext()){ ... } c.close();
ContentValues cv = new ContentValues(); cv.put("age",person.age); db.update("person", cv, "name = ?", new String[]{person.name});
db.delete("person", "age >= ?", new String[]{String.valueOf(person.age)});
<resources> <string name="app_name">DataStorageDemo</string> <string name="action_settings">Settings</string> <string name="SQLite">SQLite</string> <string name="create_table">建立表</string> <string name="insert_table">插入表</string> <string name="read_table">讀取表</string> <string name="delete_table">刪除表</string> <string name="update_table">更新表</string> </resources>
<?xml version="1.0" encoding="utf-8"?> <android.support.design.widget.CoordinatorLayout
xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:fitsSystemWindows="true" tools:context="com.zhangmiao.datastoragedemo.MainActivity"> <LinearLayout android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/SQLite" android:layout_gravity="center_horizontal" /> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal" android:layout_marginTop="@dimen/fab_margin" android:layout_marginBottom="@dimen/fab_margin" > <Button android:id="@+id/sqlite_insert" android:layout_height="wrap_content" android:layout_width="0dp" android:layout_weight="1" android:text="@string/insert_table" /> <Button android:id="@+id/sqlite_update" android:layout_height="wrap_content" android:layout_width="0dp" android:layout_weight="1" android:text="@string/update_table" /> <Button android:id="@+id/sqlite_read" android:layout_height="wrap_content" android:layout_width="0dp" android:layout_weight="1" android:text="@string/read_table" /> <Button android:id="@+id/sqlite_delete" android:layout_height="wrap_content" android:layout_width="0dp" android:layout_weight="1" android:text="@string/delete_table" /> </LinearLayout> <TextView android:id="@+id/table_info" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="@string/app_name" /> </LinearLayout> </android.support.design.widget.CoordinatorLayout>
package com.zhangmiao.datastoragedemo; /** * Created by zhangmiao on 2016/12/16. */ public class Person { public int _id; public String name; public int age; public String info; public Person() { } public Person(String name, int age, String info) { this.name = name; this.age = age; this.info = info; } }
public class SQLiteDBHelper extends SQLiteOpenHelper { private static final String DATABASE_NAME="test.db"; private static final int DATABASE_VERSION = 1; public SQLiteDBHelper(Context context){ super(context,DATABASE_NAME,null,DATABASE_VERSION); } //數據庫第一次被建立時onCreate會被調用 @Override public void onCreate(SQLiteDatabase db) { db.execSQL("CREATE TABLE IF NOT EXISTS person"+ "(_id INTEGER PRIMARY KEY AUTOINCREMENT,name VARCHAR," + "age INTEGER,info TEXT)"); } //數據庫升級 @Override public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { db.execSQL("DROP TABLE IF EXISTS person"); onCreate(db); } //數據庫降級 @Override public void onDowngrade(SQLiteDatabase db, int oldVersion, int newVersion) { onUpgrade(db,oldVersion,newVersion); } }
package com.zhangmiao.datastoragedemo; import android.content.ContentValues; import android.content.Context; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import java.util.ArrayList; import java.util.List; /** * Created by zhangmiao on 2016/12/16. */ public class SQLiteDBManager { private SQLiteDBHelper helper; private SQLiteDatabase db; public SQLiteDBManager(Context context){ helper = new SQLiteDBHelper(context); db = helper.getWritableDatabase(); } public void add(List<Person> persons){ for(int i = 0; i<persons.size();i++){ ContentValues values = new ContentValues(); Person person = persons.get(i); values.put("name",person.name); values.put("age",person.age); values.put("info",person.info); db.insert("person",null,values); } } public void updateAge(Person person){ ContentValues cv = new ContentValues(); cv.put("age",person.age); db.update("person", cv, "name = ?", new String[]{person.name}); } public void deleteOldPerson(Person person){ db.delete("person", "age >= ?", new String[]{String.valueOf(person.age)}); } public List<Person> query(){ ArrayList<Person> persons = new ArrayList<>(); Cursor c = queryTheCursor(); while (c.moveToNext()){ Person person = new Person(); person._id = c.getInt(c.getColumnIndex("_id")); person.name = c.getString(c.getColumnIndex("name")); person.age = c.getInt(c.getColumnIndex("age")); person.info = c.getString(c.getColumnIndex("info")); persons.add(person); } c.close(); return persons; } public Cursor queryTheCursor(){ Cursor c = db.rawQuery("SELECT * FROM person",null); return c; } public void closeDB(){ db.close(); } }
package com.zhangmiao.datastoragedemo; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.view.View; import android.widget.Button; import android.widget.TextView; import java.util.ArrayList; import java.util.List; public class MainActivity extends AppCompatActivity implements View.OnClickListener{ private SQLiteDBHelper mDbHelper; private List<Person> mPersons; private SQLiteDBManager mManager; private TextView mTableInfo; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mDbHelper = new SQLiteDBHelper(this); mManager = new SQLiteDBManager(this); Button sqliteInsert = (Button)findViewById(R.id.sqlite_insert); Button sqliteRead = (Button)findViewById(R.id.sqlite_read); Button sqliteUpdate = (Button)findViewById(R.id.sqlite_update); Button sqliteDelete = (Button)findViewById(R.id.sqlite_delete); mTableInfo = (TextView)findViewById(R.id.table_info); sqliteDelete.setOnClickListener(this); sqliteInsert.setOnClickListener(this); sqliteRead.setOnClickListener(this); sqliteUpdate.setOnClickListener(this); mPersons = new ArrayList<>(); initData(); } private void initData(){ String[] names = new String[]{"zhang","zhao","li","wu"}; int[] ages = new int[]{20,21,19,28}; String[] infos = new String[]{"women","men","men","women"}; for(int i = 0; i< names.length;i++){ Person person = new Person(names[i],ages[i],infos[i]); mPersons.add(person); } } @Override public void onClick(View v) { switch (v.getId()){ case R.id.sqlite_insert: mManager.add(mPersons); break; case R.id.sqlite_read: writeTableInfo(mManager.query()); break; case R.id.sqlite_update: Person person = new Person("zhang",18,"women"); mManager.updateAge(person); break; case R.id.sqlite_delete: Person person1 = new Person(); person1.age = 25; mManager.deleteOldPerson(person1); break; } } private void writeTableInfo(List<Person> persons){ String message = ""; for(int i = 0; i<persons.size();i++){ Person person = persons.get(i); message += "id: "+person._id + " name: "+ person.name +" age: "+person.age + " info: "+person.info + "\n"; } mTableInfo.setText(message); } }