Android NFC:' d'前置于NDEF有效载荷

时间:2016-07-01 14:43:59

标签: android character-encoding format nfc ndef

我有一个应用程序,可以读取MIFARE Ultralight标签上NDEF记录的有效负载。

有效负载打包到NDEF文本记录中,文本值采用以下格式:

1,10,200,Arthur smith

所以总共有3个逗号分隔3个数字和名称。

我使用NXP TagWriter格式化标签,我的应用程序可以正确读取有效负载。问题是我工作的公司使用了一个单独的公司为我们格式化标签。现在,当我的应用程序扫描这些标签时出现问题。有效负载然后读取:

d1,10,200,Arthur smith

这会导致我的应用无法处理代码。

恩智浦应用程序可以成功读取标签,如果我使用NXP TagWriter使用相同的数据重新格式化标签,那么我的应用程序将正确读取标签。

似乎有一个' d'在来自格式化公司的有效负载前面,只有我的应用程序才能看到。

有没有人知道什么是' d'是以及如何解决这个问题?

以下是我用来读取有效负载的代码:

if (NfcAdapter.ACTION_NDEF_DISCOVERED.equals(i.getAction()) || NfcAdapter.ACTION_TAG_DISCOVERED.equals(i.getAction())) {
    if(NfcScannerApplication.isCanScanNfcTag() == true){
        messages = i.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES);
        if (messages != null) {
            //setContentView(R.layout.successfulnfc);
            NfcScannerApplication.startNfcTimer();
            //Toast.makeText(this, "NFC timer set", Toast.LENGTH_LONG).show();
            Log.e(TAG, "Found " + messages.length + " NDEF messages"); // is almost always just one
            vibrate(); // signal found messages :-)

            // parse to records
            for (int i = 0; i < messages.length; i++) {
                List<Record> records = null;
                try {
                    records = new Message((NdefMessage)messages[i]);
                } catch (FormatException e) {
                    e.printStackTrace();
                }
                Log.e(TAG, "Found " + records.size() + " records in message " + i);

                for(int k = 0; k < records.size(); k++) {
                    Log.e(TAG, " Record #" + k + " is of class " + records.get(k).getClass().getSimpleName());

                    Record record = records.get(k);
                    NdefRecord ndefRecord = record.getNdefRecord();
                    byte[] arr = ndefRecord.getPayload();

                    String payload = new String(arr);

                    if(payload.length() > 0){
                        payload = payload.substring(3, payload.length());

                        Log.e(TAG, "payload = " + payload);

                        int counter = 0;
                        for( int z = 0;  z < payload.length(); z++ ) {
                            if( payload.charAt(z) == ',' ) {
                                counter++;
                            }
                        }

                        if(counter == 3){
                            //Toast.makeText(this, "comma count = 3 ", Toast.LENGTH_SHORT).show();
                            String[] splitPayload = payload.split(",");
                            String tagType = splitPayload[0];
                            String tagCompany = splitPayload[1];
                            String tagClientID = splitPayload[2];
                            String tagClientName = splitPayload[3];

3 个答案:

答案 0 :(得分:1)

您的代码无法正确解码NDEF文本记录的有效内容。您假设有效负载是UTF-8编码(Android上的默认编码),实际文本值从该解码字符串的偏移量3开始:

NdefRecord ndefRecord = record.getNdefRecord();
byte[] arr = ndefRecord.getPayload();
String payload = new String(arr);
payload = payload.substring(3, payload.length());

然而,这些假设是错误的。文本记录由状态字节,后跟变量(!)长度语言代码(US-ASCII编码)和文本值本身(以UTF-8或UTF-16编码)组成。因此,额外的&#34; d&#34;你看到的可能实际上是语言代码的一部分,或者可能是将错误的字符代码(例如状态字节的值)解码为UTF-8字符串的结果。

因此,为了获得文本值,首先要使用Status字节来查找语言代码的长度和文本值的编码:

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");

答案 1 :(得分:0)

我的猜测是,它可能是格式化公司对他们创建的标签所做的事情,以确保您的公司不能轻易地从它们切换。

但是,我不明白为什么你不能只检查有效载荷的初始“d”,然后只要发现它就剥离它。

答案 2 :(得分:0)

事实证明,NXP TagWiter应用程序中存在一个错误,该错误在“&#39;”之前。 我被迫等待恩智浦的更新解决了这个问题。

感谢您的帮助。