基于STM32(F103ZE)RFID簽到系統(tǒng)
這段時(shí)間幫學(xué)長(zhǎng)做課設(shè)摹闽,以前有幾個(gè)項(xiàng)目找到的股耽,想分享給大家根盒,去幫助課設(shè)還沒有做的。
寫的草物蝙,多多包涵炎滞。
下位機(jī)模塊用有原子哥的代碼。
目的是:用RFID-RC522去識(shí)別RFID卡(即一般飯卡)然后在LCD屏上顯示诬乞,并顯示現(xiàn)有多少人册赛,簽到簽退情況。用上位機(jī)去記錄簽到簽退時(shí)間等震嫉。有能力的話還能顯示誰沒到勤等森瘪。此外下位機(jī)還可以設(shè)置簽到時(shí)間,是否遲到等责掏。
材料
- STM32開發(fā)板(我用的是正點(diǎn)原子的戰(zhàn)艦V3 STM32F103ZET6大家也可以用自己的開發(fā)板做)
- RFID讀卡器 跟這個(gè)一樣(https://item.taobao.com/item.htm?spm=2013.1.0.0.586664e6jzoVIy&id=40316159698&source=wd&appId=8896)
- RFID卡
- keil5 寫下位機(jī)程序的
- VS2017 寫上位機(jī)程序的
- Microsoft SQL Server Tools 17 數(shù)據(jù)庫
- 串口調(diào)試助手 我用的是XCOME
然后就開始做了柜砾。
制作過程
- 硬件部分
首先要明白R(shí)FID讀卡器是怎么識(shí)別每張卡的。這里指的是數(shù)據(jù)换衬,我們不考慮怎么區(qū)分同時(shí)識(shí)別兩張卡等問題痰驱,我們只關(guān)心每張卡的數(shù)據(jù)形式,以及我們?cè)趺磪^(qū)分這些卡瞳浦;
經(jīng)過測(cè)試担映,這些卡的數(shù)據(jù)形式是16進(jìn)制。列出來一張:
04 0C 02 21 00 04 00 64 4C AF 5B 0C
注意是16進(jìn)制的我們不管他的工作原理叫潦。
不考慮兩張卡即以上同時(shí)識(shí)別蝇完。一次就是別一張。
然后將模塊的引腳連接上矗蕊。只連接VCC短蜕、GND、TX傻咖、RX四個(gè)引腳就好了朋魔,連接到板子的串口2上。因?yàn)榇?要連接上位機(jī)卿操。
我們先將模塊連接到USB轉(zhuǎn)TTL上連接到電腦上
這樣-->
然后打開串口調(diào)試助手連接警检。
還要提醒下VCC和GND千萬不要接反了K镌!
這樣-->
這里我們就能看到數(shù)據(jù)格式了扇雕,這個(gè)很重要拓售,我們以后要用到。
做完這些后就可以寫程序了镶奉。這里直說思想和難點(diǎn)础淤。
思想就是:兩個(gè)串口,串口2用來接收模塊發(fā)來的碼腮鞍,串口1用來給上位機(jī)發(fā)送信息值骇。定義一個(gè)二維數(shù)組莹菱,模塊每識(shí)別一次卡移国,向單片機(jī)發(fā)送一次數(shù)據(jù),把這個(gè)數(shù)據(jù)儲(chǔ)存到一個(gè)數(shù)組中道伟,并人數(shù)加一迹缀。然后第二次識(shí)別卡,當(dāng)識(shí)別的和第一次不同時(shí)就儲(chǔ)存到數(shù)組的另一個(gè)位置蜜徽。當(dāng)下一次識(shí)別的碼與前的某一次一樣時(shí)祝懂,就刪除這個(gè)數(shù)組。并人數(shù)減一拘鞋。然后每一次設(shè)別都將設(shè)別的卡號(hào)和簽到還是簽退情況發(fā)送給上位機(jī)砚蓬。
首先要寫兩個(gè)串口,分別都能接受數(shù)據(jù)盆色。
int fputc(int ch, FILE *f)
{
while((USART1->SR&0X40)==0);//循環(huán)發(fā)送直至發(fā)送完畢
USART1->DR = (u8) ch; //串口1要發(fā)送給上位機(jī)灰蛙,所以print函數(shù)要用串口1
return ch;
}
初始化兩個(gè)串口,串口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);//3?ê??ˉ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中斷函數(shù)
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類似隔躲,就不一一寫了摩梧。
接下來就是接收模塊的數(shù)據(jù)處理。
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);
}
這段代碼的意思就是接收出去然后去對(duì)比宣旱,一樣就把對(duì)比到的數(shù)組刪了仅父,不同就添加到一個(gè)數(shù)組。同時(shí)人數(shù)做加減浑吟。蜂鳴器響一聲笙纤。然后將接收到的數(shù)據(jù)通過串口1發(fā)送給上位機(jī)。
這樣就解決了數(shù)據(jù)處理問題组力,接下來就是顯示了省容。
顯示調(diào)用字庫。參考LD3320語音模塊忿项,有說怎么顯示漢字的蓉冈,這里就不說了城舞。顯示漢字人數(shù),簽到簽退等情況寞酿。
這樣下位機(jī)部分就完成了家夺。
- 軟件部分
即上位機(jī)部分
界面就是這樣。
上位機(jī)也不需要什么邏輯伐弹,就是對(duì)數(shù)據(jù)的處理拉馋。
然后,然后連接數(shù)據(jù)庫惨好。將數(shù)據(jù)寫入列赎,并顯示出來。
鏈接數(shù)據(jù)庫
服務(wù)器名要和你的數(shù)據(jù)庫里服務(wù)器名稱一致现拒,然后下邊選擇對(duì)應(yīng)的數(shù)據(jù)庫名稱叮阅,點(diǎn)擊測(cè)試連接。
要在窗體中添加控件DataGridView然后選擇對(duì)應(yīng)的庫龄句。
//添加數(shù)據(jù)庫 注意服務(wù)器回论,數(shù)據(jù)庫等名稱要一致
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();
//更新數(shù)據(jù)庫
this.libTableAdapter.Fill(this.vs2017DataSet.lib);
然后點(diǎn)擊查詢按鈕就可以了
因?yàn)榫驼业揭粡埧ā>椭荒苎菔疽粡埧恕?br> 這樣就成功了分歇!
這樣這個(gè)項(xiàng)目就完成了傀蓉!
最后附上完整的項(xiàng)目文件。自行下載职抡。
https://download.csdn.net/download/weixin_42320020/16594105
進(jìn)制轉(zhuǎn)載T崃恰!缚甩!
QQ:2039723308
VX:Shiboven