HarmonyOS ArkUI实战开发-NAPI异步编程

2024-04-23 05:20

本文主要是介绍HarmonyOS ArkUI实战开发-NAPI异步编程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

笔者在前 5 小节里讲述了在 OpenHarmony 上通过 NAPI 的方式实现了 JS 调用 C++的能力,但是这些实现都是同步的,本节笔者简单介绍一下 NAPI 的异步实现。

约定编程规范

ArkUI 开发框架对外提供的 API 命名是需遵守一定规范的,以 @ohos.display 模块提供的 API 为例,源码如下所示:

declare namespace display {function getDefaultDisplay(callback: AsyncCallback<Display>): void;function getDefaultDisplay(): Promise<Display>;function getDefaultDisplaySync(): Display;
}

根据该模块提供的方法,根据方法的命名规则可以得出 2 条规范:

  • 同步调用:

    • 方法名+ Sync 关键字,如:getMd5Sync():string
  • 异步调用:

    • 需要提供 AsyncCallback 和 Promise 的实现,如:getMd5(): Promise<string>getMd5(callback: AsyncCallback<Display>)

因此,我们在 index.d.ts 中声明 NAPI 方法时也按照系统约定的规范来。

定义异步方法

笔者在第 5 小结实现了 MD5 的计算,本节笔者把 MD5 的实现放在异步线程中,先在 index.d.ts 声明 JS 侧的方法,如下所示:

export const add: (a: number, b: number) => number;// 声明异步方法
export function getMd5(value: string, callback: (md5: string) => void): void;
export function getMd5(value: string): Promise<string>;// 声明同步方法
export function getMd5Sync(value: string): string;

getMd5Sync()表示同步实现 MD5 的计算,getMd5() 表示异步实现 MD5 的调用。

实现异步方法

声明完 JS 端的方法后,接着在 hello.cpp 中实现对应的方法,步骤如下:

  • 添加映射

    在 hello.cpp 的 Init() 方法里添加 JS 端的方法映射,代码如下所示:

    static napi_value Init(napi_env env, napi_value exports) {napi_property_descriptor desc[] = {{"add", nullptr, Add, nullptr, nullptr, nullptr, napi_default, nullptr},{"getMd5Sync", nullptr, GetMd5Sync, nullptr, nullptr, nullptr, napi_default, nullptr},{"getMd5", nullptr, GetMd5, nullptr, nullptr, nullptr, napi_default, nullptr},};napi_define_properties(env, exports, sizeof(desc) / sizeof(desc[0]), desc);return exports;}

"getMd5Sync" 和 GetMd5Sync 分别表示 JS 端和 C++ 端的方法,通过 napi_define_properties() 把他们映射在一起。

  • 方法实现
    getMd5() 的 C++ 端代码如下所示:
    // 定义异步线程执行中需要的上下文环境struct Md5Context {// 异步 workernapi_async_work work;// 对应 JS 端的 callback 函数napi_ref callback;// 对应 JS 端的 promise 对象napi_deferred promise;// 传递进来的参数string params;// 计算后的结果string result;};// 在子线程中执行static void doInBackground(napi_env env, void *data) {Md5Context *md5Context = (Md5Context *)data;// 模拟耗时操作,进行 MD5 计算string md5 = MD5(md5Context->params).toStr();// 计算后的 MD5 字存储到 result 中md5Context->result = md5;// 模拟耗时操作,让当前线程休眠 3 秒钟std::this_thread::sleep_for(std::chrono::seconds(3));}// 切换到主线程static void onPostExecutor(napi_env env, napi_status status, void *data) {Md5Context *md5Context = (Md5Context *)data;napi_value returnValue;if (napi_ok !=napi_create_string_utf8(env, md5Context->result.c_str(), md5Context->result.length(), &returnValue)) {delete md5Context;md5Context = nullptr;napi_throw_error(env, "-111", "napi_create_string_utf8: error");return;}if (md5Context->callback) {// 取出缓存的 js 端的 callbacknapi_value callback;if (napi_ok != napi_get_reference_value(env, md5Context->callback, &callback)) {delete md5Context;md5Context = nullptr;napi_throw_error(env, "-111", "napi_get_reference_value error");return;}napi_value tempValue;// 调用 callback,把值回调给 JS 端napi_call_function(env, nullptr, callback, 1, &returnValue, &tempValue);// 删除 callbacknapi_delete_reference(env, md5Context->callback);} else {// 以 promise 的形式回调数据if (napi_ok != napi_resolve_deferred(env, md5Context->promise, returnValue)) {delete md5Context;md5Context = nullptr;napi_throw_error(env, "-111", "napi_resolve_deferred error");}}// 删除异步任务并释放资源napi_delete_async_work(env, md5Context->work);delete md5Context;md5Context = nullptr;}static napi_value GetMd5(napi_env env, napi_callback_info info) {// 1、从 info 中读取 JS 传递过来的参数放入 args 里size_t argc = 2;napi_value args[2] = {nullptr};if (napi_ok != napi_get_cb_info(env, info, &argc, args, nullptr, nullptr)) {napi_throw_error(env, "-1001", "napi_get_cb_info error");return nullptr;}// 2、读取传入的参数类型napi_valuetype stringType = napi_undefined;if (napi_ok != napi_typeof(env, args[0], &stringType)) {napi_throw_error(env, "-1002", "napi_typeof string error");return nullptr;}// 3、传入的 string 如果为 null 或者 undefined 则抛异常if (napi_null == stringType || napi_undefined == stringType) {napi_throw_error(env, "-1003", "input params null or undefined");return nullptr;}// 4、读取传入的 string 内容长度size_t length = 0;if (napi_ok != napi_get_value_string_utf8(env, args[0], nullptr, 0, &length)) {napi_throw_error(env, "-1004", "get string length error");return nullptr;}// 5、判断传入的 string 长度是否符合if (0 == length) {napi_throw_error(env, "-1005", "string length can't be zero");return nullptr;}// 6、读取传入的 string 长度读取内容char *buffer = new char[length + 1];if (napi_ok != napi_get_value_string_utf8(env, args[0], buffer, length + 1, &length)) {delete[] buffer;buffer = nullptr;napi_throw_error(env, "-1006", "napi_get_value_string_utf8 string error");return nullptr;}// 7、读取 JS 有没有传递 callback,如果 callback 为 null 就表示是 promise 的回调方式napi_valuetype callbackType = napi_undefined;napi_status callbackStatus = napi_typeof(env, args[1], &callbackType);if (napi_ok != callbackStatus && napi_invalid_arg != callbackStatus) {delete[] buffer;buffer = nullptr;napi_throw_error(env, "-1004", "napi_typeof function error");return nullptr;}// 8、创建一个异步线程需要的数据 model,把传递过来的参数加入进去做下缓存auto context = new Md5Context();context->params = buffer;napi_value returnValue = nullptr;// 9、判断是 callback 的回调方式还是 promise 的回调方式if (napi_function == callbackType) {// 如果是 callback 的回调方式,需要创建 callback 的引用napi_ref callback;if (napi_ok != napi_create_reference(env, args[1], 1, &callback)) {delete[] buffer;delete context;buffer = nullptr;context = nullptr;napi_throw_error(env, "-11", "napi_create_reference error");return nullptr;}// 缓存 callbackcontext->callback = callback;// 临时返回一个 undefined 值给 JS 端napi_get_undefined(env, &returnValue);} else {// promise 的回调方式,创建一个 Promise 的引用napi_deferred promise;if (napi_ok != napi_create_promise(env, &promise, &returnValue)) {delete[] buffer;delete context;buffer = nullptr;context = nullptr;napi_throw_error(env, "-11", "napi_create_promise error");return nullptr;}// 缓存 promisecontext->promise = promise;}napi_value resourceName;if (napi_ok != napi_create_string_utf8(env, "GetMd5", NAPI_AUTO_LENGTH, &resourceName)) {delete[] buffer;delete context;buffer = nullptr;context = nullptr;napi_throw_error(env, "-11", "napi_create_string_utf8 resourceName error");return nullptr;}// 10、创建一个异步任务napi_async_work asyWork;napi_status status =napi_create_async_work(env, nullptr, resourceName, doInBackground, onPostExecutor, (void *)context, &asyWork);if (napi_ok != status) {delete[] buffer;delete context;buffer = nullptr;context = nullptr;napi_throw_error(env, "-11", "napi_create_async_work error");return nullptr;}// 11、保存异步任务context->work = asyWork;// 12、添加进异步队列napi_queue_async_work(env, asyWork);return returnValue;}

getMd5() 的代码比较多,笔者添加的注释比较清楚,前 6 个小步骤是对传递进来的参数做基础校验,第 7 步是根据参数判断当前异步执行的回调方式是 Promise 还是 Callback。第 8 步创建了一个 Md5Context 对象,它的作用是把当前相关参数缓存下来目的是接下来在异步线程里使用这些参数,第 9 步根据异步回调的方法创建 Promise 或者 Callback 然后把他们保存在 Md5Context 对象里。第 10 步创建一个异步任务,然后把异步任务添加进异步队列中。

napi_create_async_work() 方法的第 3 、 4 个参数需要注意,doInBackground() 方法是在异步线程中执行的,onPostExecutor() 方法在异步线程结束后切换到主线程中执行。

  • 完整代码
    hello.cpp 全部代码如下所示:
    #include <cstddef>#include <cstring>#include "napi/native_api.h"#include <js_native_api.h>#include <js_native_api_types.h>#include <node_api.h>#include <node_api_types.h>#include <string>#include <thread>#include "./md5/md5.h"// 定义异步线程执行中需要的上下文环境struct Md5Context {// 异步 workernapi_async_work work;// 对应 JS 端的 callback 函数napi_ref callback;// 对应 JS 端的 promise 对象napi_deferred promise;// 传递进来的参数string params;// 计算后的结果string result;};static void doInBackground(napi_env env, void *data) {Md5Context *md5Context = (Md5Context *)data;// 模拟耗时操作,进行 MD5 计算string md5 = MD5(md5Context->params).toStr();// 计算后的 MD5 字存储到 result 中md5Context->result = md5;// 模拟耗时操作,让当前线程休眠 3 秒钟std::this_thread::sleep_for(std::chrono::seconds(3));}static void onPostExecutor(napi_env env, napi_status status, void *data) {Md5Context *md5Context = (Md5Context *)data;napi_value returnValue;if (napi_ok !=napi_create_string_utf8(env, md5Context->result.c_str(), md5Context->result.length(), &returnValue)) {delete md5Context;md5Context = nullptr;napi_throw_error(env, "-111", "napi_create_string_utf8: error");return;}if (md5Context->callback) {// 取出缓存的 js 端的 callbacknapi_value callback;if (napi_ok != napi_get_reference_value(env, md5Context->callback, &callback)) {delete md5Context;md5Context = nullptr;napi_throw_error(env, "-111", "napi_get_reference_value error");return;}napi_value tempValue;// 调用 callback,把值回调给 JS 端napi_call_function(env, nullptr, callback, 1, &returnValue, &tempValue);// 删除 callbacknapi_delete_reference(env, md5Context->callback);} else {// 以 promise 的形式回调数据if (napi_ok != napi_resolve_deferred(env, md5Context->promise, returnValue)) {delete md5Context;md5Context = nullptr;napi_throw_error(env, "-111", "napi_resolve_deferred error");}}// 删除异步任务并释放资源napi_delete_async_work(env, md5Context->work);delete md5Context;md5Context = nullptr;}static napi_value GetMd5(napi_env env, napi_callback_info info) {// 1、从 info 中读取 JS 传递过来的参数放入 args 里size_t argc = 2;napi_value args[2] = {nullptr};if (napi_ok != napi_get_cb_info(env, info, &argc, args, nullptr, nullptr)) {napi_throw_error(env, "-1001", "napi_get_cb_info error");return nullptr;}// 2、读取传入的参数类型napi_valuetype stringType = napi_undefined;if (napi_ok != napi_typeof(env, args[0], &stringType)) {napi_throw_error(env, "-1002", "napi_typeof string error");return nullptr;}// 3、传入的 string 如果为 null 或者 undefined 则抛异常if (napi_null == stringType || napi_undefined == stringType) {napi_throw_error(env, "-1003", "input params null or undefined");return nullptr;}// 4、读取传入的 string 内容长度size_t length = 0;if (napi_ok != napi_get_value_string_utf8(env, args[0], nullptr, 0, &length)) {napi_throw_error(env, "-1004", "get string length error");return nullptr;}// 5、判断传入的 string 长度是否符合if (0 == length) {napi_throw_error(env, "-1005", "string length can't be zero");return nullptr;}// 6、读取传入的 string 长度读取内容char *buffer = new char[length + 1];if (napi_ok != napi_get_value_string_utf8(env, args[0], buffer, length + 1, &length)) {delete[] buffer;buffer = nullptr;napi_throw_error(env, "-1006", "napi_get_value_string_utf8 string error");return nullptr;}// 7、读取 JS 有没有传递 callback,如果 callback 为 null 就表示是 promise 的回调方式napi_valuetype callbackType = napi_undefined;napi_status callbackStatus = napi_typeof(env, args[1], &callbackType);if (napi_ok != callbackStatus && napi_invalid_arg != callbackStatus) {delete[] buffer;buffer = nullptr;napi_throw_error(env, "-1004", "napi_typeof function error");return nullptr;}// 8、创建一个异步线程需要的数据 model,把传递过来的参数加入进去做下缓存auto context = new Md5Context();context->params = buffer;napi_value returnValue = nullptr;// 9、判断是 callback 的回调方式还是 promise 的回调方式if (napi_function == callbackType) {// 如果是 callback 的回调方式,需要创建 callback 的引用napi_ref callback;if (napi_ok != napi_create_reference(env, args[1], 1, &callback)) {delete[] buffer;delete context;buffer = nullptr;context = nullptr;napi_throw_error(env, "-11", "napi_create_reference error");return nullptr;}// 缓存 callbackcontext->callback = callback;// 临时返回一个 undefined 值给 JS 端napi_get_undefined(env, &returnValue);} else {// promise 的回调方式,创建一个 Promise 的引用napi_deferred promise;if (napi_ok != napi_create_promise(env, &promise, &returnValue)) {delete[] buffer;delete context;buffer = nullptr;context = nullptr;napi_throw_error(env, "-11", "napi_create_promise error");return nullptr;}// 缓存 promisecontext->promise = promise;}napi_value resourceName;if (napi_ok != napi_create_string_utf8(env, "GetMd5", NAPI_AUTO_LENGTH, &resourceName)) {delete[] buffer;delete context;buffer = nullptr;context = nullptr;napi_throw_error(env, "-11", "napi_create_string_utf8 resourceName error");return nullptr;}// 10、创建一个异步任务napi_async_work asyWork;napi_status status =napi_create_async_work(env, nullptr, resourceName, doInBackground, onPostExecutor, (void *)context, &asyWork);if (napi_ok != status) {delete[] buffer;delete context;buffer = nullptr;context = nullptr;napi_throw_error(env, "-11", "napi_create_async_work error");return nullptr;}// 11、保存异步任务context->work = asyWork;// 12、添加进异步队列napi_queue_async_work(env, asyWork);return returnValue;}static napi_value GetMd5Sync(napi_env env, napi_callback_info info) {// 1、从info中取出JS传递过来的参数放入argssize_t argc = 1;napi_value args[1] = {nullptr};if (napi_ok != napi_get_cb_info(env, info, &argc, args, nullptr, nullptr)) {napi_throw_error(env, "-1000", "napi_get_cb_info error");return nullptr;}// 2、获取参数的类型napi_valuetype stringType;if (napi_ok != napi_typeof(env, args[0], &stringType)) {napi_throw_error(env, "-1001", "napi_typeof error");return nullptr;}// 3、如果参数为null或者undefined,则抛异常if (napi_null == stringType || napi_undefined == stringType) {napi_throw_error(env, "-1002", "the param can't be null");return nullptr;}// 4、获取传递的string长度size_t length = 0;if (napi_ok != napi_get_value_string_utf8(env, args[0], nullptr, 0, &length)) {napi_throw_error(env, "-1003", "napi_get_value_string_utf8 error");return nullptr;}// 5、如果传递的是"",则抛异常if (length == 0) {napi_throw_error(env, "-1004", "the param length invalid");return nullptr;}// 6、读取传递的string参数放入buffer中char *buffer = new char[length + 1];if (napi_ok != napi_get_value_string_utf8(env, args[0], buffer, length + 1, &length)) {delete[] buffer;buffer = nullptr;napi_throw_error(env, "-1005", "napi_get_value_string_utf8 error");return nullptr;}// 7、计算MD5加密操作std::string str = buffer;str = MD5(str).toStr();// 8、把C++数据转成napi_value并返回napi_value value = nullptr;const char *md5 = str.c_str();if (napi_ok != napi_create_string_utf8(env, md5, strlen(md5), &value)) {delete[] buffer;buffer = nullptr;napi_throw_error(env, "-1006", "napi_create_string_utf8 error");return nullptr;}// 9、资源清理delete[] buffer;buffer = nullptr;return value;}static napi_value Add(napi_env env, napi_callback_info info) {size_t requireArgc = 2;size_t argc = 2;napi_value args[2] = {nullptr};napi_get_cb_info(env, info, &argc, args, nullptr, nullptr);napi_valuetype valuetype0;napi_typeof(env, args[0], &valuetype0);napi_valuetype valuetype1;napi_typeof(env, args[1], &valuetype1);double value0;napi_get_value_double(env, args[0], &value0);double value1;napi_get_value_double(env, args[1], &value1);napi_value sum;napi_create_double(env, value0 + value1, &sum);return sum;}EXTERN_C_STARTstatic napi_value Init(napi_env env, napi_value exports) {napi_property_descriptor desc[] = {{"add", nullptr, Add, nullptr, nullptr, nullptr, napi_default, nullptr},{"getMd5Sync", nullptr, GetMd5Sync, nullptr, nullptr, nullptr, napi_default, nullptr},{"getMd5", nullptr, GetMd5, nullptr, nullptr, nullptr, napi_default, nullptr},};napi_define_properties(env, exports, sizeof(desc) / sizeof(desc[0]), desc);return exports;}EXTERN_C_ENDstatic 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); }

Index.ets 的测试代码如下:

    import testNapi from 'libentry.so';@Entry @Component struct Index {@State message: string = 'Hello,OpenHarmony'build() {Column({ space: 10 }) {Text(this.message).fontSize(20)Button("同步回调").onClick(() => {this.message = testNapi.getMd5Sync("Hello, OpenHarmony")})Button("异步 Callback 回调").onClick(() => {this.message = "计算中...";testNapi.getMd5("Hello, OpenHarmony", (md5: string) => {this.message = md5;});})Button("异步 Promise 回调").onClick(() => {this.message = "计算中...";testNapi.getMd5("Hello, OpenHarmony").then((md5: string) => {this.message = md5;}).catch((error: Error) => {this.message = "error: " + error;})})}.padding(10).width('100%').height("100%")}}

样例运行结果如下图所示:

小结

本节笔者简单讲述了 NAPI 的异步实现方式,下一小节笔者从源码的角度给大家讲解一下 NAPI 的实现原理,敬请期待……

码牛课堂也为了积极培养鸿蒙生态人才,让大家都能学习到鸿蒙开发最新的技术,针对一些在职人员、0基础小白、应届生/计算机专业、鸿蒙爱好者等人群,整理了一套纯血版鸿蒙(HarmonyOS Next)全栈开发技术的学习路线。大家可以进行参考学习:https://qr21.cn/FV7h05

①全方位,更合理的学习路径
路线图包括ArkTS基础语法、鸿蒙应用APP开发、鸿蒙能力集APP开发、次开发多端部署开发、物联网开发等九大模块,六大实战项目贯穿始终,由浅入深,层层递进,深入理解鸿蒙开发原理!

②多层次,更多的鸿蒙原生应用
路线图将包含完全基于鸿蒙内核开发的应用,比如一次开发多端部署、自由流转、元服务、端云一体化等,多方位的学习内容让学生能够高效掌握鸿蒙开发,少走弯路,真正理解并应用鸿蒙的核心技术和理念。

③实战化,更贴合企业需求的技术点
学习路线图中的每一个技术点都能够紧贴企业需求,经过多次真实实践,每一个知识点、每一个项目,都是码牛课堂鸿蒙研发团队精心打磨和深度解析的成果,注重对学生的细致教学,每一步都确保学生能够真正理解和掌握。

为了能让大家更好的学习鸿蒙(HarmonyOS NEXT)开发技术,这边特意整理了《鸿蒙开发学习手册》(共计890页),希望对大家有所帮助:https://qr21.cn/FV7h05

《鸿蒙开发学习手册》:https://qr21.cn/FV7h05

如何快速入门:

  1. 基本概念
  2. 构建第一个ArkTS应用
  3. ……

开发基础知识:https://qr21.cn/FV7h05

  1. 应用基础知识
  2. 配置文件
  3. 应用数据管理
  4. 应用安全管理
  5. 应用隐私保护
  6. 三方应用调用管控机制
  7. 资源分类与访问
  8. 学习ArkTS语言
  9. ……

基于ArkTS 开发:https://qr21.cn/FV7h05

  1. Ability开发
  2. UI开发
  3. 公共事件与通知
  4. 窗口管理
  5. 媒体
  6. 安全
  7. 网络与链接
  8. 电话服务
  9. 数据管理
  10. 后台任务(Background Task)管理
  11. 设备管理
  12. 设备使用信息统计
  13. DFX
  14. 国际化开发
  15. 折叠屏系列
  16. ……

鸿蒙开发面试真题(含参考答案):https://qr21.cn/FV7h05

大厂鸿蒙面试题::https://qr18.cn/F781PH

鸿蒙开发面试大盘集篇(共计319页):https://qr18.cn/F781PH

1.项目开发必备面试题
2.性能优化方向
3.架构方向
4.鸿蒙开发系统底层方向
5.鸿蒙音视频开发方向
6.鸿蒙车载开发方向
7.鸿蒙南向开发方向

这篇关于HarmonyOS ArkUI实战开发-NAPI异步编程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/927889

相关文章

Python并行处理实战之如何使用ProcessPoolExecutor加速计算

《Python并行处理实战之如何使用ProcessPoolExecutor加速计算》Python提供了多种并行处理的方式,其中concurrent.futures模块的ProcessPoolExecu... 目录简介完整代码示例代码解释1. 导入必要的模块2. 定义处理函数3. 主函数4. 生成数字列表5.

Python实例题之pygame开发打飞机游戏实例代码

《Python实例题之pygame开发打飞机游戏实例代码》对于python的学习者,能够写出一个飞机大战的程序代码,是不是感觉到非常的开心,:本文主要介绍Python实例题之pygame开发打飞机... 目录题目pygame-aircraft-game使用 Pygame 开发的打飞机游戏脚本代码解释初始化部

使用Python开发一个现代化屏幕取色器

《使用Python开发一个现代化屏幕取色器》在UI设计、网页开发等场景中,颜色拾取是高频需求,:本文主要介绍如何使用Python开发一个现代化屏幕取色器,有需要的小伙伴可以参考一下... 目录一、项目概述二、核心功能解析2.1 实时颜色追踪2.2 智能颜色显示三、效果展示四、实现步骤详解4.1 环境配置4.

华为鸿蒙HarmonyOS 5.1官宣7月开启升级! 首批支持名单公布

《华为鸿蒙HarmonyOS5.1官宣7月开启升级!首批支持名单公布》在刚刚结束的华为Pura80系列及全场景新品发布会上,除了众多新品的发布,还有一个消息也点燃了所有鸿蒙用户的期待,那就是Ha... 在今日的华为 Pura 80 系列及全场景新品发布会上,华为宣布鸿蒙 HarmonyOS 5.1 将于 7

Python使用smtplib库开发一个邮件自动发送工具

《Python使用smtplib库开发一个邮件自动发送工具》在现代软件开发中,自动化邮件发送是一个非常实用的功能,无论是系统通知、营销邮件、还是日常工作报告,Python的smtplib库都能帮助我们... 目录代码实现与知识点解析1. 导入必要的库2. 配置邮件服务器参数3. 创建邮件发送类4. 实现邮件

基于Python开发一个有趣的工作时长计算器

《基于Python开发一个有趣的工作时长计算器》随着远程办公和弹性工作制的兴起,个人及团队对于工作时长的准确统计需求日益增长,本文将使用Python和PyQt5打造一个工作时长计算器,感兴趣的小伙伴可... 目录概述功能介绍界面展示php软件使用步骤说明代码详解1.窗口初始化与布局2.工作时长计算核心逻辑3

Java Spring 中的监听器Listener详解与实战教程

《JavaSpring中的监听器Listener详解与实战教程》Spring提供了多种监听器机制,可以用于监听应用生命周期、会话生命周期和请求处理过程中的事件,:本文主要介绍JavaSprin... 目录一、监听器的作用1.1 应用生命周期管理1.2 会话管理1.3 请求处理监控二、创建监听器2.1 Ser

Apache 高级配置实战之从连接保持到日志分析的完整指南

《Apache高级配置实战之从连接保持到日志分析的完整指南》本文带你从连接保持优化开始,一路走到访问控制和日志管理,最后用AWStats来分析网站数据,对Apache配置日志分析相关知识感兴趣的朋友... 目录Apache 高级配置实战:从连接保持到日志分析的完整指南前言 一、Apache 连接保持 - 性

python web 开发之Flask中间件与请求处理钩子的最佳实践

《pythonweb开发之Flask中间件与请求处理钩子的最佳实践》Flask作为轻量级Web框架,提供了灵活的请求处理机制,中间件和请求钩子允许开发者在请求处理的不同阶段插入自定义逻辑,实现诸如... 目录Flask中间件与请求处理钩子完全指南1. 引言2. 请求处理生命周期概述3. 请求钩子详解3.1

MQTT SpringBoot整合实战教程

《MQTTSpringBoot整合实战教程》:本文主要介绍MQTTSpringBoot整合实战教程,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录MQTT-SpringBoot创建简单 SpringBoot 项目导入必须依赖增加MQTT相关配置编写