Codable items with identity?

In a word, no. This has been an unresolved design issue for several years.

Currently, there is nothing that will encode references instead of values, although it does seem feasible to write a new encoder type that will encode references.

Thanks for confirming the unresolved issue.

I wrote some untested but compilable code, below, that may help solve
the problem in a pinch, but there are some problems. I haven't dealt
with object-graph cycles except to throw if one occurs during decoding.
I assume that a Decoder will decode objects in the same order as the
Encoder encoded them, but that's not founded on any promise I have seen
in the documentation.

I provide a protocol, CodableWithIdentity, that a class conforms to by
providing both an initializer and an encode function that operate on
Keyed{De,En}codingContainer.

CodableWithIdentity relies on the user to stash an ObjectRegistry
in the userInfo property of the Encoder or Decoder. The
ObjectRegistry keeps a mapping from UInt64 to AnyObject and back.
Here is the code.

extension CodingUserInfoKey {
        public static let objectRegistry =
            CodingUserInfoKey(rawValue: "com.whitecoralislands.objectRegistry")!
}

enum ObjectCodingError : Error {
case registryMissing
case idInUse
case typeMismatch
case wrongIdState
case graphCycle
}

class ObjectRegistry {
        typealias Id = UInt64
        public enum Presence {
        case absent
        case busy
        case present(AnyObject)
        }
        var idToPresence: [Id : Presence] = [:]
        var oidToId: [ObjectIdentifier : Id] = [:]
        var nextId: Id = 0
        public init() {
        }
        public func insert(_ o: AnyObject) throws -> Id {
                let oid = ObjectIdentifier(o)
                guard oidToId[oid] == nil else {
                        throw ObjectCodingError.idInUse
                }
                let id = nextId
                defer {
                        nextId = nextId + 1
                }
                oidToId[oid] = id
                idToPresence[id] = .present(o)
                return id
        }
        public func beginInsertion(at id: Id) throws {
                guard case .absent = idToPresence[id, default: .absent] else {
                        throw ObjectCodingError.idInUse
                }
                idToPresence[id] = .busy
        }
        public func finishInsertion(of o: AnyObject, at id: Id) throws {
                switch idToPresence[id, default: .absent] {
                case .busy:
                        idToPresence[id] = .present(o)
                case .absent:
                        throw ObjectCodingError.wrongIdState
                case .present(_):
                        throw ObjectCodingError.idInUse
                }
        }
        public func lookup(_ id: Id) -> Presence {
                return idToPresence[id, default: .absent]
        }
        public func lookup(_ o: AnyObject) -> Id? {
                return oidToId[ObjectIdentifier(o)]
        }
}

struct ObjectCodingContainer<T> : Codable where T : CodableWithIdentity {
        var instance: T
        enum CodingKeys : CodingKey {
        case id
        case object
        }
        init(containing instance: T) {
                self.instance = instance
        }
        func encode(to encoder: Encoder) throws {
                var container = encoder.container(keyedBy: CodingKeys.self)
                guard let registry =
                    encoder.userInfo[.objectRegistry] as? ObjectRegistry
                    else {
                        throw ObjectCodingError.registryMissing
                }
                if let id = registry.lookup(instance) {
                        try container.encode(id, forKey: CodingKeys.id)
                        return
                }
                let id = try registry.insert(instance)
                try container.encode(id, forKey: CodingKeys.id)
                try container.encode(instance, forKey: CodingKeys.object)
        }
        init(from decoder: Decoder) throws {
                let container = try decoder.container(keyedBy: CodingKeys.self)
                guard let registry =
                    decoder.userInfo[.objectRegistry] as? ObjectRegistry
                    else {
                        throw ObjectCodingError.registryMissing
                }
                let id = try container.decode(UInt64.self,
                    forKey: CodingKeys.id)

                switch registry.lookup(id) {
                case .present(let found):
                        guard let object = found as? T else {
                                throw ObjectCodingError.typeMismatch
                        }
                        self.instance = object
                        return
                case .busy:
                        throw ObjectCodingError.graphCycle
                case .absent:
                        try registry.beginInsertion(at: id)
                        let object = try container.decode(T.self,
                                forKey: CodingKeys.object)
                        try registry.finishInsertion(of: object, at: id)
                        self.instance = object
                }
        }
}

public protocol CodableWithIdentity : AnyObject {
        associatedtype Keys : CodingKey
        init(from: inout KeyedDecodingContainer<Keys>) throws
        func encode(to: inout KeyedEncodingContainer<Keys>) throws
}

extension CodableWithIdentity {
        init(from outer: inout UnkeyedDecodingContainer) throws {
            var inner = try outer.nestedContainer(keyedBy: Self.Keys.self)
            try self.init(from: &inner)
        }
        func encode(to outer: inout UnkeyedEncodingContainer) throws {
            var inner = outer.nestedContainer(keyedBy: Self.Keys.self)
            try self.encode(to: &inner)
        }
}

extension KeyedDecodingContainer {
        func decode<T : CodableWithIdentity>(_ type: T.Type,
            forKey key: Key) throws -> T {
                    var container =
                        try nestedContainer(keyedBy: T.Keys.self, forKey: key)
                    return try T.self.init(from: &container)
        }
}

extension KeyedEncodingContainer {
        mutating func encode<T : CodableWithIdentity>(_ o: T, forKey key: Key)
            throws {
                    var container =
                        nestedContainer(keyedBy: T.Keys.self, forKey: key)
                    return try o.encode(to: &container)
        }
}

extension UnkeyedEncodingContainer {
        mutating func encode<T : CodableWithIdentity>(_ o: T)
            throws {
                    var container =
                        nestedContainer(keyedBy: T.Keys.self)
                    return try o.encode(to: &container)
        }
}

extension Encoder {
        func encode<T>(_ o: T) throws -> T where T : CodableWithIdentity {
                let container = ObjectCodingContainer<T>(containing: o)
                return container.instance
        }
}

extension Decoder {
        func decode<T>(_ type: T.Type) throws -> T
            where T : CodableWithIdentity, T : Codable {
                let container = try ObjectCodingContainer<T>(from: self)
                return container.instance
        }
}

On the decoding side, Swift's init design prevents the reconstruction of arbitrary graphs. The nearest you can get is to severely restrict the structure of the graph, or to fix up the decoded graph in a separate, manual pass — which may require exposing private members as public, making the whole process somewhat subject to abuse.

I see what you mean about the init design.

I have been thinking about performing "just-in-time" resolution
of references in an arbitrary object graph, at some small cost in
complexity and performance. Suppose that if instances of class C can
be involved in a cycle, then we replace each Codable reference to an
instance of C with a wrapped enum property that contains a reference
either to an actual C instance or the information to fetch a reference
from an ObjectRegistry:

@propertyWrapper enum Reference<C> {
case proxy(UInt64, ObjectRegistry)
case actual(C)
        var wrappedValue: C {
                mutating get {
                        switch self {
                        case .proxy(let id, let registry):
                                switch registry.lookup(id) {
                                case .present(let object):
                                        let instance = object as! C
                                        self = .actual(instance)
                                        return instance
                                default:
                                        fatalError()
                                }
                        case .actual(let object):
                                return object
                        }
                }
                set {
                        self = .actual(newValue)
                }
        }
}

Dave