文档章节

显示12个字节

laichendong
 laichendong
发布于 2014/06/25 09:27
字数 1304
阅读 34
收藏 0
如果你看到这篇文章的标题,隐约或者明确觉得不对劲儿。那,恭喜你!你知道我接下来要说什么了。 听我们的PM 美眉和研发同事在讨论问题,PM说到: “这里展示12个字节”。我一听怎么那么别扭呢?然后就问PM同事说怎么怎么回事。然后PM同事说是这么这么回事。(此处省略若干字……)最终确定这就是一个混沌的说法。因为当时我们的PM和研发同事都没有弄明白下面两个问题。才会产生这样的说法(和做法)。
  1. 中英文(不同编码)间,每个字符存储起来占用的空间不一样
  2. 中英文(不同字符)间,每个字符显示出来占用的宽度不一样
先说第一个问题。上面说的也还是比较笼统的。更加严谨的说法应该是, 同一个字符,采用不同的编码方式表示所占用的空间是可能不一样的比如:
System.out.println(Arrays.toString("东".getBytes(Charset.forName("UTF-8")))); // [-28, -72, -100]
System.out.println(Arrays.toString("东".getBytes(Charset.forName("GBK")))); // [-74, -85]
同样东东的的“东”字,用UTF-8编码表示,占用了3个字节的长度,用GBK编码表示,占用了两个字节的长度。 同一种编码方式,表示不同的字符所占用的空间也是可能不一样的。比如:
System.out.println(Arrays.toString("东".getBytes(Charset.forName("GBK")))); // [-74, -85]
System.out.println(Arrays.toString("D".getBytes(Charset.forName("GBK")))); // [68]
同样是GBK编码,表示“东”字用了2个字节,而表示大写字母“D”则只用了1个字节。这恐怕就是 流毒无穷的“ 一个中文字符算两个长度”说法的来源了。别再这样说,这是错的!如果你用的UTF-8编码,则可能是一个中文占3个甚至4个长度! 如果你用的Oracle数据库,相信会有一定的体会。Oracle对varchar类型的字段长度是按照 字节来指定的。所以一个长度为100的varchar字段,最多能存储一个str.length()==100的英文字符串。却只能存储str.length()==50(GBK编码)或str.length()==33(UTF-8编码)的中文字符串。这时候需要在数据录入时控制长度就比较悲催了。就有人出了“ 一个中文字符算两个长度”的馊主意。正确的做法应该是使用相应的编码将字符串转换成字节数组,看字节数组的长度。遗憾的是JavaScript没有提供原生的编码解码实现。 如果你使用的是varchar2类型或者是MySQL数据库,就比较开心了。他们使用的是以字符为单位指定字段的长度。所以数据库长度为100时,前端只要str.length()<=100就一定不会超长了。 关于字符编码,可以看看 维基百科的解释。 再说第二个问题,不同的字符,显示出来占用的宽度不一样。同样,上面的说法也是比较笼统的, 一个字符在屏幕上显示的宽度,同时受字号(font-size)和字体(font-family)的影响。 先说字号。一般情况我们会写:
font-size:12px;
一个字符显示在屏幕上。是一个二维的概念,有宽度和高度,那这个12px到底是说这个字符的宽度为12px还是高度为12px呢?都不是!font-size实际上设置的是字体中字符框的高度;实际的字符字形可能比这些框高或矮(通常会矮)。
<style type="text/css">
p {font-size: 80px}
</style>

<p>aaaAAA222中中中</p>
DeepinScrot-4323   可以看到,font-size设置为80px时,中文,英文,数字的实际高度都没有超过80。具体会有多高其实不用太关心,只要知道font-size越大,展示出来的字也越大就可以了。显示上的事是一个主观感受上的事。再看宽度,同样是80px的字号,小写字母,大写字母,数字和中文占的宽度也不一样,上图应该能比较明显的看出来,就不做对比图了。 再说字体,比如:
<style type="text/css">
p {font-size: 30px}
.p1{font-family:serif}
.p2{font-family:monospace}
</style>

<p class="p1">abcdefghijklmn</p>
<p class="p2">abcdefghijklmn</p>
DeepinScrot-0503 同样的内容“abcdefghijklmn ”,同样的字号“30px”,不同的字体“serif”和“mogospace”. 所占用的宽度是不一样的。由这点可以看出用str.subString(0, n) + "..." 的方式来实现“文本超长则用省略号表示”是不靠谱的。正确的做法应该是用css在展示的时候实现。
<style type="text/css">
p {
    width:100px;
    overflow: hidden;
    white-space: nowrap;
    text-overflow: ellipsis;
}
</style>

<p>我很长,我真的很长</p>
关于文本溢出问题,推荐阅读 这篇文章。其实拿大腿想想也是这样的,文本溢出是显示上的问题,按照“职责分配原则”中的“专家原则”也不能将显示的事儿交给数据处理去做。 好吧,就到这吧。希望看完这篇文章的同学不会再说出“一个中文算两个长度”和“这里显示12个字节”这样的话了。那我就功德圆满了。

© 著作权归作者所有

共有 人打赏支持
laichendong
粉丝 10
博文 85
码字总数 71483
作品 0
朝阳
程序员
私信 提问
little endian和big endian的概念解释,判定与用途

转贴地址:http://www.chinaunix.net/jh/23/823662.html ####################### //声明: 1 本帖作者是:converse ,至此感谢! 2 红色背景字体为本人添加内容! 3 黄色背景为个人阅读重点...

ll124884135
2012/04/05
0
0
innodb行宽几百字节 ,千万行记录,索引深度多少?

root@localhost:mysql3307.sock [db1] 11:30:05> show create table pageG * 1. row * Table: page Create Table: CREATE TABLE ( int(11) NOT NULL AUTO_INCREMENT, varchar(100) DEFAULT N......

jkkjkjkjkk
09/04
0
0
《超实用的Node.js代码段》连载一:获取Buffer对象字节长度

我们知道Node.js框架下的Buffer对象能够对二进制数据提供很好的支持,那么获取一个Buffer对象真实的字节长度则是必须要用到的功能了。Node.js框架为开发人员提供了一个Buffer.byteLength()方...

woiwoi
2016/04/12
46
0
MSP430学习笔记9-PS2键盘解码

PS2键盘解码的基本原理是通过外部中断读取键盘输出的串行信号,在根据扫描码进行查表解码。键盘发送往主机的信号总是在时钟的下降沿因此此中断是在下降沿触发,且时钟信号是由键盘给出,因此...

文艺小青年
2017/06/01
0
0
Oracle varchar2或char类型的byte和char的区别

Oracle定义字符串类型VARCHAR2和CHAR指定长度的用法如下: varchar2( ) 是介于1~4000之间的一个数,表示最多占用4000字节的存储空间。 char( ) 是介于1~2000之间的一个数,表示最多占用2000字...

技术小胖子
2017/11/14
0
0

没有更多内容

加载失败,请刷新页面

加载更多

Windows 环境下安装 Oracle JDK

本页面中描述了如何在 Window 环境下安装 Oracle JDK。 我们使用的版本是 Window 10,我们需要安装的版本是 Oracle JDK 8u191。 检查当前版本 在进行新的 JDK 安装之前,你需要检查下你本地的...

honeymose
31分钟前
0
0
用any-loader封装jQuery的XHR —— 随便写着玩系列

哎,都说没人用JQuery啦,叫你别写这个。 其实我也是好高骛远使用过npm上某个和某个很出名的XHR库,嗯,认识我的人都知道我喜欢喷JQ,以前天天喷,见面第一句,你还用JQ,赶紧丢了吧。但我也...

曾建凯
今天
4
0
聊聊storm的AggregateProcessor的execute及finishBatch方法

序 本文主要研究一下storm的AggregateProcessor的execute及finishBatch方法 实例 TridentTopology topology = new TridentTopology(); topology.newStream("spout1", spout......

go4it
今天
3
0
大数据教程(7.5)hadoop中内置rpc框架的使用教程

博主上一篇博客分享了hadoop客户端java API的使用,本章节带领小伙伴们一起来体验下hadoop的内置rpc框架。首先,由于hadoop的内置rpc框架的设计目的是为了内部的组件提供rpc访问的功能,并不...

em_aaron
今天
5
0
CentOS7+git+github创建Python开发环境

1.准备CentOS7 (1)下载VMware Workstation https://pan.baidu.com/s/1miFU8mk (2)下载CentOS7镜像 https://mirrors.aliyun.com/centos/ (3)安装CentOS7系统 http://blog.51cto.com/fengyuns......

枫叶云
昨天
4
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部