深入理解文字高度和行高的设置

2023-10-27

font-size设置的是什么?line-height设置的是什么?各种行高是怎么计算出来的?你真的知道吗?

1 从font-size讲起,说文字高度:

当你按住鼠标左键选中一段文字的时候,这段文字背后会有一个颜色变化的区域,这个区域可以近似的视为是这段文字的content-area(内容区域)。

又或者当我们给一行文字设置background的时候,文字背景所占的区域可以近似的视为是这段文字的content-area(内容区域)。

我们通常用content-area的高度代表文字高度 ,这是字体设计领域的一个概念。

你先别质疑这种思想!我们知道有些文字的样子喜欢往上突出,有些文字喜欢往下突,再者,行与行文字之间需要一定的空间,所以content-area在上下两个区域预留出空白的区域是很正常的,如图:

在这里插入图片描述
先说结论:

font-size 相同时,font-family 不同,得到的文字高度也不同:
在这里插入图片描述
上图表明,同样是font-size为100px,三款不同字体在浏览器画面上的高度是不同的。

我们再来分析一下,在font-size设置为100px时,Catamaran字体的那164px是怎么构成的:
在这里插入图片描述
我们假设画面由相对单位构成方便缩放拉伸,再假设现在1px对应画面上的10个相对单位。(这点你可以不用理会)

那么我们所能设置的font-size: 100px其实设置的是上图左方1000个单位(em-square)这部分的高度而已; 我们可以把em-square区域视为content-area的主体区域。

(em-square这个概念的产生是有历史原因的,但是这点我们不用理会。)
em-square之内,图中"B"上方预留的小区域,大概是考虑到有些字母模仿手写体会有起笔效果,需要一定的位置;"a"下方预留的区域是为了写类似g这样的字母的时候,下方需要一定的区域。
em-square之外,上方和下方都预留了一定空白区域,这样可以防止行与行间文字挤在一起。
事实上对于所有font-family而言,规则都是这样的,对于中文字体而言也是如此。
我们可以得到第二个结论:

font-size不能直接控制文字的高度(content-area的高度),它控制的是文字主体部分的高度(em-square的高度)。 而content-area和em-square之间的比例,完全是由字体设计师决定,作为程序员是无法统一的。

所以再经过调查, 没有属性能直接控制文字的高度,我们只能这样通过font-size间接 控制文字的高度。

2 计算一个p元素的内容高度

(我们既然说是内容高度,就直接把margin、border厚度、padding排除在外了。)

当 p 元素出现在屏幕上时,它可能包含了多行的内容,其内容高度由每一行的高度堆叠而成。这点很好理解。

2.1 line-box的默认高度

每一行可视为一个line-box。通常来讲,line-box的默认高度是由这一行的内容撑大的,这一行里面的元素谁的高度最大,line-box的高度就等于它,由它撑大:在这里插入图片描述

<p>
    Good design will be better.
    <span class="a">Ba</span>
    <span class="b">Ba</span>
    <span class="c">Ba</span>
    We get to make a consequence.
</p>

如图,我们给class=“a”、class=“b”、class="c"设置了不同的font-family或者font-size,导致这些文字的高度各不相同。每一个line-box代表每一行的区域。一个line-box的默认高度由这行内的最大高度决定。

补充:还记得吗?对于文字而言,文字高度就是它的content-area的高度。

3 line-height的作用

当我们没有设置line-height的时候,一个line-box的默认高度由这行内的最大高度决定。

所以通常情况下,line-box的高度 >= content-area的高度 > em-square的高度

当我们设置了line-height的时候,line-box的高度就是由line-height所设定的。 因此此时我们可以把line-box的高度设置得比content-area甚至em-square小,导致文字看上去挤在一起。

一行内的内容,除了文字外;通常就是行内块级元素了,通常可以分为三类:

  • img元素、input元素、svg元素等
  • display:inline-block、display:inline-table、display:inline-flex 的元素
  • 处于某种特殊格式化上下文的内联元素:例如 flexbox 元素中的子元素都处于 flex formatting context(弹性格式化上下文)中,这些子元素的 display 值都是「blockified」
    这些行内块级元素,其所占的高度通常由height,margin,border,padding盒子模型的高度来决定。

如果你将其 height 设置为 auto 的话,那么其高度的取值就是 line-height,其 content-area 的取值也是 line-height。

4 line-height是怎么计算的

官方对line-height的定义是:

两行文字基线(baseline)之间的距离。

4.1 额外谈谈"基线"

为什么是基线呢?

line-height可不可以定义为是两行文字的“中线”之间的距离呢?

可不可以定义为是两行文字的“底线”之间的距离呢?

张鑫旭认为都是都可以的。

那么为什么选基线作为定义的依据?

因为, 基线乃其他线定义之根本也。

了解“基线”大有好处。

回到刚才那幅图:
在这里插入图片描述
这么多线,都是字体设计师为了比对位置用的。这几条线,可以视为是字体设计的度衡。任何一款字体都会给出以上各种线的位置参数,所有线的位置参数都是相对于基线的位置而确定的 。我们把某款字体放到 FontForge 中,分析它的字体度量:
在这里插入图片描述

首先说明的是HHead Ascent是针对macOS的计算值,Win Ascent是针对Windows系统的计算值,这点可以不用理会。

设计一款字体的时候从基线开始定位。

我们看到,Em Size设置为1000,就是基线到基线以上1000个单位的那片区域都是em-square的区域。

Ascent设置为770,就是Ascender线距离基线770个单位。

Descent设置为230,就是Descender线距离基线230个单位。

Capital Height是680,也是相对于基线的。是大写字母顶部位置的线距离基线多少。

X Height是485,也是相对于基线而言的。是小写字母顶部位置的线距离基线多少。

Ascender线到Descender线之间的高度 加上Line Gap就是content-area的总高度。(笔者注:这点可能有误)

所以说,我们选基线来定义line-height:line-height是两行文字基线(baseline)之间的距离。

但是这个定义说了等于没说,因为它不能告诉我们怎么设置line-height啊!

还是我们早之前做的定义比较棒!!当我们设置了line-height的时候,line-box的高度就是由line-height所设定的。

4.2 line-height的计算(画重点!)

line-height的计算要根据这行文字的font-size多少 ,即是根据em-square的高度计算。本文说的"计算"就是这个意思。

w3c说,line-height可以设置的值有5种:

  • normal
  • 数字
  • 多少px(都写死了,不必计算)
  • 百分数(多少%)
  • inherit(对于该元素而言,不必计算)

4.2.1 line-height:normal的时候

我们来看看 Arial 字体,它的 em-square 是 2048,ascender 是 1854,descender 是 434,line gap 是 67。那么当 font-size: 100px 时:

line-height:normal为 100/2048*(67+1854+434) 约为 115px

4.2.2 line-height设为数字/百分数的时候

line-height设为数字/百分数的时候,line-height就是拿 数字/百分数 去乘以em-square的高度。

所以如果设置line-height:1或者line-height:100%肯定是有问题的啊——本来em-square基本上就比content-area要小,你还让line-height是em-square的高度的一倍?!行与行之间的文字绝对是要挤在一起了:
在这里插入图片描述
那怎么设置呢?反正line-height最终的计算结果超过content-area就应该比较符合常理了。

张鑫旭大神鼓励大家用数字来设置line-height。

  • 如果是博客类型的网站,line-height设置为1.5到1.6,行与行之间比较宽,适合舒适地阅读。
  • 如果是研发正规的产品,推荐大家使用20px匹配法 ,方便计算。

4.2.3 20px匹配法

为什么呢?

首先我们一般来说会设置font-size在14px到16px之间,然后content-area的高度也不会比这个高太多的吧。所以我们想着让line-height设置为20px应该够用了。而且20px方便计算。像这样:

body { font-size: 14px; line-height:??? }

只要把???结果计算出来就行啦!

20除以14等于1.42857后面还有小数位。

如果你写line-height:1.42857,虽然它乘以14非常接近20,但是没有超过20,chrome会把它设置为19px。

所以我们一般设line-height为1.4286,稍微大那么一点点,chrome把它们乘起来,然后再约等于,就是20px了。

结果写法就像这样:

body { font-size: 14px; line-height: 1.4286; }

或者

body { font-size: 14px/1.4286 'microsoft yahei'; }

4.2.4 补充:line-height: 1.5和line-height: 150%的区别?

这是一个在网上被人讲烂了的知识点了,line-height设置为数字和百分数时之间的区别:

  • line-height:1.5: 所以可继承元素根据font-size重新计算行高
  • line-height:150%: 当前元素根据font-size计算行高,继承给下面的元素
  • line-height: 1.5em :当前元素根据font-size计算行高,继承给下面的元素(与百分比效果相同)
<style>
.parent { font-size: 30px; }
.child { font-size: 60px; }
.p1 { line-height: 1.5; }
.p2 { line-height: 150%; }
</style>
<body>
<div>
<p class="parent p1">文字<span class="child">算出子级line-height为90px</span>文字</p><hr>
<p class="parent p2">文字<span class="child">算出子级line-height为15px</span>文字</p>
</div>
</body>

5 后记:

5.1 全文总结

  • 首先文字高度和font-size只有间接的关系
  • 然后,line-height的计算是根据font-size的
  • 当我们没有设置line-height的时候,一个line-box的默认高度由这行内的最大高度决定。
  • 当我们设置了line-height的时候,line-box的高度就是由line-height所设定的。

5.2 预告:文字在一行中的位置的计算?

有没有发现?根据我们上面的理论,如果想要确定一段文字在一行(line-box模型)中的位置太难了!是不是垂直居中?文字位置偏上还是偏下?要经过一系列计算才能得出结论。

进而,我们联想到,平时我们想让行内元素垂直居中的办法:设置vertical-align:middle,这种居中方法是错的吗?

嗯,是错的,这只能使得行内元素近似垂直居中。

如果想要行内的文字或者图片完美居中,就要研究清楚line-height和vertical:middle的原理了。

你可以系统地看一看张鑫旭在慕课网上的视频教程。如果急着想看结论,直接翻到他每个视频的最后一章就可以了。

或者你可以直接翻看我下一篇文章 深入理解vertical-align:middle 的结论。

本文标题:深入理解文字高度和行高的设置

文章作者:Alex Zhong

发布时间:2018-02-06, 13:15:57

原始链接:https://alexzhong22c.github.io/2018/02/06/height-calculate/

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

深入理解文字高度和行高的设置 的相关文章

  • SpringBoot——内置数据库

    简单介绍 关于数据层的三大组件 数据源 持久化技术 数据库 前两种都已经介绍过了SpringBoot的内置的解决方案 还有最后一个数据库 在SpringBoot中 内置了三款数据库 分别是 H2 HSQL Derby 这三种数据库有几个共同
  • Java 与 C++ 的区别

    近期在学习C 简单介绍下 Java 和 C 区别 Java 是纯粹的面向对象语言 所有的对象都继承自 java lang Object C 为了兼容 C 即支持面向对象也支持面向过程 Java 通过虚拟机从而实现跨平台特性 但是 C 依赖于
  • C++的std::move()

    std move 是 C 标准库中的一个函数模板 用于将对象转换为右值引用 以便支持移动语义 它位于
  • pytorch: 出现NotImplementedError 。。。。in forward

    raise NotImplementedError NotImplementedError 在利用pytorch构建model的时候 出现 NotImplementedError的错误 一般的是由于在重构nn model的时候出现了错误 主
  • 郭天祥的10天学会51单片机_第十节

    要多练 多练C语言程序 练习C语言程序编程思路 可以把别人的程序改改 找些电路焊接调试 AD985可以产生方波 三角波 正弦波 低通滤波器是通过低频阻止高频 作用是滤除杂波 液晶显示单片机状态 运算放大器放大波形 积分器将方波积分成三角波
  • 【操作系统】王道考研 p13 进程调度的时机、切换与过程、调度方式

    视频 知识总览 进程调度的时机 进程调度 就是把就绪态变为运行态 关于进程在操作系统内核程序临界区中不能进行调度与切换 临界资源 一个时间段内只允许一个进程使用的资源 各进程需要互斥地访问临界资源 临界区 访问临界资源的那段代码 举个例子
  • Java 截取String类型字符串后两位及截掉后两位

    String strhours String valueOf 123456 String strh strhours substring strhours length 2 strhours length 截取 String strm st
  • “灵巧小工具”一个将图片处理成打印纸尺寸的丰富功能完全免费无水印的图片处理工具

    今天推荐一款微信小程序 灵巧小工具 经常有打印图片需求的用户赶紧收藏了 可以先扫码体验一番 下面介绍一下它的主要功能 1 照片 1寸 2寸 5寸 6寸 支持1寸 2寸照片自动排版 生成相纸尺寸 可以选择微信聊天记录或相册中1寸或2寸照片 然
  • mac安装redis

    本文参考 mac安装redis 自我的进化http www shanxing top p 162 首先查看是否安装了homebrew brew version 如果此命令失败 那么先去安装homebrew吧 安装 从终端运行来安装redis
  • 第五课,关卡放置物体

    第五课讲了在关卡中放置物体 文件 gt 新建关卡 然后将物体放置上去 比如基本几何体的CONE 教程中用的是内容浏览器中的props文件夹里的桌子和椅子 它们都自动放置在地面上 我猜测就是物体坐标在底部 将其设置为0即可 还可以放置触发器
  • asp.net mvcview界面does not contain a definition

    在cshtml使用linq操作数据集 自动补全是可以使用Select 但是一直提示System Collection GenericList does not contain a definition for Select ccepting
  • 超全面试总结——数据仓库 超详细!!!带答案!!!持续更新中~

    超全面试总结 数据仓库 如何理解数仓 为什么要数据仓库建模 为什么要设计数据分层 通用的数据分层设计 分层的原则是什么 数据集市和数据仓库的区别 数据库和数据仓库有什么区别 维度建模三种模式 星形 雪花 星座 事实表 事实表设计方法 维度表
  • linux创建线程pthread_create函数

    函数简介 pthread create是UNIX环境创建线程函数 头文件 include
  • 目标检测常见知识点

    文章目录 1 常见目标检测结构 2 四种需要知道的特征金字塔网络 3 Focal Loss 4 FCOS 5 YOLOX beat YOLOv5 6 VFnet 7 YOLO real time model for edge devices
  • 无法连接虚拟设备 ide1:0,因为主机上没有相应的设备。您要在每次开启此虚拟机时都尝试连接此虚拟设

    运行VMware 出现以上错误 ide1 0一般是虚拟机的光驱 配置默认选项是 使用物理驱动器 而宿主机可能没有光驱 改为 使用ISO映像文件 即可

随机推荐

  • stata时间序列分析_Stata:时间序列教程

    2021年寒假Stata研讨班 高级计量经济学及Stata应用研讨班 2021空间计量研讨班 空间计量及Geoda Stata ArcGis Matlab应用 来源 综合整理自 http dss princeton edu training
  • STM32F103编译环境安装

    1 安装KEIL 安装包链接 链接 https pan baidu com s 1yeguqiSoHmycVblptGuOSA 密码 j0ym 点击这个 只需点下一步安装即可 自定义安装目录 注意在全英文目录下 出现这个关闭就行了 2 安装
  • STL之equal函数

    template
  • Golang fasthttp 为什么你这么优秀

    最近 需要使用 http client 发送数据 网上都说 fasthttp 据说是目前golang性能最好的http库 相对于自带的net http 性能说是有10倍的提升 参考 https studygolang com article
  • Firefly AIO-3399ProC开发板刷ubuntu系统 安装rknntoolkit 1.6.0 + tensorflow 2.0 + pytorch 1.5.0

    AIO 3399Pro刷ubuntu系统同时配置rknntoolkit 1 6 0 环境 由于本人所做项目需要将轻量级的深度学习算法进行部署 故前些日子购买了核心板为RK3399的深度学习开发板 用起来固然没有NVIDIA的jetson开发
  • chatgpt赋能python:建立Python开发环境的详细指南

    建立Python开发环境的详细指南 对于想要成为一名Python开发工程师的人来说 建立一个良好的开发环境是非常重要的 在安装和配置Python的过程中 您可能会遇到许多问题 并可能会出现错误 但是 请不要担心 因为这篇文章提供了一份详细的
  • keepalived高可用实战与Nginx负载均衡视频教程

    微技术 学无止境 一个专注于分享各种教程 技术 文章 分享等资源 为大家的工作 学习 提供便利 前言 大家好 我是微技术 今天给大家分享一款推荐负载均衡的学习教程 让你架构技术更加提高一点 更多深入一些技术 视频教程 keepalived
  • android 动态数据抓取,mitmproxy抓取Keep热门动态-安卓APP抓包爬虫案例

    使用mitmproxy来抓取Keep首页热门动态 鳄鱼君也是费了一天的时间去测试 在这里需要提醒大家抓包不要使用Android系统超过7 0的手机 Android系统越高手机越安全 而我们的抓包很显然是不允许的 这就是为什么你在配置了手机证
  • ajax与javascript,jquery, jquery UI

    ajax就是使用javascript语言实现的一种与服务器异步通信的方式 其核心是xmlhttprequest 像谷歌地图这种应用 如果点击某个点 需要重新刷新页面的话 用户体验很不好 这样的场所使用ajax就非常的方便 jQuery是一个
  • css媒体查询改变上边距,html - CSS宽度和边距不会与媒体查询相加

    我正在建立一个博客布局 目标是以窗口为中心 以最大宽度和最小边距为窗口 以内容为中心 我使用max width和媒体查询的组合来设置一定阈值的边距 我试着把max width和fixedmargin left和margin right放在一
  • go get & go mod找不到包问题解决

    问题一 今天在云虚机中遇到了go get执行报错的问题 报错内容如下 试了ping百度正常 说明不是DNS的问题 执行go mod tidy同样报错 最后找到问题 将GOPROXY由https proxy golang org改为了http
  • SpringBoot:自动装配提速设计

    名词约定 配置类 指使用了 Configuration Component ComponentScan Import ImportResource Bean的类 SpringBoot的设计思想就是通过一个配置类导入多个项目范围内适用的配置类
  • 《零基础入门学习Python》第087讲:Pygame:播放声音和音效

    这节课我们来谈谈 Pygame 中的 播放声音和音效 因为几乎没有任何游戏是一声不吭的 多重的感官体验更能刺激玩家的神经 没有声音的游戏就好比 不蘸番茄的薯条 尽管如此 Pygame 对于声音的处理并不是太理想 我说的是如果你想用 Pyga
  • 线性代数学习笔记——行列式的性质及拉普拉斯定理——11. 拉普拉斯定理

    这节如果不看教学视频而只看PPT的话 很难理解 这充分说明了老师的重要性 1 拉普拉斯 Laplace定理 2 基本结论 三角 对角分块矩阵行列式的计算 3 拉普拉斯定理的应用示例 求行列式 4 分块矩阵的逆的求解
  • ansible批量添加用户账户密码

    一 批量添加root用户以及密码 首先批量添加root 账号密码 不用ssh copy id root ip地址 因为如果ip比较多的情况下 ssh脚本添加会特别的麻烦 步骤1 首先要配置ansible清单 远程主机的密码这里为 12345
  • Idea上传已有项目到git

    开发经常遇到的问题是开发初期没有建立GIT仓库 开发一段时间后 需要将已有代码上传到Git 怎么将已有项目与新建的Git仓库相关联呢 借助Idea可以轻松实现 1 首先使用Git命令行 git clone XXXXX git 将项目下载 2
  • JAVA控制台输出格式

    public class PrintFormat public static void main String args Console print format System out printf format args format为指
  • Python键鼠操作自动化库PyAutoGUI简介

    PyAutoGUI是一个Python语言的键鼠自动化库 简单来说和按键精灵的功能一样 但是因为是Python的类库 所以可以使用Python代码配合一些其他类库完成更加强大的功能 下面让我为大家介绍一下吧 安装 从pip安装即可 pip i
  • 【threejs 】添加标签和射线

    three 添加标签 应用 上一篇文章我们说了世界坐标和屏幕坐标的准换那么有什么应用呢 应用 可以实现该效果 鼠标移动该模块的时候展示该模块的标签 或者可以常显 射线就是当鼠标移动到该模块该模块变化颜色 创建一个标签js文件作为封装的方法
  • 深入理解文字高度和行高的设置

    font size设置的是什么 line height设置的是什么 各种行高是怎么计算出来的 你真的知道吗 1 从font size讲起 说文字高度 当你按住鼠标左键选中一段文字的时候 这段文字背后会有一个颜色变化的区域 这个区域可以近似的