关于mPaaS中服务的一些疑问

“没有 UI 界面且通用的功能,可以设置为服务(如登录服务)”。

mPaaS定义如下:

/**
 * \code DTService 代表一个服务。
 */
@protocol DTService <NSObject>
@required
/**
 * 启动一个服务。框架在完成初始化操作后,会调用该方法。
 * 如果一个服务要启动一个应用,必须在该法被调用之后,才能启动其它的应用。
 */
- (void)start;
@optional
/**
 *  服务已经完成创建。
 */
- (void)didCreate;
/**
 *  服务即将释放。
 */
- (void)willDestroy;
@end

服务的实现,完全按照文档进行开发,有需求的可以直接查看文档

这里说一下mPaaS服务使用中的问题,

配置在MobileRuntime.plist中的服务

didCreate()没有执行回调。

无论是否是lazyLoading, didCreate完全没有回调,(start回调倒是执行了)

willDestroy()回调

上下这个服务,就类似于单例,在整个app的声明周期内都可以调用,那destroy方法何时调用?

未配置在MobileRuntime.plist中的服务

DTContext中提供了这样两个方法:

/**
 * 注册一个服务。
 *
 * @param name 服务名
 */
- (BOOL)registerService:(id)service forName:(NSString *)name;

/**
 * 反注册一个已存在的服务。
 *
 * @param name 服务名。
 */
- (void)unregisterServiceForName:(NSString *)name;

因缺乏文档,只能自己测试一下作用,看描述应该可以动态为Context添加服务,移除服务。

实际测试过程不表,只说一下结论,最终代码如下:

    ABCSceneServiceIMP * serviceIMP = [[ABCSceneServiceIMP alloc] init];

    BOOL result = [DTContextGet() registerService:serviceIMP forName:@"ABCSceneServiceIMP"];
    if (result) {
        NSDictionary * dict = ((NSDictionary *)[[DTContextGet() valueForKey:@"_serviceManager"] valueForKey:@"_serviceMap"]);
        id abcsceneservice = dict[@"ABCSceneServiceIMP"];
        [abcsceneservice setValue:ABCSceneServiceIMP.class forKey:@"cls"];
    }
  • ABCSceneServiceIMP 是我们的服务的实现类,我们手动创建了实现类,实现类对象注册为服务。(只能用这个,String\Class 都不行);

  • DTContextGet()实际获取到的十一个DTContext的子类,其中包含了window,navigationController,applicationManager和serviceManager,如下图所示:

<img src="http://kyang.oss-cn-beijing.aliyuncs.com/markdown/20210528image-20210528110551661.png" alt="image-20210528110551661" style="zoom:67%;" />

  • 通过KVC,获取添加到serviceManager中的对象(DFServiceInfo),虽然DFServiceInfo没有公开,单从控制台可以看到它的结构。如下图,cls存储的是实现类,object存储的是实现类的对象。下图是执行registerService之后)

    registerService之后

    获取到DFServiceInfo之后,我们需要对cls进行赋值,使其等于ABCSceneServiceIMP,(别问为什么,不设这个不行,DTContextGet()找不到对应服务)。

  • 使用时,跟之前一样

    id<ABCSceneService> service = [DTContextGet() findServiceByName:@"ABCSceneServiceIMP"];
    if ([service respondsToSelector:@selector(dododo)]) {
          [service performSelector:@selector(dododo)];
     }
    
  • 移除服务

    [DTContextGet() unregisterServiceForName:@"ABCSceneServiceIMP"]; 
    

问题也挺多的,

  • 按照上面的写法,start() 方法并没有执行,这边有两个解决办法,

    • 一个是ABCSceneServiceIMP创建后,手动执行start()方法, (didCreate()也可以放在这执行下)。
        ABCSceneServiceIMP * serviceIMP = [[ABCSceneServiceIMP alloc] init];
        [serviceIMP start];
        [serviceIMP didCreate];
    
    • 第二个是,把DFServiceInfo对象的object属性置为NULL,这样DFContext会重新创建Object,也就会执行start()方法, 不过didCreate()还是没有执行;
  • 手动调用- (**void**)unregisterServiceForName:(NSString *)name;方法,willDestroy()也没有进行回调。

  • 想象中的lazyLoading为NO,那么serviceManage会直接拿DFServiceInfo的object,其实并没有。

  • register方法传入一个对象,mPaaS把它赋值给DFServiceInfo的Object对象,然而,最终生效的却是cls属性,object存在,没有设置cls,也是无法findService成功的。所以重点是cls,那,为啥register方法不直接传入Class?

有没有懂得的,指点下。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容