char getKey(void) { int row, col; // 将行引脚设置为高电平,列引脚设置为输入 for (row = 0; row < ROWS; row++) { GPIO_ResetBits(GPIOA, GPIO_Pin_0 | GPIO_Pin_4 | GPIO_Pin_6 | GPIO_Pin_7); GPIO_SetBits(GPIOA, GPIO_Pin_0 << row); for (col = 0; col < COLS; col++) { if (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_4 << col) == Bit_RESET) { // 等待按键释放 while (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_4 << col) == Bit_RESET); // 返回按键对应字符 return keymap[row][col]; } } } // 没有按键按下,返回空字符 return '\0'; } 在这个基础上修改获取按键状态

时间: 2024-02-14 18:29:59 浏览: 181
根据您提供的 `getKey` 函数,以下是修改后的函数,用于获取按键状态: ```c #include <stdio.h> #include "stm32f4xx.h" #define ROWS 4 #define COLS 4 GPIO_InitTypeDef GPIO_InitStructure; void GPIO_Configuration(void) { // 行引脚配置为输出 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_4 | GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); // 列引脚配置为输入 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; GPIO_Init(GPIOA, &GPIO_InitStructure); } char GetKeyPressed(void) { char keymap[ROWS][COLS] = { {'1', '2', '3', 'A'}, {'4', '5', '6', 'B'}, {'7', '8', '9', 'C'}, {'*', '0', '#', 'D'} }; int row, col; // 将行引脚设置为低电平 GPIO_ResetBits(GPIOA, GPIO_Pin_0 | GPIO_Pin_4 | GPIO_Pin_6 | GPIO_Pin_7); // 遍历行引脚 for (row = 0; row < ROWS; row++) { // 设置当前行引脚为高电平 GPIO_SetBits(GPIOA, GPIO_Pin_0 << row); // 检测列引脚状态 for (col = 0; col < COLS; col++) { if (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_4 << col) == Bit_RESET) { // 等待按键释放 while (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_4 << col) == Bit_RESET); // 返回按键对应字符 return keymap[row][col]; } } } // 没有按键按下,返回空字符 return '\0'; } int main(void) { GPIO_Configuration(); while (1) { // 获取按键状态 char keyPressed = GetKeyPressed(); // 在这里进行您的业务逻辑 // 根据按键状态进行相应操作 // ... // 延迟一段时间,避免频繁读取按键状态 for (int i = 0; i < 100000; i++) {} } } ``` 在上面的示例代码中,我修改了行引脚和列引脚的配置,并实现了 `GetKeyPressed` 函数来获取按键状态。该函数会遍历行引脚和列引脚,检测按键是否按下,并返回对应的字符。您可以在主循环中使用该函数来获取按键状态,并根据需要进行相应的操作。请注意,示例代码中还添加了一个简单的延迟以避免频繁读取按键状态。
阅读全文

相关推荐

Main主程序 #include "sys.h" #include "delay.h" #include "adc.h" #include "gpio.h" #include "OLED_I2C.h" #include "stmflash.h" #include "ds18b20.h" #include "timer.h" #include "usart1.h" #include <stdio.h> #include <stdlib.h> #include <string.h> #include <stdbool.h> #define FLASH_SAVE_ADDR ((u32)0x0800F000) //设置FLASH 保存地址(必须为偶数) #define STM32_RX1_BUF Usart1RecBuf #define STM32_Rx1Counter RxCounter #define STM32_RX1BUFF_SIZE USART1_RXBUFF_SIZE extern const unsigned char BMP1[]; extern const unsigned char BMP2[]; char display[16]; //显示缓存区 short temperature=0; //温度 u8 setTempValue = 40; //温度上限 u8 setSmokeValue = 60; //烟雾上限 u8 sendSmsFlag = 0; //发送短信标志 u8 alarmFlag = 0x00; //蜂鸣器报警标志 u16 smoke=0; //烟雾 u8 setn=0; //记录设置按键按下的次数 bool shanshuo=0; bool shuaxin=0; bool fangdao=0; char PhoneNumber[11];//手机号码 char uniPhoneNum[44];//手机号码转码后存放数组 void UsartRx1BufClear(void) { memset(STM32_RX1_BUF, 0, STM32_RX1BUFF_SIZE);//清除缓存 STM32_Rx1Counter = 0; } /***********************************************************************************************/ /****************************** 以下为SIM800相关部分 ****************************************/ /***********************************************************************************************/ void PhoneNumChangeUnicode(char *str1,char *str2) //手机号转码 { u8 i; char *buf = str1; for(i = 0;i < 11;i ++) //发送中文短信手机号必须转码,前面需要加上003 { str2[i*4+0] = '0'; str2[i*4+1] = '0'; str2[i*4+2] = '3'; str2[i*4+3] = buf[i]; } } void gsm_atcmd_send(char *at)//GSM AT指令发送函数 { unsigned short waittry;//延时变量 do { gsm_rev_start = 0;//接收开始标志清零 gsm_rev_okflag = 0;//接收完成标志清零 waittry = 0;//延时变量清零 uart1_send((unsigned char *)at,0xFF);//串口发送内容 while(waittry ++ < 3000)//进入while延时 { if (gsm_rev_okflag == 1)//等待GSM返回OK { return;//返回出去 } delay_ms(1); } } while(gsm_rev_okflag == 0); } void gsm_init(void)//gsm初始化 { gsm_atcmd_send("AT\r\n");//测试用的 delay_ms(1000); gsm_atcmd_send("AT+CSCS=\"UCS2\"\r\n");//设置为unicode编码 delay_ms(1000); gsm_atcmd_send("AT+CMGF=1\r\n");//设置为文本模式 delay_ms(1000); gsm_atcmd_send("AT+CNMI=2,1\r\n");//来短信时提示,并存储到模块内存 delay_ms(1000); gsm_atcmd_send("AT+CMGD=1,4\r\n");//清除短信 delay_ms(1000); gsm_atcmd_send("AT+CSMP=17,0,2,25\r\n");//设置短信保留为5分钟,发送中文 } /* *number 为对方手机号 */ void gsm_send_msg(const char*number,char * content) { u8 len; unsigned char gsm_at_txbuf[60];//GSM AT命令缓存区 memset(gsm_at_txbuf, 0, 60);//缓存清零 strncpy((char *)gsm_at_txbuf,"AT+CMGS=\"",9);//将AT+CMGS=\",复制到gsm_at_txbuf memcpy(gsm_at_txbuf + 9, number, 44);//将手机号码复制到AT+CMGS=\"之后 len = strlen((char *)gsm_at_txbuf);//获取gsm_at_txbuf字符串长度 gsm_at_txbuf[len] = '"'; // AT+CMGS=\"12345678901\" gsm_at_txbuf[len + 1] = '\r'; gsm_at_txbuf[len + 2] = '\n';//gsm_at_txbuf最终的格式"AT+CMGS=\"手机号码\"\r\n" uart1_send(gsm_at_txbuf,0xFF);//发送需要接受短信的手机号码 delay_ms(1000); uart1_send((unsigned char *)content,0xFF); //发短信内容 delay_ms(100); printf("%c",0x1a); //发送结束符号 delay_ms(10); } /* *content 为短信内容 */ void sim800_send_sms(char *content) { bool send_error = 0; u16 send_count = 0; gsm_rev_okflag = 0; OLED_ShowStr(32,2,"Send Sms... ",2,0); PhoneNumChangeUnicode(PhoneNumber,uniPhoneNum); //在发送短信前,先将手机号转码 gsm_send_msg(uniPhoneNum,content);//发送短信 delay_ms(1000);//延时1秒 while(gsm_rev_okflag == 0)//等待返回OK指令 { if(send_count++ > 8000) { send_count = 0; send_error = 1; break; } delay_ms(1); }; gsm_rev_okflag = 0; if(send_error == 1) OLED_ShowStr(32,2,"Send Fail! ",2,0);//显示发送超时 else OLED_ShowStr(32,2," Send OK! ",2,0); UsartRx1BufClear(); delay_ms(1000);//延时1秒 OLED_ShowStr(32,2," ",2,0); } /***********************************************************************************************/ /****************************** end ****************************************/ /***********************************************************************************************/ void STM32_FlashCheck(void) // 检查是否是新的单片机,是的话清空存储区,否则保留 { u8 comper_str[6],i; STMFLASH_Read(FLASH_SAVE_ADDR + 0x10,(u16*)comper_str,5); comper_str[5] = '\0'; if(strstr((char *)comper_str,"FDYDZ") == NULL) //新的单片机 { STMFLASH_Write(FLASH_SAVE_ADDR + 0x10,(u16*)"FDYDZ",5); //写入“FDYDZ”,方便下次校验 delay_ms(50); STMFLASH_Write(FLASH_SAVE_ADDR + 0x40,(u16*)"12345678910",11);//存入初始手机号 delay_ms(50); } STMFLASH_Read(FLASH_SAVE_ADDR + 0x40,(u16*)PhoneNumber,11); //读出手机号 for(i = 0; i < 11 ; i++) { if(PhoneNumber[i]<'0' || PhoneNumber[i]>'9') { break; } } if(i != 11) { memset(PhoneNumber, 0 , 11); //清除缓存 sprintf(PhoneNumber,"12345678910"); } delay_ms(100); } void display_mode(void) { u8 i; //显示中文: 防盗模式 if(fangdao==1){for(i = 0;i < 4;i ++)OLED_ShowCN(i*16+32,0,i+0,1);}else {for(i = 0;i < 4;i ++)OLED_ShowCN(i*16+32,0,i+24,1);} } void displayInitInterface(void) //显示初始页面 { u8 i; for(i = 0;i < 2;i ++)OLED_ShowCN(i*16,4,i+6,0); //显示中文: 温度 for(i = 0;i < 2;i ++)OLED_ShowCN(i*16,6,i+8,0); //显示中文: 烟雾 OLED_ShowChar(32,0,':',2,0); OLED_ShowChar(32,4,':',2,0); OLED_ShowChar(32,6,':',2,0); display_mode(); } void Get_Temperature(void) //获取温度 { temperature=ReadTemperature(); if(temperature>=setTempValue) { if(!(alarmFlag&0x01)) { alarmFlag|=0x01; shanshuo = 0; sendSmsFlag = 2; //发送短信标志 } } else { alarmFlag&=0xFE; } if(temperature>=setTempValue && shanshuo) { OLED_ShowStr(40, 4," ", 2,0); } else { sprintf(display," %d",temperature); OLED_ShowStr(40, 4, (u8*)display, 2,0);//显示温度 OLED_ShowCentigrade(68, 4); //显示摄氏度 } } void Get_Smoke(void) //获取烟雾浓度 { u16 test_adc=0; /* 获取烟雾浓度 */ test_adc = Get_Adc_Average(ADC_Channel_9,10);//读取通道9的10次AD平均值 smoke = test_adc*99/4096;//转换成0-99百分比 if(smoke>=setSmokeValue) { if(!(alarmFlag&0x02)) { alarmFlag|=0x02; shanshuo = 0; sendSmsFlag = 3; //发送短信标志 } } else { alarmFlag&=0xFD; } if(smoke>=setSmokeValue && shanshuo) { OLED_ShowStr(40, 6," ", 2,0); } else { sprintf(display," %02d %%",smoke); OLED_ShowStr(40, 6, (u8*)display, 2,0);//显示温度 } } void displaySetValue(void) //显示设置值 { u8 add=2,i; if(setn==1) { OLED_ShowChar(56,4,setTempValue%100/10+'0',2,0);//显示 OLED_ShowChar(64,4,setTempValue%10+'0',2,0);//显示 } if(setn==2) { OLED_ShowChar(56,4,setSmokeValue%100/10+'0',2,0);//显示 OLED_ShowChar(64,4,setSmokeValue%10+'0',2,0);//显示 OLED_ShowChar(72,4,'%',2,0); } if(setn>=3) { for(i = 0;i < 11;i ++) { OLED_ShowChar((add++)*8,4,PhoneNumber[i],2,(setn+1)-(3+i));//显示手机号码 } } } void keyscan(void) //按键扫描 { u8 i; if(KEY1 == 0) //设置键 { delay_ms(20); if(KEY1 == 0) { while(KEY1 == 0); setn ++; if(setn == 1) { OLED_CLS();//清屏 for(i = 0;i < 4;i ++)OLED_ShowCN(i*16+32,0,i+10,0);//显示中文:设置温度 OLED_ShowCentigrade(75, 4); //显示摄氏度 } if(setn == 2) { for(i = 0;i < 4;i ++)OLED_ShowCN(i*16+32,0,i+14,0);//显示中文:设置烟雾 OLED_ShowChar(80,4,' ',2,0); } if(setn == 3) { for(i = 0;i < 6;i ++)OLED_ShowCN(i*16+16,0,i+18,0);//显示中文:设置手机号码 } if(setn >= 14) { setn = 0; OLED_CLS();//清屏 displayInitInterface(); STMFLASH_Write(FLASH_SAVE_ADDR + 0x40,(u16*)PhoneNumber,11); //退出设置,存入设置的手机号 } displaySetValue(); } } if(KEY2 == 0) //加键 { delay_ms(80); if(KEY2 == 0) { if(setTempValue < 99 && setn==1)setTempValue++; if(setSmokeValue < 99 && setn==2)setSmokeValue++; if(setn>=3) { PhoneNumber[setn-3]++; if(PhoneNumber[setn-3]>'9')PhoneNumber[setn-3]='0'; } displaySetValue(); } } if(KEY3 == 0) //减键 { delay_ms(80); if(KEY3 == 0) { if(setn==0) { fangdao=!fangdao; display_mode(); } if(setTempValue > 0 && setn==1)setTempValue--; if(setSmokeValue > 0 && setn==2)setSmokeValue--; if(setn>=3) { PhoneNumber[setn-3]--; if(PhoneNumber[setn-3]<'0')PhoneNumber[setn-3]='9'; } displaySetValue(); } } } int main(void) { bool flameFlag=0; bool SomebodyFlag=0; char SEND_BUF[400]; //发送短信缓存 delay_init(); //延时函数初始化 NVIC_Configuration(); //中断优先级配置 I2C_Configuration(); //IIC初始化 STM32_FlashCheck(); //FLASH初始化 delay_ms(200); OLED_Init(); //OLED液晶初始化 OLED_CLS(); //清屏 OLED_ShowStr(0,2," GSM Init... ",2,0); uart1_Init(9600); gsm_init();//gsm初始化 OLED_CLS();//清屏 Adc_Init(); //adc初始化 KEY_GPIO_Init(); //按键引脚初始化 SR501_GPIO_Init(); //人体红外初始化 DS18B20_GPIO_Init(); //温度初始化 DS18B20_Init(); //初始化显示 displayInitInterface(); //显示初始界面 TIM3_Init(99,719); //定时器初始化,定时1ms //Tout = ((arr+1)*(psc+1))/Tclk ; //Tclk:定时器输入频率(单位MHZ) //Tout:定时器溢出时间(单位us) while(1) { keyscan(); //按键扫描 if(setn == 0) { if(shuaxin == 1) //大概300ms刷新一次数据 { Get_Temperature(); //获取温度 Get_Smoke(); //获取烟雾 shuaxin = 0; } if(FLAME == 0) //检测到火焰 { delay_ms(10); if(FLAME == 0) { if(flameFlag == 0) { OLED_DrawBMP(88,4,120,8,(unsigned char *)BMP1); //显示火焰图片 sendSmsFlag = 1; //发送短信标志 } flameFlag = 1; } } else { if(flameFlag == 1) { OLED_ShowStr(88, 4, " ", 2,0); OLED_ShowStr(88, 6, " ", 2,0); } flameFlag = 0; } if(fangdao==1&&SR501==1) //在防盗模式下检测到有人 { if(SomebodyFlag==0) { OLED_DrawBMP(0,0,32,4,(unsigned char *)BMP2); //图显示 sendSmsFlag = 4; //发送短信标志 SomebodyFlag = 1; } } else { if(SomebodyFlag==1) { OLED_ShowStr(0, 0, " ", 2,0); OLED_ShowStr(0, 2, " ", 2,0); SomebodyFlag = 0; } } if(temperature>=setTempValue || smoke>=setSmokeValue || flameFlag || SomebodyFlag)BEEP=1;else BEEP=0; //检测到温度烟雾超标火焰蜂鸣器报警 if(temperature>=setTempValue)FAN=1;else FAN=0; //温度超标都开启风扇 if(smoke>=setSmokeValue || flameFlag)RELAY = 1; else RELAY = 0; //检测到有火或者烟雾超标,开启水泵 if(sendSmsFlag != 0) //发送短信 { char TEMP_BUF[100]; /*******************************************************************************************/ /*******************以下为短信内容处理部分,发送中文短信必须转换为Unicode码**************/ /******************************************************************************************/ memset(SEND_BUF,0,sizeof(SEND_BUF)); //清空缓冲区 switch(sendSmsFlag) { case(1): strcat(SEND_BUF,"8B66544AFF0168C06D4B5230706B7130FF01"); break; //警告!检测到火焰! case(2): strcat(SEND_BUF,"8B66544AFF016E295EA68FC79AD8FF01"); break; //警告!温度过高! case(3): strcat(SEND_BUF,"8B66544AFF0170DF96FE6D535EA68FC79AD8FF01"); break; //警告!烟雾浓度过高! case(4): strcat(SEND_BUF,"8B66544AFF0168C06D4B523067094EBAFF01"); break; //警告!检测到有人! default: break; } if(sendSmsFlag!=4) { memset(TEMP_BUF,0,sizeof(TEMP_BUF)); //清空缓冲区 sprintf(TEMP_BUF,"6E295EA6003A003%1d003%1d2103FF0C70DF96FE003A003%1d003%1d0025",temperature/10,temperature%10,smoke/10,smoke%10); strcat(SEND_BUF,TEMP_BUF); } sim800_send_sms((char *)SEND_BUF);//发送短信 sendSmsFlag = 0; /*******************************************************************************************/ /*************************** end *****************************/ /******************************************************************************************/ } } delay_ms(10); } } void TIM3_IRQHandler(void)//定时器3中断服务程序,用于记录时间 { static u16 timeCount1 = 0; if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查指定的TIM中断发生与否:TIM 中断源 { TIM_ClearITPendingBit(TIM3, TIM_IT_Update); //清除中断标志位 timeCount1++; if(timeCount1 >= 300) //300ms { timeCount1 = 0; shanshuo = !shanshuo; shuaxin = 1; } } } ADC初始化程序 #include "adc.h" #include "delay.h" //初始化ADC //这里我们仅以规则通道为例 //我们默认将开启通道0~3 void Adc_Init(void) { ADC_InitTypeDef ADC_InitStructure; GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB |RCC_APB2Periph_ADC1 , ENABLE ); //使能ADC1通道时钟 RCC_ADCCLKConfig(RCC_PCLK2_Div8); //设置ADC分频因子6 72M/8=9,ADC最大时间不能超过14M GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1|GPIO_Pin_0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚 GPIO_Init(GPIOB, &GPIO_InitStructure); ADC_DeInit(ADC1); //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值 ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式 ADC_InitStructure.ADC_ScanConvMode = DISABLE; //模数转换工作在单通道模式 ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //模数转换工作在单次转换模式 ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件而不是外部触发启动 ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐 ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC通道的数目 ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器 ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1 ADC_ResetCalibration(ADC1); //使能复位校准 while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束 ADC_StartCalibration(ADC1); //开启AD校准 while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束 } //获得ADC值 //ch:通道值 9 u16 Get_Adc(u8 ch) { //设置指定ADC的规则组通道,一个序列,采样时间 ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_71Cycles5 ); //ADC1,ADC通道,采样时间为13.5周期 ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能 while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束 return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果 } u16 Get_Adc_Average(u8 ch,u8 times) { u32 temp_val=0; u8 t; for(t=0;t<times;t++) { temp_val+=Get_Adc(ch); delay_ms(5); } return temp_val/times; } #ifndef __ADC_H #define __ADC_H #include "sys.h" void Adc_Init(void); u16 Get_Adc(u8 ch); u16 Get_Adc_Average(u8 ch,u8 times); #endif GPIO引脚初始化程序 #include "gpio.h" ////////////////////////////////////////////////////////////////////////////////// //按键舵机的GPIO设置 ////////////////////////////////////////////////////////////////////////////////// void KEY_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOC|RCC_APB2Periph_AFIO, ENABLE); //使能PABC端口时钟 GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE); //关闭JTAG模式 使PB3,PB4变成普通IO口 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12|GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15; // 端口配置 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //上拉输入 GPIO_Init(GPIOB, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15; // 端口配置 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO口速度为50MHz GPIO_Init(GPIOC, &GPIO_InitStructure); GPIO_ResetBits(GPIOC,GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15); //输出0 } void SR501_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能PA端口时钟 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; // 端口配置 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //下拉输入 GPIO_Init(GPIOA, &GPIO_InitStructure); } #ifndef __GPIO_H #define __GPIO_H #include "sys.h" #define KEY1 PBin(12) #define KEY2 PBin(13) #define KEY3 PBin(14) #define FLAME PBin(15) #define BEEP PCout(13) #define FAN PCout(14) #define RELAY PCout(15) #define SR501 PAin(0) void KEY_GPIO_Init(void);//引脚初始化 void SR501_GPIO_Init(void); #endif OLED驱动程序 /************************************************************************************ * * Description:128*64点阵的OLED显示屏驱动文件SD1306驱动IIC通信方式显示屏 * * Others: none; * * Function List: * 1. void I2C_Configuration(void) -- 配置CPU的硬件I2C * 2. void I2C_WriteByte(uint8_t addr,uint8_t data) -- 向寄存器地址写一个byte的数据 * 3. void WriteCmd(unsigned char I2C_Command) -- 写命令 * 4. void WriteDat(unsigned char I2C_Data) -- 写数据 * 5. void OLED_Init(void) -- OLED屏初始化 * 6. void OLED_SetPos(unsigned char x, unsigned char y) -- 设置起始点坐标 * 7. void OLED_Fill(unsigned char fill_Data) -- 全屏填充 * 8. void OLED_CLS(void) -- 清屏 * 9. void OLED_ON(void) -- 唤醒 * 10. void OLED_OFF(void) -- 睡眠 * 11. void OLED_ShowStr(unsigned char x, unsigned char y, unsigned char ch[], unsigned char TextSize) -- 显示字符串(字体大小有6*8和8*16两种) * 12. void OLED_ShowCN(unsigned char x, unsigned char y, unsigned char N) -- 显示中文(中文需要先取模,然后放到codetab.h中) * 13. void OLED_DrawBMP(unsigned char x0,unsigned char y0,unsigned char x1,unsigned char y1,unsigned char BMP[]) -- BMP图片 * * History: none; * *************************************************************************************/ #include "OLED_I2C.h" #include "delay.h" #include "codetab.h" /* 定义I2C总线连接的GPIO端口, 用户只需要修改下面4行代码即可任意改变SCL和SDA的引脚 */ #define RCC_I2C_PORT RCC_APB2Periph_GPIOB /* GPIO端口时钟 */ #define PORT_I2C_SCL GPIOB /* GPIO端口 */ #define PIN_I2C_SCL GPIO_Pin_6 /* GPIO引脚 */ #define PORT_I2C_SDA GPIOB /* GPIO端口 */ #define PIN_I2C_SDA GPIO_Pin_7 /* GPIO引脚 */ #define I2C_SCL_PIN GPIO_Pin_6 /* 连接到SCL时钟线的GPIO */ #define I2C_SDA_PIN GPIO_Pin_7 /* 连接到SDA数据线的GPIO */ /* 定义读写SCL和SDA的宏 */ #define I2C_SCL_1() PORT_I2C_SCL->BSRR = I2C_SCL_PIN /* SCL = 1 */ #define I2C_SCL_0() PORT_I2C_SCL->BRR = I2C_SCL_PIN /* SCL = 0 */ #define I2C_SDA_1() PORT_I2C_SDA->BSRR = I2C_SDA_PIN /* SDA = 1 */ #define I2C_SDA_0() PORT_I2C_SDA->BRR = I2C_SDA_PIN /* SDA = 0 */ #define I2C_SDA_READ() ((PORT_I2C_SDA->IDR & I2C_SDA_PIN) != 0) /* 读SDA口线状态 */ #define I2C_SCL_READ() ((PORT_I2C_SCL->IDR & I2C_SCL_PIN) != 0) /* 读SCL口线状态 */ /* ********************************************************************************************************* * 函 数 名: bsp_InitI2C * 功能说明: 配置I2C总线的GPIO,采用模拟IO的方式实现 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void bsp_InitI2C_2(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_I2C_PORT, ENABLE); /* 打开GPIO时钟 */ GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; /* 开漏输出模式 */ GPIO_InitStructure.GPIO_Pin = PIN_I2C_SCL; GPIO_Init(PORT_I2C_SCL, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = PIN_I2C_SDA; GPIO_Init(PORT_I2C_SDA, &GPIO_InitStructure); /* 给一个停止信号, 复位I2C总线上的所有设备到待机模式 */ i2c_Stop_2(); } /* ********************************************************************************************************* * 函 数 名: i2c_Start * 功能说明: CPU发起I2C总线启动信号 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_Start_2(void) { /* 当SCL高电平时,SDA出现一个下跳沿表示I2C总线启动信号 */ I2C_SDA_1(); I2C_SCL_1(); delay_us(4); I2C_SDA_0(); delay_us(4); I2C_SCL_0(); } /* ********************************************************************************************************* * 函 数 名: i2c_Start * 功能说明: CPU发起I2C总线停止信号 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_Stop_2(void) { /* 当SCL高电平时,SDA出现一个上跳沿表示I2C总线停止信号 */ I2C_SDA_0(); I2C_SCL_1(); delay_us(4); I2C_SDA_1(); delay_us(4); } /* ********************************************************************************************************* * 函 数 名: i2c_SendByte * 功能说明: CPU向I2C总线设备发送8bit数据 * 形 参: _ucByte : 等待发送的字节 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_SendByte_2(uint8_t _ucByte) { uint8_t i; /* 先发送字节的高位bit7 */ for (i = 0; i < 8; i++) { if (_ucByte & 0x80) { I2C_SDA_1(); } else { I2C_SDA_0(); } delay_us(2); I2C_SCL_1(); delay_us(2); I2C_SCL_0(); _ucByte <<= 1; /* 左移一个bit */ delay_us(2); } } /* ********************************************************************************************************* * 函 数 名: i2c_ReadByte * 功能说明: CPU从I2C总线设备读取8bit数据 * 形 参: 无 * 返 回 值: 读到的数据 ********************************************************************************************************* */ uint8_t i2c_ReadByte_2(void) { uint8_t i; uint8_t value; /* 读到第1个bit为数据的bit7 */ value = 0; for (i = 0; i < 8; i++) { value <<= 1; I2C_SCL_1();//SCL在高电平期间,数据必须保持稳定 delay_us(2); if (I2C_SDA_READ()) { value++; } I2C_SCL_0(); delay_us(1); } return value; } /* ********************************************************************************************************* * 函 数 名: i2c_WaitAck * 功能说明: CPU产生一个时钟,并读取器件的ACK应答信号 * 形 参: 无 * 返 回 值: 返回0表示正确应答,1表示无器件响应 ********************************************************************************************************* */ uint8_t i2c_WaitAck_2(void) { uint8_t tempTime; I2C_SDA_1(); /* CPU释放SDA总线 */ delay_us(1); I2C_SCL_1(); /* CPU驱动SCL = 1, 此时器件会返回ACK应答 */ delay_us(1); while(I2C_SDA_READ()) { tempTime++; if(tempTime>250) { i2c_Stop_2(); return 1; } } I2C_SCL_0(); return 0; } /* ********************************************************************************************************* * 函 数 名: i2c_Ack * 功能说明: CPU产生一个ACK信号 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_Ack_2(void) { I2C_SDA_0(); /* CPU驱动SDA = 0 */ delay_us(5); I2C_SCL_1(); /* CPU产生1个时钟 */ delay_us(5); I2C_SCL_0(); delay_us(5); I2C_SDA_1(); /* CPU释放SDA总线 */ } /* ********************************************************************************************************* * 函 数 名: i2c_NAck * 功能说明: CPU产生1个NACK信号 * 形 参: 无 * 返 回 值: 无 ********************************************************************************************************* */ void i2c_NAck_2(void) { I2C_SDA_1(); /* CPU驱动SDA = 1 */ delay_us(5); I2C_SCL_1(); /* CPU产生1个时钟 */ delay_us(5); I2C_SCL_0(); delay_us(5); } void I2C_Configuration(void) { bsp_InitI2C_2(); } void I2C_WriteByte(uint8_t addr,uint8_t data) { /* 第1步:发起I2C总线启动信号 */ i2c_Start_2(); /* 第2步:发起控制字节,高7bit是地址,bit0是读写控制位,0表示写,1表示读 */ i2c_SendByte_2(OLED_ADDRESS | I2C_WR); /* 此处是写指令 */ /* 第3步:发送ACK */ i2c_WaitAck_2(); /* 第4步:发送字节地址 */ i2c_SendByte_2(addr); i2c_WaitAck_2(); /* 第5步:开始写入数据 */ i2c_SendByte_2(data); /* 第6步:发送ACK */ i2c_WaitAck_2(); /* 发送I2C总线停止信号 */ i2c_Stop_2(); } void WriteCmd(unsigned char I2C_Command)//写命令 { I2C_WriteByte(0x00, I2C_Command); } void WriteDat(unsigned char I2C_Data)//写数据 { I2C_WriteByte(0x40, I2C_Data); } void OLED_Init(void) { delay_ms(100); //这里的延时很重要 WriteCmd(0xAE); //display off WriteCmd(0x20); //Set Memory Addressing Mode WriteCmd(0x10); //00,Horizontal Addressing Mode;01,Vertical Addressing Mode;10,Page Addressing Mode (RESET);11,Invalid WriteCmd(0xb0); //Set Page Start Address for Page Addressing Mode,0-7 WriteCmd(0xc8); //Set COM Output Scan Direction WriteCmd(0x00); //---set low column address WriteCmd(0x10); //---set high column address WriteCmd(0x40); //--set start line address WriteCmd(0x81); //--set contrast control register WriteCmd(0xff); //亮度调节 0x00~0xff WriteCmd(0xa1); //--set segment re-map 0 to 127 WriteCmd(0xa6); //--set normal display WriteCmd(0xa8); //--set multiplex ratio(1 to 64) WriteCmd(0x3F); // WriteCmd(0xa4); //0xa4,Output follows RAM content;0xa5,Output ignores RAM content WriteCmd(0xd3); //-set display offset WriteCmd(0x00); //-not offset WriteCmd(0xd5); //--set display clock divide ratio/oscillator frequency WriteCmd(0xf0); //--set divide ratio WriteCmd(0xd9); //--set pre-charge period WriteCmd(0x22); // WriteCmd(0xda); //--set com pins hardware configuration WriteCmd(0x12); WriteCmd(0xdb); //--set vcomh WriteCmd(0x20); //0x20,0.77xVcc WriteCmd(0x8d); //--set DC-DC enable WriteCmd(0x14); // WriteCmd(0xaf); //--turn on oled panel } void OLED_SetPos(unsigned char x, unsigned char y) //设置起始点坐标 { WriteCmd(0xb0+y); WriteCmd(((x&0xf0)>>4)|0x10); WriteCmd((x&0x0f)|0x01); } void OLED_Fill(unsigned char fill_Data)//全屏填充 { unsigned char m,n; for(m=0;m<8;m++) { WriteCmd(0xb0+m); //page0-page1 WriteCmd(0x00); //low column start address WriteCmd(0x10); //high column start address for(n=0;n<128;n++) { WriteDat(fill_Data); } } } void OLED_CLS(void)//清屏 { OLED_Fill(0x00); } //-------------------------------------------------------------- // Prototype : void OLED_ON(void) // Calls : // Parameters : none // Description : 将OLED从休眠中唤醒 //-------------------------------------------------------------- void OLED_ON(void) { WriteCmd(0X8D); //设置电荷泵 WriteCmd(0X14); //开启电荷泵 WriteCmd(0XAF); //OLED唤醒 } //-------------------------------------------------------------- // Prototype : void OLED_OFF(void) // Calls : // Parameters : none // Description : 让OLED休眠 -- 休眠模式下,OLED功耗不到10uA //-------------------------------------------------------------- void OLED_OFF(void) { WriteCmd(0X8D); //设置电荷泵 WriteCmd(0X10); //关闭电荷泵 WriteCmd(0XAE); //OLED休眠 } //在指定位置显示一个字符,包括部分字符 //x:0~127 //y:0~7 //chr:显示的字符 //TextSize:字符大小(1:6*8 ; 2:8*16) //mode:1,反白显示;0,正常显示 void OLED_ShowChar(unsigned char x,unsigned char y,unsigned char chr,unsigned char TextSize,u8 mode) { unsigned char c=0,i=0; c=chr-' ';//得到偏移后的值 if(TextSize == 2) { if(x>120){x=0;y++;} OLED_SetPos(x,y); for(i=0;i<8;i++) if(mode==1)WriteDat(~(F8X16[c*16+i]));else WriteDat(F8X16[c*16+i]); OLED_SetPos(x,y+1); for(i=0;i<8;i++) if(mode==1)WriteDat(~(F8X16[c*16+i+8]));else WriteDat(F8X16[c*16+i+8]); } else { if(x>126){x=0;y++;} OLED_SetPos(x,y); for(i=0;i<6;i++) if(mode==1)WriteDat(~(F6x8[c][i])); else WriteDat(F6x8[c][i]); } } //-------------------------------------------------------------- // Prototype : void OLED_ShowChar(unsigned char x, unsigned char y, unsigned char ch[], unsigned char TextSize) // Calls : // Parameters : x,y -- 起始点坐标(x:0~127, y:0~7); ch[] -- 要显示的字符串; TextSize -- 字符大小(1:6*8 ; 2:8*16) // Description : 显示codetab.h中的ASCII字符,有6*8和8*16可选择 //-------------------------------------------------------------- void OLED_ShowStr(unsigned char x, unsigned char y, unsigned char ch[], unsigned char TextSize,u8 mode) { unsigned char c = 0,i = 0,j = 0; switch(TextSize) { case 1: { while(ch[j] != '\0') { c = ch[j] - 32; if(x > 126) { x = 0; y++; } OLED_SetPos(x,y); for(i=0;i<6;i++) if(mode==1)WriteDat(~(F6x8[c][i])); else WriteDat(F6x8[c][i]); x += 6; j++; } }break; case 2: { while(ch[j] != '\0') { c = ch[j] - 32; if(x > 120) { x = 0; y++; } OLED_SetPos(x,y); for(i=0;i<8;i++) if(mode==1)WriteDat(~(F8X16[c*16+i])); else WriteDat(F8X16[c*16+i]); OLED_SetPos(x,y+1); for(i=0;i<8;i++) if(mode==1)WriteDat(~(F8X16[c*16+i+8])); else WriteDat(F8X16[c*16+i+8]); x += 8; j++; } }break; } } //-------------------------------------------------------------- // Prototype : void OLED_ShowCN(unsigned char x, unsigned char y, unsigned char N) // Calls : // Parameters : x,y -- 起始点坐标(x:0~127, y:0~7); N:汉字在codetab.h中的索引 // Description : 显示codetab.h中的汉字,16*16点阵 //mode:1,反白显示;0,正常显示 //-------------------------------------------------------------- void OLED_ShowCN(unsigned char x, unsigned char y, unsigned char N,u8 mode) { unsigned char wm=0; unsigned int adder=32*N; OLED_SetPos(x , y); for(wm = 0;wm < 16;wm++) { if(mode==1)WriteDat(~(F16x16[adder]));else WriteDat(F16x16[adder]); adder += 1; } OLED_SetPos(x,y + 1); for(wm = 0;wm < 16;wm++) { if(mode==1)WriteDat(~(F16x16[adder]));else WriteDat(F16x16[adder]); adder += 1; } } //-------------------------------------------------------------- // Prototype : void OLED_DrawBMP(unsigned char x0,unsigned char y0,unsigned char x1,unsigned char y1,unsigned char BMP[]); // Calls : // Parameters : x0,y0 -- 起始点坐标(x0:0~127, y0:0~7); x1,y1 -- 起点对角线(结束点)的坐标(x1:1~128,y1:1~8) // Description : 显示BMP位图 //-------------------------------------------------------------- void OLED_DrawBMP(unsigned char x0,unsigned char y0,unsigned char x1,unsigned char y1,unsigned char BMP[]) { unsigned int j=0; unsigned char x,y; if(y1%8==0) y = y1/8; else y = y1/8 + 1; for(y=y0;y<y1;y++) { OLED_SetPos(x0,y); for(x=x0;x<x1;x++) { WriteDat(BMP[j++]); } } } void OLED_ShowCentigrade(unsigned char x, unsigned char y)//显示℃ { unsigned char wm=0; unsigned char BUF[]={ 0x10,0x28,0x10,0xC0,0x20,0x10,0x10,0x10,0x20,0x70,0x00,0x00,0x00,0x00,0x00,0x07, 0x08,0x10,0x10,0x10,0x10,0x08,0x04,0x00,/*"℃"*/ }; OLED_SetPos(x , y); for(wm = 0;wm < 12;wm++) { WriteDat(BUF[wm]); } OLED_SetPos(x,y + 1); for(wm = 0;wm < 12;wm++) { WriteDat(BUF[wm+12]); } } #ifndef __OLED_I2C_H #define __OLED_I2C_H #include "stm32f10x.h" #define OLED_ADDRESS 0x78 //通过调整0R电阻,屏可以0x78和0x7A两个地址 -- 默认0x78 #define I2C_WR 0 /* 写控制bit */ #define I2C_RD 1 /* 读控制bit */ void bsp_InitI2C_2(void); void i2c_Start_2(void); void i2c_Stop_2(void); void i2c_SendByte_2(uint8_t _ucByte); uint8_t i2c_ReadByte_2(void); uint8_t i2c_WaitAck_2(void); void i2c_Ack_2(void); void i2c_NAck_2(void); void I2C_Configuration(void); void I2C_WriteByte(uint8_t addr,uint8_t data); void WriteCmd(unsigned char I2C_Command); void WriteDat(unsigned char I2C_Data); void OLED_Init(void); void OLED_SetPos(unsigned char x, unsigned char y); void OLED_Fill(unsigned char fill_Data); void OLED_CLS(void); void OLED_ON(void); void OLED_OFF(void); void OLED_ShowChar(unsigned char x,unsigned char y,unsigned char chr,unsigned char TextSize,u8 mode); void OLED_ShowStr(unsigned char x, unsigned char y, unsigned char ch[], unsigned char TextSize,u8 mode); void OLED_ShowCN(unsigned char x, unsigned char y, unsigned char N,u8 mode); void OLED_DrawBMP(unsigned char x0,unsigned char y0,unsigned char x1,unsigned char y1,unsigned char BMP[]); void OLED_ShowCentigrade(unsigned char x, unsigned char y); #endif TIME定时器程序 #include "timer.h" //通用定时器中断初始化 //这里时钟选择为APB1的2倍,而APB1为36M //arr:自动重装值。 //psc:时钟预分频数 //这里使用的是定时器3! void TIM3_Init(u16 arr,u16 psc) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; NVIC_InitTypeDef NVIC_InitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟使能 TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值 计数到5000为500ms TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值 10Khz的计数频率 TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式 TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位 TIM_ITConfig( //使能或者失能指定的TIM中断 TIM3, //TIM3 TIM_IT_Update , ENABLE //使能 ); NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //TIM3中断 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; //先占优先级0级 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; //从优先级2级 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能 NVIC_Init(&NVIC_InitStructure); //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器 TIM_Cmd(TIM3, ENABLE); //使能定时器3 } #ifndef __TIMER_H #define __TIMER_H #include "sys.h" void TIM3_Init(u16 arr,u16 psc); #endif DS18B20温度程序 #include "ds18b20.h" #include "delay.h" /******************************************************************************* 函数名:DS18B20_GPIO_Init 功能:初始化DS18B20引脚 输入: 输出: 返回值: 备注: *******************************************************************************/ void DS18B20_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_DS18B20_PORT, ENABLE); //使能PORTA口时钟 GPIO_InitStructure.GPIO_Pin = DS18B20_GPIO_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(DS18B20_GPIO_PORT, &GPIO_InitStructure); GPIO_SetBits(DS18B20_GPIO_PORT,DS18B20_GPIO_PIN); //输出1 } /******************************************************************************* 函数名:DS18B20_Init 功能:初始化DS18B20 输入: 输出: 返回值:初始化成功为0,不成功为1 备注: *******************************************************************************/ u8 DS18B20_Init(void) { unsigned char wait=0; DS18B20_IO_OUT(); //输出模式 DS18B20_OUT_0; //拉低 delay_us(750); //至少延时480us DS18B20_OUT_1; //拉高 delay_us(15); //15us DS18B20_IO_IN(); //输入模式 while(READ_DS18B20_IO && wait++<200)delay_us(1);//等待高电平结束 if(wait>=200)return 1; else wait=0; while(!READ_DS18B20_IO && wait++<240)delay_us(1);//等待低电平结束 if(wait>=240)return 1; else return 0; } /******************************************************************************* 函数名:DS18B20_ReadByte 功能:从DS18B20读一个字节 输入: 输出: 返回值:读取到的字节 备注: *******************************************************************************/ unsigned char DS18B20_ReadByte(void) { unsigned char i; unsigned char dat = 0; for (i=0; i<8; i++) //8位计数器 { dat >>= 1; DS18B20_IO_OUT(); //输出模式 DS18B20_OUT_0; //开始时间片 delay_us(2); //延时等待 DS18B20_OUT_1; //准备接收 DS18B20_IO_IN(); //输入模式 delay_us(12); //接收延时 if(READ_DS18B20_IO) dat |= 0x80; //读取数据 delay_us(60); //等待时间片结束 } return dat; } /******************************************************************************* 函数名:DS18B20_WriteByte 功能:写一个字节 输入:unsigned char dat 输出: 返回值: 备注: *******************************************************************************/ void DS18B20_WriteByte(unsigned char dat) { unsigned char i; unsigned char temp; DS18B20_IO_OUT();//输出模式 for (i=1; i<=8; i++) { temp = dat & 0x01; dat = dat >> 1; if (temp) { DS18B20_OUT_0; delay_us(2); DS18B20_OUT_1; //写1 delay_us(60); } else { DS18B20_OUT_0;//写0 delay_us(60); DS18B20_OUT_1; delay_us(2); } } } /************************************** 从DS18B20中获取温度值得浮点值 参数: 空 返回值: 读取到的温度值(有效范围-55.0~125.0) **************************************/ float ReadTemperature(void) { unsigned char TPH; //存放温度值的高字节 unsigned char TPL; //存放温度值的低字节 short i16=0; float f32=0; DS18B20_Init(); DS18B20_WriteByte(0xCC); //跳过ROM命令 DS18B20_WriteByte(0x44); //开始转换命令 DS18B20_Init(); DS18B20_WriteByte(0xCC); //跳过ROM命令 DS18B20_WriteByte(0xBE); //读暂存存储器命令 TPL = DS18B20_ReadByte(); //读温度低字节 TPH = DS18B20_ReadByte(); //读温度高字节 i16 = 0; i16 = (TPH<<8) |TPL; // 将高位(MSB)与低位(LSB)合并 f32 = i16 * 0.0625; // 12bit精度时温度值计算 return(f32); // 返回读取到的温度数值(float型) } #ifndef __DS18B20_H #define __DS18B20_H #include "sys.h" //如果想要修改引脚,只需修改下面的宏 #define RCC_DS18B20_PORT RCC_APB2Periph_GPIOA /* GPIO端口时钟 */ #define DS18B20_GPIO_PIN GPIO_Pin_11 #define DS18B20_GPIO_PORT GPIOA //IO方向设置(CRL寄存器对应引脚0~7,CRH寄存器对应引脚8~15) //DS18B20_GPIO_PORT->CRH&=0xFFFFFFF0为PA8引脚输出模式对应的寄存器清空 //DS18B20_GPIO_PORT->CRH|=0x00000008将(CNF8[1:0]设置为10:上拉/下拉输入模式,MODE8[1;0]设置为00:输入模式) //DS18B20_GPIO_PORT->CRH|=0x00000003将(CNF8[1:0]设置为00:通用推挽输出模式 ,MODE8[1;0]设置为11:最大50MHZ) #define DS18B20_IO_IN() {DS18B20_GPIO_PORT->CRH&=0xFFFF0FFF;DS18B20_GPIO_PORT->CRH|=0x00008000;} #define DS18B20_IO_OUT() {DS18B20_GPIO_PORT->CRH&=0xFFFF0FFF;DS18B20_GPIO_PORT->CRH|=0x00003000;} #define DS18B20_OUT_0 GPIO_ResetBits(DS18B20_GPIO_PORT, DS18B20_GPIO_PIN)//IO为低电平 #define DS18B20_OUT_1 GPIO_SetBits(DS18B20_GPIO_PORT, DS18B20_GPIO_PIN)//IO为高电平 #define READ_DS18B20_IO GPIO_ReadInputDataBit(DS18B20_GPIO_PORT, DS18B20_GPIO_PIN)//读取IO电平 void DS18B20_GPIO_Init(void); u8 DS18B20_Init(void); //初始化DS18B20 float ReadTemperature(void); //获取温度值 #endif 修改代码中使用OLED的部分,改为使用LCD1602实现

最新推荐

recommend-type

数据挖掘概述.ppt

数据挖掘概述.ppt
recommend-type

浅谈互联网+儿童文学阅读指导策略(1).docx

浅谈互联网+儿童文学阅读指导策略(1).docx
recommend-type

前端分析-202307110078988

前端分析-202307110078988
recommend-type

500强企业管理表格模板大全

在当今商业环境中,管理表格作为企业运营和管理的重要工具,是确保组织高效运作的关键。世界500强企业在管理层面的成功,很大程度上得益于它们的规范化和精细化管理。本文件介绍的“世界500强企业管理表格经典”,是一份集合了多种管理表格模板的资源,能够帮助管理者们更有效地进行企业规划、执行和监控。 首先,“管理表格”这个概念在企业中通常指的是用于记录、分析、决策和沟通的各种文档和图表。这些表格不仅仅局限于纸质形式,更多地是以电子形式存在,如Excel、Word、PDF等文件格式。它们帮助企业管理者收集和整理数据,以及可视化信息,从而做出更加精准的决策。管理表格可以应用于多个领域,例如人力资源管理、财务预算、项目管理、销售统计等。 标题中提及的“世界500强”,即指那些在全球范围内运营且在《财富》杂志每年公布的全球500强企业排行榜上出现的大型公司。这些企业通常具备较为成熟和先进的管理理念,其管理表格往往经过长时间的实践检验,并且能够有效地提高工作效率和决策质量。 描述中提到的“规范化”是企业管理中的一个核心概念。规范化指的是制定明确的标准和流程,以确保各项管理活动的一致性和可预测性。管理表格的使用能够帮助实现管理规范化,使得管理工作有据可依、有章可循,减少因个人经验和随意性带来的风险和不确定性。规范化管理不仅提高了企业的透明度,还有利于培养员工的规则意识,加强团队之间的协调与合作。 “经典”一词在这里强调的是,这些管理表格模板是经过实践验证,能够适用于大多数管理场景的基本模式。由于它们的普适性和高效性,这些表格模板被广泛应用于不同行业和不同规模的企业之中。一个典型的例子是SWOT分析表,它可以帮助企业识别内部的优势(Strengths)、弱点(Weaknesses)以及外部的机会(Opportunities)和威胁(Threats)。SWOT分析表就是一个在世界500强企业中普遍使用的管理表格。 标签中的“表格模板”则是对上述管理工具的具体描述。这些模板通常是预先设计好的,能够帮助企业管理者快速开始工作,无需从零开始制作新的表格。它们包含了一些必备的字段和格式,用户可以根据自己的具体需求对模板进行调整和填充。 文件名称列表中的“index.html”可能是压缩包内的一个网页文件,用于展示管理表格的索引或介绍。如果这是一个在线资源,它将允许用户通过网页界面访问和下载各种表格模板。而“menu”可能是一个导航文件,用来帮助用户在多个表格模板之间进行选择。“data”文件夹可能包含了实际的表格模板文件,它们可能以Excel、Word等格式存在。 总的来说,管理表格是企业成功管理不可或缺的工具。通过使用世界500强企业所采纳的管理表格模板,其他企业可以借鉴这些顶级企业的管理经验,帮助自己在管理实践中达到更高的效率和质量。通过规范化和模板化的管理表格,企业可以确保其管理活动的一致性和标准化,这对于保持竞争力和实现长期发展至关重要。
recommend-type

YOLOv8目标检测算法深度剖析:从零开始构建高效检测系统(10大秘诀)

# 1. YOLOv8目标检测算法概述 ## 1.1 YOLOv8的简介与定位 YOLOv8(You Only Look Once version 8)作为一种前沿的目标检测算法,是由YOLO系列算法演化而来。该算法特别强调快速与准确的平衡,它被设计用于实时图像识别
recommend-type

mclmcrrt9_8.dll下载

<think>我们正在处理用户关于"mclmcrrt9_8.dll"文件的下载请求。根据引用内容,这个文件是MATLAB运行时库的一部分,通常与特定版本的MATLABRuntime相关联。用户需求:下载mclmcrrt9_8.dll的官方版本。分析:1.根据引用[2]和[3],mclmcrrt9_0_1.dll和mclmcrrt9_13.dll都是MATLABRuntime的文件,版本号对应MATLAB的版本(如9_0对应R2016a,9_13对应2022b)。2.因此,mclmcrrt9_8.dll应该对应于某个特定版本的MATLAB(可能是R2016b?因为9.8版本通常对应MATLABR
recommend-type

林锐博士C++编程指南与心得:初学者快速提能

首先,这份文件的核心在于学习和提高C++编程能力,特别是针对初学者。在这个过程中,需要掌握的不仅仅是编程语法和基本结构,更多的是理解和运用这些知识来解决实际问题。下面将详细解释一些重要的知识点。 ### 1. 学习C++基础知识 - **基本数据类型**: 在C++中,需要熟悉整型、浮点型、字符型等数据类型,以及它们的使用和相互转换。 - **变量与常量**: 学习如何声明变量和常量,并理解它们在程序中的作用。 - **控制结构**: 包括条件语句(if-else)、循环语句(for、while、do-while),它们是构成程序逻辑的关键。 - **函数**: 理解函数定义、声明、调用和参数传递机制,是组织代码的重要手段。 - **数组和指针**: 学习如何使用数组存储数据,以及指针的声明、初始化和运算,这是C++中的高级话题。 ### 2. 林锐博士的《高质量的C++编程指南》 林锐博士的著作《高质量的C++编程指南》是C++学习者的重要参考资料。这本书主要覆盖了以下内容: - **编码规范**: 包括命名规则、注释习惯、文件结构等,这些都是编写可读性和可维护性代码的基础。 - **设计模式**: 在C++中合理使用设计模式可以提高代码的复用性和可维护性。 - **性能优化**: 学习如何编写效率更高、资源占用更少的代码。 - **错误处理**: 包括异常处理和错误检测机制,这对于提高程序的鲁棒性至关重要。 - **资源管理**: 学习如何在C++中管理资源,避免内存泄漏等常见错误。 ### 3. 答题与测试 - **C++C试题**: 通过阅读并回答相关试题,可以帮助读者巩固所学知识,并且学会如何将理论应用到实际问题中。 - **答案与评分标准**: 提供答案和评分标准,使读者能够自我评估学习成果,了解哪些方面需要进一步加强。 ### 4. 心得体会与实践 - **实践**: 理论知识需要通过大量编程实践来加深理解,动手编写代码,解决问题,是学习编程的重要方式。 - **阅读源码**: 阅读其他人的高质量代码,可以学习到许多编程技巧和最佳实践。 - **学习社区**: 参与C++相关社区,比如Stack Overflow、C++论坛等,可以帮助解答疑惑,交流心得。 ### 5. 拓展知识 - **C++标准库**: 学习C++标准模板库(STL),包括vector、map、list、algorithm等常用组件,是构建复杂数据结构和算法的基础。 - **面向对象编程**: C++是一种面向对象的编程语言,理解类、对象、继承、多态等概念对于写出优雅的C++代码至关重要。 - **跨平台编程**: 了解不同操作系统(如Windows、Linux)上的C++编程差异,学习如何编写跨平台的应用程序。 - **现代C++特性**: 学习C++11、C++14、C++17甚至C++20中的新特性,如智能指针、lambda表达式、自动类型推导等,可以提高开发效率和代码质量。 ### 总结 学习C++是一个系统工程,需要从基础语法开始,逐步深入到设计思想、性能优化、跨平台编程等领域。通过不断的学习和实践,初学者可以逐步成长为一个具有高代码质量意识的C++程序员。而通过阅读经典指南书籍,参与测试与评估,以及反思和总结实践经验,读者将更加扎实地掌握C++编程技术。此外,还需注意编程社区的交流和现代C++的发展趋势,这些都对于保持编程技能的前沿性和实用性是必不可少的。
recommend-type

线性代数方程组求解全攻略:直接法vs迭代法,一文搞懂

# 摘要 线性代数方程组求解是数学和工程领域中的基础而重要的问题。本文首先介绍了线性方程组求解的基础知识,然后详细阐述了直接法和迭代法两种主要的求解策略。直接法包括高斯消元法和LU分解方法,本文探讨了其理论基础、实践应用以及算法优化。迭代法则聚焦于雅可比和高斯-赛德尔方法,分析了其原理、实践应用和收敛性。通过比较分析,本文讨论了两种方法在
recommend-type

怎么下载mysql8.0.33版本

<think>我们正在处理用户关于下载并安装MySQL8.0.33版本的问题。根据引用[1]和引用[2],我们可以提供两种方法:一种是使用RPM包在CentOS上安装,另一种是使用ZIP包在Windows上安装。另外,引用[3]提到了安装过程中需要注意Python环境,但根据MySQL官方文档,Python并不是必须的,除非使用某些特定功能(如MySQLShell的Python模式)。因此,我们主要关注下载和安装步骤。用户没有明确操作系统,因此我们将分别介绍Windows和CentOS(Linux)下的安装方法。步骤概述:1.下载MySQL8.0.332.安装(根据系统不同步骤不同)3.初始化
recommend-type

C#学籍管理系统开发完成,信管专业的福音

标题中提到的“C#设计的学籍系统”涉及到几个重要的知识点。首先是“C#”,这是微软公司开发的一种面向对象的、运行在.NET框架上的高级编程语言。C#语言广泛用于开发Windows应用程序、游戏开发、分布式组件和客户端服务器应用程序等。在该标题中,它被用于构建一个学籍系统,这意味着系统的核心逻辑和功能是通过C#语言实现的。 其次是“学籍系统”,这通常是指用于管理学生个人信息、成绩、课程和学籍状态等数据的软件应用系统。学籍系统能够帮助教育机构高效地维护和更新学生档案,实现学生信息的电子化管理。它通常包括学生信息管理、成绩管理、课程安排、毕业资格审核等功能。 从描述中我们可以得知,这个学籍系统是“专门为信管打造”的。这里的“信管”很可能是对“信息管理”或者“信息系统管理”专业的简称。信息管理是一个跨学科领域,涉及信息技术在收集、存储、保护、处理、传输和安全地管理和开发信息资源方面的应用。这个系统可能是针对该专业学生的实际需求来定制开发的,包括一些特有的功能或者界面设计,以便更好地满足专业学习和实践操作的需要。 描述中还提到“请大家积极下载”,这可能意味着该学籍系统是一个开源项目,或者至少是一个允许公众访问的软件资源。由于开发者提出了“如有不足之处请大家多多包涵”,我们可以推断这个系统可能还处于测试或早期使用阶段,因此可能还不是完全成熟的版本,或者可能需要使用者反馈意见以便进行后续改进。 标签中的“C#的啊,大家注意,嘻嘻哈哈”表达了开发者轻松的态度和对C#语言的特定提及。这个标签可能是在一个非正式的交流环境中发布的,所以用词带有一定的随意性。尽管如此,它还是说明了该学籍系统是基于C#语言开发的,并提醒用户对这一点给予关注。 关于压缩包子文件的文件名称列表中,“学生成绩管理系统”直接指出了这个软件系统的主要功能之一,即管理学生的成绩。这通常包括录入成绩、查询成绩、统计分析成绩、成绩报告等功能。一个优秀的学生成绩管理系统可以让教师和学校管理人员更加高效地处理学生的成绩数据,同时也能让学生本人了解自己的学业进展。 综合以上信息,我们可以提炼出以下知识点: 1. C#语言:是一种面向对象的编程语言,适用于.NET框架,用于开发各种类型的应用程序。 2. 学籍系统:是管理学生基本信息、成绩、课程和学籍状态的软件应用系统,目的是实现学生信息的电子化管理。 3. 信息系统管理专业:该系统可能是针对信息系统管理专业的学生或教师的需求设计和开发的。 4. 开源项目或公众访问资源:鼓励用户下载使用,并接受用户的反馈和建议。 5. 学生成绩管理系统:是学籍系统的一个重要组成部分,专注于管理学生的成绩数据。 在开发一个C#设计的学籍系统时,开发者需要考虑的因素很多,比如系统的用户界面设计、数据库设计、数据安全、网络通信等。此外,系统还应该有良好的扩展性和易用性,以便未来可以根据用户反馈和新需求进行升级和优化。