前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >u8u3_u8计算后会变为u16

u8u3_u8计算后会变为u16

作者头像
Java架构师必看
发布2022-10-04 14:03:33
1.1K0
发布2022-10-04 14:03:33
举报
文章被收录于专栏:Java架构师必看Java架构师必看

大家好,我是架构君,一个会写代码吟诗的架构师。今天说一说u8u3_u8计算后会变为u16,希望能够帮助大家进步!!!

阴码+逐列 式+顺向+C51 格式

代码语言:javascript
复制
void LCD_ShowChar(u16 x,u16 y,u8 num,u8 size,u8 mode)
{ 
     							  
    u8 temp,t1,t;
	u16 y0=y;
	u8 csize=(size/8+((size%8)?1:0))*(size/2);		//得到字体一个字符对应点阵集所占的字节数 
 	num=num-' ';//得到偏移后的值(ASCII字库是从空格开始取模,所以-' '就是对应字符的字库)
	for(t=0;t<csize;t++)
	{ 
      
		if(size==12)temp=ascii_1206[num][t]; 	 	//调用1206字体
		else if(size==16)temp=ascii_1608[num][t];	//调用1608字体
		else if(size==24)temp=ascii_2412[num][t];	//调用2412字体
		else return;								//没有的字库
		for(t1=0;t1<8;t1++)
		{ 
   			    
			if(temp&0x80)LCD_DrawFRONT_COLOR(x,y,FRONT_COLOR);
			else if(mode==0)LCD_DrawFRONT_COLOR(x,y,BACK_COLOR);
			temp<<=1;
			y++;
			if(y>=tftlcd_data.height)return;		//超区域了
			if((y-y0)==size)
			{ 
   
				y=y0;
				x++;
				if(x>=tftlcd_data.width)return;	//超区域了
				break;
			}
		}  	 
	}  	    	   	 	  
} 

只听到从架构师办公室传来架构君的声音:

尘香拂马,逢谢女、城南道。有谁来对上联或下联?

从第一列开始向下 每取 8 个点作为一个字节,如果最后不足 8 个点就补满 8 位。取模顺序是从 高到低,即第一个点作为最高位。

temp其实就是一列,位与0x80取得最高位(相当于D7),如果为1则要用前景色点亮,如果为0为背景色即没有点该点的颜色,temp<<1位,相当于取了D6,y++,y相当于列扫描,一列扫完以后,x++

相当于向右进行行扫描

转载:

void LCD_ShowChar(u16 x,u16 y,u8 num,u8 size,u8 mode)

{

u8 temp,t1,t;

u16 y0=y;

u16 colortemp=POINT_COLOR;

num=num-’ ';//得到偏移后的值

if(!mode) //非叠加方式

{

for(t=0;t

{

if(size==12)temp=asc2_1206num; //调用1206字体

else temp=asc2_1608num; //调用1608字体

for(t1=0;t1<8;t1++)

{

if(temp&0x80)POINT_COLOR=colortemp;

else POINT_COLOR=BACK_COLOR;

LCD_DrawPoint(x,y);

temp<<=1;

y++;

if(x>=lcddev.width){POINT_COLOR=colortemp;return;}//超区域了

if((y-y0)==size)

{

y=y0;

x++;

if(x>=lcddev.width){POINT_COLOR=colortemp;return;}//超区域了

break;

}

}

}

}

1206字体和1608字体是两种不同的字号,就是一个横向6点,纵向12点,一个横向8点,纵向16点,从判断语句可以看出,二者对应的码表是不一样的,这个码表在工程里搜索一下,在font.h里,是一个二维数组。我们这里使用1608字体,以字母“M”为例。查一下码表中“M”对应的数组,如下:

{0x10,0x04,0x1F,0xFC,0x1F,0x00,0x00,0xFC,0x1F,0x00,0x1F,0xFC,0x10,0x04,0x00,0x00},

’ ‘是码表的开始,由字母减去’ '得到偏移量,也就是二维数组的第几行。第一个循环开始,就给临时变量temp赋值为数组的第一个值,由于是1608字体,因此也有一个数组中有16个数,第一个循环就是为了让temp遍历这16个数。

之后第二个循环开始,由于数组中的每一个数均是8位,由于每次循环都会对temp左移一位,因此第二个循环的次数是8次。经if(temp&0x80)判断,如果数字的最高位为1,则进行描点的颜色为字体颜色,如果不是1,则描点颜色为底色。(PS:描点函数LCD_DrawPoint还是很简单滴,总的来说就是向写GRAM寄存器R20h,R21h写入我们希望写的点颜色,基本操作O(∩_∩)O)

每次移位,y都会自加,第二个循环是8次,而size为16,也就是说数组中每读过两个数,y自加16次之后都会清零,然后x加1,就像列扫描一样,一列16个点结束后会进行到下一列。

超区域那部分就是说超过size了,描个点就返回啦。下面以“M”为例描述下描点的过程。

码表前8个

0x10 (0,3) 00010000 所以bit3点亮

0x04 (0,13) 00000100 上面0x10占8位,所以bit13点亮

0x1F (1,3),(1,4),(1,5),(1,6),(1,7)

0xFC (1,8),(1,9),(1,10),(1,11),(1,12),(1,13)

0x1F (2,3),(2,4),(2,5),(2,6),(2,7)

0x00 无

0x00 无

0xFC (3,8),(3,9),(3,10),(3,11),(3,12),(3,13)

码表后8个

0x1F (4,3),(4,4),(4,5),(4,6),(4,7)

0x00 无

0x1F (5,3),(5,4),(5,5),(5,6),(5,7)

0xFC (5,8),(5,9),(5,10),(5,11),(5,12),(5,13)

0x10 (6,3)

0x04 (6,13)

0x00 无

0x00 无

这个不直观啊,用matlab的scatter(x,y)描个点,正好就是TFT屏幕显示的“M”图形

代码语言:javascript
复制
此代码由Java架构师必看网-架构君整理
//显示数字,高位为0,则不显示
//x,y :起点坐标 
//len :数字的位数
//size:字体大小
//color:颜色 
//num:数值(0~4294967295); 
void LCD_ShowNum(u16 x,u16 y,u32 num,u8 len,u8 size)
{ 
            	
	u8 t,temp;
	u8 enshow=0;						   
	for(t=0;t<len;t++)
	{ 
   
		temp=(num/LCD_Pow(10,len-t-1))%10;
		if(enshow==0&&t<(len-1))
		{ 
   
			if(temp==0)
			{ 
   
				LCD_ShowChar(x+(size/2)*t,y,' ',size,0);
				continue;
			}else enshow=1; 
		 	 
		}
	 	LCD_ShowChar(x+(size/2)*t,y,temp+'0',size,0); 
	}
} 

转载:http://www.51hei.com/bbs/dpj-141896-1.html

u32 oled_pow(u8 m,u8 n)

{

u32 result=1;

while(n–)result*=m;

return result;

}

这个函数我觉得你并不理解,不然也就不会不明白temp指的是什么了。现在我给你分析一下,就按照我最开始的思路来读程序,跟着我一步一步走好了:

根据函数名,我们可以大概了解这个函数的功能是显示一个数字,传入的参数是X坐标(字符在一行的哪一个位置)、y坐标(字符显示在哪一个行)、一个待显示的数据、数据的长度和显示字符的跨度

enshow是一个使能标志,那么这个使能位是在什么时候起作用,什么时候关闭呢?

if(enshow==0&&t<(len-1))

这一句的意思你明白,是当这个enshow为0并且再数据长度范围内的时候,进行temp==0的判断,而不满足这个条件,enshow就会置1,显然,是在判断数据传输是否完成,并且在达到给定的数据长度后终止传输。

纵观函数的结构可以看出,在许可长度范围内,函数循环计算一个temp量,从最后一句

OLED_ShowChar(x+(size2/2)*t,y,temp+‘0’);

可以看到,这个temp实际上是待显示位数字,比如1,2,3…而不是数字的ASCII码,所以这里需要以‘0’做基准把数字转换为其ASCII码,可以猜测,OLED_ShowChar()函数是传入一个ASCII码,在ASCII码字模表中取模显示对应的字符。

知道了temp是什么意思,我们反过来看temp是如何求得的:

temp=(num/oled_pow(10,len-t-1))%10

这句中,num是待显示的数据,我们知道了该显示函数是按位取出num,那么这一句的作用必然是按位取数,结尾的%10(取余运算)表明所取的数是前面所求整数数据的最后一位。

现在到了最关键的部分了

oled_pow(10,len-t-1)返回的是一个什么数据?

将实参代入形参即m为累乘数据,n为数据长度-已处理位数-1

(此处的减一是为了适应C语言计数到“0”为止)

u32 oled_pow(u8 m,u8 n)

{

u32 result=1;

while(n–)result*=m;

return result;

}

翻译一下:

结果值result初始值为1;

当 在长度范围内 时,结果值=结果值*累乘数据

返回 结果值

也就是说,返回的是(10)^(显示的长度)

这样,num/oled_pow(10,len-t-1)指的是对指定长度取模(取num定长)

例如:num=1325416;显示长度为5,那么num/oled_pow(10,len-t-1)运算之后就是13。在这个结果上再对10取余就是最后一位。

由于t的变化,可以遍历每一位。

但是如果数据在变化,那当数据位数过少的时候,前面位会显示什么呢?是0。例如规定显示5位,但是实际上数据只有2位,那显示就会变成000XX,为了美观,也要把这个0干掉。

事实上,

if(temp==0)

{

OLED_ShowChar(x+(size2/2)*t,y,’ ');

continue;

}

就是干这个工作的,当取出的位为0的时候,就用空格代替改数值达到消隐无效0的作用。从上面的分析我们可以知道,这个传入的ASCII码应该是由一个char类型的值在接收,所以这里的空格会以ASCII码的形式存入形参。

我想我应该讲完了,你的三个问题答案整理如下:

1.temp=(num/oled_pow(10,len-t-1))%10;//这里算出来到底是什么数

答:该出算出的是本次要显示的位对应的数值

2.OLED_ShowChar(x+(size2/2)*t,y,’’);//这里的’'怎么和阿斯克码表对上的

答:猜测接收的形参是char类型,所以空格直接以字符型存入了(即存入的是空格的ASCII码)

3.OLED_ShowChar(x+(size2/2)*t,y,temp+‘0’); //这里又为什么+0

答:因为需要以0为基准将数字类型转为对应的ASCII码。

今天文章到此就结束了,感谢您的阅读,Java架构师必看祝您升职加薪,年年好运。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022-09-292,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档