CoreDataとSwiftを使用して、リレーションシップにオブジェクトを追加することに頭を悩ませてきました。私は途方に暮れています、なぜ私のコードが機能しないのか理解できません。 「チーム」に「イベント」を追加しようとしています。受け入れられた回答(機能するはずです)とコード(機能しない)の違いがわかりません。
Teams.Swift:
import Foundation
import CoreData
class Teams: NSManagedObject {
@NSManaged var teamName: String
@NSManaged var matches: NSSet
}
extension Teams {
func addEventToTeam(event:Event) {
//self.mutableSetValueForKeyPath("matches").addObject(event)
var matchez: NSMutableSet
matchez = self.mutableSetValueForKey("matches")
matchez.addObject(event)
//var manyRelation = self.valueForKeyPath("matches") as NSMutableSet
//manyRelation.addObject(event)
}
func getTeamName() -> String {
return teamName
}
}
コードの呼び出し(構成ビューから):
import UIKit
import CoreData
class DetailViewController: UIViewController, NSFetchedResultsControllerDelegate {
var managedObjectContext: NSManagedObjectContext? = nil
@IBOutlet weak var detailDescriptionLabel: UILabel!
var detailItem: AnyObject? {
didSet {
// Update the view.
self.configureView()
}
}
func configureView() {
// Update the user interface for the detail item.
if let detail: Event = (self.detailItem as? Event) {
//if let detail: AnyObject = self.detailItem {
if let label = self.detailDescriptionLabel {
label.text = detail.valueForKey("timeStamp").description
self.insertNewObject(self);
label.text = String(detail.getNumberOfTeams())
//detail.getTeams().
var hej: Array<Teams>
hej = detail.getTeams()
label.text = "tjosan"
for tmpTeam : Teams in hej {
label.text = label.text + ", " + tmpTeam.getTeamName()
}
}
}
if true {
}
}
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view, typically from a nib.
self.configureView()
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
var fetchedResultsController: NSFetchedResultsController {
if _fetchedResultsController != nil {
return _fetchedResultsController!
}
let fetchRequest = NSFetchRequest()
// Edit the entity name as appropriate.
let team = NSEntityDescription.entityForName("Teams", inManagedObjectContext: self.managedObjectContext)
fetchRequest.entity = team
// Set the batch size to a suitable number.
fetchRequest.fetchBatchSize = 20
// Edit the sort key as appropriate.
let sortDescriptor = NSSortDescriptor(key: "teamName", ascending: false)
let sortDescriptors = [sortDescriptor]
fetchRequest.sortDescriptors = [sortDescriptor]
// Edit the section name key path and cache name if appropriate.
// nil for section name key path means "no sections".
let aFetchedResultsController = NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: self.managedObjectContext, sectionNameKeyPath: nil, cacheName: "Master")
aFetchedResultsController.delegate = self
_fetchedResultsController = aFetchedResultsController
var error: NSError? = nil
if !_fetchedResultsController!.performFetch(&error) {
// Replace this implementation with code to handle the error appropriately.
// abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
//println("Unresolved error \(error), \(error.userInfo)")
abort()
}
return _fetchedResultsController!
}
var _fetchedResultsController: NSFetchedResultsController? = nil
func insertNewObject(sender: AnyObject) {
let context = self.fetchedResultsController.managedObjectContext
let team = self.fetchedResultsController.fetchRequest.entity
let newManagedObject = NSEntityDescription.insertNewObjectForEntityForName(team.name, inManagedObjectContext: context) as Teams
// If appropriate, configure the new managed object.
// Normally you should use accessor methods, but using KVC here avoids the need to add a custom class to the template.
newManagedObject.setValue("Lagur Namnurk", forKey: "teamName")
newManagedObject.addEventToTeam(self.detailItem as Event)
// Save the context.
var error: NSError? = nil
if !context.save(&error) {
// Replace this implementation with code to handle the error appropriately.
// abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
//println("Unresolved error \(error), \(error.userInfo)")
abort()
}
}
}
エラーメッセージ:
2014-08-13 18:38:46.651 Score Calculator 2[10538:829319] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '*** -[NSSet intersectsSet:]: set argument is not an NSSet'
*** First throw call stack:
(
0 CoreFoundation 0x00000001028a53e5 __exceptionPreprocess + 165
1 libobjc.A.dylib 0x00000001043b8967 objc_exception_throw + 45
2 CoreFoundation 0x000000010280fc6c -[NSSet intersectsSet:] + 940
3 Foundation 0x0000000102d0c4a6 NSKeyValueWillChangeBySetMutation + 156
4 Foundation 0x0000000102c804fa NSKeyValueWillChange + 386
5 Foundation 0x0000000102d0c3fb -[NSObject(NSKeyValueObserverNotification) willChangeValueForKey:withSetMutation:usingObjects:] + 310
6 CoreData 0x00000001024178d7 -[NSManagedObject(_NSInternalMethods) _includeObject:intoPropertyWithKey:andIndex:] + 551
7 CoreData 0x0000000102418294 -[NSManagedObject(_NSInternalMethods) _maintainInverseRelationship:forProperty:forChange:onSet:] + 276
8 CoreData 0x0000000102416ef2 -[NSManagedObject(_NSInternalMethods) _didChangeValue:forRelationship:named:withInverse:] + 562
9 Foundation 0x0000000102c835d6 NSKeyValueNotifyObserver + 356
10 Foundation 0x0000000102c827fd NSKeyValueDidChange + 466
11 Foundation 0x0000000102d0c7ee -[NSObject(NSKeyValueObserverNotification) didChangeValueForKey:withSetMutation:usingObjects:] + 118
12 CoreData 0x00000001024180b0 -[NSManagedObject didChangeValueForKey:withSetMutation:usingObjects:] + 80
13 CoreData 0x000000010242fa11 -[_NSNotifyingWrapperMutableSet addObject:] + 161
編集:いくつかの説明。チームとイベントには、複数対複数の順序付けられていない関係があります。
はい!答えが見つかりました!
私は、Events.Swiftクラス(関係の反対側)に新しい関数を作成しました。
私は次のコードを書きました:
import Foundation
import CoreData
class Event: NSManagedObject {
@NSManaged var timeStamp: NSDate
@NSManaged var teams: NSSet
}
extension Event {
func addTeamToEvent(team:Teams) {
var teamz = self.mutableSetValueForKey("teams")
teamz.addObject(team)
}
func getNumberOfTeams() -> Int {
return self.teams.count;
}
func getTeams() -> [Teams] {
var tmpsak: [Teams]
tmpsak = self.teams.allObjects as [Teams]
tmpsak = self.teams.allObjects as [Teams]
return tmpsak
}
}
私はそれは無関係だと思いました。ただし、getTeamsの名前をgetTeamsAsArrayに変更すると、問題が解消されました。 CoreDataは、関係のもう一方の端を埋めるときに、getTeams()という組み込み関数を使用していると思います(他のクラスはTeamsと呼ばれていたため)。誤って上書き(?)して失敗しました。
あなたの提案をありがとう、そして私はこれが他の誰かに役立つことを願っています!
やや関連性のあるメモで、同様の症状を持つバグが数年前に特定されました(そしてまだ存在しているようです)。これは、ordered多対多の関係を使用すると自動生成されたコードに表示されます。 。
Objective Cで生成されたNSManagedObjectサブクラスを使用する場合、単一のオブジェクトを追加するためのカスタムメソッドが含まれています。これらは試行錯誤されており、現在Swiftにあるものよりも優れているようです。
理論的には、作成した新しいセットで新しいセットを設定するだけで済みます。 1つのオプションは次のとおりです。
var matchobjs = matches.allObjects as [Event]
matchobjs.append(event)
matches = NSSet(array: matchobjs)
ただし、エラーは次のようになります この投稿:
私には、多対多の関係で再び何か怪しいことが起こっているように見えます...