RFID簽到系統

RFID簽到系統

這段時間幫學長做課設,以前有幾個項目找到的,想分享給大家,去幫助課設還沒有做的。
寫的草,多多包涵。
下位機模塊用有原子哥的代碼。

目的是:用RFID-RC522去識別RFID卡(即一般飯卡)然後在LCD屏上顯示,並顯示現有多少人,簽到簽退情況。用上位機去記錄簽到簽退時間等。有能力的話還能顯示誰沒到勤等。此外下位機還可以設置簽到時間,是否遲到等。


材料

然後就開始做了。

製作過程

  • 硬件部分
  1. 首先要明白RFID讀卡器是怎麼識別每張卡的。這裏指的是數據,我們不考慮怎麼區分同時識別兩張卡等問題,我們只關心每張卡的數據形式,以及我們怎麼區分這些卡;
    經過測試,這些卡的數據形式是16進制。列出來一張:
    04 0C 02 21 00 04 00 64 4C AF 5B 0C
    注意是16進制的

  2. 我們不管他的工作原理。

  3. 不考慮兩張卡即以上同時識別。一次就是別一張。

  4. 然後將模塊的引腳連接上。只連接VCC、GND、TX、RX四個引腳就好了,連接到板子的串口2上。因爲串口1要連接上位機。

我們先將模塊連接到USB轉TTL上連接到電腦上
這樣–>連接USB轉TTL
然後打開串口調試助手連接。
還要提醒下VCC和GND千萬不要接反了!!
這樣–>
串口這裏我們就能看到數據格式了,這個很重要,我們以後要用到。
做完這些後就可以寫程序了。這裏直說思想和難點。
思想就是:兩個串口,串口2用來接收模塊發來的碼,串口1用來給上位機發送信息。定義一個二維數組,模塊每識別一次卡,向單片機發送一次數據,把這個數據儲存到一個數組中,並人數加一。然後第二次識別卡,當識別的和第一次不同時就儲存到數組的另一個位置。當下一次識別的碼與前的某一次一樣時,就刪除這個數組。並人數減一。然後每一次設別都將設別的卡號和簽到還是簽退情況發送給上位機。

首先要寫兩個串口,分別都能接受數據。

int fputc(int ch, FILE *f)
{      
 while((USART1->SR&0X40)==0);//循環發送直至發送完畢
    USART1->DR = (u8) ch; //串口1要發送給上位機,所以print函數要用串口1
 return ch;
}

初始化兩個串口,串口1和串口2

void uart2_init(u32 bound)
{
  GPIO_InitTypeDef GPIO_InitStructure;
  USART_InitTypeDef USART_InitStructure;
  NVIC_InitTypeDef NVIC_InitStructure;
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); 
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);
  
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //PA.9
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; 
  GPIO_Init(GPIOA, &GPIO_InitStructure);//³õʼ»¯GPIOA.9
  
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;//PA10
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  GPIO_Init(GPIOA, &GPIO_InitStructure);
  
  NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;  
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;   
  NVIC_Init(&NVIC_InitStructure);
 
 USART_InitStructure.USART_BaudRate = bound;
 USART_InitStructure.USART_WordLength = USART_WordLength_8b;
 USART_InitStructure.USART_StopBits = USART_StopBits_1;
 USART_InitStructure.USART_Parity = USART_Parity_No;
 USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
 USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
 
  USART_Init(USART2, &USART_InitStructure); 
  USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);
  USART_Cmd(USART2, ENABLE);
}

//串口2中斷函數
void USART2_IRQHandler(void)                 
 {
 u8 Res;
#if SYSTEM_SUPPORT_OS   
 OSIntEnter();    
#endif
 if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET)  
  {
  Res =USART_ReceiveData(USART2); 
  if(USART_RX_BUF[10]!=0)
   USART_RX_STA|=0x8000;
  if((USART_RX_STA&0x8000)==0)
   {
   if(USART_RX_STA&0x4000)
    {
    if(Res!=0x0a)USART_RX_STA=0;
    else USART_RX_STA|=0x8000; 
    }
   else
    { 
    if(Res==0x0d)USART_RX_STA|=0x4000;
    else
     {
     USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
     USART_RX_STA++;
     if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0; 
     }   
    }
   }      
     } 
#if SYSTEM_SUPPORT_OS  
 OSIntExit();              
#endif
} 

串口1類似,就不一一寫了。
接下來就是接收模塊的數據處理。

  if(USART_RX_STA&0x8000)
  {
  len=USART_RX_STA&0x3fff;
   LCD_ShowString(0,100,210,24,24,"                   ");
if(len>9)
   {
    for(t=0;t<len;t++)
    {
    USART_SendData(USART1, USART_RX_BUF[t]);
     while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);
    }   
    if(USART_RX_BUF[0]==0x04&&USART_RX_BUF[1]==0x0C&&USART_RX_BUF[2]==0x02&&USART_RX_BUF[3]==0x21&&USART_RX_BUF[4]==0x00&&USART_RX_BUF[5]==0x04&&USART_RX_BUF[6]==0x00&&USART_RX_BUF[7]==0x64&&USART_RX_BUF[8]==0x4c&&USART_RX_BUF[9]==0xaf)
    {
     m++;
     times=0; 
     printf("123");
     if(m%2==1)
     {
     num++;
      LCD_ShowString(0,200,210,24,24,"                   ");
      LCD_ShowChinese(91,200,5,24,1);   
      LCD_ShowChinese(115,200,6,24,1);  
     }    
     if(m%2==0)
     {
      num--;
      LCD_ShowString(0,200,210,24,24,"                   ");
      LCD_ShowChinese(91,200,7,24,1); 
      LCD_ShowChinese(115,200,8,24,1); 
     }
    }
    else 
    {
     n++;
     times=0;
     printf("456");
     if(n%2==1)
     { 
      num++;
      LCD_ShowString(0,200,210,24,24,"                   ");
      LCD_ShowChinese(91,200,5,24,1);   
      LCD_ShowChinese(115,200,6,24,1);  
     }    
     if(n%2==0)
     {
      num--;
      LCD_ShowString(0,200,210,24,24,"                   ");
      LCD_ShowChinese(91,200,7,24,1); 
      LCD_ShowChinese(115,200,8,24,1); 
     }
    }
   }
   if(strcmp(USART_RX_BUF,"2")==0)
    LED1=!LED1;
   USART_RX_STA=0x00000;
   for(i=0;i<USART_REC_LEN;i++)
   {
    USART_RX_BUF[i]=0;
   }
   USART_RX_STA=0x00000;
   USART_RX_BUF[9]=0;
  }
  else
  {
   times++;
   if(times<=30)BEEP=1;
   else BEEP=0;
   if(times<=200)LED0=0;
   else LED0=1;
   delay_ms(10);   
  }

這段代碼的意思就是接收出去然後去對比,一樣就把對比到的數組刪了,不同就添加到一個數組。同時人數做加減。蜂鳴器響一聲。然後將接收到的數據通過串口1發送給上位機。
這樣就解決了數據處理問題,接下來就是顯示了。
顯示調用字庫。參考LD3320語音模塊,有說怎麼顯示漢字的,這裏就不說了。顯示漢字人數,簽到簽退等情況。
在這裏插入圖片描述這樣下位機部分就完成了。

  • 軟件部分

即上位機部分
c#
界面就是這樣。
上位機也不需要什麼邏輯,就是對數據的處理。
然後,然後連接數據庫。將數據寫入,並顯示出來。
鏈接數據庫
服務器名要和你的數據庫裏服務器名稱一致,然後下邊選擇對應的數據庫名稱,點擊測試連接。

數據庫
要在窗體中添加控件DataGridView然後選擇對應的庫。

//添加數據庫  注意服務器,數據庫等名稱要一致
string Connectstr = "Data Source=.;Initial Catalog=vs2017;Integrated Security=True";
mysql = new SqlConnection(Connectstr);
//查詢需要的表
SqlCommand Mycomm = new SqlCommand("Select * From lib", mysql);
mysql.Open();
label3.Text = Mycomm.ExecuteScalar().ToString();
mysql.Close();
//更新數據庫
this.libTableAdapter.Fill(this.vs2017DataSet.lib);

然後點擊查詢按鈕就可以了

查詢數據庫
因爲就找到一張卡。就只能演示一張卡了。
這樣就成功了!
這樣這個項目就完成了!
最後附上完整的項目文件。自行下載。
鏈接:https://pan.baidu.com/s/15l0hVUI6aahUY5h-ZxsmlQ
提取碼:jrm4

還有很多不足的地方,希望大佬可以指正。也希望和愛好者交流學習。
如有侵犯版權問題,立即刪除。歡迎轉載。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章