Android开发---JNI基础知识

关于JNI的使用java和C/C++的详细使用

在使用NDK开发Android应用程序时,经常会遇到JNI文件的中头文件和函数名称不正确,则使用NDK开发应用程序时,步骤如下:

1)第一步:编写java文件

-----------------------------------------------------------------------------

package com.example.demo;

public class Hello{
    public native static int add();   //static类型的方法
    public native String getString(); //非static类型的方法
}

-----------------------------------------------------------------------------

2)第二步:编译java文件

-----------------------------------------------------------------------------

使用命令:javac Hello.java

3)第三步:使用javah -jni com.example.demo.Hello生成相应JNI的头文件,生成文件如下:

---------------------------------------------------------------------------------------------------

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_example_demo_Hello */


#ifndef _Included_com_example_demo_Hello
#define _Included_com_example_demo_Hello
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_example_demo_Hello
 * Method:    add
 * Signature: ()I
 */
JNIEXPORT jint JNICALL Java_com_example_demo_Hello_add
  (JNIEnv *, jclass);     //表明该方法在java代码中是static类型的


/*
 * Class:     com_example_demo_Hello
 * Method:    getString
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_example_demo_Hello_getString
  (JNIEnv *, jobject);    //表明该方法在java代码中是非static类型的


#ifdef __cplusplus
}
#endif
#endif

---------------------------------------------------------------------------------------------------

通过第三个步骤的编译结果可以看出,生成的本地方法中,第一个参数类型是JNIEnv* ,是一个指向JNIEnv的指针类型。JNIEnv是一个指向了
一个Pointer,这个Pointer指向了一张函数表,这张函数表的每一项都是JNI中的一个函数的入口。本地的方法通过查找这张表来调用某个JNI函数,来喝
JVM交互。第二个参数,如果声明的方法为static类型的,则该参数为jclass类型;如果声明的参数为非static类型,则为jobject类型。

4)编译当前的JNI文件(Android.mk)

---------------------------------------------------------------------------------------------------

LOCAL_PATH := $(call my-dir)


include $(CLEAR_VARS)


LOCAL_MODULE    := hello-jni     //编译为动态库的名称hello-jni.so
LOCAL_SRC_FILES := hello-jni.c   //编译源文件


include $(BUILD_SHARED_LIBRARY)

----------------------------------------------------------------------------------------------------

5.Android代码下JNI编程中C和C++代码编写差别

-----------------------------------------------------------------------------------------------------------------------

android jni之C 和 C++ 函数实现的比较:
注意:jni.h头文件中对于***.c  &  ***.cpp采用不同的定义
jclass (JNICALL *GetObjectClass)   (JNIEnv *env, jobject obj);
jclass GetObjectClass(jobject obj) {
        return functions->GetObjectClass(this,obj);
}
对于***.c
1)jclass test_class = (*env)->GetObjectClass(env, obj);
2)jfieldID id_num = (*env)->GetFieldID(env, test_class, "num", "I");
对于 ***.cpp
1)jclass test_class = env->GetObjectClass(obj); 
2)jfieldID id_num = env->GetFieldID(test_class, "num", "I"); 
在 C 中,
JNI 函数调用由“(*env)->”作前缀,目的是为了取出函数指针所引用的值。
在 C++ 中,
JNIEnv 类拥有处理函数指针查找的内联成员函数。
下面将说明这个细微的差异,其中,这两行代码访问同一函数,但每种语言都有各自的语法。
C 语法:jsize len = (*env)->GetArrayLength(env,array);
C++ 语法:jsize len =env->GetArrayLength(array);

------------------------------------------------------------------------------------------------------------------------

6.编写相应的代码步骤

1)首先编写java层代码  

------------------------------------------------------------------------------------------------------------------------

package com.example.jnitest;
public class ArrayTest {
 public native String getLine(String str); 
}

------------------------------------------------------------------------------------------------------------------------

2)使用cygin命令编译如下:


使用上述命令可以在d/javacode/jnitest/src目录下生成com_example_jnitest_ArrayTest.h

3)将com_example_jnitest_ArrayTest.h头文件添加到.c文件中,编写JNI代码

--------------------------------------------------------------------------------------------------------------------------

#include "com_example_jnitest_ArrayTest.h"
#include<stdio.h>
JNIEXPORT jstring JNICALL Java_com_example_jnitest_ArrayTest_getLine
  (JNIEnv *env, jobject clazz, jstring parameter){
     char buf[128] = "初学者"; //字符缓冲  
     char *str;  
     str = (char *)(env)->GetStringUTFChars(parameter,NULL);  
     if(str == NULL)  
     {    
         return NULL;    
     }  
     printf("%s",str);  
     /** 
     **使用完了utf-8类型的字符后,我们需要释放由上面方法返回的字符串,这样可以释放被这些字符占用的内存空间,避免造成内存瘫痪  
     **/  
     (env)->ReleaseStringUTFChars(parameter, str);  
     return (env)->NewStringUTF(buf);  //该方法实例化一个UTF-8编码的本地字符串为java.lang.String类型,新创建的就是java中  
}

--------------------------------------------------------------------------------------------------------------------------------

4)使用cygin编译相应的JNI代码:

------------------------------------------------------------------------------------------------------------------------------------


5)在java文件中添加相应的动态库

----------------------------------------------------------------------------------------------------------------------------------

package com.example.jnitest;

public class ArrayTest {
    static {
        System.loadLibrary("ArrayTest");
    }
    public native String getLine(String str); 
}

------------------------------------------------------------------------------------------------------------------------------------

6)编写Android应用程序

--------------------------------------------------------------------------------------------------------------------------------------

package com.example.jnitest;


import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.widget.TextView;


public class MainActivity extends Activity {


    TextView mTextView=null;
    ArrayTest mArrayTest=null;
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mArrayTest =new ArrayTest();
        mTextView = (TextView)findViewById(R.id.textView); 
        mTextView.setText(mArrayTest.getLine("happy"));
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值