QuickJS 快速入门 (QuickJS QuickStart)
1. QuickJS 快速入门 (QuickJS QuickStart)
1.1. 简介
QuickJS是一个小型的可嵌入Javascript引擎。它支持ES2020规范,包括模块、异步生成器和代理。它还支持数学扩展,比如大整数(BigInt)、大浮点数(BigFloat)和操作符重载。
1.2. 安装
- Linux 直接下载 源码
make && make install
- MacOS X 下 makefile 有 Bug ,可以直接使用 homebrew 安装
brew install quickjs
- 执行
qjs验证安装成功
1.3. 简单使用
1.3.1. 控制台执行
qjs 进入quickjs环境,-h 获取帮助,-q 退出环境。
直接执行js:
console.log(new Date())
输出:Wed Aug 14 2019 23:51:43 GMT+0800
undefined
(function(){ return 1+1;})()
输出:2
1.3.2. js脚本执行
新建一个js脚本,名为hello.js,内容为console.log('hello world !'), 在js目录下执行
qjs hello.js
输出:hello world !
1.3.3. 编译二进制文件
将 quickjs.h、quickjs-libc.h、libquickjs.a 拷贝到js文件同目录下。
qjsc -o hello hello.js
ls
./hello
输出:hello world !
编译出来的可执行文件的大小只有569K(2019-9-18版本为900K),没有任何外部依赖,非常适合嵌入式设备使用。
1.4. 全局对象
scriptArgs输入的命令行参数,第一个参数为脚本的名称。print(...args)、console.log(...args)打印由空格和尾随换行符分隔的参数。
新建js脚本globle_obj.js
(function(){
if(typeof scriptArgs != 'undefined'){
print(scriptArgs);
console.log(scriptArgs[1]);
}
})()
qjs globle_obj.js -a 123 1234
输出:
globle_obj.js,-a,123,1234
-a
1.5. std 模块
std模块为quickjs-libc提供包装器stdlib.h和stdio.h和其他一些实用程序。
std代码示例:
创建文件std_m.js
import * as std from 'std';
var file = std.open('std_open_file.js','w');
file.puts('var file = std.open(\"std_open_file.txt\",\"w\");\n');
file.puts('file.puts(\'std_open_file line1\\n\');\n');
file.puts('file.puts(\'std_open_file line2\\n\');\n');
file.puts('file.close();\n');
file.close();
std.loadScript('std_open_file.js');
var rdfile = std.open("std_open_file.txt","r");
do{
console.log(rdfile.getline());
}while(!rdfile.eof());
rdfile.close();
执行qjs std_m.js ,目录下会生成2个新文件std_open_file.js std_open_file.txt。
控制台输出:
std_open_file line1
std_open_file line2
null
1.6. os 模块
os 模块提供操作系统特定功能:底层文件访问、信号、计时器、异步 I/O。
代码示例:
import * as os from 'os';
os.remove('hello');
os.remove('std_open_file.js');
os.remove('std_open_file.txt');
删除生成的测试文件
1.7. 自定义C模块
ES6模块完全支持。默认名称解析规则如下:
- 模块名称带有前导.或..是相对于当前模块的路径
- 模块名称没有前导.或..是系统模块,例如std或os
- 模块名称以.so结尾,是使用QuickJS C API的原生模块
使用js文件模块和系统模块,参照引用原生js模块和上面的例子即可,这里就不多赘述。
这里着重讲解如何编写自己的原生C模块,并且以导入so文件的方式在js代码中使用。
1.7.1. js数据类型在C中的定义
typedef union JSValueUnion {
int32_t int32; //整数值
double float64; //double值
void *ptr; //QuickJS引用类型的指针
} JSValueUnion; //存放于同一地址,且互斥
typedef struct JSValue {
JSValueUnion u; //存放真实数值或着其指针
int64_t tag; //JSValue类型的标示符(如 undefined 其 tag == JS_TAG_UNDEFINED)
} JSValue;
此结构定义在 quickjs.h 中。
1.7.2. c模块编写
流程如下:
- 自定义原生C函数
- 定义 QuickJS C 函数
- 定义API的函数入口名称及列表
- 定义初始化回调方法,将函数入口列表在模块中暴露
- 定义初始化模块方法,由系统自动调用,且函数名称不可更改
创建编写c_test_m.c文件:
#include "quickjs.h"
#include "stdio.h"
#include "stdlib.h"
#include "string.h"
#define JS_INIT_MODULE js_init_module
#define countof(x) (sizeof(x) / sizeof((x)[0]))
/* 自定义原生C函数 */
static double test_add(int a, double b)
{
return a + b;
}
static char *test_add_str(const char *a, double b)
{
/* 要有足够的空间来容纳要拼接的字符串,否则可能会造成缓冲溢出的错误情况 */
char instr[64];
sprintf(instr, "%.2f", b);
char *dest = malloc(128);
memset(dest, 0, 128);
strcpy(dest, a);
char *retdest = strcat(dest, instr);
return dest;
}
/*
定义 QuickJS C 函数
*ctx : 运行时上下文
this_val : this对象
argc : 入参个数
*argv : 入参列表
*/
static JSValue js_test_add(JSContext *ctx, JSValueConst this_val,
int argc, JSValueConst *argv)
{
int a;
double b;
if (JS_ToInt32(ctx, &a, argv[0]))
return JS_EXCEPTION;
if (JS_ToFloat64(ctx, &b, argv[1]))
return JS_EXCEPTION;
printf("argc = %d \n", argc);
printf("a = %d \n", a);
printf("b = %lf \n", b);
printf("argv[1].u.float64 = %lf \n", argv[1].u.float64);
return JS_NewFloat64(ctx, test_add(a, b));
}
static JSValue js_test_add_str(JSContext *ctx, JSValueConst this_val,
int argc, JSValueConst *argv)
{
if (!JS_IsString(argv[0]))
{
return JS_EXCEPTION;
}
double d;
if (JS_ToFloat64(ctx, &d, argv[1]))
return JS_EXCEPTION;
const char *jscstr = JS_ToCString(ctx, argv[0]);
printf("JS_ToCString(ctx, argv[0]) = %s \n", jscstr);
printf("argv[1].u.float64 = %lf \n", argv[1].u.float64);
char *jsret = test_add_str(jscstr, d);
return JS_NewString(ctx, jsret);
}
/* 定义API的函数入口名称及列表 */
static const JSCFunctionListEntry js_test_funcs[] = {
/* JS_CFUNC_DEF(函数入口名称,入参个数,QuickJS C 函数) */
JS_CFUNC_DEF("testAdd", 2, js_test_add),
JS_CFUNC_DEF("testAddStr", 2, js_test_add_str),
};
/* 定义初始化回调方法(由系统调用,入参格式固定),将函数入口列表 在模块中暴露 */
static int js_test_init(JSContext *ctx, JSModuleDef *m)
{
return JS_SetModuleExportList(ctx, m, js_test_funcs,
countof(js_test_funcs));
}
/* 定义初始化模块方法,由系统自动调用,且函数名称不可更改 */
JSModuleDef *JS_INIT_MODULE(JSContext *ctx, const char *module_name)
{
JSModuleDef *m;
m = JS_NewCModule(ctx, module_name, js_test_init);
if (!m)
return NULL;
JS_AddModuleExportList(ctx, m, js_test_funcs, countof(js_test_funcs));
return m;
}
将 quickjs.h、quickjs-libc.h、libquickjs.a 拷贝到当前工程目录下。
执行命令
gcc c_test_m.c libquickjs.a -fPIC -shared -o libtest.so
生成libtest.so文件。
1.7.3. 使用.so模块
创建js文件 c_test_m.js
import { testAdd , testAddStr} from 'libtest.so'
console.log('\n')
console.log(`testAdd: ${testAdd(1, 0.5)}`)
console.log('\n')
console.log(`testAddStr: ${testAddStr('Pi equal to about ', 3.14159)}`)
console.log('\n')
qjs c_test_m.js
输出:
argc = 2
a = 1
b = 0.500000
argv[1].u.float64 = 0.500000
testAdd: 1.5
JS_ToCString(ctx, argv[0]) = Pi equal to about
argv[1].u.float64 = 3.141590
testAddStr: Pi equal to about 3.14
项目地址
QuickJS 快速入门 (QuickJS QuickStart)的更多相关文章
- springboot笔记02——快速入门quickstart
前言 学习一个新的框架,往往会用一个quickstart快速入门,这次就写一下springboot的quickstart程序. 开发环境 JDK 1.8 Springboot 2.1.6 Maven ...
- 【番外篇】ASP.NET MVC快速入门之免费jQuery控件库(MVC5+EF6)
目录 [第一篇]ASP.NET MVC快速入门之数据库操作(MVC5+EF6) [第二篇]ASP.NET MVC快速入门之数据注解(MVC5+EF6) [第三篇]ASP.NET MVC快速入门之安全策 ...
- 【第一篇】ASP.NET MVC快速入门之数据库操作(MVC5+EF6)
目录 [第一篇]ASP.NET MVC快速入门之数据库操作(MVC5+EF6) [第二篇]ASP.NET MVC快速入门之数据注解(MVC5+EF6) [第三篇]ASP.NET MVC快速入门之安全策 ...
- Spring IoC反转控制的快速入门
* 下载Spring最新开发包 * 复制Spring开发jar包到工程 * 理解IoC反转控制和DI依赖注入 * 编写Spring核心配置文件 * 在程序中读取Spring配置文件,通过Spring框 ...
- Gradle用户指南(章9:Groovy快速入门)
Gradle用户指南(章9:Groovy快速入门) 你可以使用groovy插件来构建groovy项目.这个插件继承了java插件的功能,且扩展了groovy编译.你的项目可以包含groovy代码.ja ...
- TestDriven.NET – 快速入门
TestDriven.NET – 快速入门[译文] 介绍 这部分将提供一个快速的入门向导,在vs.NET的任何一个版本上面使用TestDriven.NET TDD(测试驱动开发)在你写你的代码之前,写 ...
- SuperSocket快速入门(三):实现你的AppServer和AppSession
什么是AppSession? AppSession 代表一个和客户端的逻辑连接,基于连接的操作应该定义于在该类之中.你可以用该类的实例发送数据到客户端,接收客户端发送的数据或者关闭连接.同时可以保存客 ...
- SuperSocket快速入门(二):启动程序以及相关的配置
如何快速启动第一个程序 既然是快速入门,所以,对于太深奥的知识点将不做讲解,会在后续的高级应用章节中,会对SS进行拆解.所有的实例90%都是来自SS的实例,外加本人的注释进行讲解. 一般应用而言,你只 ...
- 【转】Robot Framework 快速入门
目录 介绍 概述 安装 运行demo 介绍样例应用程序 测试用例 第一个测试用例 高级别测试用例 数据驱动测试用例 关键词keywords 内置关键词 库关键词 用户定义关键词 变量 定义变量 使用变 ...
随机推荐
- openresty开发系列27--openresty中封装redis操作
openresty开发系列27--openresty中封装redis操作 在关于web+lua+openresty开发中,项目中会大量操作redis, 重复创建连接-->数据操作-->关闭 ...
- 【转载】 十图详解tensorflow数据读取机制(附代码)
原文地址: https://zhuanlan.zhihu.com/p/27238630 何之源 深度学习(Deep Learning) 话题的优秀回答者 --------------- ...
- How to Plan and Configure YARN and MapReduce 2 in HDP 2.0
As part of HDP 2.0 Beta, YARN takes the resource management capabilities that were in MapReduce and ...
- 【计算机视觉】OpenCV篇(9) - 轮廓(寻找/绘制轮廓)
什么是轮廓? 轮廓是一系列相连的点组成的曲线,代表了物体的基本外形. 轮廓与边缘好像挺像的? 是的,确实挺像,那么区别是什么呢?简而言之,轮廓是连续的,而边缘并不全都连续(见下图示例).其实边缘主要是 ...
- webStorm 如何修改angular中html的字体颜色
请问一下:在html中怎么修改这个的颜色呢? (这个是ng中html的语法主要双括号里面的紫色太暗了,有时候都看不清是什么东东了) 修改方案:(说明:第三步需要把最后的选择框去掉,默认是选中的, ...
- zepto获取select元素当前选中option
//jQuery $("#select_city").find("option:selected"); //zepto $("#select_city ...
- SpringBoot学习笔记:自定义拦截器
SpringBoot学习笔记:自定义拦截器 快速开始 拦截器类似于过滤器,但是拦截器提供更精细的的控制能力,它可以在一个请求过程中的两个节点进行拦截: 在请求发送到Controller之前 在响应发送 ...
- Django 之上下文处理器和中间件
一.上下文处理器 上下文处理器是可以返回一些数据,在全局模板中都可以使用.比如登录后的用户信息,在很多页面中都需要使用,那么我们可以放在上下文处理器中,就没有必要在每个视图函数中都返回这个对象. 在s ...
- Django之用户注册
用户注册需要提交的信息包括: 用户名 邮箱 密码 确认密码 验证码 这里选择form表单提交信息,注册页面的响应函数就要分条件执行,get请求时要展示注册页面,post请求时要接收用户提交的信息,对信 ...
- BGP 实验
一.环境准备 1. 软件:GNS3 2. 路由:c7200 二.实验操作 实验要求: 1. 掌握 BGP 的基本配置方法. 2. 掌握如何查看 BGP 的各种配置信息. 3. 掌握基于回环口的 BGP ...