linux 下的termios读写

以下是一个简单的例子,展示如何在 Linux 下通过串口发送和接收数据:

头文件 serial_communication.h
cpp 复制代码
#ifndef SERIAL_COMMUNICATION_H
#define SERIAL_COMMUNICATION_H

#include <termios.h>
#include <fcntl.h>
#include <unistd.h>
#include <string>
#include <iostream>

class SerialPort {
public:
    SerialPort(const std::string &portName);
    ~SerialPort();
    bool openPort();
    void closePort();
    bool writeData(const std::string &data);
    std::string readData();

private:
    std::string portName;
    int fd; // 文件描述符
    struct termios oldtio, newtio; // termios 配置结构体

    bool configurePort();
};

#endif // SERIAL_COMMUNICATION_H

实现文件 serial_communication.cpp

cpp 复制代码
#include "serial_communication.h"

SerialPort::SerialPort(const std::string &portName)
    : portName(portName), fd(-1) {}

SerialPort::~SerialPort() {
    closePort();
}

bool SerialPort::openPort() {
    // 打开串口设备
    fd = open(portName.c_str(), O_RDWR | O_NOCTTY | O_NDELAY);
    if (fd == -1) {
        std::cerr << "Failed to open serial port: " << portName << std::endl;
        return false;
    }

    // 配置串口
    if (!configurePort()) {
        return false;
    }

    return true;
}

void SerialPort::closePort() {
    if (fd != -1) {
        close(fd);
    }
}

bool SerialPort::configurePort() {
    // 获取当前串口配置
    if (tcgetattr(fd, &oldtio) != 0) {
        std::cerr << "Failed to get serial port settings" << std::endl;
        return false;
    }

    // 设置新的串口参数
    newtio = oldtio;
    newtio.c_cflag = B9600 | CS8 | CLOCAL | CREAD;  // 设置波特率、数据位、启用接收
    newtio.c_iflag = IGNPAR;  // 忽略无效字符
    newtio.c_oflag = 0;       // 无特殊输出处理
    newtio.c_lflag = 0;       // 禁用行模式
    newtio.c_cc[VMIN] = 1;    // 最小字符数
    newtio.c_cc[VTIME] = 0;   // 等待时间

    // 设置串口
    if (tcsetattr(fd, TCSANOW, &newtio) != 0) {
        std::cerr << "Failed to set serial port settings" << std::endl;
        return false;
    }

    return true;
}

bool SerialPort::writeData(const std::string &data) {
    ssize_t bytesWritten = write(fd, data.c_str(), data.length());
    if (bytesWritten == -1) {
        std::cerr << "Failed to write data to serial port" << std::endl;
        return false;
    }

    return true;
}

std::string SerialPort::readData() {
    char buffer[256];
    ssize_t bytesRead = read(fd, buffer, sizeof(buffer) - 1);
    if (bytesRead == -1) {
        std::cerr << "Failed to read data from serial port" << std::endl;
        return "";
    }

    buffer[bytesRead] = '\0';  // Null-terminate the string
    return std::string(buffer);
}

使用示例:main.cpp

cpp 复制代码
#include <iostream>
#include "serial_communication.h"

int main() {
    SerialPort serial("/dev/ttyS0");  // 使用 ttyS0 串口设备(可以根据实际情况修改)

    if (!serial.openPort()) {
        return -1; // 打开串口失败
    }

    // 发送数据
    if (!serial.writeData("Hello, Serial Port!")) {
        return -1; // 写数据失败
    }

    // 读取数据
    std::string data = serial.readData();
    std::cout << "Received: " << data << std::endl;

    serial.closePort();  // 关闭串口
    return 0;
}

自动获取串口设备的方法:

可以使用 opendirreaddir 函数遍历 /dev 目录下的串口设备文件,列出所有可能的串口设备。下面是一个简单的示例,展示如何自动列出系统中所有的串口设备。

cpp 复制代码
#include <iostream>
#include <dirent.h>
#include <string>

void listSerialPorts() {
    DIR *dir = opendir("/dev");
    if (dir == nullptr) {
        std::cerr << "Failed to open /dev directory!" << std::endl;
        return;
    }

    struct dirent *entry;
    while ((entry = readdir(dir)) != nullptr) {
        // 检查文件名是否以 "tty" 开头
        std::string name(entry->d_name);
        if (name.find("ttyS") == 0 || name.find("ttyUSB") == 0) {
            std::cout << "Found serial port: /dev/" << name << std::endl;
        }
    }

    closedir(dir);
}

int main() {
    listSerialPorts();  // 列出所有串口设备
    return 0;
}
相关推荐
csdn_aspnet5 小时前
TCP/IP协议栈深度解析:从基石到前沿
服务器·网络·tcp/ip
lcreek6 小时前
Linux信号机制详解:阻塞信号集与未决信号集
linux·操作系统·系统编程
shandianchengzi6 小时前
【记录】Tailscale|部署 Tailscale 到 linux 主机或 Docker 上
linux·运维·docker·tailscale
John Song7 小时前
Linux机器怎么查看进程内存占用情况
linux·运维·chrome
sichuanwuyi7 小时前
Wydevops工具的价值分析
linux·微服务·架构·kubernetes·jenkins
持戒波罗蜜7 小时前
ubuntu20解决intel wifi 驱动问题
linux·驱动开发·嵌入式硬件·ubuntu
不做无法实现的梦~7 小时前
使用ros2来跑通mid360的驱动包
linux·嵌入式硬件·机器人·自动驾驶
梁辰兴7 小时前
计算机网络基础:虚拟专用网
服务器·网络·计算机网络·vpn·虚拟专用网·计算机网络基础·梁辰兴
点云SLAM8 小时前
C++内存泄漏检测之Windows 专用工具(CRT Debug、Dr.Memory)和Linux 专业工具(ASan 、heaptrack)
linux·c++·windows·asan·dr.memory·c++内存泄漏检测·c++内存管理
肉肉心很软8 小时前
使用onlyoffice实现文件预览编辑 + Docker一键部署流程
运维·docker·容器