swiftdata
4
总安装量
3
周安装量
#48416
全站排名
安装命令
npx skills add https://github.com/makgunay/claude-swift-skills --skill swiftdata
Agent 安装分布
opencode
3
claude-code
3
codex
3
cursor
3
github-copilot
2
windsurf
2
Skill 文档
SwiftData Persistence Patterns
Critical Constraints
- â DO NOT use Core Data syntax (
NSManagedObject,NSFetchRequest,NSPredicate) â â Use@Model,@Query,#Predicate - â DO NOT use
@FetchRequestâ â Use@Query - â DO NOT use string-based predicates â â
Use
#Predicate<ModelType>macro with Swift expressions - â DO NOT create
NSPersistentContainerâ â UseModelContainerandModelContext - â DO NOT forget
@Modelon subclasses â â Both base class AND subclasses need@Model - â DO NOT use
@Modelon structs â â@Modelis for classes only
Basic Model Definition
import SwiftData
@Model
class Trip {
@Attribute(.preserveValueOnDeletion)
var name: String
var destination: String
var startDate: Date
var endDate: Date
@Relationship(deleteRule: .cascade, inverse: \Accommodation.trip)
var accommodation: Accommodation?
init(name: String, destination: String, startDate: Date, endDate: Date) {
self.name = name
self.destination = destination
self.startDate = startDate
self.endDate = endDate
}
}
Class Inheritance
Both base and subclass need @Model. Subclass adds specialized properties.
@Model
class BusinessTrip: Trip {
var purpose: String
var expenseCode: String
var perDiemRate: Double
@Relationship(deleteRule: .cascade, inverse: \BusinessMeal.trip)
var businessMeals: [BusinessMeal] = []
init(name: String, destination: String, startDate: Date, endDate: Date,
purpose: String, expenseCode: String, perDiemRate: Double) {
self.purpose = purpose
self.expenseCode = expenseCode
self.perDiemRate = perDiemRate
super.init(name: name, destination: destination, startDate: startDate, endDate: endDate)
}
}
@Model
class PersonalTrip: Trip {
enum Reason: String, CaseIterable, Codable, Identifiable {
case family, vacation, wellness, other
var id: Self { self }
}
var reason: Reason
var notes: String?
init(name: String, destination: String, startDate: Date, endDate: Date,
reason: Reason, notes: String? = nil) {
self.reason = reason
self.notes = notes
super.init(name: name, destination: destination, startDate: startDate, endDate: endDate)
}
}
Querying with Inheritance
// All trips (base + all subclasses)
@Query(sort: \Trip.startDate) var allTrips: [Trip]
// Filter by subclass type
let businessOnly = #Predicate<Trip> { $0 is BusinessTrip }
@Query(filter: businessOnly) var businessTrips: [Trip]
// Filter by subclass property
let vacations = #Predicate<Trip> {
if let personal = $0 as? PersonalTrip {
return personal.reason == .vacation
}
return false
}
ModelContainer Setup
@main
struct MyApp: App {
var body: some Scene {
WindowGroup { ContentView() }
.modelContainer(for: [Trip.self, BusinessTrip.self, PersonalTrip.self])
}
}
// Custom configuration
let config = ModelConfiguration("MyStore", isStoredInMemoryOnly: false)
let container = try ModelContainer(for: Trip.self, configurations: config)
CRUD Operations
// Create
let trip = Trip(name: "Paris", destination: "France", startDate: .now, endDate: .now)
modelContext.insert(trip)
// Read
let descriptor = FetchDescriptor<Trip>(sortBy: [SortDescriptor(\.startDate)])
let trips = try modelContext.fetch(descriptor)
// Update â just modify properties, SwiftData auto-saves
trip.name = "Paris 2025"
// Delete
modelContext.delete(trip)
// Save explicitly
try modelContext.save()
Decision Tree: Inheritance vs Enum
Use inheritance when:
âââ Clear IS-A relationship (BusinessTrip IS-A Trip)
âââ Subclasses have significantly different properties
âââ Need both deep queries (all trips) and shallow queries (business only)
âââ Polymorphic relationships needed
Use enum/flag instead when:
âââ Only a few shared properties
âââ Distinction is simple (just a type tag)
âââ Boolean flag suffices
âââ Query strategy only focuses on specialized properties
Common Mistakes & Fixes
| Mistake | Fix |
|---|---|
@Model struct MyData |
@Model class MyData â must be class |
NSPredicate(format:) |
#Predicate<MyModel> { $0.name == "test" } |
Missing @Model on subclass |
Add @Model to both base and subclass |
@FetchRequest in SwiftUI |
@Query var items: [Item] |
| Forgetting inverse on relationship | Add inverse: \OtherModel.property |
| Not registering subclasses in container | Include all model types in .modelContainer(for:) |