往期鸿蒙5.0全套实战文章必看:(文中附带全栈鸿蒙5.0学习资料)
使用JSVM进行class相关开发
简介
使用JSVM-API接口进行class相关开发,处理JavaScript中的类,例如定义类、构造实例等。
基本概念
在使用JSVM-API接口进行class相关开发时,需要理解以下基本概念:
- 类:类是用于创建对象的模板。它提供了一种封装数据和行为的方式,以便于对数据进行处理和操作。类在JavaScript中是建立在原型(prototype)的基础上的,并且还引入了一些类独有的语法和语义。
- 实例:实例是通过类创建具体的对象。类定义了对象的结构和行为,而实例则是类的具体表现。通过实例化类,我们可以访问类中定义的属性和方法,并且每个实例都具有自己的属性值。
接口说明
接口 | 功能说明 |
---|---|
OH_JSVM_NewInstance | 通过给定的构造函数,构建一个实例。 |
OH_JSVM_GetNewTarget | 获取函数的元属性new.target。 |
OH_JSVM_DefineClass | 用于在JavaScript中定义一个类,并与对应的C类进行封装和交互。它提供了创建类的构造函数、定义属性和方法的能力,以及在C和JavaScript之间进行数据交互的支持。 |
OH_JSVM_Wrap | 在JavaScript对象中封装原生实例。稍后可以使用OH_JSVM_Unwrap()解包原生实例。 |
OH_JSVM_Unwrap | 解包先前封装在JavaScript对象中的原生实例。 |
OH_JSVM_RemoveWrap | 解包先前封装在JavaScript对象中的原生实例并释放封装。 |
使用示例
JSVM-API接口开发流程参考使用JSVM-API实现JS与C/C++语言交互开发流程,本文仅对接口对应C++相关代码进行展示。
OH_JSVM_NewInstance
通过给定的构造函数,构建一个实例。
cpp部分代码
// hello.cpp
#include <string.h>
std::string ToString(JSVM_Env env, JSVM_Value val) {
JSVM_Value jsonString;
JSVM_CALL(OH_JSVM_JsonStringify(env, val, &jsonString));
size_t totalLen = 0;
JSVM_CALL(OH_JSVM_GetValueStringUtf8(env, jsonString, nullptr, 0, &totalLen));
size_t needLen = totalLen + 1;
char* buff = new char[needLen];
std::memset(buff, 0, needLen);
JSVM_CALL(OH_JSVM_GetValueStringUtf8(env, jsonString, buff, needLen, &totalLen));
std::string str(buff);
delete[] buff;
return str;
}
// OH_JSVM_NewInstance的样例方法
static JSVM_Value NewInstance(JSVM_Env env, JSVM_CallbackInfo info) {
// 获取js侧传入的两个参数
size_t argc = 2;
JSVM_Value args[2] = {nullptr};
JSVM_CALL(OH_JSVM_GetCbInfo(env, info, &argc, args, nullptr, nullptr));
JSVM_Value result = nullptr;
// 调用OH_JSVM_NewInstance接口,实例化一个对象,将这个对象返回
JSVM_CALL(OH_JSVM_NewInstance(env, args[0], 1, &args[1], &result));
std::string str = ToString(env, result);
OH_LOG_INFO(LOG_APP, "NewInstance:%{public}s", str.c_str());
return nullptr;
}
// 通过给定的构造函数,构建一个实例。
// NewInstance注册回调
static JSVM_CallbackStruct param[] = {
{.data = nullptr, .callback = NewInstance},
};
static JSVM_CallbackStruct *method = param;
// NewInstance方法别名,供JS调用
static JSVM_PropertyDescriptor descriptor[] = {
{"newInstance", nullptr, method++, nullptr, nullptr, nullptr, JSVM_DEFAULT},
};
样例JS
const char *srcCallNative = R"JS(
function Fruit(name) {
this.name = name;
}
newInstance(Fruit, "apple");
)JS";
执行结果
在LOG中输出下面的结果:
NewInstance:{"name":"apple"}
OH_JSVM_GetNewTarget
用于获取函数的元属性new.target值。在JavaScript中,new.target是一个特殊的元属性,用于检测函数或构造函数是否是通过 'new' 运算符被调用的。
OH_JSVM_DefineClass
用于在JavaScript中定义一个类,并与对应的C类进行封装和交互。它提供了创建类的构造函数、定义属性和方法的能力,以及在C和JavaScript之间进行数据交互的支持。
cpp部分代码
// hello.cpp
#include <string>
JSVM_Value CreateInstance(JSVM_Env env, JSVM_CallbackInfo info) {
JSVM_Value newTarget;
// 获取构造函数的new.target值
JSVM_CALL(OH_JSVM_GetNewTarget(env, info, &newTarget));
OH_LOG_INFO(LOG_APP, "Create Instance");
OH_LOG_INFO(LOG_APP, "NAPI MyObject::New %{public}s", newTarget != nullptr ? "newTarget != nullptr" : "newTarget == nullptr");
JSVM_Value jsObject = nullptr;
JSVM_CALL(OH_JSVM_CreateObject(env, &jsObject));
JSVM_Value jsName = nullptr;
JSVM_CALL(OH_JSVM_CreateStringUtf8(env, "name", JSVM_AUTO_LENGTH, &jsName));
JSVM_Value jsValue = nullptr;
JSVM_CALL(OH_JSVM_CreateStringUtf8(env, "lilei", JSVM_AUTO_LENGTH, &jsValue));
JSVM_CALL(OH_JSVM_SetProperty(env, jsObject, jsName, jsValue));
return jsObject;
}
std::string ToString(JSVM_Env env, JSVM_Value val) {
JSVM_Value jsonString;
JSVM_CALL(OH_JSVM_JsonStringify(env, val, &jsonString));
size_t totalLen = 0;
JSVM_CALL(OH_JSVM_GetValueStringUtf8(env, jsonString, nullptr, 0, &totalLen));
size_t needLen = totalLen + 1;
char* buff = new char[needLen];
std::memset(buff, 0, needLen);
JSVM_CALL(OH_JSVM_GetValueStringUtf8(env, jsonString, buff, needLen, &totalLen));
std::string str(buff);
delete[] buff;
return str;
}
// 封装c++中的自定义数据结构
JSVM_Value DefineClass(JSVM_Env env, JSVM_CallbackInfo info) {
JSVM_CallbackStruct param;
param.data = nullptr;
param.callback = CreateInstance;
JSVM_Value cons;
// 用于在JavaScript中定义一个类
JSVM_CALL(OH_JSVM_DefineClass(env, "MyObject", JSVM_AUTO_LENGTH, ¶m, 0, nullptr, &cons));
JSVM_Value instanceValue = nullptr;
// 作为class的构造函数调用
JSVM_CALL(OH_JSVM_NewInstance(env, cons, 0, nullptr, &instanceValue));
std::string str = ToString(env, instanceValue);
OH_LOG_INFO(LOG_APP, "NewInstance:%{public}s", str.c_str());
// 作为普通的函数调用
JSVM_Value global;
JSVM_CALL(OH_JSVM_GetGlobal(env, &global));
JSVM_Value key;
JSVM_CALL(OH_JSVM_CreateStringUtf8(env, "Constructor", JSVM_AUTO_LENGTH, &key));
JSVM_CALL(OH_JSVM_SetProperty(env, global, key, cons));
JSVM_Value result;
JSVM_CALL(OH_JSVM_CallFunction(env, global, cons, 0, nullptr, &result));
std::string buf = ToString(env, result);
OH_LOG_INFO(LOG_APP, "NewInstance:%{public}s", buf.c_str());
return nullptr;
}
// 注册DefineClass的方法
JSVM_CallbackStruct param[] = {
{.data = nullptr, .callback = DefineClass},
};
static JSVM_CallbackStruct *method = param;
// DefineClass方法别名,供JS调用
static JSVM_PropertyDescriptor descriptor[] = {
{"defineClass", nullptr, method++, nullptr, nullptr, nullptr, JSVM_DEFAULT},
};
样例JS
const char *srcCallNative = R"JS(
defineClass();
)JS";
执行结果
在LOG中输出下面的结果:
Create Instance
NAPI MyObject::New newTarget != nullptr
NewInstance:{"name":"lilei"}
Create Instance
NAPI MyObject::New newTarget == nullptr
NewInstance:{"name":"lilei"}
OH_JSVM_Wrap
在JavaScript对象中封装原生实例。稍后可以使用OH_JSVM_Unwrap()解包原生实例
OH_JSVM_Unwrap
解包先前封装在JavaScript对象中的原生实例
OH_JSVM_RemoveWrap
解包先前封装在JavaScript对象中的原生实例并释放封装
cpp部分代码
// hello.cpp
#include <string>
// OH_JSVM_GetNewTarget、OH_JSVM_DefineClass、OH_JSVM_Wrap、OH_JSVM_Unwrap、OH_JSVM_RemoveWrap的样例方法
// 自定义类结构体Object
struct Object {
std::string name;
int32_t age;
};
// 定义一个回调函数
static void DerefItem(JSVM_Env env, void *data, void *hint) {
OH_LOG_INFO(LOG_APP, "JSVM deref_item");
(void)hint;
}
static JSVM_Value WrapObject(JSVM_Env env, JSVM_CallbackInfo info) {
OH_LOG_INFO(LOG_APP, "JSVM wrap");
Object obj;
// 设置Object属性
obj.name = "lilei";
obj.age = 18;
Object *objPointer = &obj;
// 获取回调信息中的参数数量和将要被封装的值
size_t argc = 1;
JSVM_Value toWrap = nullptr;
JSVM_CALL(OH_JSVM_GetCbInfo(env, info, &argc, &toWrap, nullptr, nullptr));
// OH_JSVM_Wrap将自定义结构Object进行封装
JSVM_CALL(OH_JSVM_Wrap(env, toWrap, reinterpret_cast<void *>(objPointer), DerefItem, NULL, NULL));
Object *data;
// OH_JSVM_UnWrap解包先前封装在JavaScript对象中的原生实例
JSVM_CALL(OH_JSVM_Unwrap(env, toWrap, reinterpret_cast<void **>(&data)));
OH_LOG_INFO(LOG_APP, "JSVM name: %{public}s", data->name.c_str());
OH_LOG_INFO(LOG_APP, "JSVM age: %{public}d", data->age);
return nullptr;
}
static JSVM_Value RemoveWrap(JSVM_Env env, JSVM_CallbackInfo info) {
OH_LOG_INFO(LOG_APP, "JSVM removeWrap");
Object obj;
// 设置Object属性
obj.name = "lilei";
obj.age = 18;
Object *objPointer = &obj;
// 获取回调信息中的参数数量和将要被封装的值
size_t argc = 1;
JSVM_Value toWrap = nullptr;
JSVM_CALL(OH_JSVM_GetCbInfo(env, info, &argc, &toWrap, nullptr, nullptr));
// 将自定义结构Object封装
JSVM_CALL(OH_JSVM_Wrap(env, toWrap, reinterpret_cast<void *>(objPointer), DerefItem, NULL, NULL));
Object *data;
// 解包先前封装的object,并移除封装
JSVM_CALL(OH_JSVM_RemoveWrap(env, toWrap, reinterpret_cast<void **>(&objPointer)));
// 检查是否已被移除
JSVM_Status status = OH_JSVM_Unwrap(env, toWrap, reinterpret_cast<void **>(&data));
if (status != JSVM_OK) {
OH_LOG_INFO(LOG_APP, "JSVM OH_JSVM_RemoveWrap success");
}
return nullptr;
}
// WrapObject、RemoveWrap注册回调
static JSVM_CallbackStruct param[] = {
{.data = nullptr, .callback = WrapObject},
{.data = nullptr, .callback = RemoveWrap},
};
static JSVM_CallbackStruct *method = param;
// WrapObject、RemoveWrap方法别名,供JS调用
static JSVM_PropertyDescriptor descriptor[] = {
{"wrapObject", nullptr, method++, nullptr, nullptr, nullptr, JSVM_DEFAULT},
{"removeWrap", nullptr, method++, nullptr, nullptr, nullptr, JSVM_DEFAULT},
};
样例JS
const char *srcCallNative = R"JS(
class Obj {};
wrapObject(new Obj());
removeWrap(new Obj());
)JS";
执行结果
在LOG中输出下面的结果:
JSVM wrap
JSVM name: lilei
JSVM age: 18
JSVM removeWrap
JSVM OH_JSVM_RemoveWrap success
JSVM deref_item