目录
首先是key.h文件
oled.h文件
oled.c
key.c文件
main.c函数
直接上代码文章末尾讲解 注文章篇幅较长可能需要点阅读时间
#define KEY_HANG1 GPIO_Pin_0
#define KEY_HANG2 GPIO_Pin_1
#define KEY_HANG3 GPIO_Pin_2
#define KEY_HANG4 GPIO_Pin_3
#define KEY1 GPIO_Pin_4
#define KEY2 GPIO_Pin_5
#define KEY3 GPIO_Pin_6
#define KEY4 GPIO_Pin_7
#define KEY1_Input GPIO_ReadInputDataBit(KEY_PROT,KEY1)
#define KEY2_Input GPIO_ReadInputDataBit(KEY_PROT,KEY2)
#define KEY3_Input GPIO_ReadInputDataBit(KEY_PROT,KEY3)
#define KEY4_Input GPIO_ReadInputDataBit(KEY_PROT,KEY4)
#define KEY_PROT GPIOA
extern u8 flag;
u16 *abc_read();
int sun1_read();
void KEY_Init(void);//IO初始化
void KEY_config(u8 *key);
u8 Read_KeyValue(void);
void Key_Input( u8 *key1);
#ifndef __OLED_H
#define __OLED_H
#include "sys.h"
#include "stdlib.h"
//-----------------OLED端口定义----------------
#define OLED_SCL_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_1)//SCL A 0
#define OLED_SCL_Set() GPIO_SetBits(GPIOB,GPIO_Pin_1)
#define OLED_SDA_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_10)//DIN A 1
#define OLED_SDA_Set() GPIO_SetBits(GPIOB,GPIO_Pin_10)
#define OLED_RES_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_7)//RES A 2
#define OLED_RES_Set() GPIO_SetBits(GPIOB,GPIO_Pin_7)
#define OLED_CMD 0 //写命令
#define OLED_DATA 1 //写数据
extern u8 flag;
void KEY_Init(void);//IO初始化
void KEY_config(u8 *key);
u8 Read_KeyValue(void);
void Key_Input( u8 *key1);
void OLED_ClearPoint(u8 x,u8 y);
void OLED_ColorTurn(u8 i);
void OLED_DisplayTurn(u8 i);
void I2C_Start(void);
void I2C_Stop(void);
void I2C_WaitAck(void);
void Send_Byte(u8 dat);
void OLED_WR_Byte(u8 dat,u8 mode);
void OLED_DisPlay_On(void);
void OLED_DisPlay_Off(void);
void OLED_Refresh(void);
void OLED_Clear(void);
void OLED_DrawPoint(u8 x,u8 y,u8 t);
void OLED_DrawLine(u8 x1,u8 y1,u8 x2,u8 y2,u8 mode);
void OLED_DrawCircle(u8 x,u8 y,u8 r);
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size1,u8 mode);
void OLED_ShowChar6x8(u8 x,u8 y,u8 chr,u8 mode);
void OLED_ShowString(u8 x,u8 y,u8 *chr,u8 size1,u8 mode);
void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size1,u8 mode);
void OLED_ShowChinese(u8 x,u8 y,u8 num,u8 size1,u8 mode);
void OLED_ScrollDisplay(u8 num,u8 space,u8 mode);
void OLED_ShowPicture(u8 x,u8 y,u8 sizex,u8 sizey,u8 BMP[],u8 mode);
void OLED_Init(void);
#endif
#include "oled.h"
#include "stdlib.h"
#include "oledfont.h"
#include "delay.h"
#include "stm32f10x.h"
#include "usart.h"
u8 OLED_GRAM[144][8];
//反显函数
void OLED_ColorTurn(u8 i)
{
if(i==0)
{
OLED_WR_Byte(0xA6,OLED_CMD);//正常显示
}
if(i==1)
{
OLED_WR_Byte(0xA7,OLED_CMD);//反色显示
}
}
//屏幕旋转180度
void OLED_DisplayTurn(u8 i)
{
if(i==0)
{
OLED_WR_Byte(0xC8,OLED_CMD);//正常显示
OLED_WR_Byte(0xA1,OLED_CMD);
}
if(i==1)
{
OLED_WR_Byte(0xC0,OLED_CMD);//反转显示
OLED_WR_Byte(0xA0,OLED_CMD);
}
}
//延时
void IIC_delay(void)
{
u8 t=3;
while(t--);
}
//起始信号
void I2C_Start(void)
{
OLED_SDA_Set();
OLED_SCL_Set();
IIC_delay();
OLED_SDA_Clr();
IIC_delay();
OLED_SCL_Clr();
IIC_delay();
}
//结束信号
void I2C_Stop(void)
{
OLED_SDA_Clr();
OLED_SCL_Set();
IIC_delay();
OLED_SDA_Set();
}
//等待信号响应
void I2C_WaitAck(void) //测数据信号的电平
{
OLED_SDA_Set();
IIC_delay();
OLED_SCL_Set();
IIC_delay();
OLED_SCL_Clr();
IIC_delay();
}
//写入一个字节
void Send_Byte(u8 dat)
{
u8 i;
for(i=0;i<8;i++)
{
if(dat&0x80)//将dat的8位从最高位依次写入
{
OLED_SDA_Set();
}
else
{
OLED_SDA_Clr();
}
IIC_delay();
OLED_SCL_Set();
IIC_delay();
OLED_SCL_Clr();//将时钟信号设置为低电平
dat<<=1;
}
}
//发送一个字节
//mode:数据/命令标志 0,表示命令;1,表示数据;
void OLED_WR_Byte(u8 dat,u8 mode)
{
I2C_Start();
Send_Byte(0x78);
I2C_WaitAck();
if(mode){Send_Byte(0x40);}
else{Send_Byte(0x00);}
I2C_WaitAck();
Send_Byte(dat);
I2C_WaitAck();
I2C_Stop();
}
//开启OLED显示
void OLED_DisPlay_On(void)
{
OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
OLED_WR_Byte(0x14,OLED_CMD);//开启电荷泵
OLED_WR_Byte(0xAF,OLED_CMD);//点亮屏幕
}
//关闭OLED显示
void OLED_DisPlay_Off(void)
{
OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
OLED_WR_Byte(0x10,OLED_CMD);//关闭电荷泵
OLED_WR_Byte(0xAE,OLED_CMD);//关闭屏幕
}
//更新显存到OLED
void OLED_Refresh(void)
{
u8 i,n;
for(i=0;i<8;i++)
{
OLED_WR_Byte(0xb0+i,OLED_CMD); //设置行起始地址
OLED_WR_Byte(0x00,OLED_CMD); //设置低列起始地址
OLED_WR_Byte(0x10,OLED_CMD); //设置高列起始地址
I2C_Start();
Send_Byte(0x78);
I2C_WaitAck();
Send_Byte(0x40);
I2C_WaitAck();
for(n=0;n<128;n++)
{
Send_Byte(OLED_GRAM[n][i]);
I2C_WaitAck();
}
I2C_Stop();
}
}
//清屏函数
void OLED_Clear(void)
{
u8 i,n;
for(i=0;i<8;i++)
{
for(n=0;n<128;n++)
{
OLED_GRAM[n][i]=0;//清除所有数据
}
}
//OLED_Refresh();//更新显示
}
//画点
//x:0~127
//y:0~63
//t:1 填充 0,清空
void OLED_DrawPoint(u8 x,u8 y,u8 t)
{
u8 i,m,n;
i=y/8;
m=y%8;
n=1<0)incx=1; //设置单步方向
else if (delta_x==0)incx=0;//垂直线
else {incx=-1;delta_x=-delta_x;}
if(delta_y>0)incy=1;
else if (delta_y==0)incy=0;//水平线
else {incy=-1;delta_y=-delta_x;}
if(delta_x>delta_y)distance=delta_x; //选取基本增量坐标轴
else distance=delta_y;
for(t=0;tdistance)
{
xerr-=distance;
uRow+=incx;
}
if(yerr>distance)
{
yerr-=distance;
uCol+=incy;
}
}
}
//x,y:圆心坐标
//r:圆的半径
void OLED_DrawCircle(u8 x,u8 y,u8 r)
{
int a, b,num;
a = 0;
b = r;
while(2 * b * b >= r * r)
{
OLED_DrawPoint(x + a, y - b,1);
OLED_DrawPoint(x - a, y - b,1);
OLED_DrawPoint(x - a, y + b,1);
OLED_DrawPoint(x + a, y + b,1);
OLED_DrawPoint(x + b, y + a,1);
OLED_DrawPoint(x + b, y - a,1);
OLED_DrawPoint(x - b, y - a,1);
OLED_DrawPoint(x - b, y + a,1);
a++;
num = (a * a + b * b) - r*r;//计算画的点离圆心的距离
if(num > 0)
{
b--;
a--;
}
}
}
//在指定位置显示一个字符,包括部分字符
//x:0~127
//y:0~63
//size1:选择字体 6x8/6x12/8x16/12x24
//mode:0,反色显示;1,正常显示
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size1,u8 mode)
{
u8 i,m,temp,size2,chr1;
u8 x0=x,y0=y;
if(size1==8)size2=6;
else size2=(size1/8+((size1%8)?1:0))*(size1/2); //得到字体一个字符对应点阵集所占的字节数
chr1=chr-' '; //计算偏移后的值
for(i=0;i>=1;
y++;
}
x++;
if((size1!=8)&&((x-x0)==size1/2))
{x=x0;y0=y0+8;}
y=y0;
}
}
//显示字符串
//x,y:起点坐标
//size1:字体大小
//*chr:字符串起始地址
//mode:0,反色显示;1,正常显示
void OLED_ShowString(u8 x,u8 y,u8 *chr,u8 size1,u8 mode)
{
while((*chr>=' ')&&(*chr<='~'))//判断是不是非法字符!
{
OLED_ShowChar(x,y,*chr,size1,mode);
if(size1==8)x+=6;
else x+=size1/2;
chr++;
}
}
//m^n
u32 OLED_Pow(u8 m,u8 n)
{
u32 result=1;
while(n--)
{
result*=m;
}
return result;
}
//显示数字
//x,y :起点坐标
//num :要显示的数字
//len :数字的位数
//size:字体大小
//mode:0,反色显示;1,正常显示
void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size1,u8 mode)
{
u8 t,temp,m=0;
if(size1==8)m=2;
for(t=0;t>=1;
y++;
}
x++;
if((x-x0)==size1)
{x=x0;y0=y0+8;}
y=y0;
}
}
//num 显示汉字的个数
//space 每一遍显示的间隔
//mode:0,反色显示;1,正常显示
void OLED_ScrollDisplay(u8 num,u8 space,u8 mode)
{
u8 i,n,t=0,m=0,r;
while(1)
{
if(m==0)
{
OLED_ShowChinese(128,24,t,16,mode); //写入一个汉字保存在OLED_GRAM[][]数组中
t++;
}
if(t==num)
{
for(r=0;r<16*space;r++) //显示间隔
{
for(i=1;i<144;i++)
{
for(n=0;n<8;n++)
{
OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
}
}
OLED_Refresh();
}
t=0;
}
m++;
if(m==16){m=0;}
for(i=1;i<144;i++) //实现左移
{
for(n=0;n<8;n++)
{
OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
}
}
OLED_Refresh();
}
}
//x,y:起点坐标
//sizex,sizey,图片长宽
//BMP[]:要写入的图片数组
//mode:0,反色显示;1,正常显示
void OLED_ShowPicture(u8 x,u8 y,u8 sizex,u8 sizey,u8 BMP[],u8 mode)
{
u16 j=0;
u8 i,n,temp,m;
u8 x0=x,y0=y;
sizey=sizey/8+((sizey%8)?1:0);
for(n=0;n>=1;
y++;
}
x++;
if((x-x0)==sizex)
{
x=x0;
y0=y0+8;
}
y=y0;
}
}
}
//OLED的初始化
void OLED_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能A端口时钟
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10|GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; //开漏输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHz
GPIO_Init(GPIOB, &GPIO_InitStructure); //初始化PA0,1
GPIO_SetBits(GPIOB,GPIO_Pin_10|GPIO_Pin_1);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHz
GPIO_Init(GPIOB, &GPIO_InitStructure); //初始化PA2
GPIO_SetBits(GPIOB,GPIO_Pin_7);
OLED_RES_Clr();
delay_ms(200);
OLED_RES_Set();
OLED_WR_Byte(0xAE,OLED_CMD);//--turn off oled panel
OLED_WR_Byte(0x00,OLED_CMD);//---set low column address
OLED_WR_Byte(0x10,OLED_CMD);//---set high column address
OLED_WR_Byte(0x40,OLED_CMD);//--set start line address Set Mapping RAM Display Start Line (0x00~0x3F)
OLED_WR_Byte(0x81,OLED_CMD);//--set contrast control register
OLED_WR_Byte(0xCF,OLED_CMD);// Set SEG Output Current Brightness
OLED_WR_Byte(0xA1,OLED_CMD);//--Set SEG/Column Mapping 0xa0左右反置 0xa1正常
OLED_WR_Byte(0xC8,OLED_CMD);//Set COM/Row Scan Direction 0xc0上下反置 0xc8正常
OLED_WR_Byte(0xA6,OLED_CMD);//--set normal display
OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)
OLED_WR_Byte(0x3f,OLED_CMD);//--1/64 duty
OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset Shift Mapping RAM Counter (0x00~0x3F)
OLED_WR_Byte(0x00,OLED_CMD);//-not offset
OLED_WR_Byte(0xd5,OLED_CMD);//--set display clock divide ratio/oscillator frequency
OLED_WR_Byte(0x80,OLED_CMD);//--set divide ratio, Set Clock as 100 Frames/Sec
OLED_WR_Byte(0xD9,OLED_CMD);//--set pre-charge period
OLED_WR_Byte(0xF1,OLED_CMD);//Set Pre-Charge as 15 Clocks & Discharge as 1 Clock
OLED_WR_Byte(0xDA,OLED_CMD);//--set com pins hardware configuration
OLED_WR_Byte(0x12,OLED_CMD);
OLED_WR_Byte(0xDB,OLED_CMD);//--set vcomh
OLED_WR_Byte(0x30,OLED_CMD);//Set VCOM Deselect Level
OLED_WR_Byte(0x20,OLED_CMD);//-Set Page Addressing Mode (0x00/0x01/0x02)
OLED_WR_Byte(0x02,OLED_CMD);//
OLED_WR_Byte(0x8D,OLED_CMD);//--set Charge Pump enable/disable
OLED_WR_Byte(0x14,OLED_CMD);//--set(0x10) disable
OLED_Clear();
OLED_WR_Byte(0xAF,OLED_CMD);
}
#include "stm32f10x.h"
#include "key.h"
#include "sys.h"
#include "delay.h"
#include "usart.h"
//按键初始化函数
void KEY_Init(void) //IO初始化
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB,ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化
GPIO_SetBits(GPIOB,GPIO_Pin_5);
GPIO_InitStructure.GPIO_Pin = KEY1|KEY2|KEY3|KEY4; //行 a 4 5 6 7
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //PA0设置成输入,默认上拉
GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.0
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3; //lie
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //a 4 5 6 7
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_ResetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3); //a 0 1 2 3
}
u8 Key = 0;
u8 flag=0;
u16 abc[9] = {0,0,0,0,0,0,0,0,0};
u8 i = 0;
u8 Flag = 0;
int sum1 = 0;
int sun1_read()
{
return sum1;
}
u16 * abd_read();
u16 *abc_read()
{
return &abc[0];
}
int Wei_Shu(u8 a)
{
int sum = 1, i = 0;
for(;i9)
{
i = 0;
}
}
}
else if(*key1 == 13){
sum1 = 0;
//
for(i=0;i<9;i++)
abc[i] = 0;
i = 0;
}
ok = 0;
}
if(led_liang==1 && sum1>0){
GPIO_SetBits(GPIOB,GPIO_Pin_5);
//delay_ms(1);
sum1--;
}
if(sum1==0){
led_liang = 0;
GPIO_ResetBits(GPIOB,GPIO_Pin_5);
}
}
void KEY_config(u8 *Key)
{
char i=1,j=1;
*Key = 0;
for(;i<5;i++){
GPIO_ResetBits(GPIOA,GPIO_Pin_0<<(i-1)); //先拉低
if((KEY1_Input == RESET) || (KEY2_Input == RESET) || (KEY3_Input == RESET) || (KEY4_Input == RESET))
{
delay_ms(10);
for(j=1;j<5;j++){
if(GPIO_ReadInputDataBit(KEY_PROT,GPIO_Pin_4<<(j-1)) == RESET){
flag = 1;
*Key = (i-1)*4+j;
while(!GPIO_ReadInputDataBit(KEY_PROT,GPIO_Pin_4<<(j-1))){};
//Key_Input(Key);
ok = 1;
break;
}
else
flag = 0;
}
}
GPIO_SetBits(GPIOA,GPIO_Pin_0<<(i-1)); //再次拉高
}
}
#include "led.h"
#include "delay.h"
#include "key.h"
#include "sys.h"
#include "usart.h"
#include "exti.h"
#include "beep.h"
#include "timer.h"
#include "stm32f10x_tim.h"
#include "oled.h"
u8 Key000=0;
//定时器3中断服务程序
void TIM3_IRQHandler(void) //TIM3中断
{
if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查TIM3更新中断发生与否
{
TIM_ClearITPendingBit(TIM3, TIM_IT_Update); //清除TIMx更新中断标志
KEY_config(&Key000);
Key_Input(&Key000);
}
}
char a[9]={0,0,0,0,0,0,0,0,0};
char *abd_xiugai(u16 abc[9]){
char i = 0;
while(abc[i]!=0){
a[i] = abc[i] + 48;
i++;
if(i==7)
break;
}
return a;
}
char b[5];
char *sun1_read_xiu_acill(int a){
//1234000s
a = a/1000;
b[3] = (char)(a%10+48);
b[2] = (char)((a/10)%10)+48;
b[1] = (char)((a/100)%10)+48;
b[0] = (char)(a/1000)+48;
b[4] = 0;
return b;
}
int main(void)
{
u8 is_anxia = 0;
delay_init(); //延时函数初始化
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
uart_init(4800); //串口初始化4800
KEY_Init();
//LED_Init(); //初始化与LED连接的硬件接口
//LED0=0; //先点亮红灯
TIM3_Int_Init(89999,2);
OLED_Init();
OLED_ColorTurn(0);//0正常显示,1 反色显示
OLED_DisplayTurn(0);//0正常显示 1 屏幕翻转显示
while(1)
{
OLED_Clear();
if(*(abc_read()+0) == 0 && sun1_read()==0)
OLED_ShowString(10,0,"P 16",16,1);
else if(sun1_read()==0)
OLED_ShowString(10,0,abd_xiugai(abc_read()),16,1);
else if(sun1_read()!=0)
OLED_ShowString(10,0,sun1_read_xiu_acill(sun1_read()),16,1);
OLED_Refresh();
}
}
/*
void TIM3_IRQHandler(void) //TIM3中断
{
if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查TIM3更新中断发生与否
{
TIM_ClearITPendingBit(TIM3, TIM_IT_Update ); //清除TIMx更新中断标志
Key_Input(&key);
}
}*/
先来个总述
接线方式 矩阵键盘的8个引脚对应GPIOA 0---7
led灯对应 GPIOB 5 可以替换为一个LS298N模块控制电机 或者其他的很多东西
oled屏幕对应接线 scl - GPIO B1
sda-GPIOB7
VCC和GND随意了
接下来是对此代码的详解
首先是对于key.h 文件的解释我们对于矩阵键盘对应IO口的一个声明,之后我们可以将对应的IO口进行简化方便使用,也使我们的代码更有可读性。
对于oled.h也是同理,这就不在啰嗦了,应该都懂吧
对于key.c 文件是本文章的核心部分
首先是对于矩阵键盘八个对应引脚的初始化这里我们选择扫描的方式来进行按键检测,但是之后我遇到了一个大问题 那就是延时的问题 会发现在我们使用oled后,时间会变得不可控。
在这里我们对于八个IO口选择 4个推挽输出 4个上拉输出的模式 在这里我将led灯对应的IO口初始化也放在这里了。对于io口的初始化就不在详细的说明 ,没什么好说的 只需要注意和.h文件的对应关系
之后我们来说说
void KEY_config(u8 *Key)
对于此函数需要一丁点的位操作知识 用到了一点<< >>的操作
void KEY_config(u8 *Key)
{
char i=1,j=1;
*Key = 0;
for(;i<5;i++){
GPIO_ResetBits(GPIOA,GPIO_Pin_0<<(i-1)); //先拉低
if((KEY1_Input == RESET) || (KEY2_Input == RESET) || (KEY3_Input == RESET) || (KEY4_Input == RESET))
{
delay_ms(10);
for(j=1;j<5;j++){
if(GPIO_ReadInputDataBit(KEY_PROT,GPIO_Pin_4<<(j-1)) == RESET){
flag = 1;
*Key = (i-1)*4+j;
while(!GPIO_ReadInputDataBit(KEY_PROT,GPIO_Pin_4<<(j-1))){};
//Key_Input(Key);
ok = 1;
break;
}
else
flag = 0;
}
}
GPIO_SetBits(GPIOA,GPIO_Pin_0<<(i-1)); //再次拉高
}
}
我们在初始化时将4路IO口设置为默认上拉 也就是一直输出高电平
所以我们只需要循环检测 电平的高低 如果有按键按下 那么图上边的一排1 必然有一路变为低电平
我们只需要在循环扫描一列4路哪一路为低电平 如果2路都为低电平则可以确认按键的位置 相当于在按键位置花了一个十字 确认位置。
好了原理已经说清了,借来下进入我们的代码部分
我们循环拉低四行的电平 那么比如我们再按下第一个按键时可以知道第一列的电平也会被拉低
这就对应了我们的前五行代码
之间插入一个按键消抖操作
暂时先写这些 , 这篇有点太长了 剩下的详解日后再更