其他分享
首页 > 其他分享> > 读取NFC标签时出现奇怪字符

读取NFC标签时出现奇怪字符

作者:互联网

我正在尝试使用Android读取NFC标签.我是养蜂人,当我接近他们时,这是为了识别我的蜂箱.我在这里搜索过,但是在阅读标签时仍然遇到问题.我想阅读文本,但是当阅读时,有一个正方形字符,并且在所需文本之前显示的字符类似“十”.

这是我正在使用的代码.我知道有效负载字节必须正确,并且我尝试更改它们但无济于事.

private static NdefMessage getTestMessage() {
    byte[] mimeBytes = "application/com.android.cts.verifier.nfc"
            .getBytes(Charset.forName("US-ASCII"));
    byte[] id = new byte[] {1, 3, 3, 7};
    byte[] payload = "CTS Verifier NDEF Push Tag".getBytes(Charset.forName("US-ASCII"));
    return new NdefMessage(new NdefRecord[] {
            new NdefRecord(NdefRecord.TNF_MIME_MEDIA, mimeBytes, id, payload)
    });
}

@Override
protected void onResume() {
    super.onResume();

    mNfcAdapter.enableForegroundDispatch(this, mPendingIntent, null, null);
    mNfcAdapter.setNdefPushMessageCallback(this, this);
}

// sending message
@Override
public NdefMessage createNdefMessage(NfcEvent event) {
    return getTestMessage();
}


private NdefMessage[] getNdefMessages(Intent intent) {
    Parcelable[] rawMessages = intent
      .getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES);
    if (rawMessages != null) {
        NdefMessage[] messages = new NdefMessage[rawMessages.length];
        for (int i = 0; i < messages.length; i++) {
            messages[i] = (NdefMessage) rawMessages[i];
        }
        return messages;
    } else {
        return null;
    }
}

static String displayByteArray(byte[] bytes) {
    String res="";
    StringBuilder builder = new StringBuilder().append("");
    for (int i = 0; i < bytes.length; i++) {
        res+=(char)bytes[i];
    }
    return res;

}

// displaying message
@Override
protected void onNewIntent(Intent intent) {
    super.onNewIntent(intent);

    NdefMessage[] messages = getNdefMessages(intent);
    edtUser.setText(displayByteArray(messages[0].toByteArray()));

    Toast.makeText(this, "NFC tag entered", Toast.LENGTH_LONG).show();
}

解决方法:

由于您尝试将整个原始NDEF记录显示为文本字符串(使用某种奇数解码方法),因此在显示消息时,您将得到奇数个附加字符:

NdefMessage[] messages = getNdefMessages(intent);
edtUser.setText(displayByteArray(messages[0].toByteArray()));

这有几个问题.首先,您通常希望使用与编写文本相同的编码来解码文本.例如,如果您使用

String text = "...";
byte[] bytes = text.getBytes(Charset.forName("US-ASCII"));

要获得给定文本字符串的US-ASCII编码字节数组,您还希望使用相同的US-ASCII编码再次将字节转换为文本字符串:

byte[] bytes = ...;
String text = new String(bytes, "US-ASCII");

其次,您将整个NDEF消息解释为文本字符串.但是,您存储在标签上的文本通常仅包含在NDEF记录的有效载荷之内.在您的情况下,前缀“十”表示您使用的语言指示为“ en”的NFC论坛文本记录(类型名称“ T”)(对于英语).因此,您需要在NDEF消息中搜索文本记录:

for (NdefRecord r : messages[0].getRecords()) {
    if (r.getTnf() == NdefRecord.TNF_WELL_KNOWN) {
        if (Arrays.equals(r.getType(), NdefRecord.RTD_TEXT)) {

一旦找到了文本记录,就可以对其文本有效载荷进行解码.有效负载由状态字节,语言字段和实际文本组成:

            byte[] payloadBytes = ndefRecord.getPayload();
            boolean isUTF8 = (payloadBytes[0] & 0x080) == 0;  //status byte: bit 7 indicates encoding (0 = UTF-8, 1 = UTF-16)
            int languageLength = payloadBytes[0] & 0x03F;     //status byte: bits 5..0 indicate length of language code
            int textLength = payloadBytes.length - 1 - languageLength;
            String languageCode = new String(payloadBytes, 1, languageLength, "US-ASCII");
            String payloadText = new String(payloadBytes, 1 + languageLength, textLength, isUTF8 ? "UTF-8" : "UTF-16");

            edtUser.setText(payloadText);
        }
    }
}

标签:ndef,character-encoding,format,nfc,android
来源: https://codeday.me/bug/20191109/2011633.html