Как получить доступ к файлу, входящему в комплект приложений в Swift?

Я знаю, что есть несколько вопросов, относящихся к этому, но они находятся в Objective-C.

Как я могу получить доступ к файлу .txt, включенному в мое приложение, используя Swift на реальном iPhone? Я хочу уметь читать и писать. Здесь являются мои файлы проекта, если вы хотите взглянуть. Я рад добавить данные, если это необходимо.

Ответ 1

Просто с помощью поиска в комплекте приложений для ресурса

var filePath = NSBundle.mainBundle().URLForResource("file", withExtension: "txt")

Однако вы не можете писать в него, потому что он находится в каталоге ресурсов приложения, и вы должны создать его в каталоге документов, чтобы записать в него

var documentsDirectory: NSURL?
var fileURL: NSURL?

documentsDirectory = NSFileManager.defaultManager().URLsForDirectory(.DocumentDirectory, inDomains: .UserDomainMask).last!
fileURL = documentsDirectory!.URLByAppendingPathComponent("file.txt")

if (fileURL!.checkResourceIsReachableAndReturnError(nil)) {
    print("file exist")
}else{
    print("file doesnt exist")
    NSData().writeToURL(fileURL!,atomically:true)
}

теперь вы можете получить к нему доступ из fileURL

РЕДАКТИРОВАТЬ - 28 августа 2018

Вот как это сделать в Swift 4.2

var filePath = Bundle.main.url(forResource: "file", withExtension: "txt")

Создать его в каталоге документов

if let documentsDirectory = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).last {
   let fileURL = documentsDirectory.appendingPathComponent("file.txt")
   do {
       if try fileURL.checkResourceIsReachable() {
           print("file exist")
       } else {
           print("file doesnt exist")
           do {
            try Data().write(to: fileURL)
           } catch {
               print("an error happened while creating the file")
           }
       }
   } catch {
       print("an error happened while checking for the file")
   }
}

Ответ 2

Swift 3, на основе Каримса ответьте.

Чтение

Вы можете читать файлы, входящие в комплект приложений через ресурс bundles:

let fileURL = Bundle.main.url(forResource:"filename", withExtension: "txt")

Запись

Однако вы не можете писать там. Вам нужно будет создать копию, желательно в папке "Документы":

func makeWritableCopy(named destFileName: String, ofResourceFile originalFileName: String) throws -> URL {
    // Get Documents directory in app bundle
    guard let documentsDirectory = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).last else {
        fatalError("No document directory found in application bundle.")
    }

    // Get URL for dest file (in Documents directory)
    let writableFileURL = documentsDirectory.appendingPathComponent(destFileName)

    // If dest file doesn’t exist yet
    if (try? writableFileURL.checkResourceIsReachable()) == nil {
        // Get original (unwritable) file’s URL
        guard let originalFileURL = Bundle.main.url(forResource: originalFileName, withExtension: nil) else {
            fatalError("Cannot find original file "\(originalFileName)" in application bundle’s resources.")
        }

        // Get original file’s contents
        let originalContents = try Data(contentsOf: originalFileURL)

        // Write original file’s contents to dest file
        try originalContents.write(to: writableFileURL, options: .atomic)
        print("Made a writable copy of file "\(originalFileName)" in "\(documentsDirectory)\\\(destFileName)".")

    } else { // Dest file already exists
        // Print dest file contents
        let contents = try String(contentsOf: writableFileURL, encoding: String.Encoding.utf8)
        print("File "\(destFileName)" already exists in "\(documentsDirectory)".\nContents:\n\(contents)")
    }

    // Return dest file URL
    return writableFileURL
}

Пример использования:

let stuffFileURL = try makeWritableCopy(named: "Stuff.txt", ofResourceFile: "Stuff.txt")
try "New contents".write(to: stuffFileURL, atomically: true, encoding: String.Encoding.utf8)

Ответ 3

Просто быстрое обновление для использования этого кода с помощью Swift 4:

Bundle.main.url(forResource:"YourFile", withExtension: "FileExtension")

И было обновлено следующее, чтобы записать запись файла:

var myData: Data!

func checkFile() {
    if let documentsDirectory = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).last {
        let fileURL = documentsDirectory.appendingPathComponent("YourFile.extension")
        do {
            let fileExists = try fileURL.checkResourceIsReachable()
            if fileExists {
                print("File exists")
            } else {
                print("File does not exist, create it")
                writeFile(fileURL: fileURL)
            }
        } catch {
            print(error.localizedDescription)
        }
    }
}

func writeFile(fileURL: URL) {
    do {
        try myData.write(to: fileURL)
    } catch {
        print(error.localizedDescription)
    }
}

Этот конкретный пример не самый гибкий, но с небольшой работой вы можете легко передать свои собственные имена файлов, расширения и значения данных.

Ответ 4

Связки читаются только. Вы можете использовать NSBundle.mainBundle().pathForResource для доступа к файлу только для чтения, но для доступа на чтение и запись вам необходимо скопировать документ в папку "Документы" или папку tmp.

Ответ 5

Связки могут быть написаны. Вы можете использовать Bundle.main.path чтобы перезаписать файл, добавив его в Copy Bundles Resource.

Project -> Target -> Build Pharse -> Copy Bundles Resource