w7+eclipse +ndk9 NDK开发 hellow ndk 笔记(java调用c)

本文详细介绍使用Eclipse进行Android NDK开发的过程,包括环境搭建、JNI调用、C/C++代码编写及编译等关键步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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文件

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变量必须包含将要编译打包进模块中的CC++源代码文件。注意,你不用在这里列出头文件和包含文件,因为编译系统将会自动为你找出依赖型的文件;仅仅列出直接传递给编译器的源代码文件就好。【注意,默认的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.youkuaiyun.com/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.youkuaiyun.com/detail/a56573016613/6217343


http://blog.youkuaiyun.com/a56573016613/article/details/14224685 c调用java

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值