成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Android本地數據存儲之Room詳細使用

移動開發 Android
@Database:Room數據庫對象。該類需要繼承自RoomDatabase,通過Room.databaseBuilder()結合單例設計模式,完成數據庫的創建工作。我們創建的Dao對象,在這里以抽象方法的形式返回,只需一行代碼即可。

Room在SQLite基礎上做了ORM封裝,使用起來類似JPA,不需要寫太多的sql。

準備,導入依賴

//room
def room_version="2.4.2"
implementation "androidx.room:room-runtime:$room_version"
annotationProcessor "androidx.room:room-compiler:$room_version"
//implementation "androidx.room:room-rxjava2:$room_version"
//implementation "androidx.room:room-rxjava3:$room_version"
//implementation "androidx.room:room-guava:$room_version"
//testImplementation "androidx.room:room-testing:$room_version"
//implementation "androidx.room:room-paging:2.5.0-alpha01"

關鍵注解說明

1、@Database:Room數據庫對象。該類需要繼承自RoomDatabase,通過Room.databaseBuilder()結合單例設計模式,完成數據庫的創建工作。我們創建的Dao對象,在這里以抽象方法的形式返回,只需一行代碼即可。

  • entities:指定該數據庫有哪些表
  • version:指定數據庫版本號,后續數據庫的升級正是依據版本號來判斷的

2、@Entity:該類與Room中表關聯起來。tableName屬性可以為該表設置名字,如果不設置,則表名與類名相同。

3、@PrimaryKey:用于指定該字段作為表的主鍵。

4、@ColumnInfo:設置該字段存儲在數據庫表中的名字并指定字段的類型;默認字段名和屬性名一樣

5、@Ignore:忽略該字段

一、使用步驟

1、創建實體類,對應數據庫中一張表,使用注解@Entity 2、創建Dao接口類,用于操作數據,使用注解@Dao;不需要實現,在編譯的時候,框架會自動生成實現類 3、創建數據庫對象Database,繼承RoomDatabase,使用單例模式返回實例 4、在Activity中使用,Room數據操作必須在異步線程中執行,所以在Activity中使用線程池執行,或者使用RxJava切換線程

使用代碼示例

1、創建實體類,對應數據庫中一張表,使用注解@Entity

@Entity
public class Person {

// 主鍵,自增長
@PrimaryKey(autoGenerate = true)
private int id;
private String name;
private String sex;
private int age;
}

2、創建Dao接口類,用于操作數據,使用注解@Dao;不需要實現,在編譯的時候,框架會自動生成實現類

@Dao
public interface PersonDao {
// 插入
@Insert
void insertPersons(Person... persons);
// 修改
@Update
void updatePersons(Person... persons);
// 刪除所有
@Query("delete from Person")
void deleteAllPersons();
// 刪除指定實體
@Delete
void deletePersons(Person... persons);
// 根據id刪除
@Query("delete from Person where id in (:ids)")
void deleteByIds(int ...ids);
// 根據id查詢
@Query("select * from Person where id in (:ids)")
List<Person> selectByIds(int ...ids);
// 查詢所有
@Query("select * from Person order by id desc")
List<Person> selectAllPersons();
}

3、創建數據庫對象Database,繼承RoomDatabase,使用單例模式返回實例

@Database(entities = {Person.class}, version = 1)
public abstract class AppDatabase extends RoomDatabase {
public abstract PersonDao personDao();

private volatile static AppDatabase instance;

public static AppDatabase getInstance(Context context){
if (instance == null) {
synchronized (DBHelper.class) {
if (instance == null) {
instance = Room.databaseBuilder(context, AppDatabase.class, "person.db").build();
}
}
}
return instance;
}
}

4、在Activity中使用

Room數據操作必須在異步線程中執行,所以在Activity中使用線程池執行

ExecutorService pool = Executors.newCachedThreadPool();

// 插入數據
public void insertRoom(View view){
AppDatabase db = AppDatabase.getInstance(getApplicationContext());

pool.execute(() -> {
PersonDao dao = db.personDao();
Person p1 = new Person("用戶1", "男", 18);
Person p2 = new Person("用戶2", "男", 28);
Person p3 = new Person("用戶3", "男", 38);
dao.insertPersons(p1, p2, p3);
});

}

// 查詢數據
public void queryRoom(View view){

AppDatabase db = AppDatabase.getInstance(getApplicationContext());

pool.execute(() -> {

PersonDao dao = db.personDao();
List<Person> list = dao.selectAllPersons();
list.forEach(p-> Log.d("test", p.toString()));

});
}

// 根據id查詢
public void queryRoomById(View view){

AppDatabase db = AppDatabase.getInstance(getApplicationContext());

pool.execute(() -> {

PersonDao dao = db.personDao();
List<Person> list = dao.selectByIds(3,4);
list.forEach(p-> Log.d("test", p.toString()));

});
}

// 刪除
public void deleteRoom(View view){

AppDatabase db = AppDatabase.getInstance(getApplicationContext());

pool.execute(() -> {

PersonDao dao = db.personDao();
dao.deleteByIds(1,2);
});
}

二、類型轉換器

SQLite支持null,integer,real,text,blob五種數據類型,實際上SQLite也接受varchar,char,decimal等數據類型,只不過在運算中或保存時會轉換成對應的5種數據類型,因此,可以將各種類型數據保存到任何字段中。

除了上述基本類型外,其他如Date、BigDecimal、或Json對象等如何存儲呢?

Room給我們提供的非常方便的類型轉換器功能。

  • @TypeConverter,定義類型轉換靜態方法
  • @TypeConverters,定義包含一組轉換方法的class類

1、創建類型轉換類型,如,Date和Long互轉

使用注解@TypeConverter聲明具體的轉換方法,每個方法必須包含一個參數,以及必須有返回值。

public class DateConverter {

@TypeConverter
public static Date toDate(Long dateLong){
return dateLong == null ? null : new Date(dateLong);
}

@TypeConverter
public static Long fromDate(Date date){
return date == null ? null : date.getTime();
}
}

2、將創建好的轉換器類,在entity上使用

使用注解@TypeConverters({DateConverter.class}),那么實體類中的所有的Date屬性都會被轉換成Long存儲,查詢取出的時候,會自動從Long轉換成Date顯示。

注意:@TypeConverters放在元素屬性、Class、Dao、Database上面

  • 放在元素屬性,只對改屬性有效
  • 放在實體Class上,對class中所有元素有效
  • 放在Dao上,對Dao的所有方法有效
  • 放在Database,對Database的所有實體和所有Dao都有效

為避免出現混亂,通常建議只在Entity或屬性上定義轉換器

@Entity
@TypeConverters({DateConverter.class})
public class BsGoods {
private static final long serialVersionUID = 1122172437556010779L;
// 主鍵
@PrimaryKey
private Long id;
private Date createdDate;
private Date updatedDate;

...
}

其他類型轉換示例,BigDecimal轉String。

如果是JavaBean等復雜對象,可以轉換成Json字符串存儲。

public class BigDecimalConverter {

@TypeConverter
public static String toStr(BigDecimal decimal) {
return decimal == null ? null : decimal.toString();
}

@TypeConverter
public static BigDecimal toDecimal(String str) {
return str == null ? null : new BigDecimal(str);
}
}

三、結合RxJava,在Activity中使用,并且更新界面UI元素

Android的界面UI元素更新,必須在主線程中執行,但是Room的數據查詢,又只能使用異常線程處理。那么如何將查詢到數據,更新到頁面控件上面呢?

這里可以結合RxJava實現流式操作,線下切換!

示例代碼,查詢所有商品數據,顯示在頁面控件上面,控件使用的是自定義的TableView,暫不展開,這里只顯示數據查詢以及顯示。

1、在Database類中定義查詢方法,傳入回調函數

public void selectAll(Consumer<List<BsGoods>> fun) {
BsGoodsDao dao = bsGoodsDao();

Observable.just("select")
.map(s -> dao.selectAll())
.subscribeOn(Schedulers.io())// 給上面的操作分配異步線程
.observeOn(AndroidSchedulers.mainThread())// 給終點分配安卓主線程
.subscribe(new Observer<List<BsGoods>>() {
@Override
public void onSubscribe(@NonNull Disposable d) {

}

@Override
public void onNext(@NonNull List<BsGoods> bsGoods) {
fun.accept(bsGoods);
}

@Override
public void onError(@NonNull Throwable e) {

}

@Override
public void onComplete() {

}
});
}

2、在Activity中使用,傳入回調函數更新界面UI

private void initializeTableViewLocal(){
BsGoodsDatabase db = BsGoodsDatabase.getInstance(getContext());
db.selectAll(list -> {
GoodsTableViewModel tableViewModel = new GoodsTableViewModel(list);
TableViewAdapter tableViewAdapter = new TableViewAdapter(tableViewModel);
mTableView.setAdapter(tableViewAdapter);
mTableView.setTableViewListener(new TableViewListener(mTableView));

tableViewAdapter.setAllItems(tableViewModel.getColumnHeaderList(), tableViewModel
.getRowHeaderList(), tableViewModel.getCellList());

});

}


責任編輯:武曉燕 來源: 今日頭條
相關推薦

2011-05-31 17:32:32

Android SharedPref

2010-12-20 09:44:36

SQLite.C#

2022-09-06 09:59:21

數據備份

2013-06-21 10:33:02

虛擬化應用存儲虛擬化

2009-03-06 10:11:30

2017-09-12 13:13:33

Android遷移測試Room

2023-12-08 08:26:05

數據存儲持久性

2018-06-11 12:53:53

LinuxStratis本地存儲

2012-12-24 09:20:48

AndoidUnity3D

2023-11-26 09:06:46

2012-04-24 10:08:12

HTML5

2021-10-27 11:33:31

數據倉庫架構

2022-09-13 15:19:20

服務器云備份存儲

2011-12-05 14:07:17

虛擬化本地存儲桌面虛擬化

2015-08-06 11:10:46

開源IaaS軟件ZStack本地存儲

2011-12-02 09:57:50

存儲虛擬化存儲虛擬化

2021-05-06 08:04:37

存儲StratisCentos 8

2015-07-09 13:47:37

IOSFMDB

2011-03-08 09:58:21

海量數據

2018-03-20 09:36:57

數據倉庫數據存儲知識
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲欧美综合 | 国产日本精品视频 | 亚洲精品视频在线 | 国产精品久久久久久久久久久久久久 | 亚洲精品一区二区三区中文字幕 | 国产精品一区二区在线播放 | 亚洲午夜av久久乱码 | 天天爽综合网 | 久久躁日日躁aaaaxxxx | 国产精品日韩欧美 | 久久与欧美 | 免费视频一区二区 | 亚洲精品日韩精品 | 欧美寡妇偷汉性猛交 | 99久久精品国产一区二区三区 | 国产在线一区观看 | 久久久涩 | 久久精品亚洲精品国产欧美 | 黄色片a级 | 日韩第一区 | 97偷拍视频 | 久日精品 | 91精品国产91久久久久久吃药 | 激情六月丁香 | 天天看天天爽 | 欧美一区在线视频 | 亚洲一区二区精品视频 | 在线观看中文视频 | 国产一二三区精品视频 | 日韩综合网 | 国产美女视频一区 | 亚洲精品一区在线观看 | 日韩精品成人 | 亚洲精品电影网在线观看 | 国产一区二区三区在线看 | 国产精品视频久久 | 九九色综合 | 久久九九色 | 欧美一区二区三区在线观看视频 | 手机av在线 | 久久久精品一区 |