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

目录
  • 前言
  • 1.导入C语言的类
  • 2.接着导入Android.mk文件
  • 3.我们配置一下build.gradle文件
  • 4.好了,此时可以编译一下项目了
  • 5.此时我们可以找一下我们生成的so包了
  • 6.将生成的so文件拷入src/main/jniLibs中
  • 7.调用C语言方法的Activity如下
  • 总结

前言

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

1.导入C语言的类

首先我们需要把C语言写的功能类放入我们的项目中。这里我直接从资料中找了一个,毕竟我不会写。路径在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_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('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文件拷入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使用内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Android通过JNI实现守护进程

    开发一个需要常住后台的App其实是一件非常头疼的事情,不仅要应对国内各大厂商的ROM,还需要应对各类的安全管家...虽然不断的研究各式各样的方法,但是效果并不好,比如任务管理器把App干掉,服务就起不来了... 网上搜寻一番后,主要的方法有以下几种方法,但都是治标不治本: 1.提高Service的优先级:这个,也只能说在系统内存不足需要回收资源的时候,优先级较高,不容易被回收,然并卵... 2.提高Service所在进程的优先级:效果不是很明显 3.在onDestroy方法里重启service:

  • Android Jni的简单使用详解

    需求介绍 相信大家在请求接口的时候,很多时候都是需要传参的,除了业务必要的字段外,还有一些恒定不变的字段,包括一些用来编码的固定字段.这些固定字段的值我们是不能直接写在项目工程中的.防止被别人反编译,抓包获取后,恶意请求,抓取我们的用户数据. 老规矩,先上图: 解决方法 这时候我们就可以利用Jni(Java Native Interface)来存储我们这些常量值,虽然也不是最安全的方式,但是也增加了反编译和抓包的难度.相对来说还是安全一点的. 具体做法就是: ① 新建一个工程,包名和你需要调用J

  • 详解Android JNI的基本使用(CMake)

    简介 什么是JNI JNI的全称是Java Native Interface:Java本地开发接口,它提供了若干的API实现了Java和其他语言的通信(主要是C和C++),目的就是Java可以调用C或C++开发的函数,C或C++也能调用Java的方法.这样有很多有点,其一就是效率,C/C++是本地语言,比java更高效:其二就是可以复用已经存在的C/C++代码:其三是Java反编译比C语言容易,一般加密算法都是用C语言编写,不容易被反编译. 什么是NDK和CMake NDK全称是Native D

  • 记录Android studio JNI开发的三种方式(推荐)

    概述 在Andorid Studio不支持JNI开发之前大家一般都是使用Eclipse开发JNI,各种配置让人觉得很蛋疼.从Andorid Studio支持JNI开发后,让我们开发JNI变的如此简单. NDK 和 JNI介绍 JNI (Java Native Interface)是一套编程接口,用来实现Java代码和其他语言(c.C++或汇编)进行交互.这里需要注意的是JNI是JAVA语言自己的特性,也就是说JNI和Android没有关系.在Windows下面用JAVA做开发也经常会用到JNI,

  • Android中的JNI数组操作教程

    前言 JNI 中有两种数组操作,基础数据类型数组和对象数组,JNI 对待基础数据类型数组和对象数组是不一样的. 基本数据类型数组 对于基本数据类型数组,JNI 都有和 Java 相对应的结构,在使用起来和基本数据类型的使用类似. 在 Android JNI 基础知识篇提到了 Java 数组类型对应的 JNI 数组类型.比如,Java int 数组对应了 jintArray,boolean 数组对应了 jbooleanArray. 如同 String 的操作一样,JNI 提供了对应的转换函数:Ge

  • Android在JNI中使用ByteBuffer的方法

    本文实例讲述了Android在JNI中使用ByteBuffer的方法.分享给大家供大家参考.具体如下: 一.ByteBuffer 定义 在NIO中,数据的读写操作始终是与缓冲区相关联的(读取时信道(SocketChannel)将数据读入缓冲区,写入时首先要将发送的数据按顺序填入缓冲区) 缓冲区是定长的,基本上它只是一个列表,它的所有元素都是基本数据类型.ByteBuffer是最常用的缓冲区,它提供了读写其他数据类型的方法,且信道的读写方法只接收ByteBuffer. ByteBuffer有以下几

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

    目录 前言 1.导入C语言的类 2.接着导入Android.mk文件 3.我们配置一下build.gradle文件 4.好了,此时可以编译一下项目了 5.此时我们可以找一下我们生成的so包了 6.将生成的so文件拷入src/main/jniLibs中 7.调用C语言方法的Activity如下 总结 前言 做Android开发的程序员应该都知道,Android的开发语言我们都是在使用JAVA(Kotlin和Flutter我们暂时不考虑).但是,有时候我们也需要使用到C语言进行一些功能的开发.这个时

  • 详解Android中Intent的使用方法

    一.Intent的用途 Intent主要有以下几种重要用途: 1. 启动Activity:可以将Intent对象传递给startActivity()方法或startActivityForResult()方法以启动一个Activity,该Intent对象包含了要启动的Activity的信息及其他必要的数据. 2. 启动Service:可以将Intent对象传递给startService()方法或bindService()方法以启动一个Service,该Intent对象包含了要启动的Service的

  • 详解Android中Notification的使用方法

    在消息通知的时候,我们经常用到两个控件Notification和Toast.特别是重要的和需要长时间显示的信息,用Notification最合适不过了.他可以在顶部显示一个图标以标示有了新的通知,当我们拉下通知栏的时候,可以看到详细的通知内容.       最典型的应用就是未看短信和未接来电的显示,还有QQ微信,我们一看就知道有一个未接来电或者未看短信,收到QQ离线信息.同样,我们也可以自定义一个Notification来定义我们自己的程序想要传达的信息. Notification我把他分为两种

  • 详解Android中解析XML的方法

    XML在各种开发中都广泛应用,Android也不例外.作为承载数据的一个重要角色,如何读写XML成为Android开发中一项重要的技能.今天就由我向大家介绍一下在Android平台下几种常见的XML解析和创建的方法. 在Android中,常见的XML解析器分别为SAX解析器.DOM解析器和PULL解析器,下面,我将一一向大家详细介绍. SAX解析器: SAX(Simple API for XML)解析器是一种基于事件的解析器,它的核心是事件处理模式,主要是围绕着事件源以及事件处理器来工作的.当事

  • 详解Android中AsyncTask的使用方法

    在Android中实现异步任务机制有两种方式,Handler和AsyncTask. Handler模式需要为每一个任务创建一个新的线程,任务完成后通过Handler实例向UI线程发送消息,完成界面的更新,这种方式对于整个过程的控制比较精细,但也是有缺点的,例如代码相对臃肿,在多个任务同时执行时,不易对线程进行精确的控制. 为了简化操作,Android1.5提供了工具类android.os.AsyncTask,它使创建异步任务变得更加简单,不再需要编写任务线程和Handler实例即可完成相同的任务

  • 实例详解ECMAScript5中新增的Array方法

    ECMAScript5 给出了一系列新的API接口,这些接口在新的浏览器中大部分是被支持的,IE9,Chrome,FirFor都支持,也有少量API不是所有浏览器都支持,以下内容仅介绍大部分被支持的API.利用新的API我们可以设计出非常靠谱的类,而且还能保持原有的javaScript的风格. ECMAScript5标准发布于2009年12月3日,它带来了一些新的,改善现有的Array数组操作的方法.(注意兼容性) 在ES5中,一共有9个Array方法:http://kangax.github.

  • 详解Android中IntentService的使用方法

    为什么我们需要IntentService ? Android中的IntentService是继承自Service类的,在我们讨论IntentService之前,我们先想一下Service的特点: Service的回调方法(onCreate.onStartCommand.onBind.onDestroy)都是运行在主线程中的.当我们通过startService启动Service之后,我们就需要在Service的onStartCommand方法中写代码完成工作,但是onStartCommand是运行

  • 详解Android中weight的使用方法

    android中对weight的学习可以说是必须的,如果UI布局仅仅使用dp与sp等等,会让布局显得极度不灵活,毕竟各个手机屏幕大小不同,更别说是还有ipad之类的了,所以也是同做本人近期做的一个小UI来分享一下weight的使用. 左边是各个屏幕的显示效果,右边是1080*1920屏幕的具体显示效果.可以看到,不管屏幕如何变化,使用weight的布局中总是填充满屏幕的,至于美观效果就不说了,直接上代码. 小编使用的android studio,eclipse用户直接复制肯定会有问题,AS用户直

  • 详解Android中Handler的使用方法

    在Android开发中,我们经常会遇到这样一种情况:在UI界面上进行某项操作后要执行一段很耗时的代码,比如我们在界面上点击了一个"下载"按钮,那么我们需要执行网络请求,这是一个耗时操作,因为不知道什么时候才能完成.为了保证不影响UI线程,所以我们会创建一个新的线程去执行我们的耗时的代码.当我们的耗时操作完成时,我们需要更新UI界面以告知用户操作完成了.所以我们可能会写出如下的代码: package ispring.com.testhandler; import android.app.

  • 详解Android中接口回调、方法回调

    在Android开发中我们很多地方都用到了方法的回调,回调就是把方法的定义和功能导入实现分开的一种机制,目的是为了解耦他的本质是基于观察者设计模式,即观察者设计模式的的简化版,例如:在下载时候的进度回调,在adapter与activity之间的回调,在javabean和fragment以及fragment之间的回调等等,回调的目的主要有两个:其一是传递数据,其二是保持数据的同步更新.常用的有两种形式,一是使用内部类的形式,得到接口的子类对象,另一种是直接实现定义的接口. 一.内部类的形式 1.在

随机推荐