python 调用JS 之 py_mini_racer 初体验

前言

之前苦于 python 使用 execjs 调用JS里的函数时,没办法动态缓存JS里的环境。然后当时的思路是将动态变化的环境提取出来,然后在写进JS代码里进行持续调用。 但是这样做实在是不够优雅!

正文

就在昨天,群里的泰迪佬告诉了我这个好用的python库, py_mini_racer。可能就会有朋友问我,为什么不用js2py这个库,虽然说这个库是能做到缓存JS动态变化的环境,但是他目前不怎么支持es6语法,尤其是我们使用浏览器沙箱时,需要执行非常庞大的代码,所以无法进行使用。
我们写一段简单的JS代码进行测试。

a = 0
function b(){
    a += 1
    return a;
}

然后写一份python demo代码,分别使用 execjs 和 py_mini_racer进行调用JS的 b 函数,打印结果a。

with open('test.js', 'r', encoding='utf-8')as f:
    jsCode = f.read()

import execjs
from py_mini_racer import MiniRacer

ctx = MiniRacer()

execjsCode = execjs.compile(jsCode)
print("execjs 第一次运行b函数=>", execjsCode.call("b"))
print("execjs 第二次运行b函数=>", execjsCode.call("b"))
print("execjs 第三次运行b函数=>", execjsCode.call("b"))

ctx.eval(jsCode)
print("py_mini_racer 第一次运行b函数=>", ctx.call("b"))
print("py_mini_racer 第二次运行b函数=>", ctx.call("b"))
print("py_mini_racer 第三次运行b函数=>", ctx.call("b"))

最后附上运行结果:

execjs 第一次运行b函数=> 1
execjs 第二次运行b函数=> 1
execjs 第三次运行b函数=> 1
py_mini_racer 第一次运行b函数=> 1
py_mini_racer 第二次运行b函数=> 2
py_mini_racer 第三次运行b函数=> 3

总结

JS里的 参数a是全局变量,execjs每次调用 b 函数时,都会重新运行一整个JS代码,导致a的值永远是从0开始的,所以调用b函数后,返回了1。而我们需要的就是每次调用 b 函数后,a的值就一直递增+1并且保存在环境中。py_mini_racer完美的解决了这个痛点,并且执行效率非常快。 朋友伙计们如果用这个库踩到了什么坑,欢迎一起研究,我也是刚用,目前一切正常。最后附上安装命令。

pip install py_mini_racer
python Python 3.12.9 (main, Feb 4 2025, 22:30:28) [Clang 18.0.3 (https://android.googlesource.com/toolchain/llvm-project d8003a456 on linux Type "help", "copyright", "credits" or "license" for more information. >>> # 在导入akshare前设置环境变量 >>> import os >>> os.environ["PYTHON_MINI_RACER_DISABLE"] = "1" # 禁用JavaScript引擎 >>> >>> import akshare as ak # 此时应跳过相关模块加载 Traceback (most recent call last): File "<stdin>", line 1, in <module> File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/akshare/__init__.py", line 4642, in <module> from akshare.stock_feature.stock_a_pe_and_pb import ( File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/akshare/stock_feature/stock_a_pe_and_pb.py", line 320, in <module> js_functions = py_mini_racer.MiniRacer() ^^^^^^^^^^^^^^^^^^^^^^^^^ File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/py_mini_racer/py_mini_racer.py", line 178, in __init__ self.__class__.ext = _build_ext_handle() ^^^^^^^^^^^^^^^^^^^ File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/py_mini_racer/py_mini_racer.py", line 125, in _build_ext_handle raise RuntimeError("Native library not available at {}".format(EXTENSION_PATH)) RuntimeError: Native library not available at /data/data/com.termux/files/usr/lib/python3.12/site-packages/py_mini_racer/libmini_racer.glibc.so >>> # 已验证可用的替代接口 >>> data = ak.stock_zh_a_spot() # 实时行情(无JS依赖 ) Exception ignored in: <function MiniRacer.__del__ at 0x72b2696520> Traceback (most recent call last): File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/py_mini_racer/py_mini_racer.py", line 315, in __del__ self.ext.mr_free_context(getattr(self, "ctx", None)) ^^^^^^^^^^^^^^^^^^^^^^^^ AttributeError: 'NoneType' object has no attribute 'mr_free_context' Traceback (most recent call last): File "<stdin>", line 1, in <module> NameError: name 'ak' is not defined >>> print(data[data['代码'] == '000001']) Traceback (most recent call last): File "<stdin>", line 1, in <module> NameError: name 'data' is not defined >>> >>> # 历史数据(推荐方式) >>> hfq_data = ak.stock_zh_a_daily(symbol="sh600000", adjust="hfq") # 复权数据 Traceback (most recent call last): File "<stdin>", line 1, in <module> NameError: name 'ak' is not defined
03-16
~ $ python Python 3.12.9 (main, Feb 4 2025, 22:30:28) [Clang 18.0.3 (https://android.googlesource.com/toolchain/llvm-project d8003a456 on linux Type "help", "copyright", "credits" or "license" for more information. >>> import akshare as ak Traceback (most recent call last): File "<stdin>", line 1, in <module> File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/akshare/__init__.py", line 4642, in <module> from akshare.stock_feature.stock_a_pe_and_pb import ( File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/akshare/stock_feature/stock_a_pe_and_pb.py", line 320, in <module> js_functions = py_mini_racer.MiniRacer() ^^^^^^^^^^^^^^^^^^^^^^^^^ File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/py_mini_racer/py_mini_racer.py", line 178, in __init__ self.__class__.ext = _build_ext_handle() ^^^^^^^^^^^^^^^^^^^ File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/py_mini_racer/py_mini_racer.py", line 125, in _build_ext_handle raise RuntimeError("Native library not available at {}".format(EXTENSION_PATH)) RuntimeError: Native library not available at /data/data/com.termux/files/usr/lib/python3.12/site-packages/py_mini_racer/libmini_racer.glibc.so >>> >>> # 使用无JavaScript依赖的接口 >>> data = ak.stock_zh_a_spot() # 实时行情接口 Exception ignored in: <function MiniRacer.__del__ at 0x72eb46a520> Traceback (most recent call last): File "/data/data/com.termux/files/usr/lib/python3.12/site-packages/py_mini_racer/py_mini_racer.py", line 315, in __del__ self.ext.mr_free_context(getattr(self, "ctx", None)) ^^^^^^^^^^^^^^^^^^^^^^^^ AttributeError: 'NoneType' object has no attribute 'mr_free_context' Traceback (most recent call last): File "<stdin>", line 1, in <module> NameError: name 'ak' is not defined >>> print(data[data['代码'] == '000001']) # 筛选上证 指数 Traceback (most recent call last): File "<stdin>", line 1, in <module> NameError: name 'data' is not defined >>> >>> # 历史数据替代方案 >>> data = ak.stock_zh_a_daily(symbol="sh000001", adjust="hfq")
03-16
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值