Windows读取NXP MiFare Ultralight C类型NFC卡片的信息
1,读取
我们需要外接一个NFC Reader让Windows可以读取NFC卡片的内容。
因为特殊原因,我们选择了Sony rc-s380 NFC Reader。相关介绍
我们需要下载并安装NFC Port Software,以便我们可以顺利使用上述NFC Reader。
安装之后,我们需要去Github获取一个NFC Port Software的.Net封装包:tijins/NfcLib
解压并运行nfc_lib_sample之后我们可以看到这样一个WindowFrom程序。

因为现在要读取的卡类型试MIFARE,所以我们勾选正确的CheckBox,其他设置保持不变,依次点击上述三个橘色方框按钮。
在红色箭头所指的方向就能看到第一个block的十六进制字符串的数据。
为了读取卡片所有block/page的信息,我们将btRead_Click的代码稍稍修改为:
private void btRead_Click(object sender, EventArgs e)
{
byte block = (byte)nudBlock.Value;
try
{
byte[] data = null;
data = new byte[NfcLib.MF_BLOCK_LENGTH];
if (card is Mifare)
{
StringBuilder readSB = new StringBuilder();
for(byte i=;i < ; i++)//我现在的卡包含45个blocks/pages
{
((Mifare)card).Read(i, data, );
readSB.AppendLine(Utility.ByteToHex(data, , data.Length));
}
tbRead.Text = readSB.ToString();
}
}
catch (Exception ex)
{
MessageBox.Show(ex.Message);
}
}
有个小遗憾就是,如果NFC Reader没有连接的时候启动app会抛出异常,而如果app已经启动再拔出然后插入NFC Reader又会无法再读卡除非重启app或者重新初始化。这个需要有点改进。
2,解析
根据上述步骤,我们得到了某张卡的信息如下:
C9 0A C9 5A 1D E1
0A C9 5A 1D E1 A0 0C
1D E1 A0 0C 0F D1
E1 A0 0C 0F D1 0B
A0 0C 0F D1 0B 6E
0F D1 0B 6E
0B 6E FE
6E FE
FE
FE (以下省略...)
然后,我们对它做一点移位:
04 C9 02 47 0A C9 5A 84 1D 48 00 00 E1 10 12 00
0A C9 5A 84 1D 48 00 00 E1 10 12 00 01 03 A0 0C
1D 48 00 00 E1 10 12 00 01 03 A0 0C 34 03 0F D1
E1 10 12 00 01 03 A0 0C 34 03 0F D1 01 0B 54 02
01 03 A0 0C 34 03 0F D1 01 0B 54 02 65 6E 32 30
34 03 0F D1 01 0B 54 02 65 6E 32 30 31 38 31 30
01 0B 54 02 65 6E 32 30 31 38 31 30 32 35 FE 00
65 6E 32 30 31 38 31 30 32 35 FE 00 00 00 00 00
31 38 31 30 32 35 FE 00 00 00 00 00 00 00 00 00
32 35 FE 00 00 00 00 00 00 00 00 00 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
可以看出,每一行只有第一列是我们需要的tag信息。
C9
0A C9 5A
1D
E1 00 A0 0C
0F D1
0B
6E FE
前面的四行是卡的UID以及制造商等信息。我们只需要关心下半段的数据。
(Tag: Lock Control TLV)
(Length: bytes)
A0 0C (Value: Information on position and function of lock bytes)
(Tag: NDEF Message TLV)
0F (Length: bytes)
D1 0B 6E
FE (Tag: Terminator TLV; has no length field)
所以,我们得到了NDEF message:D1 01 0B 54 02 65 6E 32 30 31 38 31 30 32 35
NDEF message:
D1 (Header byte of record )
(Type length: byte)
0B (Payload length: bytes)
(Type: "T")
6E (Payload field)
我们再分析最为关键的payload field,如下:
The payload field:
(Status byte: Text is UTF- encoded, Language code has a length of bytes)
6E (Language code: "en")
(Text: "")
可以用在线工具来验证一下 “32 30 31 38 31 30 32 35 ”
最后,上述步骤,用写C#的实现如下(不保证所有Mifare卡适用):
private string GetNfcTag(List<byte[]> cardContentList)
{
List<byte> cardBytes = new List<byte>();
foreach(byte[] rowCardContent in cardContentList)
{
cardBytes.AddRange(rowCardContent.Take());
} byte[] cardUid = cardBytes.Take().ToArray();
string cardUidStr = Utility.ByteToHex(cardUid, , cardUid.Length); byte[] cardMaker = cardBytes.Skip().Take().ToArray();
string cardMakerStr = Utility.ByteToHex(cardMaker, , cardMaker.Length); byte[] memoryBytes = cardBytes.Skip().ToArray(); byte lockControlByte = memoryBytes[];
byte lockByteCount = memoryBytes[];//how many bytes are the lock bytes
byte[] lockBytes = memoryBytes.Skip().Take(lockByteCount).ToArray(); int nedfLengthByteIndex = + lockByteCount + + ;
byte ndefLength = memoryBytes[nedfLengthByteIndex]; if (ndefLength == )//Tag empty
return string.Empty; byte[] ndefBytes = memoryBytes.Skip(nedfLengthByteIndex + ).Take(ndefLength).ToArray(); byte payloadHeader = ndefBytes[];
byte payloadTypeLength = ndefBytes[];
byte payloadLength = ndefBytes[];
byte[] payloadType = ndefBytes.Skip().Take(payloadTypeLength).ToArray(); byte[] payloadBytes = ndefBytes.Skip( + payloadType.Length).Take(payloadLength).ToArray(); byte languageCodeLength = payloadBytes[];
byte[] languageCode = payloadBytes.Skip().Take(languageCodeLength).ToArray();
byte[] tag = payloadBytes.Skip( + languageCode.Length).Take(payloadLength - languageCode.Length - ).ToArray(); string tagHex= Utility.ByteToHex(tag , , tag .Length);
string tagStr = Encoding.UTF8.GetString(tag);
return tagStr;
}
参考链接:
How to read binary blocks of mifare card?
Reading a NFC Mifare card raw data in android
RC-S380 非接触ICカードリーダー/ライター PaSoRi(パソリ)
Windows读取NXP MiFare Ultralight C类型NFC卡片的信息的更多相关文章
- 射频识别技术漫谈(16)——Mifare UltraLight
Mifare UltraLight又称为MF0,从UltraLight(超轻的)这个名字就可以看出来,它是一个低成本.小容量的卡片.低成本,是指它是目前市场中价格最低的遵守ISO14443A协议的芯片 ...
- php 读取文件头判断文件类型的实现代码
php代码实现读取文件头判断文件类型,支持图片.rar.exe等后缀. 例子: <?php $filename = "11.jpg"; //为图片的路径可以用d:/uploa ...
- Windows上常见的集中布尔类型的比较
博客搬到了fresky.github.io - Dawei XU,请各位看官挪步.最新的一篇是:Windows上常见的集中布尔类型的比较.
- Windows Phone 同步方式获取网络类型
原文:Windows Phone 同步方式获取网络类型 在Windows Phone 开发中有时候需要获取设备当前连接网络的类型,是Wifi,还是2G,3G,或者4G,SDK中提供获取网络类型的API ...
- python3 在 windows 读取路径多了一个\u202a 是咋回
python3 在 windows 读取路径多了一个\u202a 是咋回事
- wpf prism4 出现问题:无法加载一个或多个请求的类型。有关更多信息,请检索 LoaderExceptions 属性。
WPF Prism 框架 程序 出现 问题: 无法加载一个或多个请求的类型.有关更多信息,请检索 LoaderExceptions 属性. 1.开始以为是配置的问题,找了半天,最后原来是有个依赖类库没 ...
- 无法加载一个或多个请求的类型。有关更多信息,请检索 LoaderExceptions 属性。
新建一个MVC4的项目,引用DAL后,将DAL的连接字符串考入: <connectionStrings> <add name="brnmallEntities&qu ...
- “无法加载一个或多个请求的类型。有关更多信息,请检索 LoaderExceptions 属性 “之解决
今天在学习插件系统设计的时候遇到一个问题:“System.Reflection.ReflectionTypeLoadException: 无法加载一个或多个请求的类型. 于是百度一下,很多内容都差不多 ...
- Sqlserver查询表结构信息-字段说明、类型、长度等信息
Sqlserver 中查询表结构信息-字段说明.类型.长度等信息综合语法. SELECT 表名 = d.name,--case when a.colorder=1 then d.name else ' ...
随机推荐
- App上架流程 & 上架被拒10大原因
上架前预热 先登陆自己的开发者账号(自己提前注册好 iOS 开发者账号,这里假设你已经拥有了一个 iOS 开发者账号),进入这个页面:https://developer.apple.com/accou ...
- Coursera Algorithms Programming Assignment 5: Kd-Trees (98分)
题目地址:http://coursera.cs.princeton.edu/algs4/assignments/kdtree.html 分析: Brute-force implementation. ...
- go 简单路由实现
一.golang 路由实现的简单思路 1.http启动后,请求路径时走统一的入口函数 1.通过统一函数入口,获取request 的url路径 2.通过对url的路径分析,确定具体执行什么函数 二.统一 ...
- 参加2016华为codecraft编程精英挑战赛后感
2016年4月参加了华为的软件比赛. 关于比赛:给了一道图论的np-hard问题.刚开始完全不知道怎么入手,请教过师兄,自己也琢磨过,没有什么万全的解决方法.注意,这里说的是万全的办法.本科搞算法时候 ...
- Unity相机平滑跟随
简介 unity中经常会用到固定视角的相机跟随,然后百度发现大家都是自己写的,然后偶也写咯一个,分享一下 PS: 由于刚学C#不久,才发现delegate这个东东,也不知道对性能影响大不大,但是看MS ...
- reactnative(1) - RefreshControl 使用案例
'use strict'; import React, { Component } from 'react'; import { AppRegistry, ScrollView, StyleSheet ...
- jQuery五屏轮播手风琴切换代码
jQuery五屏轮播手风琴切换代码 在线演示本地下载
- Git的使用及托管代码到GitHub
首先Git是一个开源的分布式版本控制工具,用git创建代码仓库. 仓库(Repository)是用于保存版本管理所需信息的地方,本地代码 提交到 代码仓库中,如果需要还可以 再推送到 远程仓库中. 所 ...
- ScrollView在调试状态一点击就挂的原因(OnMouseActivate)
这几天做的一个任务是做一个Dialog,需要在这个Dialog中添加一个自定义的CSrollvew类,但是遇到一个比较扯淡的问题,程序直接运行时可以的,调试状态下一点击CSrollview就挂了.而且 ...
- JS高级——歌曲管理
1.将歌曲管理的CURD方法放到原型中 2.在构造函数中,我们只有一个属性是songList,因为音乐库不是共有的,如果将songList放入原型中,任何一个人的一次修改songList,都将把son ...