java 获取屏幕对角线尺寸,获得屏幕物理尺寸、密度及分辨率
目录(?)[+]
大家帮忙喽!
博主参加2014博客之星活动,大家帮忙投票啦!猛击这里!
通过程序去了解硬件情况是一件十分有意思的事情。很早我就研究在 WM6.5上获得屏幕物理尺寸,但一直没有成功。后来又想要在 Android 上有所突破,不过在今天之前得到的尺寸都不准确。虽然很多人认为没必要这么较真,因为貌似很多情况下用不到。不过我就当这是一件很有挑战性的事,一定要做到。对,就是这么任性。
源码中 android.view 包下的 Display 类提供了很多方法供程序员获得显示相关的信息,通过此类让我们开启了解设备屏幕之旅吧。
一、分辨率
需要注意的原来经常使用的 getHeight() 与 getWidth() 已经不推荐使用了,建议使用 getSize()来替代。
此方法原型如下:
public void getSize(Point outSize) {
synchronized (this) {
updateDisplayInfoLocked();
mDisplayInfo.getAppMetrics(mTempMetrics, mDisplayAdjustments);
outSize.x = mTempMetrics.widthPixels;
outSize.y = mTempMetrics.heightPixels;
}
}
参数是一个返回参数,用以返回分辨率的 Point,这个 Point 也比较简单,我们只需要关注 x 和 y 这两个成员就可以了。
用法如下:
private void getDisplayInfomation() {
Point point = new Point();
getWindowManager().getDefaultDisplay().getSize (point);
Log.d(TAG,"the screen size is "+point.toString());
}
结果如下:
D/MainActivity﹕ the screen size is Point(800, 1280)
此外 Display 又提供了一个 getRealSize 方法,原型如下:
public void getRealSize(Point outSize) {
synchronized (this) {
updateDisplayInfoLocked();
outSize.x = mDisplayInfo.logicalWidth;
outSize.y = mDisplayInfo.logicalHeight;
}
}
从两个方法的实现上看是有区别的,但是在通常情况下二者的返回值相同。那么差异究竟在哪里,下面做一些实验来验证一下。
首先,我将 Acitvity 设置不同的 theme,比如:
android:theme="@android:style/Theme.Black.NoTitleBar.Fullscreen"
android:theme="@android:style/Theme.NoTitleBar.Fullscreen"
结果还是相同的。
接下来将我的 Activity 父类变成 ActionBarActivity,如下: public class MainActivity extends ActionBarActivity 期望 ActionBar 会占用一些屏幕,并在程序中动态设置 Listview的Item 中的图片大小。在机缘巧合之下,结果验证了在这种情况下,getSize 返回的结果变了。
代码如下:
private void getDisplayInfomation() {
Point point = new Point();
getWindowManager().getDefaultDisplay().getSize(point);
Log.d(TAG,"the screen size is "+point.toString());
getWindowManager().getDefaultDisplay().getRealSize(point);
Log.d(TAG,"the screen real size is "+point.toString());
}
Log 如下:
D/MainActivity﹕ the screen size is Point(800, 1202)
D/MainActivity﹕ the screen real size is Point(800, 1280)
如果你不能够轻易复现也不用急,保险起见,为了得到相对正确的信息还是使用 getRealSize() 吧。
二、屏幕尺寸
设备的物理屏幕尺寸。与几年前不同,目前的手机屏幕已经大到一只手握不下了。标配早已经到了5寸屏时代。
所谓屏幕尺寸指的是屏幕对角线的长度,单位是英寸。
然而不同的屏幕尺寸是可以采用相同的分辨率的,而它们之间的区别在与密度(density)不同。
下面先介绍一下密度的概念,DPI、PPI,最后讲解一下如何根据获得的 Display 信息去求出屏幕尺寸。这是一个困扰我很久的问题了。
三、屏幕密度
屏幕密度与 DPI 这个概念紧密相连,DPI 全拼是 dots-per-inch,即每英寸的点数。也就是说,密度越大,每英寸内容纳的点数就越多。 android.util 包下有个 DisplayMetrics 类可以获得密度相关的信息。
最重要的是 densityDpi 这个成员,它有如下几个常用值:
DENSITY_LOW = 120
DENSITY_MEDIUM = 160 //默认值
DENSITY_TV = 213 //TV专用
DENSITY_HIGH = 240
DENSITY_XHIGH = 320
DENSITY_400 = 400
DENSITY_XXHIGH = 480
DENSITY_XXXHIGH = 640
举例如下:
private void getDensity() {
DisplayMetrics displayMetrics = getResources().getDisplayMetrics();
Log.d(TAG,"Density is "+displayMetrics.density+" densityDpi is "+displayMetrics.densityDpi+" height: "+displayMetrics.heightPixels+
" width: "+displayMetrics.widthPixels);
}
Log 如下:
the screen size is Point(1600, 2438)
the screen real size is Point(1600, 2560)
Density is 2.0 densityDpi is 320 height: 2438 width: 1600
有了这些信息,我们是不是就可以计算屏幕尺寸了呢?
首先求得对角线长,单位为像素。
然后用其除以密度(densityDpi)就得出对角线的长度了。
代码如下:
private void getScreenSizeOfDevice() {
DisplayMetrics dm = getResources().getDisplayMetrics();
int width=dm.widthPixels;
int height=dm.heightPixels;
double x = Math.pow(width,2);
double y = Math.pow(height,2);
double diagonal = Math.sqrt(x+y);
int dens=dm.densityDpi;
double screenInches = diagonal/(double)dens;
Log.d(TAG,"The screenInches "+screenInches);
}
Log 如下:
01-13 16:35:03.026 16601-16601/com.linc.listviewanimation D/MainActivity﹕ the screen size is Point(1600, 2438)
01-13 16:35:03.026 16601-16601/com.linc.listviewanimation D/MainActivity﹕ the screen real size is Point(1600, 2560)
01-13 16:35:03.026 16601-16601/com.linc.listviewanimation D/MainActivity﹕ Density is 2.0 densityDpi is 320 height: 2438 width: 1600 xdpi 338.666 ydpi 338.666
01-13 16:35:03.026 16601-16601/com.linc.listviewanimation D/MainActivity﹕ The screenInches 9.112922229586951
如 Log 所见,使用 heightPixels 得出的值是2483而不是正确的2560.从而使结果9.11反倒跟真实屏幕尺寸很接近。下面用正确的 height 再算一遍。
01-13 16:39:05.476 17249-17249/com.linc.listviewanimation D/MainActivity﹕ the screen size is Point(1600, 2560)
01-13 16:39:05.476 17249-17249/com.linc.listviewanimation D/MainActivity﹕ the screen real size is Point(1600, 2560)
01-13 16:39:05.476 17249-17249/com.linc.listviewanimation D/MainActivity﹕ Density is 2.0 densityDpi is 320 height: 2560 width: 1600 xdpi 338.666 ydpi 338.666
01-13 16:39:05.476 17249-17249/com.linc.listviewanimation D/MainActivity﹕ The screenInches 9.433981132056605
结果是9.43英寸,而真实值是8.91.如果再换一个设备,那么值差的更多。说明上面的计算是错误的。
那么错在哪里呢?densityDpi 是每英寸的点数(dots-per-inch)是打印机常用单位(因而也被称为打印分辨率),而不是每英寸的像素数。下面引出 PPI 这个概念。
四、PPI
Pixels per inch,这才是我要的每英寸的像素数(也被称为图像的采样率)。有了这个值,那么根据上面的公式就可以求导出屏幕的物理尺寸了。 还好 DisplayMetrics 有两个成员是 xdpi 和 ydpi,对其描述是:
/The exact physical pixels per inch of the screen in the X/Y dimension.
屏幕 X/Y 轴上真正的物理 PPI。
Yes!Got it!
为了保证获得正确的分辨率,我还是使用 getRealSize 去获得屏幕宽和高像素。所以,经过修改,代码如下:
private void getScreenSizeOfDevice2() {
Point point = new Point();
getWindowManager().getDefaultDisplay().getRealSize(point);
DisplayMetrics dm = getResources().getDisplayMetrics();
double x = Math.pow(point.x/ dm.xdpi, 2);
double y = Math.pow(point.y / dm.ydpi, 2);
double screenInches = Math.sqrt(x + y);
Log.d(TAG, "Screen inches : " + screenInches);
}
Log is as follows:
[plain] view plaincopy在CODE上查看代码片派生到我的代码片
01-13 16:58:50.142 17249-17249/com.linc.listviewanimation D/MainActivity﹕ Screen inches : 8.914015757534717
五、DIP
注意不要与上面的 DPI 混淆,这个 DIP 是 Density Independent Pixel,直译为密度无关的像素。
我们在布局文件中使用的 dp/dip 就是它。官方推荐使用 dp 是因为它会根据你设备的密度算出对应的像素。
公式为:pixel = dip*density
需要注意的是,我们在 Java 代码中对控件设置宽高是不可以设置单位的,而其自带的单位是像素。所以如果动态修改控件大小时,我们的任务就来了,那就是将像素转换为 dp。
实例代码如下:
//pixel = dip*density;
private int convertDpToPixel(int dp) {
DisplayMetrics displayMetrics = mContext.getResources().getDisplayMetrics();
return (int)(dp*displayMetrics.density);
}
private int convertPixelToDp(int pixel) {
DisplayMetrics displayMetrics = mContext.getResources().getDisplayMetrics();
return (int)(pixel/displayMetrics.density);
}
参考:
java 获取屏幕对角线尺寸,获得屏幕物理尺寸、密度及分辨率相关推荐
- Android平台根据分辨率计算屏幕尺寸,基于物理尺寸来验证手机和平板应用合并的可行性
Android平台根据分辨率计算屏幕尺寸,基于物理尺寸来验证手机和平板应用合并的可行性 出发点:Android系统本身是支持多种分辨率的,如图1所示,一般情况下要适配不同尺寸的设备只需要做两件事, ...
- android 获取屏幕的物理尺寸
计算方法:获取到屏幕的分辨率:point.x和point.y,再取出屏幕的DPI(每英寸的像素数量),计算长和宽有多少英寸,即:point.x / dm.xdpi,point.y / dm.ydpi, ...
- android获取屏幕的物理尺寸
/** * @ 获取当前手机屏幕尺寸 */ public static float getPingMuSize(Context mContext) {int densityDpi = mContext ...
- Android实战技巧之十:获得屏幕物理尺寸、密度及分辨率
大家帮忙喽! 博主参加2014博客之星活动,大家帮忙投票啦!猛击这里! 通过程序去了解硬件情况是一件十分有意思的事情.很早我就研究在WM6.5上获得屏幕物理尺寸,但一直没有成功.后来又想要在Andro ...
- Android:获得屏幕物理尺寸、密度及分辨率
一.分辨率 需要注意的原来经常使用的getHeight()与getWidth()已经不推荐使用了,建议使用getSize()来替代. 此方法原型如下: [java] view plain copy ...
- VC++获取屏幕大小第三篇 物理大小GetDeviceCaps 下
本篇文章(MoreWindows)将获取屏幕的物理大小后计算屏幕对角线长度并换算成英寸.这样可以方便大家查看自己电脑屏幕是多少英寸的.为了更方便大家使用,本文将提供电脑屏幕大小查看器下载(配源码)下载 ...
- java获取屏幕上某坐标点的颜色
全栈工程师开发手册 (作者:栾鹏) java教程全解 java获取屏幕上某坐标点的颜色.通过先获取屏幕截图,再获取坐标点颜色. public Color getScreenPixel(int x, i ...
- Java获取屏幕真实分辨率
这个问题大家可能不陌生,一搜索,就一堆答案,大多是一行代码搞定 Toolkit.getDefaultToolkit().getScreenSize(); 然而简单做了一个测试,效果如下,输出1707x ...
- JavaScript获取浏览器、元素、屏幕的宽高尺寸
版权声明 转载请告知并注明来源作者 作者:唐金健 网络昵称:御焱 掘金知乎思否专栏:优雅的前端 前言 有时候在获取浏览器.元素.屏幕的尺寸,傻傻分不清.为了让自己清晰认识,能够快速确定自己需要哪个属性 ...
最新文章
- (转)Linux系统调用和库函数调用的区别
- python单词翻译-完成自动查找翻译单词的python源代码
- jQuery1.9+中删除了live以后的替代方法
- ironpython使用opencv_连这些都不知道,就别说自己懂Python!
- Android Activity的生命周期、意图(Intent)
- 鲲鹏云实验-.NET Core 3.0-开始使用
- vmware磁盘扩容_CentOS7扩容分区(LVM)
- 光伏机器人最前线_送水、送药、送餐!哈市这些地方率先用上AI配送机器人(视频)...
- 面对tomatserver使用的webrequest
- CSS样式网页导航条
- python三国演义人物出场统计txt文本_python实例:三国演义TXT文本词频分析
- 世界观和方法论----写给学弟学妹的
- [转]MOSS 2007 整合運用 - Office System 2007 整合應用秘訣
- RBP中新增自由格式报表资源之自由格式风格
- 入侵必读:网警如何找到你
- JavaScript高级第03天笔记
- 华为禁止系统更新的方法
- 蒙特卡罗奇异值谱分析
- 在线表单设计器都有哪些优秀的功能?
- 青岛旅行规划及玩后感
热门文章
- java 正则表达式 单引号_关于Java:用Regex替换单引号和双引号
- matlab如何求极点,matlab求极点和零点
- datav轮播表updateRows方法使用
- java 热量图实现_安卓实现计步器,能够计算卡路里的消耗
- 聊聊如何学习开源项目
- 刷脸支付其大有取扫码支付而代之的架势
- 数据仓库工具之hive(1)
- mysql之explain详解(分析索引最佳使用)
- 安卓手机纯H5视频上传非常慢的问题解决
- springboot maven打包jar运行,springboot mybatis druid,demo源码