凡亿教育-婷婷
凡事用心,一起进步
打开APP
公司名片
凡亿专栏 | 使用Python高效对比多个相似的CAN DBC数据
使用Python高效对比多个相似的CAN DBC数据

尤其是当系统需求变更或多个供应商协作开发时,对比 DBC 文件以找出差异和潜在问题是至关重要的。


本文将深入探讨如何通过 Python 的 cantools 库,实现高效、准确地对比多个相似的 CAN DBC 数据。


cantools 是 Python 领域中一个强大的 CAN 协议处理库,广泛用于汽车和嵌入式系统中,用于解析、编码和解码 CAN 报文。


1、安装 cantools 模块

确保安装 cantools 库:



pip install cantools

假设有多个 DBC 文件:





dbc_file1.dbcdbc_file2.dbcdbc_file3.dbc

对比 DBC 文件时,我们一般会关注以下几个方面:


消息(Message)差异

  • 消息 ID 是否一致

  • 消息名称是否一致

  • 消息长度(DLC)是否一致


信号(Signal)差异

  • 信号数量是否一致

  • 信号名称、起始位、长度、字节序和编码格式的差异

  • 信号物理值缩放和偏移的变化


节点(Node)差异

  • 节点名称、发送和接收的消息是否一致


2、代码实现

设计一个对比代码,思路如下:

  • 加载 DBC 文件:加载并解析文件,提取消息和信号数据。

  • 多文件对比:使用嵌套循环对多个 DBC 进行两两对比,记录差异。

  • 结果展示:以结构化的形式展示差异,方便分析。


代码实现如下:



import cantoolsimport osfrom collections import defaultdictdef load_dbc(file_path):    """加载DBC文件并返回CAN数据库对象"""    try:        return cantools.database.load_file(file_path)    except Exception as e:        print(f"加载失败: {file_path} - {e}")        return Nonedef extract_message_data(db):    """提取DBC文件中的消息及信号信息"""    data = {}    for message in db.messages:        signals = {signal.name: {            "start": signal.start,            "length": signal.length,            "scale": signal.scale,            "offset": signal.offset,            "byte_order": signal.byte_order,            "is_signed": signal.is_signed        } for signal in message.signals}        data[message.name] = {            "frame_id": message.frame_id,            "length": message.length,            "signals": signals        }    return data
def compare_signals(signal1, signal2, signal_name):    """比较信号的属性差异"""    diff = {}    for attr in ["start", "length", "scale", "offset", "byte_order", "is_signed"]:        if signal1[attr] != signal2[attr]:            diff[attr] = (signal1[attr], signal2[attr])    return diff if diff else None
def compare_messages(msg_data1, msg_data2):    """比较两组消息数据"""    diff_results = defaultdict(lambda: defaultdict(dict))
    all_msg_names = set(msg_data1.keys()).union(set(msg_data2.keys()))
    for msg_name in all_msg_names:        if msg_name not in msg_data1:            diff_results[msg_name]["存在于"] = "仅在 DBC2"            continue        if msg_name not in msg_data2:            diff_results[msg_name]["存在于"] = "仅在 DBC1"            continue
        # 对比 Frame ID 和 Length        if msg_data1[msg_name]["frame_id"] != msg_data2[msg_name]["frame_id"]:            diff_results[msg_name]["frame_id"] = (msg_data1[msg_name]["frame_id"], msg_data2[msg_name]["frame_id"])        if msg_data1[msg_name]["length"] != msg_data2[msg_name]["length"]:            diff_results[msg_name]["length"] = (msg_data1[msg_name]["length"], msg_data2[msg_name]["length"])
        # 对比信号        signals1 = msg_data1[msg_name]["signals"]        signals2 = msg_data2[msg_name]["signals"]        all_signals = set(signals1.keys()).union(set(signals2.keys()))
        for sig in all_signals:            if sig not in signals1:                diff_results[msg_name]["signals"][sig] = "仅在 DBC2"            elif sig not in signals2:                diff_results[msg_name]["signals"][sig] = "仅在 DBC1"            else:                sig_diff = compare_signals(signals1[sig], signals2[sig], sig)                if sig_diff:                    diff_results[msg_name]["signals"][sig] = sig_diff
    return diff_results
def compare_multiple_dbc(dbc_files):    """对比多个 DBC 文件"""    dbcs = {file: load_dbc(file) for file in dbc_files}    dbc_data = {file: extract_message_data(db) for file, db in dbcs.items() if db}
    for i in range(len(dbc_files)):        for j in range(i   1, len(dbc_files)):            file1, file2 = dbc_files[i], dbc_files[j]            print(f" === 对比 {file1} 和 {file2} ===")            diff = compare_messages(dbc_data[file1], dbc_data[file2])            if not diff:                print("无差异")            else:                for msg, details in diff.items():                    print(f"消息: {msg}")                    for key, value in details.items():                        print(f"  - {key}: {value}")
# 示例调用dbc_files = ["dbc_file1.dbc", "dbc_file2.dbc", "dbc_file3.dbc"]compare_multiple_dbc(dbc_files)

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表凡亿课堂立场。文章及其配图仅供工程师学习之用,如有内容图片侵权或者其他问题,请联系本站作侵删。
相关阅读
进入分区查看更多精彩内容>
精彩评论

暂无评论