可以将文章内容翻译成中文,广告屏蔽插件可能会导致该功能失效(如失效,请关闭广告屏蔽插件后再试):
问题:
I have been trying to add core data. And every time I got the same error:
error: filename "EntityName +CoreDataClass.swift" used twice: '/Users/userName/Desktop/Development/MyApp/AppName/EntityName +CoreDataClass.swift' and '/Users/userName/Library/Developer/Xcode/DerivedData/AppName-dgwzrmxsetzvtedibxrazuutjwnh/Build/Intermediates/AppName.build/Debug-iphoneos/AppName.build/DerivedSources/CoreDataGenerated/Model/EntityName +CoreDataClass.swift'
I add core data using the following steps:
1.New file/ DataModel; save it in the root dir of my project
select Model.xcdatamodeld and add entity, add several attributes, save, editor/create NSManagedObjectClass Subclass.
As a result I observe 4 new files in navigator: Model.xcdatamodeld, EntityName+CoreDataProperties.swift, EntityName +CoreDataClass.swift, _COREDATA_DATAMODELNAME_+CoreDataModel.swift
their content: _COREDATA_DATAMODELNAME_+CoreDataModel.swift:
import Foundation import CoreData ___COREDATA_DATAMODEL_MANAGEDOBJECTCLASSES_IMPLEMENTATIONS___
EntityName +CoreDataClass.swift:
import Foundation import CoreData class EntityName: NSManagedObject { }
EntityName+CoreDataProperties.swift:
import Foundation import CoreData extension EntityName { @nonobjc class func fetchRequest() -> NSFetchRequest { return NSFetchRequest(entityName: "EntityName"); } @NSManaged var str: String? }
What I have tried:
1. Clean build, remove DerivedData, delete content of var/folders, restart
2. Delete generated files, displayed in navigator
All my efforts were out of luck.
What I am doing wrong?
回答1:
Xcode 8 includes automatic NSManagedObject
class generation when the model file uses the Xcode 8 file format. If you create your own subclass files, you're creating duplicates. The second file in the error message, in DerivedSources
, is the one that Xcode created automatically.
If the automatically generated files do what you need, just stop creating your own and you'll be OK.
If you want to create your own subclasses instead, you can either
- Set the "tools version" for the model file to be Xcode 7.3 or earlier to disable all code generation (this doesn't seem to change anything meaningful about the actual file contents), or
- Disable automatic generation for each entity individually by setting the "Codegen" setting to "Manual/None" for the entity.
回答2:
There are two bugs in XCode 8 here:
1 - If you change the Codegen dropdown, it's new value isn't saved in Model.xcdatamodel. You have to change something else to get it to save. For example change the class name; build; change the class name back; build again.
2 - The generated code is placed in DerivedData in the Intermediates folder, but it only happens if the folder doesn't already exist. The workaround is to do a clean then a build.
回答3:
I'm actually having the same problem (using Swift) and suspect it is a bug. If I understand correctly the OP is using Xcode's autogen NSManagedObject subclasses and is not then subsequently creating additional (superfluous) subclasses which appears to be a source of some confusion.
Steps to reproduce:
Create a new project, single view application. Tick "Use Core Data'
Create an entity in the model entity, add properties, ensure file is saved (sometimes Xcode 8 Beta throws the data if not explicitly saved)
- Select Editor -> Created NSManagedObject subclass. Tick relevant boxes
Xcode creates 3 files:
COREDATA_DATAMODELNAME_+CoreDataModel.swift. This file is corrupted and contains non-valid entries following statements to import Foundation and CoreData - the project will not compile unless this file is deleted
import Foundation import CoreData ___COREDATA_DATAMODEL_MANAGEDOBJECTCLASSES_IMPLEMENTATIONS___
EntityName+CoreDataClass.swift
EntityName+CoreDataProperties.swift
Although the editor doesn't flag any errors at this point, attempts to compile fail for the reasons listed by the OP (namely missing files with the same names but with a '.' prefix in the DerivedData folder)
If you create the NSManagedObject subclasses manually after creating your model without using Xcode's evidently bugged auto-gen, there are no issues. A bit more typing but a lot more reliable! However, you will need to start from a 'clean' project (i.e. before you attempted to auto-generate the subclasses) otherwise the error persists. Cleaning out Derived Data won't help sadly.
***** UPDATED ***** There does appear to be something rather odd going on and there does appear to be silent code generation as originally suggested (apologies) but this is very different behaviour from what one would expect. Furthermore, this code is not visible in the editor (which seems a little pointless as well as confusing). It is created in a subfolder of DerivedData > Build > Intermediates > .Build.
I can completely see why the OP was confused (as was I!) For what it's worth this 'feature' is undoubtedly an attempt to be helpful but is somewhat confusing if you're used to previous behaviour and you are offered an option to generate a visible and editable duplicate from the main menu.
So, select 'Manual/None' in the Codegen window (shown below) and then you can either use the auto-gen option in the menu bar (after deleting the 'junk') or write your own code.
回答4:
Follow these steps,
Disable automatic generation for each entity individually by setting the "Codegen" setting to "Manual/None" for the entity.
Change entity name to something else to get it to save. For example change the class name; build; change the class name back; build again.
回答5:
I got this error to,
just the change the tool version to Xcode 7.3 in your file .xcdatamodeld
and it works fine!
回答6:
For everyone having trouble getting autogen to work:
I had to set the "com.apple.syncservices.Syncable" to "NO" on the User Info settings for the Entity.
Maybe that helps out.
回答7:
Tom Harrington Answer is correct. However, there is a way to add your own functions and/or vars without doing any of the two mentioned options.
Just create an extension to the class. (make sure you name the swift file something different than the standard auto-generated NSManagesObject files.)
For example. If you have an entity called MyEntity
you could add a swift file called MyEntityExtension.swift
which could look something like this:
import Foundation import CoreData import UIKit extension MyEntity { var color: UIColor { get { return self.colorValue as! UIColor } set { if newValue.isMember(of: UIColor.self) { self.colorValue = newValue } } } }
回答8:
After trying a handful of posted answers, rebooting my machine worked for me.