RFID簽到系統
這段時間幫學長做課設,以前有幾個項目找到的,想分享給大家,去幫助課設還沒有做的。
寫的草,多多包涵。
下位機模塊用有原子哥的代碼。
目的是:用RFID-RC522去識別RFID卡(即一般飯卡)然後在LCD屏上顯示,並顯示現有多少人,簽到簽退情況。用上位機去記錄簽到簽退時間等。有能力的話還能顯示誰沒到勤等。此外下位機還可以設置簽到時間,是否遲到等。
材料
- STM32開發板(我用的是正點原子的戰艦V3 STM32F103ZET6大家也可以用自己的開發板做)
- RFID讀卡器 跟這個一樣(https://item.taobao.com/item.htm?spm=2013.1.0.0.586664e6jzoVIy&id=40316159698&source=wd&appId=8896)
- RFID卡
- keil5 寫下位機程序的
- VS2017 寫上位機程序的
- Microsoft SQL Server Tools 17 數據庫
- 串口調試助手 我用的是XCOME
然後就開始做了。
製作過程
- 硬件部分
-
首先要明白RFID讀卡器是怎麼識別每張卡的。這裏指的是數據,我們不考慮怎麼區分同時識別兩張卡等問題,我們只關心每張卡的數據形式,以及我們怎麼區分這些卡;
經過測試,這些卡的數據形式是16進制。列出來一張:
04 0C 02 21 00 04 00 64 4C AF 5B 0C
注意是16進制的 -
我們不管他的工作原理。
-
不考慮兩張卡即以上同時識別。一次就是別一張。
-
然後將模塊的引腳連接上。只連接VCC、GND、TX、RX四個引腳就好了,連接到板子的串口2上。因爲串口1要連接上位機。
我們先將模塊連接到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語音模塊,有說怎麼顯示漢字的,這裏就不說了。顯示漢字人數,簽到簽退等情況。
這樣下位機部分就完成了。
- 軟件部分
即上位機部分
界面就是這樣。
上位機也不需要什麼邏輯,就是對數據的處理。
然後,然後連接數據庫。將數據寫入,並顯示出來。
鏈接數據庫
服務器名要和你的數據庫裏服務器名稱一致,然後下邊選擇對應的數據庫名稱,點擊測試連接。
要在窗體中添加控件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
還有很多不足的地方,希望大佬可以指正。也希望和愛好者交流學習。
如有侵犯版權問題,立即刪除。歡迎轉載。