腾讯Kuikly跨端框架实战:5分钟搞定鸿蒙端Kotlin Multiplatform开发

# 腾讯Kuikly跨端框架实战:5分钟搞定鸿蒙端Kotlin Multiplatform开发 最近和几个做移动端的朋友聊天,大家普遍有个痛点:手头的项目既要维护Android和iOS,现在又多了鸿蒙这个新平台,团队资源被拉扯得厉害。每次新功能上线,都得在三个甚至更多的代码仓库里重复劳动,测试、发版周期长得让人头疼。如果你也正被这种多端开发的“泥潭”所困扰,那么今天聊的腾讯Kuikly框架,或许能给你带来一些不一样的思路。它不是一个简单的UI适配层,而是基于Kotlin Multiplatform (KMP) 技术栈,试图从根本上解决逻辑复用的问题,并且对鸿蒙HarmonyOS NEXT提供了相当深入的支持。这篇文章,我就从一个实践者的角度,带你快速走一遍用Kuikly在鸿蒙端跑起来第一个Kotlin跨平台组件的全过程,目标很明确:**用最短的时间,看到代码在鸿蒙设备上运行起来的效果**。 本文主要面向已经有一定Kotlin或Android开发经验,对跨平台方案感兴趣,特别是希望快速验证鸿蒙端可行性的开发者。我们不深入讨论过于底层的渲染原理,而是聚焦于“怎么做”——从环境准备、工程集成到编写第一个可运行的Compose UI,整个过程力求清晰、可操作。你会发现,借助成熟的工具链和清晰的步骤,让Kotlin代码跑在鸿蒙上,并没有想象中那么复杂。 ## 1. 理解Kuikly:为什么是它,以及它如何工作 在动手之前,我们有必要花几分钟了解一下Kuikly的核心设计思想。这能帮助你在后续遇到配置或代码问题时,知道该从哪个方向去排查。Kuikly本质上是一个**基于Kotlin Multiplatform的跨端开发框架**。它的野心不小,目标是实现“一码五端”,即同一套Kotlin业务逻辑和UI代码,能够编译并运行在Android、iOS、HarmonyOS、Web以及小程序平台上。 ### 1.1 架构核心:KMP与原生渲染的融合 Kuikly的架构可以粗略分为两层:**KuiklyBase**和**KuiklyUI**。 * **KuiklyBase** 是地基。它封装了Kotlin Multiplatform的通用能力,比如网络请求、本地存储、设备信息等平台相关接口的抽象。更重要的是,它处理了与各原生平台(包括鸿蒙)的桥接、工具链适配和稳定性监控。你可以把它想象成一个跨平台的“标准库”和“粘合剂”。 * **KuiklyUI** 是面向用户的部分。其中,**Kuikly Compose** 是它的声明式UI解决方案,语法上高度借鉴(兼容)Jetpack Compose,这让熟悉Android现代开发的开发者几乎可以零成本上手。 那么,Kuikly Compose在鸿蒙上是怎么把Kotlin代码变成屏幕上的像素的呢?这里有一个关键选择:**它没有使用鸿蒙官方的声明式UI框架ArkUI,而是直接调用了鸿蒙底层的命令式C API**。 > 这个设计决策初看有些反直觉,但仔细想想有其道理。直接调用C API意味着更少的抽象层,渲染指令从Kotlin/Native到系统绘制的路径更短,理论上能获得更接近原生的性能。同时,这种方案也让框架对UI控件的控制更加灵活,便于实现复杂的自定义渲染和控件混排。 简单来说,你的Compose UI代码在鸿蒙端会被Kuikly的渲染适配层“翻译”成一系列高效的C函数调用,直接驱动鸿蒙的图形系统。这个过程对开发者是透明的,你依然用着熟悉的Compose DSL写界面。 ### 1.2 横向对比:Kuikly在跨端生态中的位置 为了更清晰地定位Kuikly,我们可以将其与几个主流方案做个快速比较: | 特性维度 | Kuikly Compose | Flutter | React Native | Compose Multiplatform (CMP) | | :--- | :--- | :--- | :--- | :--- | | **核心架构** | KMP + 原生渲染(C API) | 自绘引擎(Skia) | JSI + 原生组件 | 声明式共享UI (实验性) | | **鸿蒙支持** | **官方适配,支持完整** | 依赖社区,滞后 | 实验性支持 | 尚未正式支持 | | **开发语言** | **Kotlin** | Dart | JavaScript/TypeScript | Kotlin | | **性能特点** | **接近原生** | 高,但包体积大 | 受Bridge影响 | 依赖平台实现 | | **动态化能力** | 支持页面级动态更新 | 官方支持热重载 | 依赖CodePush等 | 无 | 从上表可以看出,Kuikly的主要优势在于**语言统一(Kotlin)**、**对鸿蒙的原生级支持**以及**追求极致的性能**。对于已经投资Kotlin技术栈、尤其是Compose技术的团队,以及必须深度拥抱鸿蒙生态的项目,Kuikly提供了一个颇具吸引力的选项。 ## 2. 5分钟快速启动:从零搭建鸿蒙Kuikly工程 理论部分就此打住,我们直接进入实战环节。假设你手头已经有一个现成的鸿蒙应用工程,或者愿意新建一个,接下来的步骤将引导你完成Kuikly渲染器的集成。 ### 2.1 前期准备与环境检查 工欲善其事,必先利其器。请确保你的开发环境满足以下要求: 1. **IDE**: 鸿蒙官方IDE **DevEco Studio**。建议使用较新版本,以获得更好的Kotlin和C++支持。 2. **SDK**: 在DevEco Studio中安装HarmonyOS SDK,特别是**Native (C++)** 相关的开发工具链。 3. **基础知识**: 对鸿蒙应用的基本结构(`entry`、`oh-package.json`)、ArkTS语法有基本了解。同时,需要具备简单的C++/Node-API (NAPI) 接口调用概念。 > 提示:如果你还没有鸿蒙工程,可以在DevEco Studio中快速创建一个`Empty Ability`模板项目,应用类型选择`Application`,开发模型选择`Stage`。这个新项目将作为我们集成Kuikly的“宿主”。 ### 2.2 关键步骤:集成Kuikly渲染器依赖 集成过程的核心,是在你的鸿蒙工程中引入Kuikly的运行时库,并建立Kotlin/Native代码与ArkTS UI之间的通信桥梁。 **第一步:添加依赖声明** 打开你的鸿蒙模块(通常是`entry`)下的 `oh-package.json5` 文件。在 `dependencies` 字段中添加Kuikly渲染器的依赖。 ```json // entry/oh-package.json5 { "name": "entry", "version": "1.0.0", "dependencies": { // ... 其他已有依赖 "@kuikly-open/render": "latest" // 建议指定具体稳定版本,如 "1.0.0-beta01" } } ``` 添加后,IDE通常会提示你进行同步(Sync)或执行 `ohpm install` 命令来下载依赖包。 **第二步:配置C++(NAPI)支持** 由于Kuikly底层通过Kotlin/Native与C API交互,我们需要在鸿蒙工程中启用C++支持,并创建一个NAPI接口供ArkTS调用。 1. 在 `entry/src/main/` 目录下,确保存在 `cpp` 文件夹。如果没有,请在项目设置中为`entry`模块添加`Native C++`支持。 2. 在 `cpp` 目录下创建或编辑 `napi_init.cpp` 文件。这个文件将定义一个初始化函数,作为Kotlin/Native世界的入口。 ```cpp // entry/src/main/cpp/napi_init.cpp #include "napi/native_api.h" // 这个函数将被ArkTS调用,用于初始化Kuikly Native运行时 static napi_value InitKuikly(napi_env env, napi_callback_info info) { // 这里未来会放置Kuikly Native的初始化逻辑。 // 目前我们只需提供一个空函数,建立连接通道。 return nullptr; } EXTERN_C_START static napi_value Init(napi_env env, napi_value exports) { napi_property_descriptor desc[] = { // 将 InitKuikly 函数暴露给ArkTS,名称为 "initKuikly" {"initKuikly", nullptr, InitKuikly, nullptr, nullptr, nullptr, napi_default, nullptr}, }; napi_define_properties(env, exports, sizeof(desc) / sizeof(desc[0]), desc); return exports; } EXTERN_C_END // 模块注册,这是NAPI的标准流程 static napi_module demoModule = { .nm_version = 1, .nm_flags = 0, .nm_filename = nullptr, .nm_register_func = Init, .nm_modname = "entry", .nm_priv = ((void*)0), .reserved = { 0 }, }; extern "C" __attribute__((constructor)) void RegisterEntryModule(void) { napi_module_register(&demoModule); } ``` 3. 为了让ArkTS知道这个C++函数的存在,我们还需要类型声明。在 `cpp/types/libentry/` 目录下(可能需要手动创建`types/libentry`目录),创建 `index.d.ts` 文件。 ```typescript // entry/src/main/cpp/types/libentry/index.d.ts export const initKuikly: () => number; ``` 至此,Native层的基础桥梁就搭建好了。接下来,我们需要在ArkTS层创建管理类,将这座桥与Kuikly框架连接起来。 ## 3. 构建ArkTS适配层:连接Kuikly与鸿蒙UI Kuikly框架需要鸿蒙端提供一个“管理者”来协调生命周期和Native初始化。我们将在ArkTS中实现几个关键的类。 ### 3.1 实现Native管理器 首先,在 `entry/src/main/ets` 下创建一个 `kuikly` 目录,用于存放所有Kuikly相关的ArkTS适配代码。 然后,创建 `MyNativeManager.ets` 文件。这个类继承自Kuikly提供的 `KuiklyNativeManager`,其核心任务是重写 `loadNative` 方法,调用我们刚才在C++中定义的 `initKuikly` 函数。 ```typescript // entry/src/main/ets/kuikly/MyNativeManager.ets import { KuiklyNativeManager } from '@kuikly-open/render'; import nativeManager from 'libentry.so'; // 导入我们编译好的C++模块 class MyNativeManager extends KuiklyNativeManager { protected loadNative(): number { // 调用NAPI接口,触发Kuikly Native运行时的初始化 return nativeManager.initKuikly(); } } // 导出一个全局单例实例,供整个应用使用 const globalNativeManager = new MyNativeManager(); export default globalNativeManager as KuiklyNativeManager; ``` 这个 `globalNativeManager` 将是Kuikly框架在鸿蒙端的“锚点”。 ### 3.2 创建视图委托 接下来,创建 `KuiklyViewDelegate.ets`。这个委托类负责处理一些自定义扩展,比如注册鸿蒙原生自定义组件到Kuikly中,或者响应页面的生命周期事件。对于第一个Demo,我们可以先提供一个最小实现。 ```typescript // entry/src/main/ets/kuikly/KuiklyViewDelegate.ets import { IKuiklyViewDelegate, KRRenderModuleExportCreator, KRRenderViewExportCreator } from '@kuikly-open/render'; export class KuiklyViewDelegate extends IKuiklyViewDelegate { // 返回自定义渲染视图的创建器映射(目前为空) getCustomRenderViewCreatorRegisterMap(): Map<string, KRRenderViewExportCreator> { return new Map(); } // 返回自定义渲染模块的创建器映射(目前为空) getCustomRenderModuleCreatorRegisterMap(): Map<string, KRRenderModuleExportCreator> { return new Map(); } } ``` 现在,连接层已经准备就绪。是时候创建一个真正的页面来承载我们的Kuikly UI了。 ## 4. 编写第一个Kuikly页面:让Compose在鸿蒙上渲染 我们将改造或新建一个鸿蒙的Page,在其中嵌入Kuikly组件。这里以修改默认的 `Index` 页面为例。 ### 4.1 页面结构设计与生命周期 在 `kuikly` 目录下创建 `pages/Index.ets`(或者直接修改已有的页面)。这个页面的结构会稍复杂,因为它需要处理Kuikly控制器的初始化、参数传递以及生命周期同步。 ```typescript // entry/src/main/ets/kuikly/pages/Index.ets import { KRRecord, KRNativeRenderController, Kuikly } from '@kuikly-open/render'; import globalNativeManager from '../MyNativeManager'; import { KuiklyViewDelegate } from '../KuiklyViewDelegate'; import router from '@ohos.router'; @Entry @Component struct Index { // 1. 创建视图委托实例 private kuiklyViewDelegate: KuiklyViewDelegate = new KuiklyViewDelegate(); // 2. 用于持有Kuikly渲染控制器的引用 private kuiklyController: KRNativeRenderController | null = null; // 3. 要加载的Kuikly页面名称(由Kotlin侧定义) private pageName: string = 'MainScreen'; // 示例页面名 // 4. 传递给Kuikly页面的初始数据 private pageData?: KRRecord = {}; // 5. 控制Kuikly组件显示的标志位 @State showKuikly: boolean = false; aboutToAppear(): void { // 在实际应用中,pageName和pageData可能通过路由参数传入 // 这里为了简化,我们直接使用固定值 this.showKuikly = true; } build() { Stack() { if (this.showKuikly) { // 核心:Kuikly组件 Kuikly({ pageName: this.pageName, pageData: this.pageData ?? {}, delegate: this.kuiklyViewDelegate, contextCode: '', // 动态代码相关,初学可留空 onControllerReadyCallback: (controller: KRNativeRenderController) => { // 当Kuikly控制器准备就绪时的回调 this.kuiklyController = controller; console.info('Kuikly Render Controller is ready.'); // 可以在这里注册异常回调等 controller.registerExceptionCallback((mode, stack) => { console.error('Kuikly Render Error:', stack); }); }, nativeManager: globalNativeManager, }) .width('100%') .height('100%') } } .width('100%') .height('100%') } onPageShow(): void { // 通知Kuikly页面已显示 this.kuiklyViewDelegate.pageDidAppear(); } onPageHide(): void { // 通知Kuikly页面已隐藏 this.kuiklyViewDelegate.pageDidDisappear(); } } ``` 这个页面做了几件关键事: * 在 `build` 方法中条件渲染 `Kuikly` 组件。 * 通过 `pageName` 告诉Kuikly框架该加载哪个Kotlin端定义的Composable。 * 在 `onControllerReadyCallback` 中获取控制器实例,便于后续交互。 * 将之前创建的 `globalNativeManager` 和 `kuiklyViewDelegate` 注入组件。 * 同步鸿蒙页面的生命周期 (`onPageShow`/`onPageHide`) 给Kuikly。 ### 4.2 编写Kotlin共享端的Compose UI 鸿蒙端的架子搭好了,现在轮到“主角”登场——用Kotlin编写跨平台的UI逻辑。我们需要在Kotlin Multiplatform的共享代码模块(通常是 `commonMain`)中编写Compose代码。 假设你的KMP项目已经设置好,在 `commonMain/kotlin/` 路径下,创建一个简单的Composable函数: ```kotlin // 在KMP共享模块中,例如:com.yourcompany.kuiklydemo.commonMain import androidx.compose.runtime.Composable import androidx.compose.material3.Text import androidx.compose.material3.MaterialTheme import androidx.compose.ui.Modifier import androidx.compose.ui.unit.sp @Composable fun MainScreen() { // 这是一个极其简单的Compose UI androidx.compose.material3.Surface( modifier = Modifier.fillMaxSize(), color = MaterialTheme.colorScheme.background ) { androidx.compose.foundation.layout.Column( modifier = Modifier.fillMaxSize(), horizontalAlignment = Alignment.CenterHorizontally, verticalArrangement = Arrangement.Center ) { Text( text = "Hello, HarmonyOS from Kuikly!", fontSize = 24.sp, color = MaterialTheme.colorScheme.primary ) Text( text = "This UI is written in Kotlin Compose.", fontSize = 16.sp, color = MaterialTheme.colorScheme.onSurface ) } } } ``` 接下来,我们需要在Kotlin端注册这个Composable,使其能够被鸿蒙端通过 `pageName` 调用。这通常在共享模块或Android主模块的初始化处完成(具体取决于Kuikly框架的注册API,请参考其官方文档)。一个概念性的注册代码可能如下: ```kotlin // 在Kotlin端的某个初始化位置 Kuikly.registerScreen("MainScreen") { MainScreen() } ``` ### 4.3 编译、运行与调试 最后,也是最激动人心的步骤: 1. **编译Kotlin代码为鸿蒙库**:使用Kuikly提供的Gradle插件或工具,将你的Kotlin共享代码(包含 `MainScreen`)编译为鸿蒙系统可识别的Native库(如 `.so` 文件)。 2. **集成到鸿蒙工程**:将上一步生成的Native库文件,放置到鸿蒙工程 `entry/src/main/cpp/libs/` 目录下对应的ABI子目录中(例如 `arm64-v8a`)。 3. **修改C++初始化函数**:回头修改 `napi_init.cpp` 中的 `InitKuikly` 函数,确保它正确调用了Kuikly Native库的初始化入口。这步通常需要参考Kuikly Native SDK的头文件和文档。 4. **构建并运行鸿蒙应用**:在DevEco Studio中,选择你的鸿蒙设备或模拟器,点击运行。如果一切配置正确,应用启动后,你应该能看到一个白色的背景上显示着“Hello, HarmonyOS from Kuikly!”的文本。 > 注意:第1、3步的具体命令和API调用强烈依赖于Kuikly框架当前版本提供的工具链和文档。实践中,请务必查阅最新的官方集成指南。这里描述的是通用的流程和概念。 当看到用Kotlin Compose编写的界面在鸿蒙设备上流畅显示时,最初的集成工作就取得了关键性成功。这意味着你已经打通了从Kotlin跨平台代码到鸿蒙原生渲染的完整链路。接下来的工作,就是在这个基础上,继续构建更复杂的业务逻辑和UI交互,享受一套代码多端部署带来的效率提升。

创作声明:本文部分内容由AI辅助生成(AIGC),仅供参考

Python内容推荐

负荷预测基于贝叶斯网络的考虑不确定性的短期电能负荷预测(Python代码实现)

负荷预测基于贝叶斯网络的考虑不确定性的短期电能负荷预测(Python代码实现)

内容概要:本文围绕基于贝叶斯网络的短期电能负荷预测方法展开,重点解决电力负荷预测中存在的不确定性问题。通过构建贝叶斯网络模型,整合历史负荷数据及关键外部影响因素(如气温、湿度、节假日、时段特征等),实现了对短期电能负荷的概率化预测。该方法不仅能输出未来负荷的点估计值,还可提供预测结果的置信区间与概率分布,有效量化预测风险,显著提升了预测结果在实际电力系统调度、风险管理和决策支持中的可用性与可靠性。文中配套提供了完整的Python代码实现,涵盖数据预处理、网络结构学习、参数估计与概率推理全过程,便于读者复现、验证并进一步拓展应用。; 适合人群:具备一定Python编程能力、概率统计基础和机器学习背景,从事电力系统运行分析、能源管理、智能电网研究或相关领域的科研人员、工程师及高校研究生。; 使用场景及目标:①应用于电网公司或能源企业,提升短期负荷预测的精度与鲁棒性,优化发电计划与资源配置;②在高比例可再生能源接入的新型电力系统中,评估负荷波动带来的不确定性,辅助储能系统配置、需求响应策略制定与市场竞价决策;③作为教学与科研案例,深入理解贝叶斯网络在时间序列预测、不确定性建模与因果推理中的具体应用。; 阅读建议:建议读者在学习过程中结合提供的Python代码,动手实践模型构建的每一个环节,重点关注变量选择、网络拓扑结构的设计、条件概率表的确定以及推理算法的实现细节,并尝试在不同的实际数据集上进行测试与调优,以深刻掌握贝叶斯方法在处理复杂不确定性问题上的优势与技巧。

Kuikly框架实战:Material3导航栏[可运行源码]

Kuikly框架实战:Material3导航栏[可运行源码]

腾讯Kuikly框架是一个为跨端开发而生的框架,它基于Kotlin Multiplatform技术,能够支持包括PC、Web、小程序、鸿蒙、IOS、Android在内的六端开发。

帮你避开90%论文被拒坑的研究设计诊断工具。基于Montgomery、Goodfellow、Zobel三本领域圣经,四层诊断框架.zip

帮你避开90%论文被拒坑的研究设计诊断工具。基于Montgomery、Goodfellow、Zobel三本领域圣经,四层诊断框架.zip

一个专为本科/研究生论文写作设计的AI技能,支持工科、心理学、教育学、管理学等多学科领域,提供符合中国学术规范(GB/T 7714-2015)的论文写作、数据分析、参考文献管理一体化解决方案。

msdaora驱动-下载即用.zip

msdaora驱动-下载即用.zip

下载代码方式:https://pan.quark.cn/s/7c9f6e46935a WIN7操作系统内置了MSDAORA驱动,具体为OLE DB Provider for Oracle(MSDAORA)。对于32位WIN7系统,系统自带的msdaOra.dll驱动文件存放于路径:C:\Program Files\Common Files\System\Ole DB\目录中。(注意:WINDOWS 7及更高版本不再默认包含此驱动)。若在WIN7后续版本中需要使用该驱动,可通过将WIN7的MSDAORA驱动复制到当前电脑上实现。msdaOra驱动文件msdaOra.dll或msdaOrar.dll解压缩后,应放置于相同路径:C:\Program Files\Common Files\System\Ole DB\目录下。五、通过打开"开始-运行"功能,输入regsvr32 msdaora.dll后回车,即可完成驱动注册并解决问题。所提供的msdaora.dll文件希望能对您有所帮助。出于个人便利考虑,现将该文件存储为OleDB-MSDAORA_x32.rar格式供参考使用。

自动化浇水系统,利用Arduino UNO实时监测土壤湿度,并在MATLAB中显示实时数据.zip

自动化浇水系统,利用Arduino UNO实时监测土壤湿度,并在MATLAB中显示实时数据.zip

1.版本:matlab2014a/2019b/2024b 2.附赠案例数据可直接运行。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。

(旭辉 讯达)外圈挖沟有滑板7号线原程序.gxw

(旭辉 讯达)外圈挖沟有滑板7号线原程序.gxw

(旭辉 讯达)外圈挖沟有滑板7号线原程序.gxw

yoloe-26x-seg-v8.4.0.pt

yoloe-26x-seg-v8.4.0.pt

yoloe-26x-seg-v8.4.0.pt

贪吃蛇C语言代码 超级好玩.exe

贪吃蛇C语言代码 超级好玩.exe

源码链接: https://pan.quark.cn/s/a4b39357ea24 由C语言编写的贪吃蛇程序代码,极具趣味性,对于下载表示诚挚的谢意,多次表达深深的感激之情。

【带RL负载的全波桥式整流器】功能齐全的单相非控整流器(Simulink)

【带RL负载的全波桥式整流器】功能齐全的单相非控整流器(Simulink)

内容概要:本文详细介绍了基于Simulink构建的功能齐全的单相非控全波桥式整流器仿真模型,重点研究其在RL(电阻-电感)负载条件下的工作特性与电气行为。该模型精确模拟了交流电源通过由四个二极管构成的桥式电路转换为脉动直流电的全过程,并结合电感与电阻负载,有效展现了电流的连续性和平滑效果。模型完整涵盖了整流器的核心功能,可用于深入分析输出电压与电流的波形特征、功率因数、谐波含量等关键性能指标,是电力电子技术领域中进行整流电路原理验证、性能评估及教学演示的理想工具。; 适合人群:适用于电气工程、自动化、电力电子及相关专业的本科高年级学生、研究生,以及从事电力电子技术开发与研究的科研人员和工程师。; 使用场景及目标:①作为高校电力电子课程的教学辅助工具,用于课堂演示和学生实验仿真,加深对全波整流原理的理解;②作为科研项目中AC-DC变换环节的基准模块,用于性能测试、参数优化及新型控制策略的前期验证;③帮助使用者探究非控整流器在不同RL参数下的动态响应过程与稳态运行特性,掌握负载变化对整流效果的影响规律。; 阅读建议:建议使用者在具备电力电子技术基础知识的前提下,亲自运行并调试该Simulink模型,通过改变交流电源的电压、频率以及负载的电阻和电感值,实时观察和记录输出波形的变化,从而更深刻地理解电路的工作机理和性能特点。

串口读写最简demo CreateFile

串口读写最简demo CreateFile

下载代码方式:https://pan.quark.cn/s/ec082249541f 在Windows编程中,与硬件设备交互,如串口通信,通常需要用到系统提供的API函数。 "CreateFile 读写串口 最简demo" 是一个适用于VC6(Visual C++ 6.0)开发环境的示例项目,它演示了如何使用CreateFile函数来打开、读取和写入串行端口。 以下将详细介绍这个知识点。 `CreateFile` 是Windows API中的一个重要函数,用于创建、打开、重命名或获取设备句柄。 它的原型如下:```cppHANDLE CreateFile( LPCTSTR lpFileName, // 设备名或路径 DWORD dwDesiredAccess, // 访问模式 DWORD dwShareMode, // 共享模式 LPSECURITY_ATTRIBUTES lpSecurityAttributes, // 安全属性 DWORD dwCreationDisposition, // 创建或打开模式 DWORD dwFlagsAndAttributes, // 文件或设备属性 HANDLE hTemplateFile // 模板文件句柄);```在串口通信中,`lpFileName` 参数通常设定为`"COM1:"`、`"COM2:"`等,表示你要操作的串口。 例如,要打开`COM1`,可以这样写:```cppLPCSTR comName = "COM1:";HANDLE hSerialPort = CreateFile(comName, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATT...

基于微信小程序的房屋租赁系统+源码+文档(毕业设计&课程设计&项目开发)

基于微信小程序的房屋租赁系统+源码+文档(毕业设计&课程设计&项目开发)

基于微信小程序的房屋租赁系统+源码+文档,适合毕业设计、课程设计、项目开发。项目源码已经过严格测试,可以放心参考并在此基础上延申使用,详情见md文档 ouseServer:houseServer是服务端的代码,使用的是node+express+mysql开发 2、leasehouse:是微信小程序的代码, 3、mywx-projiect:是后台管理页面的代码,使用的是vue+element-ui开发 基于微信小程序的房屋租赁系统+源码+文档,适合毕业设计、课程设计、项目开发。项目源码已经过严格测试,可以放心参考并在此基础上延申使用,详情见md文档 ouseServer:houseServer是服务端的代码,使用的是node+express+mysql开发 2、leasehouse:是微信小程序的代码, 3、mywx-projiect:是后台管理页面的代码,使用的是vue+element-ui开发~

算法交易项目(QT/C++).zip

算法交易项目(QT/C++).zip

这是一个基于Qt和C++开发的**数值计算与可视化工具**,主要用于股价或时间序列数据的插值与拟合分析。项目名称"算法交易"更侧重于"使用算法处理行情数据"的含义,而非实际的交易系统。

辩论赛计时器,html版,适合单位比赛等,即开即用

辩论赛计时器,html版,适合单位比赛等,即开即用

即开即用,可以自定义标题,自定义背景,自定义标志,傻瓜式操作,全程可视化操作,画面精美。

信号与系统(第四版)吴大正习题答案

信号与系统(第四版)吴大正习题答案

代码转载自:https://pan.quark.cn/s/ae2b1dd9bd2a 《信号与系统》版本说明(V0.1) 王一刚 Sept.10, 2018 2018年9月,本系统正式开始上线,在接下去的日子里,希望在大家共同努力下,网络课件能够越做越好。 本课件制作工具VsCode,采用MarkDown语言设计,所以文件的后缀名均为md。 所以同学们也可以用VsCode进行查看和修改,提交。 \# 所需基础知识: git原理,参考文档progit2.pdf。 参考网站https://git-scm.com MarkDown基本格式与原理 VsCode基本操作 课程文档链接:https://.com/NIT2018/NitSignal.git VsCode显示MarkDown页面效果时,需要添加相关插件。 有个笨办法:在Debug菜单,选择Start Debugging,然后点击more。 此时在左边页面出现一堆插件,在上面搜索栏输入MarkDown,就在左边会出现一批插件,可以选择其中某几种,但是要预览的话,必须选中Markdown Preview Enhanced。 下图就是我的VsCode中整个工作页面。 image [^_^]: image 接下来,我们正式进入学习环节。 点击进入目录

OpenHarmony/applications-clock

OpenHarmony/applications-clock

Clock 是 OpenHarmony 系统中的基础时钟应用,提供闹钟、秒表、倒计时和世界时钟四大核心功能。该应用采用 ArkTS 语言开发,基于 OpenHarmony Stage 模型,支持 Phone 和 Tablet 多种设备形态,具备响应式布局、多设备适配和无障碍支持等特性。

ESXi6.7显卡直通说明.docx

ESXi6.7显卡直通说明.docx

代码下载地址: https://pan.quark.cn/s/c9f2aa48972f Single GPU Passthrough on Ryzen CPU https://liucreator.gitlab.io/zh/posts/0x0b-single-gpu-passthrough/main/ 中文版本 These are my steps to set up QEMU/KVM with GPU passthrough as of 2021 March, it works for me as I tried many times on different systems. You might need to change a few things, and feel free to ask me if you have any questions! My current setup is on Ryzen CPU with Radeon GPU, and sorry my English is not the best. Followers YouTube Channel Subscribers My Specs: CPU: AMD Ryzen 3900x 12 cores GPU: MSI Radeon RX 560 4GT LP OC RAM: 32GB 3200Mhz dual channels OS: Arch Linux 5.11 DE: KDE Plasma 5.21 * Procedure: Setting up an basic KVM on linux without VFIO Libvirt Hooks and Scripts QEMU...

易语言源码RGB-CMYK颜色值互换易语言源码

易语言源码RGB-CMYK颜色值互换易语言源码

易语言源码RGB_CMYK颜色值互换易语言源码

Centos8 百度网盘链接

Centos8 百度网盘链接

源码链接: https://pan.quark.cn/s/cf720b6761d1 centos8-fast-init 最小化安装centos8 之后, 快速初始化环境, 使新机器最短时间可以上线作为开发机/生产机使用 停止维护,最新版本为 https://.com/xisj/centos9-fast-init 一键安装 快速上线 需要先修改 vps.sh 缓解sshd长时间连接后自动断开的问题 禁止穷举ssh密码 如果被大量穷举, 可以使用正经的fail2ban 修改 /etc/fail2ban/jail.conf docker安装centos8:latest 遇到 Failed to download metadata for repo ‘AppStream’ 更换阿里源 替换yum源为ali源 centos8 对镜像的选择做的不错,一般情况下无需替换 备份 下载fedora的epel仓库 刷新数据 更新到最新 ----- 新上架机器后, 启用网卡 修改 ONBOOT=no 为 ONBOOT=yes 重启网络服务 查看ip 可选操作: 更新时区 docker内部更新时区 手动个ssh开放更多端口 手工安装gcc https://sourceforge.net/projects/mingw-w64/files/ 修改环境变量 安装ffmpeg -- ffmpeg 推荐安装ffmpeg* ,这样不会因为缺少库出现奇怪问题 安装 youtube-dl 安装秘钥实现免密登陆 利用ssh 实现内外网联通 ssh -D :7777 -N admin@example.com

最新股份分红协议范本.docx

最新股份分红协议范本.docx

最新股份分红协议范本.docx

同城-家政按摩H5小程序源码  -.zip

同城-家政按摩H5小程序源码 -.zip

同城-家政按摩H5小程序源码 -.zip

最新推荐最新推荐

recommend-type

螺旋桨参数化设计的Matlab工具集

该螺旋桨参数化设计工具基于Matlab软件平台(支持2014a、2019b及2024b版本)开发,核心功能是实现螺旋桨关键参数的调整与优化,从而辅助工程设计、科学研究及教学活动。该工具采用参数化编程模式,代码结构清晰,逻辑严谨,并配有详细注释,便于用户理解、修改和扩展。 工具适用于计算机科学、电子信息工程、数学等专业的高等教育场景,尤其适合课程设计、期末大作业和毕业设计等任务。通过调整螺旋桨的尺寸、桨叶数量、角度分布及旋转速度等参数,用户可系统研究各参数对螺旋桨性能的影响。参数化方法不仅提升了设计效率,降低了实验成本,还通过Matlab强大的数值计算与图形处理能力,实现了精确的仿真与直观的观察。 此外,工具附赠可直接运行的案例数据,帮助用户快速上手并掌握螺旋桨参数化设计的基本原理与技巧。该工具整合了自动化与智能化设计理念,对提升螺旋桨设计水平具有重要价值,是相关专业学生及工程师的实用资源。 资源来源于网络分享,仅用于学习交流使用,请勿用于商业,如有侵权请联系我删除!
recommend-type

CPF-ApplicationTPC/ohos-grpc-node

grpc 是一个适用于 OpenHarmony 的高性能远程过程调用(RPC)框架,旨在简化分布式系统中的服务通信。grpc 基于 HTTP/2 协议进行数据传输,支持简单 RPC、客户端流式 RPC、服务器流式 RPC 和双向流式 RPC 四种通信模式,具备低延迟、高吞吐、可双向通信等特性。
recommend-type

全域数据智慧梳理归集方案.pptx

全域数据智慧梳理归集方案.pptx
recommend-type

IP地址数据库(iP138测试版-txt)

下载代码方式:https://pan.quark.cn/s/a4b39357ea24 qqwry.dat 纯真(CZ88.NET)自2005年起一直为广大社区用户提供社区版IP地址库,只要获得纯真的授权就能免费使用,并不断获取后续更新的版本。 如果有需要免费版IP库的朋友可以前往纯真的官网进行申请。 纯真除了免费的社区版IP库外,还提供数据更加准确、服务更加周全的商业版IP地址查询数据。 纯真围绕IP地址,基于 网络空间拓扑测绘 + 移动位置大数据 方案,对IP地址定位、IP网络风险、IP使用场景、IP网络类型、秒拨侦测、VPN侦测、代理侦测、爬虫侦测、真人度等均有近20年丰富的数据沉淀。 IP地址解析 Demo: Perl version Changelog QQ IP数据库 纯真版 20231122 - 记录总数:547299条 - 更新日期:2023年11月22日 - 数据库版本:纯真
recommend-type

mysql数据类型变更-下载即用.zip

源码直接下载地址: https://pan.quark.cn/s/3de753365538 ### MySQL 数据类型转换详述#### 一、引言在MySQL数据库系统中,数据类型转换是一项至关重要的操作。这项技能使我们能够灵活应对不同类型的数据,特别是在执行数据检索或处理操作时,经常需要将一种数据类型转化为另一种数据类型。本文将深入探讨MySQL中的数据类型转换方式及其具体应用场景。#### 二、MySQL数据类型转换概述MySQL提供了多种途径来实现数据类型的转换,主要包括`CAST`函数、`CONVERT`函数以及直接运用算术运算等手段。接下来,我们将详细解析这些转换方法及其适用情境。#### 三、数字类型向字符串类型转换1. **运用`CONCAT`函数** 当需要将整型(例如`INT`)转化为字符串类型(例如`VARCHAR`)时,可以利用`CONCAT`函数达成这一目的。 ```sql -- 示例:将整数8转换为字符串80 SELECT CONCAT(8, 0); ``` 或者将数字转化为`CHAR`类型: ```sql -- 示例:将整数123转换为字符串 SELECT CAST(123 AS CHAR); -- 或者采用CONCAT方法 SELECT CONCAT(123, ); ```2. **运用`CAST`函数** `CAST`函数能够直接将一个表达式转换成指定的数据类型。 ```sql -- 示例:将字符串123转换为整型 SELECT CAST(123 AS SIGNED); ```#### 四、字符串类型向数字类型转换1. **运用`CAST`函数** `CAST`函数同样适用于将字符串转换成数字类型。 ```sql -- 示例:将...
recommend-type

学生成绩管理系统C++课程设计与实践

资源摘要信息:"学生成绩信息管理系统-C++(1).doc" 1. 系统需求分析与设计 在进行学生成绩信息管理系统开发前,首先需要进行系统需求分析,这是确定系统开发目标与范围的过程。需求分析应包括数据需求和功能需求两个方面。 - 数据需求分析: - 学生成绩信息:需要收集学生的姓名、学号、课程成绩等数据。 - 数据类型和长度:明确每个数据项的数据类型(如字符串、整型等)和长度,例如学号可能是字符串类型且长度为一定值。 - 描述:详细描述每个数据项的意义,以确保系统能够准确处理。 - 功能需求分析: - 列出功能列表:用户界面应提供清晰的操作指引,列出所有可用功能。 - 查询学生成绩:系统应能通过学号或姓名查询学生的成绩信息。 - 增加学生成绩信息:允许用户添加未保存的学生成绩信息。 - 删除学生成绩信息:能够通过学号或姓名删除已经保存的成绩信息。 - 修改学生成绩信息:通过学号或姓名修改已有的成绩记录。 - 退出程序:提供安全退出程序的选项,并确保所有修改都已保存。 2. 系统设计 系统设计阶段主要完成内存数据结构设计、数据文件设计、代码设计、输入输出设计、用户界面设计和处理过程设计。 - 内存数据结构设计: - 使用链表结构组织内存中的数据,便于动态增删查改操作。 - 数据文件设计: - 选择文本文件存储数据,便于查看和编辑。 - 代码设计: - 根据功能需求,编写相应的函数和模块。 - 输入输出设计: - 设计简洁明了的输入输出提示信息和操作流程。 - 用户界面设计: - 用户界面应为字符界面,方便在命令行环境下使用。 - 处理过程设计: - 设计数据处理流程,确保每个操作都有明确的处理逻辑。 3. 系统实现与测试 实现阶段需要根据设计阶段的成果编写程序代码,并进行系统测试。 - 程序编写: - 完成系统设计中所有功能的程序代码编写。 - 系统测试: - 设计测试用例,通过测试用例上机测试系统。 - 记录测试方法和测试结果,确保系统稳定可靠。 4. 设计报告撰写 最后,根据系统开发的各个阶段,撰写详细的设计报告。 - 系统描述:包括问题说明、数据需求和功能需求。 - 系统设计:详细记录内存数据结构设计、数据文件设计、代码设计、输入/输出设计、用户界面设计、处理过程设计。 - 系统测试:包括测试用例描述、测试方法和测试结果。 - 设计特点、不足、收获和体会:反思整个开发过程,总结经验和教训。 时间安排: - 第19周(7月12日至7月16日)完成项目。 - 7月9日8:00到计算机学院实验中心(三楼)提交程序和课程设计报告。 指导教师和系主任(或责任教师)需要在文档上签名确认。 系统需求分析: - 使用表格记录系统需求分析的结果,包括数据项、数据类型、数据长度和描述。 - 分析数据项如学生成绩信息、状态器、链表节点等,确定其属性和行为。 以上就是文档中提到的学生成绩信息管理系统开发的关键知识点。开发此类系统需要熟练掌握C++编程基础,了解面向对象的程序设计思想,以及熟悉文件操作和链表等数据结构的应用。此外,良好的软件开发流程意识、测试意识和文档撰写能力也是必不可少的。
recommend-type

别再手动拖拽了!用Lumerical脚本批量创建FDTD仿真结构(附完整代码)

# 告别低效建模:Lumerical脚本自动化实战指南 在光子学仿真领域,时间就是科研生命线。当同行还在GUI界面里反复点击菜单时,你已经用脚本批量生成了20组参数化结构——这不是未来场景,而是每位FDTD用户都应该掌握的基础生产力革命。本文将彻底改变你与Lumerical的交互方式,从手动拖拽的农耕时代,跃迁到自动化建模的工业文明。 ## 1. 为何脚本建模是必然选择 2019年Nature Photonics的一项研究显示,科研工作者在仿真工具上平均浪费37%的时间在重复性操作上。对于需要参数扫描的纳米光学结构设计,这个数字可能更高。手动创建10个不同尺寸的纳米柱阵列意味着: -
recommend-type

Java邮件解析任务中,如何安全高效地提取HTML邮件内容并避免硬编码、资源泄漏和类型转换异常?

<think>我们被要求优化一段Java代码,该代码用于处理邮件(特别是来自特定发件人的构建通知邮件)。代码的主要问题包括: 1. 重复获取邮件内容:在检查MIME类型后,多次调用`msg[i].getContent()`,这可能导致性能问题或流关闭异常。 2. 类型转换问题:直接将邮件内容转换为`Multipart`而不进行类型检查,可能引发`ClassCastException`。 3. 代码结构问题:逻辑嵌套过深,可读性差,且存在重复代码(如插入邮件详情的操作在两个地方都有)。 4. 硬编码和魔法值:例如在解析HTML表格时使用了硬编码的索引(如list3.get(10)),这容易因邮件
recommend-type

RH公司应收账款管理优化策略研究

资源摘要信息:"本文针对RH公司的应收账款管理问题进行了深入研究,并提出了改进策略。文章首先分析了应收账款在企业管理中的重要性,指出其对于提高企业竞争力、扩大销售和充分利用生产能力的作用。然后,以RH公司为例,探讨了公司应收账款管理的现状,并识别出合同管理、客户信用调查等方面的不足。在此基础上,文章提出了一系列改善措施,包括完善信用政策、改进业务流程、加强信用调查和提高账款回收力度。特别强调了建立专门的应收账款回收部门和流程的重要性,并建议在实际应用过程中进行持续优化。同时,文章也意识到企业面临复杂多变的内外部环境,因此提出的策略需要根据具体情况调整和优化。 针对财务管理领域的专业学生和从业者,本文提供了一个关于应收账款管理问题的案例研究,具有实际指导意义。文章还探讨了信用管理和征信体系在应收账款管理中的作用,强调了它们对于提升企业信用风险控制和市场竞争能力的重要性。通过对比国内外企业在应收账款管理上的差异,文章总结了适合中国企业实际环境的应收账款管理方法和策略。" 根据提供的文件内容,以下是详细的知识点: 1. 应收账款管理的重要性:应收账款作为企业的一项重要资产,其有效管理关系到企业的现金流、财务健康以及市场竞争力。不良的应收账款管理会导致资金链断裂、坏账损失增加等问题,严重影响企业的正常运营和长远发展。 2. 应收账款的信用风险:在信用交易日益频繁的商业环境中,企业必须对客户信用进行评估,以便采取合理的信用政策,降低信用风险。 3. 合同管理的薄弱环节:合同是应收账款管理的法律基础,严格的合同管理能够保障企业权益,减少因合同问题导致的应收账款风险。 4. 客户信用调查:了解客户的信用状况对于预测和控制应收账款风险至关重要。企业需要建立有效的客户信用调查机制,识别和筛选信用良好的客户。 5. 应收账款回收策略:企业应建立有效的账款回收机制,包括定期的账款跟进、逾期账款的催收等。同时,建立专门的应收账款回收部门可以提升回收效率。 6. 应收账款管理流程优化:通过改进企业内部管理流程,如简化审批流程、提高工作效率等措施,能够提升应收账款的管理效率。 7. 应收账款管理策略的调整和优化:由于企业的内外部环境复杂多变,因此制定的管理策略需要根据实际情况进行动态调整和持续优化。 8. 信用管理和征信体系的作用:建立和完善企业内部信用管理体系和征信体系,有助于企业更好地控制信用风险,并在市场竞争中占据有利地位。 9. 对比国内外应收账款管理实践:通过研究国内外企业在应收账款管理上的不同做法和经验,可以借鉴先进的管理理念和方法,提升国内企业的应收账款管理水平。 综上所述,本文深入探讨了应收账款管理的多个方面,为RH公司乃至其他同类型企业提供了应收账款管理的改进方向和策略,对于财务管理专业的教育和实践都具有重要的参考价值。
recommend-type

新手别慌!用BingPi-M2开发板带你5分钟搞懂Tina Linux SDK目录结构

# 新手别慌!用BingPi-M2开发板带你5分钟搞懂Tina Linux SDK目录结构 第一次拿到BingPi-M2开发板时,面对Tina Linux SDK里密密麻麻的文件夹,我完全不知道从哪下手。就像走进一个陌生的大仓库,每个货架上都堆满了工具和零件,却找不到操作手册。这种困惑持续了整整两天,直到我意识到——理解目录结构比死记硬背每个文件更重要。 ## 1. 为什么SDK目录结构如此重要 想象你正在组装一台复杂的模型飞机。如果所有零件都混在一个箱子里,你需要花大量时间寻找每个螺丝和面板。但如果有分门别类的隔层,标注着"机身部件"、"电子设备"、"紧固件",组装效率会成倍提升。Ti