跨平台:現在很多應用都是要兼顧iOS和Android兩個平台同時開發。如果兩個平台都能使用相同的資料庫,那就不用考慮内部資料的架構不同,使用Realm提供的API,可以使資料持久化層在兩個平台上無差異化的轉換。
簡單易用:Core Data 和 SQLite 備援、繁雜的知識和代碼足以吓退絕大多數剛入門的開發者,而換用 Realm,則可以極大地減少學習成本,立即學會本地化存儲的方法。毫不吹噓的說,把官方最新文檔完整看一遍,就完全可以上手開發了。
可視化:Realm 還提供了一個輕量級的資料庫檢視工具,在Mac Appstore 可以下載下傳“Realm Browser”這個工具,開發者可以檢視資料庫當中的内容,執行簡單的插入和删除資料的操作。畢竟,很多時候,開發者使用資料庫的理由是因為要提供一些所謂的“知識庫”。

“Realm Browser”這個工具調試起Realm資料庫實在太好用了,強烈推薦。
如果使用模拟器進行調試,可以通過
[RLMRealmConfiguration defaultConfiguration].fileURL
列印出Realm 資料庫位址,然後在Finder中⌘⇧G跳轉到對應路徑下,用Realm Browser打開對應的.realm檔案就可以看到資料啦.
如果是使用真機調試的話“Xcode->Window->Devices(⌘⇧2)”,然後找到對應的裝置與項目,點選Download Container,導出xcappdata檔案後,顯示包内容,進到AppData->Documents,使用Realm Browser打開.realm檔案即可.
自2012年起, Realm 就已經開始被用于正式的商業産品中了。經過4年的使用,逐漸趨于穩定。
一. Realm 安裝
使用 Realm 建構應用的基本要求:
iOS 7 及其以上版本, macOS 10.9 及其以上版本,此外 Realm 支援 tvOS 和 watchOS 的所有版本。
需要使用 Xcode 7.3 或者以後的版本。
注意 這裡如果是純的OC項目,就安裝OC的Realm,如果是純的Swift項目,就安裝Swift的Realm。如果是混編項目,就需要安裝OC的Realm,然後要把
Swift/RLMSupport.swift檔案一同編譯進去。
RLMSupport.swift這個檔案為 Objective-C 版本的 Realm 集合類型中引入了 Sequence 一緻性,并且重新暴露了一些不能夠從 Swift 中進行原生通路的 Objective-C 方法,例如可變參數 (variadic arguments)。更加詳細的說明見
官方文檔。
安裝方法就4種:
一. Dynamic Framework
注意:動态架構與 iOS 7 不相容,要支援 iOS 7 的話請檢視“靜态架構”。
下載下傳
最新的Realm發行版本,并解壓;
前往Xcode 工程的”General”設定項中,從ios/dynamic/、osx/、tvos/或者watchos/中将’Realm.framework’拖曳到”Embedded Binaries”選項中。确認Copy items if needed被選中後,點選Finish按鈕;
在單元測試 Target 的”Build Settings”中,在”Framework Search Paths”中添加Realm.framework的上級目錄;
如果希望使用 Swift 加載 Realm,請拖動Swift/RLMSupport.swift檔案到 Xcode 工程的檔案導航欄中并選中Copy items if needed;
如果在 iOS、watchOS 或者 tvOS 項目中使用 Realm,請在您應用目标的”Build Phases”中,建立一個新的”Run Script Phase”,并将
bash "${BUILT_PRODUCTS_DIR}/${FRAMEWORKS_FOLDER_PATH}/Realm.framework/strip-frameworks.sh"
這條腳本複制到文本框中。 因為要繞過
APP商店送出的bug,這一步在打包通用裝置的二進制釋出版本時是必須的。
二.CocoaPods
在項目的Podfile中,添加pod 'Realm',在終端運作pod install。
三.Carthage
1.在Carthage 中添加github "realm/realm-cocoa",運作carthage update。為了修改用以建構項目的 Swift toolchain,通過--toolchain參數來指定合适的 toolchain。--no-use-binaries參數也是必需的,這可以避免 Carthage 将預建構的 Swift 3.0 二進制包下載下傳下來。 例如:
carthage update --toolchain com.apple.dt.toolchain.Swift_2_3 --no-use-binaries
2.從 Carthage/Build/目錄下對應平台檔案夾中,将 Realm.framework拖曳到您 Xcode 工程”General”設定項的”Linked Frameworks and Libraries”頁籤中;
3.iOS/tvOS/watchOS: 在您應用目标的“Build Phases”設定頁籤中,點選“+”按鈕并選擇“New Run Script Phase”。在建立的Run Script中,填寫:
/usr/local/bin/carthage copy-frameworks
在“Input Files”内添加您想要使用的架構路徑,例如:
$(SRCROOT)/Carthage/Build/iOS/Realm.framework
因為要繞過
四.Static Framework (iOS only) Realm 的最新版本并解壓,将 Realm.framework 從 ios/static/檔案夾拖曳到您 Xcode 項目中的檔案導航器當中。確定 Copy items if needed 選中然後單擊 Finish;
在 Xcode 檔案導航器中選擇您的項目,然後選擇您的應用目标,進入到** Build Phases** 頁籤中。在 Link Binary with Libraries 中單擊 + 号然後添加libc++.dylib;
二. Realm 中的相關術語
為了能更好的了解Realm的使用,先介紹一下涉及到的相關術語。
RLMRealm:Realm是架構的核心所在,是我們建構資料庫的通路點,就如同Core Data的管理對象上下文(managed object context)一樣。出于簡單起見,realm提供了一個預設的defaultRealm( )的便利構造器方法。
RLMObject:這是我們自定義的Realm資料模型。建立資料模型的行為對應的就是資料庫的結構。要建立一個資料模型,我們隻需要繼承RLMObject,然後設計我們想要存儲的屬性即可。
關系(Relationships):通過簡單地在資料模型中聲明一個RLMObject類型的屬性,我們就可以建立一個“一對多”的對象關系。同樣地,我們還可以建立“多對一”和“多對多”的關系。
寫操作事務(Write Transactions):資料庫中的所有操作,比如建立、編輯,或者删除對象,都必須在事務中完成。“事務”是指位于write閉包内的代碼段。
查詢(Queries):要在資料庫中檢索資訊,我們需要用到“檢索”操作。檢索最簡單的形式是對Realm( )資料庫發送查詢消息。如果需要檢索更複雜的資料,那麼還可以使用斷言(predicates)、複合查詢以及結果排序等等操作。
RLMResults:這個類是執行任何查詢請求後所傳回的類,其中包含了一系列的RLMObject對象。RLMResults和NSArray類似,我們可以用下智語法來對其進行通路,并且還可以決定它們之間的關系。不僅如此,它還擁有許多更強大的功能,包括排序、查找等等操作。
三.Realm 入門——如何使用
由于Realm的API極為友好,一看就懂,是以這裡就按照平時開發的順序,把需要用到的都梳理一遍。
- 建立資料庫
- (void)creatDataBaseWithName:(NSString *)databaseName{ NSArray *docPath = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES); NSString *path = [docPath objectAtIndex:0]; NSString *filePath = [path stringByAppendingPathComponent:databaseName]; NSLog(@"資料庫目錄 = %@",filePath); RLMRealmConfiguration *config = [RLMRealmConfiguration defaultConfiguration]; config.fileURL = [NSURL URLWithString:filePath]; config.objectClasses = @[MyClass.class, MyOtherClass.class]; config.readOnly = NO; int currentVersion = 1.0; config.schemaVersion = currentVersion; config.migrationBlock = ^(RLMMigration *migration , uint64_t oldSchemaVersion) { // 這裡是設定資料遷移的block if (oldSchemaVersion < currentVersion) { } }; [RLMRealmConfiguration setDefaultConfiguration:config];}
建立資料庫主要設定RLMRealmConfiguration,設定資料庫名字和存儲地方。把路徑以及資料庫名字拼接好字元串,指派給fileURL即可。
objectClasses這個屬性是用來控制對哪個類能夠存儲在指定 Realm 資料庫中做出限制。例如,如果有兩個團隊分别負責開發您應用中的不同部分,并且同時在應用内部使用了 Realm 資料庫,那麼您肯定不希望為它們協調進行資料遷移您可以通過設定RLMRealmConfiguration的 objectClasses屬性來對類做出限制。objectClasses一般可以不用設定。
readOnly是控制是否隻讀屬性。
還有一個很特殊的資料庫,記憶體資料庫。
通常情況下,Realm 資料庫是存儲在硬碟中的,但是您能夠通過設定inMemoryIdentifier而不是設定RLMRealmConfiguration中的 fileURL屬性,以建立一個完全在記憶體中運作的資料庫。
RLMRealmConfiguration *config = [RLMRealmConfiguration defaultConfiguration];
config.inMemoryIdentifier = @"MyInMemoryRealm";
RLMRealm *realm = [RLMRealm realmWithConfiguration:config error:nil];
記憶體資料庫在每次程式運作期間都不會儲存資料。但是,這不會妨礙到 Realm 的其他功能,包括查詢、關系以及線程安全。
如果需要一種靈活的資料讀寫但又不想儲存資料的方式的話,那麼可以選擇用記憶體資料庫。(關于記憶體資料庫的性能 和 類屬性的 性能,還沒有測試過,感覺性能不會有太大的差異,是以記憶體資料庫使用場景感覺不多)
使用記憶體資料庫需要注意的是:
記憶體資料庫會在臨時檔案夾中建立多個檔案,用來協調處理諸如跨程序通知之類的事務。 實際上沒有任何的資料會被寫入到這些檔案當中,除非作業系統由于記憶體過滿, 需要清除磁盤上的多餘空間。 才會去把記憶體裡面的資料存入到檔案中。(感謝 @酷酷的哀殿 指出)
如果某個記憶體 Realm 資料庫執行個體沒有被引用,那麼所有的資料就會被釋放。是以必須要在應用的生命周期内保持對Realm記憶體資料庫的強引用,以避免資料丢失。
-
建表
Realm資料模型是基于标準 Objective‑C 類來進行定義的,使用屬性來完成模型的具體定義。
我們隻需要繼承 RLMObject或者一個已經存在的模型類,您就可以建立一個新的 Realm 資料模型對象。對應在資料庫裡面就是一張表。
#import <Realm/Realm.h> @interface RLMUser : RLMObject @property NSString *accid; //使用者注冊id @property NSInteger custId; //姓名 @property NSString *custName; //頭像大圖url @property NSString *avatarBig; @property RLMArray<Car> *cars; RLM_ARRAY_TYPE(RLMUser) // 定義RLMArray<RLMUser> @interface Car : RLMObject @property NSString *carName; @property RLMUser *owner;@endRLM_ARRAY_TYPE(Car) // 定義RLMArray<Car> @end
注意RLMObject 官方建議不要加上 Objective-C的property attributes(如nonatomic, atomic, strong, copy, weak 等等)假如設定了,這些attributes會一直生效直到RLMObject被寫入realm資料庫。
RLM_ARRAY_TYPE宏建立了一個協定,進而允許 RLMArray<Car>文法的使用。如果該宏沒有放置在模型接口的底部的話,您或許需要提前聲明該模型類。
關于RLMObject的的關系
1.對一(To-One)關系
對于多對一(many-to-one)或者一對一(one-to-one)關系來說,隻需要聲明一個RLMObject子類類型的屬性即可,如上面代碼例子,@property RLMUser *owner;
2.對多(To-Many)關系通過 RLMArray類型的屬性您可以定義一個對多關系。如上面代碼例子,@property RLMArray<Car> *cars;
3.反向關系(Inverse Relationship)
連結是單向性的。是以,如果對多關系屬性 RLMUser.cars連結了一個 Car執行個體,而這個執行個體的對一關系屬性 Car.owner又連結到了對應的這個 RLMUser執行個體,那麼實際上這些連結仍然是互相獨立的。
@interface Car : RLMObject
@property NSString *carName;
@property (readonly) RLMLinkingObjects *owners;
@end
@implementation Car
+ (NSDictionary *)linkingObjectsProperties {
return
@{
@"owners": [RLMPropertyDescriptor descriptorWithClass:RLMUser.class propertyName:@"cars"];
};
}
@end
這裡可以類比Core Data裡面xcdatamodel檔案裡面那些“箭頭”
@implementation Book
// 主鍵
+ (NSString *)primaryKey { return @"ID";}
//設定屬性預設值
+ (NSDictionary *)defaultPropertyValues{ return @{@"carName":@"測試" };}
//設定忽略屬性,即不存到realm資料庫中
+ (NSArray<NSString *> *)ignoredProperties { return @[@"ID"];}
//一般來說,屬性為nil的話realm會抛出異常,但是如果實作了這個方法的話,就隻有name為nil會抛出異常,也就是說現在cover屬性可以為空了
+ (NSArray *)requiredProperties { return @[@"name"];}
//設定索引,可以加快檢索的速度
+ (NSArray *)indexedProperties { return @[@"ID"];
}
@end
還可以給RLMObject設定主鍵primaryKey,預設值defaultPropertyValues,忽略的屬性ignoredProperties,必要屬性requiredProperties,索引indexedProperties。比較有用的是主鍵和索引。
3.存儲資料
建立對象
// (1) 建立一個Car對象,然後設定其屬性
Car *car = [[Car alloc] init];
car.carName = @"Lamborghini";
// (2) 通過字典建立Car對象
Car *myOtherCar = [[Car alloc] initWithValue:@{@"name" : @"Rolls-Royce"}];
// (3) 通過數組建立狗狗對象
Car *myThirdcar = [[Car alloc] initWithValue:@[@"BMW"]];`
注意,所有的必需屬性都必須在對象添加到 Realm 前被指派
4.增
![]()
Realm資料庫的掌握
[realm beginWriteTransaction];
[realm addObject:Car];
[realm commitWriteTransaction];
請注意,如果在程序中存在多個寫入操作的話,那麼單個寫入操作将會阻塞其餘的寫入操作,并且還會鎖定該操作所在的目前線程。
Realm這個特性與其他持久化解決方案類似,我們建議您使用該方案正常的最佳做法:将寫入操作轉移到一個獨立的線程中執行。
官方給出了一個建議:
由于 Realm 采用了 MVCC 設計架構,讀取操作并不會因為寫入事務正在進行而受到影響。除非您需要立即使用多個線程來同時執行寫入操作,不然您應當采用批量化的寫入事務,而不是采用多次少量的寫入事務。
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
RLMRealm *realm = [RLMRealm defaultRealm];
[realm transactionWithBlock:^{
[realm addObject: Car];
}];
});`
上面的代碼就是把寫事務放到子線程中去處理。
5.删
![]()
Realm資料庫的掌握 [realm beginWriteTransaction];// 删除單條記錄
[realm deleteObject:Car];// 删除多條記錄
[realm deleteObjects:CarResult];// 删除所有記錄
[realm deleteAllObjects];
[realm commitWriteTransaction];
6.改當沒有主鍵的情況下,需要先查詢,再修改資料。當有主鍵的情況下,有以下幾個非常好用的API![]()
Realm資料庫的掌握
[realm addOrUpdateObject:Car];
[Car createOrUpdateInRealm:realm withValue:@{@"id": @1, @"price": @9000.0f}];
addOrUpdateObject會去先查找有沒有傳入的Car相同的主鍵,如果有,就更新該條資料。這裡需要注意,addOrUpdateObject這個方法不是增量更新,所有的值都必須有,如果有哪幾個值是null,那麼就會覆寫原來已經有的值,這樣就會出現資料丢失的問題。
createOrUpdateInRealm:withValue:這個方法是增量更新的,後面傳一個字典,使用這個方法的前提是有主鍵。方法會先去主鍵裡面找有沒有字典裡面傳入的主鍵的記錄,如果有,就隻更新字典裡面的子集。如果沒有,就建立一條記錄。
7.查
在Realm中所有的查詢(包括查詢和屬性通路)在 Realm 中都是延遲加載的,隻有當屬性被通路時,才能夠讀取相應的資料。
查詢結果并不是資料的拷貝:修改查詢結果(在寫入事務中)會直接修改硬碟上的資料。同樣地,您可以直接通過包含在RLMResults中的RLMObject對象完成周遊關系圖的操作。除非查詢結果被使用,否則檢索的執行将會被推遲。這意味着連結幾個不同的臨時 {RLMResults} 來進行排序和比對資料,不會執行額外的工作,例如進行中間狀态。一旦檢索執行之後,或者通知子產品被添加之後, RLMResults将随時保持更新,接收 Realm 中,在背景線程上執行的檢索操作中可能所做的更改。
//從預設資料庫查詢所有的車
RLMResults<Car *> *cars = [Car allObjects];
// 使用斷言字元串查詢
RLMResults<Dog *> *tanDogs = [Dog objectsWhere:@"color = '棕黃色' AND name BEGINSWITH '大'"];
// 使用 NSPredicate 查詢
NSPredicate *pred = [NSPredicate predicateWithFormat:@"color = %@ AND name BEGINSWITH %@", @"棕黃色", @"大"];
RLMResults *results = [Dog objectsWithPredicate:pred];
// 排序名字以“大”開頭的棕黃色狗狗
RLMResults<Dog *> *sortedDogs = [[Dog objectsWhere:@"color = '棕黃色' AND name BEGINSWITH '大'"] sortedResultsUsingProperty:@"name" ascending:YES];
Realm還能支援鍊式查詢
Realm 查詢引擎一個特性就是它能夠通過非常小的事務開銷來執行鍊式查詢(chain queries),而不需要像傳統資料庫那樣為每個成功的查詢建立一個不同的資料庫伺服器通路。
RLMResults<Car *> *Cars = [Car objectsWhere:@"color = blue"];
RLMResults<Car *> *CarsWithBNames = [Cars objectsWhere:@"name BEGINSWITH 'B'"];`
8.其他相關特性
1.支援KVC和KVO
RLMObject、RLMResult以及 RLMArray都遵守鍵值編碼(Key-Value Coding)(KVC)機制。當您在運作時才能決定哪個屬性需要更新的時候,這個方法是最有用的。将 KVC 應用在集合當中是大量更新對象的極佳方式,這樣就可以不用經常周遊集合,為每個項目建立一個通路器了。
RLMResults<Person *> *persons = [Person allObjects];
[[RLMRealm defaultRealm] transactionWithBlock:^{
[[persons firstObject] setValue:@YES forKeyPath:@"isFirst"];
// 将每個人的 planet 屬性設定為“地球”
[persons setValue:@"地球" forKeyPath:@"planet"];}];
Realm 對象的大多數屬性都遵從 KVO 機制。所有 RLMObject子類的持久化(persisted)存儲(未被忽略)的屬性都是遵循 KVO 機制的,并且 RLMObject以及 RLMArray中 無效的(invalidated)屬性也同樣遵循(然而 RLMLinkingObjects屬性并不能使用 KVO 進行觀察)。
2.支援資料庫加密
// 産生随機密鑰
NSMutableData *key = [NSMutableData dataWithLength:64];
SecRandomCopyBytes(kSecRandomDefault, key.length, (uint8_t *)key.mutableBytes);
// 打開加密檔案
RLMRealmConfiguration *config = [RLMRealmConfiguration defaultConfiguration];
config.encryptionKey = key;
NSError *error = nil;
RLMRealm *realm = [RLMRealm realmWithConfiguration:config error:&error];
if (!realm) {
// 如果密鑰錯誤,`error` 會提示資料庫不可通路
NSLog(@"Error opening realm: %@", error);}
Realm 支援在建立 Realm 資料庫時采用64位的密鑰對資料庫檔案進行 AES-256+SHA2 加密。這樣硬碟上的資料都能都采用AES-256來進行加密和解密,并用 SHA-2 HMAC 來進行驗證。每次您要擷取一個 Realm 執行個體時,您都需要提供一次相同的密鑰。
不過,加密過的 Realm 隻會帶來很少的額外資源占用(通常最多隻會比平常慢10%)。
3.通知
// 擷取 Realm 通知
token = [realm addNotificationBlock:^(NSString *notification,RLMRealm * realm) { [myViewController updateUI];}];
[token stop];
// 移除通知
[realm removeNotification:self.token];
Realm 執行個體将會在每次寫入事務送出後,給其他線程上的 Realm 執行個體發送通知。一般控制器如果想一直持有這個通知,就需要申請一個屬性,strong持有這個通知。
- (void)viewDidLoad { [super viewDidLoad];
// 觀察 RLMResults 通知
__weak typeof(self) weakSelf = self;
self.notificationToken = [[Person objectsWhere:@"age > 5"]
addNotificationBlock:^(RLMResults<Person *> *results, RLMCollectionChange *change, NSError *error) {
if (error) {
NSLog(@"Failed to open Realm on background worker: %@", error); return; }
UITableView *tableView = weakSelf.tableView;
// 對于變化資訊來說,檢索的初次運作将會傳遞 nil
if (!changes) {
[tableView reloadData];
return;
}
// 檢索結果被改變,是以将它們應用到 UITableView 當中
[tableView beginUpdates];
[tableView deleteRowsAtIndexPaths:[changes deletionsInSection:0]
withRowAnimation:UITableViewRowAnimationAutomatic];
[tableView insertRowsAtIndexPaths:[changes insertionsInSection:0]
withRowAnimation:UITableViewRowAnimationAutomatic];
[tableView reloadRowsAtIndexPaths:[changes modificationsInSection:0]
withRowAnimation:UITableViewRowAnimationAutomatic];
[tableView endUpdates];
}];
}
我們還能進行更加細粒度的通知,用集合通知就可以做到。
集合通知是異步觸發的,首先它會在初始結果出現的時候觸發,随後當某個寫入事務改變了集合中的所有或者某個對象的時候,通知都會再次觸發。這些變化可以通過傳遞到通知閉包當的 RLMCollectionChange參數通路到。這個對象當中包含了受 deletions、insertions和 modifications 狀态所影響的索引資訊。
集合通知對于 RLMResults、RLMArray、RLMLinkingObjects 以及 RLMResults 這些衍生出來的集合來說,當關系中的對象被添加或者删除的時候,一樣也會觸發這個狀态變化。
4.資料庫遷移
這是Realm的優點之一,友善遷移。
對比Core Data的資料遷移,實在是友善太多了。關于iOS Core Data 資料遷移 指南請看這篇
文章資料庫存儲方面的增删改查應該都沒有什麼大問題,比較蛋疼的應該就是資料遷移了。在版本疊代過程中,很可能會發生表的新增,删除,或者表結構的變化,如果新版本中不做資料遷移,使用者更新到新版,很可能就直接crash了。對比Core Data的資料遷移比較複雜,Realm的遷移實在太簡單了。
1.新增删除表,Realm不需要做遷移2.新增删除字段,Realm不需要做遷移。Realm 會自行檢測新增和需要移除的屬性,然後自動更新硬碟上的資料庫架構。
舉個官方給的資料遷移的例子:
RLMRealmConfiguration *config = [RLMRealmConfiguration defaultConfiguration];
config.schemaVersion = 2;
config.migrationBlock = ^(RLMMigration *migration, uint64_t oldSchemaVersion){
// enumerateObjects:block: 周遊了存儲在 Realm 檔案中的每一個“Person”對象
[migration enumerateObjects:Person.className block:^(RLMObject *oldObject, RLMObject *newObject) {
// 隻有當 Realm 資料庫的架構版本為 0 的時候,才添加 “fullName” 屬性
if (oldSchemaVersion < 1) {
newObject[@"fullName"] = [NSString stringWithFormat:
@"%@ %@", oldObject[@"firstName"], oldObject[@"lastName"]];
}
// 隻有當 Realm 資料庫的架構版本為 0 或者 1 的時候,才添加“email”屬性
if (oldSchemaVersion < 2) {
newObject[@"email"] = @"";
}
// 替換屬性名
if (oldSchemaVersion < 3) {
// 重命名操作應該在調用 `enumerateObjects:` 之外完成
[migration renamePropertyForClass:Person.className oldName:@"yearsSinceBirth" newName:@"age"];
} }];
};
[RLMRealmConfiguration setDefaultConfiguration:config];
// 現在我們已經成功更新了架構版本并且提供了遷移閉包,打開舊有的 Realm 資料庫會自動執行此資料遷移,然後成功進行通路
[RLMRealm defaultRealm];
在block裡面分别有3種遷移方式,第一種是合并字段的例子,第二種是增加新字段的例子,第三種是原字段重命名的例子。
四. Realm 使用中可能需要注意的一些問題
在我從0開始接觸Realm到熟練上手,基本就遇到了多線程這一個坑。可見Realm的API文檔是多麼的友好。雖然坑不多,但是還有有些需要注意的地方。
1.跨線程通路資料庫,Realm對象一定需要建立一個
*** Terminating app due to uncaught exception 'RLMException', reason: 'Realm accessed from incorrect thread.`
******* First throw call stack:
****(**** 0 CoreFoundation 0x000000011479f34b __exceptionPreprocess + 171
**** 1 libobjc.A.dylib 0x00000001164a321e objc_exception_throw + 48
**** 2 BHFangChuang 0x000000010dd4c2b5 -[RLMRealm beginWriteTransaction] + 77
**** 3 BHFangChuang 0x000000010dd4c377 -[RLMRealm transactionWithBlock:error:] + 45
**** 4 BHFangChuang 0x000000010dd4c348 -[RLMRealm transactionWithBlock:] + 19
**** 5 BHFangChuang 0x000000010d51d7ae __71-[RealmDataBaseHelper updateUserWithLoginDate:andLogoutDate:according:]_block_invoke + 190
**** 6 libdispatch.dylib 0x00000001180ef980 _dispatch_call_block_and_release + 12
**** 7 libdispatch.dylib 0x00000001181190cd _dispatch_client_callout + 8
**** 8 libdispatch.dylib 0x00000001180f8366 _dispatch_queue_override_invoke + 1426
**** 9 libdispatch.dylib 0x00000001180fa3b7 _dispatch_root_queue_drain + 720
**** 10 libdispatch.dylib 0x00000001180fa08b _dispatch_worker_thread3 + 123
**** 11 libsystem_pthread.dylib 0x00000001184c8746 _pthread_wqthread + 1299
**** 12 libsystem_pthread.dylib 0x00000001184c8221 start_wqthread +
13****)****libc++abi.dylib: terminating with uncaught exception of type NSException**
如果程式崩潰了,出現以上錯誤,那就是因為你通路Realm資料的時候,使用的Realm對象所在的線程和目前線程不一緻。
解決辦法就是在目前線程重新擷取最新的Realm,即可。
-
自己封裝一個Realm全局執行個體單例是沒啥作用的
這個也是我之前對Realm多線程了解不清,導緻的一個誤解。
很多開發者應該都會對Core Data和Sqlite3或者FMDB,自己封裝一個類似Helper的單例。于是我也在這裡封裝了一個單例,在建立完Realm資料庫的時候strong持有一個Realm的對象。然後之後的通路中隻需要讀取這個單例持有的Realm對象就可以拿到資料庫了。
想法是好的,但是同一個Realm對象是不支援跨線程操作realm資料庫的。
Realm 通過確定每個線程始終擁有 Realm 的一個快照,以便讓并發運作變得十分輕松。你可以同時有任意數目的線程通路同一個 Realm 檔案,并且由于每個線程都有對應的快照,是以線程之間絕不會産生影響。需要注意的一件事情就是不能讓多個線程都持有同一個 Realm 對象的 執行個體 。如果多個線程需要通路同一個對象,那麼它們分别會擷取自己所需要的執行個體(否則在一個線程上發生的更改就會造成其他線程得到不完整或者不一緻的資料)。
其實RLMRealm *realm = [RLMRealm defaultRealm]; 這句話就是擷取了目前realm對象的一個執行個體,其實實作就是拿到單例。是以我們每次在子線程裡面不要再去讀取我們自己封裝持有的realm執行個體了,直接調用系統的這個方法即可,能保證通路不出錯。
3.transactionWithBlock 已經處于一個寫的事務中,事務之間不能嵌套
[realm transactionWithBlock:^{ [self.realm beginWriteTransaction]; [self convertToRLMUserWith:bhUser To:[self convertToRLMUserWith:bhUser To:nil]]; [self.realm commitWriteTransaction]; }];
transactionWithBlock 已經處于一個寫的事務中,如果還在block裡面再寫一個commitWriteTransaction,就會出錯,寫事務是不能嵌套的。
出錯資訊如下:
*** Terminating app due to uncaught exception 'RLMException', reason: 'The Realm is already in a write transaction'
******* First throw call stack:****(**** 0 CoreFoundation 0x0000000112e2d34b __exceptionPreprocess + 171
**** 1 libobjc.A.dylib 0x0000000114b3121e objc_exception_throw + 48
**** 2 BHFangChuang 0x000000010c4702b5 -[RLMRealm beginWriteTransaction] + 77
**** 3 BHFangChuang 0x000000010bc4175a __71-[RealmDataBaseHelper updateUserWithLoginDate:andLogoutDate:according:]_block_invoke_2 + 42
**** 4 BHFangChuang 0x000000010c470380 -[RLMRealm transactionWithBlock:error:] + 54
**** 5 BHFangChuang 0x000000010c470348 -[RLMRealm transactionWithBlock:] + 19
**** 6 BHFangChuang 0x000000010bc416d7 __71-[RealmDataBaseHelper updateUserWithLoginDate:andLogoutDate:according:]_block_invoke + 231
**** 7 libdispatch.dylib 0x0000000116819980 _dispatch_call_block_and_release + 12
**** 8 libdispatch.dylib 0x00000001168430cd _dispatch_client_callout + 8
**** 9 libdispatch.dylib 0x0000000116822366 _dispatch_queue_override_invoke + 1426
**** 10 libdispatch.dylib 0x00000001168243b7 _dispatch_root_queue_drain + 720
**** 11 libdispatch.dylib 0x000000011682408b _dispatch_worker_thread3 + 123
**** 12 libsystem_pthread.dylib 0x0000000116bed746 _pthread_wqthread + 1299
**** 13 libsystem_pthread.dylib 0x0000000116bed221 start_wqthread + 13****)****libc++abi.dylib: terminating with uncaught exception of type NSException**
4.建議每個model都需要設定主鍵,這樣可以友善add和update
如果能設定主鍵,請盡量設定主鍵,因為這樣友善我們更新資料,我們可以很友善的調用addOrUpdateObject: 或者 createOrUpdateInRealm:withValue:方法進行更新。這樣就不需要先根據主鍵,查詢出資料,然後再去更新。有了主鍵以後,這兩步操作可以一步完成。
5.查詢也不能跨線程查詢
RLMResults * results = [self selectUserWithAccid:bhUser.accid];
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
RLMRealm *realm = [RLMRealm defaultRealm];
[realm transactionWithBlock:^{
[realm addOrUpdateObject:results[0]];
}];
});
由于查詢是在子線程外查詢的,是以跨線程也會出錯,出錯資訊如下:
***** Terminating app due to uncaught exception 'RLMException', reason: 'Realm accessed from incorrect thread'******* First throw call stack:****(**** 0 CoreFoundation 0x000000011517a34b __exceptionPreprocess + 171
**** 1 libobjc.A.dylib 0x0000000116e7e21e objc_exception_throw + 48
**** 2 BHFangChuang 0x000000010e7c34ab _ZL10throwErrorP8NSString + 129
**** 3 BHFangChuang 0x000000010e7c177f -[RLMResults count] + 40
**** 4 BHFangChuang 0x000000010df8f3bf -[RealmDataBaseHelper convertToRLMUserWith:LoginDate:LogoutDate:To:] + 159
**** 5 BHFangChuang 0x000000010df8efc1 __71-[RealmDataBaseHelper updateUserWithLoginDate:andLogoutDate:according:]_block_invoke_2 + 81
**** 6 BHFangChuang 0x000000010e7bd320 -[RLMRealm transactionWithBlock:error:] + 54
**** 7 BHFangChuang 0x000000010e7bd2e8 -[RLMRealm transactionWithBlock:] + 19
**** 8 BHFangChuang 0x000000010df8eecf __71-[RealmDataBaseHelper updateUserWithLoginDate:andLogoutDate:according:]_block_invoke + 351
**** 9 libdispatch.dylib 0x0000000118b63980 _dispatch_call_block_and_release + 12
**** 10 libdispatch.dylib 0x0000000118b8d0cd _dispatch_client_callout + 8
**** 11 libdispatch.dylib 0x0000000118b6c366 _dispatch_queue_override_invoke + 1426
**** 12 libdispatch.dylib 0x0000000118b6e3b7 _dispatch_root_queue_drain + 720
**** 13 libdispatch.dylib 0x0000000118b6e08b _dispatch_worker_thread3 + 123
**** 14 libsystem_pthread.dylib 0x0000000118f3c746 _pthread_wqthread + 1299
**** 15 libsystem_pthread.dylib 0x0000000118f3c221 start_wqthread + 13****)****libc++abi.dylib: terminating with uncaught exception of type **
五. Realm “放棄”——優點和缺點
關于Realm的優點,在官網上也說了很多了,我感觸最深的3個優點也在文章開頭提到了。
CoreData VS Realm 的對比,可以看看
這篇文章說到使用 Realm最後的二道門檻,一是如何從其他資料庫遷移到Realm,二是Realm資料庫的一些限制。
接下來請還在考慮是否使用Realm的同學仔細看清楚,下面是你需要權衡是否要換到Realm資料庫的重要标準。(以下描述基于Realm最新版 2.0.2)
1.從其他資料庫遷移到Realm
如果從其他資料庫遷移到Realm,請看我之前寫過的一篇
,簡單的提一下蛋疼的問題,由于切換了資料庫,需要在未來幾個版本都必須維護2套資料庫,因為老使用者的資料需要慢慢從老資料庫遷移到Realm,這個有點蛋疼。遷移資料的那段代碼需要“惡心”的存在工程裡。但是一旦都遷移完成,之後的路就比較平坦了。
關于Core Data遷移過來沒有fetchedResultController的問題,這裡提一下。由于使用Realm的話就無法使用Core Data的fetchedResultController,那麼如果資料庫更新了資料,是不是隻能通過reloadData來更新tableview了?目前基本上是的,Realm提供了我們通知機制,目前的Realm支援給realm資料庫對象添加通知,這樣就可以在資料庫寫入事務送出後擷取到,進而更新UI;詳情可以參考
https://realm.io/cn/docs/swift/latest/#notification當然如果仍希望使用NSFetchedResultsController的話,那麼推薦使用RBQFetchedResultsController,這是一個替代品,位址是:
https://github.com/Roobiq/RBQFetchedResultsController目前Realm計劃在未來實作類似的效果,具體您可以參見這個PR:
http://github.com/realm/realm-cocoa/issues/687當然,如果是新的App,還在開發中,可以考慮直接使用Realm,會更爽。
以上是第一道門檻,如果覺得遷移帶來的代價還能承受,那麼恭喜你,已經踏入Realm一半了。那麼還請看第二道“門檻”。
-
Realm資料庫目前版本的限制
把使用者一部分攔在Realm門口的還在這第二道坎,因為這些限制,這些“缺點”,導緻App的業務無法使用Realm得到滿足,是以最終放棄了Realm。當然,這些問題,有些是可以靈活通過改變表結構解決的,畢竟人是活的(如果真的想用Realm,想些辦法,誰也攔不住)
Realm資料庫的掌握 1.類名稱的長度最大隻能存儲 57 個 UTF8 字元。
2.屬性名稱的長度最大隻能支援 63 個 UTF8 字元。
3.NSData以及 NSString屬性不能儲存超過 16 MB 大小的資料。如果要存儲大量的資料,可通過将其分解為16MB 大小的塊,或者直接存儲在檔案系統中,然後将檔案路徑存儲在 Realm 中。如果您的應用試圖存儲一個大于 16MB 的單一屬性,系統将在運作時抛出異常。
4.對字元串進行排序以及不區分大小寫查詢隻支援“基礎拉丁字元集”、“拉丁字元補充集”、“拉丁文擴充字元集 A” 以及”拉丁文擴充字元集 B“(UTF-8 的範圍在 0~591 之間)。
5.盡管 Realm 檔案可以被多個線程同時通路,但是您不能跨線程處理 Realms、Realm 對象、查詢和查詢結果。(這個其實也不算是個問題,我們在多線程中建立新的Realm對象就可以解決)
6.Realm對象的 Setters & Getters 不能被重載
因為 Realm 在底層資料庫中重寫了 setters 和 getters 方法,是以您不可以在您的對象上再對其進行重寫。一個簡單的替代方法就是:建立一個新的 Realm 忽略屬性,該屬性的通路起可以被重寫, 并且可以調用其他的 getter 和 setter 方法。
7.檔案大小 & 版本跟蹤
一般來說 Realm 資料庫比 SQLite 資料庫在硬碟上占用的空間更少。如果您的 Realm 檔案大小超出了您的想象,這可能是因為您資料庫中的 RLMRealm中包含了舊版本資料。為了使您的資料有相同的顯示方式,Realm 隻在循環疊代開始的時候才更新資料版本。這意味着,如果您從 Realm 讀取了一些資料并進行了在一個鎖定的線程中進行長時間的運作,然後在其他線程進行讀寫 Realm 資料庫的話,那麼版本将不會被更新,Realm 将儲存中間版本的資料,但是這些資料已經沒有用了,這導緻了檔案大小的增長。這部分空間會在下次寫入操作時被重複利用。這些操作可以通過調用writeCopyToPath:error:來實作。
解決辦法:通過調用invalidate,來告訴 Realm 您不再需要那些拷貝到 Realm 的資料了。這可以使我們不必跟蹤這些對象的中間版本。在下次出現新版本時,再進行版本更新。您可能在 Realm 使用Grand Central Dispatch時也發現了這個問題。在 dispatch 結束後自動釋放排程隊列(dispatch queue)時,排程隊列(dispatch queue)沒有随着程式釋放。這造成了直到RLMRealm 對象被釋放後,Realm 中間版本的資料空間才會被再利用。為了避免這個問題,您應該在 dispatch 隊列中,使用一個顯式的自動排程隊列(dispatch queue)。
8.Realm 沒有自動增長屬性
Realm 沒有線程/程序安全的自動增長屬性機制,這在其他資料庫中常常用來産生主鍵。然而,在絕大多數情況下,對于主鍵來說,我們需要的是一個唯一的、自動生成的值,是以沒有必要使用順序的、連續的、整數的 ID 作為主鍵。
解決辦法:
在這種情況下,一個獨一無二的字元串主鍵通常就能滿足需求了。一個常見的模式是将預設的屬性值設定為 [[NSUUID UUID] UUIDString]以産生一個唯一的字元串 ID。自動增長屬性另一種常見的動機是為了維持插入之後的順序。在某些情況下,這可以通過向某個 RLMArray中添加對象,或者使用 [NSDate date]預設值的createdAt屬性。
9.所有的資料模型必須直接繼承自RealmObject。這阻礙我們利用資料模型中的任意類型的繼承。
這一點也不算問題,我們隻要自己在建立一個model就可以解決這個問題。自己建立的model可以自己随意去繼承,這個model專門用來接收網絡資料,然後把自己的這個model轉換成要存儲到表裡面的model,即RLMObject對象。這樣這個問題也可以解決了。
Realm 允許模型能夠生成更多的子類,也允許跨模型進行代碼複用,但是由于某些 Cocoa 特性使得運作時中豐富的類多态無法使用。以下是可以完成的操作:
父類中的類方法,執行個體方法和屬性可以被它的子類所繼承
子類中可以在方法以及函數中使用父類作為參數
以下是不能完成的:
多态類之間的轉換(例如子類轉換成子類,子類轉換成父類,父類轉換成子類等)
同時對多個類進行檢索
多類容器 (RLMArray以及 RLMResults)
10.Realm不支援集合類型
這一點也是比較蛋疼。
Realm支援以下的屬性類型:BOOL、bool、int、NSInteger、long、long long、float、double、NSString、NSDate、NSData以及
被特殊類型标記的NSNumber。CGFloat屬性的支援被取消了,因為它不具備平台獨立性。
這裡就是不支援集合,比如說NSArray,NSMutableArray,NSDictionary,NSMutableDictionary,NSSet,NSMutableSet。如果伺服器傳來的一個字典,key是一個字元串,對應的value就是一個數組,這時候就想存儲這個數組就比較困難了。當然Realm裡面是有集合的,就是RLMArray,這裡面裝的都是RLMObject。
是以我們想解決這個問題,就需要把資料裡面的東西都取出來,如果是model,就先自己接收一下,然後轉換成RLMObject的model,再存儲到RLMArray裡面去,這樣轉換一遍,還是可以的做到的。
這裡列出了暫時Realm目前辦法存在的“缺點”,如果這10點,在自己的App上都能滿足業務需求,那麼這一道坎也不是問題了。
以上兩道砍請仔細衡量清楚,這裡還有一篇文章是關于更換資料庫的心得體會的,
高速公路換輪胎——為遺留系統替換資料庫考慮更換的同學也可以看看。這兩道坎如果真的不适合,過不去,那麼請放棄Realm吧!
六. Realm 到底是什麼?
大家都知道Sqlite3 是一個移動端上面使用的小型資料庫,FMDB是基于Sqlite3進行的一個封裝。
那Core Data是資料庫麼?Core Data本身并不是資料庫,它是一個擁有多種功能的架構,其中一個重要的功能就是把應用程式同資料庫之間的互動過程自動化了。有了Core Data架構以後,我們無須編寫Objective-C代碼,又可以是使用關系型資料庫。因為Core Data會在底層自動給我們生成應該最佳優化過的SQL語句。
那麼Realm是資料庫麼?
Realm 不是 ORM,也不基于 SQLite 建立,而是為移動開發者定制的全功能資料庫。它可以将原生對象直接映射到Realm的資料庫引擎(遠不僅是一個鍵值對存儲)中。
Realm 是一個
MVCC 資料庫,底層是用 C++ 編寫的。MVCC 指的是多版本并發控制。
Realm是滿足ACID的。原子性(Atomicity)、一緻性(Consistency)、隔離性(Isolation)、持久性(Durability)。一個支援事務(Transaction)的資料庫,必需要具有這四種特性。Realm都已經滿足。
1.Realm 采用MVCC的設計思想
MVCC 解決了一個重要的并發問題:在所有的資料庫中都有這樣的時候,當有人正在寫資料庫的時候有人又想讀取資料庫了(例如,不同的線程可以同時讀取或者寫入同一個資料庫)。這會導緻資料的不一緻性 - 可能當你讀取記錄的時候一個寫操作才部分結束。
有很多的辦法可以解決讀、寫并發的問題,最常見的就是給資料庫加鎖。在之前的情況下,我們在寫資料的時候就會加上一個鎖。在寫操作完成之前,所有的讀操作都會被阻塞。這就是衆所周知的讀-寫鎖。這常常都會很慢。Realm采用的是MVCC資料庫的優點就展現出來了,速度非常快。
MVCC 在設計上采用了和 Git 一樣的源檔案管理算法。你可以把 Realm 的内部想象成一個 Git,它也有分支和原子化的送出操作。這意味着你可能工作在許多分支上(資料庫的版本),但是你卻沒有一個完整的資料拷貝。Realm 和真正的 MVCC 資料庫還是有些不同的。一個像 Git 的真正的 MVCC 資料庫,你可以有成為版本樹上 HEAD 的多個候選者。而 Realm 在某個時刻隻有一個寫操作,而且總是操作最新的版本 - 它不可以在老的版本上工作。
Realm底層是B+樹實作的,在Realm團隊開源的
realm-core裡面可以看到源碼,裡面有用bpTree,這是一個B+樹的實作。B+ 樹是一種樹資料結構,是一個n叉樹,每個節點通常有多個孩子,一棵B+樹包含根節點、内部節點和葉子節點。根節點可能是一個葉子節點,也可能是一個包含兩個或兩個以上孩子節點的節點。
B+ 樹通常用于資料庫和作業系統的
檔案系統中。NTFS, ReiserFS, NSS, XFS, JFS, ReFS 和BFS等檔案系統都在使用B+樹作為中繼資料索引。B+ 樹的特點是能夠保持資料穩定有序,其插入與修改擁有較穩定的對數時間複雜度。B+ 樹元素自底向上插入。
Realm會讓每一個連接配接的線程都會有資料在一個特定時刻的快照。這也是為什麼能夠在上百個線程中做大量的操作并同時通路資料庫,卻不會發生崩潰的原因。
上圖很好的展現了Realm的一次寫操作流程。這裡分3個階段,階段一中,V1指向根節點R。在階段二中,準備寫入操作,這個時候會有一個V2節點,指向新的R',并且建立一個分支出來,A'和C'。相應的右孩子指向原來V1指向的R的右孩子。如果寫入操作失敗,就丢棄左邊這個分支。這樣的設計可以保證即使失敗,也僅僅隻丢失最新資料,而不會破壞整個資料庫。如果寫入成功,那麼把原來的R,A,C節點放入Garbage中,于是就到了第三階段,寫入成功,變成了V2指向根節點。
在這個寫入的過程中,第二階段是最關鍵的,寫入操作并不會改變原有資料,而是建立了一個新的分支。這樣就不用加鎖,也可以解決資料庫的并發問題。
正是B+樹的底層資料結構 + MVCC的設計,保證了Realm的高性能。
2.Realm 采用了 zero-copy 架構
因為 Realm 采用了 zero-copy 架構,這樣幾乎就沒有記憶體開銷。這是因為每一個 Realm 對象直接通過一個本地 long 指針和底層資料庫對應,這個指針是資料庫中資料的鈎子。
通常的傳統的資料庫操作是這樣的,資料存儲在磁盤的資料庫檔案中,我們的查詢請求會轉換為一系列的SQL語句,建立一個資料庫連接配接。資料庫伺服器收到請求,通過解析器對SQL語句進行詞法和文法語義分析,然後通過查詢優化器對SQL語句進行優化,優化完成執行對應的查詢,讀取磁盤的資料庫檔案(有索引則先讀索引),讀取命中查詢的每一行的資料,然後存到記憶體裡(這裡有記憶體消耗)。之後你需要把資料序列化成可在記憶體裡面存儲的格式,這意味着比特對齊,這樣 CPU 才能處理它們。最後,資料需要轉換成語言層面的類型,然後它會以對象的形式傳回,比如Objective-C的對象等。
這裡就是Realm另外一個很快的原因,Realm的資料庫檔案是通過memory-mapped,也就是說資料庫檔案本身是映射到記憶體(實際上是虛拟記憶體)中的,Realm通路檔案偏移就好比檔案已經在記憶體中一樣(這裡的記憶體是指虛拟記憶體),它允許檔案在沒有做反序列化的情況下直接從記憶體讀取,提高了讀取效率。Realm 隻需要簡單地計算偏移來找到檔案中的資料,然後從原始通路點傳回資料結構的值 。
正是Realm采用了 zero-copy 架構,幾乎沒有記憶體開銷,Realm核心檔案格式基于memory-mapped,節約了大量的序列化和反序列化的開銷,導緻了Realm擷取對象的速度特别高效。
-
Realm 對象在不同的線程間不能共享
Realm 對象不能線上程間傳遞的原因就是為了保證隔離性和資料一緻性。這樣做的目的隻有一個,為了速度。
由于Realm是基于零拷貝的,所有對象都在記憶體裡,是以會自動更新。如果允許Realm對象線上程間共享,Realm 會無法確定資料的一緻性,因為不同的線程會在不确定的什麼時間點同時改變對象的資料。
要想保證多線程能共享對象就是加鎖,但是加鎖又會導緻一個長時間的背景寫事務會阻塞 UI 的讀事務。不加鎖就不能保證資料的一緻性,但是可以滿足速度的要求。Realm在衡量之後,還是為了速度,做出了不允許線程間共享的妥協。
正是因為不允許對象在不同的線程間共享,保證了資料的一緻性,不加線程鎖,保證了Realm的在速度上遙遙領先。
-
真正的懶加載
大多數資料庫趨向于在水準層級存儲資料,這也就是為什麼你從 SQLite 讀取一個屬性的時候,你就必須要加載整行的資料。它在檔案中是連續存儲的。
不同的是,Realm盡可能讓 Realm 在垂直層級連續存儲屬性,你也可以看作是按列存儲。
在查詢到一組資料後,隻有當你真正通路對象的時候才真正加載進來。
- Realm 中的檔案
Realm資料庫的掌握 先來說說中間的Database File
.realm 檔案是memory mapped的,所有的對象都是檔案首位址偏移量的一個引用。對象的存儲不一定是連續的,但是Array可以保證是連續存儲。
.realm執行寫操作的時候,有3個指針,一個是current top pointer ,一個是 other top pointer ,最後一個是 switch bit。
switch bit* 标示着top pointer是否已經被使用過。如果被使用過了,代表着資料庫已經是可讀的。
the top pointer優先更新,緊接着是the switch bit更新。因為即使寫入失敗了,雖然丢失了所有資料,但是這樣能保證資料庫依舊是可讀的。
再來說說 .lock file。
.lock檔案中會包含 the shared group 的metadata。這個檔案承擔着允許多線程通路相同的Realm對象的職責。
最後說說Commit logs history
這個檔案會用來更新索引indexes,會用來同步。裡面主要維護了3個小檔案,2個是資料相關的,1個是操作management的。
總結
經過上面的分析之後,深深的感受到Realm就是為速度而生的!在保證了ACID的要求下,很多設計都是以速度為主。當然,Realm 最核心的理念就是對象驅動,這是 Realm 的核心原則。Realm 本質上是一個嵌入式資料庫,但是它也是看待資料的另一種方式。它用另一種角度來重新看待移動應用中的模型和業務邏輯。
Realm還是跨平台的,多個平台都使用相同的資料庫,是多麼好的一件事情呀。相信使用Realm作為App資料庫的開發者會越來越多。