任务管理器(TaskManager)
任务管理
- 为什么需要任务管理
- 实现思路
- 部分逻辑代码
- Task
- 延伸
为什么需要任务管理
在设计router的时候我们曾希望通过类映射的方式将一些特殊的服务例如api请求,而在之后我发现因为task任务的生命周期问题,在model回调之前整个task都已经被释放了,所以我需要一个manager来负责管理这些任务,包括了任务完成之后的释放。
实现思路
我新建了一个TaskManager的类作为管理器,考虑到我们现在的业务大部分页面的点击事件都是在这里,在这里我先添加了一个ManagerDic的可变字典,大致的结构是
{key : [TASK1,TASK2]} key对应的就是约定的类名
之后我用了KVO来进行管理,在HHTask这个类里添加了一个属性completeTask 来做判断 当任务完成 将completeTask置成yes,然后 移除在TaskManager里面的任务,移除观察者。
部分逻辑代码
首先判断是否已经有对应的数组存在,task任务
完成以后remove掉自己的任务就好,如果还没有任务就生成对应任务数组。
+ (void)addTask:(id)task key:(NSString *)key{
HHTaskManager *manager = [HHTaskManager shared];
@synchronized (self) {
if (manager.managerDic[key]) {
NSMutableArray *taskArray = manager.managerDic[key];
[taskArray addObject:task];
}
else {
NSMutableArray *taskArray = [NSMutableArray array];
[taskArray addObject:task];
[manager.managerDic setObject:taskArray forKey:key];
}
[task addObserver:manager forKeyPath:@"completeTask" options:0 context:nil];
NSMethodSignature *sig= [[task class] instanceMethodSignatureForSelector:@selector(excuteTask)];
NSInvocation *invocation=[NSInvocation invocationWithMethodSignature:sig];
invocation.target = task;
invocation.selector = @selector(excuteTask);
[invocation invoke];
}
}
上面这一部分任务的执行 其实可拓展性会比较强 如果是与UI有关系的任务可以放在主线程进行,这里可以在初始化task的时候给他一个status的状态来判断。我们现在的做法是在我们的router解析完以后有一个规则依靠规则来判断。
这里利用了KVO来检测任务完成,完成以后移除任务和观察者
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
HHTaskManager *manager = [HHTaskManager shared];
if (manager.managerDic[NSStringFromClass([object class])]) {
NSMutableArray *taskArray = manager.managerDic[NSStringFromClass([object class])];
[object removeObserver:manager forKeyPath:@"completeTask"];
[taskArray removeObject:object];
}
}
Task
typedef void (^TaskResult)(BOOL isSucceed, NSDictionary *userInfo);
@interface Task : NSObject
@property (nonatomic, assign) BOOL isNeedLogin;
@property (nonatomic, copy) TaskResult callback;
@property (nonatomic, strong)NSString *completeTask;
- (void)initWithparam:(NSDictionary *)param;
- (void)excuteTask;
//子类实现方法
- (void)excuteDetailTask;
对应的task的设计,起初想设立一定的规则模块来完善登录这个动作,后来还是决定增加一个属性来判断是否需要登录,考虑了登录以后有回调的需求。 预留了一个callback给一些任务完成以后需要有特殊回调的需求。completeTask是表示任务是否完成。因为考虑到 有api请求的需要等到api返回以后才是真正的完成了任务。这是一个基类。然后可以看一下我的其他任务子类代码。
@interface HHWapTask : Task
@property (nonatomic, strong)NSString *url;
@end
可以看到其实不同的task任务还会分解出不同的参数,在- (void)initWithparam:(NSDictionary *)param;这个方法里将你所需要的参数进行了分解。之后再执行任务即可。
延伸
刚开始我打算做的是一个任务的存储和管理,之后考虑的是一些重复的任务之间因为key 是相同的通过key我可以判断是否直接取消这个任务,在task里面可以加一个任务是否需要启动的开关,如果发现任务重复就设置为不开启,然后直接移除任务。