Android之SQLite數據存儲

来源:http://www.cnblogs.com/zhangmiao14/archive/2016/12/19/6198167.html
-Advertisement-
Play Games

一、SQLite保存數據介紹 將資料庫保存在資料庫對於重覆或者結構化數據(比如契約信息)而言是理想之選。SQL資料庫的主要原則之一是架構:資料庫如何組織正式聲明。架構體現於用於創建資料庫的SQL語句。它有助於創建伴隨類,即契約類,其以一種系統性、自記錄的方式明確指定架構佈局。 契約類是用於定義URL ...


一、SQLite保存數據介紹

  將資料庫保存在資料庫對於重覆或者結構化數據(比如契約信息)而言是理想之選。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()方法重新執行查詢得到游標。

二、使用方法

1.實現SQLiteOpenHelper

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) {
    }
}

2.插入數據

db.insert("person",null,values);

3.讀取數據

 Cursor c = db.rawQuery("SELECT * FROM person",null);
 while (c.moveToNext()){
   ...   
 }
 c.close();

4.修改數據

ContentValues cv = new ContentValues();
        cv.put("age",person.age);
        db.update("person", cv, "name = ?", new String[]{person.name});

5.刪除數據

db.delete("person", "age >= ?", new String[]{String.valueOf(person.age)});

三、小案例

1.添加strings.xml文件

<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>

2.修改activity_main.xml文件

<?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>

3.添加Person類

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;
    }
}

4.添加SQLiteDBHelper類

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);
    }
}

5.添加SQLiteDBManager類

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();
    }
}

6.修改MainActivity

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);
    }
}

代碼下載地址:https://github.com/ZhangMiao147/DataStorageDemo

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 最近為了滿足蘋果的 https 要求, 經過努力終於寫出了方法 驗證 SSL 證書是否滿足 ATS 要求 nscurl --ats-diagnostics --verbose https://你的功能變數名稱 PASS 符合要求 輸出滿足 ATS 的證書 openssl s_client -connect ...
  • 首先想強調一下“語音識別”四個字字面意義上的需求:用戶說話然後馬上把用戶說的話轉成文字顯示!,這才是開發者真正需要的功能。 做需求之前其實是先谷歌百度一下看有沒有造好的輪子直接用,結果真的很呵呵,都是標著這個庫深入學習的標題,裡面調用一下api從URL里取出一個本地語音文件進行識別,這就沒了? 最基 ...
  • 一、SharedPreferences保存數據介紹 如果有想要保存的相對較小鍵值集合,應使用SharedPreferences API。SharedPreferences對象指向包含鍵值對的文件並提供讀寫這些文件的簡單方法。每個SharedPreferences文件由框架進行管理並且可以專用或共用。 ...
  • 微信小程式提交審核需要選擇資質服務範圍,如果服務範圍不對,審核會不通過, 開發小程式之前,最好先查詢所開發小程式的資質範圍,否則無法通過微信審核。 小程式的資質範圍查詢地址,數據同步微信官方 https://weixin.hotapp.cn/weixinmob ...
  • CocoaPods是什麼? 當你開發iOS應用時,會經常使用到很多第三方開源類庫,比如JSONKit,AFNetWorking等等。可能某個類庫又用到其他類庫,所以要使用它,必須得另外下載其他類庫,而其他類庫又用到其他類庫,“子子孫孫無窮盡也”,這也許是比較特殊的情況。總之小編的意思就是,手動一個個 ...
  • 多線程下載就是將同一個網路上的原始文件根據線程個數分成均等份,然後每個單獨的線程下載對應的一部分 ...
  • 概覽屏幕 概覽屏幕 概覽屏幕(也稱為最新動態屏幕、最近任務列表或最近使用的應用)是一個系統級別 UI,其中列出了最近訪問過的 Activity 和任務。 用戶可以瀏覽該列表並選擇要恢復的任務,也可以通過滑動清除任務將其從列表中移除。 對於 Android 5.0 版本(API 級別 21),包含不同 ...
  • 廣度優先搜索 在給定圖G=(V,E)和一個特定的源頂點s的情況下,廣度優先搜索系統地探索G中的邊,以期“發現”可從s 到達的所有頂點,並計算s 到所有這些可達頂點之間的距離(即最少的邊數)。該搜索演算法同時還能生成一棵根為s、且包括所有s 的可達頂點的廣度優先樹。對從s 可達的任意頂點v,廣度優先樹中 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...