【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;
}
相关推荐
一粒黑子7 小时前
【实战解析】阿里开源 PageAgent:纯前端 GUI Agent,一行JS让网页支持自然语言操控
前端·javascript·开源
IT枫斗者7 小时前
前端部署后如何判断“页面是不是最新”?一套可落地的版本检测方案(适配 Vite/Vue/React/任意 SPA)
前端·javascript·vue.js·react.js·架构·bug
九转成圣8 小时前
Java 性能优化实战:如何将海量扁平数据高效转化为类目字典树?
java·开发语言·json
Beginner x_u8 小时前
链表专题:JS 实现原理与高频算法题总结
javascript·算法·链表
SmartRadio8 小时前
ESP32-S3 双模式切换实现:兼顾手机_路由器连接与WiFi长距离通信
开发语言·网络·智能手机·esp32·长距离wifi
laowangpython9 小时前
Rust 入门:GitHub 热门内存安全编程语言
开发语言·其他·rust·github
我叫汪枫9 小时前
在后台管理系统中,如何递归和选择保留的思路来过滤菜单
开发语言·javascript·node.js·ecmascript
_.Switch9 小时前
东方财富股票数据JS逆向:secids字段和AES加密实战
开发语言·前端·javascript·网络·爬虫·python·ecmascript
软件技术NINI9 小时前
webkit简介及工作流程
开发语言·前端·javascript·udp·ecmascript·webkit·yarn
Brendan_0019 小时前
JavaScript的Stomp.over
开发语言·javascript·ecmascript