管理O
发布于

【转载】aardio与不同编程语言之间相互调用

作者:Jacen He
链接:https://www.zhihu.com/question/353354060/answer/2561667239
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

不同编程语言之间其实可以相互调用,aardio 就支持混入很多不同的编程语言,生成的软件体积也很小。不同编程语言相互调用的代码也比较简单。下面看示例代码(下面的代码基本都可以复制粘贴,直接运行,不需要任何复杂的步骤):

aardio 直接调用系统 API 函数( 支持回调、相互调用 ):

var ret,point = ::User32.GetCursorPos({
  int x;
  int y;
})

aardio / JavaScript 相互调用( 支持 ES6 ):

import web.script;
var vm = web.script("ES6")

//导出 aardio 函数到 JavaScript
vm.external = {
	log = function(...){  
		console.log(...)
	} 
}

vm.script = /*****
function TestFunction(a,b) { 
	return a + b;
}
*****/

var ret = vm.script.TestFunction(2,3);

在 aardio 中调用 Python( Python 调用 aardio 同样简单,参考 aardio 自带范例),简单得就像在 Python 中使用 Python:

import console;
import py3; 

//导入 Python 模块。
var itertools = py3.import("itertools")

//调用 Python 函数,支持纯 aardio 类型参数
var permutations = itertools.permutations({"a","b","c"});

//调用 Python 内置函数 list
var pyList = py3.list(permutations);

//遍历 Python 对象成员
for( item in pyList.each() ){
  console.log(item); //像 aardio 对象一样使用 Python 对象
}

console.pause();

aardio 直接调用 .Net / C# 组件( 支持相互调用,支持委托这些)

import win.ui; 
var winform = win.form(text="DataGridView")

import System.Data;
var tab = System.Data.DataTable("DT"); 
tab.Columns.Add("名称"); 
tab.Rows.Add({"WangWu"}); 

import System.Windows.Forms;
var grid = System.Windows.Forms.CreateEmbed("DataGridView",winform); 
grid.ColumnHeadersHeightSizeMode = 2; 
grid.DataSource = System.Data.DataView(tab);
   
winform.show();
win.loopMessage();

aardio 直接调用 Java ( 也支持相互调用 )

import java; 
var jvm = java(); 

//加载Java类对象
HelloworldApp = jvm.import("aardio.sample.HelloworldApp");

//也可以如下自内存或文件直接加载类,
HelloworldApp = jvm.import("aardio.sample.HelloworldApp",$"\java\aardio\sample\HelloworldApp.class");

//用 Java 类创建 Java 对象
var helloworld = HelloworldApp();

//直接调用 Java 对象的方法 
var result = helloworld.test(3); 

用 aardio 编译 C 语言代码生成 DLL 执行文件,再调用 DLL 中的 C 函数:

import tcc;  
//编译 DLL
tcc.build( "/start.dll" ).code = /***
#include <windows.h> 
__declspec(dllexport) int Add( int a,int b ) 
{     
  return a + b;
} 
***/

//加载 DLL
var dll = raw.loadDll( "/start.dll",,"cdecl" );
//调用 C函数
var result = dll.Add(12,3);

在 aardio 中嵌入并调用批处理:

import console
import process.batch;

//批处理 for 遍历并拆分字符串
var bat = process.batch(`
@echo off 
for %%i in (abc,def,xyz) do echo %%i
`)
console.log(bat.read(-1))
 
console.pause()

aardio 调用 Rust 语言解析 TOML:

import console; 
import string.toml;

var str = string.toml.stringify({abc=123,d={1,2,3}});
console.log( str );

import process.code;
process.code("~\lib\string\toml\.res");
console.pause(true);

在 aardio 里嵌入 PHP,以下短短几句代码,包含了 HTTP 服务器,PHP服务端,嵌入的浏览器组件:

import win.ui;
/*DSG{{*/
var winform = win.form(text="Hello World / PHP_CGI 服务器")
/*}}*/

var code = /*
<html>
<head> 
<meta charset="utf-8">
<title>PHP 测试</title>
</head>
<body>
<?php echo '<p>Hello World / PHP_CGI 服务器</p>'; ?>
</body>
</html>
*/
string.save("/test.php",code);

import php.simpleHttpServer;
var url = php.simpleHttpServer.startUrl("/test.php");

import web.form;
var wb = web.form(winform);
wb.go(url);

winform.show();
win.loopMessage();

aardio 调用 PowerShell,并且在 PowerShell 中调用 aardio。这甚至都不用带上体积较大的 System.Management.Automation.dll,一个轻巧的 EXE 就可以搞定一切,向下兼容到 PowerShell 2.0 :

import dotNet.ps;
 
var pScript = /*

# 声明 PowerShell 参数
param($win)  

# 修改 aardio 对象属性
$win.title = "PowerShell + aardio";

# 调用 aardio 对象函数
$win.msgbox("这是 PowerShell 调用 aardio 打开的对话框。")  
*/

import win;
dotNet.ps(pScript,{ 
  win = win; //# 将 aardio 对象作为参数传给 PowerShell
});

aardio 执行 Ruby 语言代码 :

import win.ui;
/*DSG{{*/
var winform = win.form(text="执行Ruby代码")
winform.add(
edit={cls="edit";left=26;top=16;right=737;bottom=435;multiline=1;z=1}
)
/*}}*/

import process.ruby;
var out = process.ruby.exec("puts '测试UTF-8'")
winform.edit.print(out);

var out = process.ruby.eval(`[1, 2, { name: "tanaka", age: 19 }]`)
winform.edit.print(out);

winform.show();
win.loopMessage();

aardio 调用 Node.js :

import console;  
import nodeJs;

var js = /******

console.log(process.argv);

var startEnviron = require('startEnviron');
console.log(startEnviron.dest);

******/

//自动分析 JS 代码中的 require 语句并安装依赖模块
nodeJs.requireByJs(js);

//把对象传给 node.js,在 JS 代码中用 require('startEnviron') 获取。
nodeJs.startEnviron({
    src:"传个字符串",dest:{test:"嵌套的对象表,传给node.js都没问题",number:123, arr:{1,2,3} }
})

//执行JS,这里指定的启动参数在 JS 代码中可用 process.argv 获取。
var prcs = nodeJs.exec(js,"--args1=1","--args2=1");
prcs.logResponse();

console.pause(true);

aardio 调用 Fortran ( DLL 源码在 aardio 范例里有 ) :

import console

//加载 DLL , DLL 路径前加 $ 实现内存加载 DLL(发布后不需要外部 DLL 文件)
var dll = raw.loadDll($"/fortran.dll",,"cdecl");

//不声明直接调用,结构体默认传址,这不用改什么。
var c = dll.__test_MOD_addbypoint({
	int x = 22;
	int y = 3;
})
console.log(c);


//可以先声明一下,参数类型加上&声明为按引用传址(指针)
var add = dll.api("__test_MOD_add","int(int &a,int &b)")
var c = add(33,2); //Fortran 的数值参数默认都是传址(传指针)

//不声明直接调用可以用结构体取代指针
var c = dll.__test_MOD_add({int a=33},{int b=2});

//用 raw.int 创建传址数值也可以
var c = dll.__test_MOD_add(raw.int(33,true),raw.int(2,true));

//参数声明为传值时调用更简单,不声明调用时数值默认为 int 类型
var c = dll.__test_MOD_addbyval(33,2,raw.double(123));
console.log(c);

//字符串
var str = "hello"; //只读字符串,改用 raw.buffer 创建可读写字节数组
dll.__test_MOD_hello(str,#str); //注意到字符串长度传过去

console.pause(true);

aardio 调用 C++ :

import vc6;
import console; 
console.open();

var vc = vc6( "/" )  

//输入C++源码
vc.cpp = /****** 
    #include <windows.h> 
      
    struct TestInfo{
        int x;
        int y;
        BYTE name[256];
    };
    
    class CTestObject
    {
    public: 
        //注意函数声明前加上 virtual 以支持 aardio 中的 raw.interface
        virtual	void getName(char *buffer,int len); 
        virtual	void getInfo(TestInfo *pInfo);
    };
    
    void CTestObject::getName(char *buffer,int len){
        strcpy(buffer,"测试");
    }
    
    void CTestObject::getInfo(TestInfo *pInfo){
        pInfo->x = 1;
        pInfo->y = 2;
        strcpy((char *)pInfo->name,"测试");
    }
    
    extern "C" __declspec(dllexport) CTestObject* __cdecl CreateTestObject() { 
        return new CTestObject();
    }
    
    extern "C" __declspec(dllexport) void __cdecl DeleteTestObject( CTestObject* pTest) {
        delete pTest;
    }
******/

//编译生成DLL 
vc.exec(
    'cl *.cpp'
    ,'/W3' /*警告等级*/
    ,'/MD' /*使用多线程动态运行库*/
    ,'/O2 /Ot /EHsc' /*代码优化选项*/
    ,'/D "WIN32" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL"' /*定义常数和宏*/
    ,'/I"./INCLUDE"'/*指定头文件目录*/
    ,'kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib' /*导入库*/
    ,'/link /SUBSYSTEM:WINDOWS /MACHINE:X86' /*后面是链接参数 */
    ,'/out:test.dll'/*输出文件名*/
    ,'/dll' /*输出DLL*/ 
    ,'/LIBPATH:".\LIB" /LIBPATH:".\LIB2"' /*指定库目录*/
)

var dll = raw.loadDll("/test.dll",,"cdecl");

import raw.interface;
class testObject{
	ctor(){
		//创建 C++ 对象,并获取指针,注意这里使用了 P 尾标获取指针。
		var pTest = dll.CreateTestObjectP();
		
		//C++ 对象指针转换为 aardio 对象。
		this = ..raw.interface( pTest,"
    		void getName(string &buffer,int len);
    		void getInfo(struct &pInfo); 
    		","thiscall" //注意调用约定为thiscall
		)
		
		//添加析构函数	
		..table.gc(this,"delete")	
	};
	delete = function(){
		if(!owner.deleted){
			dll.DeleteTestObject( owner );
			owner.deleted = true;
		} 
	};
}

//创建对象
var obj = testObject();

//调用 C++ 函数
var name = obj.getName(25,25);
console.log(name);

//调用 C++ 函数
var info = obj.getInfo({ int x;int y;BYTE name[256]})
console.log( info.name  );

console.pause();

aardio 调用 FreeBASIC ( DLL 源码在 aardio 范例里有 ) :

//加载DLL,DLL路径前面加上$表示把DLL嵌入到程序中并通过内存加载
var dll = raw.loadDll( 
	$"\basic.dll",,"cdecl" //注意参数里指定使用 cdecl 调用约定。
);

//定义结构体,当然也可以先声明一个 class 来创建实例。
var info = {
    int x;
    INT y;
}

// 然后直接调用 API
var ret = dll.msgboxW(123,456,"测试一下好用不好用",info);


//最后打印结构体看一下值
import console;
console.log(ret);

console.dumpJson(info);
console.pause();

aardio 调用 Delphi ( DLL 源码在 aardio 范例里有) :

import win.ui;

//内存加载 DLL
var delphiDll = raw.loadDll($"\Project1.dll");
class win.ui.ctrl.delphiForm{
	ctor(parent,tParam){
		this.hwnd = delphiDll.CreateForm(parent.hwnd);
	};
	@..win.ui.ctrl.metaProperty()
}
/*DSG{{*/
var winform = win.form(text="用 Delphi 语言为 aardio 编写控件";right=507;bottom=423;bgcolor=11842740)
winform.add(
custom={cls="delphiForm";text="嵌入 Delphi 控件";left=17;top=28;right=490;bottom=211;db=1;dl=1;dr=1;dt=1;z=1};
edit={cls="edit";text="请先用 Delphi 打开此目录下的 DLL 源码工程编译生成 \Project1.dll";left=16;top=228;right=489;bottom=398;edge=1;multiline=1;z=2}
)
/*}}*/

import web.json;
winform.onTest = function(delphiStructParam){ 
	winform.edit.print("Delphi 调用了aardio 函数,参数如下:");
	winform.edit.print(delphiStructParam);
	delphiStructParam.x = 90;

	//可选返回修改后的结构体
	return delphiStructParam;
}
winform.edit.text = "";

winform.show();
win.loopMessage();

aardio 调用 R 语言:

import console; 
import process.r;

//执行 R 代码,支持 aardio 模板语法
process.r.code = /*
write("<?="这是 aardio 代码"?>",file=".data.txt");
*/

//执行 R 代码,支持 aardio 模板语法
var out = process.r.loadcode(`write("<?="这是 aardio 代码"?>",file=".data.txt");`)
 
//执行纯 R 代码,参数 @1 可以指定 R代码或 R 文件。
var out = process.r.exec(`
args=commandArgs(T);
write(args[1],file=".data.txt");

# list 有点像 aardio 中的表(table),可以包含各种数据类型, 
a <- list(hello = 1, world = "字符串" ) # <- 相当于 aardio 中的等号,  R的等号一般用于分隔键值对
print ( a[["world"]] ); # aardio 里的直接下标也是这么写
print ( a$world ); # 相当于 aardio 里的  a.world
print ( a[1] ); # 这个返回的是键值对 hello = 1,不像 aadio 中 a[1] 与 a.hello 是指向不同的元素。
print ( mode(a[1]) ); # 数据类型还是显示为 list

b <- TRUE #布尔值必须全大写
print( b ) 

# 向量
a = c(10, 20, 30, 40, 50)
print( a[1] ) #起始下标为 1 ,这跟 aardio 一样
print( a[1:4] ) # 取出第 1 项到第 4 项

# 定义函数,与 aardio 语法类似
new.function <- function(a,b,c) {
   result <- a * b + c # 类似 aardio 中的 return a * b + c #
   print(result) # 指定返回值以后,还能继续执行后面的代码,不像 aardio 函数 return 后面的代码被忽略。
}

print( new.function(2,3,1) )
`,"测试一下"); //可以添加不定个数的启动参数
console.log( out );
console.more(,true);

console.showLoading(" 正在安装 rjson 包");
process.r.require("rjson","https://mirrors.ustc.edu.cn/CRAN/");//不会重复安装

var out  = process.r.exec( `
library("rjson") # 载入 rjson 包

args <- commandArgs(T);
tab <- fromJSON(args[1], simplify=FALSE);

#不要用 print ,cat 不会加一堆不必要的东西
cat( toJSON(tab) )
`, {
  name1 = "测试一下,传对象给 R 语言";
  name2 = "这是一个 aardio 对象"
})

console.dump(out);

var rCode = /*
testabc <- function(a,b,c) {
   result <- a * b + c # 类似 aardio 中的 return a * b + c #
   print(result) # 指定返回值以后,还能继续执行后面的代码,不像 aardio 函数 return 后面的代码被忽略。
}
*/

//启动 R 
var r = process.r.startRpc(rCode);
 
//调用 R 函数
var ret  = r.testabc(2,3,1)

//打印 R 函数返回值
if(ret[["result"]]){
	console.log("R 函数返回值",ret[["result"]])
} 
console.pause(true);

 

aardio 调用 Julia :

import console;
import julia;

//调用 Julia 函数
var ret = julia.sqrt(2);
console.log(ret);

//导入 Julia 模块
julia.using("Base64");
var data = julia.Base64.base64encode("测试一下");
console.log( data );

//转换 Julia 数据类型
var buf = julia.value.build(raw.buffer("abc"));
console.log(julia.typeof(buf));

//执行 Julia 代码并获取返回值
var refs = julia.eval("refs = IdDict(");

//查看 Julia 代码错误
console.log(julia.lasterr());

console.pause();

aardio 调用 Nim 语言:

import console;
 
var nimCode = /*
 {.pragma: rtl, exportc, dynlib, cdecl.}
 
import md5
 
# Nim 双引号中的字符串,相当于 aardio 中用单引号包含的转义字符串
# aardio 中双引号包含的字符串,相当于 Nim 中的原始字符串: r"原始字符串" 
# Nim 与 aardio 都是 UTF-8 编码,aardio 的文本字符串在 Nim 中的类型为 cstring
# Nim 中 string 可以隐式转换为 cstring, cstring 加上 $ 转为 string 类型
proc build*(str: cstring, num: ptr[cint]): cstring {.rtl.} =
    num[] =  num[] * 2
    result =  md5.getMD5($str)
*/
string.save("/test.nim",nimCode )

import process.nim; 
process.nim("c --app:lib -d:release -r test.nim")

//支持改为 $"/test.dll" 内存加载 DLL,
//但这时候 test.dll 还未生成,所以示例里没有加 $
//用cdecl 调用约定的好处是:导出函数名直接可用,不会被加上修饰名
var test = raw.loadDll("/test.dll",,"cdecl")

//nim 与 aardio 的字符串都是 UTF-8 编码,UTF-8 真是到处通行,非常方便省了很多事
var build  = test.api("build","str(str,int& num)" )

//一般C语言不能这么直接返回字符串(要考虑谁释放内存)。
//但是 nim 可以投机取巧一下,nim 会自动回收内存,而这时候还来不及回收。
var str,num = build("测试abc",9)
console.log(str,num)

//用 aardio 算出 MD5 对比一下,结果一模一样
import crypt;
console.log(crypt.md5("测试abc",false))
console.pause();

aardio 调用 V 语言:

import console; 
import process.v;

console.open();

//V语言不支持中文路径,所以工程目录路径不要包含任何中文
string.save("/hello.v","
struct Point {
pub mut: //声明下面的字段公开、可变
	x int
	y int 
} 

[export: 'add'] //一定要用这句指定DLL导出函数名
pub fn add(a int,b int,mut pt &Point) int {
   pt.x = a+b
   return a+b
}" )

/*
V语言是翻译成C语言然后生成DLL,生成的DLL依赖 VC 运行库,
试了换成调用TCC编译,10KB的DLL增大到 400KB,并且运行崩溃。
*/
process.v.shared("hello.v").waitOne();

//V生成的DLL建议至少在 WIN10 上用,需要VC++2017运行库 
import sys.vc14;
sys.vc14.require(); //检测并自动安装 VC++ 运行库

//导入DLL,注意要指定 cdecl 调用约定
var dll = raw.loadDll("/hello.dll",,"cdecl")

//调用 V 函数( V是翻译为C语言,所以参考C语言的规则)
var n,pt = dll.add(12,3,{int x=1;int y =2});

//输出结果
console.log(n);
console.dumpJson(pt);

console.pause(true);

img

浏览 (522)
点赞
收藏
全部评论 1
管理O
管理O
感觉有个bug:markdown 代码块的序号宽度不统一,见图片:![ ](https://mlogclub.oss-cn-hongkong.aliyuncs.com/images/2022/07/06/778dfeed193291377c55cf5e6cb47733.png)
点赞
评论