我正在使用Mifare Ultralight C标签并以NDEF格式向他们写入数据。使用NDEF-NFC library为我的NDEF消息创建原始字节。我的NDEF消息是
var spRecord = new NdefTextRecord {
Text = "1",
LanguageCode = "en"
};
var msg = new NdefMessage { spRecord };
我得到的输出是D1 01 04 54 02 65 6E 31
(十六进制)。如果我按原样将此字节数组写入标记:
按照 Writing Ndef to NFC tag in Windows Form Application C# using ACR122U as Reader/Writer 中的建议,我修改了我的字节数组以编写03 08 D1 01 04 54 02 65 6E 31 FE 00
,而不是库生成的原始字节数组。
但是在尝试使用库将字节数组转换为NDEF消息对象时,我收到以下错误:
Ndef解析错误:第一条记录中缺少预期的消息开始。
如何正确回读NDEF消息?
答案 0 :(得分:3)
正如您已经发现的,NFC论坛类型2标签(例如MIFARE Ultralight,NTAG等)需要将NDEF消息嵌入到NDEF TLV(标签长度值)结构中。这意味着您将标记03
和NDEF消息的长度添加到消息(值)本身。因此,你得到了
+-----+--------+-------------------------+
| TAG | LENGTH | VALUE |
| 03 | 08 | D1 01 04 54 02 65 6E 31 |
+-----+--------+-------------------------+
此外,您可以添加终结者TLV(标记= FE
,长度= 00
),以指示可以跳过标记上的剩余数据区域。
您使用的NDEF库仅处理 NDEF消息,而不处理在NFC标记上存储数据所需的容器格式。因此,您需要自己处理该部分。
var msg = new NdefMessage { ... };
var msgBytes = msg.toByteArray();
var ndefTlvLen = new byte[(msgBytes.Length < 255) ? 1 : 3];
if (msgBytes.Length < 255) {
ndefTlvLen[0] = (byte)(msgBytes.Length);
} else {
ndefTlvLen[0] = (byte)0x0FF;
ndefTlvLen[1] = (byte)((msgBytes.Length >> 8) & 0x0FF);
ndefTlvLen[2] = (byte)(msgBytes.Length & 0x0FF);
}
var tagData = new byte[1 + ndefTlvLen.Length + msgBytes.Length + 2];
int offset = 0;
tagData[offset++] = (byte)0x003;
Array.Copy(ndefTlvLen, 0, tagData, offset, ndefTlvLen.Length);
offset += ndefTlvLen.Length;
Array.Copy(msgBytes, 0, tagData, offset, msgBytes.Length);
offset += msgBytes.Length;
tagData[offset++] = (byte)0x0FE;
tagData[offset++] = (byte)0x000;
var tagData = ...; // byte[]
var msg;
int offset = 0;
while (offset < tagData.Length) {
byte tag = tagData[offset++];
int len = (tagData[offset++] & 0x0FF);
if (len == 255) {
len = ((tagData[offset++] & 0x0FF) << 8);
len |= (tagData[offset++] & 0x0FF);
}
if (tag == (byte)0x03) {
var msgBytes = new byte[len];
Array.Copy(tagData, offset, msgBytes, 0, len);
msg = NdefMessage.FromByteArray(msgBytes);
} else if (tag == (byte)0xFE) {
break;
}
offset += len;
}