在项目开发中我们可能会遇到这样子的情况,比如在我们登陆的时候需要把数据发送给服务器进行比对,通常我们的做法是当用户点击按钮后,使用一个加载效果的view遮挡住当前界面,直到服务器返回数据或者超时。如果不进行遮挡,用户可能频繁的点击登录,而你又一直发送数据,这样子显然是不可取的,解决这样子的方法有很多种。
今天我们说一种方式,让按钮响应时间由自己控制。
要想达到这种效果你可能需要去了解一下什么是 Runtime
OK,如果你不是很了解也没有关系,对于这个功能用到的也不多。其中包括:
objc_getAssociatedObject(<#id object#>, <#const void *key#>)
参数一:一般都是self,调用者
参数二:你的key(key - value)
有get方法那么肯定会有set
objc_setAssociatedObject(<#id object#>, <#const void *key#>, <#id value#>, <#objc_AssociationPolicy policy#>)
参数一:self
参数二:key
参数三:value,这里要注意包装成为id类型对于int、float等基本类型
参数四:MRC基础的东西
/*
OBJC_ASSOCIATION_ASSIGN = 0,
OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1,
OBJC_ASSOCIATION_COPY_NONATOMIC = 3,
OBJC_ASSOCIATION_RETAIN = 01401,
OBJC_ASSOCIATION_COPY = 01403
*/
为什么会用到上面的方法,因为在给一个Category动态添加属性。
完成了动态添加属性之后,我们需要利用Runtime的性质去对系统的方法动动手脚了。
这里我们还需要了解三个方法 分别是:
通过这个方法可以得到系统对象方法的编号,类型是Method
class_getInstanceMethod(<#__unsafe_unretained Class cls#>, <#SEL name#>)
通过这个方法可以获取到系统类方法的编号,类型是Method
class_getClassMethod(<#__unsafe_unretained Class cls#>, <#SEL name#>)
这里我们要获取到的是对象的方法,所以用第一个,在获取到系统的方法之后呢,我们需要定义一个自己的方法,用来和系统的方法进行互换
互换两个方法的编号
method_exchangeImplementations(<#Method m1#>, <#Method m2#>)
如图所示:互换前和互换后
交换前
交换后
所以我就达到了交换的目的。
了解了上面的几个方法之后,我们就可以开始 写代码了
首先如果想要使用Runtime第一步你得先导入:
import <objc/message.h>
第二步利用Runtime动态的去添加属性
这里要重写get set方法。
通过kvc赋值,所以我们先要有key,用来存/取数据
static char * const PQ_ACCEPTTIMEKEY = "pq_acceptTime";
static char * const PQ_DELAYINTERVALKEY = "pq_delayButtonInterVal";
// getter method
- (NSTimeInterval)pq_delayButtonInterVal{
return [objc_getAssociatedObject(self, PQ_DELAYINTERVALKEY) doubleValue];
}
- (NSTimeInterval)pq_acceptTime{
return [objc_getAssociatedObject(self, PQ_ACCEPTTIMEKEY) doubleValue];
}
// setter method
- (void)setPq_delayButtonInterVal:(NSTimeInterval)pq_delayButtonInterVal{
objc_setAssociatedObject(self, PQ_DELAYINTERVALKEY, @(pq_delayButtonInterVal), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (void)setPq_acceptTime:(NSTimeInterval)pq_acceptTime{
objc_setAssociatedObject(self, PQ_ACCEPTTIMEKEY, @(pq_acceptTime), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
完成动态添加属性之后,我们要把系统的方法和我们的方法进行互换,以便于添加功能。
+ (void)load{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Method systemMethod = class_getInstanceMethod([self class], @selector(sendAction:to:forEvent:));
Method myselfMethod = class_getInstanceMethod([self class], @selector(pq_sendAction:to:forEvent:));
BOOL isAdd = class_addMethod([self class], @selector(sendAction:to:forEvent:), method_getImplementation(myselfMethod), method_getTypeEncoding(myselfMethod));
if (!isAdd) {
method_exchangeImplementations(systemMethod, myselfMethod);
}
});
}
然后我们就要写自己的方法了,这里有一个小技巧,先把系统的方法名打上,比如对于这个方法可以这样做:
先输入 - sendAction 一般打到这里系统就会提示自动补全
得到系统的方法名之后在前面添加我们的前缀,表示是我们自己的方法。比如
这里我就添加了一个前缀pq,再用下划线连接。
- (void)pq_sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event
接下来就是处理代码的实现
这里用到的两个变量一个需要对外公开(.h文件中),一个写在(.m文件中的)
@interface UIControl ()
@property (nonatomic,assign) NSTimeInterval pq_acceptTime;
@end
- (void)pq_sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event{
if (NSDate.date.timeIntervalSince1970 - self.pq_acceptTime < self.pq_delayButtonInterVal) {
NSLog(@"现在点我我也不鸟你");
return;
}
if (self.pq_delayButtonInterVal > 0) {
self.pq_acceptTime = NSDate.date.timeIntervalSince1970;
}
[self pq_sendAction:action to:target forEvent:event];
}
至此,功能就完成啦,小伙伴们可以马上使用了
自己创建一个button,然后设置如下代码,如果你发现找不到你自己定义的属性,那么你一般是没有导入头文件在ViewController中
//设置这个延时时间是5秒钟
self.clickMeBtn.pq_delayButtonInterVal = 5;