Android屏幕适配 px,dp,dpi及density的关系与深入理解

  • Post author:
  • Post category:其他







PX(pixel):


即传统计算机语言中描述的像素,在Android则代表绝对像素。


之所以Android中不推荐使用这种单位,正是因为不同生产厂商,不同品牌,不同屏幕的设备,其分辨率亦不一。




举例来说,我们现在将某个Button的width设为160px,则会出现如下情况:


在分辨率为“320宽”的设备里,该按钮显示占屏幕宽度一半;


在分辨率为“640宽”的设备里,该按钮显示占屏幕宽度的四分之一;





DPI(Dots Per Inch):


为了避免上面说到的使用px在屏幕适配中带来的问题,Android引入了一个新的单位:dp/dip。


而在理解“dp”之前,我们更有必要先了解一下另一个概念。正是:dpi。


也有人讲dpi称为“屏幕密度”。其含义则是:每英寸所打印的点数,既每一英寸的屏幕所包含的像素数。




举例来说,假设现在有一台“宽2英寸,长3英寸”的设备,则:


  • 当该设备分辨率为“320*480”,则dpi值为160。

  • 当该设备分辨率为“640*960”,则dpi值为320。


而“dpi”值越高的设备,其屏幕显示画面的效果也就越精细。




使用场景:


正是因为dpi值其代表的特性,所以android项目的资源文件下存在以下目录:


  • drawable-ldpi    ( 当dpi为120时,使用此目录下的资源)

  • drawable-mdpi    ( 当dpi为160时,使用此目录下的资源)

  • drawable-hdpi    ( 当dpi为240时,使用此目录下的资源)



  • drawable-xhdpi   ( 当dpi为320时,使用此目录下的资源)

  • drawable-xxhdpi  ( 当dpi为480时,使用此目录下的资源)


Android正是根据设备DPI值得不同,选择清晰度不同的资源使用,完成屏幕的适配。





DP/DIP(device independent pixels):


与我们之前谈到的绝对密度“px”对应,Android中引入的“dp”代表的则是“设备独立像素”。


该单位是为支持WVGA、HVGA和QVGA而使用的,其不再依赖像素本身,而是和屏幕密度相关。




在Android当中规定:


在屏幕密度为“160dpi”的情况下,则刚好“1dp = 1px”。


注:当屏幕密度为“320dpi”时,则“1dp = 2px”,以此类推…….


也正是因此,让我们得以保证了:控件在不同密度的屏幕上显示一致,既完成屏幕适配。


使用场景:


让我们回到上面说到的使用px造成的控件显示问题,此时我们将使用新的单位“dp”。于是:


  • 在分辨率320*480(既dpi为160)的设备下,则160dp等价于160px,按钮占屏幕宽的一半。

  • 在分辨率640*960(既dpi为320)的设备下,则160dp等价于320px,按钮依然占屏幕宽的一半。


Density:


就这个单词本身直接翻译的意思而言,其也代表“密度”。


但需要注意的是,在Android中,其实并非如此。


注意我们这里指的是,通过代码“context.getResources().getDisplayMetrics().density”获取的“density”值。

而通过该方法获取到的该值,实际上是等价于“dpi / 160”的一个结果值。也就是说:

“getResources().getDisplayMetrics().density” = “getResources().getDisplayMetrics().densityDpi / 160”


看到这样一个解析,聪明的人大概已经能预见什么了。我们似乎发现了某种关联:

在Android里:“dpi = 160,则1dp = 1px”、“dpi = 320,则1dp = 2px”。以此类推。

到此你已经发现,dp,px与160之间存在着某种规律:“1dp = (dpi / 160)px”

换算一下,最终得到公式:

dp = density * px

到了这里我们明白了,其实Android提供的该值,也就是为了让我们在dp与px之间做转换。

归根结底,其目的还是为了帮助我们做屏幕适配。




使用场景:

虽然使用dp在xml文件中定义控件尺寸,能够很好的帮助我们完成适配。

但很多时候,我们也会需要在Java代码中动态的去设定控件的尺寸。

但由于在代码中的尺寸设定,基本都被默认为了px单位。

所以这个时候就可以借助“density”来帮我们完成dp与px的转换,从而完成适配。


这也是为什么,我们可以在网上查到类似的工具类代码:


   public static int dip2px(Context context, float dipValue){ 
                final float scale = context.getResources().getDisplayMetrics().density; 
                return (int)(dipValue * scale + 0.5f); 
        } 
        
    public static int px2dip(Context context, float pxValue){ 
                final float scale = context.getResources().getDisplayMetrics().density; 
                return (int)(pxValue / scale + 0.5f); 
        } 

注:不要奇怪,熟悉的Java的特性的你应该明白,“+0.5f”是为了避免在类型强制转换中可能造成的精度丢失.




到了这里,我们总算小有收获。最后,通过一段代码,来验证一下我们的总结和猜想:


DisplayMetrics metrices = getResources().getDisplayMetrics();
int dpi = metrices.densityDpi;
float density = metrices.density;
float width = metrices.widthPixels;
float height = metrices.heightPixels;
		
Log.i("dpi==>", dpi+"");
Log.i("density==>", density+"");
Log.i("width==>", width+"");
Log.i("height==>", height+"");

查看打印结果:












版权声明:本文为ghost_Programmer原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。