我正在尝试在Mifare Classic 1k NFC标签上读取和写入数据。
由于this app:
,我找到了卡的密钥和访问条件 <uses-permission android:name="android.permission.NFC" />
出现在我的清单中。
这是我的代码:
Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
if(tag != null) {
Log.i("hey", Arrays.toString(tag.getTechList()));
MifareClassic mfc = MifareClassic.get(tag) ;
try {
mfc.connect();
boolean authA = mfc.authenticateSectorWithKeyA(2, MifareClassic.KEY_NFC_FORUM) ;
boolean authB = mfc.authenticateSectorWithKeyB(2, MifareClassic.KEY_DEFAULT) ;
Log.i("hey", "authA : " + authA) ;
Log.i("hey", "authB : " + authB) ;
if (authB && authA) {
byte[] bWrite = new byte[16];
byte[] hello = "hello".getBytes(StandardCharsets.US_ASCII);
System.arraycopy(hello, 0, bWrite, 0, hello.length);
mfc.writeBlock(0, bWrite);
Log.i("hey", "write : " + Arrays.toString(bWrite));
byte[] bRead = mfc.readBlock(0);
String str = new String(bRead, StandardCharsets.US_ASCII);
Log.i("hey", "read bytes : " + Arrays.toString(bRead));
Log.i("hey", "read string : " + str);
Toast.makeText(this, "read : " + str, Toast.LENGTH_SHORT).show();
Log.i("hey", "expected : " + new String(bWrite, StandardCharsets.US_ASCII));
}
mfc.close();
} catch (IOException e) {
Toast.makeText(this, "Error", Toast.LENGTH_SHORT).show();
e.printStackTrace();
Log.i("hey", "Error") ;
}
}
当我尝试这样写作和阅读时,我读到的并不是我写的。
这是logcat:
I/hey: [android.nfc.tech.NfcA, android.nfc.tech.MifareClassic, android.nfc.tech.NdefFormatable]
I/hey: authA : true
authB : true
I/hey: write : [104, 101, 108, 108, 111, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
I/hey: read byte : [-78, 0, 0, 0, 0, 0, -1, 7, -128, 105, -1, -1, -1, -1, -1, -1]
read string : �������������i������
I/hey: expected : hello����������������������
我试过不同的字符集,但它没有改变任何东西。
我还尝试只通过注释写入部分并将身份验证密钥A更改为MifareClassic.KEY_MIFARE_APPLICATION_DIRECTORY
来读取扇区0,这次我得到了这个Logcat:
I/hey: [android.nfc.tech.NfcA, android.nfc.tech.MifareClassic, android.nfc.tech.NdefFormatable]
I/hey: authA : true
authB : true
I/hey: read bytes : [-123, -121, 0, 16, 18, 8, 4, 0, 1, -64, 62, -70, 74, 124, 78, 29]
read string : �������>�J|N
有关如何修复它以正确编写和显示文本的任何想法?
答案 0 :(得分:3)
首先对第2部分进行身份验证:
boolean authA = mfc.authenticateSectorWithKeyA(2, MifareClassic.KEY_NFC_FORUM);
boolean authB = mfc.authenticateSectorWithKeyB(2, MifareClassic.KEY_DEFAULT);
然后,您尝试编写并读取块0:
mfc.writeBlock(0, bWrite);
byte[] bRead = mfc.readBlock(0);
这有几个问题:
使用密钥A和密钥B进行身份验证没有意义。只有最后一次身份验证才能确定标记的身份验证状态。由于所有扇区似乎都可以使用密钥B写入,因此您可以安全地使用第二行(仅mfc.authenticateSectorWithKeyB()
)。
您对扇区2进行身份验证,扇区2由块8,9,10和11组成。写入和读取块0在该身份验证状态下没有意义。请注意,读/写操作通常使用全局块编号。扇区仅用作身份验证/访问控制目的的逻辑单元。您可以使用mfc.sectorToBlock()
和mfc.getBlockCountInSector()
轻松计算行业编号中的块编号。
块0(扇区0)是一个特殊块,即制造商块。它通常包含在生产过程中刻录到标签中的UID,SAK和制造商信息。该块是只读的,无法更改。因此,写入该块不会产生任何影响。