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

tijins/NfcLib

About the NDEF Format

MIFARE Ultralight 系列

RC-S380 非接触ICカードリーダー/ライター PaSoRi(パソリ)

Windows读取NXP MiFare Ultralight C类型NFC卡片的信息的更多相关文章

  1. 射频识别技术漫谈(16)——Mifare UltraLight

    Mifare UltraLight又称为MF0,从UltraLight(超轻的)这个名字就可以看出来,它是一个低成本.小容量的卡片.低成本,是指它是目前市场中价格最低的遵守ISO14443A协议的芯片 ...

  2. php 读取文件头判断文件类型的实现代码

    php代码实现读取文件头判断文件类型,支持图片.rar.exe等后缀. 例子: <?php $filename = "11.jpg"; //为图片的路径可以用d:/uploa ...

  3. Windows上常见的集中布尔类型的比较

    博客搬到了fresky.github.io - Dawei XU,请各位看官挪步.最新的一篇是:Windows上常见的集中布尔类型的比较.

  4. Windows Phone 同步方式获取网络类型

    原文:Windows Phone 同步方式获取网络类型 在Windows Phone 开发中有时候需要获取设备当前连接网络的类型,是Wifi,还是2G,3G,或者4G,SDK中提供获取网络类型的API ...

  5. python3 在 windows 读取路径多了一个\u202a 是咋回

    python3 在 windows 读取路径多了一个\u202a 是咋回事

  6. wpf prism4 出现问题:无法加载一个或多个请求的类型。有关更多信息,请检索 LoaderExceptions 属性。

    WPF Prism 框架 程序 出现 问题: 无法加载一个或多个请求的类型.有关更多信息,请检索 LoaderExceptions 属性. 1.开始以为是配置的问题,找了半天,最后原来是有个依赖类库没 ...

  7. 无法加载一个或多个请求的类型。有关更多信息,请检索 LoaderExceptions 属性。

    新建一个MVC4的项目,引用DAL后,将DAL的连接字符串考入: <connectionStrings>     <add name="brnmallEntities&qu ...

  8. “无法加载一个或多个请求的类型。有关更多信息,请检索 LoaderExceptions 属性 “之解决

    今天在学习插件系统设计的时候遇到一个问题:“System.Reflection.ReflectionTypeLoadException: 无法加载一个或多个请求的类型. 于是百度一下,很多内容都差不多 ...

  9. Sqlserver查询表结构信息-字段说明、类型、长度等信息

    Sqlserver 中查询表结构信息-字段说明.类型.长度等信息综合语法. SELECT 表名 = d.name,--case when a.colorder=1 then d.name else ' ...

随机推荐

  1. 洛谷P2744 [USACO5.3]量取牛奶Milk Measuring

    题目描述 农夫约翰要量取 Q(1 <= Q <= 20,000)夸脱(夸脱,quarts,容积单位--译者注) 他的最好的牛奶,并把它装入一个大瓶子中卖出.消费者要多少,他就给多少,从不有 ...

  2. 创建APP检查更新页

    本文来源及参考:Create a check for updates page for your app. 这篇文章解释了如何创建一个简单的检查更新页,检查该用户已安装的应用程序的最新版本. 简介 这 ...

  3. 55. GridPanel中getSelectionModel详解

    转自:https://blog.csdn.net/qq_29663071/article/details/50728429 本文导读:Ext.grid.GridPanel继承自Panel,其xtype ...

  4. Finding Nemo(搜索)

    http://poj.org/problem?id=2049 题意:有一个迷宫,迷宫中有墙.门和空地.有M道墙,每一道墙用(x,y,d,t)表示,(x,y)表示墙的起始坐标,(d=1,t)表示向上t个 ...

  5. [Swift通天遁地]五、高级扩展-(12)扩展故事板中的元件添加本地化功能

    ★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★➤微信公众号:山青咏芝(shanqingyongzhi)➤博客园地址:山青咏芝(https://www.cnblogs. ...

  6. 前端性能优化---减少http请求数量和减少请求资源的大小

    减少http请求数量:就是资源的合并 减少http请求大小:就是资源的压缩   一.资源合并的原理:   资源不合并的缺点: 1.文件和文件之间有插入请求----请求a.js,b.js,c.js(三行 ...

  7. C#用Microsoft.Office.Interop.Word进行Word转PDF的问题

    之前用Aspose.Word进行Word转PDF发现'\'这个字符会被转换成'¥'这样的错误,没办法只能换个方法了.下面是Microsoft.Office.Interop.Word转PDF的方法: p ...

  8. 笨拙而诡异的 Oracle

    有这样一段 SQL 代码: 通过 C# 获取查询结果:    SQL 代码中有两个参数,且都是字符串类型,以上的 C# 代码是生成 Oracle SQL 代码所需要的参数.运行结果如下:   居然发生 ...

  9. asp.net——登录界面

    题目: 在页面中放入两个TextBox分别用于输入用户名和密码,一个显示文字用的Label,一个提交按钮Button.点击按钮提交用户名和密码,如果用户名都密码正确(比如用户名admin 密码abc1 ...

  10. Echarts 出现不明竖线解决方案

    Echarts出现了不明竖线,百思不得其解.去查相应的解决方案也没有找到. 后来自己点来点去,突然感觉像是上一个Echarts遗留的. 然后去Echarts官网看到了 clear()方法,这个方法可以 ...