Chinaunix首页 | 论坛 | 博客
  • 博客访问: 33194
  • 博文数量: 17
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 195
  • 用 户 组: 普通用户
  • 注册时间: 2014-08-27 14:42
文章分类

全部博文(17)

文章存档

2016年(12)

2014年(5)

我的朋友

分类: C/C++

2014-09-01 14:47:10

Java无法直接调用C# dll,需要通过经过桥接的方式,进行中继转发一下请求。查阅大量资料,做了大量实验,不停的调试、排错之后,期间试过jni4net,不过这个插件需要修改原有的C# DLL内容,还会生成一些额外的Java代码,jni4net侵入性太多,将它排除。最后通过管理性的C++桥接方式,成功完成了Java调用C# DLL。

在文中附上完整的代码工程,让读者做实验时参考起来非常方便。本文附上完整的Java、C++、CSharp代码工程,供大家参考。

 

实验环境:

64位 Win 7

jdk1.7.0_51

Eclipse 4.3.1

Visual Studio 2010

 

 

具体的实现步骤如下:

1,新建一个Java项目TestJNI,定义一个Java客户端类 TestJNI.java

 

package msg;

public class TestJNI {   
	   
	public native boolean MasSentMessage(String a, String b);   
	
    public native int add(int a, int b);   
    public native String submit(String a, String b);   
    public native boolean testBoolean(String a, String b);   
    static {   
        System.loadLibrary("CPP");   
    }   
    public static void main(String[] args) {   
        TestJNI t = new TestJNI();   
        
//        System.out.println(t.MasSentMessage("user", "pass"));  
//        System.out.println(t.add(2, 20));   
        System.err.println(t.submit("user", "pass"));
//        System.err.println(t.testBoolean("1", "pass"));
    }   
}

 

 

2,调用JDK里的javah命令通过TestJNI.java类生成msg_TestJNI.h文件,javah设置如下, 

 
 

 

3,在Visual Studio 2010新建一个C#项目CSharp(项目类型为Visual C# -->Windows?类库)

编辑C#文件如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace OJMain
{
    public class OJEntrance
    {
        private int result;
        public int Result
        {
            get { return result + 10; }
            set { this.result = value; }
        }
        public string submit(string str1, string str2)
        {
           Console.WriteLine("成功调用了dll");
           return "CSharp:" + str1 + ":" + str2;
        }
        public bool testBoolean(string str1, string str2)
        {
            if (str1.Equals("true"))
            {
                return true;
            }
            else
            {
                return false;
            }
        }
    }
}

 

 

4,利用Visual Studio 2010 生成dll。

如果调用C#项目的Java客户端在64位的机器上,需要重新生成dll,在Visual Studio 2010的生成参数设置如下,将目标平台设为“Any CPU”



  

5,在Visual Studio 2010中新建一个win32 dll模式的C++项目CPP,为了在Java和C#之间的调用建立通道并进行转接。

 

5.1设置两个项目属性

一,项目属性-->配置属性--> 常规:“公共语言运行时支持”设为“公共语言运行时支持(/clr)”

二,项目属性-->配置属性-->C/C++-->代码生成:“运行库”设为“多线程DLL(/MD)”

 

5.2  C++源文件的目录里放置如下内容:

  • JDK目录里的两个h文件(jni.h,jni_md.h);
  • Java项目中生成的msg_TestJNI.h;
  • C#项目CSharp中生成的CSharp.dll ;

5.3编写jstring 和 string的相互转换功能,完整代码参见附件 CPP.rar(CPP.cpp)。

// char* To jstring
jstring stringTojstring(JNIEnv* env, const char* pat)
{ 
	jclass strClass = env->FindClass("Ljava/lang/String;"); 
	jmethodID ctorID = env->GetMethodID(strClass, "", "([BLjava/lang/String;)V"); 
	jbyteArray bytes = env->NewByteArray(strlen(pat)); 
	env->SetByteArrayRegion(bytes, 0, strlen(pat), (jbyte*)pat); 
	jstring encoding = env->NewStringUTF("utf-8"); 
	return (jstring)env->NewObject(strClass, ctorID, bytes, encoding); 
}
// jstring To char*
char* jstringTostring(JNIEnv* env, jstring jstr)
{
	char* rtn = NULL;
	jclass clsstring = env->FindClass("java/lang/String");
	jstring strencode = env->NewStringUTF("utf-8");
	jmethodID mid = env->GetMethodID(clsstring, "getBytes", "(Ljava/lang/String;)[B");
	jbyteArray barr = (jbyteArray)env->CallObjectMethod(jstr, mid, strencode);
	jsize alen = env->GetArrayLength(barr);
	jbyte* ba = env->GetByteArrayElements(barr, JNI_FALSE);
	if (alen > 0)
	{
		rtn = (char*)malloc(alen + 1);
		memcpy(rtn, ba, alen);
		rtn[alen] = 0;
	}
	env->ReleaseByteArrayElements(barr, ba, 0);
	return rtn;
}
// jstring To String
String^ jstringToStr(JNIEnv* env, jstring jstr)
{
	char* str = jstringTostring(env, jstr);
	String^ value = gcnew String(str);
	free(str);
	return value;
}

// String To jstring
jstring strTojstring(JNIEnv* env, String^ rtn)
{
	pin_ptr wch = PtrToStringChars(rtn);
	size_t convertedChars = 0;
	size_t sizeInBytes = ((rtn->Length + 1) * 2);
	char *ch = (char *)malloc(sizeInBytes);
	errno_t err = wcstombs_s(&convertedChars, 
		ch, sizeInBytes,
		wch, sizeInBytes);
	jstring js = stringTojstring(env, ch);
	free(ch);
	return js;
}

 

5.4  编写C++的管理类, 完整代码参见附件 CPP.rar(CPP.cpp)

#include "jni.h"   
#include "jni_md.h"
#include "msg_TestJNI.h" 

//引入c#的库和命名空间
#using "CSharp.dll"
using namespace OJMain; 
……
…..
JNIEXPORT jint JNICALL Java_msg_TestJNI_add   
  (JNIEnv *env, jobject obj, jint a, jint b)   
{   
    //c#中的对象   
    OJEntrance ^o = gcnew OJEntrance();   
    o->Result = a + b;   
    return o->Result;   
}

JNIEXPORT jstring JNICALL Java_msg_TestJNI_submit
  (JNIEnv *env, jobject obj, jstring str1, jstring str2)  
{   
    //c#中的对象   
    OJEntrance ^o = gcnew OJEntrance();   
    return strTojstring(env, o->submit(jstringToStr(env,str1), jstringToStr(env,str2)));   
}

JNIEXPORT jboolean JNICALL Java_msg_TestJNI_testBoolean
  (JNIEnv *env, jobject obj, jstring str1, jstring str2)  
{   
    //c#中的对象   
    OJEntrance ^o = gcnew OJEntrance();   
    return o->testBoolean(jstringToStr(env,str1), jstringToStr(env,str2));  
}

 

5.5利用Visual Studio 2010生成CPP.dll

 

 

6,JDK的bin目录里放置如下内容

  • C# 项目CSharp中生成的CSharp.dll ;
  • C++项目CPP中生成的CPP.dll;

 

7,在Java项目TestJNI中 运行TestJNI,调用C# dll。

 

 
附:示例代码

java.zip 
CSharp.rar
CPP.rar

阅读(1180) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~