文档章节

显示12个字节

laichendong
 laichendong
发布于 2014/06/25 09:27
字数 1304
阅读 35
收藏 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
朝阳
程序员
私信 提问
linux命令--df命令&du命令

df命令是用来查看linux文件系统的磁盘占用情况,可以看到目前磁盘占用多少空间及还剩多少空间 其中几个常用的命令如下 命令1 df默认显示大小是以kb为单位的,可以用参数-k 1024字节为单位,-...

王小明123
2015/08/17
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
2018/09/04
0
0
Oracle 中的 sql语句

1、 上面的语句不能使用 来查找(当数据库中isKeepOnPBS存在1,0,null三种情况时),否则null值的查不出来,<>和!=都是不等于,都可以使用 2、 3、 4、查询当天的记录 5、sql查询当前时间 6、...

土龙
2014/10/18
0
0
little endian和big endian的概念解释,判定与用途

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

ll124884135
2012/04/05
0
0
Oracle varchar2或char类型的byte和char的区别

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

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

没有更多内容

加载失败,请刷新页面

加载更多

CSS 选择器参考手册

CSS 选择器参考手册 选择器 描述 [attribute] 用于选取带有指定属性的元素。 [attribute=value] 用于选取带有指定属性和值的元素。 [attribute~=value] 用于选取属性值中包含指定词汇的元素。...

Jack088
今天
1
0
数据库篇一

数据库篇 第1章 数据库介绍 1.1 数据库概述  什么是数据库(DB:DataBase) 数据库就是存储数据的仓库,其本质是一个文件系统,数据按照特定的格式将数据存储起来,用户可以对数据库中的数据...

stars永恒
今天
2
0
Intellij IDEA中设置了jsp页面,但是在访问页面时却提示404

在Intellij IDEA中设置了spring boot的jsp页面,但是在访问时,却出现404,Not Found,经过查找资料后解决,步骤如下: 在Run/Debug Configurations面板中设置该程序的Working Directory选项...

uknow8692
昨天
3
0
day24:文档第五行增内容|每月1号压缩/etc/目录|过滤文本重复次数多的10个单词|人员分组|

1、在文本文档1.txt里第五行下面增加如下内容;两个方法; # This is a test file.# Test insert line into this file. 分析:给文档后增加内容,可以用sed 来搞定;也可以用while do done...

芬野de博客
昨天
3
0
深入理解JVM—JVM内存模型

深入理解JVM—JVM内存模型 我们知道,计算机CPU和内存的交互是最频繁的,内存是我们的高速缓存区,用户磁盘和CPU的交互,而CPU运转速度越来越快,磁盘远远跟不上CPU的读写速度,才设计了内存...

onedotdot
昨天
3
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部