【js 引擎】如何使用 quickjs 中异常处理功能

开发 js 运行时,要在 c/c++ 代码中,通过调用 quickjs 的 api 来实现 native 层的异常抛出。本文讲解一下如何在quickjs中实现这个操作。

要讲清楚这个操作,需要知道 quickjs 中表示异常有两种值。

  1. 一种表示发生了异常,叫他 exception,表示为 JS_EXCEPTION
  2. 另一种表示异常具体是什么,叫他 Error,表示为 各种类型的错误。相关 API 如下
c++ 复制代码
JSValue JS_Throw(JSContext *ctx, JSValue obj);
JSValue JS_GetException(JSContext *ctx);
JS_BOOL JS_IsError(JSContext *ctx, JSValueConst val);
JS_BOOL JS_IsException(JSValueConst v)
void JS_ResetUncatchableError(JSContext *ctx);
JSValue JS_NewError(JSContext *ctx);

JSValue __js_printf_like(2, 3) JS_ThrowSyntaxError(JSContext *ctx, const char *fmt, ...);
JSValue __js_printf_like(2, 3) JS_ThrowTypeError(JSContext *ctx, const char *fmt, ...);
JSValue __js_printf_like(2, 3) JS_ThrowReferenceError(JSContext *ctx, const char *fmt, ...);
JSValue __js_printf_like(2, 3) JS_ThrowRangeError(JSContext *ctx, const char *fmt, ...);
JSValue __js_printf_like(2, 3) JS_ThrowInternalError(JSContext *ctx, const char *fmt, ...);
JSValue JS_ThrowOutOfMemory(JSContext *ctx);

抛出异常

  1. 错误发生时,我们通常返回 Execption,并将 Error 放置在 context 上。具体做法如下
c 复制代码
// 抛出异常
void js_cfunc(JSContext *ctx, JSValueConst this_val, int argc, JSValueConst *argv) {
	JSValue ret = JS_UNDEFINED;
	if (/*error occur*/) {
		JSValue err = JS_NewError(ctx);
		/*set err info, 
		ref: JS_AddIntrinsicBasicObjects 
			 JS_ThrowSyntaxError
		*/
		JS_Throw(ctx, err);
		return JS_EXCEPTION;
	}
	return ret;
}
  1. 便利函数,语法糖
    一些常见的错误,创建异常并抛出的动作可以合并成 JS_ThrowSyntaxError 等便利函数,见上。
c 复制代码
void js_cfunc(JSContext *ctx, JSValueConst this_val, int argc, JSValueConst *argv) {
	JSValue ret = JS_UNDEFINED;
	if (/*error occur*/) {
		return JS_ThrowSyntaxError(ctx, "error info", ...);
	}
	return ret;
}

处理异常

异常发生时,我们通常获取 Error,并处理错误

c 复制代码
// 处理异常
void js_cfunc(JSContext *ctx, JSValueConst this_val, int argc, JSValueConst *argv) {
	JSValue ret = JS_Call(ctx, ....);
	if (JS_IsException(ret)) {
		JSValue err = JS_GetException(ctx);
		/**
			处理异常,常见手段是
			1. 打印异常栈和异常信息:ref:函数 js_std_dump_error / eval_buf
			2. 也可以直接不处理,异常会自动传播到上层函数。其中原理,看这块代码是可以完全搞清楚的。
		*/
	}
	return ret;
}
相关推荐
艾小码3 分钟前
手把手教你实现一个EventEmitter,彻底告别复杂事件管理!
前端·javascript·node.js
TeleostNaCl3 分钟前
如何安装 Google 通用的驱动以便使用 ADB 和 Fastboot 调试(Bootloader)设备
android·经验分享·adb·android studio·android-studio·android runtime
上位机付工25 分钟前
C#与倍福TwinCAT3进行ADS通信
开发语言·c#
月阳羊30 分钟前
【硬件-笔试面试题-69】硬件/电子工程师,笔试面试题(知识点:电机驱动电路的反馈电路)
java·经验分享·嵌入式硬件·面试
励志不掉头发的内向程序员35 分钟前
STL库——二叉搜索树
开发语言·c++·学习
至此流年莫相忘1 小时前
设计模式:模板方法模式
java·开发语言·设计模式
土了个豆子的1 小时前
02.继承MonoBehaviour的单例模式基类
开发语言·visualstudio·单例模式·c#·里氏替换原则
qq_172805592 小时前
Go 自建库的使用教程与测试
开发语言·后端·golang
久绊A2 小时前
Hydra-SSH 破解安全防范
开发语言·php
阿昭L2 小时前
c++中获取随机数
开发语言·c++