ID 卡号读取

<!-- /* Font Definitions */ @font-face {font-family:宋体; panose-1:2 1 6 0 3 1 1 1 1 1; mso-font-alt:SimSun; mso-font-charset:134; mso-generic-font-family:auto; mso-font-pitch:variable; mso-font-signature:3 135135232 16 0 262145 0;} @font-face {font-family:"/@宋体"; panose-1:2 1 6 0 3 1 1 1 1 1; mso-font-charset:134; mso-generic-font-family:auto; mso-font-pitch:variable; mso-font-signature:3 135135232 16 0 262145 0;} /* Style Definitions */ p.MsoNormal, li.MsoNormal, div.MsoNormal {mso-style-parent:""; margin:0cm; margin-bottom:.0001pt; text-align:justify; text-justify:inter-ideograph; mso-pagination:none; font-size:10.5pt; mso-bidi-font-size:12.0pt; font-family:"Times New Roman"; mso-fareast-font-family:宋体; mso-font-kerning:1.0pt;} /* Page Definitions */ @page {mso-page-border-surround-header:no; mso-page-border-surround-footer:no;} @page Section1 {size:595.3pt 841.9pt; margin:72.0pt 90.0pt 72.0pt 90.0pt; mso-header-margin:42.55pt; mso-footer-margin:49.6pt; mso-paper-source:0; layout-grid:15.6pt;} div.Section1 {page:Section1;} -->

韦根协议——ID 卡号读取

2010-03-31 20:52

最近在做门禁系统,第一次听说韦根协议。我第一阶段要做的事情就是用韦根读卡器将ID 卡的卡号读出来,当我拿到读卡器和ID 卡的时候感觉无从下手。可当我把韦根协议都研究透之后想想也不过如此而已,下面简单的介绍一下韦根协议。

韦根协议是国际上统一的标准,它也是一种数据传输协议。标准的26-bit 应该是最常用的格式。此外,还有34-bit 37-bit 等格式。格式的含义如下:当给出这一串数字02888888888 ,用户并不知道这串数字的含义,但如果说这是一个电话号码的时候,那么你可能就会说:哦,028 是成都的区号,而88888888 是电话号码。呵呵,不错,这正是四川航空的服务热线。但是安防行业并不愿意把这些格式公开,而安防公司也常常变化这些格式来保证产品的保密性。

而标准26-bit 格式是一个开放式的格式,这就意味着任何人都可以购买某一特定格式的HID 卡,并且这些特定格式的种类是公开可选的。26-Bit 格式就是一个广泛使用的工业标准,并且对所有HID 的用户开放。几乎所有的门禁控制系统都接受标准的26-Bit 格式。

一、Wiegand (韦根)接口

Wiegand 接口通常由3 根线组成,它们是:数据0Data0 ),数据1Data1 )和 Data return 。这3 条线负责传输Wiegand 信号。D0D1 在没有数据输出时都保持+5V 高电平。若输出为0 ,则D0 拉低一段时间,若输出为1 ,则D1 拉低一段时间。如图:

二、标准26Wiegand 通讯协议

标准韦根输出是由26 位二进制数组成,每一位的含义如下:

1 2                   9 10                                         25 26
X X X X X X X X X X X X X X X X X X X X X X X X X X       
二进制         

1 位为2—13 位的偶校验位
2—9 位对应与电子卡HID 码的低8
10-25 位对应电子卡的PID 号码
26 位为14-25 位的奇校验位
26 位数据在读出器的韦根输出线D0D1 上输出。

三、HIDPID

HID 号码即Hidden ID code 隐含码,PID 号码即Public ID code 公开码。 PID 很容易在读出器的输出结果中找到,但HID 在读出器的输出结果中部分或者全部隐掉。HID 是一个非常重要的号码,它不仅存在于卡中,也存在于读卡器中。如果卡中的HID 与读卡器中的HID 不同的话,那么这张卡就无法在这个读卡器上正常工作。

四、Wiegand 接口硬件设计

可以将Wiegand 接口的Data0Data1 两个输出接到MCU 的两个IO 脚上,采用查询的方式接收数据,但这样接收并不可靠。比较好的方法是将Data0Data1 接到MCU 的两个中断引脚上,采用中断的方式接收数据。如图:



示例程序:

 

    #include<string.h>

    #include<reg52.h>

    #include<intrins.h>

    #define DELAY_TIME 60 /* 经实验,不要小于 50 !否则可能造成时序混乱 */

    #define TRUE 1

    #define FALSE 0   

#define uchar unsigned char

#define uint unsigned int

sbit DATA1=P3^2; // 韦根读卡器的 DATA1 接单片机的 P3^2

sbit DATA0=P3^3; // 韦根读卡器的 DATA0 接单片机的 P3^3

bit Even; // 偶检验位

bit Odd;  // 奇检验位

bit CheakEven=0;// 偶检验                                                                                                                                                                                                    

bit CheakOdd=1;// 奇检验

uchar Cnt=0;//

uchar WG[26]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};// 接收韦根数据 26

uchar WGstr[3]={0,0,0};// 存韦根 ID 卡卡号的 HID PID , 其中 HID 码为 8 位即一字节, PID 16 位两字节

uchar i;

uchar j=0;

  void Init()

  {

       EA=0;

       TMOD=0x20;

       TL1=0xfd;

       TH1=0xfd;// 设置波特率为 9600 的定时器 1 为方式 2 以及初始值                

       PCON=0x00;

       SCON=0xd8;  // 设置串口方式

       IT0=1;    // 设置外部中断 0 为下降沿触发方式

       IT1=1;    // 设置外部中断 1 为下降沿触发方式

       EX0=1;   // 开外部中断 0

       EX1=1;   // 开外部中断 1

       TR1=1;     // 开定时器 1

       EA=1;        // 开所有中断

  }

void Weigand(uchar *str)  // 26 位韦根数据转换为 3 字节数据存储

{

              uchar k;

              for(k=0;k<26;k++)

              {

                     if(k<=0)  // 读偶检验位

                     {

                            if(str[k]==0)

                                   Even=0;

                            else

                                   Even=1;

                     }

                     if(k>=25) // 读奇检验位

                     {

                            if(str[k]==0)

                                   Odd=0;

                            else

                                   Odd=1;

                     }

                     if(k<=8)  // HID 码低 8

                     {

                            if(str[k]==0x00)

                                   WGstr[0]|=0x00;

                            else

                            {

                                   WGstr[0]|=0x01;

                                   CheakEven=~ CheakEven;    //   根据 HID 码低 8 1 的个数来确定偶检验位是 1 还是 0

                            }

                            if(k<8)

                            WGstr[0]=WGstr[0]<<1;

                     }

                     if(k<=16)  // PID 码高 8

                     {

                            if(str[k]==0x00)

                                   WGstr[1]|=0x00;

                            else

                            {

                                   WGstr[1]|=0x01;

                                   CheakOdd=~CheakOdd;    根据 PID 码高 8 1 的个数来确定奇检验位是 1 还是 0

                            }

                            if(k<16)

                            WGstr[1]=WGstr[1]<<1;

                     }

                     else         // PID 码的低 8

                     {

                            if(str[k]==0x00)

                                   WGstr[2]|=0x00;

                            else

                            {

                                   WGstr[2]|=0x01;

                                   CheakOdd=~CheakOdd; //   根据 PID 码低 8 1 的个数来确定奇检验位是 1 还是 0

                            }

                            if(k<24)

                            WGstr[2]=WGstr[2]<<1;

                     }

              }

}                                                                                        

  void main()

  {

              //P1=0x55;

              Init();// 调用初始化函数

              Cnt=0;

              while(1)

              {

                     if(Cnt>=26)

                     {

                            EA=0;     // 关中断以免外部中断的干扰

                            Weigand(WG);// 将读到的 26 位韦根数据转换位 3 字节格式

                            EA=1;

                            for(j=0;j<3;j++) // 用串口调试工具显示读到的卡号

                            {

                            SBUF=WGstr[j];

                            while(TI==0);

                            TI=0;

                            WGstr[j]=0;

                            }

                            Cnt=0;

                     }    

              }

  }

  void DATA0_Interrupt(void) interrupt 2 using 1   // 外部中断 1 DATA0 数据即 0

  {

     

                     WG[Cnt]=0x00;

                     Cnt++;

 

  }

  void DATA1_Interrupt(void) interrupt 0 using 2    // 外部中断 0 DATA1 数据即 1

  {

              WG[Cnt]=0x01;

              Cnt++;

 

  }

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个简单的安卓 NFC 读取卡号的示例代码: 首先,确保在 AndroidManifest.xml 文件中添加了以下权限: ``` <uses-permission android:name="android.permission.NFC" /> ``` 然后,在相应的 Activity 中添加以下代码: ```java public class MainActivity extends Activity { private NfcAdapter nfcAdapter; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 获取 NFC 适配器 nfcAdapter = NfcAdapter.getDefaultAdapter(this); if (nfcAdapter == null) { Toast.makeText(this, "设备不支持 NFC", Toast.LENGTH_SHORT).show(); finish(); return; } // 检查 NFC 是否开启 if (!nfcAdapter.isEnabled()) { Toast.makeText(this, "请在系统设置中先启用 NFC 功能", Toast.LENGTH_SHORT).show(); finish(); return; } // 创建 PendingIntent 对象,并在检测到卡片时调用 onNewIntent 方法 PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0); IntentFilter filter = new IntentFilter(NfcAdapter.ACTION_TAG_DISCOVERED); IntentFilter[] filters = new IntentFilter[]{filter}; nfcAdapter.enableForegroundDispatch(this, pendingIntent, filters, null); } @Override protected void onResume() { super.onResume(); // 检查 NFC 是否开启 if (nfcAdapter != null && !nfcAdapter.isEnabled()) { Toast.makeText(this, "请在系统设置中先启用 NFC 功能", Toast.LENGTH_SHORT).show(); finish(); return; } } @Override protected void onPause() { super.onPause(); // 关闭前台调度 if (nfcAdapter != null) { nfcAdapter.disableForegroundDispatch(this); } } @Override protected void onNewIntent(Intent intent) { super.onNewIntent(intent); String cardNumber = ByteArrayToHexString(intent.getByteArrayExtra(NfcAdapter.EXTRA_ID)); Toast.makeText(this, "卡号: " + cardNumber, Toast.LENGTH_SHORT).show(); } /** * 将字节数组转换为十六进制字符串 * @param bytes 字节数组 * @return 十六进制字符串 */ private String ByteArrayToHexString(byte[] bytes) { StringBuilder sb = new StringBuilder(); for (byte aByte : bytes) { String hex = Integer.toHexString(aByte & 0xFF); if (hex.length() == 1) { sb.append('0'); } sb.append(hex); } return sb.toString().toUpperCase(); } } ``` 在 AndroidManifest.xml 文件中,需要添加以下声明: ```xml <activity android:name=".MainActivity" android:label="@string/app_name"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> <intent-filter> <action android:name="android.nfc.action.TAG_DISCOVERED" /> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </activity> ``` 以上代码中,主要实现了以下功能: 1. 获取 NFC 适配器,并检查设备是否支持 NFC。 2. 检查 NFC 是否开启。 3. 创建 PendingIntent 对象,并在检测到卡片时调用 onNewIntent 方法。 4. 将字节数组转换为十六进制字符串,获取卡号并显示在 Toast 中。 实际使用时,还需要根据具体的需求进行修改和完善。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值