单片机 中关于LCD 显示 的程序 中有 write_com(0x80+17); 写在液晶数据缓冲区第一行 write_com(0xc0+17

单片机 中关于LCD 显示 的程序 中有 write_com(0x80+17); 写在液晶数据缓冲区第一行 write_com(0xc0+17,第1张

这个是1602的程序,每行16个显示单元,第一行地址从0x80~0x80+16,第二行0xc0~0xc0+16,销团一个字符占一个地址,其它的地饥斗姿址也是有存储单元的,只不过不能显示出来,你向1602发送一条移位指烂绝令就看出来了。也不是每个地址都有存储单元,具体不记得了,你试试就知道了。

** 1602A液晶屏驱动程序** 晶 振 频 率:11.0592M*****************************************/#include <reg51.h>#define uchar unsigned char#define uint unsigned int sbit LCM_E=P3^5//定义接口sbit LCM_RW=P3^6sbit LCM_RS=P3^7#define LCM_Data P1//数据接口 void LCM_WriteData(uchar WDLCM)void LCM_WriteCommand(uchar WCLCM,BuysC)uchar LCM_ReadData(void)uchar LCM_ReadStatus(void)void LCM_Init(void)void DisplayOneChar(uchar X, uchar Y, uchar DData)void DisplayListChar(uchar X, uchar Y, uchar code *DData)void Delay5Ms(void)void Delay400Ms(void)uchar code uctech[] = {"1602A"}uchar code net[] = {"www.52c51.com"}void main(void){Delay400Ms()//启动等待,等LCM讲入工作状态LCM_Init()//LCM初始化Delay5Ms()//延时片刻(可不要) DisplayListChar(6, 1, uctech)DisplayListChar(0, 0, net)LCM_ReadData()//测试用句无意义LCM_Data=255while(1)} /****************************1602A写数据函数**************************//*函数原型:void LCM_WriteData(uchar WDLCM)/*函数功能:1602A写数据/*输入参数:要写入的数据/*输出参数:无/*调用模块:/**********************************************************************/ void LCM_WriteData(uchar WDLCM){LCM_ReadStatus()//检测忙LCM_Data = WDLCMLCM_RS = 1LCM_RW = 0LCM_E = 0//若晶振速度太高可以在这后加小的延时LCM_E = 0//延盯正时LCM_E = 1} /****************************1602A写指令函数**************************//*函数原型:void LCM_WriteCommand(uchar WCLCM,BuysC)/*函数迟则昌功能:1602A写指令/*输入参数:要写入的指令/*输出参数:无/*调用模块:/**********************************************************************/ void LCM_WriteCommand(uchar WCLCM,BuysC)//BuysC为0时忽略忙检测{if (BuysC) LCM_ReadStatus()//根据需要检测忙码扒LCM_Data = WCLCMLCM_RS = 0LCM_RW = 0LCM_E = 0LCM_E = 0LCM_E = 1} /****************************1602A读数据函数**************************//*函数原型:uchar LCM_ReadData(void)/*函数功能:1602A读数据/*输入参数:无/*输出参数:1602A返回的数据/*调用模块:/**********************************************************************/ uchar LCM_ReadData(void){LCM_RS = 1LCM_RW = 1LCM_E = 0LCM_E = 0LCM_E = 1return(LCM_Data)} /****************************1602A读状态函数**************************//*函数原型:uchar LCM_ReadData(void)/*函数功能:1602A读状态/*输入参数:无/*输出参数:1602A返回的状态/*调用模块:/**********************************************************************/ uchar LCM_ReadStatus(void){LCM_Data = 0xFFLCM_RS = 0LCM_RW = 1LCM_E = 0LCM_E = 0LCM_E = 1while (LCM_Data &0x80)//检测忙信号return(LCM_Data)} /****************************1602A初始化函数**************************//*函数原型:void LCM_Init(void)/*函数功能:1602A初始化/*输入参数:无/*输出参数:无/*调用模块:/**********************************************************************/ void LCM_Init(void){LCM_Data = 0LCM_WriteCommand(0x38,0)//三次显示模式设置,不检测忙信号Delay5Ms()LCM_WriteCommand(0x38,0)Delay5Ms()LCM_WriteCommand(0x38,0)Delay5Ms()LCM_WriteCommand(0x38,1)//显示模式设置,开始要求每次检测忙信号LCM_WriteCommand(0x08,1)//关闭显示LCM_WriteCommand(0x01,1)//显示清屏LCM_WriteCommand(0x06,1)// 显示光标移动设置LCM_WriteCommand(0x0C,1)// 显示开及光标设置} /******************1602A按指定位置显示一个字符函数**********************//*函数原型:void DisplayOneChar(uchar X, uchar Y, uchar DData)/*函数功能:1602A按指定位置显示一个字符/*输入参数:X坐标 Y坐标 要显示的字符/*输出参数:无/*调用模块:/**********************************************************************/ void DisplayOneChar(uchar X, uchar Y, uchar DData){Y &= 0x1X &= 0xF//限制X不能大于15,Y不能大于1if (Y) X |= 0x40//当要显示第二行时地址码+0x40X |= 0x80// 算出指令码LCM_WriteCommand(X, 0)//这里不检测忙信号,发送地址码LCM_WriteData(DData)} /*******************1602A按指定位置显示一串字符函数*********************//*函数原型:void DisplayListChar(uchar X, uchar Y, uchar code *DData)/*函数功能:1602A按指定位置显示一个字符/*输入参数:X坐标 Y坐标 要显示字符串的首地址/*输出参数:无/*调用模块:/**********************************************************************/ void DisplayListChar(uchar X, uchar Y, uchar code *DData){uchar ListLengthListLength = 0Y &= 0x1X &= 0xF//限制X不能大于15,Y不能大于1while (DData[ListLength]>0x20)//若到达字串尾则退出{if (X <= 0xF)//X坐标应小于0xF{DisplayOneChar(X, Y, DData[ListLength])//显示单个字符ListLength++X++}}} /********************5ms延时函数************************//*函数原型:void Delay5Ms(void)/*函数功能:5ms延时/*输入参数:无/*输出参数:无/*调用模块:/**********************************************************************/ void Delay5Ms(void){unsigned int TempCyc = 5552while(TempCyc--)} /********************400ms延时延时函数************************//*函数原型:void Delay400Ms(void)/*函数功能:400ms延时延时/*输入参数:无/*输出参数:无/*调用模块:/**********************************************************************/ void Delay400Ms(void){uchar TempCycA = 5unsigned int TempCycBwhile(TempCycA--){TempCycB=7269while(TempCycB--)}}

①问,RS,RW,E 的设置为 读状态时序 的过程,读出来的数据(数据口P1) 的最高位 刚好即为 液晶的 忙碌 标志位。

②问,writecontrol(unsigned condata) 函数 是个 写指令 函数;

③问,空 *** 作 是为了让数据稳定后,才使能 液晶接收数据;

④问,写指令有时间 间隔要求,太频繁的读写会丢码,所以要加延时,三条38指令,是因为安全起见,上电后马上初始化会出现液晶电源不稳而丢码;

⑤问,液晶显示 只在 更新数据 的时候刷新一次即可,液晶会维持内容,无需重复刷新。

⑥附送,③和④问,都是驱动太恶心而造成的结果,好的驱动程序不需如此 *** 作。液晶的时序要求为ns级,单片机的指令周期普遍为us级,根本不需过多累赘。

#include"reg52.h" //包含52头文件

#include"SMC1602A.h" //包含SMC1602A宏定义文件

#define BusyReadCount 10 //读忙标志等待次数

#define SMC1602_Data P0 //定义 数据接口

//sbit SMC1602_VO=P2^4 //定义 VO对比度接口

sbit SMC1602_RW=P2^5 //定义 R/W接口25

sbit SMC1602_RS=P2^6 //定义 RS接口26

sbit SMC1602_E=P2^7 //定义 E接口27

#define SMC1602_En SMC1602_E=1 //使能

#define SMC1602_Dis SMC1602_E=0 //禁止

uchar SMC1602_Read(bit read_type) //1602液晶屏读函数

{

uchar read_data

SMC1602_Dis //禁止使能

SMC1602_RW=ReadOperate //读 *** 作

SMC1602_RS=read_type //读类型:0状态,1数据

SMC1602_En //开启使能

read_data=SMC1602_Data //存储结果

SMC1602_Dis //禁止使能

return read_data //返回结果

}

void SMC1602_WriteByte(bit write_type,uchar write_data) //1602液晶屏读函数

{

uchar i=BusyReadCount

for(ii--) //延时 *** 作,为写 *** 作预留回复时间

while((SMC1602_Read(CommOperate)&BusyState) &&(++i<=BusyReadCount)) //读取忙标志(BusyReadCount次),若均忙中,则不再读取忙标志,直接执行写 *** 作

//while(SMC1602_Read(CommOperate)&BusyState) if(++i>BusyReadCount) return //读取忙标志,若BusyReadCount次均忙中,则不进行写 *** 作

//while(SMC1602_Read(CommOperate)&BusyState) //禅铅等待空闲(死等)

SMC1602_Dis //禁止使能

SMC1602_RW=WriteOperate //写 *** 作

SMC1602_RS=write_type //写类型:0指令,1数据

SMC1602_Data=write_data //写 *** 作,将 *** 作数送的数据口

SMC1602_En //开启使能

SMC1602_Dis //禁止使能

}

void SMC1602_WriteCGRAM(uchar *write_buf,uchar start_loca,uchar word_num,uchar start_addr) //SMC1602写CGRAM函数,用于自定义字符

{

uchar i,j

write_buf+=start_loca //指向"需写入数据数组"的起始位置

SMC1602_WriteByte(CommOperate,CGRAMAddr|start_addr<<3) //写CGRAM *** 作,并将CGRAM起始地址设为 start_addr

for(j=0j<word_numj++) //自定义字符数量

for(i=0i<8i++) SMC1602_WriteByte(DataOperate,*write_buf++) //写入贺衫好一个自定义字符塌春8个字节数据

}

void SMC1602_Init() //1602液晶屏初始化函数

{

uint i

SMC1602_WriteByte(CommOperate,DisplayMode) //显示模式设置:16×2显示,5×7点阵,8位数据接口

SMC1602_WriteByte(CommOperate,ScreenMode|ScreenOn) //光标模式设置:开启整体显示,开启光标显示,开启光标闪烁

SMC1602_WriteByte(CommOperate,InputMode) //输入方式设置:关闭整屏移动,开启光标正移动(+1)

SMC1602_WriteByte(CommOperate,CleanLCD) //清屏,复位光标

SMC1602_WriteByte(CommOperate,FirstCol) //定位第一行

for(i=150ii--) //等待电源稳定,否则写CGRAM数据(自定义字符)时容易丢失,uint执行周期长,用uchar将会缩短时间,不足以稳定LCD

}


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/12402142.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-25
下一篇 2023-05-25

发表评论

登录后才能评论

评论列表(0条)

保存