在 Swift 中复制 Sqlite 数据库无法正常工作

Posted

技术标签:

【中文标题】在 Swift 中复制 Sqlite 数据库无法正常工作【英文标题】:Copy Sqlite DataBase in Swift does not work properly 【发布时间】:2017-01-10 07:50:47 【问题描述】:

我正在为目标 c 使用以下代码来复制 sqlite 数据库,它工作正常。但是当我将此代码转换为 swift 时,它会在 Bool 类型上显示错误。

这是目标c代码

  - (void) copyDatabaseIfNeeded 

  //Using NSFileManager we can perform many file system operations.
   NSFileManager *fileManager = [NSFileManager defaultManager];
   NSError *error;

   NSString *dbPath = [self getDBPath];
   BOOL success = [fileManager fileExistsAtPath:dbPath];

  if(!success) 

   NSString *defaultDBPath = [[[NSBundle mainBundle] resourcePath] stringByAppendingPathComponent:@"database.sqlite"];
   success = [fileManager copyItemAtPath:defaultDBPath toPath:dbPath error:&error];

   if (!success)
      NSAssert1(0, @"Failed to create writable database file with message '%@'.", [error localizedDescription]);
   
   

  - (NSString *) getDBPath
     

   NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory , NSUserDomainMask, YES);
   NSString *documentsDir = [paths objectAtIndex:0];
   return [documentsDir   stringByAppendingPathComponent:@"database.sqlite"];
  

这是导致问题的 Swift 的 CopyDataBase。

   var fileManager = FileManager.default
   var error: Error!
   var dbPath = self.getDBPath()
   var success = fileManager.fileExists(atPath: dbPath)
   if !success 
   var defaultDBPath = URL(fileURLWithPath: Bundle.main.resourcePath!).appendingPathComponent("CapalinoDataBase.sqlite").absoluteString
   do 
    success = try fileManager.copyItem(atPath: defaultDBPath, toPath: dbPath)
    
    catch 
    
    if !success 
    assert(false, "Failed to create writable database file with message '\(error.localizedDescription)'.")
    
      

【问题讨论】:

【参考方案1】:

请试试这个。

func copyDatabse() 

        let fileMgr = FileManager.default

        if let path = Bundle.main.path(forResource: "db", ofType:"sqlite") 

            do 
                try fileMgr.copyItem(atPath: path, toPath: dbPath())
                print("Copy success")
            
            catch 
                print(error.localizedDescription )
            
        
    


    func dbPath() -> String 

        let dirPaths =  NSSearchPathForDirectoriesInDomains(.documentDirectory,.userDomainMask, true)
        let docsDir = dirPaths[0]

        let destPath = (docsDir as NSString).appendingPathComponent("/db.sqlite")

        return destPath
    

【讨论】:

【参考方案2】:

在 swift 中使用单吨类使用 SQLIte 的最佳方式。

Download example

func methodToCreateDatabase() -> NSURL? 

    let fileManager = NSFileManager.defaultManager()

    let urls = fileManager.URLsForDirectory(.DocumentDirectory, inDomains: .UserDomainMask)

    if let documentDirectory:NSURL = urls.first  // No use of as? NSURL because let urls returns array of NSURL

        // exclude cloud backup
        do 
            try documentDirectory.setResourceValue(true, forKey: NSURLIsExcludedFromBackupKey)
         catch _
            print("Failed to exclude backup")
        

        // This is where the database should be in the documents directory
        let finalDatabaseURL = documentDirectory.URLByAppendingPathComponent("contact.db")

        if finalDatabaseURL.checkResourceIsReachableAndReturnError(nil) 
            // The file already exists, so just return the URL
            return finalDatabaseURL
         else 
            // Copy the initial file from the application bundle to the documents directory
            if let bundleURL = NSBundle.mainBundle().URLForResource("contact", withExtension: "db") 

                do 
                    try fileManager.copyItemAtURL(bundleURL, toURL: finalDatabaseURL)
                 catch _ 
                    print("Couldn't copy file to final location!")
                

             else 
                print("Couldn't find initial database in the bundle!")
            
        
     else 
        print("Couldn't get documents directory!")
    

    return nil

【讨论】:

【参考方案3】:
Please try this one it is working on swift 3.0

func copyDatabaseIfNeeded() 
    //Using NSFileManager we can perform many file system operations.
    let fileManager = FileManager.default
    let error: Error?
    let dbPath: String = self.getDBPath()
    var success: Bool = fileManager.fileExists(atPath: dbPath)
    if !success 
        let defaultDBPath: String = URL(fileURLWithPath: (Bundle.main.resourcePath)!).appendingPathComponent("database.sqlite").absoluteString

        do 
            success = try fileManager.copyItem(atPath: defaultDBPath, toPath: dbPath) as Any as! Bool
        
        catch let error as NSError 
            print("Ooops! Something went wrong: \(error)")
        

        if !success 
            assert(false, "Failed to create writable database file with message '\(error?.localizedDescription)'.")
        
    


func getDBPath() -> String 
    let paths: [Any] = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true)
    let documentsDir: String = paths[0] as! String
    return URL(fileURLWithPath: documentsDir).appendingPathComponent("database.sqlite").absoluteString

【讨论】:

以上是关于在 Swift 中复制 Sqlite 数据库无法正常工作的主要内容,如果未能解决你的问题,请参考以下文章

无法使用 SQLite.swift 构建命令行项目

Android SQLite:W / System.err:java.io.FileNotFoundException无法从资产复制数据库

如何使用 Swift 从 CoreData sqlite 表中获取特定的行数据

斯威夫特和 SQLite

在tableView swift的单元格页脚视图中复制行

Swift SQLITE3 插入/选择问题