iCloud是蘋果提供的雲端服務,用戶可以將通訊錄、備忘錄、郵件、照片、音樂、視頻等備份到雲服務器並在各個蘋果設備間直接進行共享而無需關心數據同步問題,甚至即使你的設備丟失後在一台新的設備上也可以通過Apple ID登錄同步。
key-value data
:NSUserDefault
document
:Provisioning Profile
),這裡可以使用通配Bundle ID
以上2步是針對真機的,調試模擬器可以忽略 打開項目的Capabilities
,找到iCloud服務並開啟它 在iCloud服務的Service
中勾選Key-value storae
和iCloud Documents
entitlements
文件使用和NSUserDefault
差不多,都是以鍵值對的形式存儲。
#import "iCloudKeysViewController.h"
@interface iCloudKeysViewController()
@property (strong, nonatomic) NSUbiquitousKeyValueStore *keyStore;
@property (strong, nonatomic) IBOutlet UILabel *textLabel;
@end
@implementation iCloudKeysViewController
- (void)viewDidLoad
{
[super viewDidLoad];
self.textLabel.text = @"Ready Go";
//獲取iCloud配置首選項
self.keyStore = [NSUbiquitousKeyValueStore defaultStore];
//注冊通知中心,當配置發生改變的時候,發生通知
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
[center addObserver:self
selector:@selector(ubiquitousKeyValueStoreDidChange:)
name:NSUbiquitousKeyValueStoreDidChangeExternallyNotification
object:keyStore];
}
/* UI點擊,點擊改變按鈕 */
- (IBAction)changeKey
{
[self.keyStore setString:@"Hello World" forKey:@"MyString"];
[self.keyStore synchronize];
NSLog(@"Save key");
}
/* 監聽通知,當配置發生改變的時候會調用 */
- (void)ubiquitousKeyValueStoreDidChange:(NSNotification *)notification
{
NSLog(@"External Change detected");
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Change detected"
message:@"Change detected"
delegate:nil
cancelButtonTitle:@"Ok"
otherButtonTitles:nil, nil];
[alert show];
//顯示改變的信息
textLabel.text = [keyStore stringForKey:@"MyString"];
}
@end
UIDocument
類來完成,這個類提供了新建、修改、查詢文檔、打開文檔、刪除文檔的功能。 UIDocument
對文檔的新增、修改、刪除、讀取全部基於一個雲端URL來完成,對於開發者而言沒有本地和雲端之分,這樣大大簡化了開發過程。
NSFileManager
的對象方法
-(NSURL *)URLForUbiquityContainerIdentifier:(NSString *)identifier;
上面需要傳遞一個雲端存儲容器的唯一標示,這個可以去自動生成的entitlements
文件查看Ubiquity Container Identifiers
字段獲得,如果傳nil
,代表第一個容器
默認的第一個容器標識是
iCloud.$(CFBundleIdentifier)
,
其中$(CFBundleIdentifier)
代表Bundle ID
,那麼根據應用的Bundle ID
就可以得知我的第一個容器的標識是iCloud.com.liuting.icloud.iCloudTest
/* 將指定URL的文檔保存到iCloud(可以是新增或者覆蓋,通過saveOperation參數設定)*/
- (void)saveToURL:(NSURL *)url
forSaveOperation:(UIDocumentSaveOperation)saveOperation
completionHandler:(void (^)(BOOL success))completionHandler;
/* 保存操作option */
typedef NS_ENUM(NSInteger, UIDocumentSaveOperation) {
UIDocumentSaveForCreating,/* 創建 */
UIDocumentSaveForOverwriting/* 覆蓋寫入 */
};
/* 打開文檔,參數是打開文檔成功回調 */
- (void)openWithCompletionHandler:(void (^)(BOOL success))completionHandler;
/* 刪除指定URL下的文件 */
- (BOOL)removeItemAtURL:(NSURL *)URL
error:(NSError **)error;
UIDocument
在設計的時候,沒有提供統一的存儲方式來存儲數據,需要我們去繼承它,重寫2個對象方法自己操作數據
/*
保存文檔時調用
@param typeName 文檔文件類型
@param outError 錯誤信息輸出
@return 文檔數據
*/
-(id)contentsForType:(NSString *)typeName
error:(NSError **)outError;
/*
讀取數據時調用
@param contents 文檔數據
@param typeName 文檔文件類型
@param outError 錯誤信息輸出
@return 讀取是否成功
*/
-(BOOL)loadFromContents:(id)contents
ofType:(NSString *)typeName
error:(NSError **)outError;
UIDocument
保存數據的本質:NSData
或者NSFileWrapper
數據 UIDocument
讀取數據的本質:NSData
或者NSFileWrapper
數據轉化為A對應類型的數據
#import
@interface LTDocument : UIDocument
@property (strong, nonatomic) NSData *data;/*< 文檔數據 */
@end
#import "LTDocument.h"
@implementation LTDocument
#pragma mark - 重寫父類方法
/**
* 保存時調用
* @param typeName 文檔文件類型後綴
* @param outError 錯誤信息輸出
* @return 文檔數據
*/
- (id)contentsForType:(NSString *)typeName
error:(NSError *__autoreleasing *)outError
{
if (!self.data) {
self.data = [NSData data];
}
return self.data;
}
/**
* 讀取數據時調用
* @param contents 文檔數據
* @param typeName 文檔文件類型後綴
* @param outError 錯誤信息輸出
* @return 讀取是否成功
*/
- (BOOL)loadFromContents:(id)contents
ofType:(NSString *)typeName
error:(NSError *__autoreleasing *)outError
{
self.data = [contents copy];
return true;
}
@end
如果要加載iCloud中的文檔列表,就需要使用另一個類NSMetadataQuery
通常考慮到網絡的原因並不會一次性加載所有數據,而利用NSMetadataQuery
並指定searchScopes
為NSMetadataQueryUbiquitousDocumentScope
來限制查找iCloud文檔數據。 使用NSMetadataQuery
還可以通過謂詞限制搜索關鍵字等信息,並在搜索完成之後通過通知的形式通知客戶端搜索的情況。
#import "ViewController.h"
#import "LTDocument.h"
#define kContainerIdentifier @"iCloud.com.liuting.icloud.iCloudTest"
@interface ViewController ()
@property (weak, nonatomic) IBOutlet UITextField *documentField;/*< 輸入框 */
@property (weak, nonatomic) IBOutlet UILabel *documentShowLable;/*< 顯示欄 */
@property (weak, nonatomic) IBOutlet UITableView *documentTableView;/* 文檔列表 */
/* 文檔文件信息,鍵為文件名,值為創建日期 */
@property (strong, nonatomic) NSMutableDictionary *files;
@property (strong, nonatomic) NSMetadataQuery *query;/*< 查詢文檔對象 */
@property (strong, nonatomic) LTDocument *document;/*< 當前選中文檔 */
@end
/**
* 取得雲端存儲文件的地址
* @param fileName 文件名,如果文件名為nil,則重新創建一個URL
* @return 文件地址
*/
- (NSURL *)getUbiquityFileURL:(NSString *)fileName{
//取得雲端URL基地址(參數中傳入nil則會默認獲取第一個容器),需要一個容器標示
NSFileManager *manager = [NSFileManager defaultManager];
NSURL *url = [manager URLForUbiquityContainerIdentifier:kContainerIdentifier];
//取得Documents目錄
url = [url URLByAppendingPathComponent:@"Documents"];
//取得最終地址
url = [url URLByAppendingPathComponent:fileName];
return url;
}
/* 從iCloud上加載所有文檔信息 */
- (void)loadDocuments
{
if (!self.query) {
self.query = [[NSMetadataQuery alloc] init];
self.query.searchScopes = @[NSMetadataQueryUbiquitousDocumentsScope];
//注意查詢狀態是通過通知的形式告訴監聽對象的
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
[center addObserver:self
selector:@selector(metadataQueryFinish:)
name:NSMetadataQueryDidFinishGatheringNotification
object:self.query];//數據獲取完成通知
[center addObserver:self
selector:@selector(metadataQueryFinish:)
name:NSMetadataQueryDidUpdateNotification
object:self.query];//查詢更新通知
}
//開始查詢
[self.query startQuery];
}
/* 查詢更新或者數據獲取完成的通知調用 */
- (void)metadataQueryFinish:(NSNotification *)notification
{
NSLog(@"數據獲取成功!");
NSArray *items = self.query.results;//查詢結果集
self.files = [NSMutableDictionary dictionary];
//變量結果集,存儲文件名稱、創建日期
[items enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
NSMetadataItem *item = obj;
//獲取文件名
NSString *fileName = [item valueForAttribute:NSMetadataItemFSNameKey];
//獲取文件創建日期
NSDate *date = [item valueForAttribute:NSMetadataItemFSContentChangeDateKey];
NSDateFormatter *dateformate = [[NSDateFormatter alloc]init];
dateformate.dateFormat = @"YY-MM-dd HH:mm";
NSString *dateString = [dateformate stringFromDate:date];
//保存文件名和文件創建日期
[self.files setObject:dateString forKey:fileName];
}];
//表格刷新
self.documentShowLable.text = @"";
[self.documentTableView reloadData];
}
#pragma mark - UI點擊事件
/* 點擊添加文檔 */
- (IBAction)addDocument:(id)sender {
//提示信息
if (self.documentField.text.length <= 0) {
NSLog(@"請輸入要創建的文檔名");
self.documentField.placeholder = @"請輸入要創建的文檔名";
return;
}
//創建文檔URL
NSString *text = self.documentField.text;
NSString *fileName = [NSString stringWithFormat:@"%@.txt",text];
NSURL *url = [self getUbiquityFileURL:fileName];
//創建雲端文檔對象
LTDocument *document = [[LTDocument alloc] initWithFileURL:url];
//設置文檔內容
NSString *dataString = @"hallo World";
document.data = [dataString dataUsingEncoding:NSUTF8StringEncoding];
//保存或創建文檔,UIDocumentSaveForCreating是創建文檔
[document saveToURL:url
forSaveOperation:UIDocumentSaveForCreating
completionHandler:^(BOOL success)
{
if (success) {
NSLog(@"創建文檔成功.");
self.documentField.text = @"";
//從iCloud上加載所有文檔信息
[self loadDocuments];
}else{
NSLog(@"創建文檔失敗.");
}
}];
}
/* 點擊修改文檔 */
- (IBAction)saveDocument:(UIButton *)sender {
if ([sender.titleLabel.text isEqualToString:@"修改文檔"]) {
self.documentField.text = self.documentShowLable.text;
[sender setTitle:@"保存文檔" forState:UIControlStateNormal];
} else if([sender.titleLabel.text isEqualToString:@"保存文檔"]) {
[sender setTitle:@"修改文檔" forState:UIControlStateNormal];
self.documentField.placeholder = @"請輸入修改的文檔內容";
//要保存的文檔內容
NSString *dataText = self.documentField.text;
NSData *data = [dataText dataUsingEncoding:NSUTF8StringEncoding];
self.document.data = data;
//保存或創建文檔,UIDocumentSaveForOverwriting是覆蓋保存文檔
[self.document saveToURL:self.document.fileURL
forSaveOperation:UIDocumentSaveForOverwriting
completionHandler:^(BOOL success)
{
NSLog(@"保存成功!");
self.documentShowLable.text = self.documentField.text;
self.documentField.text = @"";
}];
}
}
/* 點擊刪除文檔 */
- (IBAction)removeDocument:(id)sender {
//提示信息
if (self.documentField.text.length <= 0) {
self.documentField.placeholder = @"請輸入要刪除的文檔名";
return;
}
//判斷要刪除的文檔是否存在
NSString *text = self.documentField.text;
NSString *fileName = [NSString stringWithFormat:@"%@.txt",text];
NSArray *fileNames = [self.files allKeys];
if (![fileNames containsObject:fileName]) {
NSLog(@"沒有要刪除的文檔");
return;
}
//創建要刪除的文檔URL
NSURL *url = [self getUbiquityFileURL:fileName];
NSError *error = nil;
//刪除文檔文件
[[NSFileManager defaultManager] removeItemAtURL:url error:&error];
if (error) {
NSLog(@"刪除文檔過程中發生錯誤,錯誤信息:%@",error.localizedDescription);
return;
}
//從集合中刪除
[self.files removeObjectForKey:fileName];
self.documentField.text = @"";
}
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
self.documentTableView.delegate = self;
self.documentTableView.dataSource = self;
/* 從iCloud上加載所有文檔信息 */
[self loadDocuments];
}
#pragma mark - UITableView數據源
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
return 1;
}
- (NSInteger)tableView:(UITableView *)tableView
numberOfRowsInSection:(NSInteger)section
{
return self.files.count;
}
- (UITableViewCell *)tableView:(UITableView *)tableView
cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
static NSString *identtityKey = @"myTableViewCellIdentityKey1";
UITableViewCell *cell =
[self.documentTableView dequeueReusableCellWithIdentifier:identtityKey];
if(cell == nil){
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1
reuseIdentifier:identtityKey];
}
//顯示文檔名和文檔創建日期
NSArray *fileNames = self.files.allKeys;
NSString *fileName = fileNames[indexPath.row];
cell.textLabel.text = fileName;
cell.detailTextLabel.text = [self.files valueForKey:fileName];
return cell;
}
#pragma mark - UITableView代理方法
/* 點擊文檔列表的其中一個文檔調用 */
- (void)tableView:(UITableView *)tableView
didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
UITableViewCell *cell = [self.documentTableView cellForRowAtIndexPath:indexPath];
//獲取文檔URL
NSURL *url = [self getUbiquityFileURL:cell.textLabel.text];
//創建文檔操作對象
LTDocument *document = [[LTDocument alloc] initWithFileURL:url];
self.document = document;
//打開文檔並讀取文檔內容
[document openWithCompletionHandler:^(BOOL success) {
if(success){
NSLog(@"讀取數據成功.");
NSString *dataText = [[NSString alloc] initWithData:document.data
encoding:NSUTF8StringEncoding];
self.documentShowLable.text = dataText;
}else{
NSLog(@"讀取數據失敗.");
}
}];
}
@end