Reflection is a feature that dynamically retrieves type, member information, and calls methods, properties, and so on at run time. There is little emphasis on reflection in OC because the OC Runtime is much more powerful than reflection in other languages. However, Swift is a type-safe language that does not support direct manipulation like OC, and its standard library still provides a reflection mechanism that allows us to access member information.

Swift’s reflection mechanism is based on a mechanism called Mirror. You create a Mirror object for a specific instance and then query that instance.

Mirror usage

  • Create an instance of the Mirror class. Reflecting is the instance to be reflected. The value is of Any type.

let mirror = Mirror(reflecting: LGTeacher.self)

  • Mirror can query the properties in the mirror

Look at the following code

for pro in mirror.children {
    // label value
    print("(pro.label):(pro.value)")
}
Copy the code

Mirror. children is a collection of tuples.

Implement jsonMap recursively with Mirror

Look at the code below, we use the Mirror level by level from inside to outside to retrieve the current class LGTeacher attribute information.

struct Student { let name:String } enum SexType { case Man, Woman } class LGTeacher { var age: Int = 18 var student: [Student] = [Student(name:" "),Student(name:" ")] let sex = sextype. Man (){// print(" ") //}} func test(_ mirrorObj: Any) -> Any{ let mirror = Mirror(reflecting: mirrorObj) guard ! mirror.children.isEmpty else{ return mirrorObj } var result: [String: Any] = [:] for child in mirror.children{ if let key = child.label{ result[key] = test(child.value) }else{ print("No Keys") } } return result } var reslut = test(LGTeacher()) print(reslut)Copy the code

Add the Error

// To be continued…

Mirror source code analysis

// To be continued…

Handyjson source code analysis

// To be continued…

Restore StructMetadata

In this section, we need to restore StructMetadata and print out the attribute type and value of the Struct instance to understand the principle of reflection.

Restore the structure through Swfit source code

To restore the structure of StructMetadata, we first need to start from the source, find metadata. h, search TargetStructMetadata

struct TargetStructMetadata : public TargetValueMetadata<Runtime>
Copy the code

It turns out that TargetStructMetadata has TargetStructDescriptor in it but doesn’t own it directly, so keep looking down, and then keep looking for TargetValueMetadata

struct TargetValueMetadata : public TargetMetadata<Runtime> { using StoredPointer = typename Runtime::StoredPointer; TargetValueMetadata(MetadataKind Kind, const TargetTypeContextDescriptor<Runtime> *description) : TargetMetadata<Runtime>(Kind), Description(description) {} /// An out-of-line description of the type. TargetSignedPointer<Runtime, const TargetValueTypeDescriptor<Runtime> * __ptrauth_swift_type_descriptor> Description; static bool classof(const TargetMetadata<Runtime> *metadata) { return metadata->getKind() == MetadataKind::Struct || metadata->getKind() == MetadataKind::Enum || metadata->getKind() == MetadataKind::Optional; } ConstTargetMetadataPointer<Runtime, TargetValueTypeDescriptor> getDescription() const { return Description; } typename Runtime::StoredSignedPointer getDescriptionAsSignedPointer() const { return Description; }};Copy the code

Inside this we find a member called Description, and look at TargetMetadata

struct TargetMetadata {
  using StoredPointer = typename Runtime::StoredPointer;

  /// The basic header type.
  typedef TargetTypeMetadataHeader<Runtime> HeaderType;

  constexpr TargetMetadata()
    : Kind(static_cast<StoredPointer>(MetadataKind::Class)) {}
  constexpr TargetMetadata(MetadataKind Kind)
    : Kind(static_cast<StoredPointer>(Kind)) {}

#if SWIFT_OBJC_INTEROP
protected:
  constexpr TargetMetadata(TargetAnyClassMetadata<Runtime> *isa)
    : Kind(reinterpret_cast<StoredPointer>(isa)) {}
#endif

private:
  /// The kind. Only valid for non-class metadata; getKind() must be used to get
  /// the kind value.
  StoredPointer Kind;
Copy the code

This code is too long, we just find StoredPointer Kind; Will do. So we initially restore: TargetStructMetadata as follows

Struct TargetStructMetadata {var kind: Int var typeDescriptor: UnsafeMutablePointer<TargetStructDescriptor> }Copy the code

Let’s go ahead and explore the structure of TargetStructDescriptor

class TargetStructDescriptor final
: public TargetValueTypeDescriptor<Runtime>,
Copy the code

We peel it off and we get back to TargetStructMetadata and TargetStructDescriptor

struct TargetStructDescriptor {
    var flags: Int32
    var parent: TargetRelativeDirectPointer<UnsafeRawPointer>
    var name: TargetRelativeDirectPointer<CChar>
    var accessFunctionPointer: TargetRelativeDirectPointer<UnsafeRawPointer>
    var fieldDescriptor: TargetRelativeDirectPointer<FieldDescriptor>
    var NumFields: UInt32
    var FieldOffsetVectorOffset: UInt32
}

struct TargetStructMetadata {
    var kind: Int
    var typeDescriptor: UnsafeMutablePointer<TargetStructDescriptor>
}
Copy the code

Verify reductive structure

Next, let’s verify that our structure is correct: look at the following code:

struct WYWPerson { var age:Int = 37 let name:String = "wuyanwei" let sex = false var address = "wujiazui117-601" var Birthday = ("2019", "10", "05")} var person = WYWPerson() Let PTR = unsafeBitCast(wywperson. self as any. Type, to: UnsafeMutablePointer<TargetStructMetadata>.self) let namePtr = ptr.pointee.typeDescriptor.pointee.name.getmeasureRelativeOffset() print("current class name: \(String(cString: NamePtr)) ") let numFileds = PTR. Pointee. TypeDescriptor. Pointee. NumFields print (" the number of the current structure of the attribute: \(numFileds)") // **current class name: WYWPerson** ** 5** **======= start fetch filed ======** **--- filed: age info begin ----** **--- filed: name info begin ----** **--- filed: sex info begin ----** **--- filed: address info begin ----** **--- filed: birthday info begin ----**Copy the code

The resulting print is consistent with our structure, so there is no problem with our restored structure

Getting attribute information

And you can see that the property descriptor that we restore is the property descriptor that stores the property information of the structure, so it’s easy to print out the name of the property and the data type. Note that the typeManglename holds a string that has been shuffled. And then we continue up here

let offsets = ptr.pointee.typeDescriptor.pointee.getFieldOffsets(UnsafeRawPointer(ptr).assumingMemoryBound(to: Double.self)) for i in 0.. <numFileds{ let fieldDespritor = ptr.pointee.typeDescriptor.pointee.fieldDescriptor.getmeasureRelativeOffset().pointee.fields.index(of: Int(i)).pointee.fieldName.getmeasureRelativeOffset() print("--- filed: \(String(cString: FieldDespritor) info begin ----") // let fieldOffset = offsets[Int(I)] String let typeMangleName = ptr.pointee.typeDescriptor.pointee.fieldDescriptor.getmeasureRelativeOffset().pointee.fields.index(of: Int(i)).pointee.mangledTypeName.getmeasureRelativeOffset() print("typeManglename:\(typeMangleName)") let genericVector =  UnsafeRawPointer(ptr).advanced(by: ptr.pointee.typeDescriptor.pointee.genericArgumentOffset * MemoryLayout<UnsafeRawPointer>.size).assumingMemoryBound(to: Any.Type.self) //HandyJSON let fieldType = _swift_getTypeByMangledNameInContext(typeMangleName, 256, genericContext: UnsafeRawPointer(ptr.pointee.typeDescriptor), genericArguments: UnsafeRawPointer(genericVector)? .assumingMemoryBound(to: Optional<UnsafeRawPointer>.self)) print(fieldType as Any) } /// current class name: WYWPerson Number of attributes of the current structure: 5 ======= start fetch filed ====== -- filed: age info begin ---- typeManglename:0x000000010000acb0 Optional(Swift.Int) --- filed: name info begin ---- typeManglename:0x000000010000ad2a Optional(Swift.String) --- filed: sex info begin ---- typeManglename:0x000000010000ad34 Optional(Swift.Bool) --- filed: address info begin ---- typeManglename:0x000000010000ad2a Optional(Swift.String) --- filed: birthday info begin ---- typeManglename:0x000000010000ad38 Optional((Swift.String, Swift.String, Swift.String))Copy the code

Note: here we are calling a swift standard library functions to obtain the types of the Type attribute _swift_getTypeByMangledNameInContext is swift, a call alias, by the way stated below

@_silgen_name("swift_getTypeByMangledNameInContext") public func _swift_getTypeByMangledNameInContext( _ name: UnsafeMutablePointer<CChar>, _ nameLength: Int, genericContext: UnsafeRawPointer? , genericArguments:UnsafeRawPointer?) -> Any.Type?Copy the code

Get property values

Finally, we need to get the value of the property. Here is a more convoluted way to do this, using the same dynamic assignment method as in the previous section:

  1. First find the address of the FieldOffsets by relative memory offset. This is the first address of a buffer, which in turn stores the address of the current value of each attribute (offset from the first address of the instance).
  2. Gets a value in memory based on its address.
  3. The protocol defines a method that converts value to our Self type, which is the property type, and then prints it.

Look at the following code

func getFieldOffsets(_ metadata: UnsafeRawPointer) -> UnsafePointer<Int32> {
        return UnsafeRawPointer(metadata.assumingMemoryBound(to: Int.self).advanced(by: numericCast(self.fieldOffsetVectorOffset))).assumingMemoryBound(to: Int32.self)
    }
Copy the code
//HandJSON let fieldType = _swift_getTypeByMangledNameInContext(typeMangleName, 256, genericContext: UnsafeRawPointer(ptr.pointee.typeDescriptor), genericArguments: UnsafeRawPointer(genericVector)? .assumingMemoryBound(to: Self)) print(fieldType as Any) HandJSON let type = unsafeBitCast(fieldType, to: Any.Type.self) let value = customCast(type: type) let instanceAddress = UnsafeRawPointer(withUnsafeMutablePointer(to: &person){$0}) let fieldOffset = offsets[Int(i)] print("fieldType:\(type) \nfieldValue: \(value.get(from: instanceAddress.advanced(by: Int(fieldOffset)))) ") print("--- filed: \(String(cString: fieldDespritor)) info end ---- \n")Copy the code

Implementation summary

That’s how we print struct instance attributes and attribute information. Our basic idea is summarized as follows,

Restore TargetClassMetadata structure by source code. Restore the structure of TargetClassDescriptor from source code. The metadata of the WYWPerson instance, wywPerson. self, is formatted as a pointer to the StructMetadata structure we restored using the memory map unsafeBitCast. Read the structure name from StructMetadata, find the property name of each item from the fields of the fieldDescriptor, and get the property type (the string of the type is mixed) by bridging Handyjson to get the value of the property and print it.

All the code

Related operation code

struct TargetRelativeDirectPointer<Pointee>{
    var offset: Int32
    
    mutating func getmeasureRelativeOffset() -> UnsafeMutablePointer<Pointee>{
        let offset = self.offset
        
        return withUnsafePointer(to: &self) { p in
           return UnsafeMutablePointer(mutating: UnsafeRawPointer(p).advanced(by: numericCast(offset)).assumingMemoryBound(to: Pointee.self))
        }
    }
}

struct FieldDescriptor {
    var MangledTypeName: TargetRelativeDirectPointer<CChar>
    var Superclass: TargetRelativeDirectPointer<CChar>
    var kind: UInt16
    var fieldRecordSize: Int16
    var numFields: Int32
    var fields: FiledRecordBuffer<FieldRecord>
}

struct FieldRecord {
    var fieldRecordFlags: Int32
    var mangledTypeName: TargetRelativeDirectPointer<CChar>
    var fieldName: TargetRelativeDirectPointer<UInt8>
}

struct FiledRecordBuffer<Element>{
    var element: Element
    
    mutating func buffer(n: Int) -> UnsafeBufferPointer<Element> {
        return withUnsafePointer(to: &self) {
            let ptr = $0.withMemoryRebound(to: Element.self, capacity: 1) { start in
                return start
            }
            return UnsafeBufferPointer(start: ptr, count: n)
        }
    }
    
    mutating func index(of i: Int) -> UnsafeMutablePointer<Element> {
        return withUnsafePointer(to: &self) {
            return UnsafeMutablePointer(mutating: UnsafeRawPointer($0).assumingMemoryBound(to: Element.self).advanced(by: i))
        }
    }
}
Copy the code

Restore the struct structure

struct TargetStructDescriptor { var flags: Int32 var parent: TargetRelativeDirectPointer<UnsafeRawPointer> var name: TargetRelativeDirectPointer<CChar> var accessFunctionPointer: TargetRelativeDirectPointer<UnsafeRawPointer> var fieldDescriptor: TargetRelativeDirectPointer<FieldDescriptor> var numFields: UInt32 var fieldOffsetVectorOffset: UInt32 func getFieldOffsets(_ metadata: UnsafeRawPointer) -> UnsafePointer<Int32> { return UnsafeRawPointer(metadata.assumingMemoryBound(to: Int.self).advanced(by: numericCast(self.fieldOffsetVectorOffset))).assumingMemoryBound(to: Int32.self) } var genericArgumentOffset: Int { return 2 } } struct TargetStructMetadata { var kind: Int var typeDescriptor: UnsafeMutablePointer<TargetStructDescriptor> } @_silgen_name("swift_getTypeByMangledNameInContext") public func _swift_getTypeByMangledNameInContext( _ name: UnsafeMutablePointer<CChar>, _ nameLength: Int, genericContext: UnsafeRawPointer? , genericArguments:UnsafeRawPointer?) -> Any.Type?Copy the code
struct WYWPerson { var age:Int = 37 let name:String = "wuyanwei" let sex = false var address = "wujiazui117-601" var Birthday = ("2019", "10", "05")} var person = WYWPerson() Let PTR = unsafeBitCast(wywperson. self as any. Type, to: UnsafeMutablePointer<TargetStructMetadata>.self) let namePtr = ptr.pointee.typeDescriptor.pointee.name.getmeasureRelativeOffset() print("current class name: \(String(cString: NamePtr)) ") let numFileds = PTR. Pointee. TypeDescriptor. Pointee. NumFields print (" the number of the current structure of the attribute: \(numFileds)") print("======= start fetch filed ======") let offsets = ptr.pointee.typeDescriptor.pointee.getFieldOffsets(UnsafeRawPointer(ptr)) for i in 0.. <numFileds{ let fieldDespritor = ptr.pointee.typeDescriptor.pointee.fieldDescriptor.getmeasureRelativeOffset().pointee.fields.index(of: Int(i)).pointee.fieldName.getmeasureRelativeOffset() print("--- filed: \(String(cString: fieldDespritor)) info begin ----") let typeMangleName = ptr.pointee.typeDescriptor.pointee.fieldDescriptor.getmeasureRelativeOffset().pointee.fields.index(of: Int(i)).pointee.mangledTypeName.getmeasureRelativeOffset() print("typeManglename:\(typeMangleName)") let genericVector =  UnsafeRawPointer(ptr).advanced(by: ptr.pointee.typeDescriptor.pointee.genericArgumentOffset * MemoryLayout<UnsafeRawPointer>.size).assumingMemoryBound(to: Any.Type.self) //HandJSON let fieldType = _swift_getTypeByMangledNameInContext(typeMangleName, 256, genericContext: UnsafeRawPointer(ptr.pointee.typeDescriptor), genericArguments: UnsafeRawPointer(genericVector)? .assumingMemoryBound(to: Self)) print(fieldType as Any) HandJSON let type = unsafeBitCast(fieldType, to: Any.Type.self) let value = customCast(type: type) let instanceAddress = UnsafeRawPointer(withUnsafeMutablePointer(to: &person){$0}) let fieldOffset = offsets[Int(i)] print("fieldType:\(type) \nfieldValue: \(value.get(from: instanceAddress.advanced(by: Int(fieldOffset)))) ") print("--- filed: \(String(cString: fieldDespritor)) info end ---- \n") } print("-----end")Copy the code
///result Generated result Current class Name: WYWPerson Number of attributes of the current structure: 5 ======= start fetch filed ====== -- filed: age info begin ---- typeManglename:0x000000010000ad72 Optional(Swift.Int) fieldType:Int fieldValue: 37 --- filed: age info end ---- --- filed: name info begin ---- typeManglename:0x000000010000adf6 Optional(Swift.String) fieldType:String fieldValue: wuyanwei --- filed: name info end ---- --- filed: sex info begin ---- typeManglename:0x000000010000adfa Optional(Swift.Bool) fieldType:Bool fieldValue: false --- filed: sex info end ---- --- filed: address info begin ---- typeManglename:0x000000010000adf6 Optional(Swift.String) fieldType:String fieldValue: wujiazui117-601 --- filed: address info end ---- --- filed: birthday info begin ---- typeManglename:0x000000010000adfe Optional((Swift.String, Swift.String, Swift.String)) fieldType:(String, String, String) fieldValue: ("2019", "10", "05") --- filed: birthday info end ---- -----endCopy the code