【开发者笔记】java 利用jna调用c#的dll
一、需求阐述:
如果我们的项目利用c#开发,到了开发后期需要和java组进行合作,其中有一部分业务逻辑利用c#已经code completed,那么我们可能会考虑用java来调用现成的c#dll实现需求。前几天工作上正好遇到这样一个问题,于是记下开发过程。
当然这只是个假设,具体情况具体分析,个人认为重构代码才是王道……
二、原理说明:
其实具体原理我也没弄太明白,我就根据自己的理解来说吧,抛砖引玉。
因为c#代码是托管到.net平台上的,所以java不能直接调用c#代码,于是引入C++中间件,c++项目可以设置项目为clr公共运行时,从而通过引用的方式调用c#相应方法。而jna是可以直接调用c++生成的dll的,于是大致流程就走通了。c++调用写好的c#dll,java再调用c++生成的dll中间件,大致流程就是这样了,不过其中有很多坑,下面我会细说。
三、运行平台:
系统:Windows 10 x64
开发工具:Visual Studio 2015/2017(我笔记本和公司电脑安装不同版本,我都有实现过) MyEclipse2014
SDK:jdk-x86、jdk-x64 (dll分为x86和x64平台,和jdk的版本要对应,同一台电脑装两个版本的jdk比较烦,我采用的是系统配置jdk32位调试32位dll,然后myeclipse自带64位jdk调试64位dll)
四、准备工作:
1、首先准备上述运行平台,建议选择和系统位数一致的jdk(安装vs、myeclipse或eclipse或sts);
2、下载jna.jar :JNA下载(下载jna-4.4.0.jar 和 jna-platform-4.4.0)
五、开始CODE
5.1 生成c#DLL
5.1.1 以管理员方式启动vs(项目涉及到注册com组件,必须以管理员启动才能完成),新建c#项目
  
5.1.2 设置c#项目
首先,右键刚刚新建的Invoke项目,点击属性。

继续设置项目属性。

记得保存。
然后新建需要被调用的CSharp类代码。这里我们新建一些简单的方法,为了演示效果我们分别对int、string、bool进行操作。如图:

然后右键项目,点击生成。

第一步,完成,干得漂亮。
5.2 生成c++中间件
5.2.1 新建c++项目并设置属性
      


项目新建成功,右键项目,选择属性。
  

5.2.2 书写c++代码
添加cpp文件
      
      
编辑cpp文件
/**********************************
2017-9-5 21:02:51
声明需要被java调用的方法,该方法和java接口内部方法保持一致
预处理语句目的是暴露函数供外部调用。
************************/
#ifdef MYLIBAPI
#else
#define MYLIBAPI extern "C" __declspec(dllimport)
#endif MYLIBAPI int add(int a, int b); //添加函数声明
MYLIBAPI char* getString(char* str);
MYLIBAPI int reverse(int flag); using namespace System;
using namespace Invoke;
//using namespace System::Runtime::InteropServices::m int add(int a, int b)
{
Method ^method = gcnew Method();
int result = method->add(a, b);
return result;
} char* getString(char* str)
{
String ^ paraStr = gcnew String(str);
Method ^method = gcnew Method();
String ^resultString = method->getString(paraStr);
char* result = (char*)(void*)System::Runtime::InteropServices::Marshal::StringToCoTaskMemAnsi(resultString);
return result;
}
int reverse(int flag)
{
Method ^method = gcnew Method();
int result = method->reverse(flag);
return result;
}
好了,c++和c#全部工作完成,右键生成。

复制下dll生成文件全名,一会儿java里面用。
六、编写java代码
6.1 新建java project ,注意选择和dll平台一致的jdk。然后将之前下载的两个jna的jar加载到项目里面,如图:

6.2 开始写java 代码
package com.dyi.test; import com.sun.jna.Library;
import com.sun.jna.Native; /**
* 需要引入jna-4.4.0.jar 和 jna-platform-4.4.0
* 包下载地址:https://github.com/java-native-access/jna
* @author stagebo
*
*/
public class InvokeTest {
/**
* 调用示例
* @param args
* @throws Exception
*/
public static void main(String[] args) throws Exception {
System.out.println(System.getProperty("java.version"));//输出当前jdk版本号
System.out.println(System.getProperty("sun.arch.data.model"));//输出当前jdk所用平台 CLibrary1 clib = CLibrary1.INSTANCE;
System.out.println("测试返回结果:"+clib.add(13, 13));
System.out.println("测试返回结果:"+clib.getString("this is java param."));
System.out.println("测试返回结果:"+clib.reverse(true)); }
}
/**
* 必要接口,必须包含INSTANCE实例和需要调用的方法声明。
* @author stagebo
*
*/
interface CLibrary1 extends Library {
CLibrary1 INSTANCE = (CLibrary1) Native.
loadLibrary("D:\\vs workplace\\java调用CSDLL示例\\x64\\Release\\CppDll",
CLibrary1.class); /*需要调用的方法,方法名与c++方法名相同*/
int add(int a,int b);
String getString(String a);
boolean reverse(boolean flag); }
然后我们运行:

哦豁,报错了【无效的内存访问】,因为java找到了c++dll,但是没找到c#的dll,其中c++dll我们写的全路径名,可以直接找到,那么c#的dll怎么找呢。答案是将c#的dll复制到jdk的bin目录下,jvm就能找到了。
如图我们将Invoke.dll复制到jdk的bin目录下:
  
然后再运行:
    
nice!对于常用类型中的int、string、boolean都可以顺利传递了,事实上其他类型的也可以实现,只要遵循不同语言之间的类型对应关系就可以了,具体的类型关系可以百度。
七、注意事项
7.1 java报错:Exception in thread "main" java.lang.Error: Invalid memory access
可能原因:
1、c#dll没有复制到jdk的bin目录;
2、java和c++之间数据类型不对应;
7.1.2 java报错:Exception in thread "main" java.lang.UnsatisfiedLinkError: Unable to load library 'D:\vs workplace\X86InvokeTest\Release\X86CPPDlls': Native library (win32-x86/D:\vs workplace\X86InvokeTest\Release\X86CPPDlls.dll) not found in resource path ([file:/G:/My%20Eclipse%20workplace/InvokeCSharpX86Test/bin/, file:/G:/My%20Eclipse%20workplace/InvokeCSharpX86Test/Lib/jna-4.4.0.jar, file:/G:/My%20Eclipse%20workplace/InvokeCSharpX86Test/Lib/jna-platform-4.4.0.jar])
可能原因:
1、c++dll路径不正确,建议做test时用绝对路径,这样你在c++项目编译过后不用拷贝便可以在java程序里面直接调用;
2、jdk的平台和c++项目的平台不匹配,jdk是32位那么c++dll一定也是32位的,64位也同样;
7.1.3 windows64位下编译的32位dll测试失败,暂时不清楚是不是64位系统的原因,由于我电脑虚拟机没有装上,就没有去32位系统上测试了。
================================2018-1-3 17:15:54 更新========================================================
1、提供给测试项目开源地址:
2、怎么确定c#的dll是不是成功复制到jdk的bin目录呢?换言之怎么确定自己的bin目录在哪里呢?可以在eclipse中运行的时候通过控制台看到。

2018-12-25 10:36:44 补充。发现问问题的网友有点多,一一回复不来,大家进群聊吧。

【开发者笔记】java 利用jna调用c#的dll的更多相关文章
- java 利用jna调用c#的dll
		
一.需求阐述: 如果我们的项目利用c#开发,到了开发后期需要和java组进行合作,其中有一部分业务逻辑利用c#已经code completed,那么我们可能会考虑用java来调用现成的c#dll实现需 ...
 - Atitit.java jna  调用c  c++ dll的原理与实践  总结  v2  q27
		
Atitit.java jna 调用c c++ dll的原理与实践 总结 v2 q27 1. Jna简单介绍1 2. Jna范例halo owrld1 3. Jna概念2 3.1. (1)需 ...
 - java 用JNA调用dll 参考文档
		
1 Java调用C语言动态库(JNA方式):回调函数.结构体数组传参.结构体数组返回 2jna结构体数组 JNA结构体数组 3JNA调用C语言动态链接库学习实践总结 4 Java 通过 JNA 调 ...
 - java利用JDK调用并执行js源码
		
前言: 不同开发语言之间具有通用性,更具有协作调用的可能.有时候对于一些场景会有调用js的需求,因此下面展示了一个java利用自身JDK调用js函数的demo,供感兴趣的朋友参考. js函数文件 ex ...
 - java利用反射调用类的某个方法
		
java利用反射机制 可以动态调用某个类的某个方法,在 扩展系统功能或提供对外接口时经常用的到. 代码如下: 打印类Print.java package com.test.reflct; /** * ...
 - Java利用JNI调用C/C++写成的DLL
		
前言 由于学期作业的要求,笔者需要开发一个语音识别系统.出于对Java的热爱,笔者非常想用Java来写上层程序(前台+数据库的三层),又要用到Microsoft Speech SDK,所以在这些条件下 ...
 - Java通过jna调用c++动态库
		
1 环境准备 操作系统:windows 10,x64 jna,jna-4.4.0.jar c++开发环境,vc2013 java开发环境,eclipse,jdk8 2 dll开发 通过vc2013创建 ...
 - Java使用JNA调用DLL库
		
Java调用DLL方法有三种,JNI.JNA.JNative, 本文为JNA JNA为使用jna.jar包,下载地址:http://www.java2s.com/Code/Jar/j/Download ...
 - java通过jna调用so
		
c++: FirstEliteValidate.h #pragma once void __attribute__((constructor)) startup();void __attribute_ ...
 
随机推荐
- linux使用ip能ping通,但使用域名却不能访问的解决方法
			
使用命令:yum -y update进行更新测试,一般测试结果为couldn't resolve hostmirrors.aliyun.com 解决方式参考博客couldn't resolve hos ...
 - C++标准库之mutex
			
互斥锁有可重入.不可重入之分.C++标准库中用mutex表示不可重入的互斥锁,用recursive_mutex表示可重入的互斥锁.为这两个类增加根据时间来阻塞线程的能力,就又有了两个新的互斥锁:tim ...
 - iScroll框架的使用和修改
			
iScroll 的诞生是因为手机 Webkit 浏览器(iPhone.iPod.Android 和 Pre)本身没有为固定宽度和高度的元素提供滚动内容的方法.这导致了很多网页使用 position:a ...
 - WORD里怎样能做到局部“分栏”就是一页里有的分有的不分
			
选中你要分的部分再分栏如果不想分的部分也被分了,那就可以选中不想分的那部分,选择“分栏”->“一栏” 转自:http://zhidao.baidu.com/question/9873268.ht ...
 - Yii2.0实现微信公众号后台开发
			
接入微信 Yii2后台配置 1.在app/config/params.php中配置token参数 return [ //微信接入 'wechat' =>[ 'token' => 'your ...
 - 解决myeclipse启动慢的问题
			
去掉拼写检查:windows->preferences->General->Editors->Text Editors->Spelling 将"Enable s ...
 - UNIQLO
			
UNIQLO品牌的迅销公司建立于1963年,当年是一家销售西服的小服装店.公司现任董事长兼总经理柳井正早年毕业于早稻田大学经济学专业,1972年8月进入迅销公司,1984年9月就任公司董事长兼总经理. ...
 - jQuery的end()方法使用详解
			
end()方法的定义和用法: end()方法能够回到最近的一个"破坏性"操作之前,即将匹配的元素列表变为前一次的状态.如果没有破坏性操作将返回一个空集.破坏性操作的概念:指任何改变 ...
 - 简单配置Nginx 指向本地端口,并开启SSL
			
简单配置Nginx 指向本地端口,并开启SSL,如果要开启SSL,必须使用域名去申请SSL key,一般是两个文件,一般是要收费的. # 在/etc/nginx/nginx.conf 的文件中有下面一 ...
 - Visio2010如何安装
			
双击setup. 点击我接受此协议的条款,然后点击继续. 这里选择自定义,很重要哦,不要选择立即安装,不然,一会装完后,你会找不到快捷方式的. 文件位置这里选择好存放路径,一会我们要去这里 ...