Flutter 三方库 js_wrapping 的鸿蒙化适配指南 - 实现 Dart 与 JavaScript 的无缝对象包装、支持强类型回调与属性映射

Flutter 三方库 js_wrapping 的鸿蒙化适配指南 - 实现 Dart 与 JavaScript 的无缝对象包装、支持强类型回调与属性映射

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net

Flutter 三方库 js_wrapping 的鸿蒙化适配指南 - 实现 Dart 与 JavaScript 的无缝对象包装、支持强类型回调与属性映射

前言

在进行 Flutter for OpenHarmony 的 Web 混合开发时,频繁地在 Dart 层与底层 JavaScript 环境进行数据交互是不可避免的。虽然官方提供了基本的 dart:js,但在处理复杂的 JS 对象和回调时,代码往往会变得杂乱无章。js_wrapping 提供了一个更优雅的、类型安全的包装层。本文将指导大家如何在鸿蒙端利用该库提升 JS 互操作的开发体验。

一、原理解析 / 概念介绍

1.1 基础原理

js_wrapping 是对 dart:js 的高级封装。它通过映射机制,将 JS 对象包装成 Dart 类。当你访问 Dart 类的属性时,它在底层自动调用 JS API 进行存取。

graph LR A["Dart Class (Wrapper)"] -- "属性访问/方法调用" --> B["js_wrapping 逻辑层"] B -- "Interop" --> C["JavaScript Context (WebView/Browser)"] C -- "返回值/回调" --> B B -- "自动解包" --> A 

1.2 核心优势

  • 类型安全:将动态的 JS 对象映射为具体的 Dart 类型,减少运行时错误。
  • 自动包装与解包:无需手动编写 JsObject.fromBrowserCast 等繁琐逻辑。
  • 回调支持:完美处理 JS 回调函数与 Dart Function 之间的转换。
  • 内存优化:提供生命周期管理,避免 JS 对象与 Dart 包装类之间的悬挂引用。

二、鸿蒙基础指导

2.1 适配情况

  1. 是否原生支持? 是,主要用于 Web 宿主环境(如鸿蒙 Web 组件)。
  2. 是否鸿蒙官方支持? 社区跨平台方案。
  3. 是否需要安装额外的 package? 不需要,但需确保在 Web 模式下编译。

2.2 适配代码

pubspec.yaml 中配置:

dependencies: js_wrapping: ^0.7.0 

在鸿蒙端开发时,如果你使用 Flutter WebView 处理 JS 交互,该库可以极大地简化注入脚本后的对象管理。

三、核心 API / 组件详解

3.1 核心方法

类/方法说明
JsSerializable抽象基类,用于定义映射关系
JsObjectWrapper包装原生 JS 对象的容器
getProperty(name)安全获取 JS 对象属性
invoke(method, args)执行 JS 方法并获取结果

3.2 基础配置

import 'package:js_wrapping/js_wrapping.dart'; import 'dart:js' as js; // 定义一个 JS 对象的 Dart 映射 class UserJS extends TypedJsObject { UserJS(js.JsObject proxy) : super(proxy); String get name => $get('name'); void setName(String val) => $set('name', val); } 

四、典型应用场景

4.1 映射 JS 全局变量

在鸿蒙 Web 场景中,访问注入的全局配置文件。

void loadHmosConfig() { final jsConfig = js.context['hmos_config']; if (jsConfig != null) { final wrapper = JsObjectWrapper(jsConfig); String version = wrapper.getProperty('system_version'); print('当前鸿蒙渲染引擎版本: $version'); } } 

4.2 处理复杂的 JS 回调

void registerHmosListener() { final handler = js.allowInterop((msg) { print('收到来自鸿蒙 Web 层消息: $msg'); }); // 使用 js_wrapping 将 handler 封装并传递给 JS 侧 js.context.callMethod('registerFlutterHandler', [handler]); } 

五、OpenHarmony 平台适配挑战

5.1 渲染引擎差异

鸿蒙的 Web 组件核心与标准 Chrome 略有不同。在利用 js_wrapping 进行深层次属性映射时,务必注意某些 ES6+ 特性在旧版鸿蒙系统 Webview 上的兼容性。建议尽量使用基础的对象操作。

5.2 安全性约束

鸿蒙对 Web 环境与原生环境的通信有严格的 CSP(内容安全策略)。在使用 js_wrapping 动态执行 JS 代码时,确保相关脚本来源合法,否则 invoke 方法可能会报权限拒绝错误。

六、综合实战演示

import 'package:flutter/material.dart'; import 'package:js_wrapping/js_wrapping.dart'; class JsInteropDemo extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar(title: Text('JsWrapping 鸿蒙实战')), body: Center( child: ElevatedButton( onPressed: () { // 假设外部 JS 环境有一个 window.nativeBridge 对象 try { final bridge = JsObjectWrapper(js.context['nativeBridge']); final result = bridge.invoke('getDeviceInfo', ['battery']); print('鸿蒙设备电池信息: $result'); } catch (e) { print('当前环境不支持 JS 互操作'); } }, child: Text('调用 JS 桥接方法'), ), ), ); } } 

七、总结

js_wrapping 是鸿蒙混合开发中的良伴。它通过结构化的方式治愈了“JS 互操作混乱症”,让 Dart 代码保持整洁的同时,依然能高效通过鸿蒙 Web 通信。对于需要构建大规模 JS 交互逻辑的应用,该库必不可少。

Read more

Python爬虫实战:手把手教你用 Python 爬取网易新闻每日热文,小白也能轻松上手

Python爬虫实战:手把手教你用 Python 爬取网易新闻每日热文,小白也能轻松上手

Python爬虫实战:手把手教你用 Python 爬取网易新闻每日热文,小白也能轻松上手 Python爬虫实战:手把手教你用 Python 爬取网易新闻每日热文,小白也能轻松上手,该教程详细讲解如何用 Python 爬取网易新闻每日热文,先介绍爬虫 “请求 - 解析 - 提取 - 保存” 原理及 requests、BeautifulSoup4 等必备库的安装,再逐段解析完整代码:从设置请求头模拟浏览器、发送 HTTP 请求获取网页数据,到通过关键词匹配和类名匹配双方案提取 “今日推荐” 热文,还包含数据去重、Excel 保存(按日期命名)及异常处理与调试模块。同时给出实操步骤,解答爬取不到数据、Excel 保存失败等常见问题,强调爬虫伦理与法律规范,最后提供定时爬取、多频道爬取等功能扩展建议,帮助小白轻松上手打造自动新闻采集工具。 前言     Python作为一门简洁、易读、功能强大的编程语言,

By Ne0inhk
双指针问题5(c++)

双指针问题5(c++)

概念 双指针,顾名思义,就是用两个指针解决问题。 有些问题用单指针会出现超时等问题,这时就需要用到双指针 双指针由两个指针组成,一般是左右指针,或前后指针 通过两个指针配合变化,用更短的时间高效解决问题 题目 (续上一篇,如需了解上一篇题目,请移步主页观看) 合并有序数组 #include <bits/stdc++.h> #define ll long long using namespace std; int la,lb,lab; int a[20010],b[10010]; int main() { cin>>la; for(int i = 1;i<

By Ne0inhk

面向 C++ 的现代 CMake 第二版(三)

原文:zh.annas-archive.org/md5/4abd6886e8722cebdc63cd42f86a9282 译者:飞龙 协议:CC BY-NC-SA 4.0 第八章:链接可执行文件和库 你可能会认为,一旦我们成功地将源代码编译成二进制文件,我们作为构建工程师的角色就完成了。然而,事实并非完全如此。尽管二进制文件确实包含了 CPU 执行所需的所有代码,但这些代码可能会以复杂的方式分布在多个文件中。我们不希望 CPU 在不同的文件中寻找单独的代码片段。相反,我们的目标是将这些分散的单元合并为一个文件。为了实现这一目标,我们使用了一个称为链接的过程。 快速观察可以发现,CMake 有很少的链接命令,其中target_link_libraries()是主要命令。那么,为什么要专门用一整章来讲解这个命令呢?不幸的是,计算机科学几乎没有什么事情是简单的,链接也不例外:为了获得正确的结果,我们需要了解整个过程——我们需要知道链接器是如何工作的,并掌握基本知识。我们将讨论目标文件的内部结构,重定位和引用解析机制的工作原理,以及它们的用途。

By Ne0inhk

绿盟校招C++研发工程师一面复盘

绿盟校招C++研发工程师一面复盘 1. 进程间通信方式中的共享内存为何比套接字快呢?共享内存的核心是让多个进程映射同一块物理内存到各自的虚拟地址空间,通信过程完全绕开内核的中转干预。而套接字会触发用户态到内核态的切换。用户态与内核态的切换,需要保存、恢复进程上下文,这是操作系统的核心开销之一,套接字的每一次send()/recv()都是系统调用,都要经历用户态->内核态->用户态的切换。共享内存无协议开销,数据是直接写入内存的原始字节流,无需封装任何协议头、无需计算校验和、无需处理拥塞控制,CPU开销极低。 2. 线上CPU飙升如何排查? * 首先确认是哪个进程占用CPU过高,登录服务器利用top命令查看各个进程的资源占用情况 * 确认CPU利用率很高的进程PID,假设1234为某个进程,则通过top -Hp 1234查看具体的线程 * 假设得到的线程ID是5678,再将线程ID转化为十六进制,得到十六进制的tid162e,此时利用jstack 1234 | grep 162e -A 100查看具体的栈信息。jstack命令用于生成当前时刻的线程快照。线程

By Ne0inhk