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

相关文章

基于Python开发电脑定时关机工具

《基于Python开发电脑定时关机工具》这篇文章主要为大家详细介绍了如何基于Python开发一个电脑定时关机工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 简介2. 运行效果3. 相关源码1. 简介这个程序就像一个“忠实的管家”,帮你按时关掉电脑,而且全程不需要你多做

Java中的Opencv简介与开发环境部署方法

《Java中的Opencv简介与开发环境部署方法》OpenCV是一个开源的计算机视觉和图像处理库,提供了丰富的图像处理算法和工具,它支持多种图像处理和计算机视觉算法,可以用于物体识别与跟踪、图像分割与... 目录1.Opencv简介Opencv的应用2.Java使用OpenCV进行图像操作opencv安装j

Idea调用WebService的关键步骤和注意事项

《Idea调用WebService的关键步骤和注意事项》:本文主要介绍如何在Idea中调用WebService,包括理解WebService的基本概念、获取WSDL文件、阅读和理解WSDL文件、选... 目录前言一、理解WebService的基本概念二、获取WSDL文件三、阅读和理解WSDL文件四、选择对接

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

基于Qt Qml实现时间轴组件

《基于QtQml实现时间轴组件》时间轴组件是现代用户界面中常见的元素,用于按时间顺序展示事件,本文主要为大家详细介绍了如何使用Qml实现一个简单的时间轴组件,需要的可以参考下... 目录写在前面效果图组件概述实现细节1. 组件结构2. 属性定义3. 数据模型4. 事件项的添加和排序5. 事件项的渲染如何使用

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要