C语言是静态语言,它的工作方式是通过函数调用,这样在编译时我们就已经确定程序如何运行的。而Objective-C是动态语言,它并非通过调用类的方 法来执行功能,而是给对象发送消息,对象在接收到消息之后会去找匹配的方法来运行。这种做法就把C语言在编译时的工作挪到了运行时来做,可以获得额外的灵 活性。

      在Objective-C中有个@selector,在很多地方被翻译成“选择子”。实际上,对于类的实例对象来说,类的方法是用一个数字来代表的,并非 是我们看到的一个长长的带着:这个字符的一串字符串。通过这个@selector就可以把这个方法的名字转成所对应的数字。当一个类确定后,实际上每个方 法的@selector的值就是固定的,说到这里,你一定可以想到method swizzling是什么一个东东了,没错,如果我们原来有个A方法,@selector(A)就是一个数字,我们的对象在接收到一个消息后就去查找对应 的方法并运行——如果,我们把@selector(B)的数字换成了原来@selector(A)的数字,那么此时对象虽然受到A消息,但会去运行B方 法!

      在iOS中,这是完全可以实现的,那么我们什么时候会需要这么做呢?我觉得有2个时候:

1. 破解,毋庸讳言,这绝对是破解的利器,不解释了。

2. 在开发调试过程中,如果你对某个库里的方法不确定或者觉得需要扩展的时候,你可以自己写一个去代替它。因为Objective-C是有Category的,所以扩展功能没啥必要,但调试时增加一些打印语句是很方便实际的。


     举个例子,NSString里面的lowercaseString方法,如果我不太清楚这个方法都干了什么,我就可以自己写个方法来替换它,这个方法里面增加打印语句,这样log里面就一目了然了。

     首先需要增加一个NSString的Category

  1. @interface NSString (wzTest)  
  2.   
  3. - (NSString*)myLowerString;  
  4.   
  5. @end  
  6.   
  7. @implementation NSString (wzTest)  
  8.   
  9. - (NSString*)myLowerString  
  10. {  
  11.     NSString *lowerString = [self myLowerString];  
  12.     NSLog(@"%@ => %@"self, lowerString);  
  13.     return  lowerString;  
  14. }  
  15.   
  16. @end  
      这里有一个地方解释一下,在myLowerString方法里面,看起来递归调用了自身。但是,我们会用原来的lowercaseString方法去替换自己写的myLowerString方法,所以这里并没有调用自身,而是调用了原来的lowercaseString方法。这点请注意一下。    

      其次替换系统原来的lowercaseString方法,使用runtime里面的方法。

  1. Method originalMethod = class_getInstanceMethod([NSString class], @selector(lowercaseString));  
  2. Method swapMethod = class_getInstanceMethod([NSString class], @selector(myLowerString));  
  3. method_exchangeImplementations(originalMethod, swapMethod);  
  4. NSString *testStr = @"thIs is THE Test STRING";  
  5. NSLog(@"lowerString of testStr=%@", [testStr lowercaseString]);  

      我们来看一下log的结果:

2014-05-29 22:17:55.514 testTableView[1582:a0b] thIs is THE Test STRING => this is the test string

2014-05-29 22:17:55.514 testTableView[1582:a0b] lowerString of testStr=this is the test string

      我们可以看到,系统中使用是继续使用lowercaseString方法的,不过实际执行的是我们新增的方法。当你不需要这样做的时候,关闭method swizzling方法就可以恢复了。

      我们的例子中是增加了打印语句,实际上还可以做更多地操作。这在用第三方库调试的时候是非常有用的一个方法,可以很方便的查看变量的内容或做一些其他工作。调试结束后,关闭method swizzling就可以正常的工作。