1.如何用runtime给键盘添加工具栏和按钮响应事件:
.h #import <Foundation/Foundation.h> #import <UIKit/UIKit.h> @interface KeyBoardTool : NSObject /** * 增加隐藏键盘按钮 * * @param textfield 输入框对象 */ + (void)hideKeyboard:(UITextField *)textfield; @end .m #pragma mark - 增加隐藏键盘按钮 + (void)hideKeyboard:(UITextField *)textfield { //为键盘增加工具栏 UIToolbar * topView = [[UIToolbar alloc] initWithFrame:CGRectMake(0, 0, ScreenWidth, 40)]; [topView setBarStyle:UIBarStyleDefault]; [textfield setInputAccessoryView:topView]; UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom]; btn.frame = CGRectMake(0, 5, 40, 30); [btn addTarget:self action:@selector(dismissKeyBoard:) forControlEvents:UIControlEventTouchUpInside]; [btn setBackgroundImage:[UIImage imageNamed:@"closed"] forState:UIControlStateNormal]; //将TextField绑定到button上,kTextField要和UITextField创建的对象名一致 /* 第一个参数是需要添加属性的对象; 第二个参数是属性的key; 第三个参数是属性的值; 第四个参数是一个枚举值,类似@property属性创建时设置的关键字,可从命名看出各含义 */ objc_setAssociatedObject(btn, kTextField, textfield, OBJC_ASSOCIATION_RETAIN_NONATOMIC); //UIBarButtonSystemItemFlexibleSpace自动调节按钮间距 UIBarButtonItem * leftBtn = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemFixedSpace target:self action:nil]; //如果希望左侧显示按钮,重新实例化一个按钮用 CustomView 方法设置,类似下面的rightBtn。 UIBarButtonItem *rightBtn = [[UIBarButtonItem alloc] initWithCustomView:btn]; NSArray * buttonsArray = [NSArray arrayWithObjects:leftBtn,rightBtn,nil]; [topView setItems:buttonsArray]; } + (void)dismissKeyBoard:(UIButton *)button{ //获取button上对应的属性,kTextField要和UITextField创建的对象名一致 [objc_getAssociatedObject(button, kTextField) resignFirstResponder]; }
其实这些都是固定的方法,一开始接触绑定的方法看不懂,也没什么好理解的,记住就好。
2.获取某一个类所有属性和所有方法
+ (void)getAllPropertyAndAllMethod:(id)myClass { unsigned int outCount = 0; // 获取到所有的成员变量列表 Ivar *vars = class_copyIvarList([myClass class], &outCount); // 遍历所有的成员变量 for (int i = 0; i < outCount; i++) { Ivar ivar = vars[i]; // 取出第i个位置的成员变量 const char *propertyName = ivar_getName(ivar); // 获取变量名 const char *propertyType = ivar_getTypeEncoding(ivar); // 获取变量编码类型 printf("%s/%s\n", propertyName, propertyType); } unsigned int count; //获取方法列表,所有在.m文件显式实现的方法都会被找到,包括setter+getter方法; Method *allMethods = class_copyMethodList([myClass class], &count); for(int i =0;i<count;i++) { //Method,为runtime声明的一个宏,表示对一个方法的描述 Method md = allMethods[i]; //获取SEL:SEL类型,即获取方法选择器@selector() SEL sel = method_getName(md); //得到sel的方法名:以字符串格式获取sel的name,也即@selector()中的方法名称 const char *methodname = sel_getName(sel); NSLog(@"(Method:%s)",methodname); } }
通过以上方法可以看到许多隐藏的属性和方法,但是,请慎用,比较常用的方法有设置placeholder的字体大小:
[_loginTestField setValue:[UIFont systemFontOfSize:14] forKeyPath:@"_placeholderLabel.font"];
3.改变属性值
+ (void)changeVariable { ViewController *viewVC = [[ViewController alloc]init]; NSLog(@"改变前的viewVC:%@",viewVC); unsigned int count = 0; Ivar *vars = class_copyIvarList([ViewController class], &count); Ivar ivv = vars[0]; //从第一个方法getAllVariable中输出的控制台信息,我们可以看到实例属性。 object_setIvar(viewVC, ivv, @"world"); //属性被强制改为world。 NSLog(@"改变之后的viewVC:%@",viewVC); }
从上面方法中能看出来改变的是ViewController中第一个实例属性的值。
4.增加新方法
+ (void)addNewMethod { /* 动态添加方法: 第一个参数表示Class cls 类型; 第二个参数表示待调用的方法名称; 第三个参数(IMP)addingMethod,IMP一个函数指针,这里表示指定具体实现方法addingMethod; 第四个参数表方法的参数,0代表没有参数; */ ViewController *viewVC = [[ViewController alloc]init]; class_addMethod([viewVC class], @selector(myNewMethod), (IMP)addingMethod, 0); //调用方法 [viewVC performSelector:@selector(myNewMethod)]; } //这个方法不调用,但是不写的话会报警告 - (void)myNewMethod { } //具体的实现(方法的内部都默认包含两个参数Class类和SEL方法,被称为隐式参数。) int addingMethod(id self, SEL _cmd){ NSLog(@"已新增方法:myNewMethod"); return 1; }
5.交换方法
+ (void)exchangeMethod { ViewController *viewVC = [[ViewController alloc]init]; Method method1 = class_getInstanceMethod([viewVC class], @selector(method1)); Method method2 = class_getInstanceMethod([viewVC class], @selector(method2)); //交换方法 method_exchangeImplementations(method1, method2); [viewVC method1]; }
在ViewController中有method1和method2两个方法,method_exchangeImplementations交换了这两个方法,所以当调用method1,实际上调用的是method2.
6.增加新属性
假如要给ViewController添加新属性,先创建一个继承于ViewController的category,然后下面看代码:
.h #import "ViewController.h" @interface ViewController (VVC) @property (nonatomic,strong)NSString *vccVariable; @end .m #import "ViewController+VVC.h" #import <objc/runtime.h> //runtime API的使用需要包含此头文件 const char * vcc = "vcckey"; //做为key,字符常量 必须是C语言字符串; @implementation ViewController (VVC) -(void)setVccVariable:(NSString *)vccVariable{ NSString *newStr = [NSString stringWithFormat:@"%@",vccVariable]; /* 第一个参数是需要添加属性的对象; 第二个参数是属性的key; 第三个参数是属性的值; 第四个参数是一个枚举值,类似@property属性创建时设置的关键字,可从命名看出各含义 */ objc_setAssociatedObject([ViewController class],vcc, newStr, OBJC_ASSOCIATION_COPY_NONATOMIC); } //提取属性的值: -(NSString *)vccVariable{ NSString *myVccVariable = objc_getAssociatedObject([ViewController class], vcc); return myVccVariable; } @end
Demo下载地址:https://github.com/codeliu6572/Runtime_use
学习runtime,也许看起来不是很好懂,就好比我们刚学习一门新的语言,我们从没见过这样的代码,最简单的方法就是记住,但是对于runtime,用得好的话能够省很多事,用不好的后果可能会被苹果拒绝,甚至说根本驾驭不了,博主这里只是浅层次的运用,每一门语言都博大精深,需要认真去研究。