本文转载,其来源在参考中:1,稍加修改,因为近期使用到这个模块,故而加以整理!


1.平台

  首先我使用的是 奋斗 STM32 开发板 MINI板
 基于STM32单片机光学指纹识别模块(FPM10A)全教程


2.购买指纹模块,可以获得三份资料

 1.简要使用说明
 2.使用指纹模块的功能函数
 3.FPM10A用户手册.


3.硬件搭建


 根据使用说明:FPM 10A使用标准的串口与外界通信,默认的波特率为57600,可以与任何单片机,ARM,DSP等带串口的设备进行连接,请注意电平转换,连接电脑需要进行电平转换,比如MAX232电路。
FPM10A光学指纹模块共有5个管脚
 1 为 VCC 电源的正极接 3.6V – 5.5V的电压均可。
 2 为 GND 电源的负极 接地。
 3 为 TXD 串口的发送。
 4 为 RXD 串口的接收。
 5 为 NC 悬空不需要使用。


奋斗板上已经有5V的管脚,可以直接供给指纹模块,
这里需要注意的是,指纹模块主要通过串口进行控制,模块和STM32单片机连接的时候,需要进行电平转换,
基于STM32单片机光学指纹识别模块(FPM10A)全教程

这样只要把这个转接板插入STM32,接上5V的电,就可以工作了,将模块的发送端接转接板的接收端,接收端接转接板的发送端。
这样,我们的硬件平台就搭建好了!

4.模块的测试工作

>模块成功上电后,指纹采集窗口会闪一下,表示自检正常,如果不闪,请仔细检查电源,是否接反,接错等。指纹模块使用120MHZ的DSP全速工作,工作时芯片有一些热,经过严格的测试,这是没有问题的可以放心使用,在不使用的时候可以关闭电源,以降低功耗。

5.现在我们要进入编程环节了

>指纹模块主要是通过串口进行控制,所以这里我们需要用到单片机的串口模块。
我们需要用到两个关键函数
1.使用串口发送一个字节的数据
2.使用串口接收一个字节的数据

这里我使用的STM32单片,所以这两个程序如下:

// 从 USART1 发送一个字节
void USART1_SendByte(unsigned char temp)
{USART_SendData(USART1, temp);while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);
}
// 从 USART1 读取一个字节
unsigned char USART1_ReceivByte()
{unsigned char recev;while(USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET);recev = USART_ReceiveData(USART1);return recev;
}

6.查看FPM10A用户手册 我们来实现比对一个指纹(我们这里假设指纹模块中已经存在指纹模板)

首先我们需要让指纹模块检测是否有指纹输入(也就是是否有手指放在指纹模块上检测)
我们来看手册上给的操作说明:

基于STM32单片机光学指纹识别模块(FPM10A)全教程
我们需要发送给定的数据包给模块,发送的数据已经给我们了,现在我们参看给我们的C例程

  //应答包数组unsigned char dat[18];//获得指纹图像unsigned char FP_Get_Img[6] ={0x01,0x00,0x03,0x01,0x0,0x05};//协议包头unsigned char FP_Pack_Head[6] = {0xEF,0x01,0xFF,0xFF,0xFF,0xFF};      //FINGERPRINT_获得指纹图像命令void FINGERPRINT_Cmd_Get_Img(void){unsigned char i;for(i=0;i<6;i++) //发送包头USART1_SendByte(FP_Pack_Head[i]);for(i=0;i<6;i++) //发送命令 0x1dUSART1_SendByte(FP_Get_Img[i]);for(i=0;i<12;i++)//读回应答信息dat[i]=USART1_ReceivByte();}
/*说明:这个函数就是检测是否有指纹输入的信息,根据用户手册,当确认码返回值为0时,表示成功录入,所以,我们可以有下面的函数:*/
//检测指纹模块录入指纹情况,返回00表示录入成功;02无手指;03录入失败
unsigned char test_fig()
{unsigned char fig_dat;FINGERPRINT_Cmd_Get_Img();Delay_ms1(20);fig_dat=dat[9];return(fig_dat);
}

因此,我们在主函数中可以这样调用:

void main
{if(test_fig()==0){//do something}
}

7.如何录入一个新的指纹信息呢?

步骤如下
1.获得指纹图像
2.检测是否成功的按了指纹
3.将图像转换成特征码存放在Buffer1中
4.再次获得指纹图像
5.将图像转换成特征码存放在Buffer2中
6.转换成特征码
7.存储到指定地址上

同样的,根据用户手册,我们可以得到以下这样的模块:

当调用的时候,你只要给这个函数附上两个值就可以了,例如:
unsigned char FP_add_new_user(00,01);
如果你下次再次写入这个地址,以前存储的指纹模板信息将被覆盖

 //添加一个新的指纹
unsigned char FP_add_new_user(unsigned char ucH_user,unsigned char ucL_user)
{ do{                   FINGERPRINT_Cmd_Get_Img();       //获得指纹图像} while ( dat[9]!=0x0 );            //检测是否成功的按了指纹 FINGERPRINT_Cmd_Img_To_Buffer1();   //将图像转换成特征码存放在Buffer1中do{FINGERPRINT_Cmd_Get_Img();      //获得指纹图像  } while( dat[9]!=0x0 );FINGERPRINT_Cmd_Img_To_Buffer2();   //将图像转换成特征码存放在Buffer2中FINGERPRINT_Cmd_Reg_Model();        //转换成特征码FINGERPRINT_Cmd_Save_Finger(ucH_user,ucL_user);                          return 0;
}
//存储模版到特定地址
void FINGERPRINT_Cmd_Save_Finger( unsigned char ucH_Char, unsigned char ucL_Char)
{unsigned long temp = 0;unsigned char i;FP_Save_Finger[5] = ucH_Char;FP_Save_Finger[6] = ucL_Char;for(i=0;i<7;i++)   //计算校验和temp = temp + FP_Save_Finger[i];FP_Save_Finger[7]=(temp & 0x00FF00) >> 8; //存放校验数据FP_Save_Finger[8]= temp & 0x0000FF;for(i=0;i<6;i++)   USART1_SendByte(FP_Pack_Head[i]);        //发送包头for(i=0;i<9;i++)  USART1_SendByte(FP_Save_Finger[i]) ;//发送命令将图像转换成特征码存放CHAR_buffer1for(i=0;i<12;i++)dat[i]=USART1_ReceivByte();
}

8.如何删除一个模板?

//删除所有指纹模版
void FINGERPRINT_Cmd_Delete_All_Model(void)
{unsigned char i;for(i=0;i<6;i++) //发送包头USART1_SendByte(FP_Pack_Head[i]);for(i=0;i<6;i++) //发送命令 0x1dUSART1_SendByte(FP_Delet_All_Model[i]);for(i=0;i<12;i++)//读回应答信息dat[i]=USART1_ReceivByte();
}

9.如何获取已经存取的指纹模板信息?

这个模块一共可以存储0~999枚指纹信息

//搜索全部用户999枚
void FINGERPRINT_Cmd_Search_Finger(void)
{unsigned char i;   //发送命令搜索指纹库for(i=0;i<6;i++)  {USART1_SendByte(FP_Pack_Head[i]);  }for(i=0;i<11;i++){USART1_SendByte(FP_Search[i]);  }for(i=0;i<16;i++){dat[i]=USART1_ReceivByte();}
}

根据用户手册,我们可以从应答包中得出模块中已经存在指纹数量的大小

这样,我们就轻松把指纹模块搞定!

下面我附上基于STM32单片机光学指纹识别模块(FPM10A)打包好的函数库
第一个是 FPM10A.c

\#include "stm32f10x.h"
\#include "stm32f10x_usart.h"
\#include "misc.h"
unsigned char dat[18];
//FINGERPRINT通信协议定义
unsigned char FP_Pack_Head[6] = {0xEF,0x01,0xFF,0xFF,0xFF,0xFF};                    //协议包头
unsigned char FP_Get_Img[6] = {0x01,0x00,0x03,0x01,0x0,0x05};                       //获得指纹图像
unsigned char FP_Templete_Num[6] ={0x01,0x00,0x03,0x1D,0x00,0x21 };                 //获得模版总数
unsigned char FP_Search[11]={0x01,0x0,0x08,0x04,0x01,0x0,0x0,0x03,0xA1,0x0,0xB2};   //搜索指纹搜索范围0 - 929
unsigned char FP_Search_0_9[11]={0x01,0x0,0x08,0x04,0x01,0x0,0x0,0x0,0x13,0x0,0x21};//搜索0-9号指纹
unsigned char FP_Img_To_Buffer1[7]={0x01,0x0,0x04,0x02,0x01,0x0,0x08};     //将图像放入到BUFFER1
unsigned char FP_Img_To_Buffer2[7]={0x01,0x0,0x04,0x02,0x02,0x0,0x09};     //将图像放入到BUFFER2
unsigned char FP_Reg_Model[6]={0x01,0x0,0x03,0x05,0x0,0x09};       //将BUFFER1跟BUFFER2合成特征模版
unsigned char FP_Delet_All_Model[6]={0x01,0x0,0x03,0x0d,0x00,0x11};     //删除指纹模块里所有的模版
unsigned char FP_Save_Finger[9]={0x01,0x00,0x06,0x06,0x01,0x00,0x0B,0x00,0x19};  //将BUFFER1中的特征码存放到指定的位置
unsigned char FP_Delete_Model[10]={0x01,0x00,0x07,0x0C,0x0,0x0,0x0,0x1,0x0,0x0};  //删除指定的模版//从 USART1 发送一个字节
void USART1_SendByte(unsigned char temp)
{USART_SendData(USART1, temp);while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);
}
//从 USART1 读取一个字节
unsigned char USART1_ReceivByte()
{unsigned char recev;while(USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET);recev = USART_ReceiveData(USART1);return recev;
}
//FINGERPRINT命令字
//FINGERPRINT_获得指纹图像命令
void FINGERPRINT_Cmd_Get_Img(void)
{unsigned char i;for(i=0;i<6;i++) //发送包头USART1_SendByte(FP_Pack_Head[i]);for(i=0;i<6;i++) //发送命令 0x1dUSART1_SendByte(FP_Get_Img[i]);for(i=0;i<12;i++)//读回应答信息dat[i]=USART1_ReceivByte();
}
//删除所有指纹模版
void FINGERPRINT_Cmd_Delete_All_Model(void)
{unsigned char i;for(i=0;i<6;i++) //发送包头USART1_SendByte(FP_Pack_Head[i]);for(i=0;i<6;i++) //发送命令 0x1dUSART1_SendByte(FP_Delet_All_Model[i]);for(i=0;i<12;i++)//读回应答信息dat[i]=USART1_ReceivByte();
}
//讲图像转换成特征码存放在Buffer1中
void FINGERPRINT_Cmd_Img_To_Buffer1(void)
{unsigned char i;for(i=0;i<6;i++)    //发送包头{USART1_SendByte(FP_Pack_Head[i]);  }for(i=0;i<7;i++)   //发送命令 将图像转换成 特征码 存放在 CHAR_buffer1{USART1_SendByte(FP_Img_To_Buffer1[i]);}for(i=0;i<12;i++)//读应答信息{dat[i]=USART1_ReceivByte();//把应答数据存放到缓冲区}
}
//将图像转换成特征码存放在Buffer2中
void FINGERPRINT_Cmd_Img_To_Buffer2(void)
{unsigned char i;for(i=0;i<6;i++)    //发送包头{USART1_SendByte(FP_Pack_Head[i]);  }for(i=0;i<7;i++)   //发送命令 将图像转换成 特征码 存放在 CHAR_buffer1{USART1_SendByte(FP_Img_To_Buffer2[i]);}for(i=0;i<12;i++){dat[i]=USART1_ReceivByte();//读回应答信息}
}
//将BUFFER1 跟 BUFFER2 中的特征码合并成指纹模版
void FINGERPRINT_Cmd_Reg_Model(void)
{unsigned char i;   for(i=0;i<6;i++) //包头{USART1_SendByte(FP_Pack_Head[i]);  }for(i=0;i<6;i++) //命令合并指纹模版{USART1_SendByte(FP_Reg_Model[i]);  }for(i=0;i<12;i++){dat[i]=USART1_ReceivByte();}
}
//存储模版到特定地址
void FINGERPRINT_Cmd_Save_Finger( unsigned char ucH_Char, unsigned char ucL_Char)
{unsigned long temp = 0;unsigned char i;FP_Save_Finger[5] = ucH_Char;FP_Save_Finger[6] = ucL_Char;for(i=0;i<7;i++)   //计算校验和temp = temp + FP_Save_Finger[i];FP_Save_Finger[7]=(temp & 0x00FF00) >> 8; //存放校验数据FP_Save_Finger[8]= temp & 0x0000FF;for(i=0;i<6;i++)   USART1_SendByte(FP_Pack_Head[i]);        //发送包头for(i=0;i<9;i++) USART1_SendByte(FP_Save_Finger[i]);      //发送命令 将图像转换成 特征码 存放在 CHAR_buffer1for(i=0;i<12;i++)dat[i]=USART1_ReceivByte();
}
//获得指纹模板数量
void FINGERPRINT_Cmd_Get_Templete_Num(void)
{ unsigned int i;for(i=0;i<6;i++) //包头USART1_SendByte(FP_Pack_Head[i]);//发送命令 0x1dfor(i=0;i<6;i++)USART1_SendByte(FP_Templete_Num[i]);for(i=0;i<12;i++)dat[i]=USART1_ReceivByte();
}
//搜索全部用户999枚
void FINGERPRINT_Cmd_Search_Finger(void)
{unsigned char i;   //发送命令搜索指纹库for(i=0;i<6;i++)  {USART1_SendByte(FP_Pack_Head[i]);  }for(i=0;i<11;i++){USART1_SendByte(FP_Search[i]);  }for(i=0;i<16;i++){dat[i]=USART1_ReceivByte();}
}
//搜索用户0~9枚
void FINGERPRINT_Cmd_Search_Finger_Admin(void)
{unsigned char i;   for(i=0;i<6;i++)   //发送命令搜索指纹库{USART1_SendByte(FP_Pack_Head[i]);  }for(i=0;i<11;i++){USART1_SendByte(FP_Search_0_9[i]);  }for(i=0;i<12;i++)dat[i]=USART1_ReceivByte();
}
//添加一个新的指纹
unsigned char FP_add_new_user(unsigned char ucH_user,unsigned char ucL_user)
{ do{                   FINGERPRINT_Cmd_Get_Img();       //获得指纹图像} while ( dat[9]!=0x0 );            //检测是否成功的按了指纹 FINGERPRINT_Cmd_Img_To_Buffer1();   //将图像转换成特征码存放在Buffer1中do{FINGERPRINT_Cmd_Get_Img();      //获得指纹图像  } while( dat[9]!=0x0 );FINGERPRINT_Cmd_Img_To_Buffer2();   //将图像转换成特征码存放在Buffer2中FINGERPRINT_Cmd_Reg_Model();        //转换成特征码FINGERPRINT_Cmd_Save_Finger(ucH_user,ucL_user);                          return 0;
}

第2个 FPM10A.h

\#ifndef _FPM10A_H
\#define _FPM10A_H
\#include <stdint.h>
extern unsigned char dat[18];
extern void FINGERPRINT_Cmd_Get_Img();
extern void FINGERPRINT_Cmd_Img_To_Buffer1();
extern void FINGERPRINT_Cmd_Img_To_Buffer2();
extern void FINGERPRINT_Cmd_Reg_Model();
extern void FINGERPRINT_Cmd_Delete_All_Model(void);
extern void FINGERPRINT_Cmd_Search_Finger(void);
extern void FINGERPRINT_Cmd_Get_Templete_Num(void);
extern void FINGERPRINT_Cmd_Search_Finger_Admin(void);
extern void FINGERPRINT_Cmd_Save_Finger( unsigned char ucH_Char, unsigned char ucL_Char);
extern unsigned char FP_add_new_user(unsigned char ucH_user,unsigned char ucL_user);extern void USART1_SendByte(unsigned char temp);
extern unsigned char USART1_ReceivByte();
extern void Delay_ms1(uint32_t nCount);
void Delay_nus1(uint32_t nCount)
{uint32_t j;while(nCount--){j=8;while(j--);}
}
void Delay_ms1(uint32_t nCount)
{while(nCount--)Delay_nus1(1100);
}
unsigned char test_fig()//检测指纹模块录入指纹情况,返回00表示录入成功;02无手指;03录入失败
{unsigned char fig_dat;FINGERPRINT_Cmd_Get_Img();Delay_ms1(20);fig_dat=dat[9];return(fig_dat);
}
\#endif

有了这两个东西,加入到你的工程中,就可以直接调用啦!


  1. http://blog.sina.com.cn/s/blog_804665db01015nnd.html ↩

转载于:https://www.cnblogs.com/ysmintor/p/5180653.html

基于STM32单片机光学指纹识别模块(FPM10A)全教程(基于C语言)相关推荐

  1. K_A12_007 基于STM32等单片机驱动AS608光学指纹识别模块 OLED0.96显示

    K_A12_007 基于STM32等单片机驱动AS608光学指纹识别模块 OLED0.96显示 一.资源说明 二.基本参数 参数 引脚说明 三.驱动说明 对应程序: 四.部分代码说明 1.接线引脚定义 ...

  2. 基于51单片机的指纹识别锁管理门禁密码锁系统方案原理图程序设计

    硬件电路的设计 (附文件) 3.1系统的功能分析及体系结构设计 3.1.1系统功能分析 本系统由STC89C52单片机电路+指纹模块传感器电路+LCD1602液晶显示电路+继电器电路+按键电路+电源电 ...

  3. 51驱动AS608光学指纹识别模块 12864显示

    51驱动AS608光学指纹识别模块 12864显示 AS608光学指纹识别模块 模块工作原理 1.指纹特征 2.指纹处理 模块参数 引脚说明 实验程序 硬件设备和接线 程序讲解 按键 主函数 实验步骤 ...

  4. 【雕爷学编程】Arduino动手做(141)---AS608光学指纹识别模块

    37款传感器与执行器的提法,在网络上广泛流传,其实Arduino能够兼容的传感器模块肯定是不止这37种的.鉴于本人手头积累了一些传感器和执行器模块,依照实践出真知(一定要动手做)的理念,以学习和交流为 ...

  5. delphi 数字识别_基于STM32单片机的车牌识别

    系统介绍    使用STM32F103RCT6作为主控,摄像头使用OV7670(带FIFO).STM32进行了16倍频.识别过程分别为:图像采集,二值化,识别车牌区域,字符分割,字符匹配. 识别过程分 ...

  6. stm32车牌识别_基于STM32单片机的车牌识别

    系统介绍    使用STM32F103RCT6作为主控,摄像头使用OV7670(带FIFO).STM32进行了16倍频.识别过程分别为:图像采集,二值化,识别车牌区域,字符分割,字符匹配. 识别过程分 ...

  7. AS608光学指纹识别模块 智能锁/考勤门禁开发/指纹采集模块

    一.注册指纹 /*[Arduino]168种传感器模块系列实验(资料+代码+图形+仿真)实验一百五十:AS608光学指纹识别模块 智能锁/考勤门禁开发/指纹采集模块实验之二:输入序号,注册指纹示例安装 ...

  8. 基于stm32单片机的指纹解锁门禁系统

    基于stm32单片机的指纹解锁门禁系统 1.包含三个模式,正常模式,录入模式,删除模式 正常模式下,需要管理员指纹开锁,其他人才能正常打卡,实验室人数清零后,重新需要管理员指纹开锁 录入与删除指纹模式 ...

  9. 指纹传感器的测试软件,【Arduino】168种传感器系列实验(149)-AS608光学指纹识别模块-Arduino中文社区 - Powered by Discuz!...

    [mw_shl_code=arduino,true]/* [Arduino]168种传感器模块系列实验(资料+代码+图形+仿真) 实验一百五十:AS608光学指纹识别模块+0.91寸OLED液晶屏显示 ...

最新文章

  1. Silverlight+WCF 新手实例 象棋 主界面-实时聊天区(二十五)
  2. Spring Cloud构建微服务架构(六)高可用服务注册中心
  3. 终于感觉掌握了一门重要的技术
  4. 一步步分析-C语言如何面向对象编程
  5. pagehelper插件oracle,带你学习最简单的分页插件PageHelper
  6. Python——迭代器的几个高级用法
  7. C++手写快读详解(快速读入数字)
  8. 在mac上安装cgal4.11时,QT5 和qglviewer的设置
  9. 数据结构笔记(十四)-- 串的模式匹配算法
  10. canvas-画图改进版
  11. Xib中用自动布局设置UIScrollView的ContenSize
  12. handlersocket mysql_Mysql插件之HandlerSocket的安装、配置、使用
  13. 【算法】并查集(Java)
  14. 【PPT在插入excel对象时报错】
  15. Windows Terminal 设置背景图片
  16. python海龟怎样写字又快又好看_python海龟画图
  17. QQ自带截图功能(你想不到的强大, 必看,长截图/翻译/文本识别/钉在桌面/录屏)
  18. PAT甲之初窥门径(上)
  19. 谭浩强C++ 第二章
  20. OpenNURBS 3DM Viewer

热门文章

  1. 计算机软件技术实习 项目一 简单计算器的实现(实验准备)1-(1)
  2. 【算法】如何给100亿个数字排序?
  3. python保存为xlsb_Python XLSB到CSV的转换数据类型
  4. 大数据分析工具Power BI(十):制作可视化图表的报表类型
  5. arcgis快速生成图框_ArcGIS中定义图框样式
  6. 电脑突然关机,如何恢复word文件内容
  7. Python爬虫入门教程 63-100 Python字体反爬之一,没办法,这个必须写,反爬第3篇
  8. 假如,你拥有这样的商务邮箱,你的工作方式将会改变!
  9. 511遇见易语言数组插入成员
  10. AWTK 支持 Rive 动画