我的编程空间,编程开发者的网络收藏夹
学习永远不晚

实例详解Android中JNI的使用方法

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

实例详解Android中JNI的使用方法

前言

做Android开发的程序员应该都知道,Android的开发语言我们都是在使用JAVA(Kotlin和Flutter我们暂时不考虑)。但是,有时候我们也需要使用到C语言进行一些功能的开发。这个时候我们就需要用到JNI了。

1.导入C语言的类

首先我们需要把C语言写的功能类放入我们的项目中。这里我直接从资料中找了一个,毕竟我不会写。路径在class="lazy" data-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字节
	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中使用的对方主机地址
	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);
 
	//构造netbios结构包
	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;
	}
	//这里要初始化。因为发现linux和模拟器都没问题,真机上该变量若不初始化,其值就不可预知
	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;
}

这里要注意一点,jstring Java_com_hao_cmake_MainActivity_cpuFromJNI方法中,com_hao_cmake是我们的包名,MainActivity是调用JNI的Activity名称,cpuFromJNI是对应方法的名字。

2.接着导入Android.mk文件

这个文件也是放在jni文件夹中


LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
 
# 指定so库文件的名称
LOCAL_MODULE    := jni_mix
# 指定需要编译的源文件列表
LOCAL_class="lazy" data-src_FILES := find_name.cpp
# 指定C++的编译标志
LOCAL_CPPFLAGS += -fexceptions
# 指定要加载的静态库
#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('class="lazy" data-src/main/jni/Android.mk')
    }
}
packagingOptions{
    pickFirst 'lib/arm64-v8a/libjni_mix.so'
    pickFirst 'lib/armeabi-v7a/libjni_mix.so'
}

4.好了,此时可以编译一下项目了

5.此时我们可以找一下我们生成的so包了

在build → intermediates → ndkBuild → debug → obj → local下,我们可以找到我们生成的相关配置平台的so文件

6.将生成的so文件拷入class="lazy" data-src/main/jniLibs中

这个样子的

7.调用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进行调用的全流程。

注意:在使用JNI进行调用的时候,我们的环境一定要有NDK,这个我这里就不说了,大家如果没有搭建需要上网找找搭建一下。

总结

到此这篇关于Android中JNI使用的文章就介绍到这了,更多相关Android中JNI使用内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

实例详解Android中JNI的使用方法

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android在JNI中使用ByteBuffer的方法

本文实例讲述了Android在JNI中使用ByteBuffer的方法。分享给大家供大家参考。具体如下: 一、ByteBuffer 定义 在NIO中,数据的读写操作始终是与缓冲区相关联的(读取时信道(SocketChannel)将数据读入缓冲
2022-06-06

基于Android studio3.6的JNI教程之opencv实例详解

基本环境: Android studio3.6 NDK:r14b(尽量使用该版本) Opencv3.4.1 android sdk(1)新建工程OpenCVDemo,选择,一定要选择Native c++类型,最后要选c++14支持。 (2)
2022-06-06

Android ToolBar整合实例使用方法详解

最近做项目中遇到ToolBar因为不同的界面toobar不同为了描述统一的风格。相信大家也非常清楚,大多数ToolBar包括以下几个方面左标题 左边题颜色 左标题图标等标题 标题颜色右标题 右标题颜色 右标题图标ToolBar标题 Tool
2022-06-06

Android CardView详解及使用方法和实例

Android CardView详解 Android5.0中向我们介绍了一个全新的控件–CardView,从本质上看,可以将CardView看做是FrameLayout在自身之上添加了圆角和阴影效果。请注意:CardView被包装为一种布
2022-06-06

实例讲解Android中SQLiteDatabase使用方法

SQLite数据库是android系统内嵌的数据库,小巧强大,能够满足大多数SQL语句的处理工作,而SQLite数据库仅仅是个文件而已。虽然SQLite的有点很多,但并不是如同PC端的mysql般强大,而且android系统中不允许通过JD
2022-06-06

android Handler详细使用方法实例

开发环境为android4.1.Handler使用例1这个例子是最简单的介绍handler使用的,是将handler绑定到它所建立的线程中.本次实验完成的功能是:单击Start按钮,程序会开始启动线程,并且线程程序完成后延时1s会继续启动该
2022-06-06

vue中使用echarts的方法实例详解

这篇文章主要介绍了vue中使用echarts的方法,结合实例形式详细分析了vue中使用echarts的包安装、引入、生命周期函数元素挂载等相关操作技巧与使用注意事项,需要的朋友可以参考下
2023-05-19

mysql中json_extract的使用方法实例详解

目录一、前言二、创建示例表三、基本语法- 获取jsON对象中某个key对应的value值- 获取JSON数组中某个元素- 获取JSON中的嵌套数据四、渐入佳境- 获取JSON多个路径的数据- 路径表达式*的使用- 返回NULL值- 返回错误
2023-04-13

Android中使用AndroidTestCase的方法实例

Android 使用 AndroidTestCase 的步骤 1.新建一个类继承 AndroidTestCase代码如下: public class TestAudio extends AndroidTestCase { pri
2022-06-06

详解Android中Handler的使用方法

在Android开发中,我们经常会遇到这样一种情况:在UI界面上进行某项操作后要执行一段很耗时的代码,比如我们在界面上点击了一个”下载“按钮,那么我们需要执行网络请求,这是一个耗时操作,因为不知道什么时候才能完成。为了保证不影响UI线程,所
2022-06-06

详解Android中IntentService的使用方法

为什么我们需要IntentService Android中的IntentService是继承自Service类的,在我们讨论IntentService之前,我们先想一下Service的特点: Service的回调方法(onCreate、o
2022-06-06

Android 中Context的使用方法详解

Android 中Context的使用方法详解概要:Context字面意思是上下文,位于framework package的android.content.Context中,其实该类为LONG型,类似Win32中的Handle句柄。很多方法
2023-05-30

详解Android中Notification的使用方法

在消息通知的时候,我们经常用到两个控件Notification和Toast。特别是重要的和需要长时间显示的信息,用Notification最合适不过了。他可以在顶部显示一个图标以标示有了新的通知,当我们拉下通知栏的时候,可以看到
2022-06-06

Android 中RxPermissions 的使用方法详解

Android 中RxPermissions 的使用方法详解以请求拍照、读取位置权限为例module的build.gradle: compile 'com.tbruyelle.rxpermissions2:rxpermissions:0.
2023-05-30

详解Android中AsyncTask的使用方法

在Android中实现异步任务机制有两种方式,Handler和AsyncTask。Handler模式需要为每一个任务创建一个新的线程,任务完成后通过Handler实例向UI线程发送消息,完成界面的更新,这种方式对于整个过程的控制比较精细,但
2022-06-06

详解Android中weight的使用方法

android中对weight的学习可以说是必须的,如果UI布局仅仅使用dp与sp等等,会让布局显得极度不灵活,毕竟各个手机屏幕大小不同,更别说是还有ipad之类的了,所以也是同做本人近期做的一个小UI来分享一下weight的使用。
2022-06-06

详解Android中Intent的使用方法

一、Intent的用途 Intent主要有以下几种重要用途: 1. 启动Activity:可以将Intent对象传递给startActivity()方法或startActivityForResult()方法以启动一个Activity,该I
2022-06-06

Android 中CheckBox的isChecked的使用实例详解

Android 中CheckBox的isChecked的使用实例详解 范例说明所有的网络服务在User使用之前,都需要签署同意条款,在手机应用程序、手机游戏的设计经验中,常看见CheckBox在同意条款情境的运用,其选取的状态有两种即isC
2022-06-06

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录