Android布局中長度單位的深入研究
要想使自己的布局在不同設備達到精準空置,理清理順Android布局長度單位之間關系很有必要,否則你也許會經常撓頭為什么顯示出來的布局不是自己定義的效果呢,有些東西,雖然基礎,但是弄個透徹也需要花些功夫,廢話不多說,下面開始。
1.先了解一下Android有支持哪些長度單位:
px: pixels(像素). 不同設備顯示效果相同,比如我們800*480的屏幕寬度就是 800px
dip: device independent pixels(設備獨立像素). 不同設備有不同的顯示效果,這個和設備硬件有關,通常屏幕大時,density就大,屏幕小時,density就小
屏幕實際分辨率為240px*400px時,densityDpi=120
屏幕實際分辨率為320px*533px,densityDpi=160
屏幕實際分辨率為480px*800px,densityDpi=240
而dip與px之間的換算關系是:
pixs =dips * (densityDpi/160),也就是說當densityDpi=160時,1dip=1px
sp: scaled pixels(放大像素),sp的大小取決于系統metrics.scaledDensity值大小
pt: point,是一個標準的長度單位,1pt=1/72英寸,用于印刷業(基本用不到)
pt與px的換算關系:pixs = pt*xdpi * (1.0f/72);xdpi表示1英寸像素個數
in(英寸)長度單位(基本用不到)
in與px的換算關系:pixs = in*xdpi
mm(毫米)長度單位(基本用不到)
mm與px的換算關系:pixs = mm * xdpi * (1.0f/25.4f)
2.系統獲取長度單位
看了上面具體長度單位的含義你會產生一個疑問,不同單位換算取決于系統的一些屬性,比如densityDpi的值,xdpi的值,那么系統這些值在哪里獲取了,直接看測試用例:
- public void testgetdisplay(){
- WindowManager wm = (WindowManager) this.getInstrumentation().getContext().getSystemService(Context.WINDOW_SERVICE);
- DisplayMetrics mDisplayMetrics = new DisplayMetrics();
- wm.getDefaultDisplay().getMetrics(mDisplayMetrics);
- System.out.println("display.height="+wm.getDefaultDisplay().getHeight());
- System.out.println("display.width="+wm.getDefaultDisplay().getWidth());
- System.out.println("densityDpi="+mDisplayMetrics.densityDpi);
- System.out.println("xdpi="+mDisplayMetrics.xdpi);
- System.out.println("density="+mDisplayMetrics.density);
- }
3.densityDpi與drawable-(hdpi,mdpi,ldpi)之間的關系
系統drawable有hdpi,mdpi,ldpi三個文件夾下面存放不同尺寸的圖片,使用哪個文件下的文件,與系統densityDpi值是有關系的。
densityDpi=120:ldpi densityDpi=160:mdpi densityDpi=240:hdpi
前面我又說過densityDpi取決于顯示屏,這樣你就了解了為什么不同顯示屏WVGA,HVGA,QVGA會采用不同drawable-(hdpi,mdpi,ldpi)圖片。
分辨率為240px*400px,densityDpi=120-->QVGA:ldpi
分辨率為320px*533px,densityDpi=160 -->HVGA:mdpi
分辨率為480px*800px,densityDpi=240 -->WVGA:WVGA
4.深入了解代碼
盡管了解上面這些理論值,但是有時候發現設置了不同長度單位,可顯示出來的效果卻出人預想,我曾經就碰到過這種撓頭的問題,為解決這個問題,只有深入代碼,一探究竟了。
在深入代碼前我們首先要搞清楚一個問題,那就是代碼中所有長度值的單位都是px,手上沒有現成的例子就以現在我研究的/Launcher2/res/layout-land/workspace_screen.xml為例,看一個自定義屬性:
launcher:cellWidth="105pt"
該屬性自定義了一個桌面快捷圖標的寬度,若讀者自己測試,自己寫個測試view,設置屬性:
android:layout_width="800px"
是一樣的。
當view被創建的時候,xml中的屬性值存在參數AttributeSet attrs中:
- public CellLayout(Context context, AttributeSet attrs, int defStyle)
繼續看該構造函數的實現代碼:
- public CellLayout(Context context, AttributeSet attrs, int defStyle) {
- super(context, attrs, defStyle);
- //獲取自定義屬性組CellLayout中的所有自定義屬性,關于自定義屬性,這里不作展開說明
- TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CellLayout, defStyle, 0);
- //獲取屬性cellWidth的值,長度單位將轉換為px
- mCellWidth = a.getDimensionPixelSize(R.styleable.CellLayout_cellWidth, 10);
- 。。。
- }
實現長度單位換算的關鍵代碼就在a.getDimensionPixelSize(R.styleable.CellLayout_cellWidth, 10),直接深入到關鍵代碼:
- public int getDimensionPixelSize(int index, int defValue)
- public static int complexToDimensionPixelSize(int data,DisplayMetrics metrics)
- 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;
- }
unit就是指單位類型,這個怎么來的我沒有,但我想它肯定是在解析xml是根據不同單位轉換的。