android的ndk学习(1)

android的ndk学习(1)

之前学了一段时间ndk,总觉得要总结一下。ndk使得非常方便地实现java和C与C++代码的相互沟通,合理地掌握使用ndk可以提高应用程序的执行效率,所以对于学习anndroid开发的人来说,ndk是必须掌握的工具。刚刚开始学习的时候是有点兴奋,有点害怕的,兴奋是因为之前学过C++语言,能将学过的东西结合在一起,感觉可以做出更好的东西,害怕的是之前听身边的大神说ndk在android开发中是非常难的内容之一。但是不管怎么说我还是找了本书,看了视频,找了一些电子资料,并且开始了学习ndk之路!

一,第一个程序Hello world

相对来说,使用ndk实现大量的原生方法并让他们与Java类同步很容易成为一个繁琐的任务。首先需要新建一个android项目,然后在主函数那里声明一个native方法,代码如下

public class MainActivity extends  Activity {

	public static native String test();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

    }

然后使用命令行,开始-cmd,切换到项目的src目录下再执行命令如下:

javah -d ../jni 包名.MainActivity

这时候刷新项目就会发现多了一个jni文件夹,里面有个.h的文件,打开就是一个c头文件

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

#ifndef _Included_com_example_exercise_MainActivity
#define _Included_com_example_exercise_MainActivity
#ifdef __cplusplus
extern "C" {
#endif
#undef com_example_exercise_MainActivity_MODE_PRIVATE
#define com_example_exercise_MainActivity_MODE_PRIVATE 0L
#undef com_example_exercise_MainActivity_MODE_WORLD_READABLE
#define com_example_exercise_MainActivity_MODE_WORLD_READABLE 1L
#undef com_example_exercise_MainActivity_MODE_WORLD_WRITEABLE
#define com_example_exercise_MainActivity_MODE_WORLD_WRITEABLE 2L
#undef com_example_exercise_MainActivity_MODE_APPEND
#define com_example_exercise_MainActivity_MODE_APPEND 32768L
#undef com_example_exercise_MainActivity_MODE_MULTI_PROCESS
#define com_example_exercise_MainActivity_MODE_MULTI_PROCESS 4L
#undef com_example_exercise_MainActivity_MODE_ENABLE_WRITE_AHEAD_LOGGING
#define com_example_exercise_MainActivity_MODE_ENABLE_WRITE_AHEAD_LOGGING 8L
#undef com_example_exercise_MainActivity_BIND_AUTO_CREATE
#define com_example_exercise_MainActivity_BIND_AUTO_CREATE 1L
#undef com_example_exercise_MainActivity_BIND_DEBUG_UNBIND
#define com_example_exercise_MainActivity_BIND_DEBUG_UNBIND 2L
#undef com_example_exercise_MainActivity_BIND_NOT_FOREGROUND
#define com_example_exercise_MainActivity_BIND_NOT_FOREGROUND 4L
#undef com_example_exercise_MainActivity_BIND_ABOVE_CLIENT
#define com_example_exercise_MainActivity_BIND_ABOVE_CLIENT 8L
#undef com_example_exercise_MainActivity_BIND_ALLOW_OOM_MANAGEMENT
#define com_example_exercise_MainActivity_BIND_ALLOW_OOM_MANAGEMENT 16L
#undef com_example_exercise_MainActivity_BIND_WAIVE_PRIORITY
#define com_example_exercise_MainActivity_BIND_WAIVE_PRIORITY 32L
#undef com_example_exercise_MainActivity_BIND_IMPORTANT
#define com_example_exercise_MainActivity_BIND_IMPORTANT 64L
#undef com_example_exercise_MainActivity_BIND_ADJUST_WITH_ACTIVITY
#define com_example_exercise_MainActivity_BIND_ADJUST_WITH_ACTIVITY 128L
#undef com_example_exercise_MainActivity_CONTEXT_INCLUDE_CODE
#define com_example_exercise_MainActivity_CONTEXT_INCLUDE_CODE 1L
#undef com_example_exercise_MainActivity_CONTEXT_IGNORE_SECURITY
#define com_example_exercise_MainActivity_CONTEXT_IGNORE_SECURITY 2L
#undef com_example_exercise_MainActivity_CONTEXT_RESTRICTED
#define com_example_exercise_MainActivity_CONTEXT_RESTRICTED 4L
#undef com_example_exercise_MainActivity_RESULT_CANCELED
#define com_example_exercise_MainActivity_RESULT_CANCELED 0L
#undef com_example_exercise_MainActivity_RESULT_OK
#define com_example_exercise_MainActivity_RESULT_OK -1L
#undef com_example_exercise_MainActivity_RESULT_FIRST_USER
#define com_example_exercise_MainActivity_RESULT_FIRST_USER 1L
#undef com_example_exercise_MainActivity_DEFAULT_KEYS_DISABLE
#define com_example_exercise_MainActivity_DEFAULT_KEYS_DISABLE 0L
#undef com_example_exercise_MainActivity_DEFAULT_KEYS_DIALER
#define com_example_exercise_MainActivity_DEFAULT_KEYS_DIALER 1L
#undef com_example_exercise_MainActivity_DEFAULT_KEYS_SHORTCUT
#define com_example_exercise_MainActivity_DEFAULT_KEYS_SHORTCUT 2L
#undef com_example_exercise_MainActivity_DEFAULT_KEYS_SEARCH_LOCAL
#define com_example_exercise_MainActivity_DEFAULT_KEYS_SEARCH_LOCAL 3L
#undef com_example_exercise_MainActivity_DEFAULT_KEYS_SEARCH_GLOBAL
#define com_example_exercise_MainActivity_DEFAULT_KEYS_SEARCH_GLOBAL 4L
/*
 * Class:     com_example_exercise_MainActivity
 * Method:    test
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_example_exercise_MainActivity_test
  (JNIEnv *, jclass);

/*
 * Class:     com_example_exercise_MainActivity
 * Method:    updateFile
 * Signature: (Ljava/lang/String;)V
 */
<pre name="code" class="java">JNICALL Java_com_example_exercise_MainActivity_updateFile
  (JNIEnv *, jclass, jstring)

#ifdef __cplusplus}#endif#endif


代码很长,但是我们暂时只要看

JNICALL Java_com_example_exercise_MainActivity_updateFile
  (JNIEnv *, jclass, jstring)

这就是根据我们一开始在mainactivity定义的那个native方法生成的一个方法。有了头文件,我们就可以开始写.c文件了,即实现文件,新建一个文件main.c,然后输入代码如下

#include<stdio.h>
#include<stdlib.h>
JNIEXPORT jstring JNICALL Java_com_example_exercise_MainActivity_test
(JNIEnv * env, jobject obj){

	return (*env)->NewStringUTF(env, "Hello world !");
}

返回一个字符串,这就是java与c交互的代码。

但是现在还不能直接运行,还要新建一个android.mk文件对项目进行配置,代码如下:

LOCAL_PATH :=$(call my-dir)

include $(CLEAR_VARS)
LOCAL_MODULE    := main
LOCAL_SRC_FILES := main.c
include $(BUILD_SHARED_LIBRARY)

好了,然后就是在mainactivity中使用方法了,代码如下

public class MainActivity extends  Activity {

	public static native String test();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        TextView t = (TextView)findViewById(R.id.jnitextview);
        t.setText(test());

    }
    static {
		System.loadLibrary("main");
	}

}

导入库是使用的

  static {
		System.loadLibrary("main");
	}

main是我们在android.mk中配置的一个名字,现在万事俱备,只差编译生成so文件了,我们打开cmd并且切换到项目的目录下,执行ndk-build,中间是减号,不是下划线,刷新项目就可以看到libs中多了个文件夹和里面的一个libmain.so文件,这时候就可以运行项目了!如果没有意外就会出现helloworld在手机频幕上。

二,打印log

打印log是必须掌握的只是,所以这里介绍一下怎么配置,首先是配置android.mk文件,添加一行代码

LOCAL_LDLIBS    += -llog

完整的android.mk代码如下

LOCAL_PATH :=$(call my-dir)

include $(CLEAR_VARS)
LOCAL_MODULE    := main
LOCAL_SRC_FILES := main.c
LOCAL_LDLIBS    += -llog
include $(BUILD_SHARED_LIBRARY)

然后在实现文件中添加头文件#include<android/log.h>

并且宏定义要打印log的类型

#define LOGI(...) ((void)__android_log_print(ANDROID_LOG_INFO, "native-activity", __VA_ARGS__))

#define LOGW(...) ((void)__android_log_print(ANDROID_LOG_WARN, "native-activity", __VA_ARGS__))

在代码中使用LOGI()或者LOGW,运行程序,然后就可以打印log了!更加详细的可以去看java api文档中的ndk篇。

三,总结

刚开始学jni,要配置这个要配置那个的非常麻烦,但是写了一个helloworld以后感觉配置也就那样,万事开头难啊!相信后面的学习会越来越难,但是也会越来越有意思,希望继续加油!

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-08-10 00:03:57

android的ndk学习(1)的相关文章

Android Studio NDK 学习之接受Java传入的Int数组

本博客是基于Android Studio 1.3 preview版本,且默认你已经安装了Android SDK, Android NDK. 用Android Studio新建一个工程叫AndroidJNI_IntArray,其目录结构如下: ├── AndroidJNI_IntArray.iml ├── app │   ├── app.iml │   ├── build │   ├── build.gradle │   ├── libs │   ├── proguard-rules.pro │ 

Android Studio NDK 学习之接受Java传入的字符串

本博客是基于Android Studio 1.3 preview版本,且默认你已经安装了Android SDK, Android NDK. 用Android Studio新建一个工程叫Prompt,其目录结构如下: ├── Prompt.iml ├── app │   ├── app.iml │   ├── build │   ├── build.gradle │   ├── libs │   ├── proguard-rules.pro │   └── src ├── build │   └─

Android NDK学习笔记(一) 为什么要用NDK?

NDK是什么 NDK是Native Development Kit的简称,即本地开发工具包.通过NDK,Android允许开发人员使用本地代码语言(例如C/C++)来完成应用的部分(甚至全部)功能.注意:由于翻译原因,有些地方也把Native翻译为"原生". NDK是SDK的一个补充,可以帮助你做这些事情: 生成可以在ARM CPU,Android 1.5(及以上)平台运行的JNI兼容的共享库. 将生成的共享库放置在应用程序项目路径的合适位置,使其能自动地添加进你最终的(和经过签名的)

android的JNI 、 NDK 学习!

转载的! Java Native Interface (JNI)标准是java平台的一部分,它允许Java代码和其他语言写的代码进行交互.JNI 是本地编程接口,它使得在 Java 虚拟机 (VM) 内部运行的 Java 代码能够与用其它编程语言(如 C.C++ 和汇编语言)编写的应用程序和库进行交互操作. 1.从如何载入.so档案谈起 由于Android的应用层的类都是以Java写的,这些Java类编译为Dex型式的Bytecode之后,必须靠Dalvik虚拟机(VM: Virtual Mac

【转】 Android的NDK开发(1)————Android JNI简介与调用流程

原文网址:http://blog.csdn.net/conowen/article/details/7521340 ******************************************************************************************** * author:[email protected]大钟                                                                      

android的NDK和java进行本地socket通信

关于Android应用与Framework的socket通信,相信关心这个问题的朋友们已经看过<android使用socket使底层和framework通信>这篇文章,美中不足的是作者只贴出一些关键的代码片段而并没有放出源码.我这里还是以一个能实际运行的例子为基础来讲,这样也方便大家学习. 首先看一下效果,如下图.我填写姓名"Potter",选择性别"Mr"然后点击发送,底层socket收到消息后将消息直接返回给我,我将返回的结果(Mr.Potter)直

NDK学习4: Eclipse HelloWorld

NDK学习4: Eclipse HelloWorld 1.配置Eclipse NDK环境? Window->preferences->android->ndk ? 2.新建Android工程 在工程目录上点右键->Android Tools-> add native support 这个时候工程会多出一个jni目录 ? 3.编辑Hello.cpp #include?<stdio.h> int?main(int?argc,?char*?argv[]) { ??pri

Android之NDK开发

一篇文章: Android之NDK开发 一.NDK产生的背景 Android平台从诞生起,就已经支持C.C++开发.众所周知,Android的SDK基于Java实现,这意味着基于Android SDK进行开发的第三方应用都必须使用Java语言.但这并不等同于“第三方应用只能使用Java”.在Android SDK首次发布时,Google就宣称其虚拟机Dalvik支持JNI编程方式,也就是第三方应用完全可以通过JNI调用自己的C动态库,即在Android平台上,“Java+C”的编程方式是一直都可

Android JNI/NDK开发之基本姿势&lt;一&gt;

开发环境信息 列举下本篇文章编写的Demo基本信息 操作系统 Windows 10 家庭中文版 开发工具 Android Studio 2.1 SDK new NDK new 扫盲之SDK.JDK.NDK的区别 SDK 软件开发工具包:英语全称:Software Development Kit JDK Java语言的软件开发工具包:英语全称:Java Development Kit NDK 原生软件开发工具包:英语全称:Native Development Kit:被Google称为NDK 由此