我正在逐步解决免费和公开可用的Hacking With Swift课程。不幸的是,有关“核心数据”的一章并未说明如何以一对多关系使用NSSet中的对象。我查看了文档,试图找出如何使用生成的访问器,但收效甚微。如果有人能够解释如何执行此操作,我将非常感谢您了解它!
One To Many Relationship
到目前为止我尝试过的一切都失败了!
func deleteCandy (at offsets: IndexSet) {
for offset in offsets {
let candy = countries[offset].candyArray[offset]
moc.delete(candy)
if self.moc.hasChanges{
try? moc.save()
}
}
}
}
班级:
扩展国家/地区{
@nonobjc public class func fetchRequest() -> NSFetchRequest<Country> {
return NSFetchRequest<Country>(entityName: "Country")
}
@NSManaged public var fullName: String?
public var wrappedFullName: String {
fullName ?? "Unknown Full Name"
}
@NSManaged public var shortName: String?
public var wrappedShortName: String{
shortName ?? "Unknown Short Name"
}
public var candyArray:[Candy] {
let set = candy as? Set<Candy> ?? []
return set.sorted {
$0.wrappedName < $1.wrappedName
}
}
@NSManaged public var candy: NSSet?
}
// MARK:生成的糖果访问器 扩展国家/地区{
@objc(addCandyObject:)
@NSManaged public func addToCandy(_ value: Candy)
@objc(removeCandyObject:)
@NSManaged public func removeFromCandy(_ value: Candy)
@objc(addCandy:)
@NSManaged public func addToCandy(_ values: NSSet)
@objc(removeCandy:)
@NSManaged public func removeFromCandy(_ values: NSSet)
}
ContentView @Environment(.managedObjectContext)var moc @FetchRequest(entity:Country.entity(),sortDescriptors:[])var国家:FetchedResults
var body: some View {
VStack {
List {
ForEach(countries, id: \.self) { country in
Section(header: Text(country.wrappedFullName)) {
ForEach(country.candyArray, id: \.self) { candy in
Text(candy.wrappedName)
}
}
}.onDelete(perform: deleteCandy)
}
Button("Add") {
let candy1 = Candy(context: self.moc)
candy1.name = "Mars"
candy1.origin = Country(context: self.moc)
candy1.origin?.shortName = "UK"
candy1.origin?.fullName = "United Kingdom"
let candy2 = Candy(context: self.moc)
candy2.name = "KitKat"
candy2.origin = Country(context: self.moc)
candy2.origin?.shortName = "UK"
candy2.origin?.fullName = "United Kingdom"
let candy3 = Candy(context: self.moc)
candy3.name = "Twix"
candy3.origin = Country(context: self.moc)
candy3.origin?.shortName = "UK"
candy3.origin?.fullName = "United Kingdom"
let candy4 = Candy(context: self.moc)
candy4.name = "Toblerone"
candy4.origin = Country(context: self.moc)
candy4.origin?.shortName = "CH"
candy4.origin?.fullName = "Switzerland"
try? self.moc.save()
}
}
}
func deleteCandy (at offsets: IndexSet) {
for offset in offsets {
let candy = countries[offset].candyArray[offset]
moc.delete(candy)
if self.moc.hasChanges{
try? moc.save()
}
}
}
} The UI
答案 0 :(得分:0)
我偶然发现了相同的问题,并使用了从NSManagedObject子类生成的 removeFromCandy(_ value:Candy)函数解决了这个问题。
var body: some View {
VStack {
List {
ForEach(countries, id: \.self) { country in
Section(header: Text(country.wrappedFullName)) {
ForEach(country.candyArray, id: \.self) { candy in
Text(candy.wrappedName)
}.onDelete {indexSet in
let deleteItem = self.country.candyArray[indexSet.first!]
self.country.removeFromCandy(deleteItem)
self.moc.delete(deleteItem)
do {
try self.moc.save()
} catch {
print(error)
}
}
}
}
}
Button("Add") {
let candy1 = Candy(context: self.moc)
candy1.name = "Mars"
candy1.origin = Country(context: self.moc)
candy1.origin?.shortName = "UK"
candy1.origin?.fullName = "United Kingdom"
try? self.moc.save()
}
}
}
答案 1 :(得分:0)
最后,通过DMG的闭包传递的此解决方案非常有效:
func deleteCandies(at offsets: IndexSet, from country: Country) {
for offset in offsets.sorted().reversed() {
let candyToDelete = country.candyArray[offset]
country.removeFromCandy(candyToDelete)
moc.delete(candyToDelete)
}
if moc.hasChanges{
try? moc.save()
}
}
VStack {
List {
ForEach(countries, id: \.self) { country in
Section(header: Text(country.wrappedFullName)) {
ForEach(country.candyArray, id: \.self) { candy in
Text(candy.wrappedName)
}
.onDelete(perform: { offsets in
self.deleteCandies(at: offsets, from: country)
})
}
}
}