CAN信号值解析

本文提供一种可以解析CAN信号各信号值的一种方法并进行说明。

一般情况下,高端一点的设备会计算每一个信号的值,但是接受到CAN信号的报文实际上有各种情况,如何通过设定的起始位和数据长度来获取某一信号的值,详情请参考,本文方法比较笨拙,如有更好的办法,欢迎指教。

假设,我们获取到的数据长度为8byte,即DLC为8。CAN Matrix表格如下:
在这里插入图片描述

我们约定,一种表示信号的方式如下:

在这里插入图片描述

lsb起始位
数据长度
按照上面的图来说,该信号起始位为40,长度为10。所以,该信号表示值的范围在0-1023之间,至于具体表示的物理量的含义需要有其他的文件规定。

获取到一帧数据以后,如何通过一系列的运算来得到信号值呢?

这里提供一种思路:

根据起始位和长度信息确定信息跨越的byte位
根据起始位确定LSB所在的bit位
根据跨越的byte位组合为数据
把以上组合的数据进行位与,得到信号值。
大概分为以上的4部分,以下分别说明:

设定:lsbbit,lsbbyte,msbbyte,start,length.按照以上的例子就是:
  lsbbit = 0
lsbbyte = 5
msbbyte = 4
start = 40
length = 10
计算lsbbit:
  lsbbit = start & 7

计算lsbbyte:
  lsbbyte = start >> 3

计算msbbyte:
  由于数据是向byte减少的方向进行的。所以:

msbbyte = lsbbyte - x

x为跨越的位数,也就是:lsbbit + length - 1所占的长度,如果大于7说明为另一行,即:

msbbyte = lsbbyte - ((lsbbit + length - 1) >> 3)

组合data:
  我们知道数据介于msbbyte-lsbbyte之间,那么我们可以组合这两个数据:

for(index = msbbyte -> (lsbbyte+1)):

data_merge += data[index] << (( lsbbyte - index ) << 3)

获取数据value:
  去尾:value = data_merge >> lsbbit

按位与:value = value & ((1 << length) - 1)

所以,最终得到的结果为筛选出来的信号值,然后根据该信号值关联上实际的物理值,就可以得到具体的物理数据了。

代码实现的方式有很多,大体思路为此。

很久以前写的,今天用的时候居然发现了一个bug,所以记录下来,表示注意。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
解析CAN信号,你需要先读取DBC文件,然后使用Java中的位运算和字节数组来解析CAN数据。以下是一个简单的示例代码,用于解析CAN数据中的一个信号: ```java import com.github.pires.obd.reader.io.CsvLogWriter; import com.github.pires.obd.reader.net.ObdReading; import com.github.pires.obd.reader.net.ObdService; import com.github.pires.obd.reader.net.ObdServiceConnection; import com.github.pires.obd.reader.net.ObdServiceHandler; import com.github.pires.obd.reader.net.ObdServiceReceiver; import com.github.pires.obd.reader.net.ObdServiceStatus; import com.github.pires.obd.reader.net.ObdServiceStatusListener; import com.github.pires.obd.reader.util.Log; import com.google.inject.Inject; import java.io.IOException; import java.util.concurrent.BlockingQueue; import java.util.concurrent.LinkedBlockingQueue; public class CanSignalParser { private DbcMessage message; private DbcSignal signal; public CanSignalParser(DbcMessage message, DbcSignal signal) { this.message = message; this.signal = signal; } public double getValue(byte[] data) { int startBit = signal.getStartBit(); int length = signal.getLength(); boolean littleEndian = signal.isLittleEndian(); int byteOffset = startBit / 8; int bitOffset = startBit % 8; if (littleEndian) { byteOffset = message.getLength() - byteOffset - length / 8; bitOffset = 8 - length % 8 - bitOffset; } int value = 0; for (int i = 0; i < length; i++) { int bit = (data[byteOffset] >> (bitOffset + i)) & 1; value |= bit << i; } double factor = signal.getFactor(); double offset = signal.getOffset(); double minValue = signal.getMinValue(); double maxValue = signal.getMaxValue(); double result = value * factor + offset; if (result < minValue) { result = minValue; } if (result > maxValue) { result = maxValue; } return result; } } ``` 这个例子接受一个`DbcMessage`对象和一个`DbcSignal`对象作为参数,并提供一个`getValue()`方法来计算信号。该方法接受一个字节数组,表示CAN数据,然后使用位运算和信号的起始位、长度、因子和偏移量来计算信号的实际。 你可以使用以下代码来调用这个方法: ```java CanSignalParser parser = new CanSignalParser(message, signal); double value = parser.getValue(canData); ``` 其中`message`和`signal`是从DBC文件中读取的`DbcMessage`和`DbcSignal`对象,`canData`是一个字节数组,表示CAN数据。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值