纵向取模需要自己倒叙吗_关于iOS 蓝牙 字库 发送16*16汉字点阵取模相关问题(一)...

在与硬件的蓝牙通讯过程中,硬件需要手机端发送16*16的汉字点阵过去,用于硬件直接显示汉字。这里采用的是通用的HZK16(宋体)汉字

原理

1.HZK16字库是符合GB2312国家标准的16×16点阵字库,HZK16的GB2312-80支持的汉字有6763个,符号682个。其中一级汉字有 3755个,按声序排列,二级汉字有3008个,按偏旁部首排列。

我们在一些应用场合根本用不到这么多汉字字模,所以在应用时就可以只提取部分字体作为己用。 HZK16字库里的16×16汉字一共需要256个点来显示,也就是说需要32个字节才能达到显示一个普通汉字的目的。

我们知道一个GB2312汉字是由两个字节编码的,范围为0xA1A1~0xFEFE。A1-A9为符号区,B0-F7为汉字区。每一个区有94个字符(注意:这只是编码的许可范围,不一定都有字型对应,比如符号区就有很多编码空白区域)。

下面以汉字「我」为例,介绍如何在HZK16文件中找到它对应的32个字节的字模数据。

前面说到一个汉字占两个字节,这两个中前一个字节为该汉字的区号,后一个字节为该字的位号。其中,每个区记录94个汉字,位号为该字在该区中的位置。所以要找到「我」在hzk16库中的位置就必须得到它的区码和位码。

区码:汉字的第一个字节-0xA0,因为汉字编码是从0xA0区开始的,所以文件最前面就是从0xA0区开始,要算出相对区码

位码:汉字的第二个字节-0xA0

这样我们就可以得到汉字在HZK16中的绝对偏移位置:offset = (94*(区码-1)+(位码-1))*32。

注解:

区码减1是因为数组是以0为开始而区号位号是以1为开始的

(94*(区号-1)+位号-1)是一个汉字字模占用的字节数

最后乘以32是因为汉字库文应从该位置起的32字节信息记录该字的字模信息(前面提到一个汉字要有32个字节显示)

“我”字的图示

下面开始说明手机端如何发送(这里已OC为例,其他语言可自行进行修改)

首先需要先将UTF-8编码转换成 cp936 编码,然后才能正确地索引字库中的汉字。-(Byte *)convertStringToGBKStr:(NSString *)str{

//转换成UTF-8

NSStringEncoding enc = CFStringConvertEncodingToNSStringEncoding(kCFStringEncodingGB_18030_2000);    NSData *data = [str dataUsingEncoding: enc];    Byte * bytes = (Byte *)[data bytes];    return bytes;

}

首先要确定硬件那边的取模方式

一般的取模方式

下面的方法执行完之后 会打印出来二进制 横向扫描的0,1的点阵

注意:这个0.1点阵字节为正序的。而实际硬件需要的字节需要倒序- (void)sendText:(NSString *)str{

FILE* fphzk = NULL;

int offset;

unsigned char buffer[32];

unsigned char key[8] = { 0x80,0x40,0x20,0x10,0x08,0x04,0x02,0x01 };

unsigned char word[3]; // 改成你的转码后的汉字编码//

//文件路径

NSString *path = [[NSBundle mainBundle] pathForResource:@"HZK16" ofType:@""];    fphzk = fopen(path.UTF8String, "rb");

if(fphzk == NULL){

fprintf(stderr, "error hzk16\n");

return;

}

//循环解码

NSMutableString *sendStr = [NSMutableString string];

for (int i=0; i

NSString *endStr = [str substringWithRange:NSMakeRange(i, 1)];

Byte *bytes = [self convertStringToGBKStr:endStr];        //汉字ascii编码        //0xCE, 0xD2

word[0] = bytes[0];

word[1] = bytes[1];

offset = (94*(unsigned int)(word[0]-0xa0-1)+(word[1]-0xa0-1))*32;

fseek(fphzk, offset, SEEK_SET);

fread(buffer, 1, 32, fphzk);

for(k=0; k<16; k++){

for(j=0; j<2; j++){

for(i=0; i<8; i++){

flag = buffer[k*2+j]&key[i];

printf("%s", flag?"1":"0");

}

}

printf("\n");

}

fclose(fphzk);

fphzk = NULL;

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值