一、基本概念
dip : Density independent pixels ,設(shè)備獨立像素。
dp :就是dip
px : 像素
dpi :dots per inch , 直接來說就是一英寸多少個像素點。常見取值 120,160,240。我一般稱作像素密度,簡稱密度。
density : 直接翻譯的話貌似叫 密度。常見取值 1.5 , 1.0 。
分辨率 : 橫縱2個方向的像素點的數(shù)量,常見取值 480720 ,320480,1080*1920
屏幕尺寸: 屏幕對角線的長度。
二、應(yīng)用
在android里面,獲取一個窗口的metrics,里面有這么幾個值
metrics.density;
metrics.densityDpi;
densityDpi : 就是我們常說的dpi。
density : 其實是 DPI / (160像素/英寸) 后得到的值。是不是有點奇怪,因為我?guī)Я藛挝弧?。。這個涉及到后面一個比較重要的東西,后面再說。
從上面就看得出了,DPI本身的單位也是 像素/英寸,所以density其實是沒單位的,他就是一個比例值。
而dpi的單位是 像素/英寸,比較符合物理上面的密度定義,密度不都是單位度量的值么,所以我更喜歡把dpi叫像素密度,簡稱密度,density還是就叫density。
三、各單位間轉(zhuǎn)換
- 計算dpi
比如一個機(jī)器,屏幕4寸,分辨率480X800,他的dpi能算么。
因為不知道邊長,肯定不能分開算,4是對角線長度,那直接用勾股定理算對角線像素,除以4,算出來大概是 dpi = 233 像素/英寸。
那么density就是 (233 px/inch)/(160 px/inch)=1.46 左右
順帶說下,android默認(rèn)的只有3個dpi,low、medium和high,對應(yīng) 120、160、240,如果沒有特別設(shè)置,所有的dpi都會被算成這3個,其中的default就是160。
- 計算 dp 與 px
我們寫布局的時候,肯定還是要知道1個dp到底有多少px的。
換算公式如下: dp = (DPI/(160像素/英寸))*px = density *px
注意,這里都是帶單位的。px是單位,dp是單位,density沒單位。
為了方便,假設(shè)dpi是240 像素/英寸 , 那么density就是1.5
那么就是 dp=1.5px ,注意這是帶了單位的,也就是 設(shè)備無關(guān)像素 = density 像素
- 為啥 標(biāo)準(zhǔn)dpi = 160
(1)Android Design [1] 里把主流設(shè)備的 dpi 歸成了四個檔次,120 dpi、160 dpi、240 dpi、320 dpi
實際開發(fā)當(dāng)中,我們經(jīng)常需要對這幾個尺寸進(jìn)行相互轉(zhuǎn)換(比如先在某個分辨率下完成設(shè)計,然后縮放到其他尺寸微調(diào)后輸出),一般按照 dpi 之間的比例即 2:1.5:1:0.75 來給界面中的元素來進(jìn)行尺寸定義。
也就是說如果以 160 dpi 作為基準(zhǔn)的話,只要尺寸的 DP 是 4 的公倍數(shù),XHDPI 下乘以 2,HDPI 下乘以 1.5,LDPI 下乘以 0.75 即可滿足所有尺寸下都是整數(shù) pixel 。
但假設(shè)以 240 dpi 作為標(biāo)準(zhǔn),那需要 DP 是 3 的公倍數(shù),XHDPI 下乘以 1.333,MDPI 下乘以 0.666 ,LDPI 下除以 2
而以 LDPI 和 XHDPI 為基準(zhǔn)就更復(fù)雜了,所以選擇 160 dpi
(2)這個在Google的官方文檔中有給出了解釋,因為第一款A(yù)ndroid設(shè)備(HTC的T-Mobile G1)是屬于160dpi的。
轉(zhuǎn)換如下:
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);
}
用上面兩個方法在代碼里進(jìn)行轉(zhuǎn)換即可。