lcd1602能显示汉字吗_lcd1602显示汉字程序

lyj159 发表于 2017-10-19 09:13:03 收藏 已收藏
赞(0) •  评论(0

lcd1602能显示汉字吗_lcd1602显示汉字程序

lyj159 发表于 2017-10-19 09:13:03

  lcd1602能否显示汉字

lcd1602能显示汉字吗_lcd1602显示汉字程序

  1602液晶是字符型液晶,总共有2行,每行16个字符。每个字符有效的可显示字体是5*8点阵字体。所以通常无法显示中文。中文的显示必须选择图形点阵型液晶。

  特别地:是指,显示“年月日”,之类的简单的汉字吧?最多支持8个CGRAM字符。可以自己做。调用字模代码0-7。如:“年”的CGRAM单元数据为:0FH,09H,0FH,09H,0FH,09H,13H,00H。这8个字节写入CGRAM单元00H-07H。调用字符代码0。可以看到中文“年”。

  lcd1602显示汉字的概念

lcd1602能显示汉字吗_lcd1602显示汉字程序

  首先:我们要弄清楚一个概念,其实我们使用1602显示中文汉字,就是利用其内部的自定义字节空间。

  第一行的内容对应DDRAM中80H到8FH内容 第二行的内容对应DDRAM中C0H到CFH内容

  通过查看LCD1602的用户手册,里面提到当我们需要想CGRAM写入数据时,我们需要先写入CGRAM的地址,从手册上可知,CGRAM的地址为0x40.所以当我们要写入数据时,要先写入一条地址命令 write_com(0x040);

  然后把你要写入的数据通过写数据命令写入 write_date( 你要写得数据);

  把你要显示汉字的代码存放在CGRAM里,就相当于其他常用字符一样固定在LCD里面了,把你要显示汉字的代码存放在CGRAM里,就相当于其他常用字符一样固定在LCD里面了,就相当于把你要显示汉字的代码存放在CGRAM里,就相当于其他常用字符一样固定在LCD里面了,我们都知道LCD1602内部有一个64字节的自定义CGRAM,那么好,我们就是要利用自定义字节来达到显示汉字的效果。

  我们知道一个字符占8个字节,所以在CGRAM能够存放8个字符。 也就是说,这个64字节的CGRAM是给我们提供写入数据的, 那么我们怎么把我们要的汉字写入CGRAM呢?

  DDRAM显示用ram,直接和屏幕上的点相对应。屏幕上的一个点和ddram中的一个位对应,字符屏的ddram和图形屏的ddram有一点点区别。

  CGROM 中存储了一些标准的 字符的字模编码,是液晶屏出厂时固化在控制芯片中的,用户不能改变其中的存储内容,只能读取调用,包含有标准的ASCII码、日文字符和希腊文字符。

  CGRAM 是控制芯片留给用户,用以存储用户自己设计的字模编码。

  字模编码都要先被读取到对应的DDRAM中,经中转以后,屏幕的相应位置才显示出字符。对于字符屏,要显示某个字符时,往ddram里写字符的索引(一般都是ascii码)就可以完成显示。

  lcd1602显示汉字的程序

lcd1602能显示汉字吗_lcd1602显示汉字程序

  汉字的字模:

  就是显示出来的汉子字体结构。

  要定义字模就需要知道1602的CG RAM地址和显示字符的地址 DD RAM :

   1    2    3   4   5    6   7   8   9  10 11  12 13 14  15 16

  00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F

  40 41 42 43 44 45 46 47 48 49 4A 4B 4C 4D 4E 4F

  第一行第一个字符的地址是( 0x80) ,后面的字符地址以此类推

  第二行第一个字符的地址为 ( 0x80 + 0x40),需要在第一行的基地址后面加上一个偏移 量0x40,后面的地址在此基址上再依次加一。

  void delay(unsigned int time_i)

  {

  int i;

  for (;tim》0;tim--)

     {

  for (i=0;i《10;i++);

     }

  }

  void command_out (char out_data)

  {

  RW=0;

  RS=0;

  EN=1;

  DB=out_data;

  EN=0;

  delay(10);

  }

  void OUTD(char out_data)

  {

  RS=1;

  RW=0;

  EN=1;

  DB= out_data;

  EN=0;

  delay(10);

  }

  OUTI()是写入指令的函数, OUTD()是写入数据的函数。 如:

  左上角第一个5*7的显示数据为:

  0x00,0x01,0x81,0x12,0x24,0x41,0x81,0x10 蓝色区域的相应位为1

  这样就可以写出全部4个5*7点阵的显示数据了,把它封装在一个数组中,如数组,

  hanzi[]={0x00,0x01,0x81,0x12,0x24,0x41,0x81,0x10, …… …… …… }

  之后利用命令:

  OUTI(0X40);

  for (i=0;i《64;i++)

  {

  OUTD(neu[i]);

  }

  将这个数组中的数据赋值到1602内部CGRAM中。完成后,再设置数据在1602上的 显示地址,根据图1的结构,设置好位置,如在第一行第一个字符和第二个字符以及第二行

  第一个字符和第二个字符处显示:

  Command_out (0x80+0x00);

  for (i=0;i《4;i++)

  {

  data_out (i);

  }

  Command_out(0x80+0x40);

  for (i=4;i《8;i++)

  { data_out(i);

  }

  command_out()的作用是设置显示的起始地址,data_out()的作用是将hanzi[]数组中的16 进制数据发送出去,每一个16进制数字对应之5*7点阵上的每一行内容。第1行第1个字 符需要8个16进制数据,第1行第2个字符处也需要816进制数据。还有第2行的两个字 符点阵,这样显示一个汉字需要32个16进制数据。

  下面列举一个实例(proteus仿真通过)

  //*************************************************************************************** //

  硬件连接:1602VDD接5V,VO接地,BL1接5V,BL2接地,8根数据线接P0口,RS RW E分别接P2.0、P2.1、P.4口

  //***************************************************************************************//

  #include 《reg52.h》

  #include 《string.h》

  #define Busy 0x80 //用于检测LCM状态字中的Busy标识

  #define LCM_Data P0 ;

  sbit LCM_RS=P2^0; //寄存器选择

  sbit LCM_RW=P2^1; //读/写控制

  sbit LCM_E=P2^4; //读/写使能

  int i,j;

  //自定义字符列表 //

  =====================================================================================

  unsigned char character0[8] = {0x08,0x0f,0x12,0x0f,0x0a,0x1f,0x02,0x02}, //年

  character1[8] = {0x0f,0x09,0x0f,0x09,0x0f,0x09,0x0b,0x11}, //月

  character2[8] = {0x0f,0x09,0x09,0x09,0x0f,0x09,0x09,0x0f}, //日

  characterN[8] = {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}; //日

  //==========================================================

  //=========================================================

  //延时程序

  //=======================================================

  void Delay5Ms(void)

  {

  unsigned long int TempCyc = 5552;

  while(TempCyc--);

  }

  void Delay400Ms(void)

  {

  unsigned char TempCycA = 5;

  unsigned int TempCycB;

  while(TempCycA--)

     {

  TempCycB=7269;

  while(TempCycB--);

     };

  }

  //===================================================================

  //读写子程序

  //=================================================================

  //读数据

  unsigned char ReadDataLCM(void)

  {

  LCM_RS = 1;

  LCM_RW = 1;

  LCM_E = 1;

  LCM_E = 0;

  for(i=0;i《100;i++);

  LCM_E = 1;

  return(LCM_Data);

  }

  //读状态

  unsigned char ReadStatusLCM(void)

  {

  LCM_Data = 0xFF;

  LCM_RS = 0;

  LCM_RW = 1;

  LCM_E = 1;

  LCM_E = 0;

  for(i=0;i《100;i++);

  LCM_E = 1;

  //while (LCM_Data & Busy); //检测忙信号

  return(LCM_Data);

  }

  //写数据

  void WriteDataLCM(unsigned char WDLCM)

  {

  ReadStatusLCM(); //检测忙

  LCM_Data = WDLCM;

  LCM_RS = 1;

  LCM_RW = 0;

  LCM_E = 1;

  LCM_E = 0; //若晶振速度太高可以在这后加小的延时

  for(i=0;i《100;i++);//延时

  LCM_E = 1;

  }

  //写指令

  void WriteCommandLCM(unsigned char WCLCM,BuysC) //BuysC为0时忽略忙检测

  {

  if (BuysC) ReadStatusLCM(); //根据需要检测忙

  LCM_Data = WCLCM;

  LCM_RS = 0;

  LCM_RW = 0;

  LCM_E = 1;

  LCM_E = 0;

  for(i=0;i《100;i++);

  LCM_E = 1;

  }

  //===========================================================

  //初始化子程序

  //===================================================================

  void LCMInit(void) //LCM初始化

  {

  LCM_Data = 0;

  WriteCommandLCM(0x38,0); // 三次显示模式设置,不检测忙信号

  Delay5Ms();

  WriteCommandLCM(0x38,0);

  Delay5Ms();

  WriteCommandLCM(0x38,0);

  Delay5Ms();

  WriteCommandLCM(0x38,1); // 显示模式设置,开始要求每次检测忙信号

  Delay5Ms();

  WriteCommandLCM(0x08,1); // 关闭显示

  Delay5Ms();

  WriteCommandLCM(0x01,1); // 清屏 Delay5Ms();

  WriteCommandLCM(0x06,1); // 显示光标移动设置

  Delay5Ms();

  WriteCommandLCM(0x0c,1); // 显示开及光标设置

  Delay5Ms();

  }

  //================================================================

  //按指定位置显示一个字符

  //===================================================================

  void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData)

  {

  Y &= 0x1;

  X &= 0xF; //限制X不能大于15,Y不能大于1

  if (Y) X |= 0x40; //当要显示第二行时地址码+0x40;

  X |= 0x80; //算出指令码

  WriteCommandLCM(X, 0); //这里不检测忙信号,发送地址码

  WriteDataLCM(DData);

  }

  //=================================================================

  //按指定位置显示一串字符

  //void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData)

  //说明: x(0-15):x参数 y(0-1):y参数 DData(字符串):要显示的内容(英文、数字、符号)

  //========================================================================

  void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData)

  {

  unsigned char ListLength,j; ListLength = strlen(DData);

  Y &= 0x1;

  X &= 0xF; //限制X不能大于15,Y不能大于1

  if (X 《= 0xF) //X坐标应小于0xF

     {

  for(j=0;j《ListLength;j++)

        {

  DisplayOneChar(X, Y, DData[j]); //显示单个字符 X++;

        }

     }

  }

  //==========================================================

  //显示自定义字符

  //void mychar(char xx,char yy,unsigned char *character,unsigned char saveto)

  //说明:xx(0-15):为x参数.yy(0-1):y参数.character:要显示的字符的列表地址,在程序前面有定义

  //saveto(1-7)为字符保存的RAM,每屏最多显示7个自定义字符 //(0x00-0x0h是自定义字符)

  //=============================================================

  void mychar(char xx,char yy,unsigned char *character,unsigned char saveto)

  {

  unsigned char add = (saveto《《3) | 0x40;

  unsigned char t; //临时变量,每一行的值 /*

  t=*(character+0);

  WriteCommandLCM(add, 0); //第1行 WriteDataLCM(t); t=*(character+1);

  WriteCommandLCM(add+1, 0); //第2行 WriteDataLCM(t); t=*(character+2);

  WriteCommandLCM(add+2, 0); //第3行 WriteDataLCM(t); t=*(character+3);

  WriteCommandLCM(add+3, 0); //第4行 WriteDataLCM(t); t=*(character+4);

  WriteCommandLCM(add+4, 0); //第5行 WriteDataLCM(t); t=*(character+5);

  WriteCommandLCM(add+5, 0); //第6行 WriteDataLCM(t); t=*(character+6);

  WriteCommandLCM(add+6, 0); //第7行 WriteDataLCM(t); t=*(character+7);

  WriteCommandLCM(add+7, 0); //第8行 WriteDataLCM(t); */

  for(i = 0;i《8;i++)

  {

  WriteCommandLCM(add+i, 0);

  WriteDataLCM(*(character+i));

  }

  DisplayOneChar(xx,yy,saveto); //显示字符 }

  //===============================================================

  //主函数

  //=========================================================

  main()

  {

  Delay400Ms();

  LCMInit();

  Delay400Ms(); //1602初始化

  while(1)

     {

  DisplayListChar(0,0,“This is ListChar”); DisplayListChar(0,1,“!”);

  for(j=0;j《30;j++)for(i=0;i《30000;i++); WriteCommandLCM(0x01,1); //清屏

  Delay5Ms();

  DisplayListChar(0,0,“This is OneChar:”);

  DisplayOneChar(0,1,0x4f);

  DisplayOneChar(1,1,0x6e);

  DisplayOneChar(2,1,0x65);

  DisplayOneChar(3,1,0x21);

  for(j=0;j《30;j++)for(i=0;i《30000;i++);

  WriteCommandLCM(0x01,1); //清屏

  Delay5Ms();

  DisplayListChar(0,0,“This is MyChar:”);

  mychar(0,1, character0,1);

  mychar(1,1, character1,2);

  mychar(2,1, character2,3);

  for(j=0;j《30;j++)for(i=0;i《30000;i++);

  WriteCommandLCM(0x01,1); //清屏 Delay5Ms();

     }

  }

收藏

相关话题
文章来源栏目
+关注

评论(0)

加载更多评论

参与评论

相关文章

分享到

QQ空间 QQ好友 微博
取消