android ndk-build 编译静态库libxx.a 以及Android studio openssl 静态库配置(cmake)
android ndk-build 编译静态库libxx.a
需求场景:
目前有安卓编码好的现在的openssl的两个.a,我们需要调用openssl的函数,并把功能再封装成.a;
这样使用时,在android studio jni项目 cmake 里面,需要先引用openssl的.a再引用 上面封装的.a;
如果使用so,那么,直接在android studio jni项目的 cpp里面直接调用openssl的方法,对外提供jni java接口,打包成so即可;
先来说用ndk-build 打包 libxx.a吧,并在打包的时候引用openssl的静态库.a
1. 首先当前目录下准备好 编码好的openssl静态库,如下图示,安卓平台的

2. 测试使用的源文件,我这里在.h声明一个封装的方法 ssl_sha,并且在.c 实现里面,调用 openssl 的SHA256接口;如下
加 #ifdef __cplusplus 是为了在jni调用时,cpp里面出现未定义的错误,如下 ssl_hash.h
#ifndef ssl_hash_h
#define ssl_hash_h # ifdef __cplusplus
extern "C" {
#endif #include <stdio.h> /*
* openssl sha256
* */
int ssl_sha256(const char *in_data,int in_len,char *out_hash); # ifdef __cplusplus
}
# endif #endif /* ssl_hash_h */
如下 ssl_hash.c 实现文件
#include "ssl_hash.h"
#include "openssl/sha.h" int ssl_sha256(const char *in_data,int in_len,char *out_hash)
{
if (!in_data) {
return -;
} if (!SHA256((const unsigned char *)in_data, in_len, (unsigned char *)out_hash)) {
return -;
}
return ;
}
同样这两文件也要放在 .mk的同一目录;
3. 开始编写 Android.mk,Application.mk, 如下示例
#Android.mk文件 LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
#编码后生成的.a的名称
LOCAL_MODULE := sslhash
#要编码的源文件c或cpp
LOCAL_SRC_FILES := ssl_hash.c
#依赖的静态库.a,若要使用动态库使用LOCAL_SHARED_LIBRARIES .so
LOCAL_STATIC_LIBRARIES := libs/${TARGET_ARCH_ABI}/lib/libssl.a libs/${TARGET_ARCH_ABI}/lib/libcrypto.a
#依赖的静态库的头文件
LOCAL_C_INCLUDES := libs/${TARGET_ARCH_ABI}/include
#最后生成静态库
include $(BUILD_STATIC_LIBRARY)
#Application.mk 这里测试,只编译出x86的 APP_ABI := x86
4. 所有文件都准备好之后如下图示:

5. 调用ndk-build 开始编译
~/Library/Android/sdk/ndk-bundle/ndk-build NDK_PROJECT_PATH=./ NDK_LIBS_OUT=./ APP_BUILD_SCRIPT=./Android.mk NDK_APPLICATION_MK=./Application.mk
6. 成功之后会在当前目录下生成 obj目录,里面包含生成的 静态库libxx.a

7. 编译好之后,下面需要在Android studio 中测试;用as 创建一个jni项目;
8.在工程build.gradle(Module:app) 里面添加 libs目录 (Android 视图)
externalNativeBuild {
cmake {
path "CMakeLists.txt"
}
}
sourceSets {
main {
jniLibs.srcDirs = ['libs']
}
}
并且在指定测试只编译 x86
externalNativeBuild {
cmake {
cppFlags ""
abiFilters "x86"
}
}
ndk{
abiFilters "x86"
}
9. 将之前libs目录的openssl库全部copy到 工程 jnilibs目录 (Android 视图)
并把生成的libsslhash.a放到 x86/lib下
并把ssl_hash.h头文件放到x86/include下
放到哪都行,需要在cmake里面指定引用好

10. 打开CmakeList.txt 配置openssl静态库的引用,以及配置引用生成的libsslhash.a
如下完善的cmakelist
cmake_minimum_required(VERSION 3.4.)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=gnu++11")
#这里配置指定目录libs
set(OpenSSL_DIR ${CMAKE_SOURCE_DIR}/libs) # 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. #这里引用三个静态.a, ssl,sslhash,crypto
add_library(crypto STATIC IMPORTED)
add_library(ssl STATIC IMPORTED)
add_library(sslhash STATIC IMPORTED) # 这里加载,并且找到相应的 libxxx.a
set_target_properties( # Specifies the target library.
crypto # Specifies the parameter you want to define.
PROPERTIES IMPORTED_LOCATION # Provides the path to the library you want to import.
${OpenSSL_DIR}/${ANDROID_ABI}/lib/libcrypto.a ) set_target_properties( # Specifies the target library.
ssl # Specifies the parameter you want to define.
PROPERTIES IMPORTED_LOCATION # Provides the path to the library you want to import.
${OpenSSL_DIR}/${ANDROID_ABI}/lib/libssl.a ) set_target_properties( # Specifies the target library.
sslhash # Specifies the parameter you want to define.
PROPERTIES IMPORTED_LOCATION # Provides the path to the library you want to import.
${OpenSSL_DIR}/${ANDROID_ABI}/lib/libsslhash.a) add_library( # Sets the name of the library.
native-lib # Sets the library as a shared library.
SHARED # Provides a relative path to your source file(s).
src/main/cpp/native-lib.cpp ) # Searches for a specified prebuilt library and stores the path as a
# variable. Because CMake includes system libraries in the search path by
# default, you only need to specify the name of the public NDK library
# you want to add. CMake verifies that the library exists before
# completing its build. find_library( # Sets the name of the path variable.
log-lib # Specifies the name of the NDK library that
# you want CMake to locate.
log ) # Specifies libraries CMake should link to your target library. You
# can link multiple libraries, such as libraries you define in this
# build script, prebuilt third-party libraries, or system libraries. #这里openssl需要 zlib库,加载系统的
find_library( z-lib z ) #这里指定一下咱的头文件目录,openssl头文件,以及咱封装的ssl_hash.h头文件
include_directories( ${OpenSSL_DIR}/${ANDROID_ABI}/include
${OpenSSL_DIR}/${ANDROID_ABI}/
) #最后一步连接,这里注意,一定要把sslhash,咱封装的这个.a放到ssl,crypto的前面,不然报错说方法未定义,可能因为sslhash引用了openssl的东西,
target_link_libraries( # Specifies the target library.
native-lib # Links the target library to the log library
# included in the NDK.
sslhash
ssl
crypto
${log-lib}
${z-lib})
11. 配置完成,如果编译不出问题就可以写jni调用测试了;
在 MainActitity.java里面,添加一下jni方法
public native String stringFromJNI();
//添加的测试方法
public native int sslsha(byte[] indata,int inlen,byte[] outhash);
12. 在native-lib.cpp 里面实现jni方法
#include "ssl_hash.h" extern "C"
JNIEXPORT jint JNICALL
Java_androidapp_cocoajin_com_tjni_MainActivity_sslsha(JNIEnv *env, jobject instance,
jbyteArray indata_, jint inlen,
jbyteArray outhash_) {
jbyte *indata = env->GetByteArrayElements(indata_, NULL);
jbyte *outhash = env->GetByteArrayElements(outhash_, NULL); int ret = ssl_sha256((const char *)indata,inlen,(char *)outhash); env->ReleaseByteArrayElements(indata_, indata, );
env->ReleaseByteArrayElements(outhash_, outhash, ); return ret;
} //默认系统的
extern "C"
JNIEXPORT jstring JNICALL
Java_androidapp_cocoajin_com_tjni_MainActivity_stringFromJNI(
JNIEnv *env,
jobject /* this */) {
std::string hello = "Hello from C++";
return env->NewStringUTF(hello.c_str());
}
13. 最后调用 ,这里把byte[] 转成了hexString显示在了app上
// Example of a call to a native method
TextView tv = (TextView) findViewById(R.id.sample_text); String he = "hello";
byte[] habb = new byte[];
sslsha(he.getBytes(),he.length(),habb); tv.setText(ByteHexUtil.bytesToHexString(habb));
14. 执行结果,以及验证结果


15: 参考资料与 工程下载
android ndk-build 编译静态库libxx.a 以及Android studio openssl 静态库配置(cmake)的更多相关文章
- android NDk环境编译总结
首先,这篇文章的撰写是基于很多前人的优秀的帖子,感谢他们的分享让我能够学习这么多的知识.谢谢 Android NDK开发环境的搭建 前言: Android 上,应用程序的开发,大部分基于 Java 语 ...
- Android NDK入门实例 计算斐波那契数列二生成.so库文件
上一篇文章输生成了jni头文件,里面包含了本地C代码的信息,提供我们引用的C头文件.下面实现本地代码,再用ndk-build编译生成.so库文件.由于编译时要用到make和gcc,这里很多人是通过安装 ...
- Android NDK 学习之在C中调用Java的变量和静态变量
本博客主要是在Ubuntu 下开发,且默认你已经安装了Eclipse,Android SDK, Android NDK, CDT插件. 在Eclipse中添加配置NDK,路径如下Eclipse-> ...
- 【初体验】macos下android ndk交叉编译hello world,并拷贝到android手机上执行
1.机器上以前安装了java 1.8(貌似android ndk不需要java) 2. 下载android ndk,版本是android-ndk-r14b (比较奇怪,我下载了最新的android-n ...
- Android NDK 同时编译多个Module
LOCAL_PATH := $(call my-dir) ## ## NDK 支持同时编译多个Module: ## 在配置的时候,每个Module需要 以 include $(CLEAR_VARS)开 ...
- Android NDK 同时编译多个模块
|-test |---Android.mk |---Application.mk |---sub1 |------Android.mk |------main.c |---sub2 |------An ...
- 【转】[Android] NDK独立编译——独立工具链
转载地址:https://blog.csdn.net/suningning/article/details/74510125
- Android NDK工程的编译和链接以及使用gdb进行调试
前提条件:已经安装了JDK 6.0.android SDK.NDK r9和eclipsele4.2开发环境. 推荐下载Android开发的综合套件adt-bundle-windows-x86,再下载A ...
- android ndk 编译的时候指令集的选取
android ndk在编译的时候默认生成的是thumb指令(拇指)不是arm(手臂)指令,但是有时候在看反汇编的时候,不太熟悉thumb指令或者说thumb指令看起来更费劲,需要生成arm指令,这个 ...
随机推荐
- hdu1556 树状数组区间更新单点查询板子
就是裸的区间更新: 相对于直观的线段树的区间更新,树状数组的区间更新原理不太相同:由于数组中的一个结点控制的是一块区间,当遇到更新[l,r]时,先将所有能控制到 l 的结点给更新了,这样一来就是一下子 ...
- python+selenium+Jenkins构建自动化测试
环境准备 本次使用JDK:1.8.tomcat:8.5.Jenkins:2.127 安装jdk 官方下载地址 安装教程参考 安装tomcat 和jenkins tomcat官方下载地址 jenkins ...
- Highcharts实现图形报表(我主要实现javaweb开发的图形报表)
官网网址:https://www.hcharts.cn/ 中文版的(参考起来方便,你懂的.):http://www.mamicode.com/info-detail-446038.html 网上已经有 ...
- [Reprinted] 使用Spring Data Redis操作Redis(一) 很全面
Original Address: http://blog.csdn.net/albertfly/article/details/51494080
- python全栈开发day41-background、精灵图技术、定位(相、绝、固)、z-index
一.昨日内容回顾 1.标准文档流定义: https://www.jianshu.com/p/b4d2c1dfd6e5 2.浮动和浮动的四大特性 1)脱标 2) 浮动元素相互贴靠 3)字围 4)紧凑(浮 ...
- BZOJ1195 [HNOI2006]最短母串 AC自动机 bfs
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 传送门 - BZOJ1195 题意概括 给出一堆串,然后求一个包含这些串的所有串的最短的中的字典序最小的. 题解 先造一个AC ...
- rocketMQ基本架构简介
1.RocketMQ 简介: RocketMQ前身是阿里研发的一个队列模型的消息中间件,后开源给apache基金会成为了apache的顶级开源项目,具有高性能.高可靠.高实时.分布式特点. 2.Roc ...
- Hash值破解工具(findmyhash与hash-identifier破解Hash值)
Hash值破解工具(findmyhash与hash-identifier破解Hash值) 前言: Kali Linux提供各种哈希密文破解工具,如hashcat.john.rainbows.不论哪一种 ...
- js基础梳理-究竟什么是执行上下文栈(执行栈),执行上下文(可执行代码)?
日常在群里讨论一些概念性的问题,比如变量提升,作用域和闭包相关问题的时候,经常会听一些大佬们给别人解释的时候说执行上下文,调用上下文巴拉巴拉,总有点似懂非懂,不明觉厉的感觉.今天,就对这两个概念梳理一 ...
- Java 泛型优点之编译时类型检查
Java 泛型优点之编译时类型检查 使用泛型代码要比非泛型代码更有优势,下面是 Java 官方教程对泛型其中一个优点的介绍: "Stronger type checks at compile ...