OpenHarmony轻量系统服务管理|鸿蒙业务模型重要概念
前言
在针对鸿蒙分布式任务调度的源代码分析中,发现它业务逻辑的实现围绕着三大概念展开,分别是服务(Service)、功能(Feature)和功能接口API(Iunknown)。所以理解并掌握这三个概念对于我们深入学习鸿蒙底层代码的业务逻辑有极大的帮助。下面将结合前期分析鸿蒙代码的经验,通过图文并茂的方式为读者讲解它们的基类和实例对象以及相应的作用。
结构分析
Service
服务(Service)是鸿蒙操作系统中业务逻辑的核心,是一组功能或操作的集合。下面的结构体定义了服务的相关函数指针。
1 struct Service { 2 const char *(*GetName)(Service *service); //获取服务名称 3 BOOL (*Initialize)(Service *service, Identity identity); //初始化服务 4 BOOL (*MessageHandle)(Service *service, Request *request); //处理服务的消息 5 TaskConfig (*GetTaskConfig)(Service *service); //获取服务的任务配置 6 };
Feature
一个服务可以拥有多个功能(Feature),功能是业务的执行单元。由各式各样的功能组合成一个完整的服务。下面的结构体定义了功能接口的相关函数指针。
struct Feature { const char *(*GetName)(Feature *feature); //获取功能名称 void (*OnInitialize)(Feature *feature, Service *parent, Identity identity); //初始化功能 void (*OnStop)(Feature *feature, Identity identity); //停止功能 BOOL (*OnMessage)(Feature *feature, Request *request); //处理功能消息 };
IUnknown
通过上述服务和功能的概念抽象出鸿蒙完整的业务逻辑模型。一个服务包含零个到多个功能。在鸿蒙代码中,针对服务和功能的调用还提供了统一的对外接口(Iunknown)。
1 struct IUnknown { 2 int (*QueryInterface)(IUnknown *iUnknown, int version, void **target);//查询指定版本的IUnknown接口的子类对象 3 int (*AddRef)(IUnknown *iUnknown); //添加引用计数 4 int (*Release)(IUnknown *iUnknown); //释放引用计数 5 };
实例对象
在后续服务和功能的注册、启动等过程中都是使用ServiceImpl和FeatureImpl。它们的定义如下:
功能绑定接口后的实例对象
一个功能对应一个接口,作为一个功能的实例对象
1 struct FeatureImpl { 2 Feature *feature; //功能对象 3 IUnknown *iUnknown; //对外的接口 4 };
服务绑定接口后的实例对象
一个服务至少向外提供一个API接口(当服务下面没有注册功能时,向外暴露defaultApi)。也可以提供多个API接口(当服务下面注册多个功能时,每一个功能都有一个API接口)。一个服务对应一个任务池。
1 struct ServiceImpl { 2 Service *service; //服务对象 3 IUnknown *defaultApi; //默认的对外接口 4 TaskPool *taskPool; //绑定的任务池 5 Vector features; //已注册的FeatureImpl对象 6 int16 serviceId; //服务ID,位于g_samgrImpl的vector中的下标 7 uint8 inited; //服务所处的状态 8 Operations ops; //操作信息,记录一些运维信息 9 };
服务和功能实例对象的图示
ServiceImpl:Vector的data指向一个FeatureImpl类型的指针数组,每个指针指向一个FeatureImpl对象。TaskPool指向一个TaskPool对象。图中只画出了部分重要字段。
注:任务池的绑定分为三种:
1.SHARED_TASK,根据service的优先级共享任务,此时绑定的是SamgrLiteImpl中维护的共享任务池。
2.SPECIFIED_TASK,为service绑定指定的任务池,遍历所有的ServiceImpl对象,查找相同任务配置的任务池,若找到则绑定,若未找到则创建新的任务池。
3.SINGLE_TASK,根据任务配置,创建新的任务池并绑定

注册过程
这里先简单的介绍一下服务和功能的注册,在后续文章中再详细介绍函数的调用过程。提到服务的注册我们还需要了解一个重要的数据结构SamgrLiteImpl,在代码中有一个由它定义的全局变量g_samgrImpl,这个全局变量维护了一系列的服务,也称为系统功能管理器(Samgr)。结构体定义如下:
1 struct SamgrLiteImpl { 2 SamgrLite vtbl; //SamgrLite类管理一些函数,用于注册和发现服务和功能 3 MutexId mutex; //互斥锁 4 BootStatus status; //系统功能管理器(samgr)的状态 5 Vector services; //已注册的ServiceImpl对象 6 TaskPool *sharedPool[MAX_POOL_NUM]; //服务共享的任务池 7 };
注册的过程分为接口的注册、功能的注册和服务的注册。流程如下:
1.Feature绑定IUnknown封装为FeatureImpl对象。(接口的注册)
2.将FeatureImpl对象添加到ServiceImpl的vector集合中。(功能的注册)
3.将ServiceImpl对象添加到SamgrLiteImpl的vector集合中。(服务的注册)
Samgr实例图示
SamgrLiteImpl:Vector的data指向一个ServiceImpl类型的指针数组,每个指针指向一个ServiceImpl对象。TaskPool指向一个TaskPool类型的指针数组,每个指针指向一个TaskPool,这一点与ServiceImpl中是不同的。图中只画出了部分重要字段。
注:当ServiceImpl中task的taskflags即任务类型为SHARED_TASK时,根据任务的优先级共享SamgrLiteImpl中TaskPool维护的指定任务池。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· DeepSeek 开源周回顾「GitHub 热点速览」
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了