让应用自动适配多个分辨率的屏幕,是每个android程序员的基本功,就好像前端工程师熟练编写CSS Hack一样。适配工作中一个重要的工作就是对页面的调整。

对于页面的适配,有很多的方法和技巧。比如布局中尽量使用wrapcontent ,fillparent,尽量避免具体的数字,由系统来计算合适的宽高;或者为每个分辨率写一套布局文件,设置对应分辨率下控件的宽高;

为每一个分辨率写一套布局文件虽然够独立,够简单。但是维护起来成本较高。一个页面的改动,往往涉及多个布局文件的改动,让人很痛苦。

小技巧

我们可以尝试只写一套xml布局,然后为该布局准备多套dimension文件。

说的详细一点就是,xml布局中组件的宽高,不要使用具体的数值来表示,而是配置到dimension文件中。每套dimension文件中数值的大小都是成比例计算出来。

比如在1980*1080分辨率下,定义 px15表示15px

<dimen name= "px15" >15px</ dimen>

那么在 1080 * 720分辨率下,px15要成比例缩小1.5倍, 定义px15 表示 10px

<dimen name= "px15" >10px</ dimen>

所以在xml布局文件中,我们可以这样来表示:

<LinearLayout
 
android:layout_width="@dimen/px150"
 
android:layout_height="@dimen/px15"
 
android:orientation="vertical" >
 
    ……
 
    ……
 
</LinearLayout>

这套布局文件中的LinearLayout 在1980 * 1080 分辨率下的宽高为 150 x 15  , 在 1080 * 720分辨率下的宽高就会自动变成 100 * 10

其他分辨率同理

疑问

1.有的同学会疑问,这样不就变成需要维护多套dimenson文件了?换汤不换药呀?

其实不然,对于dimension文件我们可以使用代码来控制生成,数值范围可以根据自己的情况来。其他分辨率下只需要按照相应比例,使用代码算一下即可。

编写一个这样的生成代码并不难,下篇文章我们再给出。

生成完毕后,Values 目录结构如下:

2. 按比例计算布局一定可靠吗,会不会出现混乱的现象

有可能会,这个时候就需要协调布局使用的宽高,选择合适的宽高让页面在各个分辨率下,看起来不算离谱就行,不一定严格按照设计来。大部分页面是兼容的。

以上介绍了使用dimension文件做适配。说道了使用代码自动生成所有的dimension文件,接下来我们给出相关代码。

DimensTools:

package com.example.test;
import java.io.*;
import java.util.*;
/**
* dimens数据自动生成工具
* 
*/
public class DimensTools {
     /** 源文件 */
     static String oldFilePath = "./res/values-nodpi/dimens.xml";
     /** 新生成文件路径 */
     static String filePath720 = "./res/values-1280x720/dimens.xml";
     /** 新生成文件路径 */
     static String filePath672 = "./res/values-1280x672/dimens.xml";
     /** 新生成文件路径 */
     static String filePath1080 = "./res/values-1920x1080/dimens.xml";
     /** 缩小倍数 */
     static float changes = 1.5f;
     public static void main(String[] args) {
          //生成1-1920px
          String allPx= getAllPx();
          DeleteFolder(oldFilePath);
          writeFile(oldFilePath, allPx);
          String st = convertStreamToString(oldFilePath, changes);
          DeleteFolder(filePath720);
          writeFile(filePath720, st);
          DeleteFolder(filePath672);
          writeFile(filePath672, st);
          String st1 = convertStreamToString(oldFilePath, 1f);
          DeleteFolder(filePath1080);
          writeFile(filePath1080, st1);
     }
     /** 读取文件 生成缩放后字符串 */
     public static String convertStreamToString(String filepath, float f) {
          StringBuilder sb = new StringBuilder();
          try {
               BufferedReader bf = new BufferedReader(new FileReader(filepath));
               String line = null;
               System.out.println("q1");
               String endmark = "px</dimen>";
               String startmark = ">";
               while ((line = bf.readLine()) != null) {
                    if (line.contains(endmark)) {
                         int end = line.lastIndexOf(endmark);
                         int start = line.indexOf(startmark);
                         String stpx = line.substring(start + 1, end);
                         int px = Integer.parseInt(stpx);
                         int newpx = (int) ((float) px / f);
                         String newline = line.replace(px + "px", newpx + "px");
                         sb.append(newline + "\r\n");
                    } else {
                         sb.append(line + "\r\n");
                    }
               }
               System.out.println(sb.toString());
          } catch (IOException e) {
               e.printStackTrace();
          }
          return sb.toString();
     }
     /**
     * 根据路径删除指定的目录或文件,无论存在与否
     *
     * @param sPath
     *            要删除的目录或文件
     * @return 删除成功返回 true,否则返回 false。
     */
     public static boolean DeleteFolder(String sPath) {
          File file = new File(sPath);
          // // 判断目录或文件是否存在
          if (!file.exists()) { // 不存在返回 false
               return true;
          } else {
               // 判断是否为文件
               if (file.isFile()) { // 为文件时调用删除文件方法
                    return deleteFile(sPath);
               } else { // 为目录时调用删除目录方法
               // return deleteDirectory(sPath);
               }
          }
          return false;
     }
     /** 存为新文件 */
     public static void writeFile(String filepath, String st) {
          try {
               FileWriter fw = new FileWriter(filepath);
               BufferedWriter bw = new BufferedWriter(fw);
               bw.write(st);
               bw.flush();
               bw.close();
          } catch (IOException e) {
               e.printStackTrace();
          }
     }
     /** 生成全px文件 */
     public static String getAllPx() {
          StringBuilder sb = new StringBuilder();
          try {
               sb.append("<resources>" + "\r\n");
               sb.append("<dimen name=\"screen_width\">1920px</dimen>" + "\r\n");
               sb.append("<dimen name=\"screen_height\">1080px</dimen>" + "\r\n");
               for (int i = 1; i <= 1920; i++) {
                    System.out.println("i="+i);
                    sb.append("<dimen name=\"px" + i + "\">" + i + "px</dimen>"
                              + "\r\n");
               }
               sb.append("</resources>" + "\r\n");
               System.out.println(sb.toString());
          } catch (Exception e) {
               e.printStackTrace();
          }
          return sb.toString();
     }
     /**
     * 删除单个文件
     *
     * @param sPath
     *            被删除文件的文件名
     * @return 单个文件删除成功返回true,否则返回false
     */
     public static boolean deleteFile(String sPath) {
          boolean flag = false;
          File file = new File(sPath);
          // 路径为文件且不为空则进行删除
          if (file.isFile() && file.exists()) {
               file.delete();
               flag = true;
          }
          return flag;
     }
}

使用方法:cmd下使用javac ,java命令运行。这样有点费劲哈,改天用ant写个自动脚本放上来。

注:先建立好相应的文件夹,672也按照1.5的比例缩放的。可以根据自己的需要调整。

巧用dimens适配多个分辨率的更多相关文章

  1. Android 屏幕适配之dimens适配

    Android 屏幕适配之dimens适配     转  https://blog.csdn.net/github_2011/article/details/72636851 在过去多个项目中一直使用 ...

  2. 适配各种Windows分辨率,为DPI添加感知,当在高DPI时,禁用WINFORM缩放等。

    因为现在高分屏越来越多,很多windows设备必须设置高DPI,这样很容易导致WINFORM整体错位,因此我们需要自己适配.禁止缩放 在程序配置清单 mainfest中添加如下. <assemb ...

  3. dp 密度 分辨率 屏幕 状态栏 标题栏 适配

    一篇总结的非常完善的博文:http://www.jianshu.com/p/ec5a1a30694b 屏幕像素参数相关信息表格 屏幕级别 像素密度 每英寸像素数   通常分辨率 分辨率别称    默认 ...

  4. Android适配(屏幕适配、国际化适配)-转

    首先来说一下Android的屏幕适配: 关于Android屏幕的一些基本概念知识,自行充电..在此只介绍实际开发过程中的使用 1.说到Android的屏幕适配,首当其冲的就是图片的适配 图片适配遵循两 ...

  5. android 项目学习随笔二十(屏幕适配)

    1.图片适配 放入相同名称的资源文件,机器根据不同分辨率找相近的资源 240*320 ldpi 320*480 mdpi 480*800 hdpi 720*1280 xhdpi 2.布局适配 在不同的 ...

  6. Android多分辨率适配经验总结

      Android多分辨率适配是一件很有意义但是比较麻烦的事情,网上有很多关于多分辨率适配的文章,多数文章讲解的都是整个APP的图片比较规则,可以将图片做成9图来完成多分辨率适配,但是对于一些游戏类应 ...

  7. Android 屏幕适配方式

    适配:即当前应用在相同的手机上面显示相同的效果.适配前需要首先确定当前手机所属像素密度类型(如:xhdpi.hdpi.mdpi等) 像素密度:每英寸上分布的像素点个数,单位(dpi,ppi),利用勾股 ...

  8. Android 目前最稳定和高效的UI适配方案

    Android系统发布十多年以来,关于Android的UI的适配一直是开发环节中最重要的问题,但是我看到还是有很多小伙伴对Android适配方案不了解.刚好,近期准备对糗事百科Android客户端设计 ...

  9. android 手机 多分辨率适配

    近来在做android屏幕适配这方面的工作, 今天总算有点眉目.  小记一下 基础知识就不科普了, 网上一大堆. 作为一个刚接触这方面人, 最先进入我脑子的, 是从小到大的各种屏, 小到手表, 大到街 ...

随机推荐

  1. 洛谷11月月赛round.1

    太感动了#2 thwfhk 240 (801ms) 100 100 40   又一张明信片,话说10月的怎么还没收到   P2246 SAC#1 - Hello World(升级版) 题目背景 一天, ...

  2. BZOJ1190[HNOI2007]梦幻岛宝石

    Description 给你N颗宝石,每颗宝石都有重量和价值.要你从这些宝石中选取一些宝石,保证总重量不超过W,且总价值最大为,并输出最大的总价值.数据范围:N<=100;W<=2^30, ...

  3. java多线程系类:JUC线程池:01之线程池架构

    概要 前面分别介绍了"Java多线程基础"."JUC原子类"和"JUC锁".本章介绍JUC的最后一部分的内容--线程池.内容包括:线程池架构 ...

  4. C# SaveFileDialog的用法(转载)

    #region 保存对话框private void ShowSaveFileDialog(){//string localFilePath, fileNameExt, newFileName, Fil ...

  5. 关于javascript中this的那点事

    this可谓是JavaScript中的开发神器,使用得当的话不仅有事半功倍的效果,而且代码的逼格也更高.但是既然是神器,如果你没有足够的功力的话,那么就不要使用它,否则就有可能自毁身亡.曾几何时,我偶 ...

  6. js 数组处理函数

    本文转载自有有<js 数组处理函数> concat 将参数列表连接到原数组后面形成一个新的数组并返回,原有数组不受影响. var arr = ["a","b& ...

  7. 12款经典的白富美型—jquery图片轮播插件—前端开发必备

    图片轮播是网站中的常用功能,用于在有限的网页空间内展示一组产品图片或者照片,同时还有非常吸引人的动画效果.本文向大家推荐12款实用的 jQuery 图片轮播效果插件,帮助你在你的项目中加入一些效果精美 ...

  8. Office2013插件开发Outlook篇(1)-- 第一个office2013插件

    一.环境: 下载VS2013安装,记得安装office插件开发包哦. 二.新建Outlook插件项目

  9. 【JavaScript】[bind,call,apply] (function cal(){}());声明函数立即执行

    ---恢复内容开始--- 1.js 里函数调用有 4 种模式:方法调用.正常函数调用.构造器函数调用.apply/call 调用.同时,无论哪种函数调用除了你声明时定义的形参外,还会自动添加 2 个形 ...

  10. linux基础知识与技能3

    3.2.vi的高级使用* 查找在命令模式下,输入/xxx,就可以查找到xxx * 快速切换行在命令模式下,输入:num,就可以快速切换到num行 * 设置显示行号在命令模式下,输入:set nu,就可 ...