Runtime--Instance Variables

簡介

Instance Variables(以下稱實例變量)指在一個對象生存時存在并且保存值的變量,其內存申請和釋放和對象的創建 (alloc)和銷毀(dealloc)同步進行。
比如在header文件中

@interface Data : NSObject
{
    //Instance Variables 區
}
@end

或者在implementation文件中

#import "Data.h"

@interface Data ()
{
  //Instance Variables 區
}
@end


@implementation Data
@end

標識的區域都可以聲明屬性變量。

與Runtime

注:主要介紹ARC環境下Instance Variables(以下簡稱Ivar)在Runtime的操作。
Ivar的官方聲明

/// An opaque type that represents an instance variable.
typedef struct objc_ivar *Ivar;

相關的操作有

//獲取Ivar名稱
const char *ivar_getName(Ivar v) ;

//獲取@encode()
const char *ivar_getTypeEncoding(Ivar v);

//獲取內存偏移量
ptrdiff_t ivar_getOffset(Ivar v);

下邊通過簡單的代碼示例展示相關用法,首先聲明一個簡單的類

@interface Data : NSObject
{
    NSString  *_firstName;
    NSString  *_firstName2;
}
@property (nonatomic,copy)NSString  *secondName;
@property (nonatomic,copy)NSString  *secondName2;
@end

獲取Ivar

獲取指定的Ivar

//獲取實例變量
Ivar var_Instance = class_getInstanceVariable([Data class], "_firstName");
const char *name_Instance = ivar_getName(var_Instance);
NSLog(@"var_Instance:%@",[NSString stringWithUTF8String:name_Instance]);

//輸出
var_Instance:_firstName

Ivar相關屬性的獲取

const char *name_var = ivar_getName(var_Instance);//名稱
const char *type_var = ivar_getTypeEncoding(var_Instance);//有對照表
ptrdiff_t offset_var = ivar_getOffset(var_Instance);//獲取變量內存偏移量
    
NSLog(@"Ivar Name:%@ TypeEncoding:%@ Offset:%td",[NSString stringWithUTF8String:name_var],[NSString stringWithUTF8String:type_var],offset_var);

//輸出
Ivar Name:_firstName TypeEncoding:@"NSString" Offset:8

關于輸出結果說明:
1、Name:_firstName同Ivar的名稱一樣
2、TypeEncoding:@"NSString":聲明的變量類型,同@encode()命令的結果;Type Encodings
3、Offset:8 在32和64位下不同,具體值跟指針變量在當前系統下占用空間有關,可以通過sizeof(<#expression-or-type#>)查看驗證

獲取Ivar列表

Ivar *class_copyIvarList(Class cls, unsigned int *outCount) 

示例如下

/* 獲取實例變量列表 */
unsigned int count = 0;
Ivar *list =  class_copyIvarList([Data class], &count);
for (int i = 0; i < count ; i ++)
{
   Ivar var = list[i];
   const char *name_var = ivar_getName(var);
   const char *type_var = ivar_getTypeEncoding(var);//有對照表
   ptrdiff_t offset_var = ivar_getOffset(var);//獲取變量內存偏移量

   NSLog(@"Ivar Name:%@ TypeEncoding:%@ Offset:%td",[NSString stringWithUTF8String:name_var],[NSString stringWithUTF8String:type_var],offset_var);
}
free(list);//必須free

//輸出
Ivar Name:_firstName TypeEncoding:@"NSString" Offset:8
Ivar Name:_firstName2 TypeEncoding:@"NSString" Offset:16
Ivar Name:_secondName TypeEncoding:@"NSString" Offset:24
Ivar Name:_secondName2 TypeEncoding:@"NSString" Offset:32

提示:在header和implementation文件中聲明的Ivar,以上獲取方法均可以正常獲取。

Ivar Value

對于Ivar的值操作無法以下兩種:

//獲取
id object_getIvar(id obj, Ivar ivar) ;

//設置
void object_setIvar(id obj, Ivar ivar, id value);

示例代碼如下

Data *data = [[Data alloc] init];
data.secondName = @"secondName";//只賦值一個
    
//獲取
Ivar var_1 = class_getInstanceVariable([Data class], "_secondName");
id value_1 =  object_getIvar(data, var_1);
NSLog(@"value_1:%@",value_1);
    
//設置
NSLog(@"設置前:%@",data.secondName2);
Ivar var_2 = class_getInstanceVariable([Data class], "_secondName2");
object_setIvar(data, var_2, @"secondName2");
id value =  object_getIvar(data, var_2);
NSLog(@"var_2:%@",value);
NSLog(@"設置后:%@",data.secondName2);

//輸出
value_1:secondName
設置前:(null)
var_2:secondName2
設置后:secondName2

添加Ivar

/** 
 * Adds a new instance variable to a class.
 * 
 * @return 如果成功返回YES,否則NO (比如Ivar已存在)
 *         
 * @note 只能在objc_allocateClassPair 和 before objc_registerClassPair之間調用;
 * @note 不支持已有類添加Ivar。 
 * @note 不支持metaclass.
 * @note size參數 : sizeof(pointer_type)
 * @note alignment參數 : log2(sizeof(pointer_type))
 * @note types參數 :@encode(type)
 */
 
BOOL class_addIvar(Class cls, const char *name, size_t size, 
                               uint8_t alignment, const char *types)

示例代碼

Class newClass = objc_allocateClassPair([NSObject class], "Data3", 0);
class_addIvar(newClass, "add", sizeof(NSString*), log2(sizeof(NSString*)), @encode(NSString));
objc_registerClassPair(newClass);
    

unsigned int count = 0;
Ivar *list =  class_copyIvarList(objc_getClass("Data3"), &count);
for (int i = 0; i < count ; i ++)
{
   Ivar var = list[i];
   const char *name_var = ivar_getName(var);
   const char *type_var = ivar_getTypeEncoding(var);//有對照表
   ptrdiff_t offset_var = ivar_getOffset(var);//獲取變量內存偏移量
   
   NSLog(@"Ivar Name:%@ TypeEncoding:%@ Offset:%td",[NSString stringWithUTF8String:name_var],[NSString stringWithUTF8String:type_var],offset_var);
}
free(list);

//輸出
Ivar Name:add TypeEncoding:{NSString=#} Offset:8

此部分更深層的內容參考此文章此問答
此文章的其它參考資料Objective-C Runtime

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,933評論 18 139
  • runtime 和 runloop 作為一個程序員進階是必須的,也是非常重要的, 在面試過程中是經常會被問到的, ...
    made_China閱讀 1,226評論 0 7
  • 我們常常會聽說 Objective-C 是一門動態語言,那么這個「動態」表現在哪呢?我想最主要的表現就是 Obje...
    Ethan_Struggle閱讀 2,231評論 0 7
  • 對于從事 iOS 開發人員來說,所有的人都會答出【runtime 是運行時】什么情況下用runtime?大部分人能...
    夢夜繁星閱讀 3,732評論 7 64
  • 這些年來研究國學經典的人很多。熊逸顯然算不上名頭大的。可是﹣這個預言我愿立此存照﹣歷史也許不會記住“百家講壇”的群...
    平常心de閱讀 1,197評論 1 3