編輯:關於Android編程
當我們的app有數據需要保存到本地緩存時,可以使用file,sharedpreferences,還有sqlite。
sharedpreferences其實使用xml的方式,以鍵值對形式存儲基本數據類型的數據。對於有復雜篩選查詢的
操作,file和sharedpreferences都不能滿足了。sqlite可以滿足有大量復雜查詢要求的緩存數據操作。但是sqlite的使用略復雜,代碼量很大,還好網上有很多優秀的orm框架可使用,比喻ORMlite,greenDao等。
ORMlite,greenDao這些框架都是在SQLite的基礎上封裝的ORM對象關系映射框架,簡化了代碼操作。
而今天的主角:Realm是一個可以替代SQLite以及ORM Libraries的輕量級數據庫。
相比SQLite,Realm更快並且具有很多現代數據庫的特性,比如支持JSON,流式api,數據變更通知,以及加密支持,這些都為安卓開發者帶來了方便。不多介紹,更詳細的介紹參見官網:https://realm.io/
我們重點來說說Reaml的使用,看看到底爽在哪裡。
環境配置:
1、在Project的build.gradle文件中添加依賴:
dependencies {
...
classpath "io.realm:realm-gradle-plugin:1.1.0"
...
}
apply plugin: 'com.android.application' apply plugin: 'realm-android' ....
使用:
在整個使用的過程中,Realm是主角,我們先來看看Realm類中的一段翻譯:
/**
* Realm類可以對你的持久化對象進行存儲和事務管理,可以用來創建RealmObjects實例。領域內的對象可以在任何時間查詢和讀取。
* 創建,修改和刪除等操作必須被包含在一個完整的事務裡面,後面的代碼會講到。
* 該事務確保多個實例(在多個線程)可以在一個一致的狀態和保證事務在ACID前提下,訪問相同的對象。
*
* 當一個Realm實例操作完成後,切記不要忘記調用close()方法。否則會導致本地資源無法釋放,引起OOM。
*
* Realm實例不能在不同的線程間訪問操作。確切的說,你必須在每個要使用的線程上打開一個實例
* 每個線程都會使用引用計數來自動緩存Realm實例,所以只要引用計數不達到零,
* 調用getInstance(RealmConfiguration)方法將會返回緩存的Realm實例,應該算是一個輕量級的操作。
*
* 對於UI線程來說,打開和關閉Realm實例,應當放在onCreate/onDestroy或者onStart/onStop方法中
*
* 在不同的線程間,Realm實例使用Handler機制來調整他的狀態。也就是說,Realm實例在線程中,如果沒有Looper,是不能收到更新通知的,
* 除非手動調用waitForChange()方法
*
* 在安卓Activity領域工作的一個標准模式可以在下面看到
* 在Android Activity中,Realm的標准工作模式如下:
*
*
* public class RealmApplication extends Application {
*
* \@Override
* public void onCreate() {
* super.onCreate();
*
* // The Realm file will be located in package's "files" directory.
* RealmConfiguration realmConfig = new RealmConfiguration.Builder(this).build();
* Realm.setDefaultConfiguration(realmConfig);
* }
* }
*
* public class RealmActivity extends Activity {
*
* private Realm realm;
*
* \@Override
* protected void onCreate(Bundle savedInstanceState) {
* super.onCreate(savedInstanceState);
* setContentView(R.layout.layout_main);
* realm = Realm.getDefaultInstance();
* }
*
* \@Override
* protected void onDestroy() {
* super.onDestroy();
* realm.close();
* }
* }
*
*
* Realm支持String和byte字段長度高達16MB
* 參考連接:
* ACID
* Examples using Realm
*
*/
部分源碼分析:
public final class Realm extends BaseRealm {
//默認的文件名,是啥?
public static final String DEFAULT_REALM_NAME = RealmConfiguration.DEFAULT_REALM_NAME;
//怎麼這麼熟悉呢?是RxJava?
@Override
@OptionalAPI(dependencies = {"rx.Observable"})
public Observable asObservable() {
return configuration.getRxFactory().from(this);
}
//下面這些方法,應該都能顧名思義吧
public void createAllFromJson(Class clazz, JSONArray json) {
}
public void createOrUpdateAllFromJson(Class clazz, JSONArray json) {
}
public E createOrUpdateObjectFromJson(Class clazz, JSONObject json) {
}
public E createObject(Class clazz) {
}
public E copyToRealmOrUpdate(E object) {
}
public void executeTransaction(Transaction transaction) {
}
public void delete(Class clazz) {
}
}
/** * 一個RealmConfiguration對象,可用來設置特定的Realm實例 * RealmConfiguration實例只能通過io.realm.RealmConfiguration.Builder類的build()方法來創建 * 想使用默認的RealmConfiguration實例,請使用io.realm.Realm#getDefaultInstance()方法。 * 如果想使用自己配置RealmConfiguration實例的Realm實例,需要調用Realm#setDefaultConfiguration(RealmConfiguration) * *
* 可以用下面代碼創建一個最簡單配置的實例: * RealmConfiguration config = new RealmConfiguration.Builder(getContext()).build()) * 這樣創建的實例,具有一下屬性: * *
public final class RealmConfiguration {
//默認文件名
public static final String DEFAULT_REALM_NAME = "default.realm";
//Rx工廠
private final RxObservableFactory rxObservableFactory;
//弱引用
private final WeakReference contextWeakRef;
/**
*
* 從Asset目錄中返回Realm文件名,還可以保存在Asset中?
* @return input stream to the asset file.
* @throws IOException if copying the file fails.
*/
InputStream getAssetFile() throws IOException {
Context context = contextWeakRef.get();
if (context != null) {
return context.getAssets().open(assetFilePath);
} else {
}
}
/**
* 使用app自己內置硬盤目錄來存儲Realm file。不需要任何擴展訪問權限。
* 默認目錄為:/data/data//files,這個路徑能否修改取決於供應商的具體實現
*
* @param 參數context請使用application的context.
*/
public Builder(Context context) {
if (context == null) {
throw new IllegalArgumentException("A non-null Context must be provided");
}
RealmCore.loadLibrary(context);
initializeBuilder(context.getFilesDir());
}
}
public class MyApplication extends Application {
private String realmName = "dk.realm";
@Override
public void onCreate() {
super.onCreate();
RealmConfiguration realmConfig = new RealmConfiguration.Builder(this)
.name(realmName)
//.assetFile(this,"realm file path in assets,will copy this file to Context.getFilesDir() replace an empty realm file")
.build();
Realm.setDefaultConfiguration(realmConfig);
}
}
public class TestUser extends RealmObject {
@PrimaryKey
private int userId;//id,主鍵
@Required
private String userName;//用戶姓名,必填字段
private String userPwd;//密碼
private int userAge;//年齡
private String userAddress;//住址
private String userWork;//工作
private String userSex;//性別
//private RealmList list; 集合
//...
}
public class BaseDao {
private Realm realm;
public BaseDao(Realm realm) {
this.realm = realm;
}
/**
* 添加(性能優於下面的saveOrUpdate()方法)
*
* @param object
* @return 保存或者修改是否成功
*/
public boolean insert(RealmObject object) {
try {
realm.beginTransaction();
realm.insert(object);
realm.commitTransaction();
return true;
} catch (Exception e) {
e.printStackTrace();
realm.cancelTransaction();
return false;
}
}
/**
* 添加(性能優於下面的saveOrUpdateBatch()方法)
*
* @param list
* @return 批量保存是否成功
*/
public boolean insert(List list) {
try {
realm.beginTransaction();
realm.insert(list);
realm.commitTransaction();
return true;
} catch (Exception e) {
e.printStackTrace();
realm.cancelTransaction();
return false;
}
}
//...
}
/**
* 單條保存demo
*/
public boolean addOneTest() {
boolean bl = false;
try{
realm.beginTransaction();
//在數據庫中創建一個對象,主鍵默認值為0
TestUser user = realm.createObject(TestUser.class);//(類,主鍵)
//更新數據庫各自段的值
user.setUserName("admin");
//主鍵字段的值由0更新為55。而不是直接創建了一個id為55的對象
user.setUserId(55);
//...
realm.commitTransaction();
bl = true;
}catch (Exception e){
e.printStackTrace();
realm.cancelTransaction();
}
/*try{
realm.beginTransaction();
TestUser user2 = new TestUser("hibrid", "120250", 26, "贛州", "賊", "男");
//不給id,會被默認為0
//user2.setUserId(102);
TestUser userWithId = realm.copyToRealm(user2);
realm.commitTransaction();
bl = true;
}catch (Exception e){
e.printStackTrace();
realm.cancelTransaction();
}*/
return bl;
}
//init data
public boolean init() {
/**
* 此處要注意,方法最後調用的是添加或者修改的方法。
* 如果list的數據都不給id,則第一條記錄添加成功後的id為0,後面的都在此基礎上修改。
* 最後的效果是,數據庫只有一條記錄,id為0,其他字段被更新為了最後一個對象的數據
*/
List list = new ArrayList<>();
list.add(new TestUser(0,"android", "123123", 20, "河南常德", "傳菜員", "女"));
list.add(new TestUser(1,"angel", "13588889988", 21, "雲南西雙版納", "飛行員", "男"));
list.add(new TestUser(2,"adidass", "110119", 28, "雲南德克薩斯州", "海員", "男"));
list.add(new TestUser(3,"hijack", "250250", 39, "加州電廠", "廚師", "女"));
list.add(new TestUser(4,"hibrid", "120250", 26, "贛州", "賊", "男"));
list.add(new TestUser(5,"admin", "123456", 20, "湖北漢城", "程序員", "女"));
return saveOrUpdateBatch(list);
}
/**
* 條件查詢
*
* @return 返回結果集合
*/
public RealmResults findByAnyParams(HashMap
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mRealm = Realm.getDefaultInstance();
userDao = new UserDao(mRealm);
//...
/**
* 數據庫數據更新監聽
*/
mRealm.addChangeListener(this);
}
//...
@Override
public void onChange(Realm element) {
findAll();
}
@Override
protected void onDestroy() {
userDao = null;
mRealm.close();
super.onDestroy();
}
Android學習筆記之Shared Preference
如果沒有特殊要求,我們可以使用Android提供的框架來創建系統樣式的Preference Screen,在其內部可以包含PreferenceCategory和Prefe
Android bluetooth介紹(二): android 藍牙代碼架構及其uart 到rfcomm流程
關鍵詞:藍牙blueZ UART HCI_UART H4 HCI L2CAP RFCOMM 版本:基於android4.2之前版本 bluez內核:linux/
Android 實現伸縮布局效果示例代碼
最近項目實現下面的圖示的效果,本來想用listview+gridview實現,但是貌似挺麻煩的於是就用flowlayout 來addview實現添加伸縮的效果,實現也比較
RecyclerView的通用適配器的高級使用
前言博主由於項目中頻繁的使用了V7包中的RecyclerView來代替ListView的列表展示,所以抽空基於ListView的通用適配器的原理,給RecyclerVie