CtrlK
BlogDocsLog inGet started
Tessl Logo

dpearson2699/swift-ios-skills

Agent skills for iOS, iPadOS, Swift, SwiftUI, and modern Apple framework development.

71

Quality

89%

Does it follow best practices?

Impact

No eval scenarios have been run

SecuritybySnyk

Advisory

Suggest reviewing before use

Overview
Quality
Evals
Security
Files

concurrency-patterns.mdskills/swift-concurrency/references/

Concurrency Patterns

Approachable concurrency patterns introduced in Swift 6.2+ — a philosophy shift where code stays single-threaded by default until you choose to introduce concurrency.

Contents

Core Problem Solved

In Swift 6.0/6.1, data-race safety was enforced at compile time, but the most natural code to write often produced data-race errors. Async functions on types with mutable state would implicitly hop to the global concurrent executor, causing send-safety violations even when no actual parallelism was intended.

// Swift 6.0/6.1: This produces a data-race error
class PhotoProcessor {
    func extractSticker(data: Data, with id: String?) async -> Sticker? { /* ... */ }
}

@MainActor
final class StickerModel {
    let photoProcessor = PhotoProcessor()

    func extractSticker(_ item: PhotosPickerItem) async throws -> Sticker? {
        guard let data = try await item.loadTransferable(type: Data.self) else { return nil }
        // Error: Sending 'self.photoProcessor' risks causing data races
        return await photoProcessor.extractSticker(data: data, with: item.itemIdentifier)
    }
}
// Swift 6.2: The same code compiles without error
// because extractSticker stays on the caller's actor
class PhotoProcessor {
    func extractSticker(data: Data, with id: String?) async -> Sticker? { /* ... */ }
}

@MainActor
final class StickerModel {
    let photoProcessor = PhotoProcessor()

    func extractSticker(_ item: PhotosPickerItem) async throws -> Sticker? {
        guard let data = try await item.loadTransferable(type: Data.self) else { return nil }
        return await photoProcessor.extractSticker(data: data, with: item.itemIdentifier)
    }
}

SE-0466: Default MainActor Isolation

Enable with the -default-isolation MainActor compiler flag or the Xcode 26 "Approachable Concurrency" build setting.

What it does:

  • All declarations in the module are implicitly @MainActor unless opted out.
  • Global and static variables are protected by the main actor by default.
  • Protocol conformances are implicitly isolated to @MainActor.
  • Eliminates most annotation burden for single-threaded UI code.

Recommended for: Apps, scripts, and executable targets. Not recommended for library targets that should remain actor-agnostic.

// With default MainActor isolation -- no @MainActor annotations needed:
final class StickerLibrary {
    static let shared = StickerLibrary()
}

final class StickerModel {
    let photoProcessor = PhotoProcessor()
    var selection: [PhotosPickerItem] = []
}

extension StickerModel: Exportable {
    func export() { photoProcessor.exportAsPNG() }
}

SE-0461: nonisolated(nonsending)

Nonisolated async functions stay on the caller's actor by default instead of hopping to the global concurrent executor. This is the nonisolated(nonsending) default behavior.

Key implication: Values passed into an async function are never sent outside the actor, eliminating data races without annotation.

To explicitly opt into background execution, use @concurrent.

@concurrent Attribute

Ensures a function always runs on the concurrent thread pool, freeing the calling actor for other work.

class PhotoProcessor {
    var cachedStickers: [String: Sticker] = [:]

    func extractSticker(data: Data, with id: String) async -> Sticker {
        if let sticker = cachedStickers[id] { return sticker }
        let sticker = await Self.extractSubject(from: data)
        cachedStickers[id] = sticker
        return sticker
    }

    @concurrent
    static func extractSubject(from data: Data) async -> Sticker { /* ... */ }
}

Steps to offload a function to background:

  1. Ensure the containing type is nonisolated (or the function itself).
  2. Add @concurrent to the function.
  3. Add async if not already asynchronous.
  4. Add await at call sites.
nonisolated struct PhotoProcessor {
    @concurrent
    func process(data: Data) async -> ProcessedPhoto? { /* ... */ }
}

processedPhotos[item.id] = await PhotoProcessor().process(data: data)

SE-0472: Task.immediate

Task.immediate starts executing synchronously on the current actor before any suspension point, rather than being enqueued. There is also Task.immediateDetached which combines immediate start with detached semantics.

Task.immediate { await handleUserInput() }

Use for latency-sensitive work where enqueue delay is unacceptable.

Isolated Conformances

A conformance that needs MainActor state is called an isolated conformance. The compiler ensures the conformance is only used in a matching isolation context.

protocol Exportable {
    func export()
}

extension StickerModel: @MainActor Exportable {
    func export() { photoProcessor.exportAsPNG() }
}

@MainActor
struct ImageExporter {
    var items: [any Exportable]

    mutating func add(_ item: StickerModel) {
        items.append(item)  // OK -- on MainActor
    }
}

// But in a nonisolated context:
nonisolated struct GenericExporter {
    var items: [any Exportable]

    mutating func add(_ item: StickerModel) {
        // Error: Main actor-isolated conformance of 'StickerModel' to
        // 'Exportable' cannot be used in nonisolated context
        items.append(item)
    }
}

SE-0481: weak let (Proposed — Swift 6.2+)

Immutable weak references (weak let) enable Sendable conformance for types that hold weak references, since immutability guarantees thread safety. Proposed in SE-0481; may not yet be available in shipping toolchains.

SE-0475: Transactional Observation (Observations)

Observations { } provides transactional observation of @Observable types via AsyncSequence.

for await _ in Observations { model.count } {
    print("Count changed to \(model.count)")
}

Global and Static State

Global and static variables are prone to data races. The most common protection is @MainActor:

@MainActor
final class StickerLibrary {
    static let shared = StickerLibrary()  // protected by MainActor
}

With default MainActor isolation (SE-0466), this annotation is implicit.

Migration and Build Settings

All approachable concurrency features are opt-in via:

  • Xcode 26: Swift Compiler > Concurrency section in build settings.
  • SwiftPM: swiftSettings in Package.swift using the SwiftSetting API.

Swift 6.2 includes migration tooling to help make necessary code changes automatically. See swift.org/migration for details.

Summary

The Swift 6.2 concurrency progression:

  1. Start with code that runs on the main actor by default (no data race risk).
  2. Async functions run wherever they are called from (still no data race risk).
  3. When you need performance, offload specific code with @concurrent.

skills

CHANGELOG.md

README.md

tile.json