Pyhon:串口应用及数据解析过程

news2024/10/6 14:26:00

Pyhon:串口应用及数据解析过程

串口通信是一种常用的通信协议,本文重点记录在Python中使用串口,并且以一款电源保护板的串口数据协议为例,对其进行解析,记录收发过程中对16进制数据进行转换的过程。

1. 调用串口

在Python中进行串口通信时,需要serial包的支持,通过安装

pip install pyserial

安装包,然后可以对串口进行调用,在Linux系统中和在windows系统中的调用方式基本相同,只是端口的名称有所差别

  • 打开串口:可以通过以下命令打开端口,这里打开COM7端口,设置波特率为115200
serial_port = serial.Serial("COM7", 115200, timeout=0.5) # windows 中打开方式

在Linux系统中可以采用如下方法打开端口,大同小异

serial_port = serial.Serial(
    port='/dev/ttyUSB0',    # Linux中打开方式
    baudrate=115200,
    bytesize=serial.EIGHTBITS,
    parity=serial.PARITY_NONE,
    stopbits=serial.STOPBITS_ONE,
)
# Wait a second to let the port initialize
time.sleep(1)
  • 关闭串口:在关闭串口的时候可以采用如下指令,就完成了对打开的端口进行关闭
serial_port.close()
  • 发送数据:利用串口最主要的过程就是对数据的发送和接收功能,发送数据可以用如下函数
serial_port.write()
  • 接收数据:接收过程用到的函数相对于发送会多几个,有如下函数需要使用
serial_port.flush()  # 清空缓存
data_num = serial_port.inWaiting() # 这个数据显示一次读取回来多少个数据
data = serial_port.read()  # 读串口数据

在有了这些基本函数后,需要做的是针对自己的任务来设计发送和接收的具体过程,根据所要传输的内容设计过程。在下面的测试过程中针对实际的串口数据解码过程,对电源保护板的串口数据进行解码,传输16进制的HEX数据。下面配合代码详细描述解码和测试过程。

2 测试工具准备

为了方便测试,可以采用虚拟串口,Virtual Serial Port Driver工具,工具可以有15天的试用期,有足够时间使用,不用破解,可以去下载地址下载,利用该工具可以生成一对虚拟串口,然后利用串口调试助手打开其中一个端口,Python程序中打开里一个端口,完成对于数据发送接收及解析的测试过程。

在这里插入图片描述

在生成了一对串口之后,还需要准备串口调试助手,如下UartAssist小软件,这个工具还挺好用,在5.0版本上具有很多其他的功能,可以用这个地址下载。

在这里插入图片描述

具有很多的便捷功能,
在这里插入图片描述

3 程序编写及测试

程序如下,为了方便说明分成4个部分(这个4个部分在一个.py文件中)。

    1. 导入包,打开关闭串口函数
import time
import serial
# import math
import binascii


# 在Linux系统中调用串口
# serial_port = serial.Serial(
#     port='/dev/ttyUSB0',
#     baudrate=115200,
#     bytesize=serial.EIGHTBITS,
#     parity=serial.PARITY_NONE,
#     stopbits=serial.STOPBITS_ONE,
# )
# # Wait a second to let the port initialize
# time.sleep(1)


serial_port = serial.Serial("COM7", 115200, timeout=0.5)


#  打开串口函数
def serial_open():
    if serial_port.isOpen():  # 判断串口是否成功打开
        print("打开串口成功。")
        print(serial_port.name)  # 输出串口号
    else:
        print("打开串口失败。")


#  关闭串口函数
def serial_close():
    serial_port.close()
    pass
    1. 读串口数据函数并解析数据
data_all = []
data_hex_str = []

# 读串口数据
def serial_read():
    serial_port.flush()
    while True:
        # time.sleep(100)
        data_num = serial_port.inWaiting()# 这个数据显示一次读取回来多少个数据
        # print("接收到", data_num, "个数据")
        if data_num > 0:
            time.sleep(0.1)
            print('第一次读取当前接收字节数', data_num)
            data_num = serial_port.inWaiting()  # 这个数据显示一次读取回来多少个数据
            data_all = []
            print('第二次读取当前接收字节数', data_num)
            for i in range(data_num):
                data = serial_port.read()
                data_all.append(data.hex())   # 这个地方是字符串的操作 这个操作的过程把接收到的数据都以16进制字符形式存储在这个变量中
                # data_hex_str += binascii.hexlify(data)  # 这个转换成hex字符串
                # print(data.hex())  # 这个就可以取出16进制字符的形式 如'FF'

            print('总的接收的字符串列表', data_all)

            # 在这里就有数据了 接下来是如何解析数据
            if data_all[0] == '4e' and data_all[1] == '57':
                print('解析到起始帧。。。。。。。。。')

                frame_len_str = bytes.fromhex(data_all[2] + data_all[3])  # 转成字符
                frame_len = int(frame_len_str.hex(), 16)
                print('帧长:', frame_len)

                frame_id = data_all[4] + data_all[5] + data_all[6] + data_all[7]
                print('BMS终端ID号:', frame_id)

                frame_ordword = data_all[8]
                print('命令字:', frame_ordword)

                frame_source = data_all[9]
                print('数据帧来源:', frame_source)  # 0 BMS 1 蓝牙 2 GPS 3 PC上位机

                frame_trantype = data_all[10]
                print('传输类型:', frame_trantype)  # 0 BMS 1 蓝牙 2 GPS 3 PC上位机

                if data_all[11] == '79':  ## 这里面涉及到电池数量,这个用多少设置成多少 测试用20节电池, 3个字节表示一个电池电压
                    print('开始解析单电池电压...')
                    len_0x79 = int(bytes.fromhex(data_all[12]).hex(), 16)  # 这个字节是这个表示电压的字节的长度
                    # print('共', len_0x79, '个字节')
                    battery_num = round(len_0x79/3)
                    print('共连接了', battery_num, '节电池')
                    battery_mv_str = bytes.fromhex(data_all[14] + data_all[15])  # 转成字符
                    battery_mv = int(battery_mv_str.hex(), 16)
                    print('第一节电池电压值:', battery_mv, 'mv')

                if data_all[13+len_0x79] == '80':
                    print('开始解析功率管温度...')
                    power_tube_temp_str = bytes.fromhex(data_all[74] + data_all[75])  # 转成字符
                    power_tube_temp = int(power_tube_temp_str.hex(), 16)
                    # 对温度进行转换,超过一百为负值
                    if power_tube_temp > 100:
                        power_tube_temp = 0-(power_tube_temp-100)
                    print('功率管温度为:', power_tube_temp)

                if data_all[16+len_0x79] == '81':
                    print('开始解析电池箱内温度...')
                    battery_case_temp_str = bytes.fromhex(data_all[77] + data_all[78])  # 转成字符
                    battery_case_temp = int(battery_case_temp_str.hex(), 16)
                    # 对温度进行转换,超过一百为负值
                    if battery_case_temp > 100:
                        battery_case_temp = 0 - (battery_case_temp - 100)
                    print('电池箱内温度为:', battery_case_temp)

                if data_all[19+len_0x79] == '82':
                    print('开始解析电池温度...')
                    battery_temp_str = bytes.fromhex(data_all[80] + data_all[81])  # 转成字符
                    battery_temp = int(battery_temp_str.hex(), 16)
                    # 对温度进行转换,超过一百为负值
                    if battery_temp > 100:
                        battery_temp = 0 - (battery_temp - 100)
                    print('电池温度为:', battery_temp)

                if data_all[22+len_0x79] == '83':
                    print('开始电池总电压...')
                    battery_v_sum_str = bytes.fromhex(data_all[83] + data_all[84])  # 转成字符
                    battery_v_sum = int(battery_v_sum_str.hex(), 16)
                    # 对总电压进行转换, *0.01V
                    battery_v_sum = battery_v_sum * 0.01
                    print('电池总电压为:', battery_v_sum, 'V')

                if data_all[25+len_0x79] == '84':
                    print('开始电流数据...')
                    battery_c_sum_str = bytes.fromhex(data_all[86] + data_all[87])  # 转成字符
                    battery_c_sum = int(battery_c_sum_str.hex(), 16)
                    if battery_c_sum < 32768:
                        print('放电电流为:', battery_c_sum, 'mA')
                    else:
                        battery_c_sum = battery_c_sum - 32768
                        print('充电电电流为:', battery_c_sum, 'mA')

                if data_all[28+len_0x79] == '85':
                    print('开始解析电池剩余容量...')
                    battery_left_str = bytes.fromhex(data_all[89])  # 转成字符
                    battery_left = int(battery_left_str.hex(), 16)
                    battery_left = battery_left/255
                    print('电池剩余容量为:', battery_left)  # 这个是一个百分比

                if data_all[30+len_0x79] == '86':
                    print('开始解析电池温度传感器数量...')
                    temp_sensor_num_str = bytes.fromhex(data_all[91])  # 转成字符
                    temp_sensor_num = int(temp_sensor_num_str.hex(), 16)
                    print('电池温度传感器数量为:', temp_sensor_num)

                if data_all[32+len_0x79] == '87':
                    print('开始解析电池循环使用次数...')
                    battery_ues_time_str = bytes.fromhex(data_all[93] + data_all[94])  # 转成字符
                    battery_ues_time = int(battery_ues_time_str.hex(), 16)
                    print('电池循环使用次数:', battery_ues_time)

                if data_all[35+len_0x79] == '89':
                    print('开始解析电池循环总容量...')
                    battery_ues_sumtime_str = bytes.fromhex(data_all[96]+data_all[97]+data_all[98]+data_all[99])  # 转成字符
                    battery_ues_sumtime = int(battery_ues_sumtime_str.hex(), 16)
                    print('电池循环使用次数:', battery_ues_sumtime)

                if data_all[40+len_0x79] == '8a':
                    print('开始解析电池总串数...')
                    battery_sum_num_str = bytes.fromhex(data_all[101] + data_all[102])  # 转成字符
                    battery_sum_num = int(battery_sum_num_str.hex(), 16)
                    print('电池总串数:', battery_sum_num)

                if data_all[43+len_0x79] == '8b':
                    print('开始解析电池警告信息...')
                    battery_alarm_sum_str = bytes.fromhex(data_all[104] + data_all[105])  # 转成字符
                    battery_alarm_sum = int(battery_alarm_sum_str.hex(), 16)
                    print('电池警告信息:', battery_alarm_sum)

                if data_all[46+len_0x79] == '8c':
                    print('开始解析电池状态信息...')
                    battery_state_str = bytes.fromhex(data_all[107] + data_all[108])  # 转成字符
                    battery_state = int(battery_state_str.hex(), 16)
                    print('电池状态信息:', battery_state)

                if data_all[49+len_0x79] == '8e':
                    print('开始解析总电压过压保护...')
                    voltage_high_prot_str = bytes.fromhex(data_all[110] + data_all[111])  # 转成字符
                    voltage_high_prot = int(voltage_high_prot_str.hex(), 16)
                    voltage_high_prot = voltage_high_prot*10  # 最小单位10mv
                    print('总电压过压保护:', voltage_high_prot)

                if data_all[52+len_0x79] == '8f':
                    print('开始解析总电压欠压保护...')
                    voltage_low_prot_str = bytes.fromhex(data_all[113] + data_all[114])  # 转成字符
                    voltage_low_prot = int(voltage_low_prot_str.hex(), 16)
                    voltage_low_prot = voltage_low_prot*10  # 最小单位10mv
                    print('总电压欠压保护:', voltage_low_prot, 'mV')

                if data_all[55+len_0x79] == '90':
                    print('开始解析单体过压保护电压...')
                    voltage_single_high_prot_str = bytes.fromhex(data_all[116] + data_all[117])  # 转成字符
                    voltage_single_high_prot = int(voltage_single_high_prot_str.hex(), 16)
                    print('单体过压保护电压:', voltage_single_high_prot, 'mV')

                if data_all[58+len_0x79] == '91':
                    print('开始解析单体过压恢复电压...')
                    voltage_single_high_recover_str = bytes.fromhex(data_all[119] + data_all[120])  # 转成字符
                    voltage_single_high_recover = int(voltage_single_high_recover_str.hex(), 16)
                    print('单体过压恢复电压:', voltage_single_high_recover, 'mV')

                if data_all[61+len_0x79] == '92':
                    print('开始解析单体过压保护延时...')
                    voltage_single_high_pro_delay_str = bytes.fromhex(data_all[122] + data_all[123])  # 转成字符
                    voltage_single_high_pro_delay = int(voltage_single_high_pro_delay_str.hex(), 16)
                    print('单体过压保护延时:', voltage_single_high_pro_delay, 's')

                if data_all[64+len_0x79] == '93':
                    print('开始解析单体欠压保护电压...')
                    voltage_single_low_pro_str = bytes.fromhex(data_all[125] + data_all[126])  # 转成字符
                    voltage_single_low_pro = int(voltage_single_low_pro_str.hex(), 16)
                    print('单体欠压保护电压:', voltage_single_low_pro, 'mV')

                if data_all[67+len_0x79] == '94':
                    print('开始解析单体欠压恢复电压...')
                    voltage_single_low_recover_str = bytes.fromhex(data_all[128] + data_all[129])  # 转成字符
                    voltage_single_low_recover = int(voltage_single_low_recover_str.hex(), 16)
                    print('单体欠压恢复电压:', voltage_single_low_recover, 'mV')

                if data_all[70+len_0x79] == '95':
                    print('开始解析单体欠压保护延时...')
                    voltage_single_low_pro_delay_str = bytes.fromhex(data_all[131] + data_all[132])  # 转成字符
                    voltage_single_low_pro_delay = int(voltage_single_low_pro_delay_str.hex(), 16)
                    print('单体欠压保护延时:', voltage_single_low_pro_delay, 's')

                if data_all[73+len_0x79] == '96':
                    print('开始解析电芯压差保护值...')
                    battery_vol_diff_pro_str = bytes.fromhex(data_all[134] + data_all[135])  # 转成字符
                    battery_vol_diff_pro = int(battery_vol_diff_pro_str.hex(), 16)
                    print('电芯压差保护值:', battery_vol_diff_pro, 'mV')

            else:
                print('未找到起始帧')

            # serial_port.write(data)

            # if data == "\r".encode():
                # For Windows boxen on the other end
                # serial_port.write("\n".encode())
        else:
            # print(data_all)
            a = []

在读数据的时候,先清楚掉缓存中的数据,防止缓存中的数据产生影响,然后进行了两次serial_port.inWaiting()的操作,是为了保证一次串口传输的数据都接收完全,这个保护板一次传输回来的字节可以达到267。
接收回来的数据需要按照HEX形式读取,转换成10进制数据。在这里利用的函数byets.fromhex()bytes转成str
,然后利用int把str转成10进制,得到实际的结果。可以参考注释。

    1. 写串口数据函数
# 写串口数据
def serial_write():
    data_write_1_1 = '4E'  # 起始帧
    data_write_1_2 = '57'
    data_write_1 = data_write_1_1 + data_write_1_2

    data_write_2_1 = '00'  # 帧长度
    data_write_2_2 = '13'
    data_write_2 = data_write_2_1 + data_write_2_2

    data_write_3_1 = '00'    # BMS终端号
    data_write_3_2 = '00'
    data_write_3_3 = '00'
    data_write_3_4 = '00'
    data_write_3 = data_write_3_1 + data_write_3_2 + data_write_3_3 + data_write_3_4

    data_write_4 = '06'  # 命令字
    data_write_5 = '03'  # 帧来源
    data_write_6 = '00'  # 传输类型
    data_write_7 = '00'  # 数据标识码  '00'代表全部数据类型

    data_write_8_1 = '00'    # 记录号
    data_write_8_2 = '00'
    data_write_8_3 = '00'
    data_write_8_4 = '00'
    data_write_8 = data_write_8_1 + data_write_8_2 + data_write_8_3 + data_write_8_4
    data_write_9 = '68'   # 结束标识

    # 计算校验和 这个是累加求和得到的校验和
    data_temp = (int(bytes.fromhex(data_write_1_1).hex(), 16) + int(bytes.fromhex(data_write_1_2).hex(), 16) +
                 int(bytes.fromhex(data_write_2_1).hex(), 16) + int(bytes.fromhex(data_write_2_2).hex(), 16) +
                 int(bytes.fromhex(data_write_3_1).hex(), 16) + int(bytes.fromhex(data_write_3_2).hex(), 16) +
                 int(bytes.fromhex(data_write_3_3).hex(), 16) + int(bytes.fromhex(data_write_3_4).hex(), 16) +
                 int(bytes.fromhex(data_write_4).hex(), 16) + int(bytes.fromhex(data_write_5).hex(), 16) +
                 int(bytes.fromhex(data_write_6).hex(), 16) + int(bytes.fromhex(data_write_7).hex(), 16) +
                 int(bytes.fromhex(data_write_8_1).hex(), 16) + int(bytes.fromhex(data_write_8_2).hex(), 16) +
                 int(bytes.fromhex(data_write_8_3).hex(), 16) + int(bytes.fromhex(data_write_8_4).hex(), 16) +
                 int(bytes.fromhex(data_write_9).hex(), 16))

    # print(type(data_temp), data_temp)
    # data_test1 = str(hex(data_temp))
    data_test = "{:04x}".format(data_temp)  # 这是格式化输出
    # print(type(data_test), data_test)
    # print(type(data_test_213), data_test_213)
    # data_test_byte = data_test_213.encode('utf-8')
    # data_test_byte = str.encode(data_test_213)
    # print(type(data_test_byte.decode()), data_test_byte.decode())

    data_write_10_1 = '00'  # 校验和,data_write_1 ~ data_write_9 的累加和,这两面前两个字节不用
    data_write_10_2 = '00'
    data_write_10_3 = data_test[0:2]  # ‘01’
    data_write_10_4 = data_test[2:4]  # '29'

    data_write_10 = data_write_10_1 + data_write_10_2 + data_write_10_3 + data_write_10_4

    data_write = data_write_1 + data_write_2 + data_write_3 + data_write_4 + data_write_5 + data_write_6 + data_write_7 + data_write_8 + data_write_9 + data_write_10
    data_write_b = bytes.fromhex(data_write)
    print('写串口数据')
    serial_port.write(data_write_b)

在写串口数据的时候,因为指令基本不变,所以直接用字符串赋值了,可以参考注释单独复制出来测试。

    1. mian函数

if __name__ == '__main__':
    serial_open()
    try:
        serial_write()
        serial_read()
    except KeyboardInterrupt:
        print("Exiting Program")
    finally:
        serial_close()

运行程序,可以看到程序先发送数据,在接收返回的数据进行解析

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/767034.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

浅谈HTTPS抓包原理,为什么Charles能够抓取HTTPS报文?

Charles作用其实相当于拦截器&#xff0c;当客户端和服务器通信时&#xff0c;Charles其实会先接收到服务器的证书&#xff0c;但是它会自己生成一个证书发送给客户端(不管是Web端或App应用)&#xff0c;也就是说它不仅仅是拦截&#xff0c;甚至还可以修改。 由于Charles更改了…

[数据结构 -- 手撕排序算法第六篇] 递归实现快速排序(集霍尔版本,挖坑法,前后指针法为一篇的实现方法,很能打)

目录 1、常见的排序算法 1.1 交换排序基本思想 2、快速排序的实现方法 2.1 基本思想 3 hoare&#xff08;霍尔&#xff09;版本 3.1 实现思路 3.2 思路图解 3.3 为什么实现思路的步骤2、3不能交换 3.4 hoare版本代码实现 3.5 hoare版本代码测试 4、挖坑法 4.1 实现…

SpringCloud(六)Config配置中心

一、配置中心 官方文档&#xff1a;**https://docs.spring.io/spring-cloud-config/docs/current/reference/html/ 经过前面的学习&#xff0c;我们对于一个分布式应用的技术选型和搭建已经了解得比较多了&#xff0c;但是如果我们的微服务项目需要部署很多个实例&#xff0c…

基于时域特征和频域特征组合的敏感特征集,再利用SVM或KNN传统分类器进行轴承故障诊断(python编程,代码有详细注释)

1.文件夹介绍&#xff08;使用的是CWRU数据集&#xff09; 0HP-3HP四个文件夹装载不同工况下的内圈故障、外圈故障、滚动体故障和正常轴承数据。 这里以打开0HP文件为例进行展示&#xff0c;creat_data.py是处理原始数据的脚本&#xff0c;负责将原始数据切不重叠割成1024的固…

postgresql导入导出数据库的一些问题

新建一个数据库 别忘了添加空间数据的扩展 备份之前的数据库 注意一定要自定义表&#xff0c;去掉 spatial_ref_sys &#xff0c;要不然需要先drop在创建&#xff0c;可能会报错。 一般不会去导函数&#xff0c;如果有个别自己创建的函数可以手动复制一下&#xff0c;全部导的话…

联合接地的概念和优势——通信设备的“保命秘籍”

夏日炎炎&#xff0c;强对流天气多发&#xff0c;雷电灾害也需要引起关注。雷电是大气中的超长距离放电过程。雷电有着强大的电流、炙热的高温、强烈的电磁辐射以及猛烈的冲击波&#xff0c;这让其能够瞬间变身“无敌破坏王”&#xff0c;造成雷电灾害。如若强大的瞬间过电压冲…

【Python爬虫开发基础⑭】Scrapy架构(组件介绍、架构组成和工作原理)

&#x1f680;个人主页&#xff1a;为梦而生~ 关注我一起学习吧&#xff01; &#x1f4a1;专栏&#xff1a;python网络爬虫从基础到实战 欢迎订阅&#xff01;后面的内容会越来越有意思~ &#x1f4a1;往期推荐&#xff1a; ⭐️前面比较重要的基础内容&#xff1a; 【Python爬…

【C++】开源:跨平台轻量日志库easyloggingpp

&#x1f60f;★,:.☆(&#xffe3;▽&#xffe3;)/$:.★ &#x1f60f; 这篇文章主要介绍跨平台轻量日志库easyloggingpp。 无专精则不能成&#xff0c;无涉猎则不能通。。——梁启超 欢迎来到我的博客&#xff0c;一起学习&#xff0c;共同进步。 喜欢的朋友可以关注一下&am…

剑指offer16.数值的整数次方

这道题不是easy的难度吗&#xff1f;为什么是 medium&#xff0c;看完题我就想到了用分治法把n除以2&#xff0c;分成两个子问题&#xff0c;然后子问题的解合起来就是这个问题的解&#xff0c;于是立马写了如下代码&#xff1a; class Solution {public double myPow(double …

MySQL(十一):MySQL语法-函数

原始数据 select * from employees where emp_no 10002; select * from salaries where emp_no 10002;--------------------------------------------------------------- | emp_no | birth_date | first_name | last_name | gender | hire_date | -----------------------…

duilib消息产生以及响应机制

1、duilib消息产生处理原理 win32消息产生机制 消息产生。系统将消息排列到其应该排放的线程消息队列中。线程中的消息循环调用GetMessage&#xff08;or PeekMessage&#xff09;获取消息。传送消息TranslateMessage and DispatchMessage to 窗口过程&#xff08;Windows pr…

JavaWeb课程设计项目实战(01)——项目综述

版权声明 本文原创作者&#xff1a;谷哥的小弟作者博客地址&#xff1a;http://blog.csdn.net/lfdfhl 项目背景 本项目适合JavaWeb新手作为入门学习的练手案例&#xff0c;主要技术包括&#xff1a; TomcatJSPELJSTLMySQL 该项目是一个学生信息管理系统&#xff0c;主要功能…

300-700题红题第四题和第五题

题目链接&#xff1a;332. 重新安排行程 题目链接&#xff1a;335. 路径交叉 332. 重新安排行程 是写过的详细的在回溯。只能说再写一遍还是磕磕绊绊&#xff0c;但好歹理解的比较快了。 主要就是理解一下回溯的backtrack的输出&#xff0c;不仅仅是可以输出路径的&#xff0…

竹云IDaaS | 全面融合AWS Cognito

AWS Cognito组件用户池User Pool国内暂未上线&#xff0c;身份云IDaaS可完全替代Cognito用户池&#xff0c;借助身份云进行用户身份验证并融合Coginto身份池Identity Pool&#xff0c;实现对AWS资源的无缝安全访问。 企业面临的挑战 AWS Cognito中国站暂未提供用户池User&…

Python_pymysql_与mysql交互

目录 基础功能 简单封装 源码等资料获取方法 基础功能 import pymysql from pymysql.cursors import DictCursor # 导入字典类型的游标对象# 连接数据库 db pymysql.connect(host192.168.3.109, # 数据库IP地址port3306, # 数据库端口号userroot, …

Qchart学习

目录 Qchart简介 QChartView 简介 QAbstractAxis 简介 QAbstractSeries 简介 Qchart Public Types Properties属性 Public Functions QAbstractSeries Public Types Properties Public Functions Signals信号 QAbstractAxis Properties Public Functions 主题设…

全面了解ESP-01SWiFi模块

ESP-01S是一款基于ESP8266芯片的WiFi模块&#xff0c;它提供了低成本、低功耗和高度集成的解决方案&#xff0c;适用于物联网和嵌入式应用。本文将介绍ESP-01S模块的功能和特点&#xff0c;并提供一个简单的WiFi控制示例。 目录 ESP-01S模块管脚功能&#xff1a; ESP-01S特点…

字符串列表分类求平均值

给定一字符串列表数据&#xff0c;按颜色分类计算价格平均值并写入列表。 (本笔记适合对python字符串和列表基本烂熟的 coder 翻阅) 【学习的细节是欢悦的历程】 Python 官网&#xff1a;https://www.python.org/ Free&#xff1a;大咖免费“圣经”教程《 python 完全自学教程…

【C++修炼之路】模板初阶

&#x1f451;作者主页&#xff1a;安 度 因 &#x1f3e0;学习社区&#xff1a;StackFrame &#x1f4d6;专栏链接&#xff1a;C修炼之路 文章目录 一、泛型编程二、函数模板1、概念2、格式3、函数模板实例化① 隐式② 显式 4、特性 五、类模板六、模板使内置类型"升级&q…

无人驾驶中识别颜色并跟踪的优化(加入PID算法控制)

我们了解到无人驾驶是如何去识别颜色的&#xff0c;以及无人车能够跟随颜色目标的演示。回到现实中我们发现&#xff0c;无人车的速度控制是很关键的&#xff0c;这个涉及到安全问题&#xff0c;比如等待红绿灯时&#xff0c;该减速或加速超车等这样很常见的情形&#xff0c;在…