In the first part, we talked about the API of Class and Category. In the second part, we talked about the API of Method. In this part, we talked about Ivar and Property.
4.objc_ivar or Ivar
First, let’s find the function that prints Ivar information:
Const char * _Nullable ivar_getName(Ivar _Nonnull V) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);Copy the code
Get the name of Ivar by passing in the corresponding Ivar. -(void)logIvarName:(Ivar) Ivar {if (Ivar) {const char* name = ivar_getName(Ivar); NSLog(@”name = %s”,name); } else {NSLog(@”ivar null”); }} Now that you know how to get a name, how do you get Ivar?
Ivar _Nullable
class_getInstanceVariable(Class _Nullable cls, const char * _Nonnull name)
OBJC_AVAILABLE(10.0, 2.0, 9.0, 1.0, 2.0);
Ivar _Nullable
class_getClassVariable(Class _Nullable cls, const char * _Nonnull name)
OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);
Copy the code
Class_getInstanceVariable is an instance variable named name in the CLS class. Class_getClassVariable is a CLS class variable named name. OC syntax does not have a class variable, so this method is useless. Let’s create a new Cat class and add a member variable int _age and a property @property(nonatomic,copy)NSString* name. As we know, properties automatically generate a member variable preceded by an _ (name generates _name).
-(void)getIvar {
Ivar ivar = class_getInstanceVariable(objc_getClass("Cat"), "_name");
Ivar ivar1 = class_getInstanceVariable(objc_getClass("Cat"), "_age");
[self logIvarName:ivar];
[self logIvarName:ivar1];
}
Copy the code
Running results:
2019-02-26 11:42:38.646792+0800 Run-time Demo[59730:4976606] name = _name 2019-02-26 11:42:38.646845+0800 Runtime-Demo[59730:4976606] name = _ageCopy the code
There it is, and it is a member variable. So how do you get all the member variables of a class? Use the following method:
Ivar _Nonnull * _Nullable
class_copyIvarList(Class _Nullable cls, unsigned int * _Nullable outCount)
OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);
Copy the code
BOOL sex and @property(nonatomic, strong)Person* master;
-(void)copyIvarList {
unsigned int count;
Ivar* ivars =class_copyIvarList(objc_getClass("Cat"), &count);
for (unsigned int i = 0; i < count; i++) {
Ivar ivar = ivars[i];
[self logIvarName:ivar];
}
free(ivars);
}
Copy the code
Running results:
2019-02-26 11:50:51.090761+0800 Runtime-demo [59875:4979802] name = _age 2019-02-26 11:50:51.090799+0800 Runtime-demo [59875:4979802] Name = _sex 2019-02-26 11:50:51.090809+0800 Runtime-demo [59875:4979802] name = _name 2019-02-26 11:50:51.090817+0800 Runtime-Demo[59875:4979802] name = _masterCopy the code
If you want to get the type of a member variable, use the following method:
Const char * _Nullable ivar_getTypeEncoding(Ivar _Nonnull V) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);Copy the code
Let’s try to get the type of the _name:
-(void)getTypeEncoding {
Ivar ivar = class_getInstanceVariable(objc_getClass("Cat"), "_name");
const char* type = ivar_getTypeEncoding(ivar);
NSLog(@"type = %s".type);
}
Copy the code
Running results:
type = @"NSString"
Copy the code
Name is indeed an NSString. The next three methods we look at are assigning or evaluating ivar.
Id _Nullable object_getIvar(ID _Nullable obj, Ivar _Nonnull Ivar) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0); Void object_setIvar(id _Nullable obj, Ivar _Nonnull Ivar, ID _Nullable value) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0); Void object_setIvarWithStrongDefault(id _Nullable obj, Ivar _Nonnull Ivar, ID _Nullable value) OBJC_AVAILABLE(10.12, 10.0, 10.0, 3.0, 2.0);Copy the code
Object_getIvar This method is a function that evaluates ivar. Let’s test it out:
-(void)getIvarValue {
Cat* cat = [Cat new];
Ivar ivar = class_getInstanceVariable(objc_getClass("Cat"), "_name");
NSString* name = object_getIvar(cat, ivar);
NSLog(@"Before assignment: %@",name);
cat.name = @"jack";
NSString* name2 = object_getIvar(cat, ivar);
NSLog(@"After assignment: %@",name2);
}
Copy the code
Running results:
2019-02-26 15:44:11.758498+0800 Runtime-Demo[63973:5079569] before: (NULL) 2019-02-26 15:44:11.758541+0800 Runtime-demo [63973:5079569] : set to JackCopy the code
Object_setIvar and object_setIvarWithStrongDefault are both related to memory management. To start with what they have in common, there is no difference if memory management is a known memory management style (member variables or attributes are ARC, strong, or weak). The difference is that if it is an unknown memory management mode, object_setIvar assigns the instance variable to unsafe_unretain, whereas object_setIvarWithStrongDefault assigns the instance variable to strong. First we need to understand the three concepts: strong,weak and unsafe_unretain. Strong is a strong reference that points to and owns the object, and memory is freed based on whether retainCount is 0. Weak is a weak reference that points to but does not own the object. Pointer is automatically set to nil when space is freed. Unsafe_unretain is similar to weak, except that the pointer is not set to nil when the space is freed, so there is the harm of wild Pointers. So, in ARC, these two methods work almost exactly the same. Add two new attributes, @property(nonatomic, copy)NSString* style and @property(nonatomic, copy)NSString* breed.
-(void)setIvar {
Cat* cat = [Cat new];
Ivar ivar = class_getInstanceVariable(objc_getClass("Cat"), "_breed");
Ivar ivar2 = class_getInstanceVariable(objc_getClass("Cat"), "_style");
object_setIvar(cat, ivar,@"The short");
object_setIvar(cat, ivar2,@"Lively");
NSLog(@"breed = %@",cat.breed);
NSLog(@"style = %@",cat.style);
}
Copy the code
Running results:
Breed = breed [66371:5132652] and breed = breed Runtime-demo [66371:5132652] style = LivelyCopy the code
The assignment function is perfectly handy. The following method is to get the offset of the instance variable, that is, the memory offset, and we can see the memory address of the variable.
Ptrdiff_t ivar_getOffset(Ivar _Nonnull V) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);Copy the code
Let’s test the Cat class. Let’s look at the distribution of Cat attributes and variables:
Cat.h
@interface Cat : NSObject
{
@public
int _age;
}
@property(nonatomic, copy)NSString* name;
@property(nonatomic, copy)NSString *breed;
@property(nonatomic, copy)NSString* style;
@end
Cat.m
@interface Cat()
{
BOOL _sex;
}
@property(nonatomic, strong)Person* master;
@end
@implementation Cat
@end
Copy the code
We see that there are four attributes and two member variables in the Cat class. Now we print the ptrDIFF_t for each variable by getting the list of variables
-(void)getOffset {
unsigned int count;
Ivar* ivars =class_copyIvarList(objc_getClass("Cat"), &count);
for (unsigned int i = 0; i < count; i++) {
Ivar ivar = ivars[i];
ptrdiff_t offset = ivar_getOffset(ivar);
NSLog(@"%s = %td",ivar_getName(ivar),offset);
}
free(ivars);
NSLog(@"Cat total bytes = %lu",class_getInstanceSize(objc_getClass("Cat")));
}
Copy the code
Running results:
2019-02-26 20:09:16.296160+0800 runtime-demo [17275:490666] _age = 8 2019-02-26 20:09:16.296274+0800 Runtime-Demo[17275:490666] _sex = 12 2019-02-26 20:09:16.296364+0800 Runtime-Demo[17275:490666] _name = 16 2019-02-26 20:09:16.296452+0800 Runtime-demo [17275:490666] _breed = 24 2019-02-26 20:16.296525 +0800 Runtime-demo [17275:490666] _style = 32 2019-02-26 20:09:16.296666+0800 Runtime-demo [17275:490666] _master = 40 2019-02-26 20:09:16.296765+0800 Runtime-demo [17275:490666] Total bytes of Cat = 48Copy the code
_age starts at byte 8 and takes 4 bytes. Then byte 12 starts at byte _sex and takes 4 bytes. Byte 16 is _name and takes 8 bytes. Byte 24 is _breed and takes 8 bytes. Up to 40 bytes is _master, which is 8 bytes. The amount of memory they take up is determined by their own type and memory alignment.
The following function adds variables to a dynamic class. What is a dynamic class? We talked in the first article about dynamically creating classes by using the objc_allocateClassPair function, The class_addIvar function must be added after objc_allocateClassPair before objc_registerClassPair.
BOOL
class_addIvar(Class _Nullable cls, const char * _Nonnull name, size_t size,
uint8_t alignment, const char * _Nullable types)
OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);
Copy the code
CLS is the class to which you want to add the instance variable, size is the number of bytes of memory, types is the type of the instance variable, alignment is alignment, the official documentation has a formula log2(sizeof(pointer_type)). Let’s test it out:
-(void)addIvar {
Class class = objc_allocateClassPair(objc_getClass("NSObject"), "Dog", 0);
float alignment = log2f(sizeof(int));
class_addIvar(class, "age", sizeof(int), alignment, "int");
objc_registerClassPair(class);
Ivar ivar = class_getInstanceVariable(class, "age");
NSLog(@"name = %s",ivar_getName(ivar));
NSLog(@"size = %zu",class_getInstanceSize(objc_getClass("Dog")));
}
Copy the code
Running results:
2019-02-26 20:44:46.198155+0800 runtime-demo [19229:519808] name = age 2019-02-26 20:44:46.198295+0800 Runtime-Demo[19229:519808] size = 16Copy the code
Can print out the instance variable of the new class.
The following four methods have to do with variable layout, which I find the most difficult to understand. The concept of IvarLayout is not explained in Runtime.h.
Const Uint8_t * _Nullable class_getIvarLayout(Class _Nullable CLS) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0); Const Uint8_t * _Nullable class_getWeakIvarLayout(Class _Nullable CLS) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0); Void class_setIvarLayout(Class _Nullable CLS, const Uint8_t * _Nullable Layout) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0); Void class_setWeakIvarLayout(Class _Nullable CLS, const Uint8_t * _Nullable Layout) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);Copy the code
If you want to learn more about layout, check out the IvAR Memory Layout at Runtime. I’m not going to go into detail here.
4.objc_property or objc_property_t
Properties should be the most familiar, which is equivalent to adding modifiers to instance variables and automatically generating set and GET methods, which are easy to use. The runtime structure for properties is objc_property or objc_property_t. We don’t know what the structure is, but we do know another one:
typedef struct {
const char * _Nonnull name; /**< The name of the attribute */
const char * _Nonnull value; /**< The value of the attribute (usually empty) */
} objc_property_attribute_t;
Copy the code
We can get some information about attributes indirectly through objc_property_attribute_t. And the method property_copyAttributeList method gets objC_property_attribute_t by passing it in
objc_property_attribute_t * _Nullable
property_copyAttributeList(objc_property_t _Nonnull property,
unsigned int * _Nullable outCount)
OBJC_AVAILABLE(10.7, 4.3, 9.0, 1.0, 2.0);
Copy the code
Let’s write a method to encapsulate this method:
-(void)logProperty:(objc_property_t)property {
NSLog(@"-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -");
unsigned int count;
objc_property_attribute_t* attributeList = property_copyAttributeList(property, &count);
for (unsigned int i = 0; i < count; i++) {
objc_property_attribute_t attribute = attributeList[i];
NSLog(@"name = %s",attribute.name);
NSLog(@"value = %s",attribute.value); }}Copy the code
Later we will use this method to print the information related to the property. So how do you get objc_property_t?
Objc_property_t _Nullable class_getProperty(Class _Nullable CLS, const char * _Nonnull name) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);Copy the code
@property(nonatomic, copy)NSString* name, @property(nonatomic, copy)NSString* name, @property(nonatomic, copy)NSString* style, @property(nonatomic, strong)Person* master. Let’s get the name attribute separately.
-(void)getProperty {
objc_property_t property = class_getProperty(objc_getClass("Cat"), "name");
[self logProperty:property];
}
Copy the code
Print result:
2019-02-27 09:37:17.172874+0800 runtime-demo [72525:5355290] name = T 2019-02-27 09:37:17.172916+0800 Runtime-Demo[72525:5355290] value = @"NSString"2019-02-27 09:37:17.172929+0800 runtime-demo [72525:5355290] name = C 2019-02-27 09:37:17.172950+0800 [72525:5355290] value = 2019-02-27 09:37:17.172965+0800 Runtime-demo [72525:5355290] name = N 2019-02-27 09:37:17.172975+0800 runtime-demo [72525:5355290] value = 2019-02-27 09:37:17.172985+0800 runtime-demo [72525:5355290] Name = V 2019-02-27 09:37:17.172995+0800 Runtime-demo [72525:5355290] value = _nameCopy the code
Name is T and V, T stands for type, the type of the attribute, V stands for ivar, and _name stands for ivar of the attribute. Other ones that don’t have values, those modifiers, C for copy, N for nonatomic. From this we can conclude that:
name | value | meaning |
---|---|---|
T | There are | Type of property |
V | There are | The name of the instance variable generated by the |
C | There is no | copy |
N | There is no | nonatomic |
W | There is no | weak |
& | There is no | Object types default to &, such as strong and readwrite |
R | There is no | readonly |
Note: If there is no N, it is atomic.
You can also get a list of attributes for a class. For printing convenience, we’ll just print the name of the property this time, using the property_getName method:
Const char * _Nonnull property_getName(objC_property_t _Nonnull property) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);Copy the code
Here we print the names of the following tables:
objc_property_t _Nonnull * _Nullable
class_copyPropertyList(Class _Nullable cls, unsigned int * _Nullable outCount)
OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);
Copy the code
Again, take Cat:
-(void)copyPropertyList {
unsigned int count;
objc_property_t* propertyList = class_copyPropertyList(objc_getClass("Cat"), &count);
for (unsigned int i = 0; i < count; i++) {
objc_property_t property = propertyList[i];
NSLog(@"name = %s",property_getName(property));
}
free(propertyList);
}
Copy the code
Running results:
2019-02-27 10:30:33.006299+0800 run-time Demo[73443:5379227] name = master 2019-02-27 10:30:33.00638 +0800 Run-time Demo[73443:5379227] name = name 2019-02-27 10:30:33.006348+0800 Run-time Demo[73443:5379227] name = breed 2019-02-27 10:30:33.006357+0800 Runtime-demo [73443:5379227] name = styleCopy the code
The property names are printed out, and this is distinguished from ivar. If you look for IVar by a given property, you will find the underlined one. Previously we could print all the properties of a property, and the system provided two methods:
Const char * _Nullable property_getAttributes(objC_property_t _Nonnull property) OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0); char * _Nullable property_copyAttributeValue(objc_property_t _Nonnull property, Const char * _Nonnull attributeName) OBJC_AVAILABLE(10.7, 4.3, 9.0, 1.0, 2.0);Copy the code
Let’s start by testing the property_getAttributes function
-(void)getAttributes {
objc_property_t property = class_getProperty(objc_getClass("Cat"), "name");
const char* attributes = property_getAttributes(property);
NSLog(@"attributes = %s",attributes);
}
Copy the code
Running results:
attributes = T@"NSString",C,N,V_name
Copy the code
The printed result is the same as before, this time as a string. Now look at the property_copyAttributeValue method, which gets a separate value from the attributeName.
-(void)copyAttributeValue {
objc_property_t property = class_getProperty(objc_getClass("Cat"), "name"); Ivar char* value = property_copyAttributeValue(property, property_copyAttributeValue)"V");
NSLog(@"value = %s",value);
}
Copy the code
Running results:
value = _name
Copy the code
From the previous print, this print is correct. The following two methods add or replace properties dynamically
BOOL
class_addProperty(Class _Nullable cls, const char * _Nonnull name,
const objc_property_attribute_t * _Nullable attributes,
unsigned int attributeCount)
OBJC_AVAILABLE(10.7, 4.3, 9.0, 1.0, 2.0);
void
class_replaceProperty(Class _Nullable cls, const char * _Nonnull name,
const objc_property_attribute_t * _Nullable attributes,
unsigned int attributeCount)
OBJC_AVAILABLE(10.7, 4.3, 9.0, 1.0, 2.0);
Copy the code
Add a Property of the form @property(nonatomic, copy, readOnly)NSString* MOOD. The value of T is NSString, the value of V is _mood, and the value of nonatomic is N, copy is C, and readonly is R. So we know that the attribute has T,V,C,N, and R. Ok, let’s write code!
-(void)addProperty {
unsigned int count = 5;
objc_property_attribute_t attributeList[count];
objc_property_attribute_t attribute1 ;
attribute1.name = "T";
attribute1.value = "NSString";
objc_property_attribute_t attribute2 ;
attribute2.name = "V";
attribute2.value = "_mood";
objc_property_attribute_t attribute3 ;
attribute3.name = "N";
attribute3.value = "";
objc_property_attribute_t attribute4 ;
attribute4.name = "C";
attribute4.value = "";
objc_property_attribute_t attribute5 ;
attribute5.name = "R";
attribute5.value = "";
attributeList[0] = attribute1;
attributeList[1] = attribute2;
attributeList[2] = attribute3;
attributeList[3] = attribute4;
attributeList[4] = attribute5;
BOOL isSuccess = class_addProperty(objc_getClass("Cat"), "mood", (const objc_property_attribute_t *)&attributeList, count);
NSLog(@"New % @",isSuccess? @"Success": @"Failure");
[self copyPropertyList];
objc_property_t property = class_getProperty(objc_getClass("Cat"), "mood");
const char* attributes = property_getAttributes(property);
NSLog(@"attributes = %s",attributes);
}
Copy the code
Running results:
2019-02-27 11:52:49.325561+0800 Runtime-Demo[74832:5417422] Added successfully 2019-02-27 11:52:49.325614+0800 Runtime-Demo[74832:5417422] Name = mood 2019-02-27 11:52:49.325632+0800 Runtime-Demo[74832:5417422] name = master 2019-02-27 11:52:49.325650+0800 Runtime-demo [74832:5417422] name = name 2019-02-27 11:52:49.325662+0800 Breed breed [74832:5417422] name = breed breed 2019-02-27 11:52:49.325674+0800 Runtime-Demo[74832:5417422] name = style 2019-02-27 11:52:49.325709+0800 Runtime-Demo[74832:5417422] Attributes = TNSString,V_mood,N,C,RCopy the code
The addition succeeded, and the printed property list also has mood. The printed Attributes are also fine. So class_replaceProperty and I’m going to change the name of the name property to catName. Again, let’s look at the objc_property_attribute_t list, the name property is @property(nonatomic, copy)NSString* name, if you just change the name, T,C,N stays the same, it becomes V, The value of V becomes _catName. So the code is:
-(void)replaceProperty {
unsigned int count = 4;
objc_property_attribute_t attributeList[count];
objc_property_attribute_t attribute1 ;
attribute1.name = "T";
attribute1.value = "NSString";
objc_property_attribute_t attribute2 ;
attribute2.name = "V";
attribute2.value = "_mood";
objc_property_attribute_t attribute3 ;
attribute3.name = "N";
attribute3.value = "";
objc_property_attribute_t attribute4 ;
attribute4.name = "C";
attribute4.value = "";
attributeList[0] = attribute1;
attributeList[1] = attribute2;
attributeList[2] = attribute3;
attributeList[3] = attribute4;
class_replaceProperty(objc_getClass("Cat"), "name", (const objc_property_attribute_t*)&attributeList, count);
[self copyPropertyList];
objc_property_t property = class_getProperty(objc_getClass("Cat"), "name");
const char* attributes = property_getAttributes(property);
NSLog(@"attributes = %s",attributes);
}
Copy the code
Running results:
2019-02-27 11:58:46.34340 +0800 runruntime -Demo[74940:5421075] name = master 2019-02-27 11:58:46.34340 +0800 Runtime-demo [74939:5421075] name = name 2019-02-27 11:58:46.341980+0800 Runtime-demo [74939:5421075] name = name breed 2019-02-27 11:58:46.341988+0800 Runtime-demo [74939:5421075] name = style 2019-02-27 11:58:46.342016+0800 Runtime-Demo[74939:5421075] attributes = TNSString,V_mood,N,CCopy the code
The printout was a complete surprise to me, with no catName at all, but the printout attributes were changed attributes. Why is that? We’ll look at it from the source:
struct property_t {
const char *name;
const char *attributes;
};
Copy the code
The property_t structure is divided into name and Attributes.
BOOL
class_addProperty(Class cls, const char *name,
const objc_property_attribute_t *attrs, unsigned int n)
{
return _class_addProperty(cls, name, attrs, n, NO);
}
void
class_replaceProperty(Class cls, const char *name,
const objc_property_attribute_t *attrs, unsigned int n)
{
_class_addProperty(cls, name, attrs, n, YES);
}
Copy the code
Both the class_addProperty and class_replaceProperty layers call the _class_addProperty method, but the boilervalues are passed differently. Let’s take a look at the _class_addProperty method,
static bool
_class_addProperty(Class cls, const char *name,
const objc_property_attribute_t *attrs, unsigned int count,
bool replace)
{
if(! cls)return NO;
if(! name)return NO;
property_t *prop = class_getProperty(cls, name);
if(prop && ! replace) { // already exists, refuse to replacereturn NO;
}
else if (prop) {
// replace existing
rwlock_writer_t lock(runtimeLock);
try_free(prop->attributes);
prop->attributes = copyPropertyAttributeString(attrs, count);
return YES;
}
else {
rwlock_writer_t lock(runtimeLock);
assert(cls->isRealized());
property_list_t *proplist = (property_list_t *)
malloc(sizeof(*proplist));
proplist->count = 1;
proplist->entsizeAndFlags = sizeof(proplist->first);
proplist->first.name = strdupIfMutable(name);
proplist->first.attributes = copyPropertyAttributeString(attrs, count);
cls->data()->properties.attachLists(&proplist, 1);
returnYES; }}Copy the code
Property_t *prop = class_getProperty(CLS, name); Prop = YES; replace = YES; prop = YES;
else if (prop) {
// replace existing
rwlock_writer_t lock(runtimeLock);
try_free(prop->attributes);
prop->attributes = copyPropertyAttributeString(attrs, count);
return YES;
}
Copy the code
From this paragraph we can see that this section only changes prop-> Attributes. Also not changed prop->name. Therefore, the name of the property we print does not change. The best use of class_replaceProperty, then, is to modify types or modifiers. `