Qt应用开发(安卓篇)——调用ioctl、socket等C函数

2024-01-31 19:12

本文主要是介绍Qt应用开发(安卓篇)——调用ioctl、socket等C函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、前言

        在 Qt for Android 中没办法像在嵌入式linux中一样直接使用 ioctl 等底层函数,这是因为因为 Android 平台的安全性和权限限制。

        在 Android 中,访问设备硬件和系统资源需要特定的权限,并且需要通过 Android 系统提供的 API 来进行。Android 平台为了保障系统的安全性和稳定性,限制了应用程序对底层硬件和系统的直接访问。

        Qt for Android 是建立在 Android NDK 和 Java 层之上的,它提供了一种跨平台的开发框架,允许开发者使用 C++ 和 Qt API 来开发 Android 应用程序。但是,由于 Android 平台的限制,Qt for Android 也受到了 Android 平台的限制,无法直接访问底层设备或调用底层系统函数。

        我们需要通过 Java 层的 JNI 接口来间接访问,通过 JNI 接口调用底层的系统函数或设备驱动程序,主要分两步:

  1. 实现 JNI 方法: 在 C 语言中实现这些本地方法。

  2. 加载并链接 JNI 库: 在 Qt 项目中加载并链接 JNI 库,以便在 Qt/C++ 代码中调用本地方法。

        下面我们通过一个socketcan的调用实例才讲解。

二、编写 JNI 接口

        在Android路径下,新建一个jni文件夹,新建文件socketcan_native.c,部分代码内容如下:

#include <jni.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include <net/if.h>
#include <linux/can.h>
#include <linux/can/raw.h>
#include <fcntl.h>
#include <string.h>#define	STATUS_OK					0
#define STATUS_ERR					-1static int sock_fd;
static int m_isopen;void sockcan_close()
{close(sock_fd);system(ip_cmd_can_close);m_isopen = STATUS_ERR;
}
int sockcan_open(int bitrate)
{//创建套接口sock_fd = socket(AF_CAN,SOCK_RAW,CAN_RAW);if(sock_fd < 0){return STATUS_ERR;}//绑定can0设备与套接口struct ifreq ifr;struct sockaddr_can addr;strcpy(ifr.ifr_name,"can0");ioctl(sock_fd,SIOCGIFINDEX,&ifr);ifr.ifr_ifindex = if_nametoindex(ifr.ifr_name);addr.can_family = AF_CAN;addr.can_ifindex = ifr.ifr_ifindex;if(bind(sock_fd,(struct sockaddr *)&addr,sizeof(addr))<0){perror("bind error!\n");return STATUS_ERR;}//配置int flags;flags = fcntl(sock_fd,F_GETFL,0);// flags |= O_NONBLOCK;//非阻塞flags &= ~O_NONBLOCK;//阻塞fcntl(sock_fd,F_SETFL,flags);m_isopen = STATUS_OK;return STATUS_OK;
}JNIEXPORT jint JNICALL
JJava_com_example_socketcan_SocketCANJNI_socketCanOpen(JNIEnv *env, jobject thiz, jint baudrate)
{return sockcan_open(baudrate);
}JNIEXPORT jint JNICALL
Java_com_example_socketcan_SocketCANJNI_socketCanWrite
(JNIEnv *env, jobject thiz, jint canId,jint dataLen,jint externFlag,jint remoteFlag,jbyteArray datas)
{if(sock_fd <= 0)return -1;if(dataLen > 8)return -2;//获取实例的变量array的值int nArrLen = (*env)->GetArrayLength(env,datas);char *chArr = (char*)(*env)->GetByteArrayElements(env,datas, 0);struct can_frame txframe;memcpy(txframe.data, chArr, nArrLen);txframe.can_id = canId;if(externFlag)txframe.can_id |= CAN_EFF_FLAG;if(remoteFlag)txframe.can_id |= CAN_RTR_FLAG;txframe.can_dlc = dataLen;return write(sock_fd, &txframe, sizeof(struct can_frame));
}

        如果是cpp的文件,在JNIEXPORT jint JNICALL前面需要添加extern "C" ,并且有一些指针写法需要小改。

        在Android/jni文件夹下,新建文件Android.mk,内容如下:

LOCAL_PATH := $(call my-dir)include $(CLEAR_VARS)LOCAL_MODULE    := socketcan_native
LOCAL_SRC_FILES := socketcan_native.cinclude $(BUILD_SHARED_LIBRARY)

        然后在该目录使用ndk对其进行打包生成so文件,在libs文件夹下,会生成四个不同环境下的的so文件。

5625fc6983dd4a5b99fb3600a07984be.png

 a4e6576586c54f4bba493d17270690ce.png

60f5cd37710c45fdb5b1ecf78e08673b.png

 858f5940bab048dd90563643ecc09d6a.png

986cfd8307964db8838e8c742541e157.png

三、加载并链接 JNI 库

        Qt侧声明接口,直接调用!!!

extern "C" JNIEXPORT jint JNICALL
JJava_com_example_socketcan_SocketCANJNI_socketCanOpen
(JNIEnv *env, jobject thiz, jint baudrate);extern "C" JNIEXPORT jint JNICALL
Java_com_example_socketcan_SocketCANJNI_socketCanWrite
(JNIEnv *env, jobject thiz, jint canId,jint dataLen,jint externFlag,jint remoteFlag,jbyteArray datas);
jint attachResult = QAndroidJniEnvironment::javaVM()->AttachCurrentThread(reinterpret_cast<JNIEnv**>(&env), NULL);
if (attachResult != JNI_OK) {qDebug() << "Failed to attach current thread to JVM";
}
JJava_com_example_socketcan_SocketCANJNI_socketCanOpen(env, NULL, 500000);

四:其他

        在其他的教程中,还有一种方法,QT调用java代码,java代码调用lib,在这方法中,我们需要在Android下新建src文件夹,新建SocketCANJNI.java文件,Qt側使用QAndroidJniObject::callStaticMethod调用java代码,这条路理论上也行得通,但是我没有成功实现,大家有空可以研究以下,撒花!

// SocketCANJNI.java
package com.example.socketcan;public class SocketCANJNI {static {System.loadLibrary("socketcan_native");}public native void socketCanOpen(int bitrate);
}

 

这篇关于Qt应用开发(安卓篇)——调用ioctl、socket等C函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/664779

相关文章

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

PHP应用中处理限流和API节流的最佳实践

《PHP应用中处理限流和API节流的最佳实践》限流和API节流对于确保Web应用程序的可靠性、安全性和可扩展性至关重要,本文将详细介绍PHP应用中处理限流和API节流的最佳实践,下面就来和小编一起学习... 目录限流的重要性在 php 中实施限流的最佳实践使用集中式存储进行状态管理(如 Redis)采用滑动

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

使用docker搭建嵌入式Linux开发环境

《使用docker搭建嵌入式Linux开发环境》本文主要介绍了使用docker搭建嵌入式Linux开发环境,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录1、前言2、安装docker3、编写容器管理脚本4、创建容器1、前言在日常开发全志、rk等不同

QT Creator配置Kit的实现示例

《QTCreator配置Kit的实现示例》本文主要介绍了使用Qt5.12.12与VS2022时,因MSVC编译器版本不匹配及WindowsSDK缺失导致配置错误的问题解决,感兴趣的可以了解一下... 目录0、背景:qt5.12.12+vs2022一、症状:二、原因:(可以跳过,直奔后面的解决方法)三、解决方

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

Python实战之SEO优化自动化工具开发指南

《Python实战之SEO优化自动化工具开发指南》在数字化营销时代,搜索引擎优化(SEO)已成为网站获取流量的重要手段,本文将带您使用Python开发一套完整的SEO自动化工具,需要的可以了解下... 目录前言项目概述技术栈选择核心模块实现1. 关键词研究模块2. 网站技术seo检测模块3. 内容优化分析模

Python Counter 函数使用案例

《PythonCounter函数使用案例》Counter是collections模块中的一个类,专门用于对可迭代对象中的元素进行计数,接下来通过本文给大家介绍PythonCounter函数使用案例... 目录一、Counter函数概述二、基本使用案例(一)列表元素计数(二)字符串字符计数(三)元组计数三、C

PostgreSQL简介及实战应用

《PostgreSQL简介及实战应用》PostgreSQL是一种功能强大的开源关系型数据库管理系统,以其稳定性、高性能、扩展性和复杂查询能力在众多项目中得到广泛应用,本文将从基础概念讲起,逐步深入到高... 目录前言1. PostgreSQL基础1.1 PostgreSQL简介1.2 基础语法1.3 数据库