OpenHarmony-6.IPC/RPC组件
- IPC/RPC组件机制
1.基本概念
IPC:设备内的进程间通信(Inter-Process Communication)。
RPC:设备间的进程间通信(Remote Procedure Call)。
IPC/RPC用于实现跨进程通信,不同的是前者使用Binder驱动,用于设备内的跨进程通信,后者使用软总线驱动,用于跨设备跨进程通信。需要跨进程通信的原因是因为每个进程都有自己独立的资源和内存空间,其他进程不能随意访问不同进程的内存和资源,IPC/RPC便是为了突破这一点。
Client-Server 软件模型外加 Interface 接口共同组成了 OpenHarmony 诸多子系统的架构,使其系统的稳定性、扩展性以及进程通信能力得到加强。子系统中均可见到 Interface 接口目录、Framework 客户端目录以及 Service 服务端目录,工作重点一般集中在 Service 服务端,如下图所示:
1.1.实现原理
IPC和RPC通常采用客户端-服务端(Client-Server)模型,在使用时,请求Client端进程可获取Server端所在进程的代理(Proxy),并通过此代理读写数据来实现进程间的数据通信,更具体的讲,首先客户端会建立一个服务端的代理对象,这个代理对象具备和服务端一样的功能,若想访问服务端中的某一个方法,只需访问代理对象中对应的方法即可,代理对象会将请求发送给服务端;然后服务端处理接受到的请求,处理完之后通过驱动返回处理结果给代理对象;最后代理对象将请求结果进一步返回给客户端。
如下图所示: 通常,Stub会先注册系统能力(System Ability)到系统能力管理者(System Ability Manager,缩写SAMgr)中,SAMgr负责管理这些SA并向Client提供相关的接口。Client要和某个具体的SA通信,必须先从SAMgr中获取该SA的代理Proxy对象,然后使用代理Proxy对象和SA通信。在整个通信过程中,如果使用的是IPC通信,则依赖的是Binder驱动,使用的是RPC通信,则依赖的是软总线驱动。
说明: 以下为IPC与RPC的典型使用场景:
- IPC典型使用场景在后台服务,应用的后台服务通过IPC机制提供跨进程的服务调用能力。
- RPC典型使用场景在多端协同,多端协同通过RPC机制提供远端接口调用与数据传递能力。
IPC通信机制架构图:
目录结构:
/foundation/communication/ipc
├── interfaces # 对外接口存放目录
│ └── innerkits # 对内部子系统暴露的头文件存放目录
│ ├── ipc_core # ipc 接口存放目录
│ └── libdbinder # dbinder 接口存放目录
├── ipc # ipc 框架代码
│ ├── native # ipc native 实现存放目录
│ ├── src # ipc native 源代码存放目录
│ └── test # ipc native 单元测试用例存放目录
│ └── test # ipc native 模块测试用例存放目录
├── service # dbinder 实现存放目录
│ └── dbinder # dbinder 源代码存放目录
1.2.Binder机制
Binder机制通常采用客户端-服务器(Client-Server)模型,服务请求方(Client)可获取服务提供方(Server)的代理 (Proxy),并通过此代理读写数据来实现进程间的数据通信。通常,系统能力(SystemAbility)Server侧会先注册到系统能力管理者(System Ability Manager,缩写SAMgr)中,SAMgr负责管理这些SA并向Client提供相关的接口(添加,查询,获取,删除等)。Client要和某个具体的SA通信,必须先从SAMgr中获取该SA的代理,然后使用代理和SA通信。
注:SAMgr本身也是IPC的Server端,Client通过SAMgr的代理,调用SAMgr的接口。
Binder机制架构图:
2.系统服务管理子系统
系统服务管理子系统由两部分构成:
- 系统服务框架组件(safwk):定义了SystemAbility的实现方法,并提供启动、发布等接口实现。
- 系统服务管理组件(samgr): 提供系统服务注册、查询等功能。
系统服务管理架构图如下:
代码目录:
/foundation/systemabilitymgr
│── safwk # 组件目录
│ ├── bundle.json # 组件描述及编译脚本
│ ├── etc # 配置文件
│ ├── interfaces # 对外接口目录
│ ├── services # 框架实现
│ ├── test # 测试用例
├── samgr
│ ├── bundle.json # 部件描述及编译文件
│ ├── frameworks # 框架实现存在目录
│ ├── interfaces # 接口目录
│ ├── services # 组件服务端目录
│ ├── test # 测试代码存放目录
│ ├── utils # 工具类目录
2.1.系统服务框架组件
SystemAbility实现一般采用XXX.cfg + saId.xml + libXXX.z.so的方式由init进程解析对应的XXX.cfg文件拉起SystemAbility所依赖的进程。(注:多个系统服务可能跑在同一个进程里。比如AbilityManagerService、BatteryService、WindowManagerService都在foundation进程,MMIService在独立的进程multimodalinput中。)
SystemAbility类图如下:
备注:
SystemAbility子类需要重写OnStart()和OnStop()方法,并且在OnStart()方法中调用Publish(sptr systemAbility)方法把系统服务发布出去。
2.2.系统服务实现步骤
以AbilityManagerService为例说明SystemAbility的实现。
2.2.1.定义IPC对外接口IXXX
定义该服务对外提供的能力集合函数,统一继承IPC接口类IRemoteBroker;同时声明该IPC对外接口唯一标识符DECLARE_INTERFACE_DESCRIPTOR(XXX);该标识符用于IPC通信的校验等目的。
foundation\ability\ability_runtime\interfaces\inner_api\ability_manager\include\ability_manager_interface.h:69 class IAbilityManager : public OHOS::IRemoteBroker {70 public:71 DECLARE_INTERFACE_DESCRIPTOR(u"ohos.aafwk.AbilityManager")7273 /**74 * StartAbility with want, send want to ability manager service.75 *76 * @param want, the want of the ability to start.77 * @param userId, Designation User ID.78 * @param requestCode, Ability request code.79 * @return Returns ERR_OK on success, others on failure.80 */81 virtual int StartAbility(82 const Want &want,83 int32_t userId = DEFAULT_INVAL_VALUE,84 int requestCode = DEFAULT_INVAL_VALUE) = 0;8586 ...87 }
- 定义客户端代码XXXProxy
foundation\ability\ability_runtime\services\abilitymgr\include\ability_manager_proxy.h:30 class AbilityManagerProxy : public IRemoteProxy<IAbilityManager> {31 public:32 explicit AbilityManagerProxy(const sptr<IRemoteObject> &impl) : IRemoteProxy<IAbilityManager>(impl)33 {}3435 virtual ~AbilityManagerProxy()36 {}3746 virtual int StartAbility(47 const Want &want,48 int32_t userId = DEFAULT_INVAL_VALUE,49 int requestCode = DEFAULT_INVAL_VALUE) override;5051 ...52 }
foundation\ability\ability_runtime\services\abilitymgr\src\ability_manager_proxy.cpp42 int AbilityManagerProxy::StartAbility(const Want &want, int32_t userId, int requestCode)43 {44 int error;45 MessageParcel data;46 MessageParcel reply;47 MessageOption option;4849 if (!WriteInterfaceToken(data)) {50 return INNER_ERR;51 }52 if (!data.WriteParcelable(&want)) {53 HILOG_ERROR("want write failed.");54 return INNER_ERR;55 }5657 if (!data.WriteInt32(userId)) {58 HILOG_ERROR("userId write failed.");59 return INNER_ERR;60 }6162 if (!data.WriteInt32(requestCode)) {63 HILOG_ERROR("requestCode write failed.");64 return INNER_ERR;65 }6667 error = SendRequest(AbilityManagerInterfaceCode::START_ABILITY, data, reply, option);72 return reply.ReadInt32();73 }
AbilityManagerProxy::StartAbility()实现代码中会调用Remote()->SendRequest(IAbilityManager::START_ABILITY, data, reply, option);把消息码和数据发送给服务端。
-定义服务端代码XXXStub
foundation\ability\ability_runtime\services\abilitymgr\include\ability_manager_stub.h34 class AbilityManagerStub : public IRemoteStub<IAbilityManager> {35 public:36 AbilityManagerStub();37 ~AbilityManagerStub();38 virtual int OnRemoteRequest(39 uint32_t code, MessageParcel &data, MessageParcel &reply, MessageOption &option) override;4048 virtual int DoAbilityForeground(const sptr<IRemoteObject> &token, uint32_t flag) override;4950 ...51 }
foundation\ability\ability_runtime\services\abilitymgr\src\ability_manager_stub.cpp:332 int AbilityManagerStub::OnRemoteRequest(uint32_t code, MessageParcel &data, MessageParcel &reply, MessageOption &option)333 {334 std::u16string abilityDescriptor = AbilityManagerStub::GetDescriptor();335 std::u16string remoteDescriptor = data.ReadInterfaceToken();336 if (abilityDescriptor != remoteDescriptor && extensionDescriptor != remoteDescriptor) {337 HILOG_ERROR("local descriptor is not equal to remote");338 return ERR_INVALID_STATE;339 }340341 auto itFunc = requestFuncMap_.find(code);342 if (itFunc != requestFuncMap_.end()) {343 auto requestFunc = itFunc->second;344 if (requestFunc != nullptr) {345 return (this->*requestFunc)(data, reply);346 }347 }348 HILOG_WARN("default case, need check.");349 return IPCObjectStub::OnRemoteRequest(code, data, reply, option);350 }
requestFuncMap_[START_ABILITY] = &AbilityManagerStub::StartAbilityInner。消息码START_ABILITY对应的函数为AbilityManagerStub::StartAbilityInner。
600 int AbilityManagerStub::StartAbilityInner(MessageParcel &data, MessageParcel &reply)601 {602 Want *want = data.ReadParcelable<Want>();603 if (want == nullptr) {604 HILOG_ERROR("want is nullptr");605 return ERR_INVALID_VALUE;606 }607 int32_t userId = data.ReadInt32();608 int requestCode = data.ReadInt32();609 int32_t result = StartAbility(*want, userId, requestCode);610 reply.WriteInt32(result);611 delete want;612 return NO_ERROR;613 }
StartAbility()的实现在AbilityManagerStub的实现类AbilityManagerService中。
- SystemAbility的实现类
foundation\ability\ability_runtime\services\abilitymgr\include\ability_manager_service.h76 class AbilityManagerService : public SystemAbility,77 public AbilityManagerStub,78 public AppStateCallback,79 public std::enable_shared_from_this<AbilityManagerService> {80 DECLARE_DELAYED_SINGLETON(AbilityManagerService)81 DECLEAR_SYSTEM_ABILITY(AbilityManagerService)82 public:83 void OnStart() override;84 void OnStop() override;8586 virtual void OnAddSystemAbility(int32_t systemAbilityId, const std::string& deviceId) override;8788 virtual void OnRemoveSystemAbility(int32_t systemAbilityId, const std::string& deviceId) override;8990 ServiceRunningState QueryServiceState() const;91100 virtual int StartAbility(101 const Want &want, int32_t userId = DEFAULT_INVAL_VALUE, int requestCode = DEFAULT_INVAL_VALUE) override;102 ...103 }
AbilityManagerService同时继承了SystemAbility和AbilityManagerStub。在重写的SystemAbility的接口函数OnStart()中,调用Publish(instance_)把自己发布出去。
foundation/ability/ability_runtime/services/abilitymgr/src/ability_manager_service.cpp268 void AbilityManagerService::OnStart()269 {270 if (state_ == ServiceRunningState::STATE_RUNNING) {271 HILOG_INFO("AMS has already started.");272 return;273 }274 HILOG_INFO("AMS starting.");275 if (!Init()) {276 HILOG_ERROR("Failed to init AMS.");277 return;278 }279 state_ = ServiceRunningState::STATE_RUNNING;280 /* Publish service maybe failed, so we need call this function at the last,281 * so it can't affect the TDD test program */282 instance_ = DelayedSingleton<AbilityManagerService>::GetInstance().get();287 bool ret = Publish(instance_); 292293 SetParameter(BOOTEVENT_A // In namespace OHOS true");294 WatchParameter(BOOTEVENT namespace AAFwk {} ), AAFwk::ApplicationUtil::AppFwkBootEventCallback, nullptr);295 AddSystemAbilityListener(BACKGROUND_TASK_MANAGER_SERVICE_ID);296 AddSystemAbilityListener(DISTRIBUTED_SCHED_SA_ID);297 AddSystemAbilityListener(BUNDLE_MGR_SERVICE_SYS_ABILITY_ID);298 HILOG_INFO("AMS start success.");299 }
注:在实现SystemAbility的时候,必须调用宏REGISTER_SYSTEM_ABILITY_BY_ID或者SystemAbility::MakeAndRegisterAbility()把SystemAbility注册到LocalAbilityManager中。可参考如下代码:
const bool REGISTER_RESULT = SystemAbility::MakeAndRegisterAbility(DelayedSingleton::GetInstance().get());
- SystemAbility配置
以c++实现的SA必须配置相关SystemAbility的profile配置文件才会完成SA的自动加载注册逻辑,否则没有编写配置文件的SystemAbility不会完成自动加载注册。配置方法如下:
在子系统的根目录新建一个以sa_profile为名的文件夹,然后在此文件夹中新建两个文件:一个以saId为前缀的xml文件,另外一个为BUILD.gn文件。比如AbilityManagerService,saId为ABILITY_MGR_SERVICE_ID(即180),对应的配置文件为180.xml。内容如下:
foundation/ability/ability_runtime/services/sa_profile/BUILD.gn:
14 import("//build/ohos/sa_profile/sa_profile.gni")1516 ohos_sa_profile("ams_sa_profile") {17 sources = [18 "180.json",19 "182.json",20 "183.json",21 "184.json",22 "501.json",23 ]2425 part_name = "ability_runtime"26 }
描述说明:
- part_name为相应部件名称;
- sources表示当前子系统需要配置的SystemAbility列表,可支持配置多个SystemAbility。
foundation/ability/ability_runtime/services/sa_profile/180.json:1 {2 "process": "foundation",3 "systemability": [4 {5 "name": 180,6 "libpath": "libabilityms.z.so",7 "run-on-create": true,8 "distributed": false,9 "dump_level": 110 }11 ]12 }
描述说明:
- 进程名字:该SystemAbility要运行的进程空间,此字段是必填选项。即AbilityManagerService跑在foundation进程中。
- 一个SystemAbility配置文件只能配置一个SystemAbility节点,配置多个会导致编译失败。
- SystemAbility的name为对应的saId必须与代码中注册的saId保持一致,必配项。
- libpath为SystemAbility的加载路径,必配项。
- run-on-create:true表示进程启动后即向samgr组件注册该SystemAbility;false表示按需启动,即在其他模块访问到该SystemAbility时启动,必配项。
- distributed:true表示该SystemAbility为分布式SystemAbility,支持跨设备访问;false表示只有本地跨进程访问。
- bootphase:可不设置;可以设置的值有三种:BootStartPhase、CoreStartPhase、OtherStartPhase(默认类型),三种优先级依次降低,在同一个进程中,会优先拉起注册配置BootStartPhase的SystemAbility,然后是配置了CoreStartPhase的SystemAbility,最后是OtherStartPhase;当高优先级的SystemAbility全部启动注册完毕才会启动下一级的SystemAbility的注册启动。
- dump-level:表示systemdumper支持的level等级,默认配置1。
以上步骤完成后,全量编译代码后会在out路径下生成一个以进程名为前缀的xml文件(比如foundation.xml),路径为:out\…\system\profile\foundation.xml。该文件整合了所有需要在该进程中运行的SA的saId.xml文件内容。(比如AbilityManagerService,WindowManagerService,PowerManagerService等SA的配置文件都会被集成到foundation.xml中)。
- Cfg配置文件
cfg配置文件为linux提供的native进程拉起策略,开机启动阶段由init进程解析cfg文件把目标进程拉起(动态加载的除外)。foundation进程的配置文件在systemabilitymgr子系统中。
foundation\systemabilitymgr\safwk\etc\profile\foundation.cfg34 "services" : [{35 "name" : "foundation",36 "path" : ["/system/bin/sa_main", "/system/profile/foundation.json"],37 "importance" : -20,38 "uid" : "foundation",39 "permission" : [40 "ohos.permission.INPUT_MONITORING",41 "ohos.permission.PERMISSION_USED_STATS",42 "ohos.permission.DISTRIBUTED_SOFTBUS_CENTER",43 "ohos.permission.DISTRIBUTED_DATASYNC",44 "ohos.permission.MICROPHONE",45 "ohos.permission.WRITE_CALL_LOG",46 "ohos.permission.READ_CONTACTS",47 "ohos.permission.READ_DFX_SYSEVENT",48 "ohos.permission.GRANT_SENSITIVE_PERMISSIONS",49 "ohos.permission.REVOKE_SENSITIVE_PERMISSIONS",50 "ohos.permission.MANAGE_SECURE_SETTINGS",51 "ohos.permission.GET_BUNDLE_INFO_PRIVILEGED",52 "ohos.permission.START_ABILITIES_FROM_BACKGROUND",53 "ohos.permission.ACCESS_SERVICE_DM",54 "ohos.permission.STORAGE_MANAGER",55 "ohos.permission.PROXY_AUTHORIZATION_URI",56 "ohos.permission.ABILITY_BACKGROUND_COMMUNICATION",57 "ohos.permission.USE_USER_IDM",58 "ohos.permission.MANAGE_LOCAL_ACCOUNTS",59 "ohos.permission.LISTEN_BUNDLE_CHANGE",60 "ohos.permission.GET_TELEPHONY_STATE",61 "ohos.permission.SEND_MESSAGES",62 "ohos.permission.CONNECT_CELLULAR_CALL_SERVICE",63 "ohos.permission.SET_TELEPHONY_STATE"
refer to
- https://gitee.com/openharmony/communication_ipc
- https://blog.csdn.net/procedurecode/article/details/130222081
- https://gitee.com/openharmony/docs/blob/39467f023bec8cfca8ec2f97b99039b1dbd141e5/zh-cn/application-dev/ipc/ipc-rpc-overview.md
- https://forums.openharmony.cn/forum.php?mod=viewthread&tid=2980
- https://www.51cto.com/article/701821.html
相关文章:
OpenHarmony-6.IPC/RPC组件
IPC/RPC组件机制 1.基本概念 IPC:设备内的进程间通信(Inter-Process Communication)。 RPC:设备间的进程间通信(Remote Procedure Call)。 IPC/RPC用于实现跨进程通信,不同的是前者使用Binder驱…...
自然语言处理与知识图谱的融合与应用
目录 前言1. 知识图谱与自然语言处理的关系1.1 知识图谱的定义与特点1.2 自然语言处理的核心任务1.3 二者的互补性 2. NLP在知识图谱构建中的应用2.1 信息抽取2.1.1 实体识别2.1.2 关系抽取2.1.3 属性抽取 2.2 知识融合2.3 知识推理 3. NLP与知识图谱融合的实际应用3.1 智能问答…...
Pytorch | 利用VA-I-FGSM针对CIFAR10上的ResNet分类器进行对抗攻击
Pytorch | 利用VA-I-FGSM针对CIFAR10上的ResNet分类器进行对抗攻击 CIFAR数据集VA-I-FGSM介绍相关定义算法流程 VAI-FGSM代码实现VAI-FGSM算法实现攻击效果 代码汇总vaifgsm.pytrain.pyadvtest.py 之前已经针对CIFAR10训练了多种分类器: Pytorch | 从零构建AlexNet对…...
【Java基础面试题028】Java中的hashCode和equals方法,与==操作符有什么区别?
回答重点 hashcode、equals 和 都是Java中用于比较对象的三种方式,但是它们的用途和实现还是有挺大区别的。 hashcode用于散列存储结构中确定对象的存储位置。可用于快速比较两个对象是否不同,因为如果它们的哈希码不同,那么它们肯定不相等…...
[ThinkPHP]5.0.23-Rce 1
[ThinkPHP]5.0.23-Rce 1 根据题目知道这是一个5.0.23的PHP RCE,话不多说直接上扫描器 检测出Payload url地址: ?scaptcha&test-1 Post表单参数: _method__construct&filter[]phpinfo&methodget&server[REQUEST_METHOD]1HackBar构造p…...
ByConity BSP 解锁数据仓库新未来
文章目录 前言BSP 模式简介基于 TPC-DS 的 ELT 活动测试环境登录 ECS数据查询配置 执行 02.sqlsql解释:1. 第一步:创建 wscs 临时表2. 第二步:创建 wswscs 临时表3. 第三步:对比 2001 年和 2002 年的数据子查询 1:提取…...
应对 Google Play 政策违规:开发者账号被终止解除指南
目录 解封指南 申诉文案 谷歌问题 授权书 1、授权书标题及双方信息 2、游戏信息 3、授权内容 4、双方义务与责任 5、费用与支付 5、保密条款 6、争议解决 8、其他条款 9、签字盖章 10、日期 相关推荐 解封指南 由于开发人员的疏忽,移除了读写权限的动态申请,使…...
【ES6复习笔记】Map(14)
概念 Map 是 JavaScript 中的一种数据结构,它允许你存储键值对,并且可以通过键来访问对应的值。在本教程中,我们将学习如何声明、添加、删除、获取和遍历 Map 集合。 ES6 提供了 Map 数据结构。它类似于对象,也是键值对的集合。…...
重温设计模式--8、命令模式
文章目录 命令模式的详细介绍C 代码示例C代码示例2 命令模式的详细介绍 定义与概念 命令模式属于行为型设计模式,它旨在将一个请求封装成一个对象,从而让你可以用不同的请求对客户端进行参数化,将请求的发送者和接收者解耦,并且能…...
人工智能ACA(七)——计算机视觉基础
一、自然语言处理基本介绍 1. 自然语言处理的定义 1-1 自然语言 人类使用的在社会生活中自然形成的语言 1-2 自然语言处理 目标是让计算机能够理解、解析、生成和处理人类的自然语言 包含自然语言理解和自然语言生成两部分组成 2. 自然语言处理的发展趋势 3.自然语言处理…...
RCE常见姿势
文章目录 常见漏洞执行函数:1.系统命令执行函数2.代码执行函数 命令拼接符读取文件命令绕过:空格过滤绕过关键字绕过长度过滤绕过无参数命令执行绕过无字母数字绕过利用%0A截断利用回溯绕过利用create_function()代码注入无回显RCE1.反弹shell2.dnslog外…...
CSS系列(36)-- Containment详解
前端技术探索系列:CSS Containment详解 ⚡ 致读者:探索性能优化的艺术 👋 前端开发者们, 今天我们将深入探讨 CSS Containment,这个强大的性能优化特性。 基础概念 🚀 包含类型 /* 布局包含 */ .layo…...
golang,多个proxy拉包的处理逻辑
在Go语言中,当你设置了多个代理(GOPROXY)时,Go工具链会按照你设置的顺序尝试每个代理。如果第一个代理失败,它会尝试下一个代理,直到成功获取到模块或者所有代理都尝试失败。最后,如果所有代理都…...
Vue使用Tinymce 编辑器
目录 一、下载并重新组织tinymce结构二、使用三、遇到的坑 一、下载并重新组织tinymce结构 下载 npm install tinymce^7 or yarn add tinymce^7重构目录 在node_moudles里找到tinymce文件夹,把里面文件拷贝一份放到public下,如下: -- pub…...
神经网络-AlexNet
AlexNet是在2012年的ImageNet竞赛后,整理发表的文章,也是对CNN网络的衍生。 网络结构 AlexNet网络结构如下图所示,网络分为了上下两部分,对应两个不同的GPU训练,可以更好的利用GPU算力。只有在特殊的网络层后&#x…...
《人工智能:洞察材料微观与宏观性能关系的神奇之眼》
在材料科学的广袤天地里,一个前沿且充满魅力的课题正吸引着全球科研人员的目光——如何借助人工智能的强大力量,精准模拟材料微观结构与宏观性能之间那错综复杂的关系。这不仅是解开材料性能之谜的关键钥匙,更是推动从航空航天到电子芯片等众…...
count(1)、count(_)与count(列名)的区别?
大家好,我是锋哥。今天分享关于【count(1)、count(_)与count(列名)的区别?】面试题。希望对大家有帮助; count(1)、count(_)与count(列名)的区别? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 在 SQL 中,…...
Docker 镜像加速访问方案
在数字化时代,Docker以其轻量级和便捷性成为开发者和运维人员的首选容器技术。然而自2023年5月中旬起,Docker Hub 的访问速度较慢或不稳定,这对依赖Docker Hub拉取镜像的用户来说无疑是一个挑战。本文将提供 Docker Hub 访问的一系列替代方案…...
菜鸟带新鸟——基于EPlan2022的部件库制作(3D)
设备逻辑的概念: 可在布局空间 中和其它对象上放置对象。可将其它对象放置在 3D 对象上。已放置的对象分到组件的逻辑结构中。 将此属性的整体标识为设备逻辑。可使用不同的功能创建和编辑设备逻辑。 设备的逻辑定义 定义 / 旋转 / 移动 / 翻转:组…...
笔记工具--MD-Markdown的语法技巧
MD格式,全称为Markdown格式,是一种轻量级标记语言文件,主要用于创建格式化文本。以下是对MD格式的详细解释: 一、定义与特点 定义:MD文件是一种纯文本格式的文件,使用简单的符号或语法来标记标题、列表、…...
【ES6复习笔记】生成器(11)
什么是生成器函数 生成器函数是一种特殊的函数,它可以在执行过程中暂停并保存当前状态,然后在需要时恢复执行。生成器函数通过 yield 关键字来实现暂停和恢复执行的功能。 生成器函数的基本用法 定义生成器函数:使用 function* 关键字来定…...
【解决报错】AttributeError: ‘NoneType‘ object has no attribute ‘group‘
学习爬虫时,遇到如下报错: 报错原因: 正则表达式的 search 或 finditer 方法没有找到任何匹配项,可能是换行符处理不当等。 解决方法如下: 在正则表达式末尾加上re.S即可,re.S是一个编译标志,…...
UE5.3 C++ Ceiusm中的POI 制作3DUI 结合坐标转化
一.核心思路WidgetComponent CesiumGloberAnchor 二.先制作POI 创建C Actor来制作,APOI。直接上代码 #pragma once#include "CoreMinimal.h" #include "GameFramework/Actor.h" #include "CesiumGlobeAnchorComponent.h" #includ…...
机器学习系列(一)——K-近邻算法
1. 算法定义 KNN 算法属于基于实例的机器学习方法。在对未知数据进行分类或回归之前,我们不需要对数据进行显式的训练或建立复杂的模型。它的核心思想是:对一个新的样本点,寻找在特征空间上与其最相似的 K 个已知数据点,采取“投…...
Android unitTest 单元测试用例编写(初始)
文章目录 了解测试相关库导入依赖库新建测试文件示例执行查看结果网页结果其他 本片讲解的重点是unitTest,而不是androidTest哦 了解测试相关库 androidx.compose.ui:ui-test-junit4: 用于Compose UI的JUnit 4测试库。 它提供了测试Compose UI组件的工具和API。 and…...
uni-app 跨端开发精美开源UI框架推荐
🎀🎀🎀uni-app 跨端开发系列 🎀🎀🎀 一、uni-app 组成和跨端原理 二、uni-app 各端差异注意事项 三、uni-app 离线本地存储方案 四、uni-app UI库、框架、组件选型指南 五、uni-app 蓝牙开发 六、uni-app …...
【微服务】微服务之Feign 与 Ribbon
文章目录 强烈推荐引言优点Feign示例什么是Ribbon?Ribbon 的优点Netflix Feign 和 Ribbon整合Feign 与 Ribbon 的关系Feign 与 Ribbon 结合使用的示例配置文件(application.yml)说明: Feign 与 Ribbon 结合使用的应用场景1. 动态服…...
【C语言】成绩等级制
将成绩分为A、B、C、D、E等级。具体的等级划分如下: A:90分及以上B:80分到89分C:70分到79分D:60分到69分E:60分以下 #include <stdio.h> int main() {float score 0;printf("请输入学生成绩&a…...
Flutter 插件开发入门
1、初识 Flutter Plugin Flutter 的插件类似于我们在 Android 中说的第三方库,通过使用插件,可以借助插件中的代码实现一些额外功能。 Flutter 的插件以 package 的形式存在,使用 package 的目的是为了达到模块化,可以让代码被共…...
2024.12.25在腾讯云服务器上使用docker部署flask
2024.12.25在腾讯云服务器上使用docker部署flask 操作系统:Ubuntu 根据腾讯云的说明文档安装 Docker 并配置镜像加速源,注意需要安装腾讯云的加速源,使用官网的加速源连接极其不稳定,容易导致运行失败。使用哪个公司的云服务器就…...
2024 年12月英语六级CET6听力原文(Lecture部分)
2024 年12月英语六级CET6听力原文(Long Conersation和Passage) 1 牛津大学关于普遍道德准则的研究及相关观点与建议 译文 2 食物颜色对味觉体验及大脑预期的影响 译文 3 财务资源对意义与幸福之间关系的影响研究 译文...
centos 释放系统预留内存并关闭Kdump服务
背景:Kdump是Linux系统的一种内核崩溃转储机制,它允许在系统发生内核崩溃(例如内核panic)时,捕获内存的转储信息,从而帮助事后分析故障原因。该过程需要一块预留内存(称为crashkernel内存&#…...
基于WEB的房屋出租管理系统设计
摘 要 随着城市化程度的推进,越来越多的人涌入城市,同时也带来的旺盛的租房需求,传统的房屋出租管理依赖人 工记录的方式难以满足人们对房屋出租管理的需求。因此,本文根据房屋出租信息化的需求设计一款基于房屋出租 的管理系统。…...
云边端架构的优势是什么?面临哪些挑战?
一、云边端架构的优势 降低网络延迟:在传统集中式架构中,数据需传输到云计算中心处理,导致网络延迟较高。而云边端架构将计算和存储推向边缘设备,可在离用户更近的地方处理数据,大大降低了网络延迟,提升了用…...
clickhouse解决suspiciously many的异常
1. 问题背景 clickhouse安装在虚拟机上,持续写入日志时,突然关机,然后重启,会出现clickhouse可以正常启动,但是查询sql语句,提示suspiciously many异常,如图所示 2. 问题修复 touch /data/cl…...
爬虫 APP 逆向 ---> shopee(虾皮) 电商
shopee 泰国站点:https://shopee.co.th/ shopee 网页访问时,直接弹出使用 app 登录查看,那就登录 shopee 泰国站点 app。 手机抓包:分类接口 接口:https://mall.shopee.co.th/api/v4/pages/get_category_tree 请求参…...
用例图和活动图的区别与联系
在软件开发过程中,需求分析是至关重要的一步。为了更好地理解和描述系统的功能需求,开发人员通常会使用各种图形化工具。其中,用例图和活动图是两种非常常用的工具。虽然它们都用于描述系统的行为,但各自具有不同的特点和适用场景…...
接口请求中调试可以看到Origin,其具体的作用
接口请求中带 Origin 是浏览器在跨域请求时自动添加的一个 HTTP 请求头,用于标识请求的来源(源)。Origin 头的主要作用是支持跨域资源共享(CORS)和增强安全性。 Origin 的含义 Origin 包含发起请求的源信息࿰…...
知识图谱+大模型:打造全新智慧城市底层架构
在数字化时代,智慧城市的建设正迎来新一轮的变革。本文将探讨如何结合知识图谱和大模型技术,构建智慧城市的全新底层架构,以应对日益增长的数据量和复杂性,提升城市管理的智能化水平。 知识图谱:智慧城市的知识库 知识…...
Android基于Path的addRoundRect,Canvas剪切clipPath简洁的圆形图实现,Kotlin(2)
Android基于Path的addRoundRect,Canvas剪切clipPath简洁的圆形图实现,Kotlin(2) import android.content.Context import android.graphics.BitmapFactory import android.graphics.Canvas import android.graphics.Path import a…...
计算机图形学知识点汇总
一、计算机图形学定义与内容 1.图形 图形分为“图”和“形”两部分。 其中,“形”指形体或形状,存在于客观世界和虚拟世界,它的本质是“表示”;而图则是包含几何信息与属性信息的点、线等基本图元构成的画面,用于表达…...
【EtherCATBasics】- KRTS C++示例精讲(2)
EtherCATBasics示例讲解 目录 EtherCATBasics示例讲解结构说明代码讲解 项目打开请查看【BaseFunction精讲】。 结构说明 EtherCATBasics:应用层程序,主要用于人机交互、数据显示、内核层数据交互等; EtherCATBasics.h : 数据定义…...
AI 简史:从神经元到现代大模型
AI 简史:从神经元到现代大模型 人工智能 (AI) 和深度学习 (DL) 在过去的几十年中飞速发展,推动了计算机视觉、自然语言处理和机器人等领域的进步。今年的诺贝尔物理学奖更是颁给了美国科学家约翰霍普菲尔德 (John Hopfield)和英国科学家杰弗…...
Kotlin入门到深入加强(1):基本介绍和第一个程序
吧, 一.关于本章和Kotlin 本章内容如要简单介绍一下Kotlin和创建并执行第一个Kotlin程序 什么是Kotlin,优势是什么,它和JAVA的关系又是什么 Kotlin是一种静态类型的编程语言,它运行在Java虚拟机上,并且可以与现有的Java代码无缝集成。Kotl…...
走进 Web3:探索分布式网络的未来
随着互联网的不断演变,我们正站在一个全新阶段的门槛上——Web3。它是对传统互联网(Web2)的一次深刻反思与升级,标志着一个去中心化、更加自主的数字世界的到来。Web3不仅仅是技术的革新,更代表着对互联网价值体系的根…...
信管通低代码信息管理系统应用平台
目前,国家统一要求事业单位的电脑都要进行国产化替代,替代后使用的操作系统都是基于linux的,所有以前在WINDOWS下运行的系统都不能使用了,再者,各单位的软件都很零散,没有统一起来。需要把日常办公相关的软…...
C语言从入门到放弃教程
C语言从入门到放弃 1. 介绍1.1 特点1.2 历史与发展1.3 应用领域 2. 安装2.1 编译器安装2.2 编辑器安装 3. 第一个程序1. 包含头文件2. 主函数定义3. 打印语句4. 返回值 4. 基础语法4.1 注释4.1.1 单行注释4.1.2 多行注释 4.2 关键字4.2.1 C语言标准4.2.2 C89/C90关键字…...
硬件设计:RS232电平标准
RS232是一种常用的串行通信接口标准,主要用于计算机和外部设备之间的数据传输。以下是RS232电平标准的详细介绍: 1. 电气特性: 信号电平: 逻辑1 (MARK):-3V到-15V之间。通常,-5V到-15V之间为有效的逻辑1。…...
前端初学基础
一.Web开发 前端三件 HTML ,页面展现 CSS,样式 JS(JavaScript),动起来 二,HTML 1.HTML概念 网页,网站中的一个页面,网页是构成网站的基本元素,是承载各种网站应用的平台。通俗的说,网站就…...
在 CentOS 7 上安装 Node.js 20 并升级 GCC、make 和 glibc
在 CentOS 7 上安装 Node.js 20 并升级 GCC、make 和 glibc 📖 前言 在 CentOS 7 上使用 NVM 安装 Node.js 后,可能会遇到如下问题: node: /lib64/libm.so.6: version GLIBC_2.27’ not found (required by node) node: /lib64/libc.so.6:…...