SWIG 多语言接口变换 【转】
一. SWIG 是Simple Wrapper and Interface Generator的缩写,是一个帮助使用C或者C++编写的软件创建其他编语言的API的工具。例如,我想要为一个C++编写的程序创建.NET API,一般情况下我必须使用托管C++(Managed C++)去编写大量的代码才能生成它的.NET API。有了SWIG,这个机械的工作将变得非常简单。你只须要使用一个接口文件告诉SWIG要为那些类创建.NET API,SWIG就会自动帮你生成它的.NET API。
当 然,SWIG不仅仅支持创建.NET API。最新版本的SWIG支持常用脚本语言Perl、PHP、Python、Tcl、Ruby和非脚本语言C#, Common Lisp (CLISP, Allegro CL, CFFI, UFFI), Java, Modula-3, OCAML以及R,甚至是编译器或者汇编的计划应用(Guile, MzScheme, Chicken)。也就是说可以用Swig将C++接口封装为java等语言可以调用的形式。
二. 环境搭建
1. 下载swigwin,在E:/lib目录下解压,即把swig安装到E:/lib/swig目录中。
2. 新建一个Win32 Console Application,注意在Application Settings中选择DLL以及空项目。完成后将工程的配置改成release
3. 工具 —> 选项 —> 项目和解决方案 —> VC++目录 —> 添加E:/lib/swigwin。即把swig添加到VC的可执行目录。
----------------若封装成Python可用的API,执行如下步骤----------------------------
4. 下载python2.5,安装到D:/Program Files目录,并在环境变量PATH中加入D:/Program Files/python2.5。
5. 把D:/Program Files/python2.5/include加入VC的Include路径,将D:/Program Files/python2.5/libs加入VC的Library路径。
-----------------若封装成Java可用的API,执行如下步骤--------------------------------
6. 下载JDK,安装到D:/Program Files目录,并在环境变量PATH中加入D:/Program Files/Java/jdk1.6.0_10/bin(如果之前安装过Oracle,要将D:/Program Files/Java/jdk1.6.0_10/bin放在Oracle/jre/1.3.1/bin前面,否则在编译的时候默认会选择oracle的 jdk)
7. 把D:/Program Files/Java/jdk1.6.0_10/include/win32 和 D:/Program Files/Java/jdk1.6.0_10/include 加入到加入VC的Include路径。
三. 接口文件
要在C/C++工程中创建***.i 的接口文件,告诉SWIG要为那些类的那些方法创建API。
接口文件注解:
1. 模块名由指定的%module来给出(或者用-module命令行选项).这段指示性文字必须写在文件的头部并且在使用时将这个模块名作为扩展模块对象来 使用(此外,这个模块名经常在目标语言中被定义成一个命名空间来使用)。如果模块名在命令行已经被给出了,系统将不考虑由%module标示的模块名了。
对于python:module的名字指定了生成文件xxx.py的xxx名字,
对于java:module的名字指定了生成文件xxx.java的xxx名字
2. 所有在%{...%}块内的东西将被简单作为结果逐字拷贝到SWIG创建的wrapper(包装)文件中。这部分大部分被用来包括头文件和生成 wrapper代码需要的其它声明。这里很重要的强调一点因为你在一个SWIG的输入文件中包含了一个声明,这个声明并不自动显示在生成的wrapper 代码中,因此你需要确信你确实把正确的头文件在%{ ... %}部分中。这里应该指出SWIG不解析和解释附在%{ ... %}部分的文字。SWIG的%{...%}内的语法和语义很类似于输入文件中的声明部分 。
3. 如果打算为类中所有方法创建API,那么有一个非常简单的办法,在接口文件的类声明部分使用%include标记。SWIG将对%include所指定的文件进行语法分析,类中所有公有方法(Public Method)都将在API中暴露。
/* SwigTest.i */
%module SwigTest
%{
#include "SwigTest.h"
%}
%include “SwigTest.h” //不要和#用混
-------------------------- swig库模块访问部分标准C++库包括STL的方法-------------------
SWIG对于一些语言模块的支持使较全面的但是对很少用到的库则支持的很少。
下面就是表示了C++类和支持的C++库 以及SWIG接口文件的对应表
C++ class C++ Library file SWIG Interface library file
std::deque deque std_deque.i
std::list list std_list.i
std::map map std_map.i
std::pair utility std_pair.i
std::set set std_set.i
std::string string std_string.i
std::vector vector std_vector.i
因此,当C / C++代码中用到这些库时,可将swig对应的接口文件添加到自己的接口文
件中。如:%include "std_string.i",
库文件完全识别C++的命名空间。如果你输出std::string 或 将它重命名为另一种类型。请确认你将此重命名声明包含到了你的接口文件中。例如:
%module example
%include "std_string.i"
using namespace std;
typedef std::string String;
...
void foo(string s, const String &t); // std_string typemaps still applied
当
封装java调用的api且传递的参数中含有中文时,由于c++中的String是使用单字节编码,而java中String是使用Unicode编码,
所以为了传递时不出现乱码,可以包含%include
"std_wstring.i",因为wstring使用的是wchar_t类型,这是宽字符,用于满足非ASCII字符的要求,例如Unicode编
码。
当封装python调用的api且传递的参数中含有中文时,就不用考虑这个编码问题,可以直接使用c++中的String。
四. 编译模块
1.写接口执行命令
有
了接口文件以后要对接口文件进行编译,右键点击接口文件,修改它的属性,使用自定义编译工具(Custom Build
Tool),命令行(Command Line)内容为swig.exe -c++ -python SwigTest.i
,输出(Outputs)为$(InputName)_wrapper.cpp;
echo JAVA_INCLUDE: %JAVA_INCLUDE%
echo JAVA_BIN: %JAVA_BIN%
echo on
swig.exe -c++ -python SwigTest.i
命令行参数说明:
①
为了编译java
或python模块,必须包含它们的include和bin目录,可以在PATH中设置JAVA_INCLUDE、JAVA_BIN、
PYTHON_INCLUDE、PYTHON_BIN等,其中include可以在工程属性中设置,如果工程属性中已经设置好,此处不用再包含头两句。
②“ swig.exe ”表示调用你安装的swig的可执行文件,之前已经将其目录加入到VC的可执行目录中,所以此处就可以不用再写路径,否则要找到swig路径才行。
③ “-c++” 表示要封装C++代码(不写默认是封装C代码),
④“-python”表示要封装成Python接口(Swig还可以封装成Java、Ruby等接口),
⑤ “$(InputName)_wrapper.cpp”表示指定要生成的C++代码文件的名字。
swig的命令还有其他的一些,目前用到的如 –package命令,用法可以为:
swig -java -package com.swig -outdir com/swig example.i
在生成java的api时,这个命令使生成的java文件包含在某个包中。
2.编译(右键点击接口文件—>编译)
对于生成python可用的API:执行上述命令会生成两个新的文件,一个是SwigTest.py,一个是SwigTest_wrapper.cpp。
对于生成java可用的API:执行上述命令会生成Java类文件:SwigTestJNI.java,SwigTest.java
和c文件SwigTest_wrapper.cpp。
3.将xxx_wrapper.cpp加入到工程
4.修改工程属性:
-----------------------封装java可用的api---------------------
LinkeràGeneralàOutput File改成SwigTest.dll
Build EventsàPost-Build EventàCommand Line改成:
echo on
"%JAVA_BIN%/javac" *.java
这个命令作用是编译完后调用javac将当前编译路径下的所有.java文件编译成.class二进制文件。
-----------------------封装python可用的api---------------------
LinkeràGeneralàOutput File改成_SwigTest.pyd(注意一定要有下划线)
还有一些配置在VC2005里是默认的,但在VC2003里不是的,请引起注意,包括
① C/C++àCode GenerationàRuntime Library要选择Multi-threaded DLL (/MD)
② C/C++àLanguageàTreat wchar_t as Built-in Type要选择yes(如果转换不涉及wchar等,则无需处理这一项)
③ C/C++àLanguageàEnable Run-Time Type Info要选择yes
5.生成工程
五.利用生成的api就可以在对应的脚本语言(java、python等)中使用啦。
windows环境2.0.7版:http://prdownloads.sourceforge.net/swig/swigwin-2.0.7.zip
SWIG 多语言接口变换 【转】的更多相关文章
- 使用SWIG将C++接口转换成Java接口
PS:此文章仅作为个人记录使用,代码属于私密,故无法公开: 以C++类classifier为例,文件保存于百度网盘 https://pan.baidu.com/s/1c2AwhaS(需密码) 系统:U ...
- Swift中对C语言接口缓存的使用以及数组、字符串转为指针类型的方法
由于Swift编程语言属于上层编程语言,而Swift中由于为了低层的高性能计算接口,所以往往需要C语言中的指针类型,由此,在Swift编程语言刚诞生的时候就有了UnsafePointer与Unsafe ...
- C语言接口与实现实例
一个模块有两部分组成:接口和实现.接口指明模块要做什么,它声明了使用该模块的代码可用的标识符.类型和例程,实现指明模块是如何完成其接口声明的目标的,一个给定的模块通常只有一个接口,但是可能会有许多种实 ...
- [转]SQLITE3 C语言接口 API 函数简介
SQLITE3 C语言接口 API 函数简介 说明:本说明文档属作者从接触 SQLite 开始认识的 API 函数的使用方法, 由本人翻译, 不断更新. /* 2012-05-25 */ int sq ...
- 基于Oracle OCI的数据访问C语言接口ORADBI .
基于Oracle OCI的数据访问C语言接口ORADBI cheungmine@gmail.com Mar. 22, 2008 ORADBI是我在Oracle OCI(Oracle 调用接口)基础 ...
- 18 A GIF decoder: an exercise in Go interfaces 一个GIF解码器:go语言接口训练
A GIF decoder: an exercise in Go interfaces 一个GIF解码器:go语言接口训练 25 May 2011 Introduction At the Googl ...
- opencv的C语言接口和C++接口差别(入门篇)
opencv是一个开源的图像处理库,最经典的1.0版本号提供的接口都是C语言接口. 后来的opencv2.x版本号保留了C语言接口,可是提供了C++接口,当中的C语言接口仅仅是为了向后兼容,而C++接 ...
- GO语言学习(十八)Go 语言接口
Go 语言接口 Go 语言提供了另外一种数据类型即接口,它把所有的具有共性的方法定义在一起,任何其他类型只要实现了这些方法就是实现了这个接口. 实例 /* 定义接口 */ type interface ...
- C语言学习书籍推荐《C语言接口与实现:创建可重用软件的技术》下载
<C语言接口与实现:创建可重用软件的技术>概念清晰.实例详尽,是一本有关设计.实现和有效使用C语言库函数,掌握创建可重用C语言软件模块技术的参考指南.书中提供了大量实例,重在阐述如何用一种 ...
随机推荐
- C# 在子窗体调用父窗体的值(转)
1.在父窗体中 public delegate void SetVisiableHandler();//定义委托类型 注意此委托定义在namespace下一级,在form类外 private void ...
- GeoIP2 数据库更新地址
GeoIP2 数据库更新地址 数据库文件下载网页地址 http://dev.maxmind.com/geoip/geoip2/geolite2/ http://geolite.maxmind.com/ ...
- 回文后缀(suffix)
回文后缀(suffix) 题目描述 给定字符集大小 SS ,问有多少个长度为 NN 的字符串不存在长度 >1>1 的回文后缀. 答案对 MM 取模. 输入格式 第一行两个正整数 n, kn ...
- [解决方案]IIS7.5 报错:无法启动计算机“."上的服务W3SVC
报错场景: 在云服务器上,正常使用着,突然今天一打开网站就都用不了了,上去服务器一看,IIS中站点被停止了,我还怀疑是回收的问题,结果一直启动无果,我打算重启来解决这个问题,重启后发现所有站点都变成停 ...
- OpenNI2安装
1.Openni2:从官网下载linux用zip压缩包,解压再解压2.终端转到解压目录下,找到install.sh文件,执行$sudo ./install.sh 3.执行后,生成OpenNIDevEn ...
- vue项目--favicon设置以及动态修改favicon
最近写公司项目时,动态更新favicon 动态更新之前需要有一个默认的favicon. 目前vue-cli搭建的vue项目里面已经有了一个static文件夹,存放静态文件. favicon图片放到该文 ...
- js对象和jq对象互相转换
1.DOM 对象转成 jQuery 对象 var v = document.getElementById("v"); //DOM对象 var $v = $(v); //jQuery ...
- bzoj4897 [Thu Summer Camp2016]成绩单
传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4897 [题解] 第一次看这题想的是f[l,r]的区间dp发现仅记录这两个好像不能转移啊 会出 ...
- 汕头市队赛 SRM 08 B
B-3 SRM 08 描述 给长度为 n 的数列 A 和长度为 m 的数列 B,问有多少长度为 m 的数列 C 满足 输入格式 第一行俩整数 n 和 m 第二行 n 个整数 ,表示数列 A 第三行 m ...
- log4net配置,正在用
<?xml version="1.0" encoding="utf-8" ?> <log4net> <appender name= ...