串口传输文件与点阵汉字的字模读取与显示

一、连接串口

本次实验两个USB TO TTL,将两个USB TO TTL的RX,TX引脚交叉连接,并将两个USB接口接上一台笔记本电脑(这样做是为了模拟两个计算机之间串口传输)
在这里插入图片描述
在这里插入图片描述

二、传输文件

双开串口调试助手,我这里可以看到COM8,COM9,波特率这些都默认即可
在这里插入图片描述
点击任意窗口的右下角的加载文件
在这里插入图片描述
随便选择一个图片即可
然后打开串口,发送数据即可
在这里插入图片描述

三、汉字点阵字库

汉字编码分为区位码和机内码,
区位码
所有的国标汉字及符号分配在一个 94 行、94 列的方 阵中,方阵的每一行称为一个“区”,编号为 01 区到 94 区,每一列称为一个“位”,编号为 01 位到 94 位,方阵中的每一个汉字和符号所在的区号和位号组合在一起形成的四个阿拉 伯数字就是它们的“区位码”。区位码的前两位是它的区号,后两位是它的位号。用区位码就 可以唯一地确定一个汉字或符号,反过来说,任何一个汉字或符号也都对应着一个唯一的 区位码。汉字“母”字的区位码是 3624,表明它在方阵的 36 区 24 位。
机内码
汉字的机内码是指在计算机中表示一个汉字的编码。机内码与区位码稍有区别。如上所 述,汉字区位码的区码和位码的取值均在 1~94 之间,如直接用区位码作为机内码,就会 与基本 ASCII 码混淆。为了避免机内码与基本 ASCII 码的冲突,需要避开基本 ASCII 码 中的控制码(00H~1FH),还需与基本 ASCII 码中的字符相区别。为了实现这两点,可以 先在区码和位码分别加上 20H,在此基础上再加 80H(此处“H”表示前两位数字为十六进制 数)。经过这些处理,用机内码表示一个汉字需要占两个字节,分别 称为高位字节和低位字 节,这两位字节的机内码按如下规则表示:
高位字节 = 区码 + 20H + 80H(或区码 + A0H)
低位字节 = 位码 + 20H + 80H(或位码 + AOH)
由于汉字的区码与位码的取值范围的十六进制数均为 01H~5EH(即十进制的 01~94),所以汉字的高位字节与低位字节的取值范围则为 A1H~FEH(即十进制的 161~254)。 例如,汉字“啊”的区位码为 1601,区码和位码分别用十六进制表示即为 1001H,它 的机内码的高位字节为 B0H,低位字节为 A1H,机内码就是 B0A1H
点阵字库结构.
在汉字的点阵字库中,每个字节的每个位都代表一个汉字的一个点,每个汉字都是由一个矩形的点阵组成,0代表没有,1代表有点,将0和1分别用不同颜色画出,就形成了一个汉字,常用的点阵矩阵有1212, 1414, 16*16三种字库。. 字库根据字节所表示点的不同有分为横向矩阵和纵向矩阵,目前多数的字库都是横向矩阵的存储方式 (用得最多的应该是早期UCDOS字库),纵向矩阵一般是因为有某些液晶是采用纵向扫描显示法,为了提高显示速度,于是便把字库矩阵做成纵向,省得在显示时还要做矩阵转换。

四、项目完成

打开Ubuntu,在桌面新建一个文件,在这个文件里面放入需要的文件
在这里插入图片描述
在cpp文件里写入以下代码

#include<iostream>

#include<opencv/cv.h>

#include"opencv2/opencv.hpp"

#include<opencv/cxcore.h>

#include<opencv/highgui.h>

#include<math.h>

using namespace cv;

using namespace std;

void paint_chinese(Mat& image,int x_offset,int y_offset,unsigned long offset);

void paint_ascii(Mat& image,int x_offset,int y_offset,unsigned long offset);

void put_text_to_image(int x_offset,int y_offset,String image_path,char* logo_path);

int main(){

    String image_path="xjc.jpeg";//图片路径

    char* logo_path=(char*)"logo.txt";//学号姓名路径

    put_text_to_image(20,300,image_path,logo_path);

    return 0;

}



void paint_ascii(Mat& image,int x_offset,int y_offset,unsigned long offset){

    //绘制的起点坐标

	Point p;

	p.x = x_offset;

	p.y = y_offset;

	 //存放ascii字膜

	char buff[16];           

	//打开ascii字库文件

	FILE *ASCII;

	if ((ASCII = fopen("Asci0816.zf", "rb")) == NULL){

		printf("Can't open ascii.zf,Please check the path!");

		//getch();

		exit(0);

	}

	fseek(ASCII, offset, SEEK_SET);

	fread(buff, 16, 1, ASCII);

	int i, j;

	Point p1 = p;

	for (i = 0; i<16; i++)                  //十六个char

	{

		p.x = x_offset;

		for (j = 0; j < 8; j++)              //一个char八个bit

		{

			p1 = p;

			if (buff[i] & (0x80 >> j))    /*测试当前位是否为1*/

			{

				/*

					由于原本ascii字膜是8*16的,不够大,

					所以原本的一个像素点用4个像素点替换,

					替换后就有16*32个像素点

					ps:感觉这样写代码多余了,但目前暂时只想到了这种方法

				*/

				circle(image, p1, 0, Scalar(0, 0, 255), -1);

				p1.x++;

				circle(image, p1, 0, Scalar(0, 0, 255), -1);

				p1.y++;

				circle(image, p1, 0, Scalar(0, 0, 255), -1);

				p1.x--;

				circle(image, p1, 0, Scalar(0, 0, 255), -1);

				

			}						

			p.x+=2;            //原来的一个像素点变为四个像素点,所以x和y都应该+2

		}

		p.y+=2;

	}

}



void paint_chinese(Mat& image,int x_offset,int y_offset,unsigned long offset){//在图片上画汉字

    Point p;

    p.x=x_offset;

    p.y=y_offset;

    FILE *HZK;

    char buff[72];//72个字节,用来存放汉字的

    if((HZK=fopen("HZKs2424.hz","rb"))==NULL){

        printf("Can't open HZKf2424.hz,Please check the path!");

        exit(0);//退出

    }

    fseek(HZK, offset, SEEK_SET);/*将文件指针移动到偏移量的位置*/

    fread(buff, 72, 1, HZK);/*从偏移量的位置读取72个字节,每个汉字占72个字节*/

    bool mat[24][24];//定义一个新的矩阵存放转置后的文字字膜

    int i,j,k;

    for (i = 0; i<24; i++)                 /*24x24点阵汉字,一共有24行*/

	{

		for (j = 0; j<3; j++)                /*横向有3个字节,循环判断每个字节的*/

			for (k = 0; k<8; k++)              /*每个字节有8位,循环判断每位是否为1*/

				if (buff[i * 3 + j] & (0x80 >> k))    /*测试当前位是否为1*/

				{

					mat[j * 8 + k][i] = true;          /*为1的存入新的字膜中*/

				}

				else {

					mat[j * 8 + k][i] = false;

				}

	}

    for (i = 0; i < 24; i++)

	{

		p.x = x_offset;

		for (j = 0; j < 24; j++)

		{		

			if (mat[i][j])

				circle(image, p, 1, Scalar(255, 0, 0), -1);		  //写(替换)像素点

			p.x++;                                                //右移一个像素点

		}

		p.y++;                                                    //下移一个像素点

	}

}



void put_text_to_image(int x_offset,int y_offset,String image_path,char* logo_path){//将汉字弄上图片
//x和y就是第一个字在图片上的起始坐标
    //通过图片路径获取图片

    Mat image=imread(image_path);

    int length=18;//要打印的字符长度

    unsigned char qh,wh;//定义区号,位号

    unsigned long offset;//偏移量

    unsigned char hexcode[30];//用于存放记事本读取的十六进制,记得要用无符号

    FILE* file_logo;

    if ((file_logo = fopen(logo_path, "rb")) == NULL){

		printf("Can't open txtfile,Please check the path!");

		//getch();

		exit(0);

	}

    fseek(file_logo, 0, SEEK_SET);

    fread(hexcode, length, 1, file_logo);

    int x =130,y =290;//x,y:在图片上绘制文字的起始坐标

    for(int m=0;m<length;){

        if(hexcode[m]==0x23){

            break;//读到#号时结束

        }

        else if(hexcode[m]>0xaf){

            qh=hexcode[m]-0xaf;//使用的字库里是以汉字啊开头,而不是以汉字符号开头

            wh=hexcode[m+1] - 0xa0;//计算位码

            offset=(94*(qh-1)+(wh-1))*72L;

            paint_chinese(image,x,y,offset);

            /*

            计算在汉字库中的偏移量

            对于每个汉字,使用24*24的点阵来表示的

            一行有三个字节,一共24行,所以需要72个字节来表示

            如赵字

            区位码是5352

            十六进制位3534

            机内码就是d5d4

            d5-af=38(十进制),因为是从汉字啊开始的,所以减去的是af而不是a0,38+15等于53与区码相对应

            d4-a0=52

            */

            m=m+2;//一个汉字的机内码占两个字节,

            x+=24;//一个汉字为24*24个像素点,由于是水平放置,所以是向右移动24个像素点

        }

        else{//当读取的字符为ASCII码时

        wh=hexcode[m];

        offset=wh*16l;//计算英文字符的偏移量

        paint_ascii(image,x,y,offset);

        m++;//英文字符在文件里表示只占一个字节,所以往后移一位就行了

        x+=16;
 }



    }

    cv::imshow("image", image);

    cv::waitKey();

}


在终端打开输入以下命令

g++ test.cpp -o test `pkg-config --cflags --libs opencv`

再输入./test运行
在这里插入图片描述

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值