本文地址: http://blog.csdn.net/caroline_wendy
JNI: Java Native Interface, 实现Java和C/C++的互通.
在Android上使用JNI的方法. 时间:2014.9.3
环境: 必须使用标准Eclipse, 安装Android的环境, 才可以使用NDT插件.
Eclipse Standard/SDK Version: Luna Release (4.4.0);
Android: ADT-23.0.3.zip; NDT: GNU Make 3.81;
基础安装略过.
方法:
首先新建JNI的接口类, 包含使用的静态方法. 位置: 项目->src->[package]->JniClient.java
/** * */ package com.example.hellomyjni; /** * @author Administrator * */ public class JniClient { static public native String sayName(); }
进入项目文件夹, 生成JNI的头文件, 使用命令:
"javah -classpath bin/classes -d jni com.example.hellomyjni.JniClient"
命令解析:
javah 生成头文件;
-classpath 使用类的位置(bin/classes), 都是.class文件;
-d jni 需要生成JNI的类(com.example.hellomyjni.JniClient), 包括[package].[classname].
按F5刷新项目, 项目会自动生成jni文件夹, 并包含一个头文件"com_example_hellomyjni_JniClient.h".
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class com_example_hellomyjni_JniClient */ #ifndef _Included_com_example_hellomyjni_JniClient #define _Included_com_example_hellomyjni_JniClient #ifdef __cplusplus extern "C" { #endif /* * Class: com_example_hellomyjni_JniClient * Method: sayName * Signature: ()Ljava/lang/String; */ JNIEXPORT jstring JNICALL Java_com_example_hellomyjni_JniClient_sayName (JNIEnv *, jclass); #ifdef __cplusplus } #endif #endif
右键点击项目在Android Tools里面点击"Add NativeSupport", 就会自动生成: HelloMyJni.cpp和Android.mk.
Android.mk不需要修改:
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_MODULE := HelloMyJni LOCAL_SRC_FILES := HelloMyJni.cpp include $(BUILD_SHARED_LIBRARY)
即修改HelloMyJni.cpp.
首先, 添加头文件, 即JNI生成的头文件, #include "com_example_hellomyjni_JniClient.h"; 此时头文件的报错消失.
其次, 复制JNIEXPORT函数, 并填写参数名称.
在函数写程序, 并调用其他C++程序. 注意的是此时的编译环境自动为C++.
#include <jni.h> #include "com_example_hellomyjni_JniClient.h" JNIEXPORT jstring JNICALL Java_com_example_hellomyjni_JniClient_sayName (JNIEnv *env, jclass) { return env->NewStringUTF("I‘m Spike!"); }
默认为输出为"HelloWorld".
在res->layout->activity_main.xml中, 为TextView添加id, 则可以调用, 如android:id="@+id/text_view". 其他不做修改.
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context="com.example.hellomyjni.MainActivity" > <TextView android:id="@+id/text_view" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/hello_world" /> </RelativeLayout>
首先修改输出字符, 即重新指定字符:
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); TextView tv = (TextView) findViewById(R.id.text_view); tv.setText(JniClient.sayName()); }
static { System.loadLibrary("HelloMyJni"); }
全部:
package com.example.hellomyjni; import android.app.Activity; import android.os.Bundle; import android.view.Menu; import android.view.MenuItem; import android.widget.TextView; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); TextView tv = (TextView) findViewById(R.id.text_view); tv.setText(JniClient.sayName()); } @Override public boolean onCreateOptionsMenu(Menu menu) { // Inflate the menu; this adds items to the action bar if it is present. getMenuInflater().inflate(R.menu.main, menu); return true; } @Override public boolean onOptionsItemSelected(MenuItem item) { // Handle action bar item clicks here. The action bar will // automatically handle clicks on the Home/Up button, so long // as you specify a parent activity in AndroidManifest.xml. int id = item.getItemId(); if (id == R.id.action_settings) { return true; } return super.onOptionsItemSelected(item); } static { System.loadLibrary("HelloMyJni"); } }
此时项目应该没有任何错误和警告. 输出:
原文地址:http://blog.csdn.net/caroline_wendy/article/details/39032551