`
zhaohaolin
  • 浏览: 1017247 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

Jni中C++和Java的参数传递

    博客分类:
  • JAVA
 
阅读更多

Jni中C++和Java的参数传递

如何使用JNI的一些基本方法和过程在网上多如牛毛,如果你对Jni不甚了解,不知道Jni是做什么的,如何建立一个基本的jni程序,或许可以参考下面下面这些文章:
利用VC++6.0实现JNI的最简单的例子  
JNI入门教程之HelloWorld篇
SUN JNI Tutorial

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

定义Native Java类:

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

// 硬盘信息
struct    {
    
char  name[ 256 ];
    
int  serial;
}
DiskInfo;

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

class  DiskInfo  {
    
// 名字
     public  String name;

    
// 序列号
     public   int  serial;
}

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

/* ***************** 定义本地方法 ******************* */
    
// 输入常用的数值类型(Boolean,Byte,Char,Short,Int,Float,Double)
     public  native  void  displayParms(String showText,  int  i, boolean bl);

    
// 调用一个静态方法
     public  native  int  add( int  a,  int  b);

    
// 输入一个数组
     public  native  void  setArray(boolean[] blList);

    
// 返回一个字符串数组
     public  native String[] getStringArray();

    
// 返回一个结构
     public  native DiskInfo getStruct();

    
// 返回一个结构数组
     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函数。每个函数接受(作为参数)域或方法的类,它们的名称,符号和它们对应返回的jfieldIDjmethodID

表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数组:

// 返回一个结构数组,返回一个硬盘信息的结构数组
JNIEXPORT jobjectArray JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_getStructArray
(JNIEnv 
* env, jobject _obj)
{
    
// 申明一个object数组 
    jobjectArray args  =   0 ;
    
    
// 数组大小
    jsize        len  =   5 ;

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

    
// 新建object数组
    args  =  (env) -> NewObjectArray(len, objClass,  0 );

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

    
// 获取Java中的实例类
    jclass objectClass  =  (env) -> FindClass( " com/sundy/jnidemo/DiskInfo " );
    
    
// 获取类中每一个变量的定义
    
// 名字
    jfieldID str  =  (env) -> GetFieldID(objectClass, " name " , " Ljava/lang/String; " );
    
// 序列号
    jfieldID ival  =  (env) -> GetFieldID(objectClass, " serial " , " I " );

    
// 给每一个实例的变量付值,并且将实例作为一个object,添加到objcet数组中
     for ( int   i = 0 ; i  <  len; i ++  )
    
{
        
// 给每一个实例的变量付值
        jstring jstr  =  WindowsTojstring(env, " 我的磁盘名字是 D: " );
        
// (env)->SetObjectField(_obj,str,(env)->NewStringUTF("my name is D:"));
        (env) -> SetObjectField(_obj,str,jstr);
        (env)
-> SetShortField(_obj,ival, 10 );

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

    
// 返回object数组
     return  args;

 }



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

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

/* *

*  TODO Jni 中一个从Java到C/C++参数传递测试类
*
*  @author 刘正伟(sundy)
*  @see 
http://www.cnweblog.com/sundy
*  @see mailto:sundy26@126.com
*  @version 1.0
*  @since 2005-4-30

*  修改记录:
*  
*  日期              修改人                 描述
*  ----------------------------------------------------------------------------------------------
*
*
*
*/

//  JniManage.cpp : 定义 DLL 应用程序的入口点。
//
package com.sundy.jnidemo;
#include 
" stdafx.h "

#include 
< stdio.h >
#include 
< math.h >
#include 
" jni.h "
#include 
" jni_md.h "

#include 
" ./head/Base.h "
#include 
" head/wmi.h "
#include 
" head/com_sundy_jnidemo_ChangeMethodFromJni.h "   // 通过javah –jni javactransfer 生成
#include  < stdio.h >
#include 
" stdlib.h "
#include 
" string.h "

#pragma comment (lib,
" BaseInfo.lib " )
#pragma comment (lib,
" jvm.lib " )
// 硬盘信息
struct    {
    
char  name[ 256 ];
    
int  serial;
}
DiskInfo;
/* BOOL APIENTRY DllMain( HANDLE hModule, 
                       DWORD  ul_reason_for_call, 
                       LPVOID lpReserved
                     )
{
    LPTSTR  strName = new CHAR[256] ;
    (*GetHostName)(strName);
    printf("%s\n",strName);
    delete [] strName;

    return TRUE;
}
*/

// 将jstring类型转换成windows类型
char *  jstringToWindows( JNIEnv  * env, jstring jstr );
// 将windows类型转换成jstring类型
jstring WindowsTojstring( JNIEnv *  env,  char *  str );

// 主函数
BOOL WINAPI DllMain(HANDLE hHandle, DWORD dwReason, LPVOID lpReserved)
{
    
return  TRUE;
}

// 输入常用的数值类型 Boolean,Byte,Char,Short,Int,Float,Double
JNIEXPORT  void  JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_displayParms
(JNIEnv 
* env, jobject obj, jstring s, jint i, jboolean b)
{
    
const   char *  szStr  =  (env) -> GetStringUTFChars(s,  0  );
    printf( 
" String = [%s]\n " , szStr );
    printf( 
" int = %d\n " , i );
    printf( 
" boolean = %s\n " , (b == JNI_TRUE  ?   " true "  :  " false " ) );
    (env)
-> ReleaseStringUTFChars(s, szStr );
}


// 调用一个静态方法,只有一个简单类型输出
JNIEXPORT jint JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_add
(JNIEnv 
* env, jobject, jint a, jint b)
{
    
int  rtn  =  ( int )(a  +  b);
    
return  (jint)rtn;
}


/// /输入一个数组,这里输入的是一个Boolean类型的数组
JNIEXPORT  void  JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_setArray
(JNIEnv 
* env, jobject, jbooleanArray ba)
{
    jboolean
*  pba  =  (env) -> GetBooleanArrayElements(ba,  0  );
    jsize len 
=  (env) -> GetArrayLength(ba);
    
int  i = 0 ;
    
//  change even array elements
     for ( i = 0 ; i  <  len; i += 2  )
    
{
        pba[i] 
=  JNI_FALSE;
        printf( 
" boolean = %s\n " , (pba[i] == JNI_TRUE  ?   " true "  :  " false " ) );
    }

    (env)
-> ReleaseBooleanArrayElements(ba, pba,  0  );
}


/// /返回一个字符串数组
JNIEXPORT jobjectArray JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_getStringArray
(JNIEnv 
* env, jobject)
{
    jstring      str;
    jobjectArray args 
=   0 ;
    jsize        len 
=   5 ;
    
char *         sa[]  =   " Hello, " " world! " " JNI " " is " " fun "  } ;
    
int           i = 0 ;
    args 
=  (env) -> NewObjectArray(len,(env) -> FindClass( " java/lang/String " ), 0 );
    
for ( i = 0 ; i  <  len; i ++  )
    
{
        str 
=  (env) -> NewStringUTF(sa[i] );
        (env)
-> SetObjectArrayElement(args, i, str);
    }

    
return  args;
}


// 返回一个结构,这里返回一个硬盘信息的简单结构类型
JNIEXPORT jobject JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_getStruct
(JNIEnv 
* env, jobject obj)
{
    
/*  下面为获取到Java中对应的实例类中的变量 */

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

    
// 获取类中每一个变量的定义
    
// 名字
    jfieldID str  =  (env) -> GetFieldID(objectClass, " name " , " Ljava/lang/String; " );
    
// 序列号
    jfieldID ival  =  (env) -> GetFieldID(objectClass, " serial " , " I " );


    
// 给每一个实例的变量付值
    (env) -> SetObjectField(obj,str,(env) -> NewStringUTF( " my name is D: " ));
    (env)
-> SetShortField(obj,ival, 10 );
    
    
return  obj;
}


// 返回一个结构数组,返回一个硬盘信息的结构数组
JNIEXPORT jobjectArray JNICALL Java_com_sundy_jnidemo_ChangeMethodFromJni_getStructArray
(JNIEnv 
* env, jobject _obj)
{
    
// 申明一个object数组 
    jobjectArray args  =   0 ;
    
    
// 数组大小
    jsize        len  =   5 ;

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

    
// 新建object数组
    args  =  (env) -> NewObjectArray(len, objClass,  0 );

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

    
// 获取Java中的实例类
    jclass objectClass  =  (env) -> FindClass( " com/sundy/jnidemo/DiskInfo " );
    
    
// 获取类中每一个变量的定义
    
// 名字
    jfieldID str  =  (env) -> GetFieldID(objectClass, " name " , " Ljava/lang/String; " );
    
// 序列号
    jfieldID ival  =  (env) -> GetFieldID(objectClass, " serial " , " I " );

    
// 给每一个实例的变量付值,并且将实例作为一个object,添加到objcet数组中
     for ( int   i = 0 ; i  <  len; i ++  )
    
{
        
// 给每一个实例的变量付值
        jstring jstr  =  WindowsTojstring(env, " 我的磁盘名字是 D: " );
        
// (env)->SetObjectField(_obj,str,(env)->NewStringUTF("my name is D:"));
        (env) -> SetObjectField(_obj,str,jstr);
        (env)
-> SetShortField(_obj,ival, 10 );

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

    
// 返回object数组
     return  args;

 }


// 将jstring类型转换成windows类型
char *  jstringToWindows( JNIEnv   * env, jstring jstr )
{
    
int  length  =  (env) -> GetStringLength(jstr );
    
const  jchar *  jcstr  =  (env) -> GetStringChars(jstr,  0  );
    
char *  rtn  =  ( char * )malloc( length * 2 + 1  );
    
int  size  =   0 ;
    size 
=  WideCharToMultiByte( CP_ACP,  0 , (LPCWSTR)jcstr, length, rtn,(length * 2 + 1 ), NULL, NULL );
    
if ( size  <=   0  )
        
return  NULL;
    (env)
-> ReleaseStringChars(jstr, jcstr );
    rtn[size] 
=   0 ;
    
return  rtn;
}

// 将windows类型转换成jstring类型
jstring WindowsTojstring( JNIEnv *  env,  char *  str )
{
    jstring rtn 
=   0 ;
    
int  slen  =  strlen(str);
    unsigned 
short   *  buffer  =   0 ;
    
if ( slen  ==   0  )
        rtn 
=  (env) -> NewStringUTF(str ); 
    
else
    
{
        
int  length  =  MultiByteToWideChar( CP_ACP,  0 , (LPCSTR)str, slen, NULL,  0  );
        buffer 
=  (unsigned  short   * )malloc( length * 2   +   1  );
        
if ( MultiByteToWideChar( CP_ACP,  0 , (LPCSTR)str, slen, (LPWSTR)buffer, length )  > 0  )
            rtn 
=  (env) -> NewString(  (jchar * )buffer, length );
    }

    
if ( buffer )
        free( buffer );
    
return  rtn;
}


Java 测试native代码
这没有什么多说的,看代码吧

// 主测试程序
     public   static   void  main(String[] args)  {
        ChangeMethodFromJni changeJni 
=   new  ChangeMethodFromJni();

        
// 输入常用的数值类型(string int boolean)
        System. out
                .println(
" ------------------输入常用的数值类型(string int boolean)----------- " );
        changeJni.displayParms(
" Hello World! " 100 true );

        
// 调用一个静态方法
        System. out .println( " ------------------调用一个静态方法----------- " );
        
int  ret  =  changeJni.add( 12 20 );
        System.
out .println( " The result is:  "   +  String.valueOf(ret));

        
// 输入一个数组
        System. out .println( " ------------------输入一个数组----------- " );
        boolean[] blList 
=   new  boolean[]  true false true  } ;
        changeJni.setArray(blList);

        
// 返回一个字符串数组
        System. out .println( " ------------------返回一个字符串数组----------- " );
        String[] strList 
=  changeJni.getStringArray();
        
for  ( int  i  =   0 ; i  <  strList.length; i ++ {
            System.
out .print(strList[i]);
        }

        System.
out .println();

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

        
// 返回一个结构
        DiskInfo disk  =  changeJni.getStruct();
        System.
out .println( " name: "   +  disk.name);
        System.
out .println( " Serial: "   +  disk.serial);

        
// 返回一个结构数组

        System.
out .println( " ------------------返回一个结构数组 ----------- " );
        DiskInfo[] diskList 
=  changeJni.getStructArray();
        
for  ( int  i  =   0 ; i  <  diskList.length; i ++ {
            System.
out .println( " name: "   +  diskList[i].name);
            System.
out .println( " Serial: "   +  diskList[i].serial);
        }


    }


注:本程序在VS2003,eclipse (jse5.0) winxp sp2编译通过

————————————————————————————————————————————

分享到:
评论

相关推荐

    JNI中C和Java参数传递详细例子

    本文将深入探讨在JNI环境中,C/C++和Java之间如何进行参数传递,特别关注于结构体的传递。 #### 二、JNI基础概念 在开始之前,我们先简要回顾一下JNI的基本概念: - **JNIEnv**: 一个指针,提供了JNI函数接口。 - *...

    Jni中C++和Java的参数传递.pdf

    参数和返回值类型需要根据JNI类型映射进行转换。例如,Java的`boolean`对应C/C++的`jboolean`,`String`对应`jstring`,数组类型对应`jobjectArray`等。 例如,实现`displayParms`方法,可能如下: ```c++ ...

    JNI开发Java和C/C++互相传递List集合

    JNI开发Java和C/C++互相传递List集合, 可以参考: Java从C/C++获取List集合对象:...Java传递List集合对象到C/C++ https://blog.csdn.net/niuba123456/article/details/80994168

    C++库封装JNI接口-实现java调用c++

    在跨平台的软件开发中,有时我们需要...这个过程需要理解Java和C++之间的数据类型转换,以及如何在两种语言之间传递参数和返回值。熟练掌握JNI可以极大地扩展Java应用的功能,尤其是在需要高性能计算或利用硬件特性时。

    Qt for Android 用JNI来使C++和Java互调(目前最新版全网只有这个可用)

    4. **通信接口设计**:为了在Qt和Java之间传递数据,可以创建自定义的数据结构,如JNI局部引用、全局引用或弱全局引用,以管理对象生命周期。 5. **加载动态库**:在Android的Activity或Service中,使用System....

    jni之c++和java互编

    此外,如果需要传递参数,还需了解如何在C++和Java之间转换数据类型,例如从`jobject`到Java对象的引用,从`jint`到`int`等。 3. **JNI函数和数据类型**:JNI定义了一系列的函数,如`FindClass`, `GetMethodID`, `...

    Jni中C和Java的参数传递.doc

    需要注意的是,Java 中的对象在传递给 C++ 时,需要通过 JNI 提供的 API 来获取其字段值。此外,由于 Java 和 C++ 数据类型的差异,例如 Java 的字符串是 `jstring` 类型,而 C++ 中是 `char*`,所以需要进行适当的...

    3.jni_c++调用java中的方法

    在本主题中,我们将深入探讨如何使用C++通过JNI来调用Java中的方法,以及如何实现C++与JavaScript的互调。这在跨平台开发、性能优化或利用现有C/C++库时尤其有用。 首先,我们需要理解JNI的基本结构。JNI接口定义了...

    Java通过JNI和c++对象互传

    在这个示例中,我们将使用JNI来实现Java代码和C++代码之间的对象传递。 在这个示例中,我们定义了一个Java类`WifiManager`,它具有两个native方法:`test()`和`getScanResult()`。`test()`方法返回一个字符串,而`...

    JNI与C++数据类型传递示例(包括ArrayList对象、ArrayList嵌套返回)

    一个C++(Ubuntu16.04+QT5.9.1)通过JNI,调用JAVA类及方法的示例。通过JNI传递和返回多种类型的参数,boolean ,int,String,ArrayList,ArrayList嵌套ArrayList&lt;ArrayList&lt;String&gt;&gt;等。

    java jni 传递结构体

    文档里描述了如何通过jni方法在java与c++代码之间传递非基本类型数据

    C++JNI多线程回调java

    1. **初始化JNIEnv**:在C++中,我们需要获取到`JNIEnv`指针,这是JNI的核心,提供了调用Java方法和访问Java对象的接口。 2. **注册本地方法**:在`JNI_OnLoad`函数中,C++代码会注册需要的本地方法,这些方法将被...

    java使用JNI调用C++动态链接库(初学必备)

    Java 使用 JNI(Java Native Interface)调用 C++动态链接库是一种常见的技术,它允许 Java 代码与本地 C 或 C++代码交互,以利用 C++的高性能特性或利用已有的 C++库。以下是对标题和描述中涉及的知识点的详细解释...

    JNI 参数传递 Android 自定义对象

    本篇文章将深入探讨JNI中参数传递的基本数据类型、自定义对象以及系统对象。 一、基本数据类型的参数传递 在JNI中,基本数据类型如int、float、char等的传递相对简单。Java中的基本类型在C/C++中都有对应的类型。...

    JNI学习二:字符串参数传递与返回值

    在本示例中,我们关注的是JNI中的字符串参数传递和返回值处理,这对于那些需要在Java和C/C++之间进行深度集成的项目非常重要。 1. **JNI基本概念** JNI是Java平台提供的一个接口,它使得开发者可以编写本地方法...

    用JNI实现java和C++的相互调用

    在Java编程环境中,有时我们需要利用C或C++的高性能特性,这时可以借助Java Native Interface (JNI) 进行Java和C++之间的交互调用。JNI允许Java代码调用本地(非Java)代码,并且反之亦然。下面我们将详细介绍如何用...

    jni操作list集合,来存储对象

    - 在C/C++中实现JNI方法,创建和操作Java对象。 - 注意对象序列化和内存管理。 - 处理可能出现的异常。 通过以上步骤,我们可以使用JNI在Android应用中高效地处理Java的List集合,包括存储自定义对象。这在某些需要...

    [Android] 图片JNI(C++\Java)高斯模糊 多线程

    具体实现时,首先在Java层创建一个JNI方法,传递图像数据和参数,然后在C++层调用高斯模糊或堆栈模糊函数。处理完成后,C++通过JNI回调Java方法,将结果传回Java层,并在主线程中更新UI。 总结来说,使用JNI结合C++...

    androidjni编程,java和c层的互相通信传递数据

    Android JNI(Java Native Interface)编程是Android开发中的一个重要部分,它允许Java代码和其他语言(如C/C++)编写的代码进行交互。JNI在Android平台上扮演着桥梁的角色,使得开发者能够利用C/C++的强大性能和...

Global site tag (gtag.js) - Google Analytics