Linux+QT+SocketCAN:使用信号槽机制实现数据收发

本文介绍了如何在Linux+QT环境下利用SocketCAN库,通过信号槽机制实现CAN数据的收发。在Ubuntu16.04上,使用QT 5.13.1,创建一个单独的线程处理CAN消息的接收,同时通过对象间的信号槽连接,简化Motor对象与CAN对象的数据交互。
摘要由CSDN通过智能技术生成

Linux+QT+SocketCAN:使用信号槽机制实现数据收发

最近在考虑采用面对对象的方式重新搭建机器人的主控程序框架,虽然之前的框架也是有这种思想在里面,但是总感觉还是有程序化编程的影子,很多地方的处理都不太理想,而且虽然之前也是一直在采用QT Creator作为IDE开发程序,但是实际上并没有怎么用QT的库,只是单纯的作为一个IDE来用,未免也是空守宝山而不入。
因此这次决定充分利用QT的库和信号槽机制来对程序框架进行重新构建。

0.开发环境

  1. HOST平台: Ubuntu16.04
  2. ARM平台:at91sam9x35
  3. QT版本: 5.13.1

1.分析

针对Linux上的CAN通信,Socket CAN提供了一种十分简单方便的解决方案,QT中并没有像QTcpSocket一样的库可以使用,因此仍然使用linux内核中提供的库。
考虑到电机等设备会循环向主控返回电机的状态信息,CAN的接收工作十分耗时,因此选择新建一个继承QObject的类,将其Moveto一个QThread中单独执行循环接收工作。
考虑到CAN和Motor对象均为Robot类的对象成员,而各个Motor对象需要调用CAN的发送接口向其他对象发送数据,为避免各对象成员之间的数据交互造成各种麻烦,因此在各对象成员中建立signal与slot,在Robot中进行connect。
eg:
Robot具有两个对象成员CAN和Motor

class Robot:public QObject
{
   
	Q_OBJECT
public:
	Robot();
	~Robot();
	...
	CAN *mCAN;
	Motor *Wheel_1;
	...
}

CAN中构建数据发送槽

class CAN:public QObject
{
   
	Q_OBJECT
	...
public slots:
    void Transmit(can_frame pFrame);
    ...
}

Motor中构建数据发送信号

class Motor:public QObject
{
   
	Q_OBJECT
	...
signals:
	void sendCanMeg(can_frame pFrame);
	...
}

Robot中将其连接

Robot::Robot()
{
   
	...
	mCAN = new CAN("can1");
	Wheel_1= new Motor(WHEEL_1,SPEED_MODE)
	connect(Wheel_1,&Motor::sendCanMeg,mCAN ,&CAN::Transmit);
	...
}

2.源码

2.1 can.h

#ifndef CAN_H
#define CAN_H
#include <QObject>
#include <QThread>
#include <QtCore>
#include <fcntl.h>        /*文件控制定义*/
#include <net/if.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <linux/can.h>
#include <linux/can/raw.h>
#include <Utilities.h>
#include <QDebug>
#include <unistd.h>
// CAN接收线程
class CAN_RcvThread :public QObject
{
   
    Q_OBJECT
public:
    CAN_RcvThread(QObject *parent = 0) ;
public slots:
    void RcvMegThread(int s);
    void stopRcvThread();
private:
    bool isStop=
  • 7
    点赞
  • 50
    收藏
    觉得还不错? 一键收藏
  • 8
    评论
Linux下面QT写Can通信程序,网络上有很多例子都是互相抄的;本人因为项目原因,通过源代码Socket函数,写了完整的程序,含有2个案例;分享给大家; 这里主要是包含几个步骤,1:绑定Socket;2:cna/can1的设置,3:波特率的设置(如果发送和接收波特率不一致肯定不行的);4:发送;5:接收处理; 下面就贴出部分代码: void MyWindow::startcan(int number) { int ret = 0; //can先关闭 设置好波特率后 再开启can if(number == 0) //can0 { system("ifconfig can0 down");//先关闭 system("ip link set can0 up type can bitrate 50000 triple-sampling on");//设置波特率 system("ifconfig can0 up");//再开启 } else //can1 { system("ifconfig can1 down");//先关闭 system("ip link set can1 up type can bitrate 50000 triple-sampling on");//设置波特率 system("ifconfig can1 up");//再开启 } socket = ::socket(PF_CAN,SOCK_RAW,CAN_RAW); struct ifreq ifr; strcpy((char *)(ifr.ifr_name),number == 0 ? "can0" : "can1"); ioctl(socket,SIOCGIFINDEX,&ifr); addr.can_family = AF_CAN; addr.can_ifindex = ifr.ifr_ifindex; ret = bind(socket,(struct sockaddr*)&addr,sizeof(addr)); if (ret OpenCan(50000); ArmCan = new Thread(zyhapi,socket,port); ArmCan->start(); } void MyWindow::stopcan(int number) { if(ArmCan) { ArmCan->stop(); ArmCan->terminate(); ArmCan->wait(); } zyhapi->CloseCan(number); } //发送 void MyWindow::on_sendbtn_clicked() { /* struct can_frame frame; memset(&frame,0,sizeof(struct can_frame)); std::string str=ui->edit->text().toStdString(); if(str.length() > 8) { QMessageBox::about(this,"error","length of send string must less than 8 bytes"); return; } */ struct can_frame frame; memset(&frame,0,sizeof(struct can_frame)); char buf[8]={0X20,0XFF,0X01,0X02,0X03,0X04,0XFF,0XFF}; frame.can_id = 0x00000020;//发出去的帧ID即:0X00000020 frame.can_dlc = 8;//帧数据长度 for(int i=0;i<frame.can_dlc;i++) { frame.data[i]=buf[i];//帧数据 } //frame.can_id = 0x123; //strcpy((char*)frame.data,str.c_str()); //frame.can_dlc = str.length(); sendto(socket,&frame,sizeof(struct can_frame),0,(struct sockaddr*)&addr,sizeof(addr)); /* struct can_frame frame; char buf[8]={0X20,0XFF,0X01,0X02,0X03,0X04,0XFF,0XFF}; frame.can_id = 0x00000020;//发出去的帧ID即:0X00000020 frame.can_dlc = 8;//帧数据长度 for(int i=0;iWriteCan(frame.can_id,frame.data,frame.can_dlc);//发数据,通过can端口-- 暂时默认 CAN0 端口 */ }
### 回答1: Linux是一种自由和开放源代码的操作系统,它支持多种架构和平台,包括服务器和嵌入式系统。Qt是一种跨平台的C++应用程序开发框架,可以帮助开发人员在Windows、Linux、macOS等系统上开发GUI应用程序。SocketCAN是Linux内核中的一个网络协议族,用于控制CAN总线。 在Linux系统中,可以使用Qt开发一个GUI应用程序来控制CAN总线。通过SocketCAN协议,应用程序可以与CAN总线进行通信,并控制其中的设备。Qt提供了很多网络编程API,可以轻松实现SocketCAN协议的连接和通信。 使用QtSocketCAN开发应用程序需要掌握C++编程和网络编程的相关知识。开发人员需要了解SocketCAN协议的基本原理和使用方法,以及Qt框架的相关API。此外,还需要了解CAN总线上使用的物理层和数据链路层协议。 总之,LinuxQtSocketCAN是三个关键技术,可以帮助开发人员在Linux平台上控制CAN总线和设备。通过这些技术,可以实现很多实时、高效和可靠的应用程序,用于控制工业自动化、汽车电子等领域。 ### 回答2: Linux是一款开放源代码的操作系统,由于它的高稳定性、可靠性和安全性,它在嵌入式系统、服务器端、移动设备以及超级计算机等领域拥有广泛的应用。而QT则是一个跨平台的GUI开发框架,它能够实现C++开发的应用程序在Windows、MacOS、Linux等各种操作系统上运行。所以Linux QT socketcan就是一个用QT开发的能够支持socketcan的程序。 socketcan则是Linux内核提供的can协议的socket接口,它支持CAN总线的读写操作,常用于CAN总线通信的应用程序中。相较于一些基于串口的通信方式,CAN总线具有更高的性能和可靠性,能够支持多种设备共享网络带宽,因此广泛应用于汽车电子、工业自动化、机器人等领域。 在嵌入式开发中,通过Linux QT socketcan可以实现与CAN总线进行通信,开发出各种CAN通信的应用程序,例如读取传感器数据、控制执行器行动、监测车辆状态等。Qt提供了各种工具、控件和类库,程序员可以使用它们快速开发易于维护和扩展的GUI界面,同时也能够支持各种平台可移植性,加快了开发的速度。 综上所述,Linux QT socketcan是一款强大的工具,它将操作系统、GUI框架和CAN总线通信集成在一起,为嵌入式开发带来了高效和便捷的应用程序开发体验。 ### 回答3: Linux Qt SocketCAN是一种用于开发嵌入式系统的开源工具链,它结合了Linux操作系统、Qt应用程序框架和SocketCAN通信协议。SocketCAN是Linux内核中实现的一种CAN协议封装层,它提供了统一的接口让用户在各种CAN软硬件之间轻松切换。 Linux Qt SocketCAN可以方便地开发出各种汽车电子控制单元(ECU)、机器人、医疗设备和工业自动化等嵌入式系统应用。它提供了丰富的开发工具和库,支持多语言、多平台,并具有高效性、安全性和可靠性。开发人员可以使用Qt Creator等IDE进行开发和调试,还可以运用SocketCAN库,采用CANOpen、J1939等协议进行通信。 同时,Linux Qt SocketCAN实现了多种技术创新,如基于SocketCAN及Linux内核RT-Preempt的硬实时能力、使用QtQuick进行界面设计、支持NFC数据交互等。这些技术优势使得Linux Qt SocketCAN成为了嵌入式系统领域的先锋。 总之,Linux Qt SocketCAN为开发嵌入式系统提供了可靠、高效、安全的解决方案,是一种值得深入研究和应用的开源工具链。
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值