来聊一聊Android适配吧

窗外的山竹,在电线杆上多嘴 你说这一句 很有末日的感觉。山竹来了,连饭都没得吃,简直不要太恐怖。

说在前面

为什么要做屏幕适配?我前几天在郭霖大大的公众号里面看到一张图片,我觉得很有说明性的。

由于Android系统的开放性,任何用户、开发者、硬件厂商、运营商都可以对Android系统和硬件进行定制,修改成他们想要的样子。 上图就代表了各大品牌的手机碎片化的现状。随着Android设备的增多,设备碎片化、系统碎片化、屏幕尺寸碎片化、屏幕碎片化的程度也在不断加深。

当Android系统、屏幕尺寸、屏幕密度出现碎片化的时候,就很容易出现同一元素在不同手机上显示不同的问题。有的时候可能在4.3寸屏幕上面的样子是完美的,但是当其安装到5.7屏幕上面的就会出现很大的问题。

为了保证不同分辨率、屏幕大小的用户体验效果一致,从而引发了适配的课题

基本概念想·

1、 像素:

  • 含义:通常所说的像素,就是CCD/CMOS上光电感应元件的数量,一个感光元件经过感光,光电信号转换,A/D转换等步骤以后,在输出的照片上就形成一个点,我们如果把影像放大数倍,会发现这些连续色调其实是由许多色彩相近的小方点所组成,这些小方点就是构成影像的最小单位“像素”(Pixel)。简而言之,像素就是手机屏幕的最小构成单元。

  • 单位:px(pixel),1px = 1像素点 一般情况下UI设计师的设计图会以px作为统一的计量单位。

2、 分辨率:

  • 含义:手机在横向、纵向上的像素点数总和 一般描述成 宽*高 ,即横向像素点个数 * 纵向像素点个数(如1080 x 1920)。

  • 单位:px(pixel),1px = 1像素点

3、 屏幕尺寸(in):

  • 含义:手机对角线的物理尺寸

  • 单位 英寸(inch),一英寸大约2.54cm 常见的尺寸有4.7寸、5寸、5.5寸、6寸

4、 屏幕像素密度(dpi):

  • 含义:每英寸的像素点数。 例如每英寸内有160个像素点,则其像素密度为160dpi。

  • 单位:dpi(dots per inch)

  • 计算公式: 像素密度 = 像素 / 尺寸 (dpi = px / in)

  • 标准屏幕像素密度(mdpi): 每英寸长度上还有160个像素点(160dpi),即称为标准屏幕像素密度(mdpi)。


  • 屏幕尺寸、分辨率、像素密度三者关系:
    一部手机的分辨率是宽x高,屏幕大小是以寸为单位,那么三者的关系是:


5、 密度无关像素(dp):

  • 含义:density-independent pixel,叫dp或dip,与终端上的实际物理像素点无关

  • 单位:dp,可以保证在不同屏幕像素密度的设备上显示相同的效果,是安卓特有的长度单位。

6、 独立比例像素(sp):

  • 含义:scale-independent pixel,叫sp或sip

  • 单位:sp,字体大小专用单位 Android开发时用此单位设置文字大小,可根据字体大小首选项进行缩放; 推荐使用12sp、14sp、18sp、22sp作为字体大小,不推荐使用奇数和小数,容易造成精度丢失,12sp以下字体太小。

7、sp 与 dp 的区别:

  • dp只跟屏幕的像素密度有关;

  • sp和dp很类似但唯一的区别是,Android系统允许用户自定义文字尺寸大小(小、正常、大、超大等等),当文字尺寸是“正常”时1sp=1dp=0.00625英寸,而当文字尺寸是“大”或“超大”时,1sp>1dp=0.00625英寸。类似我们在windows里调整字体尺寸以后的效果——窗口大小不变,只有文字大小改变。

追到android源码,发现系统内部用applyDimension() (路径:android.util.TypedValue.applyDimension())将所有单位都转换成px 再处理:

  /**
     * Converts an unpacked complex data value holding a dimension to its final floating 
     * point value. The two parameters <var>unit</var> and <var>value</var>
     * are as in {@link #TYPE_DIMENSION}.
     *  
     * @param unit The unit to convert from.
     * @param value The value to apply the unit to.
     * @param metrics Current display metrics to use in the conversion -- 
     *                supplies display density and scaling information.
     * 
     * @return The complex floating point value multiplied by the appropriate 
     * metrics depending on its unit. 
     */
    public static float applyDimension(int unit, float value,
                                       DisplayMetrics metrics)
    {
        switch (unit) {
        case COMPLEX_UNIT_PX:
            return value;
        case COMPLEX_UNIT_DIP:
            return value * metrics.density;
        case COMPLEX_UNIT_SP:
            return value * metrics.scaledDensity;
        case COMPLEX_UNIT_PT:
            return value * metrics.xdpi * (1.0f/72);
        case COMPLEX_UNIT_IN:
            return value * metrics.xdpi;
        case COMPLEX_UNIT_MM:
            return value * metrics.xdpi * (1.0f/25.4f);
        }
        return 0;
    }

      可以发现dp和sp的区别在于density和scaledDensity两个值上;

    /**
     * The logical density of the display.  This is a scaling factor for the
     * Density Independent Pixel unit, where one DIP is one pixel on an
     * approximately 160 dpi screen (for example a 240x320, 1.5"x2" screen), 
     * providing the baseline of the system's display. Thus on a 160dpi screen 
     * this density value will be 1; on a 120 dpi screen it would be .75; etc.
     *  
     * <p>This value does not exactly follow the real screen size (as given by 
     * {@link #xdpi} and {@link #ydpi}, but rather is used to scale the size of
     * the overall UI in steps based on gross changes in the display dpi.  For 
     * example, a 240x320 screen will have a density of 1 even if its width is 
     * 1.8", 1.3", etc. However, if the screen resolution is increased to 
     * 320x480 but the screen size remained 1.5"x2" then the density would be 
     * increased (probably to 1.5).
     *
     * @see #DENSITY_DEFAULT
     */
    public float density;

    /**
     * A scaling factor for fonts displayed on the display.  This is the same
     * as {@link #density}, except that it may be adjusted in smaller
     * increments at runtime based on a user preference for the font size.
     */
    public float scaledDensity;

屏幕适配的方案:

  • 图片适配

关于图片适配呢?我两个点我要提及:

1、 资源位置:

前面的介绍我们可以知道:在AndroidStudio的资源目录res下有五个层级图片文件夹,分别用来存放不同分辨率的图片:

drawable-ldpi :低分辨率(用的少了,一般不再用)

drawable-mdpi:中分辨率

drawable-hdpi:高分辨率

drawable-xdpi:较高分辨率

drawable-xxdpi:超级高分辨率

drawable-xxxhpi:顶级分辨率

在对应的文件夹下放置不同分辨率的图片就可以很好的对图片进行适配。

随着屏幕越来越大,推荐xxdpi的一套切图,这样就可以向下和向上兼容,节省资源。

2、 图片拉伸:

如果对于Android有一定的了解的开发者,都可能会晓得imageview有一个scaleType用来适配图片的大小:

  • android:scaleType=“center” 保持原图的大小,显示在ImageView的中心。当原图的size大于ImageView的size时,多出来的部分被截掉。

  • android:scaleType=“center_inside” 以原图正常显示为目的,如果原图大小大于ImageView的size,就按照比例缩小原图的宽高,居中显示在ImageView中。如果原图size小于ImageView的size,则不做处理居中显示图片。

  • android:scaleType=“center_crop” 以原图填满ImageView为目的,如果原图size大于ImageView的size,则与center_inside一样,按比例缩小,居中显示在ImageView上。如果原图size小于ImageView的size,则按比例拉升原图的宽和高,填充ImageView居中显示。

  • android:scaleType=“matrix” 不改变原图的大小,从ImageView的左上角开始绘制,超出部分做剪切处理。

  • androd:scaleType=“fit_xy” 把图片按照指定的大小在ImageView中显示,拉伸显示图片,不保持原比例,填满ImageView.

  • android:scaleType=“fit_start” 把原图按照比例放大缩小到ImageView的高度,显示在ImageView的start(前部/上部)。

  • android:sacleType=“fit_center” 把原图按照比例放大缩小到ImageView的高度,显示在ImageView的center(中部/居中显示)。

  • android:scaleType=“fit_end” 把原图按照比例放大缩小到ImageView的高度,显示在ImageVIew的end(后部/尾部/底部)

  • 布局适配

关于布局适配呢?我也有两点要进行提及:

1、 建议:

不要使用绝对布局,使用相对布局和线性布局来代替绝对布局。

2、 资源位置:

在屏幕适配中,我们也可以和图片一样:根据手机大小不一样的手机建立不同的布局,比如说创建两个文件夹:

  • layout-800 * 480
  • layout-1280 * 720
    手机会根据分辨率去找设定的不同大小的layout的布局。
  • Contractlayout适配:

我在这里呢? 就不讲这个了,详情呢?请查看我前面的文章ConstraintLayout用法详解。顺便也关注一下

  • 尺寸适配(dimens适配)

在屏幕适配中,我们也可以和图片一样:根据手机大小不一样的手机建立不同的dimens,比如说创建两个文件夹:

  • values-400*320
  • values-800*480
    手机会根据分辨率去找设定的不同大小的dimens的参数。
  • 权重适配:

当布局占满屏幕宽或高的时候,子布局可以使用权重适配。例如LinearLayout中的weight属性。

  • dp dp dp

android的单位dp本身就有适配的功能,如下图所示:



所以用好dp,本身就是一种适配,下面附一下dp转px的方法 供日后查看:

fun Int.dp2Px(context: Context): Int =
        TypedValue.applyDimension(
                TypedValue.COMPLEX_UNIT_DIP,
                this.toFloat(),
                context.resources.displayMetrics
        ).toInt()

fun Int.px2Dp(context: Context): Int =
        (this.toFloat() / context.resources.displayMetrics.density).toInt()

fun Int.sp2Px(context: Context): Int =
        TypedValue.applyDimension(
                TypedValue.COMPLEX_UNIT_SP,
                this.toFloat(),
                context.resources.displayMetrics
        ).toInt()

fun Int.px2Sp(context: Context): Int =
        (this.toFloat() / context.resources.displayMetrics.scaledDensity).toInt()

fun Float.dp2Px(context: Context): Int =
        TypedValue.applyDimension(
                TypedValue.COMPLEX_UNIT_DIP,
                this,
                context.resources.displayMetrics
        ).toInt()

fun Float.sp2Px(context: Context): Int =
        TypedValue.applyDimension(
                TypedValue.COMPLEX_UNIT_SP,
                this,
                context.resources.displayMetrics
        ).toInt()
  • 代码的适配:

       //获取手机屏幕的宽和高
        val widthPixels = resources.displayMetrics.widthPixels
        val heightPixels = resources.displayMetrics.heightPixels
        //给button设置宽和高
        val layoutParams = bt_main_button.getLayoutParams()
        layoutParams.width = widthPixels / 2
        layoutParams.height = heightPixels / 2
        bt_main_button.setLayoutParams(layoutParams)

由上面代码可知:我们可以计算屏幕的大小 来确定控件的大小 这样就能达到适配的效果。

说在最后:

适配是一门大学问,无论是刚入行的新手 还是入行多年的老手 对于适配其实或多或少都有点怵的。算了,我不知道哔哔什么了,窗外的那个吊车被台风吹到了我的窗前,我现在有点虚 时刻都在注意着它的“走位”,生怕他一个e闪将我留在这里。不写了不写了。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容