屏幕适配要点

安卓中度量单位介绍

inch:英寸,表示手机屏幕对角线的长度

px:像素,平时大家说的屏幕分辨率的单位,如1920*1080,表示屏幕竖直方向有1920个像素点,横向有1080个像素点

dpi:表示每英寸像素点的个数,计算方法:对角线像素点个数 / 屏幕的物理长度,根据不同的dpi安卓将设备分为几个常用的显示级别:从小到大:
ldpi、mdpi、hdpi、xhdpi、xxhdpi

ldpi mdpi hdpi xhdpi xxhdpi xxxhdpi
dpi 0-120 120-160 160-240 240-320 320-480 480-640

安卓中代码获取屏幕dpi

private void getDisplayInfo(){
    Resources resources=getResources();
    DisplayMetrics displayMetrics =  resources.getDisplayMetrics();
    float density = displayMetrics.density;
    int densityDpi = displayMetrics.densityDpi;
    System.out.println("----> density=" + density);  —-> density=3.0 
    System.out.println("----> densityDpi=" + densityDpi);  —-> densityDpi=480
}

代码获取的densityDpi代码屏幕的dpi,density是一个整数,它其实代表的是densityDpi与160的倍数,因为mdpi是安卓中的基准屏幕密度,而其值为160
而为什么要用mdpi作为基准屏幕密度呢?这就要提到另一个度量单位dp
dp: 表示与密度无关的像素,使用dp做为单位,可以在不同像素密度下达到自适应效果(但是dp只是在相同屏幕尺寸不同屏幕密度下有较好的适配效果,对于不同尺寸的屏幕却无能为力),而dp与dpi对应的关系如下表所示:

ldpi mdpi hdpi xhdpi xxhdpi xxxhdpi
dpi 0-120 120-160 160-240 240-320 320-480 480-640
比例 1dp=0.75px 1dp=1px 1dp=1.5px 1dp=2px 1dp=3px 1dp=4px

如表所示:在mdpi的时候,dp与px的比例关系是1:1关系,所以将mdpi作为基准屏幕密度

drawable下不同适配的不同分辨率的图片文件

在res下有drawable-ldpi、drawable-mdpi、drawable-hdpi、drawable-xhdpi、drawable-xxhdpi、drawable-xxxdpi文件夹,他与屏幕自身的屏幕密度是怎样适配的呢?
我们可以根据图片放在不同的文件夹下获取到对应的图片的dpi

private void getDrawableFolderDensity(){
    TypedValue typedValue = new TypedValue();
    Resources resources=mContext.getResources();
    int id = getResources().getIdentifier(imageName, "drawable" , packageName);
    resources.openRawResource(id, typedValue);
    int density=typedValue.density;
    System.out.println("----> density="+density);
}

如果将图片放入drawable-ldpi,则其TypedValue.density 的值为120
如果将图片放入drawable-mdpi,则其TypedValue.density的值为160
相对于的 drawable-hdpi为240
相对于的drawable-xdpi为320
相对于的drawable-xxdpi为480
最后系统会根据屏幕自身的屏幕密度去对应的文件夹中去找到对应的图片,主要代码如下

public static Bitmap decodeResource(Resources res, int id, Options opts) {
    Bitmap bm = null;
    InputStream is = null;
    try {
        final TypedValue value = new TypedValue();
        is = res.openRawResource(id, value);
        bm = decodeResourceStream(res, value, is, null, opts);
    } catch (Exception e) {

    } finally {
        try {
            if (is != null) is.close();
        } catch (IOException e) {

        }
    }

    if (bm == null && opts != null && opts.inBitmap != null) {
        throw new IllegalArgumentException("Problem decoding into existing bitmap");
    }
    return bm;
}

调用decodeResourceStream( )方法

public static Bitmap decodeResourceStream(Resources res, TypedValue value,
        InputStream is, Rect pad, Options opts) {
    if (opts == null) {
        opts = new Options();
    }
    if (opts.inDensity == 0 && value != null) {
        final int density = value.density;    //这里得到该图片对应的dpi值
        if (density == TypedValue.DENSITY_DEFAULT) { //default为0
            opts.inDensity = DisplayMetrics.DENSITY_DEFAULT;  //default为160
        } else if (density != TypedValue.DENSITY_NONE) {  //none为无穷大
            opts.inDensity = density;    //如果得到的图片的density的值不为0也不为无穷大,则将得到的图片的dpi赋值到optiotn中
        }
    }       
    if (opts.inTargetDensity == 0 && res != null) {
        opts.inTargetDensity = res.getDisplayMetrics().densityDpi;    //这里将屏幕密度赋值过去
    }     
    return decodeStream(is, pad, opts);
}
  1. 调用decodeStream()方法,在该方法中会调用decodeStreamInternal();它又会继续调用nativeDecodeStream( ),该方法是native的;在BitmapFactory.cpp可见这个方法内部又调用了doDecode()它的核心源码如下:
static jobject doDecode(JNIEnv*env,SkStreamRewindable*stream,jobject padding,jobject options) {
......
if (env->GetBooleanField(options, gOptions_scaledFieldID)) {
    const int density = env->GetIntField(options, gOptions_densityFieldID);
    const int targetDensity = env->GetIntField(options, gOptions_targetDensityFieldID);
    const int screenDensity = env->GetIntField(options, gOptions_screenDensityFieldID);
    if (density != 0 && targetDensity != 0 && density != screenDensity) {
        scale = (float) targetDensity / density;    //这里赋值了缩放比例:屏幕的像素密度/图片缩放文件夹下的像素密度
    }
}
}
const bool willScale = scale != 1.0f;
......
SkBitmap decodingBitmap;
if (!decoder->decode(stream, &decodingBitmap, prefColorType,decodeMode)) {
return nullObjectReturn("decoder->decode returned false");
}
int scaledWidth = decodingBitmap.width();    //这里得到原始图片的宽度
int scaledHeight = decodingBitmap.height();
if (willScale && decodeMode != SkImageDecoder::kDecodeBounds_Mode) {
scaledWidth = int(scaledWidth * scale + 0.5f);  //对图片进行缩放,这里为什么要加0.5f ?
scaledHeight = int(scaledHeight * scale + 0.5f);
}
if (willScale) {
const float sx = scaledWidth / float(decodingBitmap.width());  //再重新计算图片的缩放比例,用来缩放canvas
const float sy = scaledHeight / float(decodingBitmap.height());
......
SkPaint paint;
SkCanvas canvas(*outputBitmap);
canvas.scale(sx, sy);
canvas.drawBitmap(decodingBitmap, 0.0f, 0.0f, &paint);
}
......
}

总结:图片适配过程,系统会优先根据当前的屏幕密度去对应的相同密度的drawable文件夹下面找对应的图片,如果找不到,则会去比当前密度更高的文件夹下去找(毕竟在更高的屏幕密度下,图片缩放后不会失真),高的找不到,再去找低的(注意:drawable和mdpi-drawable两个文件夹下的屏幕密度是相同的,所以去低像素密度文件夹下找时,在找完mdpi后,会先去drawable文件夹下找,找不到再去ldpi下去找)如果在别的像素密度的文件夹下找到图片,则会用当前屏幕密度与drawable文件夹密度之商来缩放文件夹下的图片。例如:当前屏幕密度为hdpi(240dpi),图片放在xhdpi的drawable文件夹下,则系统拿到的图片将会缩放 240 / 320倍数,最终拿缩放后的图片显示在屏幕上。
http://blog.csdn.net/xiebudong/article/details/37040263

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

推荐阅读更多精彩内容

  • 本文参考自: Google的官方权威适配文档 郭霖:Android官方提供的支持不同屏幕大小的全部方法 Storm...
    M悇芐冋忆阅读 12,794评论 5 56
  • 本文记录一些适配问题的研究,基础概念不做过多介绍。 Android在做屏幕适配的时候一般考虑两个因素:分辨率和dp...
    developerzjy阅读 7,264评论 1 24
  • 几组概念 分辨率屏幕上物理像素的总数。添加对多种屏幕的支持时, 应用不会直接使用分辨率;而只应关注通用尺寸和密度组...
    acc8226阅读 535评论 0 2
  • 屏幕适配 屏幕适配的概念 碎片化既是 Android 的优势和弱点,也是开发者们头疼的问题,同时也为 Androi...
    s酸菜阅读 9,743评论 9 58
  • 明月寄回来,机屏画蓝天。 邀请绿水青山,喜笑闹门前。 慈孝辈,微信吹,快递物资温暖。疼爱怜早晚。 人聚金风起,月增...
    万里千年阅读 630评论 34 59