C51中,HZK16汉字库的使用(mydows's Blog转载)
定义如下:
unsigned char str[]=""
在运行时str被初始化为2个字节长度,内容为GBK码,为:0xCE(区码),0xD2(位码)。
使用如下换算公式得到HZK16文件中的地址,从该位置开始的顺序32字节为的字模。
    ADD=【(区码-0xa1单片机printf函数×0x5e + (位码-0xa1)】×0x20
按照上面的计算方法,的字模地址:0x216E0 。他的C语言字模为:0x04,0x80,0x0E,0xA0,0x78,0x90,0x08,
0x90,0x08,0x84,0xFF,0xFE,0x08,0x80,
0x08,0x90,0x0A,0x90,0x0C,0x60,0x18,
0x40,0x68,0xA0,0x09,0x20,0x0A,0x14,
0x28,0x14,0x10,0x0C
        HZK16字库是符合GB2312标准的16×16点阵字库,HZK16GB2312-80支持的汉字有6763个,符号682个。其中一级汉字有3755个,按声序排列,二级汉字有3008个,按偏旁部首排列。我们在一些应用场合根本用不到这么多汉字字模,所以在应用时就可以只提取部分字体作为己用。
HZK16字库里的16×16汉字一共需要256个点来显示,也就是说需要32个字节才能达到显示一个普通汉字的目的。
我们知道一个GB2312汉字是由两个字节编码的,范围为A1A1~FEFEA1-A9为符号区,B0F7为汉字区。每一个区有94个字符(注意:这只是编码的许可范围,不一定都有字型对应,比如符号区就有很多编码空白区域)。下面以汉字为例,介绍如何在HZK16文件中到它对应的32个字节的字模数据。
前面说到一个汉字占两个字节,这两个中前一个字节为该汉字的区号,后一个字节为该字的位号。其中,每个区记录94个汉字,位号为该字在该区中的位置。所以要到hzk16库中的位置就必须得到它的区码和位码。(为了区别使用了区码和区号,其实是一个东西,别被我误导了)
区码:区号(汉字的第一个字节)-0xa0    (因为汉字编码是从0xa0区开始的,所以文件最前面就是从0xa0区开始,要算出相对区码)
位码:位号(汉字的第二个字节)-0xa0
这样我们就可以得到汉字在HZK16中的绝对偏移位置:
offset=(94*(区码-1)+(位码-1))*32
注解:1、区码减1是因为数组是以0为开始而区号位号是以1为开始的
            2(94*(区号-1)+位号-1)是一个汉字字模占用的字节数
          3、最后乘以32是因为汉字库文应从该位置起的32字节信息记录该字的字模信息(前面提到一个汉字要有32个字节显示)
有了偏移地址就可以从HZK16中读取汉字编码了,剩下的就是文件操作了,就不说了,要看代码(汉字)就是下面的:hzk16汉字库的简单读写程序,是一个最简单的c语言程序。
                                                hzk16汉字库的简单读写程序
hzk汉字点阵   
   
  int    i,j,k;   
  unsigned    char    incode[3]="";    //    要读出的汉字   
  unsigned    char    qh,wh;   
  unsigned    long    offset;   
  //    占两个字节,    取其区位号   
  qh    =    incode[0]    -    0xa0;/ /获得区码           
  wh    =    incode[1]    -    0xa0;  / /获得位码               
  offset    =    (94*(qh-1)+(wh-1))*32;          /    *得到偏移位置*    /   
     
  FILE    *HZK;   
  char    mat[32];   
  if((HZK=fopen("hzk16",    "rb"))    ==    NULL)   
  {   
  printf("Can't    Open    hzk16\n");   
  exit(0);   
  }   
  fseek(HZK,    offset,    SEEK_SET);   
  fread(mat,    32,    1,    HZK);
//显示
for(j=0;j<16;j++)
        for(i=0;i<2;i++)
            for(k=0;k<8;k++)
                if(mat[j][i]&(0x80>>k)) /*测试为1的位则显示*/
                    {
                            printf("%s",'#');
                        }else{  printf("%s",'-');
                              }

    fclose(HZK);
    fclose(fp);
                                      图形LCD模块ACM19264ASB的汉字显示
1引言
在基于单片机的智能系统中,汉字显示模块是很重要的一个组成部分,它应用广泛、操作容易、调试简便。
然而,在单片机上显示汉字也存在几个问题。首先,单片机资源有限,我们不能为了显示汉字占用太多的资源;其次,汉字存储读取比较繁琐,使用不方便;第三,汉字是通过点阵显示出来的,往往与LCD写入方式不一样,这就得进行转换和调整。
值得注意的是,基于单片机的汉字显示不能在字符LCD上实现。使用图形LCD有很多优点,
不仅能显示汉字,而且可以实现汉字动态移动和上下滚屏,实现汉字与图形的混合显示,同时功耗低。
2基于单片机的汉字显示原理
2.1汉字字模
汉字一般是以点阵式存储的,如16×1624×24点阵(即汉字的字模),每个汉字由32字节(16点阵)或72字节(24点阵)描述。根据汉字的不同字体,也可分为宋体字模、楷体字模、黑体字模等等。
汉字的字模其实是汉字字形的图形化。对于16点阵字模,就是把汉字写在一个16×16的网格内,汉字的笔画能过某网格时该网格就对应1,否则该网格对应0,这样每一网格均对应10,把对应1的网格连起来看,就是这个汉字。汉字就是这样通过字节表示点阵存储在字库中的。
为了方便查所需汉字的点阵,每个汉字都与一个双字节的内码一一对应。通过汉字的内码可以计算出它的点阵起始字节。现以16点阵为例说明。
先由内码计算出它在汉字库中的区位码,计算公式为:
区码=内码第一字节-160
位码=内码第二字节-160
再由区位码可以得到它在汉字库中字模第一个字节的位置:
(区码×94+位码)×32 于是,可以向后连续读出由32个字节组成的该字的点阵数据。
2.2汉字显示
汉字占用资源太多(如16点阵,每个汉字就需32字节),因而通常把汉字库放在EEPROM里,需要显示某个汉字时,先算出它的区位码,再求出点阵起始位置,从EEPROM中顺序调出该字的点阵数据,存在缓冲区里,最后依次送往LCD显示,描出该字。需要说明的是汉字存储方式与LCD显示方式有一定差别。
本文使用另一种显示方法,即事先将程序用到的汉字、符号和数码(为了节省显示空间,可以将数码压成8×16点阵),编成一个文本文件,用一段小程序做出相应小的汉字库,这个小
字库的汉字点阵数据取自于一般汉字库。再经过转换和调整,得到新的汉字库,最后把新字库固化在EEPROM中。单片机只需按序号读出点阵字节,送往LCD即可显示所需汉字。减轻了单片机的负担,去除了繁琐的查内码、求起始位置、转换、调整等工作,提高了系统可靠性。
116点阵汉字字库存储方式
3自定义小字库的制作
典型的汉字库可选用UCDOS下的字库,如16点阵字库HZK16。需要256K空间,用了较大的EEPROM,又不方便读取,而实际应用中需要的汉字又非常少,因而我们可以自己制作小的汉字库,在这个小字库里只包含系统需要的汉字。这样,一方面节省读取时间,另一方面大大地节省了资源。
限于篇幅,这里仅仅给出流程图(假定事先将所需汉字写到了一个文本文件),如图1所示。
将整个汉字字库存放在EPROME2PROM内,程序根据要显示汉字的机内码来调用汉字字模。
某些高端单片机,如MotorolaM68300系列32位单片机,寻址范围可达8M,液晶显示常用的16×16汉字库二进制数据文件为两百多k,将汉字字库存入大容量的E2PROM,通过地址线可寻址到汉字库中的每一个汉字。