ios App沙盒(sandbox)模型文件目錄詳解 Swift3版本


 iOS 中的沙盒機制介紹

iOS 中的沙盒機制(SandBox)是一種安全體系。
每個 iOS 應用程序都有一個單獨的文件系統(存儲空間),而且只能在對應的文件系統中進行操作,此區域被稱為沙盒。所有的非代碼文件都要保存在此,例如屬性文件 plist、文本文件、圖像、圖標、媒體資源等。

iPhone沙箱模型的四個文件夾介紹和功能

1、Documents 目錄:您應該將所有de應用程序數據文件寫入到這個目錄下。這個目錄用於存儲用戶數據或其它應該定期備份的信息。用戶自己保存的文件在documents文件里

2、AppName.app 目錄:這是應用程序的程序包目錄,包含應用程序的本身。由於應用程序必須經過簽名,所以您在運行時不能對這個目錄中的內容進行修改,否則可能會使應用程序無法啟動。

3、Library 目錄:這個目錄下有兩個子目錄:Caches 和 Preferences
Preferences 目錄:包含應用程序的偏好設置文件。您不應該直接創建偏好設置文件,而是應該使用NSUserDefaults類來取得和設置應用程序的偏好.
Caches 目錄:用於存放應用程序專用的支持文件,保存應用程序再次啟動過程中需要的信息。

4、tmp 目錄:這個目錄用於存放臨時文件,保存應用程序再次啟動過程中不需要的信息。

 

補充1:對於上述描述可以這樣舉例理解,一個記事本應用,用戶寫的東西需要保存起來,這些東西是用戶自行生成的,則需要放在 Documents 目錄里。一個新聞應用,如果需要從服務器下載東西展示給用戶看,下載的東西就放在 Library/Caches 目錄里。蘋果審核對這個要求很嚴格,主要原因是 iCloud 的同步問題。

補充2:如果想知道真機或者模擬器 App 沙盒路徑,可通過在項目中執行下述代碼打印獲取:

獲取這些目錄路徑的方法

1,獲取Home目錄路徑的函數:
let homeDir = NSHomeDirectory()

2,獲取Documents目錄路徑的方法:(swift3)

路徑結果:

3,獲取Caches目錄路徑的方法:

 

路徑結果:

4,獲取tmp目錄路徑的方法:

let tmpDir = NSTemporaryDirectory();

5,獲取應用程序程序包中資源文件路徑的方法:

程序包(NSBundle)
iOS 應用都是通過 bundle 進行封裝的,可以狹隘地將 bundle 理解為上述沙盒中的 AppName.app 文件。在 Finder 中,會把 bundle 當做一個文件顯示從而防止用戶誤操作導致程序文件損壞,但其實內部是一個目錄,包含了圖像、媒體資源、編譯好的代碼、nib 文件等,這個目錄稱為 main bundle。
Cocoa 提供了 NSBundle 類封裝了 bundle 操作。

通過NSBundle可以獲取各種資源文件,注意在Build Phases - Copy Bundle Rresources下面需要有這個資源文件,有些類型的文件不會自動加在這里

 let paths3 = Bundle.main.path(forResource: "20160504182134006", ofType: "pcm")

代碼中的mainBundle類方法用於返回一個代表應用程序包的對象。

NSFileManager文件管理介紹

使用 FileManager 可以對沙盒中的目錄、文件進行操作。

使用NSFileManager創建文件夾和文件

使用FileManager在documents目錄下創建文件夾和文件,方法如下:

執行結果為(Direct為文件夾,hi為Direct文件下的文件):

將對象寫入文件

可以通過寫(至:)方法,可以創建文件並將對象寫入,對象包括字符串,的NSString,UIImage的,的NSArray,NSDictionary的等。

(1)把字符串保存到文件(其實執行的是創建文件並將字符串寫入並保存到文件中)

(2)把圖片保存到文件路徑下,(1)(2)並列如下圖:

1,遍歷一個目錄下的所有文件

假設用戶文檔下有如下文件和文件夾:test1.txt、fold1/test2.txt

 

 

(1)首先我們獲取用戶文檔目錄路徑

1

2

3

4

let manager = FileManager.default

let urlForDocument = manager.urls(for: .documentDirectory, in:.userDomainMask)

let url = urlForDocument[0] as URL

print(url)

 

 

(2)對指定路徑執行淺搜索,返回指定目錄路徑下的文件、子目錄及符號鏈接的列表

1

2

let contentsOfPath = try? manager.contentsOfDirectory(atPath: url.path)

print("contentsOfPath: \(contentsOfPath)")

 

 

(3)類似上面的,對指定路徑執行淺搜索,返回指定目錄路徑下的文件、子目錄及符號鏈接的列表

1

2

3

let contentsOfURL = try? manager.contentsOfDirectory(at: url,

                        includingPropertiesForKeys: nil, options: .skipsHiddenFiles)

print("contentsOfURL: \(contentsOfURL)")

 

 

(4)深度遍歷,會遞歸遍歷子文件夾(但不會遞歸符號鏈接)

1

2

let enumeratorAtPath = manager.enumerator(atPath: url.path)

print("enumeratorAtPath: \(enumeratorAtPath?.allObjects)")

 

 

(5)類似上面的,深度遍歷,會遞歸遍歷子文件夾(但不會遞歸符號鏈接)

1

2

3

let enumeratorAtURL = manager.enumerator(at: url, includingPropertiesForKeys: nil,

                                         options: .skipsHiddenFiles, errorHandler:nil)

print("enumeratorAtURL: \(enumeratorAtURL?.allObjects)")

 

 

(6)深度遍歷,會遞歸遍歷子文件夾(包括符號鏈接,所以要求性能的話用enumeratorAtPath)

1

2

let subPaths = manager.subpaths(atPath: url.path)

print("subPaths: \(subPaths)")

 

 

2,判斷文件或文件夾是否存在

1

2

3

let fileManager = FileManager.default

let filePath:String = NSHomeDirectory() + "/Documents/hangge.txt"

let exist = fileManager.fileExists(atPath: filePath)

 

3,復制文件 

(1)方法1

1

2

3

4

5

let fileManager = FileManager.default

let homeDirectory = NSHomeDirectory()

let srcUrl = homeDirectory + "/Documents/hangge.txt"

let toUrl = homeDirectory + "/Documents/copyed.txt"

try! fileManager.copyItem(atPath: srcUrl, toPath: toUrl)

 

(2)方法2

1

2

3

4

5

6

7

8

9

10

// 定位到用戶文檔目錄

let manager = FileManager.default

let urlForDocument = manager.urls( for:.documentDirectory, in:.userDomainMask)

let url = urlForDocument[0]

 

// 將test.txt文件拷貝到文檔目錄根目錄下的copyed.txt文件

let srcUrl = url.appendingPathComponent("test.txt")

let toUrl = url.appendingPathComponent("copyed.txt")

 

try! manager.copyItem(at: srcUrl, to: toUrl)

 

4,移動文件

(1)方法1

1

2

3

4

5

let fileManager = FileManager.default

let homeDirectory = NSHomeDirectory()

let srcUrl = homeDirectory + "/Documents/hangge.txt"

let toUrl = homeDirectory + "/Documents/moved/hangge.txt"

try! fileManager.moveItem(atPath: srcUrl, toPath: toUrl)

 

(2)方法2

1

2

3

4

5

6

7

8

9

// 定位到用戶文檔目錄

let manager = FileManager.default

let urlForDocument = manager.urls( for: .documentDirectory, in:.userDomainMask)

let url = urlForDocument[0]

 

let srcUrl = url.appendingPathComponent("test.txt")

let toUrl = url.appendingPathComponent("copyed.txt")

// 移動srcUrl中的文件(test.txt)到toUrl中(copyed.txt)

try! manager.moveItem(at: srcUrl, to: toUrl)

 

5,刪除文件

(1)方法1

1

2

3

4

let fileManager = FileManager.default

let homeDirectory = NSHomeDirectory()

let srcUrl = homeDirectory + "/Documents/hangge.txt"

try! fileManager.removeItem(atPath: srcUrl)

 

(2)方法2

1

2

3

4

5

6

7

8

// 定位到用戶文檔目錄

let manager = FileManager.default

let urlForDocument = manager.urls(for: .documentDirectory, in:.userDomainMask)

let url = urlForDocument[0]

 

let toUrl = url.appendingPathComponent("copyed.txt")

// 刪除文檔根目錄下的toUrl路徑的文件(copyed.txt文件)

try! manager.removeItem(at: toUrl)

 

6,刪除目錄下所有的文件

(1)方法1:獲取所有文件,然后遍歷刪除

1

2

3

4

5

6

let fileManager = FileManager.default

let myDirectory = NSHomeDirectory() + "/Documents/Files"

let fileArray = fileManager.subpaths(atPath: myDirectory)

for fn in fileArray!{

    try! fileManager.removeItem(atPath: myDirectory + "/\(fn)")

}

 

(2)方法2:刪除目錄后重新創建該目錄

1

2

3

4

5

let fileManager = FileManager.default

let myDirectory = NSHomeDirectory() + "/Documents/Files"

try! fileManager.removeItem(atPath: myDirectory)

try! fileManager.createDirectory(atPath: myDirectory, withIntermediateDirectories: true,

                                 attributes: nil)

7,讀取文件

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

let manager = FileManager.default

let urlsForDocDirectory = manager.urls(for: .documentDirectory, in:.userDomainMask)

let docPath = urlsForDocDirectory[0]

let file = docPath.appendingPathComponent("test.txt")

 

//方法1

let readHandler = try! FileHandle(forReadingFrom:file)

let data = readHandler.readDataToEndOfFile()

let readString = String(data: data, encoding: String.Encoding.utf8)

print("文件內容: \(readString)")

 

//方法2

let data2 = manager.contents(atPath: file.path)

let readString2 = String(data: data2!, encoding: String.Encoding.utf8)

print("文件內容: \(readString2)")

 

8,在任意位置寫入數據

1

2

3

4

5

6

7

8

9

10

let manager = FileManager.default

let urlsForDocDirectory = manager.urls(for:.documentDirectory, in:.userDomainMask)

let docPath = urlsForDocDirectory[0]

let file = docPath.appendingPathComponent("test.txt")

 

let string = "添加一些文字到文件末尾"

let appendedData = string.data(using: String.Encoding.utf8, allowLossyConversion: true)

let writeHandler = try? FileHandle(forWritingTo:file)

writeHandler!.seekToEndOfFile()

writeHandler!.write(appendedData!)

 

9,文件權限判斷

1

2

3

4

5

6

7

8

9

10

11

12

13

let manager = FileManager.default

let urlForDocument = manager.urls(for: .documentDirectory, in:.userDomainMask)

let docPath = urlForDocument[0]

let file = docPath.appendingPathComponent("test.txt")

 

let readable = manager.isReadableFile(atPath: file.path)

print("可讀: \(readable)")

let writeable = manager.isWritableFile(atPath: file.path)

print("可寫: \(writeable)")

let executable = manager.isExecutableFile(atPath: file.path)

print("可執行: \(executable)")

let deleteable = manager.isDeletableFile(atPath: file.path)

print("可刪除: \(deleteable)")

 

10,獲取文件屬性(創建時間,修改時間,文件大小,文件類型等信息)

1

2

3

4

5

6

7

let manager = FileManager.default

let urlForDocument = manager.urls(for: .documentDirectory, in:.userDomainMask)

let docPath = urlForDocument[0]

let file = docPath.appendingPathComponent("test.txt")

 

let attributes = try? manager.attributesOfItem(atPath: file.path) //結果為Dictionary類型

print("attributes: \(attributes!)")

 

 

attributes 中獲取具體的屬性:

1

2

3

print("創建時間:\(attributes![FileAttributeKey.creationDate]!)")

print("修改時間:\(attributes![FileAttributeKey.modificationDate]!)")

print("文件大小:\(attributes![FileAttributeKey.size]!)")

 

 

11,文件/文件夾比較

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

let manager = FileManager.default

let urlForDocument = manager.urls(for: .documentDirectory, in:.userDomainMask)

let docPath = urlForDocument[0]

let contents = try! manager.contentsOfDirectory(atPath: docPath.path)

 

//下面比較用戶文檔中前面兩個文件是否內容相同(該方法也可以用來比較目錄)

let count = contents.count

if count > 1 {

    let path1 = docPath.path + "/" + (contents[0] as String)

    let path2 = docPath.path + "/" + (contents[1] as String)

    let equal = manager.contentsEqual(atPath: path1,andPath:path2)

    print("path1:\(path1)")

    print("path2:\(path2)")

    print("比較結果: \(equal)")

}

article end

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM