OpenHarmony实战:标准系统移植指南

本文描述了移植一块开发板的通用步骤,和具体芯片相关的详细移植过程无法在此一一列举。后续社区还会陆续发布开发板移植的实例供开发者参考。

定义开发板

本文以移植名为MyProduct的开发板为例讲解移植过程,假定MyProduct是MyProductVendor公司的开发板,使用MySoCVendor公司生产的MySOC芯片作为处理器。

定义产品

//vendor/MyProductVendor/{product_name}名称的目录下创建一个config.json文件,该文件用于描述产品所使用的SOC 以及所需的子系统。配置如下:

//vendor/MyProductVendor/MyProduct/config.json

{"product_name": "MyProduct","version": "3.0","type": "standard","target_cpu": "arm","ohos_version": "OpenHarmony 1.0","device_company": "MyProductVendor","board": "MySOC","enable_ramdisk": true,"subsystems": [{"subsystem": "ace","components": [{ "component": "ace_engine_lite", "features":[] }]},...]
}

主要的配置内容

配置项说明
product_name(必填)产品名称
version(必填)版本
type(必填)配置的系统级别,包含(small、standard等)
target_cpu(必填)设备的CPU类型(根据实际情况,这里的target_cpu也可能是arm64 、riscv、 x86等)
ohos_version(选填)操作系统版本
device_company(必填)device厂商名
board(必填)开发板名称
enable_ramdisk(必填)是否启动ramdisk
kernel_type(选填)内核类型
kernel_version(选填)kernel_type与kernel_version在standard是固定的不需要写
subsystems(必填)系统需要启用的子系统。子系统可以简单理解为一块独立构建的功能块。
product_company不体现在配置中,而是目录名,vendor下一级目录就是product_company,BUILD.gn脚本依然可以访问。

已定义的子系统可以在“//build/subsystem_config.json”中找到。当然你也可以定制子系统。

这里建议先拷贝Hi3516DV300 开发板的配置文件,删除掉 hisilicon_products 这个子系统。这个子系统为Hi3516DV300 SOC编译内核,显然不适合MySOC。

移植验证

至此,你可以使用如下命令,启动你产品的构建了:

./build.sh --product-name MyProduct 

构建完成后,可以在//out/{device_name}/packages/phone/images目录下看到构建出来的OpenHarmony镜像文件。

内核移植

这一步需要移植Linux内核,让Linux内核可以成功运行起来。

为SOC添加内核构建的子系统

修改文件//build/subsystem_config.json增加一个子系统。配置如下:

  "MySOCVendor_products": {"project": "hmf/MySOCVendor_products","path": "device/MySOCVendor/MySOC/build","name": "MySOCVendor_products","dir": "device/MySOCVendor"},

接着需要修改定义产品的配置文件//vendor/MyProductVendor/MyProduct/config.json,将刚刚定义的子系统加入到产品中。

编译内核

源码中提供了Linux 4.19的内核,归档在//kernel/linux-4.19。本节以该内核版本为例,讲解如何编译内核。

在子系统的定义中,描述了子系统构建的路径path,即//device/MySOCVendor/MySOC/build。这一节会在这个目录创建构建脚本,告诉构建系统如何构建内核。

建议的目录结构:

├── build
│ ├── kernel
│ │     ├── linux
│ │           ├──standard_patch_for_4_19.patch // 基于4.19版本内核的补丁
│ ├── BUILD.gn
│ ├── ohos.build

BUILD.gn是subsystem构建的唯一入口。

期望的构建结果

文件文件说明
$root_build_dir/packages/phone/images/uImage内核镜像
$root_build_dir/packages/phone/images/ubootbootloader镜像

移植验证

启动编译,验证预期的kernel镜像是否成功生成。

用户态启动引导

  1. 用户态进程启动引导总览。

    zh-cn_image_0000001199805369

    系统上电加载内核后,按照以下流程完成系统各个服务和应用的启动:

    1. 内核启动init进程,一般在bootloader启动内核时通过设置内核的cmdline来指定init的位置;如上图所示的"init=/init root/dev/xxx"。
    2. init进程启动后,会挂载tmpfs,procfs,创建基本的dev设备节点,提供最基本的根文件系统。
    3. init继续启动ueventd监听内核热插拔事件,为这些设备创建dev设备节点;包括block设备各个分区设备都是通过此事件创建。
    4. init进程挂载block设备各个分区(system,vendor),开始扫描各个系统服务的init启动脚本,并拉起各个SA服务。
    5. samgr是各个SA的服务注册中心,每个SA启动时,都需要向samgr注册,每个SA会分配一个ID,应用可以通过该ID访问SA。
    6. foundation是一个特殊的SA服务进程,提供了用户程序管理框架及基础服务;由该进程负责应用的生命周期管理。
    7. 由于应用都需要加载JS的运行环境,涉及大量准备工作,因此appspawn作为应用的孵化器,在接收到foundation里的应用启动请求时,可以直接孵化出应用进程,减少应用启动时间。
  2. init。

    init启动引导组件配置文件包含了所有需要由init进程启动的系统关键服务的服务名、可执行文件路径、权限和其他信息。每个系统服务各自安装其启动脚本到/system/etc/init目录下。

    新芯片平台移植时,平台相关的初始化配置需要增加平台相关的初始化配置文件/vendor/etc/init/init.{hardware}.cfg;该文件完成平台相关的初始化设置,如安装ko驱动,设置平台相关的/proc节点信息。

    init相关进程代码在//base/startup/init_lite目录下,该进程是系统第一个进程,无其它依赖。

    初始化配置文件具体的开发指导请参考 init启动子系统概述。

HDF驱动移植

LCD

HDF为LCD设计了驱动模型。支持一块新的LCD,需要编写一个驱动,在驱动中生成模型的实例,并完成注册。

这些LCD的驱动被放置在//drivers/hdf_core/framework/model/display/driver/panel目录中。

  1. 创建Panel驱动

    在驱动的Init方法中,需要调用RegisterPanel接口注册模型实例。如:

    int32_t XXXInit(struct HdfDeviceObject *object)
    {struct PanelData *panel = CreateYourPanel();// 注册if (RegisterPanel(panel) != HDF_SUCCESS) {HDF_LOGE("%s: RegisterPanel failed", __func__);return HDF_FAILURE;}return HDF_SUCCESS;
    }struct HdfDriverEntry g_xxxxDevEntry = {.moduleVersion = 1,.moduleName = "LCD_XXXX",.Init = XXXInit,
    };HDF_INIT(g_xxxxDevEntry);
  2. 配置加载panel驱动产品的所有设备信息被定义在文件//vendor/MyProductVendor/MyProduct/config/device_info/device_info.hcs中。修改该文件,在display的host中,名为device_lcd的device中增加配置。

    注意:moduleName要与panel驱动中的moduleName相同。

    root {...display :: host {device_lcd :: device {deviceN :: deviceNode {policy = 0;priority = 100;preload = 2;moduleName = "LCD_XXXX";}}}
    }

    更详细的驱动开发指导,请参考LCD。

触摸屏

本节描述如何移植触摸屏驱动。触摸屏的驱动被放置在//drivers/hdf_core/framework/model/input/driver/touchscreen目录中。移植触摸屏驱动主要工作是向系统注册ChipDevice模型实例。

  1. 创建触摸屏器件驱动

    在目录中创建名为touch_ic_name.c的文件。代码模板如下:注意:请替换ic_name为你所适配芯片的名称。

    #include "hdf_touch.h"static int32_t HdfXXXXChipInit(struct HdfDeviceObject *device)
    {ChipDevice *tpImpl = CreateXXXXTpImpl();if(RegisterChipDevice(tpImpl) != HDF_SUCCESS) {ReleaseXXXXTpImpl(tpImpl);return HDF_FAILURE;}return HDF_SUCCESS;
    }struct HdfDriverEntry g_touchXXXXChipEntry = {.moduleVersion = 1,.moduleName = "HDF_TOUCH_XXXX",.Init = HdfXXXXChipInit,
    };HDF_INIT(g_touchXXXXChipEntry);

    其中ChipDevice中要提供若干方法。

    方法实现说明
    int32_t (*Init)(ChipDevice *device)器件初始化
    int32_t (*Detect)(ChipDevice *device)器件探测
    int32_t (*Suspend)(ChipDevice *device)器件休眠
    int32_t (*Resume)(ChipDevice *device)器件唤醒
    int32_t (*DataHandle)(ChipDevice *device)从器件读取数据,将触摸点数据填写入device->driver->frameData中
    int32_t (*UpdateFirmware)(ChipDevice *device)固件升级
  2. 配置产品,加载器件驱动

    产品的所有设备信息被定义在文件//vendor/MyProductVendor/MyProduct/config/device_info/device_info.hcs中。修改该文件,在名为input的host中,名为device_touch_chip的device中增加配置。注意:moduleName 要与触摸屏驱动中的moduleName相同。

    deviceN :: deviceNode {policy = 0;priority = 130;preload = 0;permission = 0660;moduleName = "HDF_TOUCH_XXXX";deviceMatchAttr = "touch_XXXX_configs";
    }

    更详细的驱动开发指导,请参考TOUCHSCREEN。

WLAN

Wi-Fi驱动分为两部分,一部分负责管理WLAN设备,另一个部分负责处理WLAN流量。HDF WLAN分别为这两部分做了抽象。目前支持SDIO接口的WLAN芯片。

图1 WLAN芯片

zh-cn_image_0000001188241031

支持一款芯片的主要工作是实现一个ChipDriver驱动。实现HDF_WLAN_CORE和NetDevice提供的接口。主要需要实现的接口有:

接口定义头文件说明
HdfChipDriverFactory//drivers/hdf_core/framework/include/wifi/hdf_wlan_chipdriver_manager.hChipDriver的Factory,用于支持一个芯片多个Wi-Fi端口
HdfChipDriver//drivers/hdf_core/framework/include/wifi/wifi_module.h每个WLAN端口对应一个HdfChipDriver,用来管理一个特定的WLAN端口
NetDeviceInterFace//drivers/hdf_core/framework/include/net/net_device.h与协议栈之间的接口,如发送数据、设置网络接口状态等

建议适配按如下步骤操作:

  1. 创建HDF驱动建议将代码放置在//device/MySoCVendor/peripheral/wifi/chip_name/,文件模板如下:

    static int32_t HdfWlanXXXChipDriverInit(struct HdfDeviceObject *device) {static struct HdfChipDriverFactory factory = CreateChipDriverFactory();struct HdfChipDriverManager *driverMgr = HdfWlanGetChipDriverMgr();if (driverMgr->RegChipDriver(&factory) != HDF_SUCCESS) {HDF_LOGE("%s fail: driverMgr is NULL!", __func__);return HDF_FAILURE;}return HDF_SUCCESS;
    }struct HdfDriverEntry g_hdfXXXChipEntry = {.moduleVersion = 1,.Init = HdfWlanXXXChipDriverInit,.Release = HdfWlanXXXChipRelease,.moduleName = "HDF_WIFI_CHIP_XXX"
    };HDF_INIT(g_hdfXXXChipEntry);

    在CreateChipDriverFactory中,需要创建一个HdfChipDriverFactory,接口如下:

    接口说明
    const char *driverName当前driverName
    int32_t (*InitChip)(struct HdfWlanDevice *device)初始化芯片
    int32_t (*DeinitChip)(struct HdfWlanDevice *device)去初始化芯片
    void (_ReleaseFactory)(struct HdfChipDriverFactory _factory)释放HdfChipDriverFactory对象
    struct HdfChipDriver _(_Build)(struct HdfWlanDevice *device, uint8_t ifIndex)创建一个HdfChipDriver;输入参数中,device是设备信息,ifIndex是当前创建的接口在这个芯片中的序号
    void (_Release)(struct HdfChipDriver _chipDriver)释放chipDriver
    uint8_t (*GetMaxIFCount)(struct HdfChipDriverFactory *factory)获取当前芯片支持的最大接口数

    HdfChipDriver需要实现的接口有:

    接口说明
    int32_t (*init)(struct HdfChipDriver *chipDriver, NetDevice *netDev)初始化当前网络接口,这里需要向netDev提供接口NetDeviceInterFace
    int32_t (*deinit)(struct HdfChipDriver *chipDriver, NetDevice *netDev)去初始化当前网络接口
    struct HdfMac80211BaseOps *opsWLAN基础能力接口集
    struct HdfMac80211STAOps *staOps支持STA模式所需的接口集
    struct HdfMac80211APOps *apOps支持AP模式所需要的接口集
  2. 编写配置文件,描述驱动支持的设备。

    在产品配置目录下创建芯片的配置文件//vendor/MyProductVendor/MyProduct/config/wifi/wlan_chip_chip_name.hcs

    注意: 路径中的vendor_name、product_name、chip_name请替换成实际名称。

    模板如下:

    root {wlan_config {chip_name :& chipList {chip_name :: chipInst {match_attr = "hdf_wlan_chips_chip_name"; /* 这是配置匹配属性,用于提供驱动的配置根 */driverName = "driverName"; /* 需要与HdfChipDriverFactory中的driverName相同*/sdio {vendorId = 0x0296;deviceId = [0x5347];}}}}
    }
  3. 编写配置文件,加载驱动。

    产品的所有设备信息被定义在文件//vendor/MyProductVendor/MyProduct/config/device_info/device_info.hcs中。修改该文件,在名为network的host中,名为device_wlan_chips的device中增加配置。

    注意:moduleName 要与触摸屏驱动中的moduleName相同。

    deviceN :: deviceNode {policy = 0;preload = 2;moduleName = "HDF_WLAN_CHIPS";deviceMatchAttr = "hdf_wlan_chips_chip_name";serviceName = "driverName";
    }
  4. 构建驱动

    • 创建内核菜单在//device/MySoCVendor/peripheral目录中创建Kconfig文件,内容模板如下:

      config DRIVERS_WLAN_XXXbool "Enable XXX WLAN Host driver"default ndepends on DRIVERS_HDF_WIFIhelpAnswer Y to enable XXX Host driver. Support chip xxx

      接着修改文件//drivers/hdf_core/adapter/khdf/linux/model/network/wifi/Kconfig,在文件末尾加入如下代码将配置菜单加入内核中,如:

      source "../../../../../device/MySoCVendor/peripheral/Kconfig"
    • 创建构建脚本

      //drivers/hdf_core/adapter/khdf/linux/model/network/wifi/Makefile文件末尾增加配置,模板如下:

      HDF_DEVICE_ROOT := $(HDF_DIR_PREFIX)/../device
      obj-$(CONFIG_DRIVERS_WLAN_XXX) += $(HDF_DEVICE_ROOT)/MySoCVendor/peripheral/build/standard/

      当在内核中开启DRIVERS_WLAN_XXX开关时,会调用//device/MySoCVendor/peripheral/build/standard/中的makefile。

最后

有很多小伙伴不知道学习哪些鸿蒙开发技术?不知道需要重点掌握哪些鸿蒙应用开发知识点?而且学习时频繁踩坑,最终浪费大量时间。所以有一份实用的鸿蒙(HarmonyOS NEXT)资料用来跟着学习是非常有必要的。 

这份鸿蒙(HarmonyOS NEXT)资料包含了鸿蒙开发必掌握的核心知识要点,内容包含了ArkTS、ArkUI开发组件、Stage模型、多端部署、分布式应用开发、音频、视频、WebGL、OpenHarmony多媒体技术、Napi组件、OpenHarmony内核、Harmony南向开发、鸿蒙项目实战等等)鸿蒙(HarmonyOS NEXT)技术知识点。

希望这一份鸿蒙学习资料能够给大家带来帮助,有需要的小伙伴自行领取,限时开源,先到先得~无套路领取!!

获取这份完整版高清学习路线,请点击→纯血版全套鸿蒙HarmonyOS学习资料

鸿蒙(HarmonyOS NEXT)最新学习路线

  •  HarmonOS基础技能

  • HarmonOS就业必备技能 
  •  HarmonOS多媒体技术

  • 鸿蒙NaPi组件进阶

  • HarmonOS高级技能

  • 初识HarmonOS内核 
  • 实战就业级设备开发

有了路线图,怎么能没有学习资料呢,小编也准备了一份联合鸿蒙官方发布笔记整理收纳的一套系统性的鸿蒙(OpenHarmony )学习手册(共计1236页)鸿蒙(OpenHarmony )开发入门教学视频,内容包含:ArkTS、ArkUI、Web开发、应用模型、资源分类…等知识点。

获取以上完整版高清学习路线,请点击→纯血版全套鸿蒙HarmonyOS学习资料

《鸿蒙 (OpenHarmony)开发入门教学视频》

《鸿蒙生态应用开发V2.0白皮书》

图片

《鸿蒙 (OpenHarmony)开发基础到实战手册》

OpenHarmony北向、南向开发环境搭建

图片

 《鸿蒙开发基础》

  • ArkTS语言
  • 安装DevEco Studio
  • 运用你的第一个ArkTS应用
  • ArkUI声明式UI开发
  • .……

图片

 《鸿蒙开发进阶》

  • Stage模型入门
  • 网络管理
  • 数据管理
  • 电话服务
  • 分布式应用开发
  • 通知与窗口管理
  • 多媒体技术
  • 安全技能
  • 任务管理
  • WebGL
  • 国际化开发
  • 应用测试
  • DFX面向未来设计
  • 鸿蒙系统移植和裁剪定制
  • ……

图片

《鸿蒙进阶实战》

  • ArkTS实践
  • UIAbility应用
  • 网络案例
  • ……

图片

 获取以上完整鸿蒙HarmonyOS学习资料,请点击→纯血版全套鸿蒙HarmonyOS学习资料

总结

总的来说,华为鸿蒙不再兼容安卓,对中年程序员来说是一个挑战,也是一个机会。只有积极应对变化,不断学习和提升自己,他们才能在这个变革的时代中立于不败之地。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/790251.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

RC滤波电路

RC滤波电路 综述:本文简单讲述了RC低通滤波电路和RC高通滤波电路。 滤波电路是指过滤输入信号中不需要的信号,保留需要的信号。 一、RC低通滤波电路 1.定义:RC低通滤波电路:保留低频信号,衰减高频信号。 2.截止频率…

突破校园网限速:使用 iKuai 多拨分流负载均衡 + Clash 代理(内网带宽限制通用)

文章目录 1. 简介2. iKuai 部署2.1 安装 VMware2.2 安装 iKuai(1) 下载固件(2) 安装 iKuai 虚拟机(3) 配置 iKuai 虚拟机(4) 配置 iKuai(5) 配置多拨分流 2.3 测试速度 3. Clash 部署3.1 准备工作(1) 配置磁盘分区(2) 安装 Docker(3) 安装 Clash(4) 设置代理 1. 简介 由于博主…

014——超声波模块驱动开发Plus(基于I.MX6uLL、SR04和poll机制)

目录 一、基础知识 二、分析为什么打印会影响中断 三、驱动程序 四、应用程序 五、验证及其它 一、基础知识 013——超声波模块驱动开发(基于I.MX6uLL与SR04)-CSDN博客 二、分析为什么打印会影响中断 asmlinkage __visible int printk(const ch…

Vue的双向绑定v-model详细介绍

使用: 比如用户在登录注册时需要提交账号密码; 比如用户创建,更新时,需要提交一些数据; v-model指令可以在表单 input、textarea以及select元素上创建双向绑定; 它会根据控件类型自动选取正确的方法来更…

一条SQL查询语句是如何执行的

这是专栏的第一篇文章,我想来跟你聊聊 MySQL 的基础架构。我们经常说,看一个事儿千万不要直接陷入细节里,你应该先鸟瞰其全貌,这样能够帮助你从高维度理解问题。同样,对于 MySQL 的学习也是这样。平时我们使用数据库&a…

Windows11安装MySql-8.0.36安装详细教程(保姆级教程)

之前一直用的mysql5.7,最近导入一个项目一直报错,经查阅发现数据库mysql版本太老,今天特地重头下载安装配置一下,做个记录供大家参考。 下载安装包: 下载地址:https://dev.mysql.com/downloads/ 进入后选…

【linux】公共服务器如何清理过多的.cache缓存

【linux】公共服务器如何清理过多的.cache缓存. 【先赞后看养成习惯】求关注+点赞+收藏😀 问题叙述:用的公共服务器,管理员反映.cache缓存过大,让我清理一下 .cache介绍:在Linux系统中,.cache目录通常用于存储应用程序运行时生成的缓存文件。这些文件包括临时文件、预览…

C++STL--排序算法

sort 使用快速排序,平均性能好O(nlogn),但最差情况可能很差O(n^2)。不稳定。 sort(v.begin(),v.end());//对v容器进行排序,默认升序 sort(v.begin(),v.end(),greater<int>());//降序排序对于支持随机访问的迭代器的容器&#xff0c; 都可以利用sort算法直接对其进行排序…

滑动窗口算法 - LC76 最小覆盖子串

接上文滑窗基础题&#xff1a;滑动窗口算法 - LC3 无重复字符的最长子串-CSDN博客&#xff0c;介绍了滑窗的基础题目和滑窗解法模板&#xff0c;这次带来滑窗进阶题解。 76. 最小覆盖子串 困难 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果…

基于SSM的师生交流平台

目录 背景 技术简介 系统简介 界面预览 背景 传统的师生互动平台主要依赖于面对面的线下交流&#xff0c;用户必须亲自到场以获取和交流相关信息。然而&#xff0c;随着信息技术的广泛传播&#xff0c;众多教育机构开始转向线上发展&#xff0c;寻求更多样化的发展途径。线…

day17-二叉树part04

110.平衡二叉树 &#xff08;优先掌握递归&#xff09;后序遍历 左右中 class Solution {public boolean isBalanced(TreeNode root) {return getHeight(root) ! -1;}//递归三部曲 确定方法的参数与返回值private int getHeight(TreeNode root){//明确终止条件if(root null){r…

数据库管理-第168期 惯性(20240402)

数据库管理168期 2024-04-02 数据库管理-第168期 惯性&#xff08;20240402&#xff09;1 加法的惯性2 创新的惯性3 长期的责任总结 数据库管理-第168期 惯性&#xff08;20240402&#xff09; 作者&#xff1a;胖头鱼的鱼缸&#xff08;尹海文&#xff09; Oracle ACE Associa…

JS-优先级相关出现的一个小问题

console.log("嵌套函数中调用fn this值被赋给self" self o);这个输出会是 false。 在 JavaScript 中&#xff0c;比较两个对象是否相等时&#xff0c;实际上比较的是它们在内存中的引用地址&#xff0c;而不是它们的属性值。即使两个对象有相同的属性值&#xff…

C++11:声明 初始化

C11&#xff1a;声明 & 初始化 初始化{ }初始化initializer_list 声明autodecltypenullptr 初始化 { }初始化 在C98中&#xff0c;允许使用花括号{ }对数组或者结构体元素进行统一的列表初始化。 用{ }初始化数组&#xff1a; int arr[] { 1, 2, 3, 4, 5 };用{ }初始化…

详解2024年阿里云服务器租用价格表,最新报价

阿里云服务器租用价格表2024年最新&#xff0c;云服务器ECS经济型e实例2核2G、3M固定带宽99元一年&#xff0c;轻量应用服务器2核2G3M带宽轻量服务器一年61元&#xff0c;ECS u1服务器2核4G5M固定带宽199元一年&#xff0c;2核4G4M带宽轻量服务器一年165元12个月&#xff0c;2核…

C++中的List容器用法详解

文章目录 C中的List容器用法详解List 的特点List 的重要接口用法介绍1.创建和初始化Listlist 2.插入元素push_backpush_forntinsert 删除元素pop_backpop_fontclearerase 遍历List迭代器遍历范围for遍历 排序Listsort 反转Listreverse 转移Listsplice 去重unique 合并merge 总结…

网站可扩展架构设计——中台

从公众号转载&#xff0c;关注微信公众号掌握更多技术动态 --------------------------------------------------------------- 一、中台简介 1.传统项目架构的痛点 (1)重复造轮子 各项目相对独立&#xff0c;许多项目在重复造轮子&#xff0c;让项目本身越来越臃肿&#xf…

ts之基本类型,联合类型,函数,类的概念

1.ts中基础类型 数字 字符串 布尔 数组 元组 any never void null undefined 枚举 1.最基础的就是数字 字符串 布尔 number 和 大Number的区别 js特性 装箱的概念 xxx.xxx&#xff0c;string,boolean同理 let num1: number 1; let num2: Number 1; // 用来描述实例的 类也可…

精通并发【基础四】:创建线程的几种方式

在 Java 中&#xff0c;多线程编程是一种常见且强大的编程范式&#xff0c;它允许程序同时执行多个任务。创建线程是多线程编程的基础&#xff0c;Java 提供了几种不同的方法来创建和启动线程。本文将介绍三种常用的创建线程的方法&#xff1a;继承 Thread 类、实现 Runnable 接…

[Leetcode笔记] 动态规划相关

前言 写题目写到了一些和动态规划相关的内容&#xff0c;所以在这里记录一下 LCR 089 题解思路 总的来说&#xff0c;就是用一个数组去存储当前的最优解&#xff0c;然后从0开始一路向上顺推过去&#xff0c;求得最后一位的最优解。 class Solution { public:int rob(vect…