01.Jni开发流程_java调用C/C++

(创建于2017/11/8)

JNI(Java Native Interface)
Java调用C/C++,C/C++调用Java的一套API

1.编写native方法

public class JniUtils {
    public static native String getStringFromC();
    public native String getStringFromC2();
}

2.javah命令,生成.h头文件

cd 进入到src目录下,使用命令生成头文件 javah 包名+类型(如 com.renzhenming.utils.JniUtils)

3.复制.h头文件到CPP工程中

右键->添加现有项->将头文件添加到vs中的头文件目录中,不要直接复制,直接复制无效

4.复制jni.h和jni_md.h文件到CPP工程中

注意include的时候,<>与“”的灵活使用,假设生成的头文件是这样的(有时候Javah命令生成的头文件中没有方法名,只有一些预编译的东西,不知何故)

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

#ifndef _Included_com_renzhenming_bsdiff_JniUtils
#define _Included_com_renzhenming_bsdiff_JniUtils
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_renzhenming_bsdiff_JniUtils
 * Method:    getStringFromC
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_renzhenming_bsdiff_JniUtils_getStringFromC
  (JNIEnv *, jclass);

/*
 * Class:     com_renzhenming_bsdiff_JniUtils
 * Method:    getStringFromC2
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_renzhenming_bsdiff_JniUtils_getStringFromC2
  (JNIEnv *, jobject);

#ifdef __cplusplus
}
#endif
#endif

5.实现.h头文件中声明的函数

假设如上边一样,我们生成了头文件,那么我们需要在自己的c文件中定义实现这两个方法

#include "com_renzhenming_bsdiff_JniUtils.h"    //引入头文件

JNIEXPORT jstring JNICALL Java_com_renzhenming_bsdiff_JniUtils_getStringFromC
(JNIEnv *env, jclass jcls) {
    return (*env)->NewStringUTF(env, "aaaaaaa");  //得到字符串
}

JNIEXPORT jstring JNICALL Java_com_renzhenming_bsdiff_JniUtils_getStringFromC2
(JNIEnv *env, jobject jobj) {
    return (*env)->NewStringUTF(env, "bbbbbb");//得到字符串
}

6.visual studio生成dll文件
37761343.png

如上点击debug出现下拉框,选择配置管理器,将我们的活动解决方案设置为x64


37833718.png

右键项目名->属性,设置常规下的项目默认值下的配置类型选择动态库(.dll),然后


37905203.png

点击生成,选择生成解决方案,这时候我们的dll动态库就生成在指定目录中了
37963765.png

7.vs生成dll动态库的时候容易出现的问题和解决方法

1.使用了过时的函数

严重性 代码  说明  项目  文件  行   禁止显示状态
错误  C4996   'open': The POSIX name for this item is deprecated. Instead, use the ISO C and C++ conformant name: _open. See online help for details. ndk_update  c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 263 
    
解决办法:给所在的类添加宏定义
define _CRT_NONSTDC_NO_DEPRECATE
或者给整个项目添加右键项目名->属性->配置属性->c/c++->命令行,在其他选项中添加 -D _CRT_NONSTDC_NO_DEPRECATE,确定即可,这样会给整个项目设置这个配置,不再需要在每个类中分别添加了
2.使用了不安全的函数

严重性 代码  说明  项目  文件  行   禁止显示状态
错误  C4996   'open': This function or variable may be unsafe. Consider using _sopen_s instead. To disable deprecation, use _CRT_SECURE_NO_WARNINGS. See online help for details. ndk_update  c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 263 
    
解决办法:同上的方式添加宏定义,_CRT_SECURE_NO_WARNINGS
3.安全性检查相关的问题

bsdiff是外国程序员写的,可能使用的工具不是vs,而vs对代码安全性检查比较严格,所以导致这个问题
在linux系统下不会报这个错误

严重性 代码  说明  项目  文件  行   禁止显示状态
错误  C4703   使用了可能未初始化的本地指针变量“old”   ndk_update  c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 266 
错误  C4703   使用了可能未初始化的本地指针变量“V” ndk_update  c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 273 
错误  C4703   使用了可能未初始化的本地指针变量“_new”  ndk_update  c:\users\renzhenming\documents\visual studio 2015\projects\ndk_update\ndk_update\bsdiff.cpp 295 
    
解决办法:右键项目名->属性->配置属性->c/c++->常规->SDL检查选择否
4.重复引用的问题

这个错误的原因是因为项目中既添加了bsdiff.cpp也添加了bspatch.cpp文件,导致出现定义的重复问题
也就是二者不可同时存在,移除一个

严重性 代码  说明  项目  文件  行   禁止显示状态
错误  LNK2005 "void __cdecl err(int,char const *)" (?err@@YAXHPEBD@Z) 已经在 bsdiff.obj 中定义  ndk_update  c:\Users\renzhenming\Documents\Visual Studio 2015\Projects\ndk_update\ndk_update\bspatch.obj    1   
错误  LNK2005 "void __cdecl errx(int,char const *)" (?errx@@YAXHPEBD@Z) 已经在 bsdiff.obj 中定义    ndk_update  c:\Users\renzhenming\Documents\Visual Studio 2015\Projects\ndk_update\ndk_update\bspatch.obj    1   
错误  LNK1169 找到一个或多个多重定义的符号  ndk_update  c:\Users\renzhenming\Documents\Visual Studio 2015\Projects\ndk_update\x64\Debug\ndk_update.exe  1   
    
解决办法:针对这个项目的编译,不要让两者同时存在

8.配置dll文件所在目录到环境变量

我们可以设置一个专门的目录来存放我们的dll动态库,然后将这个目录加入环境变量,当然我们也可以直接将dll文件复制到我们eclipse中Java工程的根目录下,然后load

9.重启Eclipse,加载动态库调用c方法

如果设置了环境变量,那么需要重启eclipse,如果直接复制到工程下的,则不需要,然后我们就可以调用了,调用代码如下


//工具类
package com.renzhenming.bsdiff;

public class JniUtils {
    
    static {
        System.loadLibrary("JniTest");
    }
    public static native String getStringFromC();
    public native String getStringFromC2();
}

//main函数调用
public class DemoJni {

    public static void main(String[] args) {
        
        String value = JniUtils.getStringFromC();
        System.out.println(value);
        JniUtils u = new JniUtils();
        String value2 = u.getStringFromC2();
        System.out.println(value2);
    }

}

C的函数名称:Java_完整类名_函数名

.a
.dll 共享代码

JNIEnv

在C中:
JNIEnv 结构体指针别名
env二级指针

在C++中:
JNIEnv 是一个结构体的别名
env 一级指针

1.为什么需要传入JNIEnv,函数执行过程中需要JNIEnv
2.C++为什么没有传入?this
3.C++只是对C的那一套进行的封装,给一个变量赋值为指针,这个变量是二级指针

为什么c中JNIEnv是二级指针,而C++中是一级指针

Jni方法中的参数解释:

//函数实现
JNIEXPORT jstring JNICALL Java_com_dongnaoedu_jni_JniTest_getStringFromC
(JNIEnv *env, jclass jcls){
    //JNIEnv 结构体指针
    //env二级指针
    //代表Java运行环境,调用Java中的代码
    //在C中:
    //JNIEnv 结构体指针别名
    //env二级指针
    //在C++中:
    //JNIEnv 是一个结构体的别名
    //env 一级指针
    
    //jclass 是一个结构体指针,代表native方法所属类的class对象(XXX.class)
    //访问这个类的属性方法都需要用到这个jclass
    //typedef struct _jobject *jobject;
    //typedef jobject jclass;
    return (*env)->NewStringUTF(env,"C String");
    
    
    //每个native函数都至少有两个参数(JNIEnv*,jclass或者jobject)
    //1.当native方法为静态方法时,jclass代表native方法所属类的class对象(XXX.class)
    //2.当native方法为非静态时,jobject代表native方法所属的对象
    
    如下:
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350

推荐阅读更多精彩内容