经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 移动开发 » Android » 查看文章
实例详解Android中JNI的使用方法
来源:jb51  时间:2021/8/5 17:20:36  对本文有异议

前言

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

1.导入C语言的类

首先我们需要把C语言写的功能类放入我们的项目中。这里我直接从资料中找了一个,毕竟我不会写。路径在src/main/jni中

find_name.cpp

  1. #include <jni.h>
  2. #include <string.h>
  3. #include <stdio.h>
  4. #include <stdlib.h>
  5. #include <string.h>
  6. #include <netdb.h>
  7. #include <sys/stat.h>
  8. #include <sys/types.h>
  9. #include <sys/select.h>
  10. #include <sys/socket.h>
  11. #include <netinet/in.h>
  12. #include <arpa/inet.h>
  13. #define send_MAXSIZE 50
  14. #define recv_MAXSIZE 1024
  15. struct NETBIOSNS {
  16. unsigned short int tid; //unsigned short int 占2字节
  17. unsigned short int flags;
  18. unsigned short int questions;
  19. unsigned short int answerRRS;
  20. unsigned short int authorityRRS;
  21. unsigned short int additionalRRS;
  22. unsigned char name[34];
  23. unsigned short int type;
  24. unsigned short int classe;
  25. };
  26. char *getNameFromIp(const char *ip);
  27. extern "C"
  28. jstring Java_com_hao_cmake_MainActivity_cpuFromJNI(JNIEnv* env, jobject thiz, jstring ip) {
  29. const char* str_ip;
  30. str_ip = env->GetStringUTFChars(ip, 0);
  31. return env->NewStringUTF(getNameFromIp(str_ip));
  32. }
  33. char *getNameFromIp(const char *ip) {
  34. char str_info[1024] = { 0 };
  35. struct sockaddr_in toAddr; //sendto中使用的对方地址
  36. struct sockaddr_in fromAddr; //在recvfrom中使用的对方主机地址
  37. char send_buff[send_MAXSIZE];
  38. char recv_buff[recv_MAXSIZE];
  39. memset(send_buff, 0, sizeof(send_buff));
  40. memset(recv_buff, 0, sizeof(recv_buff));
  41. int sockfd; //socket
  42. unsigned int udp_port = 137;
  43. int inetat;
  44. if ((inetat = inet_aton(ip, &toAddr.sin_addr)) == 0) {
  45. sprintf(str_info, "[%s] is not a valid IP address\n", ip);
  46. return str_info;
  47. }
  48. if ((sockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0) {
  49. sprintf(str_info, "%s socket error sockfd=%d, inetat=%d\n", ip, sockfd, inetat);
  50. return str_info;
  51. }
  52. bzero((char*) &toAddr, sizeof(toAddr));
  53. toAddr.sin_family = AF_INET;
  54. toAddr.sin_addr.s_addr = inet_addr(ip);
  55. toAddr.sin_port = htons(udp_port);
  56. //构造netbios结构包
  57. struct NETBIOSNS nbns;
  58. nbns.tid = 0x0000;
  59. nbns.flags = 0x0000;
  60. nbns.questions = 0x0100;
  61. nbns.answerRRS = 0x0000;
  62. nbns.authorityRRS = 0x0000;
  63. nbns.additionalRRS = 0x0000;
  64. nbns.name[0] = 0x20;
  65. nbns.name[1] = 0x43;
  66. nbns.name[2] = 0x4b;
  67. int j = 0;
  68. for (j = 3; j < 34; j++) {
  69. nbns.name[j] = 0x41;
  70. }
  71. nbns.name[33] = 0x00;
  72. nbns.type = 0x2100;
  73. nbns.classe = 0x0100;
  74. memcpy(send_buff, &nbns, sizeof(nbns));
  75. int send_num = 0;
  76. send_num = sendto(sockfd, send_buff, sizeof(send_buff), 0,
  77. (struct sockaddr *) &toAddr, sizeof(toAddr));
  78. if (send_num != sizeof(send_buff)) {
  79. sprintf(str_info,
  80. "%s sendto() error sockfd=%d, send_num=%d, sizeof(send_buff)=%d\n",
  81. ip, sockfd, send_num, sizeof(send_buff));
  82. shutdown(sockfd, 2);
  83. return str_info;
  84. }
  85. int recv_num = recvfrom(sockfd, recv_buff, sizeof(recv_buff), 0,
  86. (struct sockaddr *) NULL, (socklen_t*) NULL);
  87. if (recv_num < 56) {
  88. sprintf(str_info, "%s recvfrom() error sockfd=%d, recv_num=%d\n", ip,
  89. sockfd, recv_num);
  90. shutdown(sockfd, 2);
  91. return str_info;
  92. }
  93. //这里要初始化。因为发现linux和模拟器都没问题,真机上该变量若不初始化,其值就不可预知
  94. unsigned short int NumberOfNames = 0;
  95. memcpy(&NumberOfNames, recv_buff + 56, 1);
  96. char str_name[1024] = { 0 };
  97. unsigned short int mac[6] = { 0 };
  98. int i = 0;
  99. for (i = 0; i < NumberOfNames; i++) {
  100. char NetbiosName[16];
  101. memcpy(NetbiosName, recv_buff + 57 + i * 18, 16);
  102. //依次读取netbios name
  103. if (i == 0) {
  104. sprintf(str_name, "%s", NetbiosName);
  105. }
  106. }
  107. sprintf(str_info, "%s|%s|", ip, str_name);
  108. for (i = 0; i < 6; i++) {
  109. memcpy(&mac[i], recv_buff + 57 + NumberOfNames * 18 + i, 1);
  110. sprintf(str_info, "%s%02X", str_info, mac[i]);
  111. if (i != 5) {
  112. sprintf(str_info, "%s-", str_info);
  113. }
  114. }
  115. return str_info;
  116. }

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

2.接着导入Android.mk文件

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

  1. LOCAL_PATH := $(call my-dir)
  2. include $(CLEAR_VARS)
  3. # 指定so库文件的名称
  4. LOCAL_MODULE := jni_mix
  5. # 指定需要编译的源文件列表
  6. LOCAL_SRC_FILES := find_name.cpp
  7. # 指定C++的编译标志
  8. LOCAL_CPPFLAGS += -fexceptions
  9. # 指定要加载的静态库
  10. #LOCAL_WHOLE_STATIC_LIBRARIES += android_support
  11. # 指定需要链接的库
  12. LOCAL_LDLIBS := -llog
  13. include $(BUILD_SHARED_LIBRARY)
  14. $(call import-module, android/support)

3.我们配置一下build.gradle文件

android  ->  defaultConfig 下添加

  1. externalNativeBuild{
  2. ndkBuild{
  3. abiFilters "arm64-v8a","armeabi-v7a"
  4. }
  5. }

android 下添加

  1. externalNativeBuild {
  2. ndkBuild {
  3. path file('src/main/jni/Android.mk')
  4. }
  5. }
  6. packagingOptions{
  7. pickFirst 'lib/arm64-v8a/libjni_mix.so'
  8. pickFirst 'lib/armeabi-v7a/libjni_mix.so'
  9. }

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

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

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

6.将生成的so文件拷入src/main/jniLibs中

这个样子的

7.调用C语言方法的Activity如下

  1. public class MainActivity extends AppCompatActivity {
  2. public native String cpuFromJNI(String ip);
  3. static {
  4. System.loadLibrary("jni_mix");
  5. }
  6. @Override
  7. protected void onCreate(Bundle savedInstanceState) {
  8. super.onCreate(savedInstanceState);
  9. setContentView(R.layout.activity_main);
  10. String str = cpuFromJNI("192.168.0.163");
  11. Toast.makeText(this,str,Toast.LENGTH_SHORT).show();
  12. }
  13. }

这样我们就完成了用C语言类生成so包,并使用JNI进行调用的全流程。

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

总结

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

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号