1. 程式人生 > >Realm 在Android中應用

Realm 在Android中應用

轉載:http://www.cnblogs.com/RaphetS/p/5996265.html

目錄
1、Realm簡介
2、環境配置
3、在Application中初始化Realm
4、建立實體
5、增刪改查
6、非同步操作
7、Demo地址(https://github.com/RaphetS/DemoRealm )

一、Realm簡介

資料庫Realm,是用來替代sqlite的一種解決方案,它有一套自己的資料庫儲存引擎,比sqlite更輕量級,擁有更快的速度,並且具有很多現代資料庫的特性,比如支援JSON,流式api,資料變更通知,自動資料同步,簡單身份驗證,訪問控制,事件處理,最重要的是跨平臺,目前已有Java,Objective C,Swift,React-Native,Xamarin這五種實現。

本篇文章用的版本為Realm 2.0.2(官方文件)

二、環境配置

(1) 在專案的build檔案加上

buildscript {
    repositories {
        jcenter()
    }
    dependencies {
        classpath "io.realm:realm-gradle-plugin:2.0.2"
    }
}

(2) 在app的build檔案加上

apply plugin: 'realm-android'

三、初始化Realm

(1) 在Application的oncreate()方法中Realm.init()
public class
MyApplication extends Application
{ @Override public void onCreate() { super.onCreate(); Realm.init(this); } }
(2)在Application的oncreate()方法中對Realm進行相關配置

①使用預設配置

public class MyApplication extends Application {
  @Override
  public void onCreate() {
    super.onCreate();
    // The Realm file will be located in Context.getFilesDir() with name "default.realm"
Realm.init(this); RealmConfiguration config = new RealmConfiguration.Builder().build(); Realm.setDefaultConfiguration(config); } }

②使用自定義配置

public class MyApplication extends Application {
  @Override
  public void onCreate() {
    super.onCreate();
    Realm.init(this);
    RealmConfiguration config = new  RealmConfiguration.Builder()
                                         .name("myRealm.realm")
                                         .deleteRealmIfMigrationNeeded()
                                         .build();
    Realm.setDefaultConfiguration(config);
  }
}
(3)在AndroidManifest.xml配置自定義的Application
<application
  android:name=".MyApplication"
  ...
/>

四、建立實體

(1)新建一個類繼承RealmObject
public class Dog extends RealmObject {
    private String name;
    private int age;
    
    @PrimaryKey
    private String id;


    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }
}

多對多的關係:

public class Contact extends RealmObject {
    public String name;
    public RealmList<Email> emails;
}

public class Email extends RealmObject {
    public String address;
    public boolean active;
}
(2)其他相關說明

1、支援的資料型別:
boolean, byte, short, int, long, float, double, String, Date and byte[]
在Realm中byte, short, int, long最終都被對映成long型別

2、註解說明

@PrimaryKey

①欄位必須是String、 integer、byte、short、 int、long 以及它們的封裝類Byte, Short, Integer, and Long

②使用了該註解之後可以使用copyToRealmOrUpdate()方法,通過主鍵查詢它的物件,如果查詢到了,則更新它,否則新建一個物件來代替。

③使用了該註解將預設設定(@index)註解

④使用了該註解之後,建立和更新資料將會慢一點,查詢資料會快一點。

@Required

資料不能為null

@Ignore

忽略,即該欄位不被儲存到本地

@Index

為這個欄位新增一個搜尋引擎,這將使插入資料變慢、資料增大,但是查詢會變快。建議在需要優化讀取效能的情況下使用。

五、增

(1)實現方法一:事務操作
型別一 :新建一個物件,並進行儲存
Realm realm=Realm.getDefaultInstance();

realm.beginTransaction();
User user = realm.createObject(User.class); // Create a new object
user.setName("John");
user.setEmail("[email protected]");
realm.commitTransaction();
 型別二:複製一個物件到Realm資料庫
 
Realm realm=Realm.getDefaultInstance();

User user = new User("John");
user.setEmail("[email protected]");

// Copy the object to Realm. Any further changes must happen on realmUser
realm.beginTransaction();
realm.copyToRealm(user);
realm.commitTransaction();
(2)實現方法二:使用事務塊
Realm  mRealm=Realm.getDefaultInstance();

final User user = new User("John");
user.setEmail("[email protected]");

mRealm.executeTransaction(new Realm.Transaction() {
            @Override
            public void execute(Realm realm) {
            
            realm.copyToRealm(user);
               
            }
        });

六、刪

    Realm  mRealm=Realm.getDefaultInstance();

    final RealmResults<Dog> dogs=  mRealm.where(Dog.class).findAll();

        mRealm.executeTransaction(new Realm.Transaction() {
            @Override
            public void execute(Realm realm) {
            
                Dog dog=dogs.get(5);
                dog.deleteFromRealm();
                //刪除第一個資料
                dogs.deleteFirstFromRealm();
                //刪除最後一個數據
                dogs.deleteLastFromRealm();
                //刪除位置為1的資料
                dogs.deleteFromRealm(1);
                //刪除所有資料
                dogs.deleteAllFromRealm();
            }
        });

同樣也可以使用同上的beginTransaction和commitTransaction方法進行刪除

七、改

Realm  mRealm=Realm.getDefaultInstance();

Dog dog = mRealm.where(Dog.class).equalTo("id", id).findFirst();
mRealm.beginTransaction();
dog.setName(newName);
mRealm.commitTransaction();

同樣也可以用事物塊來更新資料

八、查

(1)查詢全部

查詢結果為RealmResults

    public List<Dog> queryAllDog() {
        Realm  mRealm=Realm.getDefaultInstance();
    
        RealmResults<Dog> dogs = mRealm.where(Dog.class).findAll();
        
        return mRealm.copyFromRealm(dogs);
    }
(2)條件查詢
    public Dog queryDogById(String id) {
        Realm  mRealm=Realm.getDefaultInstance();
    
        Dog dog = mRealm.where(Dog.class).equalTo("id", id).findFirst();
        return dog;
    }

常見的條件如下(詳細資料請查官方文件):

  • between(), greaterThan(), lessThan(), greaterThanOrEqualTo() & lessThanOrEqualTo()

  • equalTo() & notEqualTo()

  • contains(), beginsWith() & endsWith()

  • isNull() & isNotNull()

  • isEmpty() & isNotEmpty()

(3)對查詢結果進行排序
 /**
     * query (查詢所有)
     */
    public List<Dog> queryAllDog() {
        RealmResults<Dog> dogs = mRealm.where(Dog.class).findAll();
        /**
         * 對查詢結果,按Id進行排序,只能對查詢結果進行排序
         */
        //增序排列
        dogs=dogs.sort("id");
        //降序排列
        dogs=dogs.sort("id", Sort.DESCENDING);
        return mRealm.copyFromRealm(dogs);
    }
(4)其他查詢

sum,min,max,average只支援整型資料欄位

 /**
     *  查詢平均年齡
     */
    private void getAverageAge() {
         double avgAge=  mRealm.where(Dog.class).findAll().average("age");
    }

    /**
     *  查詢總年齡
     */
    private void getSumAge() {
      Number sum=  mRealm.where(Dog.class).findAll().sum("age");
        int sumAge=sum.intValue();
    }

    /**
     *  查詢最大年齡
     */
    private void getMaxId(){
      Number max=  mRealm.where(Dog.class).findAll().max("age");
        int maxAge=max.intValue();
    }

九、非同步操作

大多數情況下,Realm的增刪改查操作足夠快,可以在UI執行緒中執行操作。但是如果遇到較複雜的增刪改查,或增刪改查操作的資料較多時,就可以子執行緒進行操作。

(1)非同步增:
    private void addCat(final Cat cat) {
      RealmAsyncTask  addTask=  mRealm.executeTransactionAsync(new Realm.Transaction() {
            @Override
            public void execute(Realm realm) {
                realm.copyToRealm(cat);
            }
        }, new Realm.Transaction.OnSuccess() {
            @Override
            public void onSuccess() {
                ToastUtil.showShortToast(mContext,"收藏成功");
            }
        }, new Realm.Transaction.OnError() {
            @Override
            public void onError(Throwable error) {
                ToastUtil.showShortToast(mContext,"收藏失敗");
            }
        });

    }

最後在銷燬Activity或Fragment時,要取消掉非同步任務

@Override
    protected void onDestroy() {
        super.onDestroy();
       if (addTask!=null&&!addTask.isCancelled()){
            addTask.cancel();
        }
    }
(2)非同步刪
    private void deleteCat(final String id, final ImageView imageView){
      RealmAsyncTask  deleteTask=   mRealm.executeTransactionAsync(new Realm.Transaction() {
            @Override
            public void execute(Realm realm) {
                Cat cat=realm.where(Cat.class).equalTo("id",id).findFirst();
                cat.deleteFromRealm();

            }
        }, new Realm.Transaction.OnSuccess() {
            @Override
            public void onSuccess() {
                ToastUtil.showShortToast(mContext,"取消收藏成功");
            }
        }, new Realm.Transaction.OnError() {
            @Override
            public void onError(Throwable error) {
                ToastUtil.showShortToast(mContext,"取消收藏失敗");
            }
        });

    }

最後在銷燬Activity或Fragment時,要取消掉非同步任務

@Override
    protected void onDestroy() {
        super.onDestroy();
       if (deleteTask!=null&&!addTask.isCancelled()){
            deleteTask.cancel();
        }
    }
(3)非同步改
RealmAsyncTask  updateTask=   mRealm.executeTransactionAsync(new Realm.Transaction() {
            @Override
            public void execute(Realm realm) {
                Cat cat=realm.where(Cat.class).equalTo("id",mId).findFirst();
                cat.setName(name);
            }
        }, new Realm.Transaction.OnSuccess() {
            @Override
            public void onSuccess() {
                ToastUtil.showShortToast(UpdateCatActivity.this,"更新成功");
             
            }
        }, new Realm.Transaction.OnError() {
            @Override
            public void onError(Throwable error) {
                ToastUtil.showShortToast(UpdateCatActivity.this,"失敗成功");
            }
        });

最後在銷燬Activity或Fragment時,要取消掉非同步任務

@Override
    protected void onDestroy() {
        super.onDestroy();
       if (updateTask!=null&&!addTask.isCancelled()){
            updateTask.cancel();
        }
    }
(4)非同步查

     RealmResults<Cat>   cats=mRealm.where(Cat.class).findAllAsync();
        cats.addChangeListener(new RealmChangeListener<RealmResults<Cat>>() {
            @Override
            public void onChange(RealmResults<Cat> element) {
               element= element.sort("id");
                List<Cat> datas=mRealm.copyFromRealm(element);
              
            }
        });

最後在銷燬Activity或Fragment時,要取消掉非同步任務

 @Override
    protected void onDestroy() {
        super.onDestroy();
        cats.removeChangeListeners();
  
    }

Demo是以本地收藏為應用場景的,實現了對Realm的增刪改查等操作,以及非同步的增刪改查操作,過程中也踩了不少的坑。


相關推薦

RealmAndroid應用

轉載:http://www.cnblogs.com/RaphetS/p/5996265.html 目錄 1、Realm簡介 2、環境配置 3、在Application中初始化Realm 4、建立實體 5、增刪改查 6、非同步操作 7、Demo地址(https://git

Android應用安裝分析

generate upgrade 版本 線程 title 回調函數 ebe children 應用商店 #1 安裝方式 1 安裝系統APK和預制APK時,通過PMS的構造函數中安裝,即第一次開機時安裝應用,沒有安裝界面。 2 網絡下載安裝,通過應用商店等,即調用Packa

Android應用TabLayout實現頂部Tab小選單加滑動效果

TabLayout控制元件是2015年google大會上,google釋出了新的Android Support Design庫,裡面包含了幾個新的控制元件,其中就有一個TabLayout,它就可以完成TabPageIndicator的效果,而且還是官方的,最好的是它可以相容

Android應用程式獲得系統簽名許可權

       在Android中呼叫系統SystemClock.setCurrentTimeMillis()函式,和使用Instrumentation類的時候會需要系統許可權。        需要在AndroidManifest.xml中manifest裡面新增

如果在android應用activemq

      由於andoid專案中有大量需要用到訊息推送,平臺端在實現訊息推送選擇activemq,為了能使用訊息推送,決定研究一下如果在android端實現activemq的訊息推送。        這個問題說難不難,說易也不易,平臺端開發人員選擇activemq並且認為

Android 如何從一個App啟動另外一個App(如啟動支付界面、啟動地圖界面、應用商場下載App等場景)

!= oid 等等 信息 fault next -a return 5.1   假定兩個App,分別是A和B,當A運行某個功能需要啟動B,一種是啟動B應用,一種直接進入B的某個Activity。搜了很多資料,沒有一個完整的。下面就A--Android5.1.1、B--And

Android應用程序自動安裝到手機內存及判斷應用程序是否安裝在SDCard

gen targe art 默認 使用 安裝 ack 在外 static 上次寫了個widget程序,安裝好後準備把它添加到手機主頁上,發現窗口小部件中找不到我剛剛安裝的那個widget,網上查資料才發現系統窗口小部件只能先加載掃描安裝在內存的應用程序,把屬於widget的

Android的ListView的應用

and xbm pyo evc isf f11 info won ogg 玖w7拍hx仲每62菲獻0http://shufang.docin.com/nprrm06928 O2屏zz稈Q扇棧綠2oeuhttp://tushu.docin.com/vsxx68589

Android沈浸式狀態欄的應用

ack get 實際應用 back ide real server 主界面 resources 在Android5.0版本後,谷歌公司為Android系統加入了很多新特性,刷新了Android用戶的體驗度。而其中的一個新特性就是沈浸式狀態欄。那麽問題來了,很多非移動端的小夥

android應用圖片保存到系統相冊並顯示

context sca new 場景 adc finally sep r+ ace 我應用到的場景是程序中在視頻通訊時截圖,將截圖保存到本地相冊中 /*** @param bmp 獲取的bitmap數據 * @param picName 自定義的圖片名*/ public

23種設計模式在Android應用

ets ros 而不是 auto 排隊 private eth mail 記錄 所有江湖偶遇,都是宿命相逢 ----《逆水寒》,只是覺得文案不錯,就用了。哈哈! 一.設計原則: 單一職責原則(SRP):任何一個對象都應給只有一個單獨的職責(“低耦合,高內聚”)裏氏替換原則(

淺談Android幸運快三平臺出租的meta-data及其應用

key 引用 平臺 name 如何 Coding pri sch xxxxx 在日常幸運快三平臺出租 haozbbs.com Q1446595067 的Android開發中,AndroidManifest中總會出現一些標簽,或是第三方SDK配置信息,或是系統配置,不禁讓人

Android中將應用裝到/system/app

分享一下我老師大神的人工智慧教程!零基礎,通俗易懂!http://blog.csdn.net/jiangjunshow 也歡迎大家轉載本篇文章。分享知識,造福人民,實現我們中華民族偉大復興!        

探究 lua 在 Android 應用

根據前面的文章 Android 與 Lua 可以大概瞭解 Lua 。在 Android(移動裝置)中,可以通過靈活載入 Lua 指令碼,使應用更加靈活,輕鬆面對多變的需求。luajava 在 jni 層主要實現了5個方法,藉助這5個方法lua幾乎可以使用所有的java類了。 Java 和 lua 互動 關

AndroidAPP應用冷啟動黑白屏原因 優化解決方案

冷啟動 前言 應用啟動 冷啟動流程 問題原因 解決方法 優化 前言 做過APP開發,尤其是複雜專案的同學應該會經歷過APP在桌面點選冷啟動的時候,你以為會順利開啟應用首頁,但是出現在你眼前的

關於android 獲取應用程式的包資訊

 Android系統為我們提供了很多服務管理的類,包括ActivityManager、PowerManager(電源管理)、AudioManager(音訊管理) 等。除此之外,還提供了一個PackageManger管理類,它的主要職責是管理應用程式包。 通過它,我們就可以獲取應用程

Android單例模式的應用

單例模式在我們日常開發中或多或少都會遇見,使用場景也是多種多樣。今天根據專案中需求,正好用到了單例模式。在此記錄一下。 一、跟隨app存活 有什麼方式,可以讓一個變數跟隨app存活呢?是一個單例模式維護的靜態工具類。靜態工具類程式碼一般都使用單例模式來實現,可以防止建立多

Android實現靜態的預設安裝和解除安裝應用

                最近好長時間都沒有寫blog了,主要是因為最近工作上的事以及下載Android原始碼的事耽誤的(下載原始碼這件事會在後續的blog中寫道,這個真的很有意義呀~~),那麼今天來寫點什麼呢?主要的靈感來自於早上看新聞看到一篇文章說有一款應用在後臺中解除安裝使用者手機中的所有瀏覽器的

Android應用圖示微技巧,8.0系統應用圖示的適配

大家好,2018年的第一篇文章到的稍微有點遲,也是因為在上一個Glide系列結束之後一直還沒想到什麼好的新題材。 現在已經進入了2018年,Android 8.0系統也逐漸開始普及起來了。三星今年推出的最新旗艦機Galaxy S9已經搭載了Android 8.0系統,緊

androidCheckBox的高階應用之button屬性

在我們普通的應用中checkBox裡有多個Item時,頁面中會自動為我們提供複選框的按鈕,但我們有時會遇到這要的情況,CheckBox應用的物件不需要多個Item而是每個CheckBox對應平行的每一個選項,而且要求作出按鈕(buuton)有CSS屬性改變的效果,此時就必須