我有一个应用程序,可以读取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];
答案 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;”之前。 我被迫等待恩智浦的更新解决了这个问题。
感谢您的帮助。