core data提供了一种内存持久化存储和三种磁盘存储持久化存储的方式。
注意:xml存储在ios是不支持的
Store type | Speed | Object graph in memory | Other factors |
---|---|---|---|
XML (atomic) | Slow | Whole | Externally parsable |
Binary (atomic) | Fast | Whole | N/A |
SQLite | Fast | Partial | N/A |
In-memory | Fast | Whole | No on-disk storage required |
Important
尽管core data支持sqlite作为一种存储类型,存储形式-像其它的native core data存储-是私有的。你不能使用native sqlite api来创建sqlite数据库。也不能直接通过core data来使用这个数据库。你也不能对现存的core data sqlite 存储使用native sqlite api。如果你有一个现存的sqlite数据库,你需要把他import到core data存储。
core data对于来源于不信任元数据的持久化存储是没有保护机制的。并且也不能侦测是否文件被恶意修改。sqlite较为安全,但也不是绝对的。为了安全考虑,应该使用encrypted加密等方式。
支持的排序selectors包含compare:
,caseInsensitiveCompare:
, localizedCompare:
, localizedCaseInsensitiveCompare:
,和localizedStandardCompare:
.后者推荐使用。你不能对sqlite存储的临时属性进行排序。
There are additional constraints on the predicates you can use with the SQLite store:
You cannot necessarily translate arbitrary SQL queries into predicates.
You can have only one to-many element in a key path in a predicate.
For example, no toOne.toMany.toMany
, or toMany.toOne.toMany
type constructions (they evaluate to sets of sets) are allowed. As a consequence, in any predicate sent to the SQL store, there may be only one operator (and one instance of that operator) from ALL
, ANY
, and IN
.
CoreData supports a noindex:
that can be used to drop indices in queries passed to SQLite. This is done primarily for performance reasons: SQLite uses a limited number of indices per query, and noindex: allows the user to preferentially specify which indexes should not be used. See NSPredicate
documentation regarding function expressions.
To summarize: Byte-range locking file systems have the best concurrent read/write support; these include HFS+, AFP, and NFS. File systems with simple file locking are also supported, but do not allow for as much concurrent read/write access by multiple processes. Simple file locking systems include SMB and DOS. The SQLite store does not support writing to WebDAV file-systems.
Simply deleting a record from a SQLite store does not necessarily result in a reduction in the size of the file. If enough items are removed to free up a page in the database file, SQLite’s automatic database vacuuming will reduce the size of the file as it rearranges the data to remove that page. Similarly, the file size is reduced if you remove an item that itself occupies multiple pages (such as a thumbnail image).
An SQLite file is organized as a collection of pages. The data within those pages is managed through B-trees, not as simple fixed-length records. This format is more efficient for searching and for overall storage, because it allows SQLite to optimize how it stores both data and indexes in a single file. This format is also the foundation of SQLite’s data integrity (transaction and journaling) mechanism. However, the cost of this design is that some delete operations may leave holes in the file and impact read and write performance. If you delete some data and add other data, the holes left by the deleted data may be filled by the added data, or the file may be vacuumed to compact its data, whichever SQLite considers most appropriate based on the operations you’re performing.
When Core Data saves a SQLite store, SQLite updates just part of the store file. Loss of that partial update would be catastrophic, so ensure that the file is written correctly before your application continues. Unfortunately, doing partial file updates means that in some situations saving even a small set of changes to a SQLite store can take considerably longer than saving to, say, an XML store. For example, where saving to an XML file might take less than a hundredth of a second, saving to a SQLite store may take almost half a second. This data loss risk is not an issue for XML or binary stores. Because writes to these stores are typically atomic, it is less likely that data loss involves corruption of the file, and the old file is not deleted until the new has been successfully written.
Important
In OS X the fsync
command does not guarantee that bytes are written, so SQLite sends a F_FULLFSYNC
request to the kernel to ensure that the bytes are actually written through to the drive platter. This request causes the kernel to flush all buffers to the drives and causes the drives to flush their track caches. Without this, there is a significantly large window of time within which data will reside in volatile memory. If system failure occurs you risk data corruption.
You can migrate a store from one type or location to another (for example, for a Save As operation) using the NSPersistentStoreCoordinator
method migratePersistentStore:toURL:options:withType:error:
. After invocation of this method, the original store is removed from the coordinator; thus the persistent store is no longer a useful reference. The method is illustrated in the following code fragment, which shows how you can migrate a store from one location to another. If the old store type is XML, then the example also converts the store to SQLite.
NSPersistentStoreCoordinator *psc = [[self managedObjectContext] persistentStoreCoordinator];
NSURL *oldURL = <#URL identifying the location of the current store#>;
NSURL *newURL = <#URL identifying the location of the new store#>;
NSError *error = nil;
NSPersistentStore *xmlStore = [psc persistentStoreForURL:oldURL];
NSPersistentStore *sqLiteStore = [psc migratePersistentStore:xmlStore
toURL:newURL
options:nil
withType:NSSQLiteStoreType
error:&error];
guard let psc = managedObjectContext.persistentStoreCoordinator else {
fatalError("Failed to load persistent store")
}
let oldURL = NSURL(fileURLWithPath: "oldURL")
let newURL = NSURL(fileURLWithPath: "newURL")
guard let xmlStore = psc.persistentStoreForURL(oldURL) else {
fatalError("Failed to reference old store")
}
do {
try psc.migratePersistentStore(xmlStore, toURL: newURL, options:nil, withType:NSSQLiteStoreType)
} catch {
fatalError("Failed to migrate store: \(error)")
}
To migrate a store, Core Data:
Creates a temporary persistence stack.
Mounts the old and new stores.
Loads all objects from the old store.
Migrates the objects to the new store.
The objects are given temporary IDs, then assigned to the new store. The new store then saves the newly assigned objects (committing them to the external repository).
Informs other stacks that the object IDs have changed (from the old to the new stores), which keeps the stack running after a migration.
Unmounts the old store.
Returns the new store.
An error can occur if:
You provide invalid parameters to the method
Core Data cannot add the new store
Core Data cannot remove the old store
In the latter two cases, you get the same errors that you would get if you called addPersistentStore:
or removePersistentStore:
directly. If an error occurs when the store is being added or removed, treat this as an exception, because the persistence stack is likely to be in an inconsistent state.
If a failure occurs during the migration itself, instead of an error you get an exception. In these cases, Core Data unwinds cleanly and there should be no repair work necessary. You can examine the exception description to determine what went wrong — possible errors range widely from "disk is full" and "permissions problems" to "The SQLite store became corrupted" and "Core Data does not support cross store relationships".
A store’s metadata provides additional information about the store that is not directly associated with any of the entities in the store.
The metadata is represented by a dictionary. Core Data automatically sets key-value pairs to indicate the store type and its UUID. You can create additional custom keys for your application, or provide a standard set of keys such as kMDItemKeywords
to support Spotlight indexing (if you also write a suitable importer).
Be careful about what information you put into metadata. Spotlight imposes a limit to the size of metadata and replicating an entire document in metadata is probably not useful. However, if you create a URL to identify a particular object in a store (using URIRepresentation
), the URL may be useful to include as metadata.
There are two ways to get the metadata for a store:
Given an instance of a persistent store, get its metadata using the NSPersistentStoreCoordinator
instance method metadataForPersistentStore:
.
Retrieve metadata from a store without the overhead of creating a persistence stack by using the NSPersistentStoreCoordinator
class method, metadataForPersistentStoreOfType:URL:error:
.
There is an important difference between these approaches. The instance method, metadataForPersistentStore:
, returns the metadata as it currently is in your program, including any changes that may have been made since the store was last saved. The class method, metadataForPersistentStoreOfType:URL:error:
, returns the metadata as it is currently represented in the store itself. If there are pending changes to the store, the returned value may therefore be out of sync.
There are two ways you can set the metadata for a store:
Given an instance of a persistent store, set its metadata using the NSPersistentStoreCoordinator
instance method, setMetadata:forPersistentStore:
.
Set the metadata without the overhead of creating a persistence stack by using the NSPersistentStoreCoordinator
class method, setMetadata:forPersistentStoreOfType:URL:error:
.
There is again an important difference between these approaches. If you use setMetadata:forPersistentStore:
, you must save the store (through a managed object context) before the new metadata is saved. If you use setMetadata:forPersistentStoreOfType:URL:error:
, however, the metadata is updated immediately, and the last-modified date of the file is changed.
This difference has particular implications if you use NSPersistentDocument
on OS X. If you update the metadata using setMetadata:forPersistentStoreOfType:URL:error:
while you are actively working on the persistent store (that is, while there are unsaved changes), then when you save the document you will see a warning, “This document's file has been changed by another application since you opened or saved it.” To avoid this, you should instead use setMetadata:forPersistentStore:
. To find the document’s persistent store, you typically ask the persistent store coordinator for its persistent stores (persistentStores
), and use the first item in the returned array to set the metadata. When you use setMetadata:forPersistentStoreOfType:URL:error:
the action is treated as if the change occurred externally to your Core Data stack.
Because Core Data manages the values for NSStoreTypeKey
and NSStoreUUIDKey
in the same metadata, make a mutable copy of any existing metadata before setting your own keys and values, as illustrated in the following code fragment:
NSURL *url = [NSURL fileURLWithPath:@"url to store"];
NSPersistentStore *store = [self.managedObjectContext.persistentStoreCoordinator persistentStoreForURL:url];
NSMutableDictionary *metadata = [[store metadata] mutableCopy];
metadata[@"MyKeyWord"] = @"MyStoredValue";
[store setMetadata:metadata];
let url = NSURL(fileURLWithPath: "url to store")
guard let store = managedObjectContext.persistentStoreCoordinator?.persistentStoreForURL(url) else {
fatalError("Failed to retrieve store from \(url)")
}
var metadata = store.metadata
metadata["MyKeyWord"] = "MyStoredValue"
store.metadata = metadata