swift - 当尝试从 iOS9 的数据库中获取超过 512 个项目时,该方法会挂起。但它适用于 iOS10 及更高版本

标签 swift sqlite cocoapods ios9 fetch

我有一个从数据库获取数据的方法,该方法适用于 iOS 10 及更高版本的所有版本,但对于 iOS 9,如果我尝试获取超过 512 个项目,该方法会在调用 fetch 后挂起,代码如下:

internal func getAll(_ dictId:Int) -> [Word]? {
    let dictFetch = NSFetchRequest<NSFetchRequestResult>(entityName: "Words")
    dictFetch.predicate = NSPredicate(format: "dictionary = %@", dictId)
    dictFetch.fetchLimit = 513
        do {
            print("getAll before fetching")
            let res = try self.moc_.fetch(dictFetch) as? [Word]
            print("getAll after fetching")
            return res
        } catch {
            print("error: \(error)")
        }
    return nil
}

如果我将 fetchLimit 设置为 512 或更少,那么此代码可以正常工作并且我会得到日志:

getAll before fetching
getAll after fetching

但是如果我将 fetchLimit 设置为 513 或更多,那么这段代码会给出以下日志:

getAll before fetching

没有任何错误,只是挂起。

为了测试,我只在AppDelegate中调用了这个方法(项目中没有任何其他方法)。结果一样。我也尝试过异步调用它。相同的。 此外,我还更改了“字典”的输入值并进行排序,以检查这是否是由于表中的某个元素造成的。同样,不超过 512 个元素。 表“Word”包含 6000 多个项目,在 iOS 10 及更高版本中一切正常。它也挂起而没有“fetchLimit”。 请给我任何建议。

最佳答案

我找到了解决问题的方法。对于 iOS 9 及更低版本的设备,创建了一个循环,从数据库中获取 500 个项目,直到获取所有必要的数据。 也许有人会派上用场。

    // CHECK VERSION OF IOS
let fetchLimit = NSString(string: UIDevice.current.systemVersion).doubleValue >= 10 ? -1 : 500

internal func getAll(_ dictId:Int) -> [Word]? {
    return getAll(dict, nil, nil)
}

internal func getAll(_ dict:Dictionary,_ aRes:[Word]?,_ lastReg:NSDate?) -> [Word]? {
    let dictFetch = NSFetchRequest<NSFetchRequestResult>(entityName: "Words")
    if lastReg != nil {
        dictFetch.predicate = NSPredicate(format: "dictionary = %@ AND registration > %@", dict, lastReg!)
    } else {
        dictFetch.predicate = NSPredicate(format: "dictionary = %@", dictId)
    }
    let sort = NSSortDescriptor(key: "registration", ascending: true)
    dictFetch.sortDescriptors = [sort]
    if fetchLimit > 0 {
        dictFetch.fetchLimit = fetchLimit
    }

    do {
        if let res = try self.moc_.fetch(dictFetch) as? [Word] {
            var aRes_ = aRes != nil ? aRes! + res : res
            if res.count == fetchLimit {
                return self.getAll(dict, aRes_, aRes_[aRes_.count - 1].registration)
            }
            return aRes_
        }
        return aRes
    } catch {
        Titles.l(self, "getAll_, error:  \(error)")
    }
    return nil
}

关于swift - 当尝试从 iOS9 的数据库中获取超过 512 个项目时,该方法会挂起。但它适用于 iOS10 及更高版本,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51636244/

相关文章:

swift - 如何为下面提到的 IBAction 编写 Kiwi 测试用例

sqlite - 如何使用jOOQ通过SQLite正确生成日期和时间类型

swift - 多次错误 "failed to import bridging header"

ios - Pod 安装了 Alamofire 4.4,但是 Xcode 8.3 没有自动完成类

swift - Xcode 10 : how to open swift code use "open quickly (shift+command+O)"?

ios - 使用 Bundle Identifier(无 url scheme)点击按钮打开应用程序

python-3.x - cookie sqlite错误1555如何解决? ( python 和 Selenium )

ios - SCLAlertView 添加按钮问题

swift - 为什么 'throws' 在 Swift 中不是类型安全的?

ruby-on-rails - 这个 Rails 文件存储在哪里?数据库/开发.sqlite3