[LZPImageimageNamed:@"111"];
这里的代码中,如果后面的名字是错的,不会有警告或者报错,即使是运行,也什么都不会做;
//所以在这里给系统的UIImage类的imageNamed方法添加一个提醒的功能,当图片加载失败的时候有一个提醒;
1.首先,给UIImage定义一个子类,重写imageNamed方法,如下:
//如果加载失败,就会有一个提醒,但是需要在使用UIimage的类中导入自定义UIImage的子类的头文件,使用子类中的imageNamed方法,如果不想导入头件怎么办?
+(UIImage*)imageNamed:(NSString*)name:
UIImage* image = [superimageNamed:name];
if(image ==nil) {
NSLog(@"加载失败");
returnnil;
}
returnimage;
}
2.如果给系统UIImage类写分类,重写其imageNamed方法,则其系统的原来imageNamed中的功能就会被分类中的imageNamed方法覆盖,不可行;
所以考虑自定义方法,如下:
//但是这样也还是会导入头文件;
没有达到预期的效果;
#import"UIImage+check.h"
@implementationUIImage (check)
+(instancetype)LZP_imageNamed:(NSString*)name{
UIImage* image = [selfimageNamed:name];
if(image ==nil) {
NSLog(@"图片加载失败");
returnnil;
}
returnimage;
}
@end
3.为了达到我们的目的,给系统的UIImage类的imageNamed方法添加功能,使用runtime交换方法:
#import"UIImage+check.h"
#import
@implementationUIImage (check)
//当程序运行的时候,会将代码加载到内存中,这时候就会调用load方法;而且在程序的整个生命周期内只会调用一次load方法;
//这里先描述一下 [p run];
//一个实例对象调用其一个对象方法的过程:
会根据run方法的编码找到对象p的isa属性指向到的p对象的类对象的对象方法列表中,找到对应的方法的地址,然后根据这个地址在代码区找到其对应的类方法实现;
//再进一步说:[Person eat];;
//会根据eat方法的编码找到Person的isa属性指向的元类的类方法列表中,找到对应的方法地址,然后根据这个地址到代码区找到其对应的类方法实现;
//注意:对象方法保存在类对象中;类方法保存在元类中;
在load方法中修改交换两个方法,其实是交换了两个方法的地址,即在调用LZPimageNamed的时候,执行的是imageNamed,调用imageNamed的时候,执行的是LZPimageNamed方法;
注意:这里实在元类中的类方法列表中将其地址交换;
+(void)load{
MethodimageNamed =class_getClassMethod(self,@selector(imageNamed:));
MethodLZPimageNamed =class_getClassMethod(self,@selector(LZP_imageNamed:));
method_exchangeImplementations(imageNamed, LZPimageNamed);
}
+(instancetype)LZP_imageNamed:(NSString*)name{
UIImage* image = [selfLZP_imageNamed:name];
//如果在其他类中调用UIImage的imageNamed方法,就会来到这个方法,这个方法中也会调用LZP_imageNamed方法,但是imageNamed方法会执行;
然后进行下面的判断,刚好得到想要的结果,perfect!
if(image ==nil) {
NSLog(@"图片加载失败");
returnnil;
}
returnimage;
}
@end
//这里LZP_imageNamed方法中不能调用imageNamed方法,因为imageNamed实质已经是LZP_imageNamed方法了,会出项死循环,调用其自己的时候,实质是调用imageNamed方法,imageNamed中没有代用LZP_imageNamed,这样就不会楚翔死循环;