Swift - iOS10 - CoreData - Today Widget 中的 FetchRequest 总是 return 0 个元素
Swift - iOS10 - CoreData - FetchRequest in Today Widget always return 0 elements
我尝试在我的主应用程序中使用 CoreData,我想通过 Today Extension 访问它,稍后在我的 WatchApp 中访问它。
我为我的 Today Widget 添加了一个新目标 - 而不是为这两个目标激活了 AppGroups。当然是同一个appGroup。
然后我构建了一个 "SharedCode" 框架。在我的 Main 和 Today 应用程序中共享助手 classes 和我的 CoreDataStack class。
我构建了这篇博文中提到的 CoreDataStack:https://swifting.io/blog/2016/09/25/25-core-data-in-ios10-nspersistentcontainer/
我还添加了这段代码来更改目录:
final class PersistentContainer: NSPersistentContainer {
internal override class func defaultDirectoryURL() -> URL {
var url = super.defaultDirectoryURL()
if let newURL =
FileManager.default.containerURL(
forSecurityApplicationGroupIdentifier: CoreDataServiceConsts.applicationGroupIdentifier) {
url = newURL
}
return url
}
}
这是我的 CoreDataStack 助手 class:
struct CoreDataServiceConsts {
static let applicationGroupIdentifier = "nnnnnnnnn.group.is.xnd.xyz"
static let modelName = "Weight"
}
public class CoreDataStack {
public static let shared = CoreDataStack()
public var errorHandler: (Error) -> Void = {_ in }
//#1
public lazy var persistentContainer: NSPersistentContainer = {
let container = NSPersistentContainer(name: CoreDataServiceConsts.modelName)
container.loadPersistentStores(completionHandler: { [weak self](storeDescription, error) in
if let error = error {
NSLog("CoreData error \(error), \(error._userInfo)")
self?.errorHandler(error)
}
})
return container
}()
//#2
public lazy var viewContext: NSManagedObjectContext = {
return self.persistentContainer.viewContext
}()
//#3
// Optional
public lazy var backgroundContext: NSManagedObjectContext = {
return self.persistentContainer.newBackgroundContext()
}()
//#4
public func performForegroundTask(_ block: @escaping (NSManagedObjectContext) -> Void) {
self.viewContext.perform {
block(self.viewContext)
}
}
//#5
public func performBackgroundTask(_ block: @escaping (NSManagedObjectContext) -> Void) {
self.persistentContainer.performBackgroundTask(block)
}
public func saveContext () {
guard viewContext.hasChanges else { return }
do {
try viewContext.save()
} catch let error as NSError {
print("Unresolved error \(error), \(error.userInfo)")
}
}
}
我为 NSPersistentContainer 构建了一个子class:
final class PersistentContainer: NSPersistentContainer {
internal override class func defaultDirectoryURL() -> URL {
var url = super.defaultDirectoryURL()
if let newURL = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: CoreDataServiceConsts.applicationGroupIdentifier) {
url = newURL
}
return url
}
}
在我的主应用程序中,我使用 NSFetchedResultController - 添加和获取实体工作得很好 :)
现在我想从 CoreData 读取 - 我使用 FetchRequest:
func fetchWeights() {
let fetchRequest: NSFetchRequest<Measure> = Measure.fetchRequest()
let dateSort = NSSortDescriptor(key: #keyPath(Measure.date), ascending: false)
fetchRequest.sortDescriptors = [dateSort]
fetchRequest.returnsObjectsAsFaults = false
do {
let results = try CoreDataStack.shared.viewContext.fetch(fetchRequest)
if results.count > 0 {
// There are found Weigts
print("weights FOUND: \(results.count)")
} else {
// Nil Weights found
print("NO weights: \(results.count)")
}
} catch let error as NSError {
print("Fetch error: \(error) description: \(error.userInfo)")
}
}
没有提取请求总是 return 零个对象。
但是我的 CoreData 中有实体 - 在我的主应用程序中有显示。
有什么想法吗?
我不知道我做错了什么..
谢谢你的帮助。
您已经定义了一个要覆盖的 PersistentContainer
子类
defaultDirectoryURL
,但您的 CoreDataStack
仍在为您的惰性容器使用 NSPersistentContainer
类型中的构建。
这应该可以解决问题:
public class CoreDataStack {
public static let shared = CoreDataStack()
public var errorHandler: (Error) -> Void = {_ in }
//#1
public lazy var persistentContainer: PersistentContainer = {
let container = PersistentContainer(name: CoreDataServiceConsts.modelName)
container.loadPersistentStores(completionHandler: { [weak self](storeDescription, error) in
if let error = error {
NSLog("CoreData error \(error), \(error._userInfo)")
self?.errorHandler(error)
}
})
return container
}()
干杯
迈克尔
我尝试在我的主应用程序中使用 CoreData,我想通过 Today Extension 访问它,稍后在我的 WatchApp 中访问它。
我为我的 Today Widget 添加了一个新目标 - 而不是为这两个目标激活了 AppGroups。当然是同一个appGroup。
然后我构建了一个 "SharedCode" 框架。在我的 Main 和 Today 应用程序中共享助手 classes 和我的 CoreDataStack class。
我构建了这篇博文中提到的 CoreDataStack:https://swifting.io/blog/2016/09/25/25-core-data-in-ios10-nspersistentcontainer/
我还添加了这段代码来更改目录:
final class PersistentContainer: NSPersistentContainer {
internal override class func defaultDirectoryURL() -> URL {
var url = super.defaultDirectoryURL()
if let newURL =
FileManager.default.containerURL(
forSecurityApplicationGroupIdentifier: CoreDataServiceConsts.applicationGroupIdentifier) {
url = newURL
}
return url
}
}
这是我的 CoreDataStack 助手 class:
struct CoreDataServiceConsts {
static let applicationGroupIdentifier = "nnnnnnnnn.group.is.xnd.xyz"
static let modelName = "Weight"
}
public class CoreDataStack {
public static let shared = CoreDataStack()
public var errorHandler: (Error) -> Void = {_ in }
//#1
public lazy var persistentContainer: NSPersistentContainer = {
let container = NSPersistentContainer(name: CoreDataServiceConsts.modelName)
container.loadPersistentStores(completionHandler: { [weak self](storeDescription, error) in
if let error = error {
NSLog("CoreData error \(error), \(error._userInfo)")
self?.errorHandler(error)
}
})
return container
}()
//#2
public lazy var viewContext: NSManagedObjectContext = {
return self.persistentContainer.viewContext
}()
//#3
// Optional
public lazy var backgroundContext: NSManagedObjectContext = {
return self.persistentContainer.newBackgroundContext()
}()
//#4
public func performForegroundTask(_ block: @escaping (NSManagedObjectContext) -> Void) {
self.viewContext.perform {
block(self.viewContext)
}
}
//#5
public func performBackgroundTask(_ block: @escaping (NSManagedObjectContext) -> Void) {
self.persistentContainer.performBackgroundTask(block)
}
public func saveContext () {
guard viewContext.hasChanges else { return }
do {
try viewContext.save()
} catch let error as NSError {
print("Unresolved error \(error), \(error.userInfo)")
}
}
}
我为 NSPersistentContainer 构建了一个子class:
final class PersistentContainer: NSPersistentContainer {
internal override class func defaultDirectoryURL() -> URL {
var url = super.defaultDirectoryURL()
if let newURL = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: CoreDataServiceConsts.applicationGroupIdentifier) {
url = newURL
}
return url
}
}
在我的主应用程序中,我使用 NSFetchedResultController - 添加和获取实体工作得很好 :)
现在我想从 CoreData 读取 - 我使用 FetchRequest:
func fetchWeights() {
let fetchRequest: NSFetchRequest<Measure> = Measure.fetchRequest()
let dateSort = NSSortDescriptor(key: #keyPath(Measure.date), ascending: false)
fetchRequest.sortDescriptors = [dateSort]
fetchRequest.returnsObjectsAsFaults = false
do {
let results = try CoreDataStack.shared.viewContext.fetch(fetchRequest)
if results.count > 0 {
// There are found Weigts
print("weights FOUND: \(results.count)")
} else {
// Nil Weights found
print("NO weights: \(results.count)")
}
} catch let error as NSError {
print("Fetch error: \(error) description: \(error.userInfo)")
}
}
没有提取请求总是 return 零个对象。 但是我的 CoreData 中有实体 - 在我的主应用程序中有显示。
有什么想法吗? 我不知道我做错了什么.. 谢谢你的帮助。
您已经定义了一个要覆盖的 PersistentContainer
子类
defaultDirectoryURL
,但您的 CoreDataStack
仍在为您的惰性容器使用 NSPersistentContainer
类型中的构建。
这应该可以解决问题:
public class CoreDataStack {
public static let shared = CoreDataStack()
public var errorHandler: (Error) -> Void = {_ in }
//#1
public lazy var persistentContainer: PersistentContainer = {
let container = PersistentContainer(name: CoreDataServiceConsts.modelName)
container.loadPersistentStores(completionHandler: { [weak self](storeDescription, error) in
if let error = error {
NSLog("CoreData error \(error), \(error._userInfo)")
self?.errorHandler(error)
}
})
return container
}()
干杯 迈克尔