1、應用沙盒 sandbox
1. 什麼是沙盒
每個iOS應用都有自己的應用沙盒(應用沙盒就是應用的文件夾),與其他文件系統隔離。應用必須待在自己的沙盒裡,其他應用不能訪問該沙盒。
2.應用沙盒結構
應用程序包: 包含了所有的 資源文件和 可執行文件
Documents: 保存應用運行時生成的需要持久化的數據,iTunes同步設備時會備份該目錄。例如,游戲應用可將游戲存檔保存在該目錄。保存相對重要的數據tmp: 保存應用運行時所需的臨時數據,使用完畢後再將相應的文件從該目錄刪除。應用沒有運行時,系統也可能會清除該目錄下的文件。iTunes同步設備時不會備份該目錄。保存不重要的並且大的數據。Library/Caches: 保存應用運行時生成的需要持久化的數據,iTunes同步設備時不會備份該目錄。一般存儲體積大、不需要備份的非重要數據Library/ Preference: 保存應用的所有偏好設置,iOS的Settings(設置)應用會在該目錄中查找應用的設置信息。iTunes同步設備時會備份該目錄。該目錄由系統管理,無需我們來管理。通常用來存儲一些基本的軟件配置信息, 比如記住密碼、自動登錄等。3. 應用沙盒目錄的常見獲取方式
沙盒目錄: NSString*home = NSHomeDirectory();
Documents: (2種)
1、利用沙盒跟目錄拼接"Documents"字符串
NSString *home = NSHomeDirectory();
NSString*documents = [home stringByAppendingPathComponent:@"Documents"];
// 不建議采用,因為新版本的操作系統可能會修改目錄名
2、利用NSSearchPathForDirectoriesInDomains函數
//NSUserDomainMask代表從用戶文件夾下找
//YES 代表展開路徑中的波浪字符“~”
NSArray*array = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,NSUserDomainMask,YES);
//在iOS中,只有一個目錄跟傳入的參數匹配,所以這個集合裡面只有一個元素
NSString*documents = [arrayobjectAtIndex:0];
tmp:
Library/Caches:(跟Documents類似的2種方法)NSString*tmp= NSTemporaryDirectory();
1、利用山河根目錄拼接"Caches"字符串
2、利用NSSearchPathForDirectoriesInDomains函數(將函數的第2個參數改為:NSCachesDirectory即可)
Library/Preference:
通過NSUserDefaults類存取該目錄下的設置信息
2、iOS數據存儲的常用方式
XML屬性列表(plist)歸檔Preference(偏好設置) 本質還是通過“plist”來存儲數據, 但是使用更簡單(無需關注文件、文件夾路徑和名稱)
NSKeyedArchiver歸檔(NSCoding) 把任何對象,直接保存為文件 SQLite3 當非常大量的數據存儲時使用 Core Data 對SQLite的封裝1、屬性列表
屬性列表: 屬性列表是一種XML格式的文件,拓展名為plist
如果對象是NSString、NSDictionary、NSArray、NSData、NSNumber等類型,就可以使用writeToFile:atomically:方法直接將對象寫到屬性列表文件中
歸檔 NSDictionary
// 將數據封裝成字典 NSMutableDictionary *dict = [NSMutableDictionary dictionary]; [dict setObject:@"母雞" forKey:@"name"]; [dict setObject:@"15013141314" forKey:@"phone"]; [dict setObject:@"27" forKey:@"age"]; // 將字典持久化到Documents/stu.plist文件中 [dict writeToFile:path atomically:YES];恢復 NSDictionary
// 讀取Documents/stu.plist的內容,實例化NSDictionary NSDictionary *dict = [NSDictionary dictionaryWithContentsOfFile:path]; NSLog(@"name:%@", [dict objectForKey:@"name"]); NSLog(@"phone:%@", [dict objectForKey:@"phone"]); NSLog(@"age:%@", [dict objectForKey:@"age"]);
2、偏好設置
每個應用都有個NSUserDefaults實例,通過它來存取偏好設置
獲取用戶偏好設置 保存信息NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; [defaults setObject:@"itcast" forKey:@"username"]; [defaults setFloat:18.0f forKey:@"text_size"]; [defaults setBool:YES forKey:@"auto_login"];
讀取偏好設置
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; NSString *username = [defaults stringForKey:@"username"]; float textSize = [defaults floatForKey:@"text_size"]; BOOL autoLogin = [defaults boolForKey:@"auto_login"];
注意:UserDefaults設置數據時,不是立即寫入,而是根據時間戳定時地把緩存中的數據寫入本地磁盤。所以調用了set方法之後數據有可能還沒有寫入磁盤應用程序就終止了。出現以上問題,可以通過調用synchornize方法強制寫入
[defaults synchornize];
3、NSKeyedArchiver
如果對象是NSString、NSDictionary、NSArray、NSData、NSNumber等類型,可以直接用NSKeyedArchiver進行歸檔和恢復 不是所有的對象都可以直接用這種方法進行歸檔,只有遵守了NSCoding協議的對象才可以 NSCoding協議
encodeWithCoder:
每次歸檔對象時,都會調用這個方法。一般在這個方法裡面指定如何歸檔對象中的每個實例變量,可以使用encodeObject:forKey:方法歸檔實例變量
initWithCoder:
每次從文件中恢復(解碼)對象時,都會調用這個方法。一般在這個方法裡面指定如何解碼文件中的數據為對象的實例變量,可以使用decodeObject:forKey方法解碼實例變量
歸檔 NSArray
編碼:[NSKeyedArchiver archiveRootObject:arraytoFile:path]
解碼:
NSArray *array = [NSKeyedUnarchiver unarchiveObjectWithFile:path];
歸檔 Person 對象
Person:
@interface Person : NSObject@property (nonatomic, copy) NSString *name; @property (nonatomic, assign) int age; @property (nonatomic, assign) float height; @end
@implementation Person - (void)encodeWithCoder:(NSCoder *)encoder { [encoder encodeObject:self.name forKey:@"name"]; [encoder encodeInt:self.age forKey:@"age"]; [encoder encodeFloat:self.height forKey:@"height"]; } - (id)initWithCoder:(NSCoder *)decoder { self.name = [decoder decodeObjectForKey:@"name"]; self.age = [decoder decodeIntForKey:@"age"]; self.height = [decoder decodeFloatForKey:@"height"]; return self; } - (void)dealloc { [super dealloc]; [_name release]; } @end歸檔
Person *person = [[[Person alloc] init] autorelease]; person.name = @"name"; person.age = 30; person.height = 1.80f; [NSKeyedArchiver archiveRootObject:person toFile:path];解碼
Person *person = [NSKeyedUnarchiver unarchiveObjectWithFile:path];
注意:如果父類也遵守了NSCoding協議,請注意: 應該在encodeWithCoder:方法中加上一句
[super encodeWithCode:encode];
確保繼承的實例變量也能被編碼,即也能被歸檔
應該在initWithCoder:方法中加上一句self = [super initWithCoder:decoder];
確保繼承的實例變量也能被解碼,即也能被恢復