STM32单片机(9) LCD1602显示屏输出实验

【转载请注明出处:http://blog.csdn.net/leytton/article/details/38436775

核心代码由kingsraywii提供,本文作者对其进行整合、更详细地注释和部分代码改进, 添加了芯片ID获取、字符串输出和printf重定向输出功能

注:使用普中科技开发板测试时,需要拔掉Boot1插口 接5V电压,重启

/*******************************************************************************
*	
* 软件功能:	LCD1602实验(软件延时方式)
* 
*******************************************************************************/
#include "stm32f10x.h"
#include "delay.h"
#include <stdio.h>


/*
	定义宏变量
	lcd_RS = data/command(H/L) 数据/命令操作
	lcd_RW = read/write(H/L)   读写操作
	lcd_Enable 使能操作
	lcd_GPIO lcd1602接在了GPIOX口
*/
//接口
#define lcd_GPIO 		GPIOB
#define lcd_busyflag    GPIO_Pin_15 //DB7对应的口
#define lcd_pin_RS 		GPIO_Pin_1			 
#define lcd_pin_RW 		GPIO_Pin_2
#define lcd_pin_Enable 	GPIO_Pin_0
//状态
#define lcd_DATA 	1
#define lcd_COMMAND 0
#define lcd_READ 	1
#define lcd_WRITE 	0
//功能
#define lcd_RS(n)	  	n ? GPIO_SetBits(lcd_GPIO,lcd_pin_RS) : GPIO_ResetBits(lcd_GPIO,lcd_pin_RS)
#define lcd_RW(n)		n ? GPIO_SetBits(lcd_GPIO,lcd_pin_RW) : GPIO_ResetBits(lcd_GPIO,lcd_pin_RW)
#define lcd_Enable(n)   n ? GPIO_SetBits(lcd_GPIO,lcd_pin_Enable) : GPIO_ResetBits(lcd_GPIO,lcd_pin_Enable)

void RCC_Configuration(void);
void GPIO_Configuration(void);
void Lcd_Write(u8 type,u8 buf); 
void Lcd_WriteString(u8 *buf);
int Lcd_isbusy(void);
void Lcd_Init(void);
int fputc(int ch, FILE *f);
void getSys_ID(void);

/*************************************************
函数: int main(void)
功能: main主函数
参数: 无
返回: 无
**************************************************/
int main(void)
{
  int i=0;
  unsigned char* display_1={"chip ID:"};
  //unsigned char* display_2={"www.llqqww.com"};

  RCC_Configuration();
  GPIO_Configuration();
  delay_init(72);

  Lcd_Init();
	//while(1){
		while(Lcd_isbusy());       //检测忙信号
		Lcd_Write(lcd_COMMAND,0x80);    //第一行第一个字开始
		for(i=0;display_1[i]!='\0';i++){
			while(Lcd_isbusy());       //检测忙信号
			Lcd_Write(lcd_DATA,display_1[i]);
			//delay_ms(500);
		}
	    //while(Lcd_isbusy());       //检测忙信号
		//Lcd_Write(lcd_COMMAND,0xC0); //第二行第一个字开始
		//Lcd_WriteString("www.llqqww.com");
		getSys_ID();
		/*for(i=0;display_2[i]!='\0';i++){
			Lcd_Write(lcd_DATA,display_2[i]);
			delay_ms(500);
		} */
	//	Lcd_Write(lcd_COMMAND,0x01);	 //清屏
//	} 
}


/*
函数: void RCC_Configuration(void)
功能: 复位和时钟控制 配置
*/
void RCC_Configuration(void)
{
	ErrorStatus HSEStartUpStatus;                    //定义外部高速晶体启动状态枚举变量
	RCC_DeInit();                                    //复位RCC外部设备寄存器到默认值
	RCC_HSEConfig(RCC_HSE_ON);                       //打开外部高速晶振
	HSEStartUpStatus = RCC_WaitForHSEStartUp();      //等待外部高速时钟准备好
	if(SUCCESS == HSEStartUpStatus)                  //外部高速时钟已经准别好
	{
		FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //开启FLASH预读缓冲功能,加速FLASH的读取。所有程序中必须的用法.位置:RCC初始化子函数里面,时钟起振之后
		FLASH_SetLatency(FLASH_Latency_2);                    //flash操作的延时
		  	
		RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);  //配置PLL时钟 == 外部高速晶体时钟(16MHz)  * 9 = 72MHz	 
		RCC_PLLCmd(ENABLE);                                   //使能PLL时钟
		
		RCC_HCLKConfig(RCC_SYSCLK_Div1);               //配置AHB(HCLK)时钟等于==SYSCLK(系统时钟)
		RCC_PCLK2Config(RCC_HCLK_Div1);                //配置APB2(PCLK2)钟==AHB时钟
		RCC_PCLK1Config(RCC_HCLK_Div2);                //配置APB1(PCLK1)钟==AHB1/2时钟
		
		while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET){}    //等待PLL时钟就绪
		
		RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);            //配置系统时钟 = PLL时钟
		while(RCC_GetSYSCLKSource() != 0x08){}                  //检查PLL时钟是否作为系统时钟
	}
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_AFIO, ENABLE);  //允许GPIOB、AFIO时钟
}

/*
函数: void GPIO_Configuration(void)
功能: GPIO初始化
*/
void GPIO_Configuration(void)
{

	GPIO_InitTypeDef GPIO_InitStructure;        //定义GPIO初始化结构体
	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_All  ;  //设置初始化引脚
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //设置响应速率
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  //设置I/O模式
	GPIO_Init(GPIOB, &GPIO_InitStructure); //调用MDK初始化GPIOX口

	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9 | GPIO_Pin_10 | GPIO_Pin_11 |GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15; //所有GPIO为同一类型端口
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;	 //open drain
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;	 //输出的最大频率为50HZ
	GPIO_Init(GPIOB, &GPIO_InitStructure);   //初始化GPIOB端口 

}



/*
	函数名:Lcd_Write
	功能:向lcd1602模块写入命令/数据
	参数1: type:
			lcd_COMMAND	写入命令
			lcd_DATA	写入数据
	参数2: buf 8个二进制位的命令/数据
*/
void Lcd_Write(u8 type,u8 buf) 
{
	lcd_RS(type);
	lcd_RW(lcd_WRITE);		  //写入 命令/数据操作
	lcd_Enable(0);
	delay_us(2); //根据使用手册地址建立时间30ns 这里延长延时至1 us
 
	lcd_GPIO->BSRR |= (buf<<8) & 0xff00;     	//利用BSRR寄存器,将需要置1 的口置1
	lcd_GPIO->BRR  |= ((~buf)<<8) & 0xff00;	   	//利用BRR寄存器,将不要置1,即为0的口置为0
	delay_us(2);  //根据使用手册数据建立时间20ns 这里延长延时至1 us
	
	lcd_Enable(1); //使能写入
	delay_us(2);  //根据使用手册脉冲宽度为150ns 这里用了1 us
	lcd_Enable(0); //结束写入
	delay_ms(15);
}
void Lcd_WriteString(u8* buf)
{
	int i=0;
	for(i=0;buf[i]!='\0';i++){
	        while(Lcd_isbusy());       //检测忙信号
			Lcd_Write(lcd_DATA,buf[i]);
			//delay_ms(500);
	}	
}
int Lcd_isbusy()
{
	int result;
	lcd_RS(lcd_COMMAND);
	lcd_RW(lcd_READ);
	lcd_Enable(1);
	delay_us(2);
	//result = GPIO_ReadInputDataBit(lcd_GPIO,lcd_busyflag);
	result = (GPIO_ReadInputData(lcd_GPIO) & 0x8000)>>8;  

	lcd_Enable(0);
	return result;
}
/*
	初始化LCD1602
*/
void Lcd_Init()
{
	delay_ms(15);					//1、延时15ms
	Lcd_Write(lcd_COMMAND,0x38);	//2、写指令38H(不检测忙信号)
	delay_ms(5);		 			//3、延时5ms
	Lcd_Write(lcd_COMMAND,0x38);	//4、写指令38H(不检测忙信号)
	delay_ms(5);					//5、延时5ms
	Lcd_Write(lcd_COMMAND,0x38);    //6、写指令38H(不检测忙信号) 
	                                //7、以后每次写指令、读/写数据之前均需检测忙信号							  
	//delay_ms(15);					//延时15ms
	while(Lcd_isbusy());		    //检测忙信号
	Lcd_Write(lcd_COMMAND,0x38);	//8、写指令38H  显示模式设置 设置16X2显示 5X7点阵 8位数据接口
	
	//delay_ms(15);					//延时15ms
	while(Lcd_isbusy());		    //检测忙信号
	Lcd_Write(lcd_COMMAND,0x0c);    //9.显示开/关,及光标设置 此处为0000 1100 即开显示、不显示光标、光标不闪烁
									/* 指令码 0000 1DCB
									   D=1 开显示 D=0 关显示  C=1 显示光标 C=0 不显示光标 B=1光标闪烁 B=0 光标不闪烁
									*/
	//delay_ms(15);					//延时15ms
	while(Lcd_isbusy());		    //检测忙信号	 
	Lcd_Write(lcd_COMMAND,0x06);    //10.写一个指针加1,即控制从左至右显示还是从右至左 此处为0000 0110
									/* 指令码 0000 01NS
									   N=1 读/写一个数据后,指针和光标加1 N=0 读/写一个数据后,指针和光标减1  
									   S=1 写入新数据后显示屏整体移1个字符  S=0 写入新数据后显示屏不移动
									*/
	//delay_ms(15);				    //延时15ms
	while(Lcd_isbusy());		    //检测忙信号
	Lcd_Write(lcd_COMMAND,0x01);    //11.清屏
}

int fputc(int ch, FILE *f)
{
	while(Lcd_isbusy());       //检测忙信号
	Lcd_Write(lcd_DATA,((u8)ch));
	return (ch);
}

void getSys_ID(void)
{
	u8  Sys_ID[12],i;
	for(i=0;i<12;i++)
	{
		Sys_ID[i]=*(u8*)(0x1FFFF7E8+i);  //产品唯一身份标识寄存器(96位)
		printf("%0.2X",Sys_ID[i]); 
		if(i==3)
		{
			 while(Lcd_isbusy());       //检测忙信号
			 Lcd_Write(lcd_COMMAND,0xC0); //第二行第一个字开始
		}
	}

}


  • 9
    点赞
  • 40
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
以下是在STM32单片机使用LCD1602和DS18B20显示温度的代码: ``` #include "stm32f10x.h" #include "delay.h" #include "lcd1602.h" #include "ds18b20.h" int main(void) { uint8_t temperature_string[7] = {0}; // 存储温度值转换为字符串后的结果 // 初始化LCD1602显示屏 lcd1602_init(); // 初始化DS18B20温度传感器 ds18b20_init(GPIOB, GPIO_Pin_8); // 无限循环 while(1) { float temperature = ds18b20_read_temp(GPIOB, GPIO_Pin_8); // 读取温度 // 将温度转换为带一位小数的字符串形式 int integer_part = (int)temperature; int decimal_part = (int)(temperature * 10) % 10; sprintf((char*)temperature_string, "%d.%dC", integer_part, decimal_part); // 在LCD1602显示屏显示温度 lcd1602_write_string(0, 0, "Temperature:"); lcd1602_write_string(0, 1, (char*)temperature_string); // 延迟一段时间再进行下一次循环,避免过于频繁的读取温度传感器 delay_ms(1000); } } ``` 其,`delay.h`和`delay.c`文件可以用外部提供的延时库,也可以自行编写。`lcd1602.h`和`lcd1602.c`文件是LCD1602显示屏的驱动代码,提供了一些常用的函数,如`lcd1602_init()`用于初始化LCD1602显示屏,`lcd1602_write_string()`用于在指定位置上显示字符串。`ds18b20.h`和`ds18b20.c`文件是DS18B20温度传感器的驱动代码,提供了一些常用的函数,如`ds18b20_init()`用于初始化DS18B20温度传感器,`ds18b20_read_temp()`用于读取温度。在以上代码,GPIOB的第8个引脚分别连接到DS18B20温度传感器的DATA引脚,用于获取温度值并显示LCD1602显示屏上。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值