RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOD|RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1|RCC_APB2Periph_AFIO, ENABLE);

时间: 2023-11-06 19:08:21 浏览: 293
这段代码是用于STM32的外设时钟使能的。其中,RCC_APB2PeriphClockCmd是一个函数,用于使能APB2总线上的外设时钟。在这个函数中,第一个参数是需要使能的外设,第二个参数是使能或禁用该外设。在这个例子中,GPIOB、GPIOD、GPIOA、USART1和AFIO这些外设的时钟被使能了。这些外设包括GPIO口、串口1和AFIO(用于重映射)等。
相关问题

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC|RCC_APB2Periph_GPIOD, ENABLE);

### 启用 STM32 GPIO 时钟 为了启用 STM32 中 GPIOA, GPIOB, GPIOC 和 GPIOD 的 APB2 时钟,可以使用 `RCC_APB2PeriphClockCmd` 函数。此函数用于使能或失能指定的 APB2 总线上的外设时钟。 #### 使用 `RCC_APB2PeriphClockCmd` 函数 该函数原型如下: ```c void RCC_APB2PeriphClockCmd(uint32_t RCC_APB2Periph, FunctionalState NewState); ``` 其中参数说明: - `RCC_APB2Periph`: 需要使能或失能的外设时钟。 - `NewState`: 设置为 `ENABLE` 或 `DISABLE` 来控制状态。 对于不同的 GPIO 端口,对应的宏定义分别为 `RCC_APB2Periph_GPIOA`, `RCC_APB2Periph_GPIOB`, `RCC_APB2Periph_GPIOC`, 和 `RCC_APB2Periph_GPIOD`. 具体实现代码如下所示: ```c #include "stm32f10x.h" int main(void) { /* Enable the peripheral clocks of GPIOA, GPIOB, GPIOC and GPIOD */ RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD, ENABLE); while (1) { // 主循环程序... } } ``` 上述代码片段展示了如何通过组合多个端口的宏来一次性启动所需的所有 GPIO 时钟[^1]。 此外,在遇到类似 “RCC_APB2Periph_GPIOC is undefined”的错误时,需确保项目中已正确定义并包含了标准外设驱动库的相关头文件,并且在预处理器设置里加入了 `USE_STDPERIPH_DRIVER` 宏定义[^3].

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOD, ENABLE); //使能PB,PE端口时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE); GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO口速度为50MHz GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8; GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12|GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15; GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化

这段代码是基于STM32微控制器的GPIO初始化代码,主要完成了以下几个步骤: 1. **使能时钟**:通过`RCC_APB2PeriphClockCmd`函数使能相关外设(如GPIOA、GPIOB、GPIOD和AFIO)的时钟。 2. **配置GPIO模式**:设置GPIO引脚为推挽输出模式,并设定最大输出速度为50MHz。 3. **初始化具体GPIO引脚**:分别对GPIOA的Pin8和GPIOB的Pin12、Pin13、Pin14、Pin15进行初始化。 以下是详细的代码解释和功能分析: --- ### 代码实现与解释 #### 1. 使能时钟 ```c RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOD, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE); ``` - `RCC_APB2PeriphClockCmd` 是一个库函数,用于控制APB2总线上外设的时钟。 - 参数1:指定需要使能时钟的外设。这里启用了GPIOA、GPIOB、GPIOD和AFIO的时钟。 - 参数2:`ENABLE` 表示开启时钟,`DISABLE` 表示关闭时钟。 - **作用**:在使用任何外设之前,必须先使能其时钟,否则外设无法工作。 --- #### 2. 配置GPIO结构体 ```c GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; // 推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; // IO口速度为50MHz ``` - `GPIO_InitStructure` 是一个结构体变量,用于存储GPIO的配置参数。 - `GPIO_Mode_Out_PP` 表示将GPIO配置为推挽输出模式。 - `GPIO_Speed_50MHz` 表示设置GPIO的最大输出速度为50MHz(适用于高速应用)。 --- #### 3. 初始化GPIOA的Pin8 ```c GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8; GPIO_Init(GPIOA, &GPIO_InitStructure); // 初始化 ``` - `GPIO_InitStructure.GPIO_Pin` 指定了要初始化的具体引脚,这里是GPIOA的Pin8。 - `GPIO_Init` 是一个库函数,用于根据`GPIO_InitStructure`中的配置参数初始化指定的GPIO端口。 --- #### 4. 初始化GPIOB的Pin12~Pin15 ```c GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15; GPIO_Init(GPIOB, &GPIO_InitStructure); // 初始化 ``` - 这里同时配置了GPIOB的Pin12、Pin13、Pin14和Pin15为推挽输出模式。 - 使用按位或运算符`|`可以一次性指定多个引脚。 --- ### 总结 上述代码完成的主要任务是: 1. 使能GPIOA、GPIOB、GPIOD和AFIO的时钟。 2. 配置GPIOA的Pin8和GPIOB的Pin12~Pin15为推挽输出模式,最大输出速度为50MHz。 3. 使用库函数`GPIO_Init`完成具体的GPIO初始化。 --- ### 示例扩展 如果需要将GPIOB的Pin12配置为输入模式,可以修改如下: ```c GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; // 浮空输入 GPIO_Init(GPIOB, &GPIO_InitStructure); ``` 如果需要读取GPIOB的Pin12状态,可以使用以下代码: ```c uint8_t pin_state = GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_12); if (pin_state == Bit_SET) { // Pin12为高电平 } else { // Pin12为低电平 } ``` --- ###
阅读全文

相关推荐

请帮我设计一些代码实现控制stm32小车直线行驶,我现在有motor模块代码如下#include "motor.h" #include "stm32f4xx.h" //pb7:3;pa4:5;pd7:6;pd6:4; void motor_init() { RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB | RCC_AHB1Periph_GPIOA | RCC_AHB1Periph_GPIOD,ENABLE); GPIO_InitTypeDef G; G.GPIO_Mode = GPIO_Mode_OUT; G.GPIO_OType = GPIO_OType_PP; G.GPIO_Pin = GPIO_Pin_7 | GPIO_Pin_4 | GPIO_Pin_6 | GPIO_Pin_5; G.GPIO_Speed = GPIO_Medium_Speed; GPIO_Init(GPIOB,&G); GPIO_Init(GPIOA,&G); GPIO_Init(GPIOD,&G); GPIO_SetBits(GPIOB,GPIO_Pin_7); GPIO_SetBits(GPIOA,GPIO_Pin_4); GPIO_SetBits(GPIOD,GPIO_Pin_6); GPIO_SetBits(GPIOD,GPIO_Pin_7); } void motor_up() { GPIO_SetBits(GPIOB,GPIO_Pin_7); GPIO_ResetBits(GPIOA,GPIO_Pin_4); GPIO_ResetBits(GPIOD,GPIO_Pin_6); GPIO_SetBits(GPIOD,GPIO_Pin_7); } void motor_right() { GPIO_SetBits(GPIOB,GPIO_Pin_7); GPIO_ResetBits(GPIOA,GPIO_Pin_4); GPIO_SetBits(GPIOD,GPIO_Pin_6); GPIO_ResetBits(GPIOD,GPIO_Pin_7); } void motor_left() { GPIO_ResetBits(GPIOB,GPIO_Pin_7); GPIO_SetBits(GPIOA,GPIO_Pin_4); GPIO_ResetBits(GPIOD,GPIO_Pin_6); GPIO_SetBits(GPIOD,GPIO_Pin_7); } void motor_down() { GPIO_ResetBits(GPIOB,GPIO_Pin_7); GPIO_SetBits(GPIOA,GPIO_Pin_4); GPIO_SetBits(GPIOD,GPIO_Pin_6); GPIO_ResetBits(GPIOD,GPIO_Pin_7); } void motor_stop(){ GPIO_SetBits(GPIOB,GPIO_Pin_7);//???? GPIO_SetBits(GPIOD,GPIO_Pin_6);//???? GPIO_SetBits(GPIOA,GPIO_Pin_4);//????? GPIO_SetBits(GPIOD,GPIO_Pin_7);//????? }请把接下来的控制小车直线行驶的模块写出来以及主模块main的代码发出来,.c.h代码都发出来

unsigned char Usart3RecBuf[USART3_RXBUFF_SIZE];//´®¿Ú2½ÓÊÕÊý¾Ý»º´æ unsigned int Rx3Counter = 0; //´®¿Ú2ÊÕµ½Êý¾Ý±ê־λ bool rev_start = 0; //½ÓÊÕ¿ªÊ¼±êÖ¾ bool rev_stop = 0; //½ÓÊÕÍ£Ö¹±êÖ¾ void USART3_Config(void){ //GPIO¶Ë¿ÚÉèÖà GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3, ENABLE); //ʹÄÜUSART3£¬GPIOAʱÖÓ RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //ʹÄÜGPIOBʱÖÓ //USART3_TX GPIOB.10 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; //PB10 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //¸´ÓÃÍÆÍìÊä³ö GPIO_Init(GPIOB, &GPIO_InitStructure);//³õʼ»¯GPIOB.10 //USART3_RX GPIOB.11³õʼ»¯ GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;//PB11 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//¸¡¿ÕÊäÈë GPIO_Init(GPIOB, &GPIO_InitStructure);//³õʼ»¯GPIOB.11 //Usart3 NVIC ÅäÖà NVIC_InitStructure.NVIC_IRQChannel = USART3_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0 ;//ÇÀÕ¼ÓÅÏȼ¶ NVIC_InitStructure.NVIC_IRQChannelSubPriority = 4; //×ÓÓÅÏȼ¶4 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQͨµÀʹÄÜ NVIC_Init(&NVIC_InitStructure); //¸ù¾ÝÖ¸¶¨µÄ²ÎÊý³õʼ»¯VIC¼Ä´æÆ÷ //USART ³õʼ»¯ÉèÖà USART_InitStructure.USART_BaudRate = 9600;//´®¿Ú²¨ÌØÂÊ USART_InitStructure.USART_WordLength = USART_WordLength_8b;//×Ö³¤Îª8λÊý¾Ý¸ñʽ 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(USART3, &USART_InitStructure); //³õʼ»¯´®¿Ú3 USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);//¿ªÆô´®¿Ú½ÓÊÜÖÐ¶Ï USART_Cmd(USART3, ENABLE); 把这些代码改为hal库可以使用的

根据以下程序,分别概括成几个模块的软件设计:#include "stm32f10x.h" #include "1602.h" #include "math.h" typedef unsigned char u8_t; typedef signed char s8_t; typedef unsigned short u16_t; typedef signed short s16_t; typedef unsigned long u32_t; typedef signed long s32_t; //位带操作,实现51类似的GPIO控制功能 //具体实现思想,参考<<CM3权威指南>>第五章(87页~92页). //IO口操作宏定义 #define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x2000000+((addr &0xFFFFF)<<5)+(bitnum<<2)) #define MEM_ADDR(addr) *((volatile unsigned long *)(addr)) #define BIT_ADDR(addr, bitnum) MEM_ADDR(BITBAND(addr, bitnum)) //IO口地址映射 #define GPIOA_ODR_Addr (GPIOA_BASE+12) //0x4001080C #define GPIOB_ODR_Addr (GPIOB_BASE+12) //0x40010C0C #define GPIOC_ODR_Addr (GPIOC_BASE+12) //0x4001100C #define GPIOD_ODR_Addr (GPIOD_BASE+12) //0x4001140C #define GPIOE_ODR_Addr (GPIOE_BASE+12) //0x4001180C #define GPIOF_ODR_Addr (GPIOF_BASE+12) //0x40011A0C #define GPIOG_ODR_Addr (GPIOG_BASE+12) //0x40011E0C #define GPIOA_IDR_Addr (GPIOA_BASE+8) //0x40010808 #define GPIOB_IDR_Addr (GPIOB_BASE+8) //0x40010C08 #define GPIOC_IDR_Addr (GPIOC_BASE+8) //0x40011008 #define GPIOD_IDR_Addr (GPIOD_BASE+8) //0x40011408 #define GPIOE_IDR_Addr (GPIOE_BASE+8) //0x40011808 #define GPIOF_IDR_Addr (GPIOF_BASE+8) //0x40011A08 #define GPIOG_IDR_Addr (GPIOG_BASE+8) //0x40011E08 //IO口操作,只对单一的IO口! //确保n的值小于16! #define PAout(n) BIT_ADDR(GPIOA_ODR_Addr,n) //输出 #define PAin(n) BIT_ADDR(GPIOA_IDR_Addr,n) //输入 #define PBout(n) BIT_ADDR(GPIOB_ODR_Addr,n) //输出 #define PBin(n) BIT_ADDR(GPIOB_IDR_Addr,n) //输入 #define PCout(n) BIT_ADDR(GPIOC_ODR_Addr,n) //输出 #define PCin(n) BIT_ADDR(GPIOC_IDR_Addr,n) //输入 #define PDout(n) BIT_ADDR(GPIOD_ODR_Addr,n) //输出 #define PDin(n) BIT_ADDR(GPIOD_IDR_Addr,n) //输入 #define PEout(n) BIT_ADDR(GPIOE_ODR_Addr,n) //输出 #define PEin(n) BIT_ADDR(GPIOE_IDR_Addr,n) //输入 #define PFout(n) BIT_ADDR(GPIOF_ODR_Addr,n) //输出 #define PFin(n) BIT_ADDR(GPIOF_IDR_Addr,n) //输入 #define PGout(n) BIT_ADDR(GPIOG_ODR_Addr,n) //输出 #define PGin(n) BIT_ADDR(GPIOG_IDR_Addr,n) //输入 //以下为汇编函数 void WFI_SET(void); //执行WFI指令 void INTX_DISABLE(void);//关闭所有中断 void INTX_ENABLE(void); //开启所有中断 void MSR_MSP(u32 addr); //设置堆栈地址 #define EnableINT() INTX_ENABLE() #define DisableINT() INTX_DISABLE() static u8 fac_us=0; //us延时倍乘数 static u16 fac_ms=0; //ms延时倍乘数,在ucos下,代表每个节拍的ms数 void delay_init() { #if SYSTEM_SUPPORT_OS //如果需要支持OS. u32 reload; #endif SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8); //选择外部时钟 HCLK/8 fac_us=SystemCoreClock/8000000; //为系统时钟的1/8 #if SYSTEM_SUPPORT_OS //如果需要支持OS. reload=SystemCoreClock/8000000; //每秒钟的计数次数 单位为M reload*=1000000/delay_ostickspersec; //根据delay_ostickspersec设定溢出时间 //reload为24位寄存器,最大值:16777216,在72M下,约合1.86s左右 fac_ms=1000/delay_ostickspersec; //代表OS可以延时的最少单位 SysTick->CTRL|=SysTick_CTRL_TICKINT_Msk; //开启SYSTICK中断 SysTick->LOAD=reload; //每1/delay_ostickspersec秒中断一次 SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk; //开启SYSTICK #else fac_ms=(u16)fac_us*1000; //非OS下,代表每个ms需要的systick时钟数 #endif } //延时nus //nus为要延时的us数. void delay_us(u32 nus) { u32 temp; SysTick->LOAD=nus*fac_us; //时间加载 SysTick->VAL=0x00; //清空计数器 SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ; //开始倒数 do { temp=SysTick->CTRL; }while((temp&0x01)&&!(temp&(1<<16))); //等待时间到达 SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk; //关闭计数器 SysTick->VAL =0X00; //清空计数器 } //延时nms //注意nms的范围 //SysTick->LOAD为24位寄存器,所以,最大延时为: //nms<=0xffffff*8*1000/SYSCLK //SYSCLK单位为Hz,nms单位为ms //对72M条件下,nms<=1864 void delay_ms(u16 nms) { u32 temp; SysTick->LOAD=(u32)nms*fac_ms; //时间加载(SysTick->LOAD为24bit) SysTick->VAL =0x00; //清空计数器 SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ; //开始倒数 do { temp=SysTick->CTRL; }while((temp&0x01)&&!(temp&(1<<16))); //等待时间到达 SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk; //关闭计数器 SysTick->VAL =0X00; //清空计数器 } void USART1_init(void) { USART_InitTypeDef USART_InitStructure; GPIO_InitTypeDef GPIO_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_AFIO , ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(GPIOA, &GPIO_InitStructure); USART_InitStructure.USART_BaudRate = 9600; 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_Tx | USART_Mode_Rx; //USART1 USART_Init(USART1, &USART_InitStructure); USART_Cmd(USART1, ENABLE); USART_ClearITPendingBit(USART1, USART_IT_RXNE); USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); USART_ITConfig(USART1, USART_IT_PE, ENABLE); USART_ITConfig(USART1, USART_IT_ERR, ENABLE); USART1->DR = (0X55 & (uint16_t)0x01FF); while((USART1->SR&0X40)==0); NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); } void UsartSendData1(unsigned char *send_buff,unsigned long int length) { unsigned long int i = 0; delay_ms(1); for(i = 0;i < length;i ++) { USART1->DR = (send_buff[i] & (uint16_t)0x01FF); while((USART1->SR&0X40)==0); } delay_ms(1); } int GQ=0;//保存光强 int LX=0;//光类型 0白光 1黄光 2混合光 int DW=0;//档位 0~5档 int LPWM=0;//亮度 0~100 int moshi=0;//0手动 1自动 2声控 #define LED_B PBout(8) #define LED_H PBout(7) #define C_YR PBin(6) void GPIO_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOC,ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_AFIO, ENABLE); //ENABLEPBCLK GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable , ENABLE); //Disable jtag RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE); //LCD1602 管脚 D0~D7 双向I/O GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15| GPIO_Pin_3| GPIO_Pin_4| GPIO_Pin_5; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; GPIO_Init(GPIOB, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8| GPIO_Pin_11| GPIO_Pin_12| GPIO_Pin_15; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; GPIO_Init(GPIOA, &GPIO_InitStructure); //ADC GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7|GPIO_Pin_0; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; GPIO_Init(GPIOA, &GPIO_InitStructure); //LCD1602 管脚 控制 推挽模式 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12 |GPIO_Pin_13|GPIO_Pin_14; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(GPIOB, &GPIO_InitStructure); //按键 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9|GPIO_Pin_10|GPIO_Pin_11; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; GPIO_Init(GPIOB, &GPIO_InitStructure); //LED灯 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7|GPIO_Pin_8; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(GPIOB, &GPIO_InitStructure); } /* ADC配置 */ void ADC_Set(unsigned char ch) { ADC_InitTypeDef ADC_InitStructure;//ADC结构体变量//注意在一个语句快内变量的声明要放在可执行语句的前面,否则出错,因此要放在ADC1_GPIO_Config();前面 ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//ADC1和ADC2工作在独立模式 ADC_InitStructure.ADC_ScanConvMode = DISABLE; //使能扫描 ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;//ADC转换工作在连续模式 ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//由软件控制转换,不使用外部触发 ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//转换数据右对齐 ADC_InitStructure.ADC_NbrOfChannel = 1;//转换通道为1 ADC_Init(ADC1, &ADC_InitStructure); //初始化ADC if(ch == ADC_Channel_7) { ADC_RegularChannelConfig(ADC1, ADC_Channel_7, 1, ADC_SampleTime_239Cycles5); } if(ch == ADC_Channel_6) { ADC_RegularChannelConfig(ADC1, ADC_Channel_6, 1, ADC_SampleTime_239Cycles5); } ADC_Cmd(ADC1, ENABLE);//使能ADC1 ADC_ITConfig(ADC1, ADC_IT_EOC, ENABLE); ADC_SoftwareStartConvCmd(ADC1, ENABLE); } #define ADC_TIMES 10 #define ADCLOST 2 unsigned int GetADC(unsigned char ch) { unsigned int i, j; unsigned int buf[ADC_TIMES]; unsigned long int sum; long int temp; static long int XQ = 0; ADC_Set(ch); for(i=0; i < ADC_TIMES; i++) { buf[i] = ADC_GetConversionValue(ADC1);; delay_ms(10); } for(i = 0; i < ADC_TIMES - 1; i++) { for(j = i + 1; j < ADC_TIMES; j++) { if(buf[i] > buf[j]) { temp = buf[i]; } } } sum = 0; for(i = ADCLOST; i < ADC_TIMES - ADCLOST; i++) { sum += buf[i]; } temp = sum / (ADC_TIMES - 2 * ADCLOST); if( (XQ>=temp && (XQ-temp)<=1 ) || (XQ<temp && (temp-XQ)<=1)) { temp = XQ; } else { XQ = temp; } return (unsigned int)(temp>>4); } void Display_1(int T1,int T2,int T3,int T4,int T5) { char xx=0,yy=0; //光强 WrByte1602(yy,xx++,'G'); //. WrByte1602(yy,xx++,'Q'); WrByte1602(yy,xx++,'='); WrByte1602(yy,xx++,T1%1000/100+'0'); WrByte1602(yy,xx++,T1%100/10+'0'); WrByte1602(yy,xx++,T1%10+'0'); WrByte1602(yy,xx++,' '); //光类型 WrByte1602(yy,xx++,'T'); //. WrByte1602(yy,xx++,'Y'); WrByte1602(yy,xx++,'P'); WrByte1602(yy,xx++,'E'); WrByte1602(yy,xx++,'-'); if(T2==0)//0混合 1白光 2黄光 { WrByte1602(yy,xx++,'B'); WrByte1602(yy,xx++,'H'); } else if(T2==1)//0混合 1白光 2黄光 { WrByte1602(yy,xx++,'B'); WrByte1602(yy,xx++,' '); } else if(T2==2)//0混合 1白光 2黄光 { WrByte1602(yy,xx++,'H'); WrByte1602(yy,xx++,' '); } WrByte1602(yy,xx++,' '); xx=0,yy=1; //档位 WrByte1602(yy,xx++,'D'); //. WrByte1602(yy,xx++,'W'); WrByte1602(yy,xx++,'='); WrByte1602(yy,xx++,T3%10+'0'); WrByte1602(yy,xx++,' '); //亮度 WrByte1602(yy,xx++,'L'); //. WrByte1602(yy,xx++,'D'); WrByte1602(yy,xx++,'='); WrByte1602(yy,xx++,T4%1000/100+'0'); WrByte1602 if(T5==0)//0手动 1自动 2声控 { WrByte1602(yy,xx++,'S'); WrByte1602(yy,xx++,'D'); } else if(T5==1)//0手动 1自动 2声控 { WrByte1602(yy,xx++,'Z'); WrByte1602(yy,xx++,'D'); } else if(T5==2)//0手动 1自动 2声控 { WrByte1602(yy,xx++,'S'); WrByte1602(yy,xx++,'K'); } WrByte1602(yy,xx++,' '); } //int GQ=0;//保存光强 //int LX=0;//光类型 0白光 1黄光 2混合光 //int DW=0;//档位 0~5档 //int LPWM=0;//亮度 0~100 //int moshi=0;//0手动 1自动 2声控 void anjiansaom(void) { static uchar K1,K2,K3; static uchar K1Flag=0,K2Flag=0,K3Flag=0; K1 = GPIO_ReadInputDataBit(GPIOB , GPIO_Pin_9); K2 = GPIO_ReadInputDataBit(GPIOB , GPIO_Pin_11); K3 = GPIO_ReadInputDataBit(GPIOB , GPIO_Pin_10); //手动/自动 if(K1==0) { K1Flag = 1; } else { if(K1Flag) { K1Flag = 0; if(moshi==1||moshi==2)//0手动 1自动 2声控 { moshi=0; } else { moshi=1; } } } //亮度 if(K2==0) { K2Flag = 1; } else { if(K2Flag) { K2Flag = 0; if(moshi==0)//0手动 1自动 2声控 { DW++; if(DW>5) { DW=0; } LPWM=DW*20; } } } { if(K3Flag) { K3Flag = 0; //光类型 0白光 1黄光 2混合光 LX++; if(LX>2) { LX=0; } } } } void TimerConfig(TIM_TypeDef* TIMx,unsigned long int time) { NVIC_InitTypeDef NVIC_InitStructure; TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; RCC_APB1PeriphClockCmd( RCC_APB1Periph_TIM2,ENABLE); RCC_APB1PeriphClockCmd( RCC_APB1Periph_TIM3,ENABLE); TIM_DeInit(TIMx); TIM_TimeBaseStructure.TIM_Period = time * 2; /* 自动重装载寄存器周期的值(计数值) */ /* 累计 TIM_Period个频率后产生一个更新或者中断 */ TIM_TimeBaseStructure.TIM_Prescaler= (1000- 1); /* 时钟预分频数 例如:时钟频率=72MHZ/(时钟预分频+1) */ TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1; /* 采样分频 */ TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up; /* 向上计数模式 */ TIM_TimeBaseInit(TIMx, &TIM_TimeBaseStructure); TIM_ClearFlag(TIMx, TIM_FLAG_Update); /* 清除溢出中断标志 */ TIM_ITConfig(TIMx,TIM_IT_Update,ENABLE); NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); NVIC_InitStr NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); } /** * @brief 关闭或者开启指定定时器 * @param TIMx: where x can be 1 to 17 to select the TIM peripheral * @param NewState: new state of the TIMx peripheral. * This parameter can be: ENABLE or DISABLE. * @retval None */ void TimerSW( TIM_TypeDef* TIMx, FunctionalState NewState ) { TIM_SetCounter(TIMx,0); TIM_Cmd(TIMx,NewState); } int main(void) { float temp; SystemInit(); SystemCoreClockUpdate(); delay_init(); //引脚配置 GPIO_Configuration(); //关闭LED灯 LED_B=0; LED_H=0; delay_ms(1000); //初始化LCD1602 Init1602(); TimerConfig(TIM2,1); TimerSW(TIM2, ENABLE); //串口初始化 USART1_init(); while (1) { delay_ms(20); //读取ADC的值,然后计算光强 ADC_Set(0); //光强 //先读取ADC值 GQ =(255-GetADC(0))/2-60; if(GQ<0) { GQ=0; } GQ = GQ*2; if(GQ>100) { GQ = 100; } //按键扫描 anjiansaom(); //光强 光类型 档位 亮度 模式 Display_1(GQ,LX,DW,LPWM,moshi); //自动模式光强控制 //首先得有人才会亮 if(moshi==1) { if(C_YR==1 )//传感器输出高电平代表有人 { //根据亮度计算光亮 //越暗 越亮 LPWM=100-GQ; } else { LPWM=0;//关灯 } } } } /** * @brief 定时器2中断服务函数 * @param TIMx: where x can be 1 to 17 to select the TIM peripheral * @param NewState: new state of the TIMx peripheral. * This parameter can be: ENABLE or DISABLE. * @retval None */ void TIM2_IRQHandler(void) { static int i=0; if ( TIM_GetITStatus(TIM2 , TIM_IT_Update) != RESET ) { TIM_ClearITPendingBit(TIM2 , TIM_FLAG_Update); // PCout(13)=~PCout(13); //这里实现PWM 波形的产生 i++; if(i<=LPWM) { //开灯 //暖光 还是白光 还是混合 if(LX==0)//0混合 1白光 2黄光 { } } else { //关灯 LED_B=0;LED_H=0; } //不能超过100 if(i>100) { i=0; } } } //串口1接收中断 void USART1_IRQHandler(void) { char shuju=0; if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { USART_ClearITPendingBit(USART1, USART_IT_RXNE); //保存收到的数据 shuju=USART_ReceiveData(USART1); //读取接收到的数据 //收到数据1 //你好小灯 唤醒 if(shuju==1)//进入声控模式 { moshi=2; } if(moshi==2) { //收到数据2 //开灯 if(shuju==2)//开灯 { if(LPWM==0) { DW=3; LPWM=DW*20; } } else if(shuju==3)//关灯 { DW=0; LPWM=DW*20; } else if(shuju==4)//亮一点 { DW++; if(DW>5) } } } if(USART_GetITStatus(USART1, USART_IT_PE) != RESET) { USART_ClearITPendingBit(USART1, USART_IT_PE); } if (USART_GetFlagStatus(USART1, USART_IT_LBD) != RESET) { USART_ClearITPendingBit(USART1, USART_IT_LBD); } if(USART_GetFlagStatus(USART1, USART_FLAG_ORE) != RESET) { USART_ReceiveData(USART1); USART_ClearFlag(USART1, USART_FLAG_ORE); } if(USART_GetFlagStatus(USART1, USART_FLAG_NE) != RESET) { USART_ClearFlag(USART1, USART_FLAG_NE); } if(USART_GetFlagStatus(USART1, USART_FLAG_FE) != RESET) { USART_ClearFlag(USART1, USART_FLAG_FE); } if(USART_GetFlagStatus(USART1, USART_FLAG_PE) != RESET) { USART_ClearFlag(USART1, USART_FLAG_PE); } if (USART_GetITStatus(USART1, USART_IT_TC) != RESET) { USART_ClearITPendingBit(USART1, USART_IT_TC); } } #ifdef USE_FULL_ASSERT /** * @brief Reports the name of the source file and the source line number * where the assert_param error has occurred. * @param file: pointer to the source file name * @param line: assert_param error line source number * @retval None */ void assert_failed(uint8_t* file, uint32_t line) { /* User can add his own implementation to report the file name and line number, ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */ /* Infinite loop */ while (1) {} } #endif /******************* (C) COPYRIGHT 2009 STMicroelectronics *****END OF FILE****/ #include "stm32f10x.h" #include "1602.h" #define uchar unsigned char #define uint unsigned int uchar ASCII[]={0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39};//2 /* PA8~PA15为数据口 */ #define LCD1602_RS_0 GPIO_ResetBits(GPIOB , GPIO_Pin_12) #define LCD1602_RS_1 GPIO_SetBits(GPIOB , GPIO_Pin_12) #define LCD1602_RW_0 GPIO_ResetBits(GPIOB , GPIO_Pin_13) #define LCD1602_RW_1 GPIO_SetBits(GPIOB , GPIO_Pin_13) #define LCD1602_EN_0 GPIO_ResetBits(GPIOB , GPIO_Pin_14) #define LCD1602_EN_1 GPIO_SetBits(GPIOB , GPIO_Pin_14) #define LCD1602_D0_0 GPIO_ResetBits(GPIOB , GPIO_Pin_15) #define LCD1602_D0_1 GPIO_SetBits(GPIOB , GPIO_Pin_15) #define LCD1602_D1_0 GPIO_ResetBits(GPIOA , GPIO_Pin_8) #define LCD1602_D1_1 GPIO_SetBits(GPIOA , GPIO_Pin_8) #define LCD1602_D2_0 GPIO_ResetBits(GPIOA , GPIO_Pin_11) #define LCD1602_D2_1 GPIO_SetBits(GPIOA , GPIO_Pin_11) #define LCD1602_D3_0 GPIO_ResetBits(GPIOA , GPIO_Pin_12) #define LCD1602_D3_1 GPIO_SetBits(GPIOA , GPIO_Pin_12) #define LCD1602_D4_0 GPIO_ResetBits(GPIOA , GPIO_Pin_15) #define LCD1602_D4_1 GPIO_SetBits(G #define LCD1602_D6_0 GPIO_ResetBits(GPIOB , GPIO_Pin_4) #define LCD1602_D6_1 GPIO_SetBits(GPIOB , GPIO_Pin_4) #define LCD1602_D7_0 GPIO_ResetBits(GPIOB , GPIO_Pin_5) #define LCD1602_D7_1 GPIO_SetBits(GPIOB , GPIO_Pin_5) #define GET_LCD1602_D0 GPIO_ReadInputDataBit(GPIOB , GPIO_Pin_15) #define GET_LCD1602_D1 GPIO_ReadInputDataBit(GPIOA , GPIO_Pin_8) #define GET_LCD1602_D2 GPIO_ReadInputDataBit(GPIOA , GPIO_Pin_11) #define GET_LCD1602_D3 GPIO_ReadInputDataBit(GPIOA , GPIO_Pin_12) #define GET_LCD1602_D4 GPIO_ReadInputDataBit(GPIOA , GPIO_Pin_15) #define GET_LCD1602_ void SetLCD1602DValue(unsigned char value) { if(value&0x01) { LCD1602_D0_1; } else { LCD1602_D0_0; } if(value&0x02) { LCD1602_D1_1; } { LCD1602_D2_1; } else { LCD1602_D2_0; } if(value&0x08) { LCD1602_D3_1; } else { LCD1602_D3_0; } if(value&0x10) { LCD1602_D4_1; } else { LCD1602_D4_0; } if(value&0x20) { LCD1602_D5_1; } else { LCD1602_D5_0; } if(value&0x40) if(value&0x80) { LCD1602_D7_1; } else { LCD1602_D7_0; } } unsigned char GetLCD1602DValue(void) { unsigned char dat; dat = 0x00; if(GET_LCD1602_D0) { dat = dat |0x01; } if(GET_LCD1602_D1) { dat = dat |0x02; } if(GET_LCD1602_D2) { dat = dat |0x04; } if(GET_LCD1602_D3) { dat = dat |0x08; } if(GET_LCD1602_D4) { dat = dat |0x10; } if(GET_LCD1602_D5) { dat = dat |0x20; } if(GET_LCD1602_D6) { dat = dat |0x40; } if(GET_LCD1602_D7) { dat = dat |0x80; } return(dat); } /*******************通用延时子程序**************************/ void mDelay(uint i) { for (;i>0;i--) {uchar j=10;while(--j);} } void LCD1602DTest(void) { LCD1602_RS_1; LCD1602_RW_1; LCD1602_EN_1; SetLCD1602DValue(0x00); mDelay(100); mDelay(100); LCD1602_EN_0; SetLCD1602DValue(0xFF); mDelay(100); mDelay(100); LCD1602_RS_0; LCD1602_RW_0; LCD1602_EN_0; SetLCD1602DValue(0x00); } /************************************************************* 函数名称:读忙状态子函数 全局变量:无 参数说明:无 返回说明:无 版 本:1.0 说 明:读取LCD1602忙状态 **************************************************************/ void Read1602() //读忙状态 { long int i=90000; //建一个循环变量避免器件发生故障停在这里 SetLCD1602DValue(0xff); LCD1602_RS_0; LCD1602_RW_1; //设置LCD为读取数据状态 LCD1602_EN_1; //使能LCD,高电平 while ((i--)&&(GetLCD1602DValue()&0x80)); //检测数据口最高位状态,为0则空闲 LCD1602_EN_0; //关闭使能 } /************************************************************** 函数名称:写操作子函数 全局变量:无 参数说明:Dat为数据,command为指令(1为写数据,0为写指令) 返回说明:无 版 本:1.0 说 明:往LCD1602写入数据、指令 ***************************************************************/ void Write1602(uchar Dat,unsigned char command) { Read1602(); SetLCD1602DValue(Dat); if(command) { LCD1602_RS_1; //RS为1写数据、为0写指令 } else { LCD1602_RS_0; } LCD1602_RW_0; //RW为低,进行写操作 LCD1602_EN_1; LCD1602_EN_0; //E端控制一个高脉冲 } /*************************************************************** 函数名称:LCD1602初始化子函数 全局变量:无 参数说明:无 返回说明:无 版 本:1.0 说 明:设置工作模式、清屏、开显示 ***************************************************************/ void Init1602(void) { LCD1602DTest(); mDelay(10); mDelay(10); Write1602(0x38,0); //8位点阵方式 mDelay(10); Write1602(0x38,0); mDelay(10); Write1602(0x38,0); mDelay(10); Write1602(0x38,0); mDelay(10); Write1602(0x01,0); //清屏 mDelay(10); Write1602(0x0c,0); //开显示,光标不显示 } void WrByte1602_EX(uchar x,uchar y,uchar dat) { Write1602(0x80+(x<<6)+y,0); //合并生成地址 Write1602(dat,1); //写入数据 } void WrByte1602(uchar x,uchar y,uchar dat) { Write1602(0x80+(x<<6)+y,0); //合并生成地址 Write1602(dat,1); //写入数据 } /*************************************************************** 函数名称:LCD1602整行字符串写入子函数 全局变量:无 参数说明:x为写入的行(0或1),*p为写入的字符串数组 返回说明:无 版 本:1.0 说 明:在LCD1602任意行写入字符串 ****************************************************************/ void LCD_Write_String(uchar x1,uchar x,uchar *p) { x1 = 1; Write1602(0x80+(x<<6),0); //合并生成行首地址 while(*p) {Write1602(*p,1);p++;} //逐个字符写入 } #include "stm32f10x.h" #include "1602.h" #define uchar unsigned char #define uint unsigned int uchar ASCII[]={0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39};//2 /* PA8~PA15为数据口 */ #define LCD1602_RS_0 GPIO_ResetBits(GPIOB , GPIO_Pin_12) #define LCD1602_RS_1 GPIO_SetBits(GPIOB , GPIO_Pin_12) #define LCD1602_RW_0 GPIO_ResetBits(GPIOB , GPIO_Pin_13) #define LCD1602_RW_1 GPIO_SetBits(GPIOB , GPIO_Pin_13) #define LCD1602_EN_0 GPIO_ResetBits(GPIOB , GPIO_Pin_14) #define LCD1602_EN_1 GPIO_SetBits(GPIOB , GPIO_Pin_14) #define LCD1602_D0_0 GPIO_ResetBits(GPIOB , GPIO_Pin_15) #define LCD1602_D0_1 GPIO_SetBits(GPIOB , GPIO_Pin_15) #define LCD1602_D1_0 GPIO_ResetBits(GPIOA , GPIO_Pin_8) #define LCD1602_D1_1 GPIO_SetBits(GPIOA , GPIO_Pin_8) #define LCD1602_D2_0 GPIO_ResetBits(GPIOA , GPIO_Pin_11) #define LCD1602_D2_1 GPIO_SetBits(GPIOA , GPIO_Pin_11) #define LCD1602_D3_0 GPIO_ResetBits(GPIOA , GPIO_Pin_12) #define LCD1602_D3_1 GPIO_SetBits(GPIOA , GPIO_Pin_12) #define LCD1602_D4_0 GPIO_ResetBits(GPIOA , GPIO_Pin_15) #define LCD1602_D4_1 GPIO_SetBits(GPIOA , GPIO_Pin_15) #define LCD1602_D5_0 GPIO_ResetBits(GPIOB , GPIO_Pin_3) #define LCD1602_D5_1 GPIO_SetBits(GPIOB , GPIO_Pin_3) #define LCD1602_D6_0 GPIO_ResetBits(GPIOB , GPIO_Pin_4) #define LCD1602_D6_1 GPIO_SetBits(GPIOB , GPIO_Pin_4) #define LCD1602_D7_0 GPIO_ResetBits(GPIOB , GPIO_Pin_5) #define LCD1602_D7_1 GPIO_SetBits(GPIOB , GPIO_Pin_5) #define GET_LCD1602_D0 GPIO_ReadInputDataBit(GPIOB , GPIO_Pin_15) #define GET_LCD1602_D1 GPIO_ReadInputDataBit(GPIOA , GPIO_Pin_8) #define GET_LCD1602_D2 GPIO_ReadInputDataBit(GPIOA , GPIO_Pin_11) #define GET_LCD1602_D3 GPIO_ReadInputDataBit(GPIOA , GPIO_Pin_12) #define GET_LCD1602_D4 GPIO_ReadInputDataBit(GPIOA , GPIO_Pin_15) #define GET_LCD1602_D5 GPIO_ReadInputDataBit(GPIOB , GPIO_Pin_3) #define GET_LCD1602_D6 GPIO_ReadInputDataBit(GPIOB , GPIO_Pin_4) #define GET_LCD1602_D7 GPIO_ReadInputDataBit(GPIOB , GPIO_Pin_5) void SetLCD1602DValue(unsigned char value) { if(value&0x01) { LCD1602_D0_1; } else { LCD1602_D0_0; } if(value&0x02) { LCD1602_D1_1; } else { LCD1602_D1_0; } if(value&0x04) { LCD1602_D2_1; } else { LCD1602_D2_0; } if(value&0x08) { LCD1602_D3_1; } else { LCD1602_D3_0; } if(value&0x10) { LCD1602_D4_1; } else { LCD1602_D4_0; } if(value&0x20) { LCD1602_D5_1; } else { LCD1602_D5_0; } if(value&0x40) { LCD1602_D6_1; } else { LCD1602_D6_0; } if(value&0x80) { LCD1602_D7_1; } else { LCD1602_D7_0; } } unsigned char GetLCD1602DValue(void) { unsigned char dat; dat = 0x00; if(GET_LCD1602_D0) { dat = dat |0x01; } if(GET_LCD1602_D1) { dat = dat |0x02; } if(GET_LCD1602_D2) { dat = dat |0x04; } if(GET_LCD1602_D3) { dat = dat |0x08; } if(GET_LCD1602_D4) { dat = dat |0x10; } if(GET_LCD1602_D5) { dat = dat |0x20; } if(GET_LCD1602_D6) { dat = dat |0x40; } if(GET_LCD1602_D7) { dat = dat |0x80; } return(dat); } /*******************通用延时子程序**************************/ void mDelay(uint i) { for (;i>0;i--) {uchar j=10;while(--j);} } void LCD1602DTest(void) { LCD1602_RS_1; LCD1602_RW_1; LCD1602_EN_1; SetLCD1602DValue(0x00); mDelay(100); mDelay(100); LCD1602_RS_0; LCD1602_RW_0; LCD1602_EN_0; SetLCD1602DValue(0xFF); mDelay(100); mDelay(100); LCD1602_RS_0; LCD1602_RW_0; LCD1602_EN_0; SetLCD1602DValue(0x00); } /************************************************************* 函数名称:读忙状态子函数 全局变量:无 参数说明:无 返回说明:无 版 本:1.0 说 明:读取LCD1602忙状态 **************************************************************/ void Read1602() //读忙状态 { long int i=90000; //建一个循环变量避免器件发生故障停在这里 SetLCD1602DValue(0xff); LCD1602_RS_0; LCD1602_RW_1; //设置LCD为读取数据状态 LCD1602_EN_1; //使能LCD,高电平 while ((i--)&&(GetLCD1602DValue()&0x80)); //检测数据口最高位状态,为0则空闲 LCD1602_EN_0; //关闭使能 } /************************************************************** 函数名称:写操作子函数 全局变量:无 参数说明:Dat为数据,command为指令(1为写数据,0为写指令) 返回说明:无 版 本:1.0 说 明:往LCD1602写入数据、指令 ***************************************************************/ void Write1602(uchar Dat,unsigned char command) { Read1602(); SetLCD1602DValue(Dat); if(command) { LCD1602_RS_1; //RS为1写数据、为0写指令 } else { LCD1602_RS_0; } LCD1602_RW_0; //RW为低,进行写操作 LCD1602_EN_1; LCD1602_EN_0; //E端控制一个高脉冲 } /*************************************************************** 函数名称:LCD1602初始化子函数 全局变量:无 参数说明:无 返回说明:无 版 本:1.0 说 明:设置工作模式、清屏、开显示 ***************************************************************/ void Init1602(void) { LCD1602DTest(); mDelay(10); mDelay(10); Write1602(0x38,0); //8位点阵方式 mDelay(10); Write1602(0x38,0); mDelay(10); mDelay(10); Write1602(0x01,0); //清屏 mDelay(10); Write1602(0x0c,0); //开显示,光标不显示 } void WrByte1602_EX(uchar x,uchar y,uchar dat) { Write1602(0x80+(x<<6)+y,0); //合并生成地址 Write1602(dat,1); //写入数据 } void WrByte1602(uchar x,uchar y,uchar dat) { Write1602(0x80+(x<<6)+y,0); //合并生成地址 Write1602(dat,1); //写入数据 } /*************************************************************** 函数名称:LCD1602整行字符串写入子函数 全局变量:无 参数说明:x为写入的行(0或1),*p为写入的字符串数组 返回说明:无 版 本:1.0 说 明:在LCD1602任意行写入字符串 ****************************************************************/ void LCD_Write_String(uchar x1,uchar x,uchar *p) { x1 = 1; Write1602(0x80+(x<<6),0); //合并生成行首地址 while(*p) {Write1602(*p,1);p++;} //逐个字符写入 }

找不到错在哪,但是温度就是不变 /* USER CODE BEGIN Header */ /** ****************************************************************************** * @file : main.c * @brief : Main program body ****************************************************************************** * @attention * * Copyright (c) 2025 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file * in the root directory of this software component. * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** */ /* USER CODE END Header */ /* Includes ------------------------------------------------------------------*/ #include "main.h" /* Private includes ----------------------------------------------------------*/ /* USER CODE BEGIN Includes */ #include "LCD.h" #include "DS18B20.h" /* USER CODE END Includes */ /* Private typedef -----------------------------------------------------------*/ /* USER CODE BEGIN PTD */ /* USER CODE END PTD */ /* Private define ------------------------------------------------------------*/ /* USER CODE BEGIN PD */ /* USER CODE END PD */ /* Private macro -------------------------------------------------------------*/ /* USER CODE BEGIN PM */ /* USER CODE END PM */ /* Private variables ---------------------------------------------------------*/ /* USER CODE BEGIN PV */ /* USER CODE END PV */ /* Private function prototypes -----------------------------------------------*/ void SystemClock_Config(void); static void MX_GPIO_Init(void); /* USER CODE BEGIN PFP */ /* USER CODE END PFP */ /* Private user code ---------------------------------------------------------*/ /* USER CODE BEGIN 0 */ /* USER CODE END 0 */ /** * @brief The application entry point. * @retval int */ int main(void) { /* USER CODE BEGIN 1 */ float T; /* USER CODE END 1 */ /* MCU Configuration--------------------------------------------------------*/ /* Reset of all peripherals, Initializes the Flash interface and the Systick. */ HAL_Init(); /* USER CODE BEGIN Init */ /* USER CODE END Init */ /* Configure the system clock */ SystemClock_Config(); /* USER CODE BEGIN SysInit */ /* USER CODE END SysInit */ /* Initialize all configured peripherals */ MX_GPIO_Init(); /* USER CODE BEGIN 2 */ lcd9648_init(); DS18B20_ConvertT(); lcd9648_clear(); lcd_show_string(0,0,96,48,16,"Temperature:"); lcd_reflash_gram(); // lcd9648_init(); // lcd9648_clear(); // lcd_show_string(0,0,96,48,12,"Hello World!"); // lcd_show_string(0,14,96,48,16,"Temperature:"); // //lcd_show_fontHZ(0, 30,"���пƼ�"); // lcd_reflash_gram(); /* USER CODE END 2 */ /* Infinite loop */ /* USER CODE BEGIN WHILE */ while (1) { DS18B20_ConvertT(); T=DS18B20_ReadT(); if(T<0){ lcd_show_char(0,1*16,'-',16,0); lcd_reflash_gram(); T=-T; } else{ lcd_show_char(0,1*16,'+',16,0); lcd_reflash_gram(); } lcd_show_num(1*8,1*16,(int)T,3,16,0); lcd_show_char(4*8,1*16,'.',16,0); lcd_show_num(5*8,1*16,(unsigned long)(T*10000)%10000,4,16,0); lcd_reflash_gram(); /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ HAL_Delay(500); } /* USER CODE END 3 */ } /** * @brief System Clock Configuration * @retval None */ void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct = {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; /** Initializes the RCC Oscillators according to the specified parameters * in the RCC_OscInitTypeDef structure. */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(); } /** Initializes the CPU, AHB and APB buses clocks */ RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK) { Error_Handler(); } } /** * @brief GPIO Initialization Function * @param None * @retval None */ static void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; /* USER CODE BEGIN MX_GPIO_Init_1 */ /* USER CODE END MX_GPIO_Init_1 */ /* GPIO Ports Clock Enable */ __HAL_RCC_GPIOC_CLK_ENABLE(); __HAL_RCC_GPIOB_CLK_ENABLE(); __HAL_RCC_GPIOA_CLK_ENABLE(); /*Configure GPIO pin Output Level */ HAL_GPIO_WritePin(GPIOC, SCL_Pin|SDA_Pin|RS_Pin|CS_Pin |RST_Pin, GPIO_PIN_RESET); /*Configure GPIO pin Output Level */ HAL_GPIO_WritePin(DQ_GPIO_Port, DQ_Pin, GPIO_PIN_RESET); /*Configure GPIO pins : SCL_Pin SDA_Pin RS_Pin CS_Pin RST_Pin */ GPIO_InitStruct.Pin = SCL_Pin|SDA_Pin|RS_Pin|CS_Pin |RST_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOC, &GPIO_InitStruct); /*Configure GPIO pin : DQ_Pin */ GPIO_InitStruct.Pin = DQ_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(DQ_GPIO_Port, &GPIO_InitStruct); /* USER CODE BEGIN MX_GPIO_Init_2 */ HAL_GPIO_WritePin(DQ_GPIO_Port,DQ_Pin,GPIO_PIN_SET); /* USER CODE END MX_GPIO_Init_2 */ } /* USER CODE BEGIN 4 */ /* USER CODE END 4 */ /** * @brief This function is executed in case of error occurrence. * @retval None */ void Error_Handler(void) { /* USER CODE BEGIN Error_Handler_Debug */ /* User can add his own implementation to report the HAL error return state */ __disable_irq(); while (1) { } /* USER CODE END Error_Handler_Debug */ } #ifdef USE_FULL_ASSERT /** * @brief Reports the name of the source file and the source line number * where the assert_param error has occurred. * @param file: pointer to the source file name * @param line: assert_param error line source number * @retval None */ void assert_failed(uint8_t *file, uint32_t line) { /* USER CODE BEGIN 6 */ /* User can add his own implementation to report the file name and line number, ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */ /* USER CODE END 6 */ } #endif /* USE_FULL_ASSERT */

ds18b20.c文件#include “ds18b20.h” #include “user_SysTick.h” /******************************************************************************* 函 数 名 : DS18B20_IO_IN 函数功能 : DS18B20_IO输入配置 输 入 : 无 输 出 : 无 *******************************************************************************/ void DS18B20_IO_IN(void) { GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin=DS18B20_PIN; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU; GPIO_Init(DS18B20_PORT,&GPIO_InitStructure); } /******************************************************************************* 函 数 名 : DS18B20_IO_OUT 函数功能 : DS18B20_IO输出配置 输 入 : 无 输 出 : 无 *******************************************************************************/ void DS18B20_IO_OUT(void) { GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin=DS18B20_PIN; GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP; GPIO_Init(DS18B20_PORT,&GPIO_InitStructure); } /******************************************************************************* 函 数 名 : DS18B20_Reset 函数功能 : 复位DS18B20 输 入 : 无 输 出 : 无 *******************************************************************************/ void DS18B20_Reset(void) { DS18B20_IO_OUT(); //SET PG11 OUTPUT DS18B20_DQ_OUT=0; //拉低DQ SysTick_Delay_Us(750); //拉低750us DS18B20_DQ_OUT=1; //DQ=1 SysTick_Delay_Us(15); //15US } /******************************************************************************* 函 数 名 : DS18B20_Check 函数功能 : 检测DS18B20是否存在 输 入 : 无 输 出 : 1:未检测到DS18B20的存在,0:存在 *******************************************************************************/ u8 DS18B20_Check(void) { u8 retry=0; DS18B20_IO_IN();//SET PG11 INPUT while (DS18B20_DQ_IN&&retry<200) { retry++; SysTick_Delay_Us(1); }; if(retry>=200) return 1; else retry=0; while (!DS18B20_DQ_IN&&retry<240) { retry++; SysTick_Delay_Us(1); }; if(retry>=240) return 1; return 0; } /******************************************************************************* 函 数 名 : DS18B20_Read_Bit 函数功能 : 从DS18B20读取一个位 输 入 : 无 输 出 : 1/0 *******************************************************************************/ u8 DS18B20_Read_Bit(void) // read one bit { u8 data; DS18B20_IO_OUT();//SET PG11 OUTPUT DS18B20_DQ_OUT=0; SysTick_Delay_Us(2); DS18B20_DQ_OUT=1; DS18B20_IO_IN();//SET PG11 INPUT SysTick_Delay_Us(12); if(DS18B20_DQ_IN) data=1; else data=0; SysTick_Delay_Us(50); return data; } /******************************************************************************* 函 数 名 : DS18B20_Read_Byte 函数功能 : 从DS18B20读取一个字节 输 入 : 无 输 出 : 一个字节数据 *******************************************************************************/ u8 DS18B20_Read_Byte(void) // read one byte { u8 i,j,dat; dat=0; for (i=1;i<=8;i++) { j=DS18B20_Read_Bit(); dat=(j<<7)|(dat>>1); } return dat; } /******************************************************************************* 函 数 名 : DS18B20_Write_Byte 函数功能 : 写一个字节到DS18B20 输 入 : dat:要写入的字节 输 出 : 无 *******************************************************************************/ void DS18B20_Write_Byte(u8 dat) { u8 j; u8 testb; DS18B20_IO_OUT();//SET PG11 OUTPUT; for (j=1;j<=8;j++) { testb=dat&0x01; dat=dat>>1; if (testb) { DS18B20_DQ_OUT=0;// Write 1 SysTick_Delay_Us(2); DS18B20_DQ_OUT=1; SysTick_Delay_Us(60); } else { DS18B20_DQ_OUT=0;// Write 0 SysTick_Delay_Us(60); DS18B20_DQ_OUT=1; SysTick_Delay_Us(2); } } } /******************************************************************************* 函 数 名 : DS18B20_Start 函数功能 : 开始温度转换 输 入 : 无 输 出 : 无 *******************************************************************************/ void DS18B20_Start(void)// ds1820 start convert { DS18B20_Reset(); DS18B20_Check(); DS18B20_Write_Byte(0xcc); DS18B20_Write_Byte(0x44); } /******************************************************************************* 函 数 名 : DS18B20_Init 函数功能 : 初始化DS18B20的IO口 DQ 同时检测DS的存在 输 入 : 无 输 出 : 1:不存在,0:存在 *******************************************************************************/ u8 DS18B20_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(DS18B20_PORT_RCC,ENABLE); GPIO_PinRemapConfig(GPIO_Remap_SWJ_Disable,ENABLE); GPIO_InitStructure.GPIO_Pin=DS18B20_PIN; GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP; GPIO_Init(DS18B20_PORT,&GPIO_InitStructure); DS18B20_Reset(); return DS18B20_Check(); } /******************************************************************************* 函 数 名 : DS18B20_GetTemperture 函数功能 : 从ds18b20得到温度值 输 入 : 无 输 出 : 温度数据 *******************************************************************************/ float DS18B20_GetTemperture(void) { u16 temp; u8 a,b; float value; DS18B20_Start(); // ds1820 start convert DS18B20_Reset(); DS18B20_Check(); DS18B20_Write_Byte(0xcc); DS18B20_Write_Byte(0xbe); a=DS18B20_Read_Byte(); b=DS18B20_Read_Byte(); temp=b; temp=(temp<<8)+a; if((temp&0xf800)==0xf800) { temp=(~temp)+1; value=temp*(-0.0625); } else { value=temp*0.0625; } return value; } ds18b20.h文件#ifndef DS18B20_H #define DS18B20_H #include “stm32f10x.h” /* DS18B20时钟端口、引脚定义 */ #define DS18B20_PORT GPIOA #define DS18B20_PIN (GPIO_Pin_6) #define DS18B20_PORT_RCC RCC_APB2Periph_GPIOA #define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x2000000+((addr&0xFFFFF)<<5)+(bitnum<<2)) #define MEM_ADDR(addr) *((volatile unsigned long *)(addr)) #define BIT_ADDR(addr, bitnum) MEM_ADDR(BITBAND(addr, bitnum)) //F103XXIO 口地址映射 #define GPIOA_ODR_Addr (GPIOA_BASE+0x0C) //0x40020014 #define GPIOB_ODR_Addr (GPIOB_BASE+0x0C) //0x40020414 #define GPIOC_ODR_Addr (GPIOC_BASE+0x0C) //0x40020814 #define GPIOD_ODR_Addr (GPIOD_BASE+0x0C) //0x40020C14 #define GPIOE_ODR_Addr (GPIOE_BASE+0x0C) //0x40021014 #define GPIOA_IDR_Addr (GPIOA_BASE+0x08) //0x40020010 #define GPIOB_IDR_Addr (GPIOB_BASE+0x08) //0x40020410 #define GPIOC_IDR_Addr (GPIOC_BASE+0x08) //0x40020810 #define GPIOD_IDR_Addr (GPIOD_BASE+0x08) //0x40020C10 #define GPIOE_IDR_Addr (GPIOE_BASE+0x08) //0x40021010 //F103XXIO 口操作,只对单一的IO 口,n 的值范围[0-15] #define PAout(n) BIT_ADDR(GPIOA_ODR_Addr,n) //GPIOA 某一位输出 #define PAin(n) BIT_ADDR(GPIOA_IDR_Addr,n) //GPIOA 某一位输入 #define PBout(n) BIT_ADDR(GPIOB_ODR_Addr,n) //GPIOB 某一位输出 #define PBin(n) BIT_ADDR(GPIOB_IDR_Addr,n) //GPIOB 某一位输入 #define PCout(n) BIT_ADDR(GPIOC_ODR_Addr,n) //GPIOC 某一位输出 #define PCin(n) BIT_ADDR(GPIOC_IDR_Addr,n) //GPIOC 某一位输入 #define PDout(n) BIT_ADDR(GPIOD_ODR_Addr,n) //GPIOD 某一位输出 #define PDin(n) BIT_ADDR(GPIOD_IDR_Addr,n) //GPIOD 某一位输入 #define PEout(n) BIT_ADDR(GPIOE_ODR_Addr,n) //GPIOE 某一位输出 #define PEin(n) BIT_ADDR(GPIOE_IDR_Addr,n) //GPIOE 某一位输入 ////IO操作函数 #define DS18B20_DQ_OUT PAout(6) //数据输出 #define DS18B20_DQ_IN PAin(6) //数据输入 u8 DS18B20_Init(void); //初始化DS18B20 float DS18B20_GetTemperture(void); //获取温度 void DS18B20_Start(void); //开始温度转换 void DS18B20_Write_Byte(u8 dat);//写入一个字节 u8 DS18B20_Read_Byte(void); //读出一个字节 u8 DS18B20_Read_Bit(void); //读出一个位 u8 DS18B20_Check(void); //检测是否存在DS18B20 void DS18B20_Reset(void); //复位DS18B20 #endif 告诉我温度传感器的工作原理?他是怎么被STM32控制的?它和STM32之间是怎么通信的?

#include "oled.h" #include "stdlib.h" #include "oledfont.h" //OLED的显存 //存放格式如下. //[0]0 1 2 3 ... 127 //[1]0 1 2 3 ... 127 //[2]0 1 2 3 ... 127 //[3]0 1 2 3 ... 127 //[4]0 1 2 3 ... 127 //[5]0 1 2 3 ... 127 //[6]0 1 2 3 ... 127 //[7]0 1 2 3 ... 127 /********************************************** //IIC Start **********************************************/ /********************************************** //IIC Start **********************************************/ void IIC_Start() { OLED_SCLK_Set() ; OLED_SDIN_Set(); OLED_SDIN_Clr(); OLED_SCLK_Clr(); } /********************************************** //IIC Stop **********************************************/ void IIC_Stop() { OLED_SCLK_Set() ; // OLED_SCLK_Clr(); OLED_SDIN_Clr(); OLED_SDIN_Set(); } void IIC_Wait_Ack() { //GPIOB->CRH &= 0XFFF0FFFF; //设置PB12为上拉输入模式 //GPIOB->CRH |= 0x00080000; // OLED_SDA = 1; // delay_us(1); //OLED_SCL = 1; //delay_us(50000); /* while(1) { if(!OLED_SDA) //判断是否接收到OLED 应答信号 { //GPIOB->CRH &= 0XFFF0FFFF; //设置PB12为通用推免输出模式 //GPIOB->CRH |= 0x00030000; return; } } */ OLED_SCLK_Set() ; OLED_SCLK_Clr(); } /********************************************** // IIC Write byte **********************************************/ void Write_IIC_Byte(unsigned char IIC_Byte) { unsigned char i; unsigned char m,da; da=IIC_Byte; OLED_SCLK_Clr(); for(i=0;i<8;i++) { m=da; // OLED_SCLK_Clr(); m=m&0x80; if(m==0x80) {OLED_SDIN_Set();} else OLED_SDIN_Clr(); da=da<<1; OLED_SCLK_Set(); OLED_SCLK_Clr(); } } /********************************************** // IIC Write Command **********************************************/ void Write_IIC_Command(unsigned char IIC_Command) { IIC_Start(); Write_IIC_Byte(0x78); //Slave address,SA0=0 IIC_Wait_Ack(); Write_IIC_Byte(0x00); //write command IIC_Wait_Ack(); Write_IIC_Byte(IIC_Command); IIC_Wait_Ack(); IIC_Stop(); } /********************************************** // IIC Write Data **********************************************/ void Write_IIC_Data(unsigned char IIC_Data) { IIC_Start(); Write_IIC_Byte(0x78); //D/C#=0; R/W#=0 IIC_Wait_Ack(); Write_IIC_Byte(0x40); //write data IIC_Wait_Ack(); Write_IIC_Byte(IIC_Data); IIC_Wait_Ack(); IIC_Stop(); } void OLED_WR_Byte(unsigned dat,unsigned cmd) { if(cmd) { Write_IIC_Data(dat); } else { Write_IIC_Command(dat); } } /******************************************** // fill_Picture ********************************************/ void fill_picture(unsigned char fill_Data) { unsigned char m,n; for(m=0;m<8;m++) { OLED_WR_Byte(0xb0+m,0); //page0-page1 OLED_WR_Byte(0x00,0); //low column start address OLED_WR_Byte(0x10,0); //high column start address for(n=0;n<128;n++) { OLED_WR_Byte(fill_Data,1); } } } /***********************Delay****************************************/ void Delay_50ms(unsigned int Del_50ms) { unsigned int m; for(;Del_50ms>0;Del_50ms--) for(m=6245;m>0;m--); } void Delay_1ms(unsigned int Del_1ms) { unsigned char j; while(Del_1ms--) { for(j=0;j<123;j++); } } //坐标设置 void OLED_Set_Pos(unsigned char x, unsigned char y) { OLED_WR_Byte(0xb0+y,OLED_CMD); OLED_WR_Byte(((x&0xf0)>>4)|0x10,OLED_CMD); OLED_WR_Byte((x&0x0f),OLED_CMD); } //开启OLED显示 void OLED_Display_On(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC命令 OLED_WR_Byte(0X14,OLED_CMD); //DCDC ON OLED_WR_Byte(0XAF,OLED_CMD); //DISPLAY ON } //关闭OLED显示 void OLED_Display_Off(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC命令 OLED_WR_Byte(0X10,OLED_CMD); //DCDC OFF OLED_WR_Byte(0XAE,OLED_CMD); //DISPLAY OFF } //清屏函数,清完屏,整个屏幕是黑色的!和没点亮一样!!! void OLED_Clear(void) { uint8_t i,n; for(i=0;i<8;i++) { OLED_WR_Byte (0xb0+i,OLED_CMD); //设置页地址(0~7) OLED_WR_Byte (0x00,OLED_CMD); //设置显示位置—列低地址 OLED_WR_Byte (0x10,OLED_CMD); //设置显示位置—列高地址 for(n=0;n<128;n++)OLED_WR_Byte(0,OLED_DATA); } //更新显示 } void OLED_On(void) { uint8_t i,n; for(i=0;i<8;i++) { OLED_WR_Byte (0xb0+i,OLED_CMD); //设置页地址(0~7) OLED_WR_Byte (0x00,OLED_CMD); //设置显示位置—列低地址 OLED_WR_Byte (0x10,OLED_CMD); //设置显示位置—列高地址 for(n=0;n<128;n++)OLED_WR_Byte(1,OLED_DATA); } //更新显示 } //在指定位置显示一个字符,包括部分字符 //x:0~127 //y:0~63 //mode:0,反白显示;1,正常显示 //size:选择字体 16/12 void OLED_ShowChar(uint8_t x,uint8_t y,uint8_t chr,uint8_t Char_Size) { unsigned char c=0,i=0; c=chr-' ';//得到偏移后的值 if(x>Max_Column-1){x=0;y=y+2;} if(Char_Size ==16) { OLED_Set_Pos(x,y); for(i=0;i<8;i++) OLED_WR_Byte(F8X16[c*16+i],OLED_DATA); OLED_Set_Pos(x,y+1); for(i=0;i<8;i++) OLED_WR_Byte(F8X16[c*16+i+8],OLED_DATA); } else { OLED_Set_Pos(x,y); for(i=0;i<6;i++) OLED_WR_Byte(F6x8[c][i],OLED_DATA); } } //m^n函数 uint32_t oled_pow(uint8_t m,uint8_t n) { uint32_t result=1; while(n--)result*=m; return result; } //显示2个数字 //x,y :起点坐标 //len :数字的位数 //size:字体大小 //mode:模式 0,填充模式;1,叠加模式 //num:数值(0~4294967295); void OLED_ShowNum(uint8_t x,uint8_t y,uint32_t num,uint8_t len,uint8_t size2) { uint8_t t,temp; uint8_t enshow=0; for(t=0;t<len;t++) { temp=(num/oled_pow(10,len-t-1))%10; if(enshow==0&&t<(len-1)) { if(temp==0) { OLED_ShowChar(x+(size2/2)*t,y,' ',size2); continue; }else enshow=1; } OLED_ShowChar(x+(size2/2)*t,y,temp+'0',size2); } OLED_Clear(); } //显示一个字符号串 void OLED_ShowString(uint8_t x,uint8_t y,uint8_t *chr,uint8_t Char_Size) { unsigned char j=0; while (chr[j]!='\0') { OLED_ShowChar(x,y,chr[j],Char_Size); x+=8; if(x>120){x=0;y+=2;} j++; } OLED_Clear(); } //显示汉字 void OLED_ShowCHinese(uint8_t x,uint8_t y,uint8_t no) { uint8_t t,adder=0; OLED_Set_Pos(x,y); for(t=0;t<16;t++) { OLED_WR_Byte(Hzk[2*no][t],OLED_DATA); adder+=1; } OLED_Set_Pos(x,y+1); for(t=0;t<16;t++) { OLED_WR_Byte(Hzk[2*no+1][t],OLED_DATA); adder+=1; } } /***********功能描述:显示显示BMP图片128×64起始点坐标(x,y),x的范围0~127,y为页的范围0~7*****************/ 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_Set_Pos(x0,y); for(x=x0;x<x1;x++) { OLED_WR_Byte(BMP[j++],OLED_DATA); } } } //初始化SSD1306 void OLED_Init(void) { // GPIO_InitTypeDef GPIO_InitStructure; // // RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能A端口时钟 // GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5|GPIO_Pin_7; // GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出 // GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHz // GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化GPIOD3,6 // GPIO_SetBits(GPIOA,GPIO_Pin_5|GPIO_Pin_7); Delay_Ms(800); OLED_WR_Byte(0xAE,OLED_CMD);//--display off 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 OLED_WR_Byte(0xB0,OLED_CMD);//--set page address OLED_WR_Byte(0x81,OLED_CMD); // contract control OLED_WR_Byte(0xFF,OLED_CMD);//--128 OLED_WR_Byte(0xA1,OLED_CMD);//set segment remap OLED_WR_Byte(0xA6,OLED_CMD);//--normal / reverse OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64) OLED_WR_Byte(0x3F,OLED_CMD);//--1/32 duty OLED_WR_Byte(0xC8,OLED_CMD);//Com scan direction OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset OLED_WR_Byte(0x00,OLED_CMD);// OLED_WR_Byte(0xD5,OLED_CMD);//set osc division OLED_WR_Byte(0x80,OLED_CMD);// OLED_WR_Byte(0xD8,OLED_CMD);//set area color mode off OLED_WR_Byte(0x05,OLED_CMD);// OLED_WR_Byte(0xD9,OLED_CMD);//Set Pre-Charge Period OLED_WR_Byte(0xF1,OLED_CMD);// OLED_WR_Byte(0xDA,OLED_CMD);//set com pin configuartion OLED_WR_Byte(0x12,OLED_CMD);// OLED_WR_Byte(0xDB,OLED_CMD);//set Vcomh OLED_WR_Byte(0x30,OLED_CMD);// OLED_WR_Byte(0x8D,OLED_CMD);//set charge pump enable OLED_WR_Byte(0x14,OLED_CMD);// OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel } 写一个MAIN函数就行

最新推荐

recommend-type

(完整word版)《网站设计与建设》教案-顾伟.doc

(完整word版)《网站设计与建设》教案-顾伟.doc
recommend-type

assimp-5.2.4.tar

assimp-5.2.4.tar
recommend-type

顺丰快递单号远程验证码获取查询系统

资源下载链接为: https://pan.quark.cn/s/d9ef5828b597 顺丰快递单号查询系统采用PHP+cURL架构,服务器端通过cURL模拟浏览器向顺丰接口发起HTTP请求。cURL流程:curl_init初始化会话→curl_setopt设定URL、方法、头信息、Cookie、POST字段、超时→curl_exec抓取响应→curl_getinfo读取状态→curl_errno捕获异常→curl_close释放句柄。若需登录,可先用POST提交账号密码并保存Cookie;遇验证码则借助GD或OpenCV预处理图片后调用OCR识别,必要时用训练好的模型自动填写。查询参数仅快递单号,返回结果经解析后展示,同时做好错误提示与速度优化,兼容多终端。整套方案展示了PHP抓取远程数据、处理登录及验证码的完整思路,对爬虫、自动化测试均有借鉴意义。
recommend-type

Web2.0新特征图解解析

Web2.0是互联网发展的一个阶段,相对于早期的Web1.0时代,Web2.0具有以下显著特征和知识点: ### Web2.0的定义与特点 1. **用户参与内容生产**: - Web2.0的一个核心特征是用户不再是被动接收信息的消费者,而是成为了内容的生产者。这标志着“读写网络”的开始,用户可以在网络上发布信息、评论、博客、视频等内容。 2. **信息个性化定制**: - Web2.0时代,用户可以根据自己的喜好对信息进行个性化定制,例如通过RSS阅读器订阅感兴趣的新闻源,或者通过社交网络筛选自己感兴趣的话题和内容。 3. **网页技术的革新**: - 随着技术的发展,如Ajax、XML、JSON等技术的出现和应用,使得网页可以更加动态地与用户交互,无需重新加载整个页面即可更新数据,提高了用户体验。 4. **长尾效应**: - 在Web2.0时代,即使是小型或专业化的内容提供者也有机会通过互联网获得关注,这体现了长尾理论,即在网络环境下,非主流的小众产品也有机会与主流产品并存。 5. **社交网络的兴起**: - Web2.0推动了社交网络的发展,如Facebook、Twitter、微博等平台兴起,促进了信息的快速传播和人际交流方式的变革。 6. **开放性和互操作性**: - Web2.0时代倡导开放API(应用程序编程接口),允许不同的网络服务和应用间能够相互通信和共享数据,提高了网络的互操作性。 ### Web2.0的关键技术和应用 1. **博客(Blog)**: - 博客是Web2.0的代表之一,它支持用户以日记形式定期更新内容,并允许其他用户进行评论。 2. **维基(Wiki)**: - 维基是另一种形式的集体协作项目,如维基百科,任何用户都可以编辑网页内容,共同构建一个百科全书。 3. **社交网络服务(Social Networking Services)**: - 社交网络服务如Facebook、Twitter、LinkedIn等,促进了个人和组织之间的社交关系构建和信息分享。 4. **内容聚合器(RSS feeds)**: - RSS技术让用户可以通过阅读器软件快速浏览多个网站更新的内容摘要。 5. **标签(Tags)**: - 用户可以为自己的内容添加标签,便于其他用户搜索和组织信息。 6. **视频分享(Video Sharing)**: - 视频分享网站如YouTube,用户可以上传、分享和评论视频内容。 ### Web2.0与网络营销 1. **内容营销**: - Web2.0为内容营销提供了良好的平台,企业可以通过撰写博客文章、发布视频等内容吸引和维护用户。 2. **社交媒体营销**: - 社交网络的广泛使用,使得企业可以通过社交媒体进行品牌传播、产品推广和客户服务。 3. **口碑营销**: - 用户生成内容、评论和分享在Web2.0时代更易扩散,为口碑营销提供了土壤。 4. **搜索引擎优化(SEO)**: - 随着内容的多样化和个性化,SEO策略也必须适应Web2.0特点,注重社交信号和用户体验。 ### 总结 Web2.0是对互联网发展的一次深刻变革,它不仅仅是一个技术变革,更是人们使用互联网的习惯和方式的变革。Web2.0的时代特征与Web1.0相比,更加注重用户体验、社交互动和信息的个性化定制。这些变化为网络营销提供了新的思路和平台,也对企业的市场策略提出了新的要求。通过理解Web2.0的特点和应用,企业可以更好地适应互联网的发展趋势,实现与用户的深度互动和品牌的有效传播。
recommend-type

【C++编程新手必看】:一步步带你制作出风靡全球的“别踩白块儿”游戏

# 摘要 本文全面介绍C++编程在游戏开发中的应用,涵盖了从基础概念到具体实现的多个方面。首先,文章提供了游戏开发环境的搭建指南,包括编译器配置和开发工具的选择。随后,重点介绍了游戏主循环和基本框架的构建,强调了事件处理和渲染技术。在游戏逻辑和交互设计方面,本文阐述了界面布局、事件响应和游戏状态管理的核心实现。为了提升游戏体验,本文还探讨了添加音效和背景音乐以及开发高级游戏特性的方法。最后,文章介绍了性能优化和跨平台发布的过程,包括游戏的打包和针对不同平台的发布策略。本文旨在为C++游戏开发者提供一个实用的开发指南,帮助他们从零开始构建出性能优化、跨平台兼容的游戏。 # 关键字 C++编程;
recommend-type

使用scikit-learn训练模型来预测鸢尾花种类

<think>我们使用scikit-learn库来训练一个分类模型预测鸢尾花种类。鸢尾花数据集是scikit-learn内置的经典数据集,包含三种鸢尾花的四个特征(花萼长度、花萼宽度、花瓣长度、花瓣宽度)和对应的类别标签。步骤:1.导入必要的库和数据集2.数据探索(可选,但建议了解数据)3.数据预处理(如划分训练集和测试集、标准化等)4.选择模型(例如KNN、随机森林等)5.训练模型6.模型评估7.使用模型进行预测下面以KNN(K近邻)算法为例,展示完整流程。</think>### 使用 scikit-learn 训练鸢尾花分类模型 以下是基于 scikit-learn 训练鸢尾花分类模型的
recommend-type

WWF工作流设计器C#源码解析及演示

### WWF工作流设计器控件C#源码知识点 #### 1. WWF(Windows Workflow Foundation)概述 WWF是微软公司推出的一个工作流框架,作为.NET Framework的一部分。它提供了一套丰富的API,用于设计、执行和管理工作流。工作流可以用于各种应用程序,包括Web应用、服务和桌面应用,使得开发者能够将复杂的业务逻辑以工作流的形式表现出来,简化业务流程自动化和管理。 #### 2. 工作流设计器控件(Workflow Designer Control) 工作流设计器控件是WWF中的一个组件,主要用于提供可视化设计工作流的能力。它允许用户通过拖放的方式在界面上添加、配置和连接工作流活动,从而构建出复杂的工作流应用。控件的使用大大降低了工作流设计的难度,并使得设计工作流变得直观和用户友好。 #### 3. C#源码分析 在提供的文件描述中提到了两个工程项目,它们均使用C#编写。下面分别对这两个工程进行介绍: - **WorkflowDesignerControl** - 该工程是工作流设计器控件的核心实现。它封装了设计工作流所需的用户界面和逻辑代码。开发者可以在自己的应用程序中嵌入这个控件,为最终用户提供一个设计工作流的界面。 - 重点分析:控件如何加载和显示不同的工作流活动、控件如何响应用户的交互、控件状态的保存和加载机制等。 - **WorkflowDesignerExample** - 这个工程是演示如何使用WorkflowDesignerControl的示例项目。它不仅展示了如何在用户界面中嵌入工作流设计器控件,还展示了如何处理用户的交互事件,比如如何在设计完工作流后进行保存、加载或执行等。 - 重点分析:实例程序如何响应工作流设计师的用户操作、示例程序中可能包含的事件处理逻辑、以及工作流的实例化和运行等。 #### 4. 使用Visual Studio 2008编译 文件描述中提到使用Visual Studio 2008进行编译通过。Visual Studio 2008是微软在2008年发布的集成开发环境,它支持.NET Framework 3.5,而WWF正是作为.NET 3.5的一部分。开发者需要使用Visual Studio 2008(或更新版本)来加载和编译这些代码,确保所有必要的项目引用、依赖和.NET 3.5的特性均得到支持。 #### 5. 关键技术点 - **工作流活动(Workflow Activities)**:WWF中的工作流由一系列的活动组成,每个活动代表了一个可以执行的工作单元。在工作流设计器控件中,需要能够显示和操作这些活动。 - **活动编辑(Activity Editing)**:能够编辑活动的属性是工作流设计器控件的重要功能,这对于构建复杂的工作流逻辑至关重要。 - **状态管理(State Management)**:工作流设计过程中可能涉及保存和加载状态,例如保存当前的工作流设计、加载已保存的工作流设计等。 - **事件处理(Event Handling)**:处理用户交互事件,例如拖放活动到设计面板、双击活动编辑属性等。 #### 6. 文件名称列表解释 - **WorkflowDesignerControl.sln**:解决方案文件,包含了WorkflowDesignerControl和WorkflowDesignerExample两个项目。 - **WorkflowDesignerControl.suo**:Visual Studio解决方案用户选项文件,该文件包含了开发者特有的个性化设置,比如窗口布局、断点位置等。 - **Thumbs.db**:缩略图缓存文件,由Windows自动生成,用于存储文件夹中的图片缩略图,与WWF工作流设计器控件功能无关。 - **WorkflowDesignerExample**:可能是一个文件夹,包含了示例工程相关的所有文件,或者是示例工程的可执行文件。 - **EULA.txt**:最终用户许可协议文本文件,通常说明了软件的版权和使用许可条件。 综上所述,该文件集包含了WWF工作流设计器控件的完整C#源码以及相应的Visual Studio项目文件,开发者可以利用这些资源深入理解WWF工作流设计器控件的工作机制,并将其应用于实际的项目开发中,实现工作流的设计和管理功能。
recommend-type

CAD数据在ANSA中:完美修复几何数据的策略与方法

# 摘要 CAD数据的准确性和几何完整性对于产品设计和工程分析至关重要。本文首先强调了CAD数据和几何修复的重要性,随后介绍了ANSA软件在处理CAD数据中的角色。通过详细的分析,本文探讨了几何数据的常见问题,以及有效的手动和自动修复技术。文中还提供了几何修复在实际应用中的案例分析,并讨论了行业未来对几何修复技术的需求和新兴技术趋势。文章旨在为CAD数据几何修复提供全面的理论知识、诊断方法和修复策略,并
recommend-type

编写verilog代码实现以上的规格化功能

<think>我们有一个16位的二进制数,其格式如下:-a15:尾数符号位(1位)-a14-a5:尾数数值位(10位)-a4:指数符号位(1位)-a3-a0:指数数值位(4位)规格化的目标:确保尾数数值位的最高位a14为1(即尾数的绝对值在[0.5,1)范围内)。步骤:1.提取输入信号的各个部分:尾数符号位、尾数数值位、指数符号位、指数数值位。2.将尾数数值位(10位)视为无符号整数M(范围0到1023),我们需要通过左移操作使得M的最高位为1(即M>=512)。同时记录左移的位数(shift_count)。3.调整指数:新的指数=原指数-shift_count(因为尾数左移相当于乘以2^sh
recommend-type

探索ARM9 2410开发板与wince5.0系统的高级实验

标题中的“周立功ARM (magicarm2410) 高级实验”指明了文档内容涉及周立功品牌下的ARM9 2410开发板的高级使用实验。ARM9 2410是基于ARM920T内核的处理器,广泛应用于嵌入式系统开发。周立功是一家在电子与嵌入式系统领域内具有影响力的公司,提供嵌入式教学和开发解决方案。MagicARM2410是该公司的某型号开发板,可能专为教学和实验设计,携带了特定的实验内容,例如本例中的“eva例程”。 描述提供了额外的背景信息,说明周立功ARM9 2410开发板上预装有Windows CE 5.0操作系统,以及该开发板附带的EVA例程。EVA可能是用于实验教学的示例程序或演示程序。文档中还提到,虽然书店出售的《周立功 ARM9开发实践》书籍中没有包含EVA的源码,但该源码实际上是随开发板提供的。这意味着,EVA例程的源码并不在书籍中公开,而是需要直接从开发板上获取。这对于那些希望深入研究和修改EVA例程的学生和开发者来说十分重要。 标签中的“magicarm2410”和“周立功ARM”是对文档和开发板的分类标识。这些标签有助于在文档管理系统或资料库中对相关文件进行整理和检索。 至于“压缩包子文件的文件名称列表:新建文件夹”,这表明相关文件已经被打包压缩,但具体的文件内容和名称没有在描述中列出。我们仅知道压缩包内至少存在一个“新建文件夹”,这可能意味着用户需要进一步操作来查看或解压出文件夹中的内容。 综合以上信息,知识点主要包括: 1. ARM9 2410开发板:一款基于ARM920T内核的处理器的嵌入式开发板,适用于教学和项目实验。 2. Windows CE 5.0系统:这是微软推出的专为嵌入式应用设计的操作系统,提供了一个可定制、可伸缩的、实时的操作环境。 3. EVA例程:一个嵌入式系统开发的教学或实验示例程序。它可能被设计用于演示特定功能或技术,如显示、控制或通信。 4. 开发实践书籍与源码提供:《周立功 ARM9开发实践》一书可能详细介绍了ARM9 2410开发板的使用方法,但书中的内容不包含EVA例程的源码,源码需要通过其他途径获得。 5. 文件打包压缩:文档可能以压缩包的形式存在,包含了需要的内容,但具体内容未知,需要解压缩之后才能查看。 了解这些知识点后,对于从事嵌入式系统开发的工程师或者学生来说,可以更好地利用周立功 ARM9 2410开发板进行学习和实验,尤其是可以进行更深入的研究和实验设计,通过EVA例程的源码来理解嵌入式系统的运行机制和程序结构。同时,也能够使用Windows CE 5.0系统环境来开发相应的应用程序。