ios开发runtime学习二:runtime交换方法

#import "ViewController.h"

/*
    Runtime(交换方法):主要想修改系统的方法实现

    需求:

    比如说有一个项目,已经开发了2年,忽然项目负责人添加一个功能,每次UIImage加载图片,告诉我是否加载成功

    当系统提供的控件不能满足我们的需求的时候,我们可以

    1:通过继承系统控件,重写系统的方法,来扩充子类的行为(super的调用三种情况)
    2:当需要为系统类扩充别的属性或是方法的时候,与哪个类有关系,就为哪个类创建分类。3:利用runtime修改系统的类,增加属性,交换方法,消息机制,动态增加方法

    解决方法:1:重写系统的方法:新建类继承系统的类,重写系统的方法(要是覆盖父类的行为就不需要调用super,或是在super方法之下调用:在保留父类super原有的行为后,扩充子类自己的行为,代码写在super之上,可以修改super要传递的参数,例如重写setframe,要是想保留父类的行为就不要忘记调用super)。弊端:需要在每个类中都需要引入头文件 2:写分类:为哪个系统的类扩充属性和方法,就为哪个类写分类 3:利用runtime底层的实现来修改或是访问系统的类:增加属性,交换方法,消息机制,动态增加方法

    3:本需求利用runtime:不需要导入头文件,调用的还是系统类原来的方法,只是利用了runtime的交换方法。
     给系统的imageNamed添加功能,只能使用runtime(交互方法)
     1.给系统的方法添加分类
     2.自己实现一个带有扩展功能的方法
     3.交互方法,只需要交互一次,

     1.自定义UIImage
     2.UIImage添加分类

 */

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];

    // imageNamed => xmg_imageNamed 交互这两个方法实现
    UIImage *image = [UIImage imageNamed:@"1.png"];

}

@end
#import "UIImage+Image.h"
#import <objc/message.h>

/**
 *  总结:
    1:  + (void)load与+ (void)initialize的区别:+ (void)load:当类加载进内存的时候调用,而且不管有没有子类,都只会调用一次,在main函数之前调用,用途:1:可以新建类在该类中实现一些配置信息 2:runtime交换方法的时候,因为只需要交换一次方法,所有可以在该方法中实现交换方法的代码,用于只实现一次的代码  2:+ (void)initialize:当类被初始化的时候调用,可能会被调用多次,若是没有子类,则只会调用一次,若是有子类的话,该方法会被调用多次,若是子类的继承关系,先会调用父类的+ (void)initialize方法,然后再去调用子类的+ (void)initialize方法(若是继承关系,调用某个方法的时候,先会去父类中查找,若是父类中没有方法的实现就去子类中查找) 用途:1:在设置导航栏的全局背景的时候,只需要设置一次,可以重写该方法设置,最好是在该方法判断子类,若是自己,则实现设置全局导航栏的方法,若不是自己则跳过实现。2:在创建数据库代码的时候,可以在该方法中去创建,保证只初始化一次数据库实例,也可以用dispatch或是懒加载的方法中初始化数据库实例,也能保证只初始化一次数据库实例。其中也可以在+ (void)initialize方法中用dispatch也能保证即使有子类也只会初始化一次

   2:交换方法:1:获取某个类的方法:class_getClassMethod:第一个参数:获取哪个类的方法 第二个参数:SEL:获取哪个方法

  Method imageNamedMethod = class_getClassMethod(self, @selector(imageNamed:));

 // 交互方法:runtime
  method_exchangeImplementations(imageNamedMethod, xmg_imageNamedMethod);

 也就是外部调用xmg_imageNamed就相当于调用了imageNamed,调用imageNamed就相当于调用了xmg_imageNamed

 3:在分类中,最好不要重写系统方法,一旦重写,把系统方法实现给干掉,因为分类不是继承父类,而是继承NSObject,super没有改类的方法,所以就直接覆盖掉了父类的行为

 */
@implementation UIImage (Image)
// 把类加载进内存的时候调用,只会调用一次
+ (void)load
{
    // self -> UIImage
    // 获取imageNamed
    // 获取哪个类的方法
    // SEL:获取哪个方法
    Method imageNamedMethod = class_getClassMethod(self, @selector(imageNamed:));
    // 获取xmg_imageNamed
    Method xmg_imageNamedMethod = class_getClassMethod(self, @selector(xmg_imageNamed:));

    // 交互方法:runtime
    method_exchangeImplementations(imageNamedMethod, xmg_imageNamedMethod);
    // 调用imageNamed => xmg_imageNamedMethod
    // 调用xmg_imageNamedMethod => imageNamed
}

// 会调用多次
//+ (void)initialize
//{
//    static dispatch_once_t onceToken;
//    dispatch_once(&onceToken, ^{
//
//    });
//
//}

// 在分类中,最好不要重写系统方法,一旦重写,把系统方法实现给干掉

//+ (UIImage *)imageNamed:(NSString *)name
//{
//    // super -> 父类NSObject
//
//}

// 1.加载图片
// 2.判断是否加载成功
+ (UIImage *)xmg_imageNamed:(NSString *)name
{
    // 图片
   UIImage *image = [UIImage xmg_imageNamed:name];

    if (image) {
        NSLog(@"加载成功");
    } else {
        NSLog(@"加载失败");
    }

    return image;
}

@end

runtime 的实现原理:

二:通过继承重写实现:每次都需要导入头文件,而且项目中很多地方都得需要修改

#import <UIKit/UIKit.h>

@interface XMGImage : UIImage

@end
#import "XMGImage.h"

@implementation XMGImage

// 重写方法:想给系统的方法添加额外功能
+ (UIImage *)imageNamed:(NSString *)name
{
    // 真正加载图片:调用super初始化一张图片
    UIImage *image = [super imageNamed:name];

    if (image) {
        NSLog(@"加载成功");
    } else {
        NSLog(@"加载失败");
    }

    return image;

}

@end

* 开发使用场景:系统自带的方法功能不够,给系统自带的方法扩展一些功能,并且保持原有的功能。

* 方式一:继承系统的类,重写方法.

* 方式二:使用runtime,交换方法.

```

@implementation ViewController

- (void)viewDidLoad {

[super viewDidLoad];

// Do any additional setup after loading the view, typically from a nib.

// 需求:给imageNamed方法提供功能,每次加载图片就判断下图片是否加载成功。

// 步骤一:先搞个分类,定义一个能加载图片并且能打印的方法+ (instancetype)imageWithName:(NSString *)name;

// 步骤二:交换imageNamed和imageWithName的实现,就能调用imageWithName,间接调用imageWithName的实现。

UIImage *image = [UIImage imageNamed:@"123"];

}

@end

@implementation UIImage (Image)

// 加载分类到内存的时候调用

+ (void)load

{

// 交换方法

// 获取imageWithName方法地址

Method imageWithName = class_getClassMethod(self, @selector(imageWithName:));

// 获取imageWithName方法地址

Method imageName = class_getClassMethod(self, @selector(imageNamed:));

// 交换方法地址,相当于交换实现方式

method_exchangeImplementations(imageWithName, imageName);

}

// 不能在分类中重写系统方法imageNamed,因为会把系统的功能给覆盖掉,而且分类中不能调用super.

// 既能加载图片又能打印

+ (instancetype)imageWithName:(NSString *)name

{

// 这里调用imageWithName,相当于调用imageName

UIImage *image = [self imageWithName:name];

if (image == nil) {

NSLog(@"加载空的图片");

}

return image;

}

@end

```

时间: 2024-12-14 18:15:44

ios开发runtime学习二:runtime交换方法的相关文章

ios开发网络学习二:URL转码以及字典转模型框架MJExtension的使用

一:url转码,当url中涉及到中文的时候,要考虑转码,用UTF8对中文的url进行转码 #import "ViewController.h" @interface ViewController () @end @implementation ViewController #pragma mark ---------------------- #pragma mark Events -(void)touchesBegan:(NSSet<UITouch *> *)touche

IOS开发---菜鸟学习之路--(二)-数据获取

http://www.cnblogs.com/PleaseInputEnglish/p/3432024.html IOS开发---菜鸟学习之路--(二)-数据获取,布布扣,bubuko.com

iOS开发-定制多样式二维码

iOS开发-定制多样式二维码 二维码/条形码是按照某种特定的几何图形按一定规律在平台(一维/二维方向上)分布的黑白相间的图形纪录符号信息.使用若干个与二进制对应的几何形体来表示文字数值信息. 最常见的二维码功能包括信息获取.网站跳转.电商交易.手机支付等等,其拥有密度小.信息容量大.容错能力强.成本低.制作难度低等优点.在移动开发中,二维码的地位也越来越重要,掌握二维码的基本操作是重要的本领之一. 在iOS7之后,苹果自身集成了二维码的生成和读取功能.生成二维码包括以下步骤 1.导入CoreIm

从零开始学ios开发(十二):Table Views(中)UITableViewCell定制

我们继续学习Table View的内容,这次主要是针对UITableViewCell,在前一篇的例子中我们已经使用过UITableViewCell,一个默认的UITableViewCell包含imageView.textLabel.detailTextLabel等属性,但是很多时候这些默认的属性并不能满足需要,其实更多的时候我们想自己制定UITableViewCell的内容,这篇学习的就是制定自己的UITableViewCell. UITableViewCell继承自UIView,因此它可以加载

从零开始学ios开发(十二):Table Views(上)

这次学习的控件非常重要且非常强大,是ios应用中使用率非常高的一个控件,可以说几乎每个app都会使用到它,它就是功能异常强大的Table Views.可以打开你的iphone中的phone.Messages.Contacts.Mail.Settings等等等等,这些都用到了Table Views. 在Table Views中,Table是用来显示一系列数据的,每条数据占用且只占用一行(一个table cell),在ios中没有规定table到底可以容纳多少行数据,也就是说,只要内存足够多,tab

iOS开发UINavigation系列二——UINavigationItem

iOS开发UINavigation系列二--UINavigationItem 一.引言 UINavigationItem是导航栏上用于管理导航项的类,在上一篇博客中,我们知道导航栏是通过push与pop的堆栈操作来对item进行管理的,同样,每一个Item自身也有许多属性可供我们进行自定制.这篇博客,主要讨论UINavigationItem的使用方法. UINavigationBar:http://my.oschina.net/u/2340880/blog/527706. 二.来说说UINavi

IOS开发隐藏键盘的4种方法

IOS开发隐藏键盘的4种方法 开发的app中常会用到键盘输入完成后隐藏键盘的情况.在此总结了4种方法来实现我们的目的. 方法一--delegate方式 第一种隐藏键盘的方法,称为delegate三步法: 1. 遵循(委托/代理); 2. 调用; 3. 关联头文件(ViewController.h)中遵循代理 . 代码如下: #import <UIKit/UIKit.h> @interface ViewController : UIViewController<UITextFieldDel

Cmdlet开发与学习(二)

继续上一篇的内容,既然已经定义好了命令,那么我们还可以参数绑定,定义自己的参数对象. 3.定义属性,即cmdlet参数对象       参数绑定时基于cmdlet元数据进行的,cmdlet元数据是在命令发现过程中得到的.首先,基于cmdlet的类型(包括可执行程序,脚本文件,内置函数等),PowerShell实例化一个cmdlet.然后,通过查询cmdlet元数据中的参数信息,判断参数列表是否合法,或者,该命令参数是否需要参数值. 参数绑定的实现相当简单,只需要对属性加上Parameter标示符

关于iOS开发的学习

关于iOS开发的学习,打个比方就像把汽车分解:    最底层的原料有塑料,钢铁    再用这些底层的东西造出来发动机,座椅    最后再加上写螺丝,胶水等,把汽车就拼起来了iOS基本都是英文的资料,也由于封闭,文档写的相当好.在遇到新框架的时候:    弄明白框架的功能    去文档里搜搜 框架的 Programming Guide 很有用    要弄明白框架类的继承结构写iOS的程序不一定都是用OBJC,很多框架是用C写的.学习iOS开发基础可以按照下面两个方面学:    基础 (原料 钢铁 

iOS开发之删除过期Provisioning Profiles方法

2014-05-15 22:02 by Jeff Li 前言 系列文章:[传送门] 马上快要期末考试了,为了学点什么.就准备这系列的博客,记录复习的成果. 正文-计数  概率 概率论研究随机事件.它源于赌徒的研究.即使是今天,概率论也常用于赌博.随机事件的结果是否只凭运气呢?高明的赌徒发现了赌博中的规律.尽管我无法预知事件的具体结果,但我可以了解每种结果出现的可能性.这是概率论的核心. "概率"到底是什么?这在数学上还有争议."频率派"认为概率是重复尝试多次,某种结