實(shí)例詳解Android中JNI的使用方法
前言
做Android開發(fā)的程序員應(yīng)該都知道,Android的開發(fā)語言我們都是在使用JAVA(Kotlin和Flutter我們暫時(shí)不考慮)。但是,有時(shí)候我們也需要使用到C語言進(jìn)行一些功能的開發(fā)。這個(gè)時(shí)候我們就需要用到JNI了。
1.導(dǎo)入C語言的類
首先我們需要把C語言寫的功能類放入我們的項(xiàng)目中。這里我直接從資料中找了一個(gè),畢竟我不會寫。路徑在src/main/jni中
find_name.cpp
#include <jni.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <netdb.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define send_MAXSIZE 50
#define recv_MAXSIZE 1024
struct NETBIOSNS {
unsigned short int tid; //unsigned short int 占2字節(jié)
unsigned short int flags;
unsigned short int questions;
unsigned short int answerRRS;
unsigned short int authorityRRS;
unsigned short int additionalRRS;
unsigned char name[34];
unsigned short int type;
unsigned short int classe;
};
char *getNameFromIp(const char *ip);
extern "C"
jstring Java_com_hao_cmake_MainActivity_cpuFromJNI(JNIEnv* env, jobject thiz, jstring ip) {
const char* str_ip;
str_ip = env->GetStringUTFChars(ip, 0);
return env->NewStringUTF(getNameFromIp(str_ip));
}
char *getNameFromIp(const char *ip) {
char str_info[1024] = { 0 };
struct sockaddr_in toAddr; //sendto中使用的對方地址
struct sockaddr_in fromAddr; //在recvfrom中使用的對方主機(jī)地址
char send_buff[send_MAXSIZE];
char recv_buff[recv_MAXSIZE];
memset(send_buff, 0, sizeof(send_buff));
memset(recv_buff, 0, sizeof(recv_buff));
int sockfd; //socket
unsigned int udp_port = 137;
int inetat;
if ((inetat = inet_aton(ip, &toAddr.sin_addr)) == 0) {
sprintf(str_info, "[%s] is not a valid IP address\n", ip);
return str_info;
}
if ((sockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0) {
sprintf(str_info, "%s socket error sockfd=%d, inetat=%d\n", ip, sockfd, inetat);
return str_info;
}
bzero((char*) &toAddr, sizeof(toAddr));
toAddr.sin_family = AF_INET;
toAddr.sin_addr.s_addr = inet_addr(ip);
toAddr.sin_port = htons(udp_port);
//構(gòu)造netbios結(jié)構(gòu)包
struct NETBIOSNS nbns;
nbns.tid = 0x0000;
nbns.flags = 0x0000;
nbns.questions = 0x0100;
nbns.answerRRS = 0x0000;
nbns.authorityRRS = 0x0000;
nbns.additionalRRS = 0x0000;
nbns.name[0] = 0x20;
nbns.name[1] = 0x43;
nbns.name[2] = 0x4b;
int j = 0;
for (j = 3; j < 34; j++) {
nbns.name[j] = 0x41;
}
nbns.name[33] = 0x00;
nbns.type = 0x2100;
nbns.classe = 0x0100;
memcpy(send_buff, &nbns, sizeof(nbns));
int send_num = 0;
send_num = sendto(sockfd, send_buff, sizeof(send_buff), 0,
(struct sockaddr *) &toAddr, sizeof(toAddr));
if (send_num != sizeof(send_buff)) {
sprintf(str_info,
"%s sendto() error sockfd=%d, send_num=%d, sizeof(send_buff)=%d\n",
ip, sockfd, send_num, sizeof(send_buff));
shutdown(sockfd, 2);
return str_info;
}
int recv_num = recvfrom(sockfd, recv_buff, sizeof(recv_buff), 0,
(struct sockaddr *) NULL, (socklen_t*) NULL);
if (recv_num < 56) {
sprintf(str_info, "%s recvfrom() error sockfd=%d, recv_num=%d\n", ip,
sockfd, recv_num);
shutdown(sockfd, 2);
return str_info;
}
//這里要初始化。因?yàn)榘l(fā)現(xiàn)linux和模擬器都沒問題,真機(jī)上該變量若不初始化,其值就不可預(yù)知
unsigned short int NumberOfNames = 0;
memcpy(&NumberOfNames, recv_buff + 56, 1);
char str_name[1024] = { 0 };
unsigned short int mac[6] = { 0 };
int i = 0;
for (i = 0; i < NumberOfNames; i++) {
char NetbiosName[16];
memcpy(NetbiosName, recv_buff + 57 + i * 18, 16);
//依次讀取netbios name
if (i == 0) {
sprintf(str_name, "%s", NetbiosName);
}
}
sprintf(str_info, "%s|%s|", ip, str_name);
for (i = 0; i < 6; i++) {
memcpy(&mac[i], recv_buff + 57 + NumberOfNames * 18 + i, 1);
sprintf(str_info, "%s%02X", str_info, mac[i]);
if (i != 5) {
sprintf(str_info, "%s-", str_info);
}
}
return str_info;
}
這里要注意一點(diǎn),jstring Java_com_hao_cmake_MainActivity_cpuFromJNI方法中,com_hao_cmake是我們的包名,MainActivity是調(diào)用JNI的Activity名稱,cpuFromJNI是對應(yīng)方法的名字。
2.接著導(dǎo)入Android.mk文件
這個(gè)文件也是放在jni文件夾中
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) # 指定so庫文件的名稱 LOCAL_MODULE := jni_mix # 指定需要編譯的源文件列表 LOCAL_SRC_FILES := find_name.cpp # 指定C++的編譯標(biāo)志 LOCAL_CPPFLAGS += -fexceptions # 指定要加載的靜態(tài)庫 #LOCAL_WHOLE_STATIC_LIBRARIES += android_support # 指定需要鏈接的庫 LOCAL_LDLIBS := -llog include $(BUILD_SHARED_LIBRARY) $(call import-module, android/support)
3.我們配置一下build.gradle文件
android -> defaultConfig 下添加
externalNativeBuild{
ndkBuild{
abiFilters "arm64-v8a","armeabi-v7a"
}
}
android 下添加
externalNativeBuild {
ndkBuild {
path file('src/main/jni/Android.mk')
}
}
packagingOptions{
pickFirst 'lib/arm64-v8a/libjni_mix.so'
pickFirst 'lib/armeabi-v7a/libjni_mix.so'
}
4.好了,此時(shí)可以編譯一下項(xiàng)目了
5.此時(shí)我們可以找一下我們生成的so包了
在build → intermediates → ndkBuild → debug → obj → local下,我們可以找到我們生成的相關(guān)配置平臺的so文件
6.將生成的so文件拷入src/main/jniLibs中
這個(gè)樣子的

7.調(diào)用C語言方法的Activity如下
public class MainActivity extends AppCompatActivity {
public native String cpuFromJNI(String ip);
static {
System.loadLibrary("jni_mix");
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
String str = cpuFromJNI("192.168.0.163");
Toast.makeText(this,str,Toast.LENGTH_SHORT).show();
}
}
這樣我們就完成了用C語言類生成so包,并使用JNI進(jìn)行調(diào)用的全流程。
注意:在使用JNI進(jìn)行調(diào)用的時(shí)候,我們的環(huán)境一定要有NDK,這個(gè)我這里就不說了,大家如果沒有搭建需要上網(wǎng)找找搭建一下。
總結(jié)
到此這篇關(guān)于Android中JNI使用的文章就介紹到這了,更多相關(guān)Android中JNI使用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android實(shí)現(xiàn)跳轉(zhuǎn)第三方百度地圖導(dǎo)航
在眾多地圖導(dǎo)航產(chǎn)品中,百度地圖以其精準(zhǔn)的導(dǎo)航和豐富的本地生活數(shù)據(jù)受到廣泛歡迎,本項(xiàng)目介紹如何在 Android 中構(gòu)造 Intent 調(diào)用百度地圖導(dǎo)航,希望對大家有所幫助2025-04-04
Android使用AlertDialog創(chuàng)建對話框
這篇文章主要為大家詳細(xì)介紹了Android使用AlertDialog創(chuàng)建對話框的方法料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12
Android BottomNavigationView底部導(dǎo)航效果
這篇文章主要為大家詳細(xì)介紹了Android BottomNavigationView底部導(dǎo)航效果的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01
Android中查看USB連接的外接設(shè)備信息的代碼實(shí)例
這篇文章主要介紹了Android中查看USB連接的外接設(shè)備信息的代碼實(shí)例,需要的朋友可以參考下2014-04-04
Android EditText設(shè)置邊框的操作方法
這篇文章主要介紹了Android EditText設(shè)置邊框,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-12-12
使用adb命令向Android模擬器中導(dǎo)入通訊錄聯(lián)系人的方法
這篇文章主要介紹了使用adb命令向Android模擬器中導(dǎo)入通訊錄聯(lián)系人的方法,實(shí)例分析了導(dǎo)入通訊錄存儲文件的技巧,需要的朋友可以參考下2015-01-01
深入了解Android Okio的超時(shí)機(jī)制
Okio是一個(gè)IO庫,底層基于Java原生的輸入輸出流實(shí)現(xiàn)。但原生的輸入輸出流并沒有提供超時(shí)的檢測機(jī)制。而Okio實(shí)現(xiàn)了這個(gè)功能,本文就來為大家詳細(xì)講講2023-02-02
Android基于騰訊云實(shí)時(shí)音視頻仿微信視頻通話最小化懸浮
這篇文章主要為大家詳細(xì)介紹了Android基于騰訊云實(shí)時(shí)音視頻實(shí)現(xiàn)類似微信視頻通話最小化懸浮,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11

