如何將Android Room封裝成一個相似Redis的緩存數據庫的效果。java
Android Room封裝緩存。若是你還在用SharedPreferences,試試使用這種方式?android
Android Room是一個很是棒的數據庫框架,提供了明確地Dao層來操做數據庫(愈來愈像spring data jpa啦),這裏我來告訴你如何使用Room封裝出一個相似Redis的使用方式(寫事後端的童鞋應該懂)。主要目標,把它弄成K/V數據庫的效果。spring
allprojects {
repositories {
mavenCentral()
google()
jcenter()
maven { url "https://jitpack.io" }
maven { url "https://maven.google.com" }
}
}
dependencies {
implementation 'android.arch.persistence.room:runtime:1.0.0'
annotationProcessor 'android.arch.persistence.room:compiler:1.0.0'
}
複製代碼
緩存表數據庫
@Entity(tableName = "cache")
public class CacheEntity {
@PrimaryKey(autoGenerate = true)
public int id;
public String key;
public String value;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getKey() {
return key;
}
public void setKey(String key) {
this.key = key;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
}
複製代碼
DatabaseConfig.classjson
@Database(entities = {CacheEntity.class}, version = 1, exportSchema = false)
public abstract class DatabaseConfig extends RoomDatabase {
public abstract CacheDao mCacheDao();
}
複製代碼
DatabaseInitialize.class後端
public class DatabaseInitialize {
private static DatabaseConfig databaseConfig;
public DatabaseInitialize(){}
public static void init(Context context) {
// 生成數據庫實例
databaseConfig = Room
.databaseBuilder(context.getApplicationContext(), DatabaseConfig.class, "huiminyougou")
.allowMainThreadQueries() // 容許主線程中查詢
.build();
}
static DatabaseConfig getDatabaseConfig() {
if (databaseConfig == null) {
throw new NullPointerException("DatabaseInitialize.init(context) has not call, remember call this function in your Application.class");
}
return databaseConfig;
}
}
複製代碼
DatabaseSession.classapi
public final class DatabaseSession {
private DatabaseSession(){}
public static DatabaseConfig get() {
return DatabaseInitialize.getDatabaseConfig();
}
}
複製代碼
@Dao
public interface CacheDao {
@Insert
void insertCaches(CacheEntity... cacheEntities);
@Update
void updateCaches(CacheEntity... cacheEntities);
@Query("SELECT * FROM cache WHERE `key` = :key LIMIT 0,1")
CacheEntity findByKey(String key);
@Delete
void deleteCaches(CacheEntity... cacheEntities);
@Query("SELECT * FROM cache")
CacheEntity[] findAll();
}
複製代碼
public final class CacheService {
private CacheService() {
}
private static CacheDao getRepository() {
return DatabaseSession.get().mCacheDao();
}
/**
* 設置緩存
*
* @param key
* @param value
*/
public static void set(String key, String value) {
CacheEntity cacheEntity;
cacheEntity = getRepository().findByKey(key);
if (cacheEntity == null) {
cacheEntity = new CacheEntity();
cacheEntity.setKey(key);
cacheEntity.setValue(value);
getRepository().insertCaches(cacheEntity);
} else {
cacheEntity.setValue(value);
getRepository().updateCaches(cacheEntity);
}
}
/**
* 設置緩存
*
* @param key
* @param value
*/
public static void set(String key, Object value) {
CacheEntity cacheEntity = getRepository().findByKey(key);
if (cacheEntity == null) {
cacheEntity = new CacheEntity();
cacheEntity.setKey(key);
String jsonValue = GsonHelper.toJson(value);
cacheEntity.setValue(jsonValue);
getRepository().insertCaches(cacheEntity);
} else {
String jsonValue = GsonHelper.toJson(value);
cacheEntity.setValue(jsonValue);
getRepository().updateCaches(cacheEntity);
}
}
/**
* 獲取緩存
*
* @param key
* @return
*/
public static String get(String key) {
CacheEntity cacheEntity = getRepository().findByKey(key);
if (cacheEntity == null) {
return null;
}
return cacheEntity.getValue();
}
/**
* 獲取緩存對象
*
* @param key
* @param classOfT
* @param <T>
* @return
*/
public static <T> T get(String key, Class<T> classOfT) {
CacheEntity cacheEntity = getRepository().findByKey(key);
if (cacheEntity == null) {
return null;
}
String jsonValue = cacheEntity.getValue();
return GsonHelper.toObject(jsonValue, classOfT);
}
/**
* 刪除緩存
*
* @param key
*/
public static void delete(String key) {
CacheEntity cacheEntity = getRepository().findByKey(key);
if (cacheEntity != null) {
getRepository().deleteCaches(cacheEntity);
}
}
/**
* 刪除所有緩存
*/
public static void clearAll() {
CacheEntity[] cacheEntities = getRepository().findAll();
if (cacheEntities != null && cacheEntities.length != 0) {
getRepository().deleteCaches(cacheEntities);
}
}
}
複製代碼
GsonHelper.class緩存
public class GsonHelper {
private GsonHelper() {
// no instance
}
private static Gson gson = new GsonBuilder().disableHtmlEscaping().create();
public static String toJson(Object o) {
return gson.toJson(o);
}
public static <T> T toObject(String json, Class<T> classOfT) {
return gson.fromJson(json, classOfT);
}
public static <T> List<T> toList(String json, Class<? extends T[]> clazz) {
T[] array = gson.fromJson(json, clazz);
return Arrays.asList(array);
}
}
複製代碼
@Override
public void onCreate() {
super.onCreate();
DatabaseInitialize.init(this);
}
複製代碼
這裏以一個網絡請求獲取首頁數據爲例: 使用的是retrofit+rxjava,在請求成功後,將響應實體存入數據庫中。網絡
mGoodsRequestService.goodsShowIndex(lng, lat)
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new BaseHttpObserver<GoodsIndex>() {
@Override
public void onSubscribe(Disposable d) {
mView.onRequestSubscribe(d);
}
@Override
public void onSuccess(String msg, GoodsIndex data) {
CacheService.set(Constant.CacheKey.HOME_INDEX, data);
mView.onGoodsIndexRespond(data);
mView.onRequestComplete();
}
@Override
public void onFailure(String msg) {
}
@Override
public void onNetworkError(Throwable e) {
mView.onRequestError(e);
}
});
複製代碼
在頁面啓動伊始,從數據庫中拿到緩存數據:框架
GoodsIndex goodsIndex = CacheService.get(Constant.CacheKey.HOME_INDEX, GoodsIndex.class);
if (goodsIndex != null) {
mView.onGoodsIndexRespond(goodsIndex);
}
複製代碼
這裏的service層處理得並非很好,可是Android上沒有相似@Autowired這樣的註解,網絡不少關於Dagger2的文章也是雲裏霧裏,後面思考一下如何用AOP + Dagger2 封裝一個相似的@Autowired出來。有哪位大佬能夠指點一下^_^