• 1、定义本地【native】方法。通常情况下,应单独定义一个类来封装所有native方法。native方法相当于一个【接口】中的方法,只有方法声明,没有方法体。
  • 2、在项目根目录下创建【jni文件夹】,将利用【javah】命令生成的【.h】头文件拷到jni目录中。然而在实际测试中发现,这个文件没有任何卵用,不拷贝也没问题。
  • 3、在jni目录中编写所需的【.c】文件,这一步才是核心,在c代码中,实现上面定义的本地方法。其中ndk工具中提供的<jni.h>头文件中详细定义了JNI【基本数据类型】和【本地等效类型】之间的转换。
  • 4、在jni目录中创建【Android.mk】文件,主要是声明所引用的.c文件和生成的.so库的文件名
  • 5、创建【Application.mk】文件,主要是声明所有支持的平台。每增加一个支持的架构,编译后就会在lib目录下生成一个相应架构平台的目录。
  • 6、使用【ndk】工具编译生成【.so动态链接库文件】。当完成这一步后,如果我们以后不需要再重新编译,我们就可以直接删除【jni】目录。另外发布的时候这些文件都是应该删掉的。
  • 7、使用时先 System.loadLibrary("hello"),然后直接调用本地native方法即可。注意,加载库文件时的文件名要去掉前面的lib和后面的.so

1、定义本地【native】方法

定义本地方法,通常情况下,应单独定义一个类来封装所有native方法
/** 存放native方法的类 */
public class MyNativeMethods {
    private static MyNativeMethods mEmployee;
    private MyNativeMethods() {
    }
    public static MyNativeMethods getInstance() {
        if (mEmployee == null) {
            mEmployee = new MyNativeMethods();
        }
        return mEmployee;
    }
    //相当于在java代码中定义了一个接口,然后用C语言实现了此接口
    public native String helloFromC();
    public native int passwordFromC(int x, int y);
}

2、利用javah命令生成的.h头文件

1、定位到工程的【 src目录】下
cd/d E:\HelloFromC\src

2、对native方法所在类执行【javah】命令,其中javah后面的类文件的格式是【包名.类名】
javah com.bqt.hellofromc.MyNativeMethods
执行过程如下:

报错的原因是因为MyNativeMethods中有中文字符,即使是注释也会报错,且发现一处中文就报错一次
但是这并不影响处理结果,所以大可不用理会

3、在项目根目录下创建【jni文件夹】,将生成的【.h文件】拷贝到jni目录下
在实际测试中发现,这个文件没有任何卵用,不拷贝也没问题。可能是因为我测试的代码比较简单吧。


文件的内容为
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_bqt_hellofromc_MyNativeMethods */
#ifndef _Included_com_bqt_hellofromc_MyNativeMethods
#define _Included_com_bqt_hellofromc_MyNativeMethods
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_bqt_hellofromc_MyNativeMethods
 * Method:    helloFromC
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_bqt_hellofromc_MyNativeMethods_helloFromC
  (JNIEnv *, jobject);
/*
 * Class:     com_bqt_hellofromc_MyNativeMethods
 * Method:    passwordFromC
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_com_bqt_hellofromc_MyNativeMethods_passwordFromC
  (JNIEnv *, jobject, jint, jint);
#ifdef __cplusplus
}
#endif
#endif  

3、在jni目录中编写所需的.c文件【这一步才是核心】

在项目根目录下创建【jni文件夹】,在jni文件夹中创建一个【.c文件】,在c代码中,实现上面定义的本地方法
在.c文件中引入上面生成的.h文件(非必须,个人觉得有用的东西就是,它帮我们自动生成了符合JNI规范的方法声明,我们只需把方法声明拷走就行了)
#include "com_bqt_hellofromc_MyNativeMethods.h"  

以下为.c中的代码
#include <stdio.h>
#include <stdlib.h>
#include <jni.h>//必须添加的头文件

jstring Java_com_bqt_hellofromc_MyNativeMethods_helloFromC(JNIEnv* env, jobject obj) { //【返回值】【方法名】【参数列表】返回值类型jstring就是java中的string
    char* cstr = "hello from c"; //  char*  在c中可用来表示一个字符串。注意,这里绝对不能有中文
    jstring jstr = (*env)->NewStringUTF(env, cstr);
    return jstr;
}

JNIEXPORT jint JNICALL Java_com_bqt_hellofromc_MyNativeMethods_passwordFromC(JNIEnv *env, jobject obj, jint a, jint b) {
//JNIEXPORT和JNICALL都是JNI的关键字,表明函数是被JNI调用的;JNIEXPORT 表示输出类型;JNICALL表示参数的压栈顺序;貌似都可以省略
    return a + b + 10000; //c中的int占用字节数在不同环境下可能不同,可能是0-65535,所以,稍微大一点的数(十万级别)都得用double
}

4、在jni目录中创建Android.mk文件,主要是声明所引用的.c文件和生成的.so库的文件名

在工程的jni目录下创建一个【Android.mk文件】,在里面定义打包成函数【库的名字】及对应的【c代码的文件名】
LOCAL_PATH := $(call my-dir)
# C/C++代码所在目录,也就是我们的jni目录,不必修改
include $(CLEAR_VARS)
LOCAL_MODULE    := hello
# 对应打包成函数库的名字,编译器会自动在前面加上lib,在后面加上.so,最终结果就是libhello.so
LOCAL_SRC_FILES := hello.c
# 对应的c代码的文件名,即hello.c  
include $(BUILD_SHARED_LIBRARY)

5、创建Application.mk文件,主要是声明所有支持的平台

默认只会生成支持arm平台的动态链接库文件,若项目要支持arm外的平台,需在jni目录中添加【Application.mk】文件,并加上以下内容
APP_ABI := armeabi armeabi-v7a x86
#Application.mk文件的目的是,描述在你的应用程序中所有需要的模块(即静态库或动态库)
#APP_ABI 的值以空格区分,代表要支持的架构,默认值为【armeabi】。其他架构,ARMv7 【armeabi-v7a】;IA-32【 x86】
#每增加一个架构,编译后都会在lib目录下生成一个相应的文件夹,文件夹下的文件都是同名的.so文件(当然文件内容不一样)

编译成功后,会在lib目录下生成对应的多个文件夹及.so文件

6、使用ndk工具编译生成.so成动态链接库文件

可以使用cygwin工具编译,其中下面第三步中为Android工程的根目录
cd ../.. cd cygdrive/ cd D/Users/Android_workspace/HelloFromC ndk-build 或直接用cmd编译,注意要先在"系统变量path"中增加NDK工具所在路径,如【D:\Android\android-ndk-r10d】 cd/d D:\Users\Android_workspace\HelloFromC ndk-build
过程:


编译成功后会在工程libs/armeabi目录下生成一个【libhello.so】文件
其中,eabi的含义为:Embedded Application Binary Interface(嵌入式应用二进制接口)

细心观察会发现,除了自动生成了libs目录下的.so文件外,还产生了一个比libs目录大很多倍的obj目录,并且其结构及内容也很像

网上找到的资料解释为:
As part of the build process过程, the files in the libs folder have been stripped剥皮 of symbols符号 and debugging information. So you'll want to keep two copies of each of your .so files: One from the libs folder to install on the Android device, and one from the obj folder to install for GDB一个调试工具 to get symbols from.

简单来说就是,obj下的是带符号和调试信息的,lib下的是去掉这些庞大信息后的动态链接库文件,在安装到Android设备上时只需libs目录下的.so文件即可

当完成这一步后,如果我们以后不需要再重新编译,我们就可以直接删除【jni】目录和【obj】目录了

以下是是发布前的工程结构


7、使用时,直接调用本地native方法即可

在java代码中加载.so类库,之后就可以调用本地方法了
public class MainActivity extends ListActivity {
    static {
        System.loadLibrary("hello");// 在java代码中引入libs目录下的库函数,文件名为【libhello.so】。注意,引入时的文件名要去掉前面的lib和后面的.so
    }

    private TextView tv_info;
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        String[] array = { "调用C中的无参方法,返回一个字符串", "调用C中的有参方法,返回处理1+2后的值", };
        tv_info = new TextView(this);
        tv_info.setTextColor(Color.BLUE);
        tv_info.setTextSize(TypedValue.COMPLEX_UNIT_SP, 16);
        tv_info.setPadding(20, 10, 20, 10);
        getListView().addFooterView(tv_info);
        setListAdapter(new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, new ArrayList<String>(Arrays.asList(array))));
    }
    @Override
    protected void onListItemClick(ListView l, View v, int position, long id) {
        switch (position) {
        case 0:
            String stringFromC = MyNativeMethods.getInstance().helloFromC();
            tv_info.setText(stringFromC);
            break;
        case 1:
            int intFromC = MyNativeMethods.getInstance().passwordFromC(1, 2);
            tv_info.setText(intFromC+"");
            break;
        }
    }


附件列表

JNI 详细使用 基础【步骤】的更多相关文章

  1. android之ListView,详细介绍实现步骤,举例,自定义listview适配器

    android之ListView,详细介绍实现步骤,举例,自定义listview适配器 本文来源于www.ifyao.com禁止转载!www.ifyao.com android中如何使用listVie ...

  2. 部署Bookinfo示例程序详细过程和步骤(基于Kubernetes集群+Istio v1.0)

    部署Bookinfo示例程序详细过程和步骤(基于Kubernetes集群+Istio v1.0) 部署Bookinfo示例程序   在下载的Istio安装包的samples目录中包含了示例应用程序. ...

  3. 测试那些事儿—Linux搭建环境基础步骤

    Linux搭建环境基础步骤 准备工具:SecureCRT工具(Linux工具,连接服务器)FTP传输工具(上传文件到服务器)MySQL连接工具 安装包(以下文件均为压缩包rpm格式和tar.gz):J ...

  4. 接下来将介绍C#如何设置子窗体在主窗体中居中显示,本文提供详细的操作步骤,需要的朋友可以参考下

    接下来将介绍C#如何设置子窗体在主窗体中居中显示,本文提供详细的操作步骤,需要的朋友可以参考下 其实表面上看是很简单的 开始吧,现在有两个窗体Form1主窗体,Form2子窗体 而且我相信大部分人都会 ...

  5. 详细故障排除步骤:针对 Azure 中到 Windows VM 的远程桌面连接问题

    本文提供详细的故障排除步骤,用于为基于 Windows 的 Azure 虚拟机诊断和修复复杂的远程桌面错误. Important 若要消除更常见的远程桌面错误,请务必先阅读远程桌面的基本故障排除文章, ...

  6. JNI 详细使用步骤 上手示例

    1.定义本地native方法 定义本地方法,通常情况下,应单独定义一个类来封装所有native方法 /** 存放native方法的类 */ public class MyNativeMethods { ...

  7. js最详细的基础,jquery 插件最全的教材

    一.Js的this,{},[] this是Javascript语言的一个关键字,随着函数使用场合的不同,this的值会发生变化.但是有一个总的原则,那就是this指的是调用的函数自己. { } 大括号 ...

  8. Android(java)学习笔记259:JNI之NDK开发步骤

    1. NDK开发步骤(回忆一下HelloWorld案例): (1)创建工程 (2)定义native方法 (3)创建jni文件夹 (4)创建c源文件放到jni文件夹 (5)拷贝jni.h头文件到jni目 ...

  9. JNI 详细解释

    JNI事实上,Java Native Interface缩写,那是,java本地接口.它提供了许多API实现和Java和其它语言的通信(主要是C&C++). 或许不少人认为Java已经足够强大 ...

随机推荐

  1. 迭代器模式 与 C# IEnumerator/IEnumerable

    Part1 迭代器模式 与 接口 IEnumerable IEnumerator interface IEnumerable { IEnumerator GetEnumerator(); } // 泛 ...

  2. 自然语言处理系列-4条件随机场(CRF)及其tensorflow实现

    前些天与一位NLP大牛交流,请教其如何提升技术水平,其跟我讲务必要重视“NLP的最基本知识”的掌握.掌握好最基本的模型理论,不管是对日常工作和后续论文的发表都有重要的意义.小Dream听了不禁心里一颤 ...

  3. JavaScript基础-DAY1

    JavaScript介绍 你不知道它是什么就学?这就是一个网页嵌入式脚本语言...仅此而已 JavaScript组成 一个完整的 JavaScript 实现是由以下 3 个不同部分组成的: 核心(EC ...

  4. 1008 Elevator (20)(20 point(s))

    problem The highest building in our city has only one elevator. A request list is made up with N pos ...

  5. Linux驱动之IIC总线

    <作用> 电子设备中有很多IIC设备之间需要进行相互通信,这样就产生了IIC总线,常用来实现设备之间的数据通信.   <IIC总线结构> IIC总线只有两条线,一条是串行数据线 ...

  6. django组件之contenttype(一)

    方式1:适用于1张表和另一张表要关联的时候. 1.路飞学成表设计: 2.将2个价格策略表合并1张表. 3.如果再加一张表,那价格策略表的表结构会发生改变.  这样不合理的,我们的表结构一般设计完就不会 ...

  7. BZOJ 3572: [Hnoi2014]世界树 虚树 树形dp

    https://www.lydsy.com/JudgeOnline/problem.php?id=3572 http://hzwer.com/6804.html 写的时候参考了hzwer的代码,不会写 ...

  8. NEUQ OJ 2004:追梦之人 (计数数位dp)

    2004: 追梦之人 描述 题目描述: 为了纪念追梦人,粉丝们创造了一种新的数——“追梦数”.追梦数要满足以下两个条件:1.数字中不能出现“7”2.不能被7整除.比如:777和4396就不是追梦数,而 ...

  9. POJ 2356 Find a multiple 抽屉原理

    从POJ 2356来体会抽屉原理的妙用= =! 题意: 给你一个n,然后给你n个数,让你输出一个数或者多个数,让这些数的和能够组成n: 先输出一个数,代表有多少个数的和,然后再输出这些数: 题解: 首 ...

  10. 介绍一下开源项目FastAnimationWithPOP

    介绍一下开源项目FastAnimationWithPOP JUL 23RD, 2014 这是一个非常easy的动画框架,基于Facebook的POP库. 使用它你就能够在故事版中以0行代码的代价来加入 ...