java jni 参数_Jni中C++和Java的参数传递

如何使用JNI的一些基本方法和过程在网上多如牛毛,如果你对Jni不甚了解,不知道Jni是做什么的,如何建立一个基本的jni程序,或许可以参考下面下面这些文章:

利用VC++6.0实现JNI的最简单的例子这些资料的例子中,大多数只是输入一些简单的参数,获取没有参数。而在实际的使用过程中,往往需要对参数进行处理转换。才可以被C/C++程序识别。比如我们在C++中有一个结构(Struct)DiskInfo ,需要传递一个类似于DiskInfo *pDiskInfo的参数,类似于在C++这样参数如何传递到Java中呢?下面我们就来讨论C++到Java中方法的一些常见参数的转换:

定义Native Java类:

如果你习惯了使用JNI,你就不会觉得它难了。既然本地方法是由其他语言实现的,它们在Java中没有函数体。但是,所有本地代码必须用本地关键词声明,成为Java类的成员。假设我们在C++中有这么一个结构,它用来描述硬盘信息:

773ae66b7206f4db6286910ff9e6e4a8.gif

//

硬盘信息

83e7eae1993e7ee592fd15715f34ef57.gif

struct

{

dfb2de336ec3b73b1a96a6048f939cdf.gif

char

name[

256

];

dfb2de336ec3b73b1a96a6048f939cdf.gif

int

serial;

7325b0f18b89a91add45cac758a7cb3b.gif}

DiskInfo;

那么我们需要在Java中定义一个类来与之匹配,声明可以写成这样:

83e7eae1993e7ee592fd15715f34ef57.gif

class

DiskInfo

{

dfb2de336ec3b73b1a96a6048f939cdf.gif

//

名字

dfb2de336ec3b73b1a96a6048f939cdf.gif

public

String name;

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif

//

序列号

dfb2de336ec3b73b1a96a6048f939cdf.gif

public

int

serial;

7325b0f18b89a91add45cac758a7cb3b.gif}

在这个类中,申明一些Native的本地方法,来测试方法参数的传递,分别定义了一些函数,用来传递结构或者结构数组,具体定义如下面代码:

83e7eae1993e7ee592fd15715f34ef57.gif

/*

***************** 定义本地方法 *******************

*/

773ae66b7206f4db6286910ff9e6e4a8.gif

//

输入常用的数值类型(Boolean,Byte,Char,Short,Int,Float,Double)

773ae66b7206f4db6286910ff9e6e4a8.gif

public

native

void

displayParms(String showText,

int

i, boolean bl);

773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif

//

调用一个静态方法

773ae66b7206f4db6286910ff9e6e4a8.gif

public

native

int

add(

int

a,

int

b);

773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif

//

输入一个数组

773ae66b7206f4db6286910ff9e6e4a8.gif

public

native

void

setArray(boolean[] blList);

773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif

//

返回一个字符串数组

773ae66b7206f4db6286910ff9e6e4a8.gif

public

native String[] getStringArray();

773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif

//

返回一个结构

773ae66b7206f4db6286910ff9e6e4a8.gif

public

native DiskInfo getStruct();

773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif

//

返回一个结构数组

773ae66b7206f4db6286910ff9e6e4a8.gif

public

native DiskInfo[] getStructArray();

编译生成C/C++头文件

定义好了Java类之后,接下来就要写本地代码。本地方法符号提供一个满足约定的头文件,使用Java工具Javah可以很容易地创建它而不用手动去创建。你对Java的class文件使用javah命令,就会为你生成一个对应的C/C++头文件。

1、在控制台下进入工作路径,本工程路径为:E:\work\java\workspace\JavaJni。

2、运行javah 命令:javah -classpath E:\work\java\workspace\JavaJni com.sundy.jnidemo ChangeMethodFromJni

本文生成的C/C++头文件名为: com_sundy_jnidemo_ChangeMethodFromJni.h

在C/C++中实现本地方法

生成C/C++头文件之后,你就需要写头文件对应的本地方法。注意:所有的本地方法的第一个参数都是指向JNIEnv结构的。这个结构是用来调用JNI函数的。第二个参数jclass的意义,要看方法是不是静态的(static)或者实例(Instance)的。前者,jclass代表一个类对象的引用,而后者是被调用的方法所属对象的引用。

返回值和参数类型根据等价约定映射到本地C/C++类型,如表JNI类型映射所示。有些类型,在本地代码中可直接使用,而其他类型只有通过JNI调用操作。

表A

Java

类型

本地类型

描述

boolean

jboolean

C/C++8位整型

byte

jbyte

C/C++带符号的8位整型

char

jchar

C/C++无符号的16位整型

short

jshort

C/C++带符号的16位整型

int

jint

C/C++带符号的32位整型

long

jlong

C/C++带符号的64位整型e

float

jfloat

C/C++32位浮点型

double

jdouble

C/C++64位浮点型

Object

jobject

任何Java对象,或者没有对应java类型的对象

Class

jclass

Class对象

String

jstring

字符串对象

Object[]

jobjectArray

任何对象的数组

boolean[]

jbooleanArray

布尔型数组

byte[]

jbyteArray

比特型数组

char[]

jcharArray

字符型数组

short[]

jshortArray

短整型数组

int[]

jintArray

整型数组

long[]

jlongArray

长整型数组

float[]

jfloatArray

浮点型数组

double[]

jdoubleArray

双浮点型数组

※     JNI类型映射

使用数组:

JNI通过JNIEnv提供的操作Java数组的功能。它提供了两个函数:一个是操作java的简单型数组的,另一个是操作对象类型数组的。

因为速度的原因,简单类型的数组作为指向本地类型的指针暴露给本地代码。因此,它们能作为常规的数组存取。这个指针是指向实际的Java数组或者Java数组的拷贝的指针。另外,数组的布置保证匹配本地类型。

为了存取Java简单类型的数组,你就要要使用GetXXXArrayElements函数(见表B),XXX代表了数组的类型。这个函数把Java数组看成参数,返回一个指向对应的本地类型的数组的指针。

表B

函数

Java

数组类型

本地类型

GetBooleanArrayElements

jbooleanArray

jboolean

GetByteArrayElements

jbyteArray

jbyte

GetCharArrayElements

jcharArray

jchar

GetShortArrayElements

jshortArray

jshort

GetIntArrayElements

jintArray

jint

GetLongArrayElements

jlongArray

jlong

GetFloatArrayElements

jfloatArray

jfloat

GetDoubleArrayElements

jdoubleArray

jdouble

JNI数组存取函数

当你对数组的存取完成后,要确保调用相应的ReleaseXXXArrayElements函数,参数是对应Java数组和GetXXXArrayElements返回的指针。如果必要的话,这个释放函数会复制你做的任何变化(这样它们就反射到java数组),然后释放所有相关的资源。

为了使用java对象的数组,你必须使用GetObjectArrayElement函数和SetObjectArrayElement函数,分别去get,set数组的元素。GetArrayLength函数会返回数组的长度。

使用对象

JNI提供的另外一个功能是在本地代码中使用Java对象。通过使用合适的JNI函数,你可以创建Java对象,get、set 静态(static)和实例(instance)的域,调用静态(static)和实例(instance)函数。JNI通过ID识别域和方法,一个域或方法的ID是任何处理域和方法的函数的必须参数。

表C列出了用以得到静态(static)和实例(instance)的域与方法的JNI函数。每个函数接受(作为参数)域或方法的类,它们的名称,符号和它们对应返回的jfieldID或jmethodID。

表C

函数

描述

GetFieldID

得到一个实例的域的ID

GetStaticFieldID

得到一个静态的域的ID

GetMethodID

得到一个实例的方法的ID

GetStaticMethodID

得到一个静态方法的ID

※域和方法的函数

如果你有了一个类的实例,它就可以通过方法GetObjectClass得到,或者如果你没有这个类的实例,可以通过FindClass得到。符号是从域的类型或者方法的参数,返回值得到字符串,如表D所示。

表D

Java

类型

符号

boolean

Z

byte

B

char

C

short

S

int

I

long

L

float

F

double

D

void

V

objects对象

Lfully-qualified-class-name;L类名

Arrays数组

[array-type [数组类型

methods方法

(argument-types)return-type(参数类型)返回类型

※确定域和方法的符号

下面我们来看看,如果通过使用数组和对象,从C++中的获取到Java中的DiskInfo 类对象,并返回一个DiskInfo数组:

773ae66b7206f4db6286910ff9e6e4a8.gif//返回一个结构数组,返回一个硬盘信息的结构数组773ae66b7206f4db6286910ff9e6e4a8.gifJNIEXPORT jobjectArray JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_getStructArray

773ae66b7206f4db6286910ff9e6e4a8.gif(JNIEnv*env, jobject _obj)

83e7eae1993e7ee592fd15715f34ef57.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif//申明一个object数组dfb2de336ec3b73b1a96a6048f939cdf.gifjobjectArray args=0;

dfb2de336ec3b73b1a96a6048f939cdf.gif    

dfb2de336ec3b73b1a96a6048f939cdf.gif//数组大小dfb2de336ec3b73b1a96a6048f939cdf.gifjsize        len=5;

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//获取object所属类,一般为ava/lang/Object就可以了dfb2de336ec3b73b1a96a6048f939cdf.gifjclass objClass=(env)->FindClass("java/lang/Object");

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//新建object数组dfb2de336ec3b73b1a96a6048f939cdf.gifargs=(env)->NewObjectArray(len, objClass,0);

dfb2de336ec3b73b1a96a6048f939cdf.gif

eee1f5d949a2febc1195073b15d83c84.gif/*下面为获取到Java中对应的实例类中的变量*/dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//获取Java中的实例类dfb2de336ec3b73b1a96a6048f939cdf.gifjclass objectClass=(env)->FindClass("com/sundy/jnidemo/DiskInfo");

dfb2de336ec3b73b1a96a6048f939cdf.gif    

dfb2de336ec3b73b1a96a6048f939cdf.gif//获取类中每一个变量的定义

dfb2de336ec3b73b1a96a6048f939cdf.gif//名字dfb2de336ec3b73b1a96a6048f939cdf.gifjfieldID str=(env)->GetFieldID(objectClass,"name","Ljava/lang/String;");

dfb2de336ec3b73b1a96a6048f939cdf.gif//序列号dfb2de336ec3b73b1a96a6048f939cdf.gifjfieldID ival=(env)->GetFieldID(objectClass,"serial","I");

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//给每一个实例的变量付值,并且将实例作为一个object,添加到objcet数组中dfb2de336ec3b73b1a96a6048f939cdf.giffor(inti=0; i

eee1f5d949a2febc1195073b15d83c84.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif//给每一个实例的变量付值dfb2de336ec3b73b1a96a6048f939cdf.gifjstring jstr=WindowsTojstring(env,"我的磁盘名字是 D:");

dfb2de336ec3b73b1a96a6048f939cdf.gif//(env)->SetObjectField(_obj,str,(env)->NewStringUTF("my name is D:"));dfb2de336ec3b73b1a96a6048f939cdf.gif(env)->SetObjectField(_obj,str,jstr);

dfb2de336ec3b73b1a96a6048f939cdf.gif        (env)->SetShortField(_obj,ival,10);

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//添加到objcet数组中dfb2de336ec3b73b1a96a6048f939cdf.gif(env)->SetObjectArrayElement(args, i, _obj);

6dc3fe7f8bf4968d95592c6d180bd2f7.gif    }dfb2de336ec3b73b1a96a6048f939cdf.gif//返回object数组dfb2de336ec3b73b1a96a6048f939cdf.gifreturnargs;

dfb2de336ec3b73b1a96a6048f939cdf.gif

7325b0f18b89a91add45cac758a7cb3b.gif }

全部的C/C++方法实现代码如下:

83e7eae1993e7ee592fd15715f34ef57.gif/*dfb2de336ec3b73b1a96a6048f939cdf.gif*

dfb2de336ec3b73b1a96a6048f939cdf.gif* 一缕阳光(sundy)版权所有,保留所有权利。

7325b0f18b89a91add45cac758a7cb3b.gif*/83e7eae1993e7ee592fd15715f34ef57.gif/**

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif*  TODO Jni 中一个从Java到C/C++参数传递测试类

dfb2de336ec3b73b1a96a6048f939cdf.gif*

dfb2de336ec3b73b1a96a6048f939cdf.gif*  @author 刘正伟(sundy)

dfb2de336ec3b73b1a96a6048f939cdf.gif*  @seehttp://www.cnweblog.com/sundydfb2de336ec3b73b1a96a6048f939cdf.gif*  @see mailto:sundy26@126.com

dfb2de336ec3b73b1a96a6048f939cdf.gif*  @version 1.0

dfb2de336ec3b73b1a96a6048f939cdf.gif*  @since 2005-4-30

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif*  修改记录:

dfb2de336ec3b73b1a96a6048f939cdf.gif*  

dfb2de336ec3b73b1a96a6048f939cdf.gif*  日期              修改人                 描述

dfb2de336ec3b73b1a96a6048f939cdf.gif*  ----------------------------------------------------------------------------------------------

dfb2de336ec3b73b1a96a6048f939cdf.gif*

dfb2de336ec3b73b1a96a6048f939cdf.gif*

dfb2de336ec3b73b1a96a6048f939cdf.gif*

7325b0f18b89a91add45cac758a7cb3b.gif*/773ae66b7206f4db6286910ff9e6e4a8.gif//JniManage.cpp : 定义 DLL 应用程序的入口点。

773ae66b7206f4db6286910ff9e6e4a8.gif//

773ae66b7206f4db6286910ff9e6e4a8.gifpackage com.sundy.jnidemo;

773ae66b7206f4db6286910ff9e6e4a8.gif#include"stdafx.h"773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif#include773ae66b7206f4db6286910ff9e6e4a8.gif#include773ae66b7206f4db6286910ff9e6e4a8.gif#include"jni.h"773ae66b7206f4db6286910ff9e6e4a8.gif#include"jni_md.h"773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif#include"./head/Base.h"773ae66b7206f4db6286910ff9e6e4a8.gif#include"head/wmi.h"773ae66b7206f4db6286910ff9e6e4a8.gif#include"head/com_sundy_jnidemo_ChangeMethodFromJni.h"//通过javah –jni javactransfer 生成773ae66b7206f4db6286910ff9e6e4a8.gif#include773ae66b7206f4db6286910ff9e6e4a8.gif#include"stdlib.h"773ae66b7206f4db6286910ff9e6e4a8.gif#include"string.h"773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif#pragma comment (lib,"BaseInfo.lib")

773ae66b7206f4db6286910ff9e6e4a8.gif#pragma comment (lib,"jvm.lib")

773ae66b7206f4db6286910ff9e6e4a8.gif//硬盘信息83e7eae1993e7ee592fd15715f34ef57.gifstruct{

dfb2de336ec3b73b1a96a6048f939cdf.gifcharname[256];

dfb2de336ec3b73b1a96a6048f939cdf.gifintserial;

7325b0f18b89a91add45cac758a7cb3b.gif}DiskInfo;

83e7eae1993e7ee592fd15715f34ef57.gif/*BOOL APIENTRY DllMain( HANDLE hModule, 

dfb2de336ec3b73b1a96a6048f939cdf.gif                       DWORD  ul_reason_for_call, 

dfb2de336ec3b73b1a96a6048f939cdf.gif                       LPVOID lpReserved

dfb2de336ec3b73b1a96a6048f939cdf.gif                     )

dfb2de336ec3b73b1a96a6048f939cdf.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif    LPTSTR  strName = new CHAR[256] ;

dfb2de336ec3b73b1a96a6048f939cdf.gif    (*GetHostName)(strName);

dfb2de336ec3b73b1a96a6048f939cdf.gif    printf("%s\n",strName);

dfb2de336ec3b73b1a96a6048f939cdf.gif    delete [] strName;

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif    return TRUE;

7325b0f18b89a91add45cac758a7cb3b.gif}*/773ae66b7206f4db6286910ff9e6e4a8.gif//将jstring类型转换成windows类型773ae66b7206f4db6286910ff9e6e4a8.gifchar*jstringToWindows( JNIEnv*env, jstring jstr );

773ae66b7206f4db6286910ff9e6e4a8.gif//将windows类型转换成jstring类型773ae66b7206f4db6286910ff9e6e4a8.gifjstring WindowsTojstring( JNIEnv*env,char*str );

773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif//主函数773ae66b7206f4db6286910ff9e6e4a8.gifBOOL WINAPI DllMain(HANDLE hHandle, DWORD dwReason, LPVOID lpReserved)

83e7eae1993e7ee592fd15715f34ef57.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gifreturnTRUE;

7325b0f18b89a91add45cac758a7cb3b.gif}773ae66b7206f4db6286910ff9e6e4a8.gif//输入常用的数值类型 Boolean,Byte,Char,Short,Int,Float,Double773ae66b7206f4db6286910ff9e6e4a8.gifJNIEXPORTvoidJNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_displayParms

773ae66b7206f4db6286910ff9e6e4a8.gif(JNIEnv*env, jobject obj, jstring s, jint i, jboolean b)

83e7eae1993e7ee592fd15715f34ef57.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gifconstchar*szStr=(env)->GetStringUTFChars(s,0);

dfb2de336ec3b73b1a96a6048f939cdf.gif    printf("String = [%s]\n", szStr );

dfb2de336ec3b73b1a96a6048f939cdf.gif    printf("int = %d\n", i );

dfb2de336ec3b73b1a96a6048f939cdf.gif    printf("boolean = %s\n", (b==JNI_TRUE?"true":"false") );

dfb2de336ec3b73b1a96a6048f939cdf.gif    (env)->ReleaseStringUTFChars(s, szStr );

7325b0f18b89a91add45cac758a7cb3b.gif}773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif//调用一个静态方法,只有一个简单类型输出773ae66b7206f4db6286910ff9e6e4a8.gifJNIEXPORT jint JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_add

773ae66b7206f4db6286910ff9e6e4a8.gif(JNIEnv*env, jobject, jint a, jint b)

83e7eae1993e7ee592fd15715f34ef57.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gifintrtn=(int)(a+b);

dfb2de336ec3b73b1a96a6048f939cdf.gifreturn(jint)rtn;

7325b0f18b89a91add45cac758a7cb3b.gif}773ae66b7206f4db6286910ff9e6e4a8.gif

83e7eae1993e7ee592fd15715f34ef57.gif输入一个数组,这里输入的是一个Boolean类型的数组

773ae66b7206f4db6286910ff9e6e4a8.gifJNIEXPORTvoidJNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_setArray

773ae66b7206f4db6286910ff9e6e4a8.gif(JNIEnv*env, jobject, jbooleanArray ba)

83e7eae1993e7ee592fd15715f34ef57.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif    jboolean*pba=(env)->GetBooleanArrayElements(ba,0);

dfb2de336ec3b73b1a96a6048f939cdf.gif    jsize len=(env)->GetArrayLength(ba);

dfb2de336ec3b73b1a96a6048f939cdf.gifinti=0;

dfb2de336ec3b73b1a96a6048f939cdf.gif//change even array elementsdfb2de336ec3b73b1a96a6048f939cdf.giffor( i=0; i

eee1f5d949a2febc1195073b15d83c84.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif        pba[i]=JNI_FALSE;

dfb2de336ec3b73b1a96a6048f939cdf.gif        printf("boolean = %s\n", (pba[i]==JNI_TRUE?"true":"false") );

6dc3fe7f8bf4968d95592c6d180bd2f7.gif    }dfb2de336ec3b73b1a96a6048f939cdf.gif    (env)->ReleaseBooleanArrayElements(ba, pba,0);

7325b0f18b89a91add45cac758a7cb3b.gif}773ae66b7206f4db6286910ff9e6e4a8.gif

83e7eae1993e7ee592fd15715f34ef57.gif返回一个字符串数组

773ae66b7206f4db6286910ff9e6e4a8.gifJNIEXPORT jobjectArray JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_getStringArray

773ae66b7206f4db6286910ff9e6e4a8.gif(JNIEnv*env, jobject)

83e7eae1993e7ee592fd15715f34ef57.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif    jstring      str;

dfb2de336ec3b73b1a96a6048f939cdf.gif    jobjectArray args=0;

dfb2de336ec3b73b1a96a6048f939cdf.gif    jsize        len=5;

eee1f5d949a2febc1195073b15d83c84.gifchar*sa[]={"Hello,","world!","JNI","is","fun"};

dfb2de336ec3b73b1a96a6048f939cdf.gifinti=0;

dfb2de336ec3b73b1a96a6048f939cdf.gif    args=(env)->NewObjectArray(len,(env)->FindClass("java/lang/String"),0);

dfb2de336ec3b73b1a96a6048f939cdf.giffor( i=0; i

eee1f5d949a2febc1195073b15d83c84.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif        str=(env)->NewStringUTF(sa[i] );

dfb2de336ec3b73b1a96a6048f939cdf.gif        (env)->SetObjectArrayElement(args, i, str);

6dc3fe7f8bf4968d95592c6d180bd2f7.gif    }dfb2de336ec3b73b1a96a6048f939cdf.gifreturnargs;

7325b0f18b89a91add45cac758a7cb3b.gif}773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif//返回一个结构,这里返回一个硬盘信息的简单结构类型773ae66b7206f4db6286910ff9e6e4a8.gifJNIEXPORT jobject JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_getStruct

773ae66b7206f4db6286910ff9e6e4a8.gif(JNIEnv*env, jobject obj)

83e7eae1993e7ee592fd15715f34ef57.gif{

eee1f5d949a2febc1195073b15d83c84.gif/*下面为获取到Java中对应的实例类中的变量*/dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//获取Java中的实例类dfb2de336ec3b73b1a96a6048f939cdf.gifjclass objectClass=(env)->FindClass("com/sundy/jnidemo/DiskInfo");

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//获取类中每一个变量的定义

dfb2de336ec3b73b1a96a6048f939cdf.gif//名字dfb2de336ec3b73b1a96a6048f939cdf.gifjfieldID str=(env)->GetFieldID(objectClass,"name","Ljava/lang/String;");

dfb2de336ec3b73b1a96a6048f939cdf.gif//序列号dfb2de336ec3b73b1a96a6048f939cdf.gifjfieldID ival=(env)->GetFieldID(objectClass,"serial","I");

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//给每一个实例的变量付值dfb2de336ec3b73b1a96a6048f939cdf.gif(env)->SetObjectField(obj,str,(env)->NewStringUTF("my name is D:"));

dfb2de336ec3b73b1a96a6048f939cdf.gif    (env)->SetShortField(obj,ival,10);

dfb2de336ec3b73b1a96a6048f939cdf.gif    

dfb2de336ec3b73b1a96a6048f939cdf.gifreturnobj;

7325b0f18b89a91add45cac758a7cb3b.gif}773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif//返回一个结构数组,返回一个硬盘信息的结构数组773ae66b7206f4db6286910ff9e6e4a8.gifJNIEXPORT jobjectArray JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_getStructArray

773ae66b7206f4db6286910ff9e6e4a8.gif(JNIEnv*env, jobject _obj)

83e7eae1993e7ee592fd15715f34ef57.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif//申明一个object数组dfb2de336ec3b73b1a96a6048f939cdf.gifjobjectArray args=0;

dfb2de336ec3b73b1a96a6048f939cdf.gif    

dfb2de336ec3b73b1a96a6048f939cdf.gif//数组大小dfb2de336ec3b73b1a96a6048f939cdf.gifjsize        len=5;

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//获取object所属类,一般为ava/lang/Object就可以了dfb2de336ec3b73b1a96a6048f939cdf.gifjclass objClass=(env)->FindClass("java/lang/Object");

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//新建object数组dfb2de336ec3b73b1a96a6048f939cdf.gifargs=(env)->NewObjectArray(len, objClass,0);

dfb2de336ec3b73b1a96a6048f939cdf.gif

eee1f5d949a2febc1195073b15d83c84.gif/*下面为获取到Java中对应的实例类中的变量*/dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//获取Java中的实例类dfb2de336ec3b73b1a96a6048f939cdf.gifjclass objectClass=(env)->FindClass("com/sundy/jnidemo/DiskInfo");

dfb2de336ec3b73b1a96a6048f939cdf.gif    

dfb2de336ec3b73b1a96a6048f939cdf.gif//获取类中每一个变量的定义

dfb2de336ec3b73b1a96a6048f939cdf.gif//名字dfb2de336ec3b73b1a96a6048f939cdf.gifjfieldID str=(env)->GetFieldID(objectClass,"name","Ljava/lang/String;");

dfb2de336ec3b73b1a96a6048f939cdf.gif//序列号dfb2de336ec3b73b1a96a6048f939cdf.gifjfieldID ival=(env)->GetFieldID(objectClass,"serial","I");

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//给每一个实例的变量付值,并且将实例作为一个object,添加到objcet数组中dfb2de336ec3b73b1a96a6048f939cdf.giffor(inti=0; i

eee1f5d949a2febc1195073b15d83c84.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif//给每一个实例的变量付值dfb2de336ec3b73b1a96a6048f939cdf.gifjstring jstr=WindowsTojstring(env,"我的磁盘名字是 D:");

dfb2de336ec3b73b1a96a6048f939cdf.gif//(env)->SetObjectField(_obj,str,(env)->NewStringUTF("my name is D:"));dfb2de336ec3b73b1a96a6048f939cdf.gif(env)->SetObjectField(_obj,str,jstr);

dfb2de336ec3b73b1a96a6048f939cdf.gif        (env)->SetShortField(_obj,ival,10);

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//添加到objcet数组中dfb2de336ec3b73b1a96a6048f939cdf.gif(env)->SetObjectArrayElement(args, i, _obj);

6dc3fe7f8bf4968d95592c6d180bd2f7.gif    }dfb2de336ec3b73b1a96a6048f939cdf.gif//返回object数组dfb2de336ec3b73b1a96a6048f939cdf.gifreturnargs;

dfb2de336ec3b73b1a96a6048f939cdf.gif

7325b0f18b89a91add45cac758a7cb3b.gif }773ae66b7206f4db6286910ff9e6e4a8.gif

773ae66b7206f4db6286910ff9e6e4a8.gif//将jstring类型转换成windows类型773ae66b7206f4db6286910ff9e6e4a8.gifchar*jstringToWindows( JNIEnv*env, jstring jstr )

83e7eae1993e7ee592fd15715f34ef57.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gifintlength=(env)->GetStringLength(jstr );

dfb2de336ec3b73b1a96a6048f939cdf.gifconstjchar*jcstr=(env)->GetStringChars(jstr,0);

dfb2de336ec3b73b1a96a6048f939cdf.gifchar*rtn=(char*)malloc( length*2+1);

dfb2de336ec3b73b1a96a6048f939cdf.gifintsize=0;

dfb2de336ec3b73b1a96a6048f939cdf.gif    size=WideCharToMultiByte( CP_ACP,0, (LPCWSTR)jcstr, length, rtn,(length*2+1), NULL, NULL );

dfb2de336ec3b73b1a96a6048f939cdf.gifif( size<=0)

dfb2de336ec3b73b1a96a6048f939cdf.gifreturnNULL;

dfb2de336ec3b73b1a96a6048f939cdf.gif    (env)->ReleaseStringChars(jstr, jcstr );

dfb2de336ec3b73b1a96a6048f939cdf.gif    rtn[size]=0;

dfb2de336ec3b73b1a96a6048f939cdf.gifreturnrtn;

7325b0f18b89a91add45cac758a7cb3b.gif}773ae66b7206f4db6286910ff9e6e4a8.gif//将windows类型转换成jstring类型773ae66b7206f4db6286910ff9e6e4a8.gifjstring WindowsTojstring( JNIEnv*env,char*str )

83e7eae1993e7ee592fd15715f34ef57.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gif    jstring rtn=0;

dfb2de336ec3b73b1a96a6048f939cdf.gifintslen=strlen(str);

dfb2de336ec3b73b1a96a6048f939cdf.gif    unsignedshort*buffer=0;

dfb2de336ec3b73b1a96a6048f939cdf.gifif( slen==0)

dfb2de336ec3b73b1a96a6048f939cdf.gif        rtn=(env)->NewStringUTF(str ); 

dfb2de336ec3b73b1a96a6048f939cdf.gifelseeee1f5d949a2febc1195073b15d83c84.gif{

dfb2de336ec3b73b1a96a6048f939cdf.gifintlength=MultiByteToWideChar( CP_ACP,0, (LPCSTR)str, slen, NULL,0);

dfb2de336ec3b73b1a96a6048f939cdf.gif        buffer=(unsignedshort*)malloc( length*2+1);

dfb2de336ec3b73b1a96a6048f939cdf.gifif( MultiByteToWideChar( CP_ACP,0, (LPCSTR)str, slen, (LPWSTR)buffer, length )>0)

dfb2de336ec3b73b1a96a6048f939cdf.gif            rtn=(env)->NewString(  (jchar*)buffer, length );

6dc3fe7f8bf4968d95592c6d180bd2f7.gif    }dfb2de336ec3b73b1a96a6048f939cdf.gifif( buffer )

dfb2de336ec3b73b1a96a6048f939cdf.gif        free( buffer );

dfb2de336ec3b73b1a96a6048f939cdf.gifreturnrtn;

7325b0f18b89a91add45cac758a7cb3b.gif}773ae66b7206f4db6286910ff9e6e4a8.gif

Java 测试native代码

这没有什么多说的,看代码吧

773ae66b7206f4db6286910ff9e6e4a8.gif//主测试程序83e7eae1993e7ee592fd15715f34ef57.gifpublicstaticvoidmain(String[] args){

dfb2de336ec3b73b1a96a6048f939cdf.gif        ChangeMethodFromJni changeJni=newChangeMethodFromJni();

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//输入常用的数值类型(string int boolean)dfb2de336ec3b73b1a96a6048f939cdf.gifSystem.outdfb2de336ec3b73b1a96a6048f939cdf.gif                .println("------------------输入常用的数值类型(string int boolean)-----------");

dfb2de336ec3b73b1a96a6048f939cdf.gif        changeJni.displayParms("Hello World!",100,true);

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//调用一个静态方法dfb2de336ec3b73b1a96a6048f939cdf.gifSystem.out.println("------------------调用一个静态方法-----------");

dfb2de336ec3b73b1a96a6048f939cdf.gifintret=changeJni.add(12,20);

dfb2de336ec3b73b1a96a6048f939cdf.gif        System.out.println("The result is:"+String.valueOf(ret));

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//输入一个数组dfb2de336ec3b73b1a96a6048f939cdf.gifSystem.out.println("------------------输入一个数组-----------");

eee1f5d949a2febc1195073b15d83c84.gif        boolean[] blList=newboolean[]{true,false,true};

dfb2de336ec3b73b1a96a6048f939cdf.gif        changeJni.setArray(blList);

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//返回一个字符串数组dfb2de336ec3b73b1a96a6048f939cdf.gifSystem.out.println("------------------返回一个字符串数组-----------");

dfb2de336ec3b73b1a96a6048f939cdf.gif        String[] strList=changeJni.getStringArray();

eee1f5d949a2febc1195073b15d83c84.giffor(inti=0; i

dfb2de336ec3b73b1a96a6048f939cdf.gif            System.out.print(strList[i]);

6dc3fe7f8bf4968d95592c6d180bd2f7.gif        }dfb2de336ec3b73b1a96a6048f939cdf.gif        System.out.println();

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif        System.out.println("------------------返回一个结构-----------");

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//返回一个结构dfb2de336ec3b73b1a96a6048f939cdf.gifDiskInfo disk=changeJni.getStruct();

dfb2de336ec3b73b1a96a6048f939cdf.gif        System.out.println("name:"+disk.name);

dfb2de336ec3b73b1a96a6048f939cdf.gif        System.out.println("Serial:"+disk.serial);

dfb2de336ec3b73b1a96a6048f939cdf.gif

dfb2de336ec3b73b1a96a6048f939cdf.gif//返回一个结构数组dfb2de336ec3b73b1a96a6048f939cdf.gifdfb2de336ec3b73b1a96a6048f939cdf.gif        System.out.println("------------------返回一个结构数组 -----------");

dfb2de336ec3b73b1a96a6048f939cdf.gif        DiskInfo[] diskList=changeJni.getStructArray();

eee1f5d949a2febc1195073b15d83c84.giffor(inti=0; i

dfb2de336ec3b73b1a96a6048f939cdf.gif            System.out.println("name:"+diskList[i].name);

dfb2de336ec3b73b1a96a6048f939cdf.gif            System.out.println("Serial:"+diskList[i].serial);

6dc3fe7f8bf4968d95592c6d180bd2f7.gif        }dfb2de336ec3b73b1a96a6048f939cdf.gif

7325b0f18b89a91add45cac758a7cb3b.gif    }

原文:http://www.blogjava.net/china-qd/archive/2006/04/29/44002.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值