w7+eclipse +ndk9 NDK开发 hellow ndk 笔记(java调用c)
2013-09-03 18:32
211 查看
1 下载eclipse
2 安装adt
3 下载android ndk r9版本
4 下载jdk 这里我用的是1.6
5 下载android sdk
下载完毕后配置相应路径,环境变量
完成后
创建android 项目,项目结构如下
创建jni文件夹,目录和src平级
右击项目Properties,选择Builders-》new-》Program
然后配置main,输入Name,Location,Working Directory
配置Refush
配置Build Options,Specify Resource.选择项目的jni文件夹
Build Options其它选项配置
创建JniClient.java文件
创建Point.java文件
cmd下定位到这个目录(classes目录),执行
javah -jni com.ndk.test.JniClient会生成
com_ndk_test_JniClient.h 文件,把该文件放jni文件夹内
内容如下
在jni文件夹下创建com_ndk_test_JniCliet.c文件
在jni文件加下创建Android.mk文件
MainActivity代码
完成以后项目结构如下
接下来就和android项目一样直接运行即可
运行成功以后我们既可以把这个项目在bin->armeabi->libTestNdk.so文件 在其他app里面使用了
copy armeabi文件夹到目标项目libs下,然后copy JniCliet.java和Poibt.java文件
注意因为so文件读取的路径都是写好的,所以复制的时候这2个文件包名不能变即com.ndk.test
另外这个demo有一个自定义的类Point,使用的时候变量x和y改成 public float x;public float y;
使用的代码和上面MainActivity一样
下面来说说这个demo的一些语法
Android.mk
LOCAL_PATH := $(call my-dir)
一个Android.mk file首先必须定义好LOCAL_PATH变量。它用于在开发树中查找源文件。在这个例子中,宏函数’my-dir’, 由编译系统提供,用于返回当前路径(即包含Android.mk
file文件的目录)。
include $( CLEAR_VARS)
CLEAR_VARS由编译系统提供,指定让GNU MAKEFILE为你清除许多LOCAL_XXX变量(例如 LOCAL_MODULE, LOCAL_SRC_FILES, LOCAL_STATIC_LIBRARIES, 等等...),
除LOCAL_PATH 。这是必要的,因为所有的编译控制文件都在同一个GNU MAKE执行环境中,所有的变量都是全局的。
LOCAL_MODULE := helloworld
LOCAL_MODULE变量必须定义,以标识你在Android.mk文件中描述的每个模块。名称必须是唯一的,而且不包含任何空格。注意编译系统会自动产生合适的前缀和后缀,换句话说,一个被命名为'foo'的共享库模块,将会生成'libfoo.so'文件。
重要注意事项
如果你把库命名为‘libhelloworld’,编译系统将不会添加任何的lib前缀,也会生成libhelloworld.so,这是为了支持来源于Android平台的源代码的Android.mk文件,如果你确实需要这么做的话。
LOCAL_SRC_FILES变量必须包含将要编译打包进模块中的C或C++源代码文件。注意,你不用在这里列出头文件和包含文件,因为编译系统将会自动为你找出依赖型的文件;仅仅列出直接传递给编译器的源代码文件就好。【注意,默认的C++源码文件的扩展名是’.cpp’. 指定一个不同的扩展名也是可能的,只要定义LOCAL_DEFAULT_CPP_EXTENSION变量,不要忘记开始的小圆点(也就是定义为‘.cxx’,而不是‘cxx’)(当然这一步我们一般不会去改它)】
include $(BUILD_SHARED_LIBRARY)
BUILD_SHARED_LIBRARY是编译系统提供的变量,指向一个GNU Makefile脚本(应该就是在build/core目录下的shared_library.mk),负责收集自从上次调用'include
$(CLEAR_VARS)'以来,定义在LOCAL_XXX变量中的所有信息,并且决定编译什么,如何正确地去做。并根据其规则生成静态库。同理对于静态库。
com_ndk_test_JniClient.c文件
//#define LOGI(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)
定义log类型需要添加
#include "utils/Log.h" //这个是在android源码环境使用
#include <android/log.h>//这个在NDK编译时使用
其它类型定义
#undef LOG // 取消默认的LOG
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,LOG_TAG,__VA_ARGS__) // 定义LOG类型
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__) // 定义LOG类型
#define LOGW(...) __android_log_print(ANDROID_LOG_WARN,LOG_TAG,__VA_ARGS__) // 定义LOG类型
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__) // 定义LOG类型
#define LOGF(...) __android_log_print(ANDROID_LOG_FATAL,LOG_TAG,__VA_ARGS__) // 定义LOG类型
使用__android_log_write(ANDROID_LOG_INFO,
"MyNdkDemo", "found class Point");
如果要支持log还需要在mk文件中加入LOCAL_LDLIBS
:= -llog
JNIEXPORT jfloat JNICALL Java_com_ndk_test_JniClient_distance(JNIEnv*
env, jobject thiz, jobject a,jobject b)
这里JNIEXPORT和JNICALL都是JNI的关键字,表示此函数是要被JNI调用的,jfloat标示java的float。
com_ndk_test包名 JniClient类名 distance方法名
jobject a对象 JNIEnv* env, jobject thiz这2个无需关系所有方法都有
该项目demo地址
http://download.csdn.net/detail/a56573016613/6207831
以上是c文件如果c++我们需要改mk文件属性如下
LOCAL_PATH := $(call my-dir)
LOCAL_CPP_EXTENSION := .cpp
include $(CLEAR_VARS)
LOCAL_MODULE := TestCpp
LOCAL_SRC_FILES := TestCpp.cpp
include $(BUILD_SHARED_LIBRARY)
项目地址
http://download.csdn.net/detail/a56573016613/6217343
http://blog.csdn.net/a56573016613/article/details/14224685 c调用java
2 安装adt
3 下载android ndk r9版本
4 下载jdk 这里我用的是1.6
5 下载android sdk
下载完毕后配置相应路径,环境变量
完成后
创建android 项目,项目结构如下
创建jni文件夹,目录和src平级
右击项目Properties,选择Builders-》new-》Program
然后配置main,输入Name,Location,Working Directory
配置Refush
配置Build Options,Specify Resource.选择项目的jni文件夹
Build Options其它选项配置
创建JniClient.java文件
package com.ndk.test; public class JniClient {
static { System.loadLibrary("TestNdk"); } static public native String AddStr(String strA, String strB); static public native int AddInt(int a, int b); static public native float distance(Point a, Point b); }
创建Point.java文件
package com.ndk.test; public class Point { float x; float y; }运行一下项目这个时候在项目目录JniClient.class文件,在项目路径下面的bin\classes里面
cmd下定位到这个目录(classes目录),执行
javah -jni com.ndk.test.JniClient会生成
com_ndk_test_JniClient.h 文件,把该文件放jni文件夹内
内容如下
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class com_ndk_test_JniClient */ #ifndef _Included_com_ndk_test_JniClient #define _Included_com_ndk_test_JniClient #ifdef __cplusplus extern "C" { #endif /* * Class: com_ndk_test_JniClient * Method: AddStr * Signature: (Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String; */ JNIEXPORT jstring JNICALL Java_com_ndk_test_JniClient_AddStr (JNIEnv *, jclass, jstring, jstring); /* * Class: com_ndk_test_JniClient * Method: AddInt * Signature: (II)I */ JNIEXPORT jint JNICALL Java_com_ndk_test_JniClient_AddInt (JNIEnv *, jclass, jint, jint); JNIEXPORT jfloat JNICALL Java_com_ndk_test_JniClient_distance (JNIEnv* env, jobject thiz, jobject a,jobject b); #ifdef __cplusplus } #endif #endif
在jni文件夹下创建com_ndk_test_JniCliet.c文件
#include "com_ndk_test_JniClient.h" #include <stdlib.h> #include <stdio.h> #include <android/log.h> #define LOG_TAG "MOONJNI" //#define LOGI(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__) #ifdef __cplusplus extern "C" { #endif /* * Class: com_ndk_test_JniClient * Method: AddStr * Signature: (Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String; */ JNIEXPORT jstring JNICALL Java_com_ndk_test_JniClient_AddStr (JNIEnv *env, jclass arg, jstring instringA, jstring instringB) { jstring str = (*env)->NewStringUTF(env, "HelloWorld from JNI !"); return str; } /* * Class: com_ndk_test_JniClient * Method: AddInt * Signature: (II)I */ JNIEXPORT jint JNICALL Java_com_ndk_test_JniClient_AddInt (JNIEnv *env, jclass arg, jint a, jint b) { return a + b; } //第一个参数JNIEnv* env和第二个参数jobject thiz都是必须的,后面的才是Java中传递进来的参数 static const char *classPathName = "com/ndk/test/Point"; JNIEXPORT jfloat JNICALL Java_com_ndk_test_JniClient_distance(JNIEnv* env, jobject thiz, jobject a,jobject b) { //jclass point_class = (*env)->FindClass(env, classPathName); jclass point_class = (*env)->GetObjectClass(env, a); if (point_class == NULL) { //printf("class not found"); //__android_log_write(ANDROID_LOG_INFO, "MyNdkDemo", "class Point not found"); return 0; } else { //__android_log_write(ANDROID_LOG_INFO, "MyNdkDemo", "found class Point"); } jfieldID field_x = (*env)->GetFieldID(env, point_class, "x", "F"); jfieldID field_y = (*env)->GetFieldID(env, point_class, "y", "F"); jfloat ax = (*env)->GetFloatField(env, a, field_x); jfloat ay = (*env)->GetFloatField(env, a, field_y); jfloat bx = (*env)->GetFloatField(env, b, field_x); jfloat by = (*env)->GetFloatField(env, b, field_y); // return ax+ay+bx+by; return sqrtf(powf(bx-ax, 2) + powf(by-ay, 2)); } #ifdef __cplusplus } #endif
在jni文件加下创建Android.mk文件
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_MODULE := TestNdk LOCAL_SRC_FILES := com_ndk_test_JniClient.c include $(BUILD_SHARED_LIBRARY)
MainActivity代码
package com.ndk.test; import android.os.Bundle; import android.app.Activity; import android.widget.TextView; public class MainActivity extends Activity { /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // setContentView(R.layout.main); String str = JniClient.AddStr("prefix", "suffix"); int iSum = JniClient.AddInt(5, 2); String strSum = "5 + 7 = " + iSum; TextView tv1 = new TextView(this); tv1.setText(str+"dis="+getDistance()); setContentView(tv1); } private float getDistance() { Point a = new Point(); a.x = 3; a.y = 3; Point b = new Point(); b.x = 5; b.y = 5; float d = JniClient.distance(a,b); return d; } }
完成以后项目结构如下
接下来就和android项目一样直接运行即可
运行成功以后我们既可以把这个项目在bin->armeabi->libTestNdk.so文件 在其他app里面使用了
copy armeabi文件夹到目标项目libs下,然后copy JniCliet.java和Poibt.java文件
注意因为so文件读取的路径都是写好的,所以复制的时候这2个文件包名不能变即com.ndk.test
另外这个demo有一个自定义的类Point,使用的时候变量x和y改成 public float x;public float y;
使用的代码和上面MainActivity一样
下面来说说这个demo的一些语法
Android.mk
LOCAL_PATH := $(call my-dir)
一个Android.mk file首先必须定义好LOCAL_PATH变量。它用于在开发树中查找源文件。在这个例子中,宏函数’my-dir’, 由编译系统提供,用于返回当前路径(即包含Android.mk
file文件的目录)。
include $( CLEAR_VARS)
CLEAR_VARS由编译系统提供,指定让GNU MAKEFILE为你清除许多LOCAL_XXX变量(例如 LOCAL_MODULE, LOCAL_SRC_FILES, LOCAL_STATIC_LIBRARIES, 等等...),
除LOCAL_PATH 。这是必要的,因为所有的编译控制文件都在同一个GNU MAKE执行环境中,所有的变量都是全局的。
LOCAL_MODULE := helloworld
LOCAL_MODULE变量必须定义,以标识你在Android.mk文件中描述的每个模块。名称必须是唯一的,而且不包含任何空格。注意编译系统会自动产生合适的前缀和后缀,换句话说,一个被命名为'foo'的共享库模块,将会生成'libfoo.so'文件。
重要注意事项
如果你把库命名为‘libhelloworld’,编译系统将不会添加任何的lib前缀,也会生成libhelloworld.so,这是为了支持来源于Android平台的源代码的Android.mk文件,如果你确实需要这么做的话。
LOCAL_SRC_FILES变量必须包含将要编译打包进模块中的C或C++源代码文件。注意,你不用在这里列出头文件和包含文件,因为编译系统将会自动为你找出依赖型的文件;仅仅列出直接传递给编译器的源代码文件就好。【注意,默认的C++源码文件的扩展名是’.cpp’. 指定一个不同的扩展名也是可能的,只要定义LOCAL_DEFAULT_CPP_EXTENSION变量,不要忘记开始的小圆点(也就是定义为‘.cxx’,而不是‘cxx’)(当然这一步我们一般不会去改它)】
include $(BUILD_SHARED_LIBRARY)
BUILD_SHARED_LIBRARY是编译系统提供的变量,指向一个GNU Makefile脚本(应该就是在build/core目录下的shared_library.mk),负责收集自从上次调用'include
$(CLEAR_VARS)'以来,定义在LOCAL_XXX变量中的所有信息,并且决定编译什么,如何正确地去做。并根据其规则生成静态库。同理对于静态库。
com_ndk_test_JniClient.c文件
//#define LOGI(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)
定义log类型需要添加
#include "utils/Log.h" //这个是在android源码环境使用
#include <android/log.h>//这个在NDK编译时使用
其它类型定义
#undef LOG // 取消默认的LOG
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,LOG_TAG,__VA_ARGS__) // 定义LOG类型
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__) // 定义LOG类型
#define LOGW(...) __android_log_print(ANDROID_LOG_WARN,LOG_TAG,__VA_ARGS__) // 定义LOG类型
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__) // 定义LOG类型
#define LOGF(...) __android_log_print(ANDROID_LOG_FATAL,LOG_TAG,__VA_ARGS__) // 定义LOG类型
使用__android_log_write(ANDROID_LOG_INFO,
"MyNdkDemo", "found class Point");
如果要支持log还需要在mk文件中加入LOCAL_LDLIBS
:= -llog
JNIEXPORT jfloat JNICALL Java_com_ndk_test_JniClient_distance(JNIEnv*
env, jobject thiz, jobject a,jobject b)
这里JNIEXPORT和JNICALL都是JNI的关键字,表示此函数是要被JNI调用的,jfloat标示java的float。
com_ndk_test包名 JniClient类名 distance方法名
jobject a对象 JNIEnv* env, jobject thiz这2个无需关系所有方法都有
该项目demo地址
http://download.csdn.net/detail/a56573016613/6207831
以上是c文件如果c++我们需要改mk文件属性如下
LOCAL_PATH := $(call my-dir)
LOCAL_CPP_EXTENSION := .cpp
include $(CLEAR_VARS)
LOCAL_MODULE := TestCpp
LOCAL_SRC_FILES := TestCpp.cpp
include $(BUILD_SHARED_LIBRARY)
项目地址
http://download.csdn.net/detail/a56573016613/6217343
http://blog.csdn.net/a56573016613/article/details/14224685 c调用java
相关文章推荐
- w7+eclipse +ndk9 NDK开发 hellow ndk 笔记(c调用java)
- [原创]Android 基于NDK的JNI开发 C调用java和java调用C的进阶教程
- Android-NDK开发之第四个例子--用C/C++调用Java
- NDK开发java调用c(三)
- NDK开发(1)——JAVA通过JNI调用C代码详细步骤
- ndk开发中,在c中调用java代码
- Android NDK (学习笔记八) —— Java代码与C代码间方法的调用
- Android 基于NDK的JNI开发 C调用java和java调用C的进阶教程
- Android-NDK开发之第四个例子--用C/C++调用Java
- Android Studio NDK开发之一(Java调用C的方法)
- NDK开发c调用java代码(四)
- Eclipse下NDK开发(java调用so文件)
- Android-NDK开发之基础--Android JNI实例代码(一)-- 在JNI中执行Java方法--C/C++调用Java
- Eclipse下NDK开发(java调用so文件)
- NDK开发学习笔记(2):JNI访问Java中的方法
- Android(java)学习笔记259:JNI之NDK开发步骤
- NDK开发之c++调用java
- 【学习Android NDK开发】Java通过JNI调用native方法
- Android-NDK开发之第四个例子--用C/C++调用Java
- Android-java调用本地方法返回字符串显示在界面上/NDK-JNI开发实例(二)