preface

In OC development, we can’t avoid dealing with JSON data, interface data filling, we can use the native NSDictionary value, which is not very elegant, so we initialize a model class through a dictionary, value by attribute name, but this does not seem very automatic. Then, there are JSON automatic transformation framework, such as MJExtension, YYModel and other high-performance transformation framework. However, MJExtension is easy to use. After introducing MJExtension, you can call a line of code. Since MJExtension adds a category to NSObject, there is no need to modify the inheritance relationship of data model to use this framework, and there is no code intrusion. MJExtension can easily convert JSON to model, customize aliases, customize transformations, archive files, and more.

Realize the principle of

By reading the source code, the principle of MJExtension is to use runtime to get all the attribute names of this class and all its parent classes, and then get the corresponding value in the data dictionary according to the attribute name, and then set the value of the attribute by setValue:forKey.

The following core code and runtime API are in MJExtension

// ***** to get the parent Class, return Class class_getSuperclass(Class _Nullable CLS) // ***** to get the attribute list, Class_copyPropertyList (Class _Nullable CLS, unsigned int * _Nullable outCount) // Example: unsigned int outCount = 0; objc_property_t *properties = class_copyPropertyList(c, &outCount); // ***** gets the property name, returns char *, which can be converted to nsString.property_getName (objc_property_t _Nonnull property) // Example: objc_property_t property; NSString *name = @(property_getName(property)); // ***** gets the attribute of the property, which returns the char * type, which can be converted to NSString. Property_getAttributes (objC_property_t _Nonnull Property) // Example: objc_property_t property; NSString *attrs = @(property_getAttributes(property)); [object setValue:value forKey:key]; // use KVO to assign a value to the property of the data model.Copy the code

Frame mind mapping

Introduction to core Code

NSObject+MJKeyValue

Main function: Iterate over all attributes of the class (including those defined in the parent class), encapsulate as MJProperty type return. The value is taken from the data dictionary according to the property name defined in MJProperty (here the key value in the dictionary is required to be the same as the property name) and assigned to the property of model.

/** Core code: */ - (instancetype)mj_setKeyValues:(id)keyValues context:(NSManagedObjectContext *)context {// get the JSON object...... // Calls back a method written at runtime to return a list of properties via encapsulated methods. [clazz mj_enumerateProperties:^(MJProperty *property, BOOL *stop) {@try {// 0. Check whether the property is ignored...... // 1 PropertyKeysForClass: clazz];... / / if there is no value, Simply return the if (! Value | | value = = [NSNull null]) return; / / 2. Complex processing MJPropertyType * type = property. The type;... / / 3. The assignment [property setValue:value forObject:self]; } @catch (NSException *exception) { ...... } }]; . return self; }Copy the code

NSObject+MJProperty

Main function: get property and encapsulate it into the concrete implementation of MJProperty.

#pragma mark - -- public method -- + (void)mj_enumerateProperties:(MJPropertiesEnumeration)enumeration {// get member variable...... // Iterate over the member variable...... } #pragma mark - public method + (NSMutableArray *)mj_properties {NSMutableArray *cachedProperties = [self mj_propertyDictForKey:&MJCachedPropertiesKey][NSStringFromClass(self)]; if (cachedProperties == nil) { if (cachedProperties == nil) { cachedProperties = [NSMutableArray array]; [self mj_enumerateClasses:^(__unsafe_unretained Class c, For (unsigned int I = 0; I <outCount; I ++) {......} // 3 free(properties); }]; [self mj_propertyDictForKey:&MJCachedPropertiesKey][NSStringFromClass(self)] = cachedProperties; } } return cachedProperties; }Copy the code

NSObject+MJClass

Main function: traverse all the parent class, used to obtain the attributes of the parent class.

+ (void)mj_enumerateClasses:(MJClassesEnumeration)enumeration { // 1. If (enumeration == nil) return; BOOL stop = NO; // 3. Class c = self; // 4. Start traversing each class while (c &&! Stop) {// 4.1. Run enumeration(c, &stop); // get the parent class c = class_getSuperclass(c); if ([MJFoundation isClassFromFoundation:c]) break; }}Copy the code

Other function code introduction

  1. Set a whitelist of class attributes to redirect or ignore certain attributes.
#pragma mark - Property whitelist configuration /** * The property names in this array are converted to dictionaries and models ** @param allowedPropertyNames are converted to dictionaries and models */ + (void)mj_setupAllowedPropertyNames:(MJAllowedPropertyNames)allowedPropertyNames; #pragma mark - Property blacklist configuration /** * Property names in this array are ignored: dictionary and model conversions are not done ** @param ignoredPropertyNames in this array are ignored: Not to convert the dictionary and model * / + (void) mj_setupIgnoredPropertyNames (MJIgnoredPropertyNames) ignoredPropertyNames;Copy the code

NSObject+MJKeyValue

  1. Caching of class attributes that are parsed once and not parsed again
+ (NSMutableArray *)mj_properties {// Get the cached property array NSMutableArray *cachedProperties = [self dictForKey:&MJCachedPropertiesKey][NSStringFromClass(self)]; If (cachedProperties == nil) {// Not cached, initialize array. // Parse propertys and add to cachedProperties array....... // Cache attribute array [self dictForKey:&MJCachedPropertiesKey][NSStringFromClass(self)] = cachedProperties; } return cachedProperties; }Copy the code

(3) dynamic change certain values characteristic of attribute values In its own definition of data model can be realized through mj_newValueFromOldValue: property: method to replace some attribute values.

For example, write the following code in CutsomModel to assign a property named CutsomProperty to CutsomValue;

- (id)mj_newValueFromOldValue:(id)oldValue property:(MJProperty *)property { if ([property.name isEqualToString:@"CutsomProperty"]) { return @"CutsomValue"; } else { return oldValue; }}Copy the code

NSObject+MJKeyValue source code parsing

- (instancetype)mj_setKeyValues:(id)keyValues context:(NSManagedObjectContext *)context { ...... [clazz mj_enumerateProperties:^(MJProperty *property, BOOL *stop) {@try {...... // If model implements mj_newValueFromOldValue: The property method, will return the new value. Id newValue = [clazz mj_getNewValueFromObject: self oldValue: the value property: the property]; the if (newValue != value) {// there is a newValue for the filter [property setValue:newValue forObject:self]; return;}.....} @catch (NSException *exception) { ...}}]; return self; }Copy the code

Attribute Type Description

//An int NSString *const MJPropertyTypeInt = @"i"; //A short NSString *const MJPropertyTypeShort = @"s"; //A float NSString *const MJPropertyTypeFloat = @"f"; // A double NSString *const MJPropertyTypeDouble = @"d"; //A long`l` is treated as a 32-bit quantity on 64-bit programs. NSString *const MJPropertyTypeLong = @"l"; //A long long NSString *const MJPropertyTypeLongLong = @"q"; //A char NSString *const MJPropertyTypeChar = @"c"; //A C++ bool or a C99 _Bool NSString *const MJPropertyTypeBOOL1 = @"c"; NSString *const MJPropertyTypeBOOL2 = @"b"; //A character string (char *) NSString *const MJPropertyTypePointer = @"*"; NSString *const MJPropertyTypeIvar = @"^{objc_ivar=}"; NSString *const MJPropertyTypeMethod = @"^{objc_method=}"; // An unknown type (among other things, this code is used for function pointers) NSString *const MJPropertyTypeBlock = @"@?" ; // A class object (Class) NSString *const MJPropertyTypeClass = @"#"; //A method selector (SEL) NSString *const MJPropertyTypeSEL = @":"; // An object (whether statically typed or typed id) NSString *const MJPropertyTypeId = @"@";Copy the code

Automatic archive unfile based on Runtime

If there are many model attributes, it is still quite cumbersome to manually implement the archiving of each attribute. The Runtime iterates through the member variables and calls KVC to achieve automatic archiving of each attribute

@implementation NSObject (MJCoding) - (void)mj_encode:(NSCoder *)encoder { Class clazz = [self class]; NSArray *allowedCodingPropertyNames = [clazz mj_totalAllowedCodingPropertyNames]; NSArray *ignoredCodingPropertyNames = [clazz mj_totalIgnoredCodingPropertyNames]; [clazz mj_enumerateProperties:^(MJProperty *property, BOOL * stop) {/ / check ignored the if (allowedCodingPropertyNames count &&! [allowedCodingPropertyNames containsObject:property.name]) return; if ([ignoredCodingPropertyNames containsObject:property.name]) return; id value =  [property valueForObject:self]; if (value == nil) return; [encoder encodeObject:value forKey:property.name]; }]; } - (void)mj_decode:(NSCoder *)decoder { Class clazz = [self class]; NSArray *allowedCodingPropertyNames = [clazz mj_totalAllowedCodingPropertyNames]; NSArray *ignoredCodingPropertyNames = [clazz mj_totalIgnoredCodingPropertyNames]; [clazz mj_enumerateProperties:^(MJProperty *property, BOOL * stop) {/ / check ignored the if (allowedCodingPropertyNames count &&! [allowedCodingPropertyNames containsObject:property.name]) return; if ([ignoredCodingPropertyNames containsObject:property.name]) return; id value = [decoder decodeObjectForKey: the property name]; the if (value = = nil) {/ / compatible with previous MJExtension version value = [decoder decodeObjectForKey:[@"_" stringByAppendingString:property.name]]; } if (value == nil) return; [property setValue:value forObject:self]; }]; } @endCopy the code

Specific use, in. M add MJExtensionCodingImplementation macro definition

@implementation xxx
  
MJExtensionCodingImplementation
  
@end
Copy the code

Commonly used method

Custom property alias

/ * * * in the name of the attribute into other key to * * @ return values in the dictionary of the key is the attribute name in the dictionary, the value from the dictionary key values with * / + (NSDictionary *) mj_replacedKeyFromPropertyName; / * * * in the name of the attribute into other key values in the dictionary of * * @ the return value from a dictionary with key * / + (id) mj_replacedKeyFromPropertyName121: [propertyName nsstrings *)Copy the code

The model contains the array model

/** * the key in the @return dictionary is the name of the array property, Value is an array containing the model Class (Class or NSString) */ + (NSDictionary *)mj_objectClassInArray;Copy the code

Dictionary model

/** * Create a model with a dictionary * @param keyValues dictionary (NSDictionary, NSData, NSString) * @return new object */ + (instancetype)mj_objectWithKeyValues:(id)keyValues;Copy the code

Model to dictionary

/** * convert model to dictionary * @return dictionary */ - (NSMutableDictionary *)mj_keyValues;Copy the code

Called when the dictionary transformation model is complete

- (void)mj_didConvertToObjectWithKeyValues:(NSDictionary *)keyValues;

Copy the code

Custom conversion that returns a new value

- (id)mj_newValueFromOldValue:(id)oldValue property:(MJProperty *)property;

Copy the code

conclusion

Excellent framework design ability, each class has its single function, through combination and unity, to achieve a set of friendly and easy to use API interface, very convenient to get started. The Runtime technology is powerful and widely used.