NDK&JNI开发总结
NDK&JNI开发总结
简介
附个不错的博客 https://www.jianshu.com/p/87ce6f565d37
在Android Framework中,需要提供一种媒介或 桥梁,将Java层(上层)与C/C++层(下层)有机的联系起来,使得他们互相协调完成某些任务。而充当这种媒介的就是Java本地接口(JNI,Java Native Interface)。JNI提供一些列的接口,允许Java类与C/C++等本地编辑语言(在JNI中,这些语言被称为 本地语言)编写的应用 程序、模块 、库进行交互操作。比如,在Java类中使用C语言库中的函数或在C语言中使用 Java类库,都需要借助JNI。
Android NDK是一个开发工具集,提供一系列工具快速开发C/C++的动态库,并能自动将 .so/.dll 和 Java 应用一起打包到Apk;
NDK提供工具可以方便JNI调用C/C++,而且提供了交叉编译器可以修改.mk文件生成特定CPU平台的动态库,并能将so和java应用一起打包到apk中;简单说就是JNI负责Java与C/C++进行互相操作,NDK提供工具方便在Android平台使用JNI;
JNI的优点
- 可以调用c/c++代码,效率和性能好
- JNI编译的so库反编译难度更大
java调用c/c++代码
extern "C"
JNIEXPORT jstring JNICALL
native_test_ffmpeg(JNIEnv *env,jobject thiz){
return env->NewStringUTF("native_test_ffmpeg");
}
//nativeTest是java的native方法名,native_test_ffmpeg是c/c++的实现方法,"()Ljava/lang/String;"代表此方法输入参数为空,输出参数为String
static JNINativeMethod g_methods[] = {
{"nativeTest","()Ljava/lang/String;",(void*)native_test_ffmpeg}
};
//把c/c++方法和java方法关联起来,不需要按照“Java_包名_类名_方法名”规则来生成方法
jint JNI_OnLoad(JavaVM *vm, void *reserved) {
JNIEnv *env = NULL;
vm->GetEnv((void**)&env,JNI_VERSION_1_6);
jclass pJclass = env->FindClass(JNI_CLASS_PATH);
env->RegisterNatives(pJclass,g_methods,sizeof(g_methods)/sizeof(g_methods[0]));
return JNI_VERSION_1_6;
}
signature说明
输入参数在"()"内,输出参数在"()"外,多个参数顺序存放,数组用"["标识。例如:
([Student;)[Student; => Student[] xxx(Student[]){}
(I)S => short xxx(int i){}
基本数据类型对应符号如下:
boolean Z
byte B
char C
short S
int I
long L
float F
double D
void V
java类对应:
L包路径/类名
c/c++调用java代码
extern "C"
JNIEXPORT jstring JNICALL
Java_com_example_ffmpegtest_MainActivity_stringFromJNI2(JNIEnv *env, jobject thiz) {
//step1 找到类
jclass sJclass = env->FindClass(JNI_CLASS_PATH_2);
//step2 找到方法
jmethodID method_init_id = env->GetMethodID(sJclass, "<init>", "()V");//找到构造方法
jmethodID method_set_id = env->GetMethodID(sJclass, "setYear", "(I)V");//找到构造方法
jmethodID method_get_id = env->GetMethodID(sJclass, "getYear", "()I");//找到构造方法
//step3 new object
jobject obj = env->NewObject(sJclass, method_init_id);
//step4 调用相应方法
env->CallVoidMethod(obj,method_set_id,18);
jint year = env->CallIntMethod(obj, method_get_id);
char temp[] = {0,};
sprintf(temp,"%d",year);
std::string helloF = "hello ffmpeg from C++, year =";
helloF.append(temp);
return env->NewStringUTF(helloF.c_str());
}
CMakeLists.txt配置
cmake_minimum_required(VERSION 3.10.2)
# Declares and names the project.
#项目名称,也就是我安卓项目的名字
project("ffmpegtest")
# Creates and names a library, sets it as either STATIC
# or SHARED, and provides the relative paths to its source code.
# You can define multiple libraries, and CMake builds them for you.
# Gradle automatically packages shared libraries with your APK.
# 方式一(验证不通过)
# 引入FFmpeg的库文件,设置内部的方式引入,指定库的目录是 -L 指定具体的库-l
# 这种方式方便快捷
#set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -L../../../libs/${CMAKE_ANDROID_ARCH_ABI}")
#设置ffmpeg库所在路径的变量
set(FF ${CMAKE_SOURCE_DIR}/libs/${CMAKE_ANDROID_ARCH_ABI})
add_library(avcodec SHARED IMPORTED) #三个参数,第一个,库名称;第二个,SHARED动态库或STATIC静态库;第三个,IMPORTED或xx.cpp
set_target_properties(avcodec PROPERTIES IMPORTED_LOCATION ${FF}/libavcodec.so) #来源,此处是添加外部引入的库libs/libavcodec.so->avcodec
add_library(avformat SHARED IMPORTED)
set_target_properties(avformat PROPERTIES IMPORTED_LOCATION ${FF}/libavformat.so)
add_library(avutil SHARED IMPORTED)
set_target_properties(avutil PROPERTIES IMPORTED_LOCATION ${FF}/libavutil.so)
add_library(swscale SHARED IMPORTED)
set_target_properties(swscale PROPERTIES IMPORTED_LOCATION ${FF}/libswscale.so)
add_library(swresample SHARED IMPORTED)
set_target_properties(swresample PROPERTIES IMPORTED_LOCATION ${FF}/libswresample.so)
add_library(avfilter SHARED IMPORTED)
set_target_properties(avfilter PROPERTIES IMPORTED_LOCATION ${FF}/libavfilter.so)
add_library( native-lib
SHARED
${CMAKE_SOURCE_DIR}/src/main/cpp/native-lib.cpp)#来源是从native-lib.cpp文件编译
find_library( log-lib log )
find_library( android-lib android )
include_directories(${CMAKE_SOURCE_DIR}/libs/includes)
target_link_libraries( #目标库
native-lib
# 把ffmpeg的动态库以来进来
avformat
avcodec
avutil
avfilter
swscale
swresample
${log-lib}
${android-lib})
项目名称,也就是我安卓项目的名字
project("ffmpegtest")
三个参数,第一个,库名称;第二个,SHARED动态库或STATIC静态库;第三个,IMPORTED(外部引入)或xx.cpp(从CPP编译)
add_library( native-lib SHARED ${CMAKE_SOURCE_DIR}/src/main/cpp/native-lib.cpp)
add_library(avcodec SHARED IMPORTED)
来源,此处是添加外部引入的库libs/libavcodec.so->avcodec
set_target_properties(avcodec PROPERTIES IMPORTED_LOCATION ${FF}/libavcodec.so)
添加依赖的库
find_library( android-lib android )
依赖库文件路径
include_directories(${CMAKE_SOURCE_DIR}/libs/includes)
把动态库链接到一个库,第一个参数目标库,后面参数都是要链接的库
target_link_libraries( #目标库
native-lib
# 把ffmpeg的动态库以来进来
avformat
avcodec
avutil
avfilter
swscale
swresample
${log-lib}
${android-lib})
NDK&JNI开发总结的更多相关文章
- AndroidStudio如何配置NDK/JNI开发环境
参考文章: http://www.th7.cn/Program/Android/201509/550864.shtml http://www.open-open.com/lib/view/open14 ...
- 【android 开 发 】 - Android studio 下 NDK Jni 开发 简单例子
Android 开发了一段时间,一方面 ,感觉不留下点什么.有点对不起自己, 另一方面,好记性不如烂笔头,为了往后可以回头来看看,就当做是笔记,便决定开始写博客.废话不多说 ! 今天想搞一搞 ndk ...
- Android studio 下 NDK Jni 开发 简单例子
1. 创建一个新的工程 2. 创建一个新的类 JniText.java 点击Build--Make Project 后 选中工程 点击F4键 sdk location 中 Android ...
- Android Studio上NDK/JNI开发环境问题
基础环境: 操作系统 —— Windows 7 Android Studio —— 1.5.1(android-studio-bundle-141.2456560-windows.exe) NDK — ...
- NDK Jni 开发(1)
1. 学习地址 http://my.oschina.net/lifj/blog/177087 http://www.cnblogs.com/devinzhang/archive/2012/02/29/ ...
- Android游戏开发实践(1)之NDK与JNI开发03
Android游戏开发实践(1)之NDK与JNI开发03 前面已经分享了两篇有关Android平台NDK与JNI开发相关的内容.以下列举前面两篇的链接地址,感兴趣的可以再回顾下.那么,这篇继续这个小专 ...
- Android游戏开发实践(1)之NDK与JNI开发01
Android游戏开发实践(1)之NDK与JNI开发01 NDK是Native Developement Kit的缩写,顾名思义,NDK是Google提供的一套原生Java代码与本地C/C++代码&q ...
- Android游戏开发实践(1)之NDK与JNI开发02
Android游戏开发实践(1)之NDK与JNI开发02 承接上篇Android游戏开发实践(1)之NDK与JNI开发01分享完JNI的基础和简要开发流程之后,再来分享下在Android环境下的JNI ...
- 史上最易懂的Android jni开发资料--NDK环境搭建
谷歌改良了ndk的开发流程,对于Windows环境下NDK的开发,如果使用的NDK是r7之前的版本,必须要安装Cygwin才能使用NDK.而在NDKr7开始,Google的Windows版的NDK提供 ...
随机推荐
- 《Clojure编程》笔记 第2章 函数式编程
目录 背景简述 第2章 函数式编程 背景简述 本人是一个自学一年Java的小菜鸡,理论上跟大多数新手的水平差不多,但我入职的新公司是要求转Clojure语言的.坊间传闻:通常情况下,最好是有一定Jav ...
- logback日志打印sql
今天整合springboot2 + mybatis + logback 遇到了在日志中sql打印不出来的坑,在网上找了好久,都不是我遇到的问题,这里吐槽一下下现在的博客质量,好多都是抄袭的,也没有标注 ...
- Java入门(2)
阅读书目:Java入门经典(第7版) 作者:罗格斯·卡登海德 一个简单的计算平方根的程序: 1 package com.java24hours; 2 3 public class Root { 4 p ...
- Pytest学习(六) - conftest.py结合接口自动化的举例使用
一.conftest.py作用 可以理解成存放fixture的配置文件 二.conftest.py配置fixture注意事项 pytest会默认读取conftest.py里面的所有fixture co ...
- NIO源码分析:SelectionKey
SelectionKey SelectionKey,选择键,在每次通道注册到选择器上时都会创建一个SelectionKey储存在该选择器上,该SelectionKey保存了注册的通道.注册的选择器.通 ...
- 1redis介绍
一,概述 是一种nosql数据库,保存在内存中,同时redis可以把内存同时保存到磁盘,即可以把数据持久化.支持较多的数据类型,string,list(队列和栈),set,sorted set,has ...
- binary hacks读数笔记(file命令与magic file)
file命令的作用是用于检验文件的类型,并打印至终端.file命令检验文件类型按以下顺序来完成: 检验文件系统(Filesystem)中支持的文件类型. 检验magic file规则. 检验文件内容的 ...
- uboot——初始化阶段
start.S |-------------设置cpu状态 |--------------开cache |--------------获得启动方式 |------------------------- ...
- 磁盘构造/msdos分区(fdisk)格式化(mkfs)和挂载
分区不是必要的,分区是与系统盘分开,防止数据丢失. 磁盘使用流程:查看磁盘(fdisk -l)---分区---格式化(创建文件系统)----挂载(自动挂载) 分区表类型:msdos(一般是系统分区) ...
- mds的cpu占用问题分析以及解决办法
前言 mds是ceph里面处理文件接口的组件,一旦使用文件系统,不可避免的会出现一种场景就是目录很多,目录里面的文件很多,而mds是一个单进程的组件,现在虽然有了muti mds,但稳定的使用的大部分 ...