編輯:關於android開發
將數據庫保存在數據庫對於重復或者結構化數據(比如契約信息)而言是理想之選。SQL數據庫的主要原則之一是架構:數據庫如何組織正式聲明。架構體現於用於創建數據庫的SQL語句。它有助於創建伴隨類,即契約類,其以一種系統性、自記錄的方式明確指定架構布局。
契約類是用於定義URL、表格和列名稱的常數的容器。契約類允許跨同一軟件包中的所有其他類使用相同的常數。可以在一個位置更改列名稱並使其在整個代碼中傳播。組織契約類的一種良好方法是將對於整個數據庫而言是全局性的定義放入類的根級別。然後為枚舉其列的每個表格創建內部類。
SQLiteOpenHelper類中有一組有用的API。當使用此類獲取對數據庫的引用時,系統將只在需要之時而不是應用啟動過程中執行可能長期運行的操作:創建和更新數據庫。由於它們可能長期運行,因此需要確保在後台線程中調用getWritableDatabase()或getReadableDatabase(),比如使用AsyncTask或IntentService。
SQLite是輕量級嵌入式數據庫引擎,它支持SQL語言,並且只利用很少的內存就有很好的性能。此外它還是開源的,任何人都可以使用它。許多開源項目(Mozilla,PHP,Python)都使用了SQLite。SQLite由以下幾個組件組成:SQL編譯器、內核、後端以及附件。SQLite通過利用虛擬數據庫引擎(VDBE)。使調試、修改和擴展SQLite的內核變得更加方便。
特點:面向資源有限的設備,沒有服務器進程,所有數據存放在同一文件中跨平台,可自由復制。優點是高效,Android運行時環境包含了完整的SQLite。
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),一個代表正在使用的數據庫模型版本的整數。
(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);
}
}
Android 交錯 GridView,androidgridview
Android 交錯 GridView,androidgridview下載 Demo 交錯 GridView 交錯 GridView 只是具有不等大小行、多個列的 Gr
ArrayAdapter適配器的用法,模擬QQ發消息界面。,arrayadapter適配器
ArrayAdapter適配器的用法,模擬QQ發消息界面。,arrayadapter適配器 1 import java.util.ArrayList; 2 3 im
自定義控件之圓頭像,自定義控件
自定義控件之圓頭像,自定義控件先上代碼: package com.andy.oschina_android.widget; import android.conten
Android 調用百度地圖API,androidapi
Android 調用百度地圖API,androidapi一、到 百度地圖開發平台下載SDK http://lbsyun.baidu.com/index.php?title
Android移動APP開發筆記——Cordova(PhoneGap)通過CordovaPlugin插件調用 Activity 實例,phonegapcordova
Android移動APP開發筆記——Cordova(PhoneGap)通