STM32—LCD1602

news2025/7/19 6:25:43

LCD1602(Liquid Crystal Display)是一种工业字符型液晶,能够同时显示 16×02 即 32 字符(16列两行)
在这里插入图片描述

第 1 脚: VSS 为电源地

第 2 脚: VDD 接 5V 正电源

第 3 脚: VL 为液晶显示器对比度调整端,接正电源时对比度最弱,接地时对比度最高,对比度 过高时会产生“鬼影”,使用时可以通过一个
10K 的电位器调整对比度。

第 4 脚:RS 为寄存器选择,高电平时选择数据寄存器、低电平时选择指令寄存器。

第 5 脚:R/W 为读写信号线,高电平时进行读操作,低电平时进行写操作。当 RS 和 R/W 共 同为低电平时可以写入指令或者显示地址,当RS 为低电平 R/W 为高电平时可以读忙信号, 当 RS 为高电平 R/W 为低电平时可以写入数据。

第 6 脚:E 端为使能端,当 E 端由高电平跳变成低电平时,液晶模块执行命令。

第 7-14 脚:D0~D7 为 8 位双向数据线。

第 15 脚:背光源正极。

第 16 脚:背光源负极。

在这里插入图片描述
在这里插入图片描述
开发逻辑

  • 在哪显示

在这里插入图片描述
例如第二行第一个字符的地址是 40H,那么是否直接写入 40H 就可以将光标定位在第二行第一个字符的位置呢?这样不行,因为写入显示地址时要求最高位 D7 恒定为高电平 1 所以实际写入的数据应该是 01000000B(40H) +10000000B(80H)=11000000B(C0H)

  • 显示什么
    在这里插入图片描述
    读写操作时序
    读操作
    在这里插入图片描述
    检测忙信号的时候需要读时序
/*当 RS 为低电平 R/W 为高电平时可以读忙信号读忙信号和光标地址。
BF(bit7):为忙标志位,高电平表示忙,此时模块不能接收命令或者数据,
如果为低电平表示不忙。
*/

void check_busy()//检测忙信号
{
	char tmp = 0x80;
	dataBuff = 0x80;
	while(tmp & 0x80){//检测BF,1忙 0闲
	
	RS = 0;
	RW = 1;
	
	EN = 0;
	_nop_();
	EN = 1;
	_nop_();
	_nop_();
	tmp = dataBuff; 
	_nop_();
	EN = 0;
	_nop_();
	}
}

写操作在这里插入图片描述
数序参数在这里插入图片描述

/*
RS 为寄存器选择,高电平时选择数据寄存器、低电平时选择指令寄存器。

*/
void Write_Cmd_Func(char cmd)
{
	check_busy();
	RS = 0;
	RW = 0;
	
	EN = 0;
	_nop_();
	
	dataBuff = cmd;
	_nop_();
	
	EN = 1;
	_nop_();
	_nop_();
	EN = 0;
	_nop_();
	
}

void Write_Data_Func(char dataShow)
{
	check_busy();
	RS = 1;
	RW = 0;
	
	EN = 0;
	_nop_();
	
	dataBuff = dataShow;
	_nop_();
	
	EN = 1;
	_nop_();
	_nop_();
	EN = 0;
	_nop_();
}

LCD1602的初始化函数

void LCD1602_INIT()
{
 //(1)延时 15ms

 Delay15ms();

//(2)写指令 38H(不检测忙信号) 

 Write_Cmd_Func(0x38);

//(3)延时 5ms

 Delay5ms();

//(4)以后每次写指令,读/写数据操作均需要检测忙信号
	check_busy();

//(5)写指令 38H:显示模式设置

 Write_Cmd_Func(0x38);

//(6)写指令 08H:显示关闭

 Write_Cmd_Func(0x08);

//(7)写指令 01H:显示清屏

 Write_Cmd_Func(0x01);

//(8)写指令 06H:显示光标移动设置

 Write_Cmd_Func(0x06);

//(9)写指令 0CH:显示开及光标设置}

 Write_Cmd_Func(0x0c);
}

C51实验显示:hello world

#include "reg52.h"
#include "intrins.h"

#define dataBuff P0

sbit RS = P2^6;
sbit RW = P2^5;
sbit EN = P2^7;

/*当 RS 为低电平 R/W 为高电平时可以读忙信号读忙信号和光标地址。
BF(bit7):为忙标志位,高电平表示忙,此时模块不能接收命令或者数据,
如果为低电平表示不忙。
*/

void check_busy()//检测忙信号
{
	char tmp = 0x80;
	dataBuff = 0x80;
	while(tmp & 0x80){//检测BF,1忙 0闲
	
	RS = 0;
	RW = 1;
	
	EN = 0;
	_nop_();
	EN = 1;
	_nop_();
	_nop_();
	tmp = dataBuff; 
	_nop_();
	EN = 0;
	_nop_();
	}
}

/*
RS 为寄存器选择,高电平时选择数据寄存器、低电平时选择指令寄存器。

*/
void Write_Cmd_Func(char cmd)
{
	check_busy();
	RS = 0;
	RW = 0;
	
	EN = 0;
	_nop_();
	
	dataBuff = cmd;
	_nop_();
	
	EN = 1;
	_nop_();
	_nop_();
	EN = 0;
	_nop_();
	
}

void Write_Data_Func(char dataShow)
{
	check_busy();
	RS = 1;
	RW = 0;
	
	EN = 0;
	_nop_();
	
	dataBuff = dataShow;
	_nop_();
	
	EN = 1;
	_nop_();
	_nop_();
	EN = 0;
	_nop_();
}


void Delay5ms() //@11.0592MHz

{
 unsigned char i, j;
 i = 9;
 j = 244;
 do

 {
 while (--j);
 } while (--i);
}

void Delay15ms() //@11.0592MHz

{
 unsigned char i, j;
 i = 27;
 j = 226;
 do

 {
 while (--j);
 } while (--i);
}


void LCD1602_INIT()
{
 //(1)延时 15ms

 Delay15ms();

//(2)写指令 38H(不检测忙信号) 

 Write_Cmd_Func(0x38);

//(3)延时 5ms

 Delay5ms();

//(4)以后每次写指令,读/写数据操作均需要检测忙信号
	check_busy();

//(5)写指令 38H:显示模式设置

 Write_Cmd_Func(0x38);

//(6)写指令 08H:显示关闭

 Write_Cmd_Func(0x08);

//(7)写指令 01H:显示清屏

 Write_Cmd_Func(0x01);

//(8)写指令 06H:显示光标移动设置

 Write_Cmd_Func(0x06);

//(9)写指令 0CH:显示开及光标设置}

 Write_Cmd_Func(0x0c);
}

void lcd1602_Show_line(int row,int col,char *SendData)
{
	switch(row){
		case 1:
						Write_Cmd_Func(0x80+col);
					while(*SendData){
						Write_Data_Func(*SendData);
						SendData++;
					}
					break;
		
		case 2:
					Write_Cmd_Func(0x80+0x40+col);
					while(*SendData){
						Write_Data_Func(*SendData);
						SendData++;
					}
					break;
		
	}
	
}

void main()
{
	LCD1602_INIT();
	
	lcd1602_Show_line(1,3,"hello world");

	

}

STM32HAL库

RS、RW、EN三根信号线经常需要进行拉高/拉低操作,可以进行封装

#define RS_GPIO_Port GPIOB
#define RW_GPIO_Port GPIOB
#define EN_GPIO_Port GPIOB
#define RS_Pin GPIO_PIN_1
#define RW_Pin GPIO_PIN_2
#define EN_Pin GPIO_PIN_10

#define RS_HIGH HAL_GPIO_WritePin(RS_GPIO_Port, RS_Pin, GPIO_PIN_SET)

#define RS_LOW HAL_GPIO_WritePin(RS_GPIO_Port, RS_Pin, GPIO_PIN_RESET)

#define RW_HIGH HAL_GPIO_WritePin(RW_GPIO_Port, RW_Pin, GPIO_PIN_SET)

#define RW_LOW HAL_GPIO_WritePin(RW_GPIO_Port, RW_Pin, GPIO_PIN_RESET)

#define EN_HIGH HAL_GPIO_WritePin(EN_GPIO_Port, EN_Pin, GPIO_PIN_SET)

#define EN_LOW HAL_GPIO_WritePin(EN_GPIO_Port, EN_Pin, GPIO_PIN_RESET)

如何将一个字节的数据按位一次性发送到GPIOA的8个管脚?

GPIOA->ODR = cmd;

#define RS_GPIO_Port GPIOB
#define RW_GPIO_Port GPIOB
#define EN_GPIO_Port GPIOB
#define RS_Pin GPIO_PIN_1
#define RW_Pin GPIO_PIN_2
#define EN_Pin GPIO_PIN_10
#define RS_HIGH HAL_GPIO_WritePin(RS_GPIO_Port, RS_Pin, GPIO_PIN_SET)

#define RS_LOW HAL_GPIO_WritePin(RS_GPIO_Port, RS_Pin, GPIO_PIN_RESET)

#define RW_HIGH HAL_GPIO_WritePin(RW_GPIO_Port, RW_Pin, GPIO_PIN_SET)

#define RW_LOW HAL_GPIO_WritePin(RW_GPIO_Port, RW_Pin, GPIO_PIN_RESET)

#define EN_HIGH HAL_GPIO_WritePin(EN_GPIO_Port, EN_Pin, GPIO_PIN_SET)

#define EN_LOW HAL_GPIO_WritePin(EN_GPIO_Port, EN_Pin, GPIO_PIN_RESET)

void Write_Cmd_Func(uint8_t cmd)
{

 RS_LOW;

 RW_LOW;

 EN_LOW;

 GPIOA->ODR = cmd;

 HAL_Delay(5);

 EN_HIGH;

 HAL_Delay(5);

 EN_LOW;
}

void Write_Data_Func(uint8_t dataShow)
{

 RS_HIGH;

 RW_LOW;

 EN_LOW;

 GPIOA->ODR = dataShow;

 HAL_Delay(5);

 EN_HIGH;

 HAL_Delay(5);

 EN_LOW;
}

void LCD1602_INIT(void)
{

//(1)延时 15ms

 HAL_Delay(15);

//(2)写指令 38H(不检测忙信号) 

 Write_Cmd_Func(0x38);

//(3)延时 5ms

 HAL_Delay(5);

//(4)以后每次写指令,读/写数据操作均需要检测忙信号

//(5)写指令 38H:显示模式设置

 Write_Cmd_Func(0x38);

//(6)写指令 08H:显示关闭
 Write_Cmd_Func(0x08);

//(7)写指令 01H:显示清屏

 Write_Cmd_Func(0x01);

//(8)写指令 06H:显示光标移动设置

 Write_Cmd_Func(0x06);

//(9)写指令 0CH:显示开及光标设置}

 Write_Cmd_Func(0x0c);
}

void LCD1602_showLine(char row, char col, char *string)
{

 switch(row){

 case 1:

 Write_Cmd_Func(0x80+col);

 while(*string){

 Write_Data_Func(*string);

 string++;

 }

 break;

 case 2:

 Write_Cmd_Func(0x80+0x40+col);

 while(*string){

 Write_Data_Func(*string);

 string++;

 }

 break;

 }
}

main函数里:

 //char position = 0x80 + 0x05;

 //char dataShow = 'C';

 LCD1602_INIT();

 //Write_Cmd_Func(position);//选择要显示的地址

 //Write_Data_Func(dataShow);//发送要显示的字符

 LCD1602_showLine(1,5,"hello world");

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/404885.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C语言实现快速排序(hoare法、挖坑法、前后指针法与非递归实现)——不看后悔系列

目录 1. hoare法 方法与步骤 代码实现 2. 挖坑法 方法与步骤 代码实现 3. 前后指针法 方法与步骤 代码实现 4. 快速排序的缺点与优化 1.快速排序的缺点 2.快速排序的优化 ① 三数取中法选 key 代码实现 ② 小区间优化 代码实现 5. 快速排序的非递归实现 附录…

数据结构与算法基础(王卓)(16):KMP算法详解(代码实现)

实现代码的过程中 具体细节、问题: (1):关于写Get_next函数的标题: 现象: PPT上写的是: void get_next(SString T, int &next[]) 然而并不能运行,而当我们去掉了引用符号&…

记录踩过的坑-Git

Git命令克隆很慢原命令:git clone -b r1.13.0 https://github.com/tensorflow/models.git现在替换为:git clone -b r1.13.0 https://github.com.cnpmjs.org/tensorflow/models.git也就是把原 URL 中的 github.com 替换为 github.com.cnpmjs.org&#xff…

设计模式第9式:迭代器模式

前言 我们有很多种方法可以把对象集中到一个集合中,比如列表、堆栈、散列表中。每种集合都有自己的特点和使用时机,但都有一个共同的需求:用户想要遍历这些对象。同时我们并不想用户看到集合的实现,本文将讲解如何让用户遍历对象…

JVM垃圾回收器详解

垃圾收集器没有在规范中进行过多的规定,可以由不同的厂商、不同版本的JVM来实现。由于JDK的版本处于高速迭代过程中,因此Java发展至今已经衍生了众多的GC版本。从不同角度分析垃圾收集器,可以将GC分为不同的类型。1、垃圾回收器分类1.1、按线…

国际安全领域顶会NDSS 2023录稿整理 (下)

隐私计算研习社 NDSS是网络和系统安全领域的四大顶级国际学术会议(BIG4)之一,第三十届会议于2023年2月27日到3月3日,在美国圣迭戈举办。本文将接着整理剩余论文,并对论文进行分类,感兴趣的小伙伴可以访问论…

【Linux】信号+再谈进程地址空间

目录 一、Linux中的信号 1、Linux中的信号 2、进程对信号的处理 3、信号的释义 二、信号的捕捉 1、信号的捕捉signal() 2、信号的捕捉sigaction() 三、信号如何产生? 1、kill()用户调用kill向操作系统发送信号 通过命令行参数模仿写一个kill命令 2、rais…

pinctrl和gpio子系统

一、pinctrl子系统简介Linux驱动讲究驱动分离与分层,pinctrl和gpio子系统就是驱动分离与分层思想下的产物,pinctrl子系统主要工作内容如下:获取设备树中的pin信息根据获取到的pin信息来设置pin的复用功能根据获取到的pin信息来设置pin的电气属…

Day914.安全认证架构演进:单块阶段 -SpringBoot与K8s云原生微服务实践

安全认证架构演进:单块阶段 Hi,我是阿昌,今天学习记录的是关于安全认证架构演进:单块阶段的内容。 讲到安全认证的内容,就必然会提到两个点:认证 和 授权。 认证:我是谁授权:我能…

Spring中的事务@Transactional

Transactional可以添加在方法上 添加在方法上时,表示该方法出现了异常或者报错,而导致之前数据库没有进行回滚事件,也就是说如果在方法中,有报错,但是添加了Transactional 则会开始回滚。 Transactional 在异常被捕获…

剑指 Offer 29. 顺时针打印矩阵

剑指 Offer 29. 顺时针打印矩阵 难度:middle\color{orange}{middle}middle 题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字。 示例 1: 输入:matrix [[1,2,3],[4,5,6],[7,8,9]] 输出:[1,2…

【OpenCV技能树】——OpenCV基础

前言: 😊😊😊欢迎来到本博客😊😊😊 目前正在进行 OpenCV技能树的学习,OpenCV是学习图像处理理论知识比较好的一个途径,至少比看书本来得实在。本专栏文章主要记录学习Op…

Apache apisix默认密钥漏洞(CVE-2020-13945)

目录漏洞描述影响版本漏洞复现声明:本文仅供学习参考,其中涉及的一切资源均来源于网络,请勿用于任何非法行为,否则您将自行承担相应后果,本人不承担任何法律及连带责任。漏洞描述 Apache APISIX 是一个动态、实时、高…

LearnOpenGL-模型加载-3.渲染模型

本人刚学OpenGL不久且自学,文中定有代码、术语等错误,欢迎指正 我写的项目地址:https://github.com/liujianjie/LearnOpenGLProject 文章目录模型加载重要代码读取3D模型递归处理结点的网格加载纹理优化程序代码Model类加载模型流程例子1&…

【运筹优化】拉格朗日松弛 次梯度算法求解整数规划问题 + Java调用Cplex实战

文章目录一、拉格朗日松弛二、次梯度算法三、案例实战一、拉格朗日松弛 当遇到一些很难求解的模型,但又不需要去求解它的精确解,只需要给出一个次优解或者解的上下界,这时便可以考虑采用松弛模型的方法加以求解。 对于一个整数规划问题&…

Python交通标志识别基于卷积神经网络的保姆级教程(Tensorflow)

项目介绍 TensorFlow2.X 搭建卷积神经网络(CNN),实现交通标志识别。搭建的卷积神经网络是类似VGG的结构(卷积层与池化层反复堆叠,然后经过全连接层,最后用softmax映射为每个类别的概率,概率最大的即为识别…

chromium 50 chromium57 版本编译启用 widevine 功能

本实验中 chrome 版本为 57.0.2987.98 01 chromium57 在 win11 版本中启用 widevine 功能 01.01 启用 enable_widevine 选项生成 widevine 相关动态库 在chromium 57 版本中,编译时秩序设置 enable_widevinetrue 即可生成 widevinecdm.dll 和 widevinecdmadapter…

windows下maven更新/安装

写在前面: 我的maven学习的比较早,后面windows文件管理分类的时候,把学习用的全部在一个文件夹了,而这个又不好移动进去。 正好也更新一下maven的版本了,不过和重新安装好像差不多了。 现在写的windows的以后,在看看要…

若依框架部署从零开始2023版(前后端分离)

前言电脑最近重装了一次系统,目前什么都没有安装,记录一下从零开始部署前后端分离版本的若依框架系统先去官网把若依源码拉下来代码克隆若依目前已经有很多的版本了,因为现在开发比较流行前后端分离,因此这里演示前后端分离版本点…

外卖点餐系统小程序 PHP+UniAPP

一、介绍 本项目是给某大学餐厅开发的外面点餐系统,该项目针对校内的学生,配送由学校的学生负责配送。因此,该项目不同于互联网的外卖点餐系统。 该系统支持属于 Saas 系统,由平台端、商家端、用户端、以及配送端组成。 其中&a…