深入理解移动端适配与探究其解决方案

最近在重构公司的一个移动端项目,除了需要对新项目进行前端技术栈的搭建外,还需要考虑的一个重要问题就是移动端适配,关于移动端适配的理解我之前一直是处于一种比较朦胧的状态(知其然而不知其所以然),所以最近又做了进一步的学习,在该博文中谈谈我对移动端适配的理解。

在这篇博文中,我会先对移动端设备的一些基础概念做一些解释,包括设备独立像素、设备像素比和 viewport ,然后在此基础上对移动端适配做深入的说明,最后是探究移动端适配的解决方案。

设备独立像素(dip)、设备像素比(dpr)

首先让我们来了解一下关于移动设备中的各个概念:

  • 设备像素:即物理像素,指设备能控制显示的最小单位,就是显示屏上一个个的像素点。
  • 屏幕尺寸:指屏幕的对角线长度,单位是英寸,1英寸=2.54厘米。
  • 屏幕分辨率:指手机屏幕的物理像素点数,一般以「纵向物理像素点数*横向物理像素点数」表示。如 iphone 6 的屏幕分辨率为 1334 * 750。
  • 屏幕像素密度(dpi/ppi):指手机屏幕上每英寸物理像素点数。其值与屏幕分辨率和屏幕尺寸有关,计算公式是:dpi = √(纵向物理像素点数² + 横向物理像素点数²) / 屏幕尺寸。

设备独立像素(dip)也称为逻辑像素、密度独立像素,指独立于设备的用于逻辑上衡量像素的单位。

设备像素比(dpr)指的是物理像素与设备独立像素的比例。在程序中则可以通过 window.devicePixelRatio 来获取,该属性是只读的,但不是常量,对浏览器的一些操作会改变这个值。

那么各个移动设备的设备像素比是怎么得出来的呢?

设备像素比是跟该移动设备的屏幕像素密度有关的。一般来讲,设备像素比是屏幕像素密度除以 160 的整数倍,即 dpr = Math.floor(dpi / 160) = Math.floor(√(纵向物理像素点数²+横向物理像素点数²) / 屏幕尺寸 / 160) 。

如 iphone 6 尺寸为 4.7 英寸,屏幕分辨率为 1334 * 750,那么我们可以得出 iphone 6 的设备像素比为:Math.floor(√(1334²+750²) / 4.7/160) = 2。

前面说到,设备像素比指的是物理像素与设备独立像素的比例,所以在知道了移动设备的设备像素比之后,我们便可以得出该移动设备的设备独立像素,即设备独立像素 = 物理像素 / 设备像素比

如 iphone 6 的横向分辨率是 750,设备像素比是 2,那么可以得出 iphone 6 的逻辑宽度(即以设备独立像素来计算的移动设备的宽度)是 375px 。

viewport

viewprot 指的是移动设备浏览器中放置页面的一个虚拟的窗口,该窗口可大于或小于移动设备的可视区域。

一般移动设备默认都是 viewprot 大于其可视区域,这样不会破坏没有针对移动设备优化的网页的布局,用户可以通过平移和缩放来看网页的其他部分,大部分移动设备默认的 viewport 为 980px(这里的 px 指的就是设备独立像素)。

我们在进行移动端页面的重构时,经常会加上如下一句代码:

<meta name="viewport" content="width=device-width,initial-scale=1.0,maximum-scale=1.0,user-scalable=no"/>

这句代码的作用是将移动设备的 viewport 设置为该设备的逻辑宽度,同时初始缩放值为 1 最大缩放值为1,并且不允许用户进行缩放。

让我们来看一下 viewport 中的属性:

当我们把 viewport 中的 width 设置为 width-device 时,viewport 的宽度等于移动设备的逻辑宽度,而如果设置 initial-scale 为 1 也会得到相同结果。一般来讲我们会同时设置这两个属性,当这两个属性因为值不同而产生不同的效果时,浏览器则会取得两者中较大的值。

下面举个例子来帮助理解 viewport 属性的作用:

<style>
html,body{
  margin:0;
  padding:0;
}
.box{
  width:100px;
  height:100px;
  background:blue;
}
</style>

<div class="box"></div>

在谷歌浏览器中以 iphone 6 的调试效果如下:

可以看到,在没有加上 viewport 属性的情况下,iphone 6 中 viewport 的默认值是 980px,前面说过,这个是以设备独立像素来计算的,也就是和我们平时在样式代码中写的 px 是一样的,代码中定义了一个宽高都为 100px 的蓝色块,所以从横向上来看的话,蓝色块在屏幕中占了 100/980,如图所示。

而如果我们设置了 viewport 属性的话:

<meta name="viewport" content="width=device-width,initial-scale=1.0,maximum-scale=1.0,user-scalable=no"/>

效果如下:

可以看到,在加上了 viewport 属性后,移动设备的 viewport 宽度被设置成了 device-width,即该移动设备的逻辑宽度 ,在 iphone 6 中 device-width 值为 375px,所以从横向上来看的话,蓝色块在屏幕中占了 100/375,如图所示。

深入理解移动端适配

前面我们已经理解了设备独立像素、设备像素比和 viewport 的一些概念,那么要深入地去理解移动端适配,我们需要先弄清楚设备像素比的作用是什么。

举一个最容易理解的例子。iphone 3 和 iphone 4 的屏幕尺寸是一样的,但是 iphone 3 的屏幕分辨率是 480 * 320,而 iphone 4 的屏幕分辨率是 960 * 640,iphone 4 的横纵向分辨率分别是 iphone 3 横纵向分辨率的两倍。

假设 iphone 3 和 iphone 4 的设备像素比都是 1 的话,那么可以得到它们的逻辑宽度分别是 320px 和 640px,从上面加了 viewport 属性的例子来看的话,在 iphone 3 中蓝色块的宽度占比是 100/320,而在 iphone 4 中占比是 100/640,我们就会看到同样的代码在不同的移动设备中的显示差别很大。

而实际上 iphone 3 的设备像素比是 1,iphone 4 的设备像素比是 2,所以我们可以得到它们的逻辑宽度都是 320px,所以实际上蓝色块在这两款不同的移动设备中显示的大小是一样的。

所以我们可以得出,使用 px 像素单位时,同样样式的代码在不同屏幕分辨率的移动设备中显示的情况基本一致(注意这里是基本一致而不是完全一致),是因为移动设备中的设备像素比将设备独立像素转换了。

而为什么说在不同分辨率的移动设备中显示的情况基本一致而不会完全一致呢?是因为存在很多不同屏幕分辨率的移动设备,它们的逻辑宽度并不都是一样的。

如 iphone 6 的分辨率是 1334 * 750,设备像素比是 2,所以其逻辑宽度是 375px,从上面加了 viewport 属性的例子来看的话,蓝色块占比是 100/375,可以看到其蓝色块在屏幕中的占比跟 iphone 3、iphone 4 是不一样的。

所以我们可以知道,设备像素比在一定程度上帮助我们进行了移动端适配。实际上,我们还需要通过一些其它方案来对移动端进行适配。我们需要进行移动端适配的根本原因是存在多种不同逻辑宽度的移动设备

我们对移动端进行适配的一个终极目标就是,使其在不同移动设备中所占的比例是大致相同(对于上图例子来说,就是使蓝色块在不同的移动设备中所占的比例差不多)。

移动端适配解决方案

最后,是探究移动端适配的解决方案,这里我要为 《从网易与淘宝的font-size思考前端设计稿与工作流》 这篇博文打 call,写得真的很不错。在这篇博文中,分别介绍了拉勾、网易和淘宝的移动端适配解决方案,拉勾是通过设置样式的百分比来达到在一定程度上对不同分辨率的移动设备进行适配,这种方式适合简单项目的适配;而网易和淘宝使用的则是众所周知的方案,使用 rem 进行适配。

相对而言,我觉得网易的方案更加容易理解和实践,所以在我的项目中也是参考网易的方案来实现移动端适配的。而对于淘宝的移动端适配方案,则显得更加精巧,感兴趣的童鞋可以自行阅读理解哦。

那么,rem 是什么呢?rem(font size of the root element),意思即根据根元素的 font-size 来设置字体的大小。跟 px 一样,它是 CSS 中的一个样式单位,会根据根元素的 font-size 值来转换成 px 单位,公式为:px = rem * html(font-size)。

html{
  font-size:10px;
}
div{
  width:2rem;  // 2*10=20px
}

所以我们实现移动端适配的核心思想就是:使用 rem 作为样式单位,根据不同分辨率的移动设备设置根元素的 font-size 值

那么问题来了,如何合理地设置根元素的 font-size 值呢?

以 iphone 6 的设计稿为基准,即设计稿横向分辨率为 750,取 100 为参照数(即在使用 rem 时与使用 px 时相差 100 的倍数),则我们可以知道 html 的宽度为 7.5rem(750 / 100),而我们知道 iphone 6 的逻辑宽度是 375px,所以 html 的宽度也为 375px,那么此时 7.5rem * html(font-size) = 375px,所以可以得出 html(font-size) = 375 / 7.5,即 html(font-size) = deviceWidth / 7.5

通过 js 来设置根元素的 font-size

var deviceWidth = document.documentElement.clientWidth;
document.documentElement.style.fontSize = deviceWidth / 7.5 + 'px';

当然,这里有一个前提,就是设置 viewprot 宽度为移动设备的逻辑宽度

<meta name="viewport" content="width=device-width,initial-scale=1.0,maximum-scale=1.0,user-scalable=no"/>

而当 deviceWidth 大于 750px 时,我们应该去访问的是 pc 版的页面,所以当 deviceWidth 大于 750px 时我们不应该再改变根元素的 font-size 值,完整的代码如下

var deviceWidth = document.documentElement.clientWidth;
if(deviceWidth > 750) deviceWidth = 750;
document.documentElement.style.fontSize = deviceWidth / 7.5 + 'px';

而为了使我们在书写样式的时候跟设计稿的大小更加契合,可以通过 sass 的 function 来设置一个 px 与 rem 之间的转换函数

@function pxToRem($num) {
  @return ($num/100) * 1rem;
}

当设计稿中有一个宽高都为 100px 的元素时,我们便可以如下写样式

div{
  width:pxToRem(100);
  height:pxToRem(100);
}

还需要注意的是为了使字体在不同分辨率的移动设备中看起来更加舒适,其大小不应该用 rem,而应该使用 px。

最后,总结一下我在项目中使用的移动端适配方案:

  • 将 viewport 宽度设置为移动设备逻辑宽度;
  • 使用 js 根据不同分辨率的移动设备来设置根元素的 font-size 值,注意移动端与 pc 端的临界值;
  • 在样式中字体使用 px 单位,而其它元素使用 rem 单位;
  • 使用 sass 中的 function 来设置一个 px 与 rem 之间的转换函数;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容