> Le 3 août 2017 à 02:09, Jordan Rose via swift-evolution 
> <swift-evolution@swift.org> a écrit :
> 
> P.S. There's a reason why Decodable uses an initializer instead of a 
> factory-like method on the type but I can't remember what it is right now. I 
> think it's something to do with having the right result type, which would 
> have to be either 'Any' or an associated type if it wasn't just 'Self'. (And 
> if it is 'Self' then it has all the same problems as an initializer and would 
> require extra syntax.) Itai would know for sure.

For anyone interested, factory methods *can* retroactivaly be added to existing 
classes. This is how the SQLite library GRDB.swift is able to decode classes 
hierarchies like NSString, NSNumber, NSDecimalNumber, etc. from SQLite values:

The protocol for types that can instantiate from SQLite values has a factory 
method:

    public protocol DatabaseValueConvertible {
        /// Returns a value initialized from *dbValue*, if possible.
        static func fromDatabaseValue(_ dbValue: DatabaseValue) -> Self?
    }

Having Foundation classes implement it uses various techniques:

1. "Casting" (Data to NSData, or NSDate to Date, depending on which type 
provides the root conformance)

    // Workaround Swift inconvenience around factory methods of non-final 
classes
    func cast<T, U>(_ value: T) -> U? {
        return value as? U
    }
    
    extension NSData : DatabaseValueConvertible {
        public static func fromDatabaseValue(_ dbValue: DatabaseValue) -> Self? 
{
            // Use Data conformance
            guard let data = Data.fromDatabaseValue(dbValue) else {
                return nil
            }
            return cast(data)
        }
    }

    // Derives Date conformance from NSDate, for example
    extension ReferenceConvertible where Self: DatabaseValueConvertible, 
Self.ReferenceType: DatabaseValueConvertible {
        public static func fromDatabaseValue(_ dbValue: DatabaseValue) -> Self? 
{
            return ReferenceType.fromDatabaseValue(dbValue).flatMap { cast($0) }
        }
    }


2. Using magic Foundation initializers (magic because the code below compiles 
even if those are not *required* initializers). Works for NSNumber, 
NSDecimalNumber, NSString:

    extension NSNumber : DatabaseValueConvertible {
        public static func fromDatabaseValue(_ dbValue: DatabaseValue) -> Self? 
{
            switch dbValue.storage {
            case .int64(let int64):
                return self.init(value: int64)
            case .double(let double):
                return self.init(value: double)
            default:
                return nil
            }
        }
    }

    extension NSString : DatabaseValueConvertible {
        public static func fromDatabaseValue(_ dbValue: DatabaseValue) -> Self? 
{
            // Use String conformance
            guard let string = String.fromDatabaseValue(dbValue) else {
                return nil
            }
            return self.init(string: string)
        }
    }

The magic about Foundation initializers above makes me doubt that this 
technique is general enough for Decodable to profit from it, though. Yes it 
runs on Linux, so I'm not even sure if objc runtime is required or not. I'm 
clueless ???????

Gwendal Roué

_______________________________________________
swift-evolution mailing list
swift-evolution@swift.org
https://lists.swift.org/mailman/listinfo/swift-evolution

Reply via email to