添加链接
link管理
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接
相关文章推荐
纯真的熊猫  ·  什么时候 MySQL ...·  1 年前    · 


在最近的项目中,需要获取手机当前Cpu温度,网上也有很多文章介绍,这里记录一下。

获取CPU温度的方式:

在Android系统中,获取CPU温度的方法并没有直接提供给我们开发者,我们可以通过两种方式来获取Cpu温度:

  • 1、 通过读取手机传感器 sensor 的温度近似于手机CPU温度 (当然这种方式只是一个近似的值,并不准确,同时还需要手机具备相应的传感器)
  • 2、 通过读取CPU信息来获取 (这种方式相较于前一种方式获取到的数据准确很多,但是还是有一定的局限性。)

通过两种方式获取温度:

传感器(Sensor):

众所周知,我们的Android手机有很多的传感器(陀螺仪,距离,温度传感等),如果我们可以判断手机上有我们需要的温度传感器,我们就可以通过温度传感器获取到我们手机的大致温度。而要想知道我们手机上是否具备我们所需的传感器,那我们就需要去遍历一下我们手机设备的传感器了。

  • 使用SensorManager,我们可以遍历出设备所有的传感器,名字,类型等。
SensorManager mSmanager = (SensorManager) getSystemService(Context.SENSOR_SERVICE);
List<Sensor> allSensors = sm.getSensorList(Sensor.TYPE_ALL);
SensorManager mSmanager = (SensorManager) getSystemService(Context.SENSOR_SERVICE);
List<Sensor> allSensors = sm.getSensorList(Sensor.TYPE_ALL);

通过 getSensorList 方法我们就可以获取到我们所有的传感器。 获取到所有传感器列表之后,我们怎样判断传感器的类型呢?

for (Sensor s : allSensors) {
        if (Build.VERSION.SDK_INT > Build.VERSION_CODES.KITKAT) {
            typeName = SensorTypeName.getSensorTypeName(s.getType()) + " " + s.getStringType();
            if (s.getStringType().toUpperCase().indexOf("TEMP") > 0) {
                // 可以看到,这里将包含有TEMP关键字的sensor付给了变量mTempSensor 
                // 而这个mTempSensor 就是我们需要的温度传感器
                mTempSensor = s;
        } else {
            typeName = SensorTypeName.getSensorTypeName(s.getType()) + " " + s.getType();
        sb.append(String.format("\t类型:%s\n", typeName));
        sb.append(String.format("\t设备名称:%s\n", s.getName()));
        sb.append(String.format("\t设备版本:%s\n", s.getVersion()));
        sb.append(String.format("\t供应商:%s\n", s.getVendor()));
        sb.append("\n");
    // 这里我们将所有的传感器都放在一个subStr中,方便查看我们的结果
    tx1.setText(sb.toString());
    // 如果传感器不为空,那么我们就可添加一个监听,获取传感器的温度情况
    if (mTempSensor != null) {
        sm.registerListener(mSensorEventListener, mTempSensor
                , SensorManager.SENSOR_DELAY_GAME);
    // 温度传感器的监听器
    private final SensorEventListener mSensorEventListener = new SensorEventListener() {
      @Override
      public void onSensorChanged(SensorEvent event) {
          if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH) {
              if (event.sensor.getStringType().toUpperCase().indexOf("TEMP") > 0) {
                  /*温度传感器返回当前的温度,单位是摄氏度(°C)。*/
                  float temperature = event.values[0];
                  Log.e("temperature: ", temperature);
                  sm.unregisterListener(mSensorEventListener, mTempSensor);
      @Override
      public void onAccuracyChanged(Sensor sensor, int accuracy) {
    for (Sensor s : allSensors) {
        if (Build.VERSION.SDK_INT > Build.VERSION_CODES.KITKAT) {
            typeName = SensorTypeName.getSensorTypeName(s.getType()) + " " + s.getStringType();
            if (s.getStringType().toUpperCase().indexOf("TEMP") > 0) {
                // 可以看到,这里将包含有TEMP关键字的sensor付给了变量mTempSensor 
                // 而这个mTempSensor 就是我们需要的温度传感器
                mTempSensor = s;
        } else {
            typeName = SensorTypeName.getSensorTypeName(s.getType()) + " " + s.getType();
        sb.append(String.format("\t类型:%s\n", typeName));
        sb.append(String.format("\t设备名称:%s\n", s.getName()));
        sb.append(String.format("\t设备版本:%s\n", s.getVersion()));
        sb.append(String.format("\t供应商:%s\n", s.getVendor()));
        sb.append("\n");
    // 这里我们将所有的传感器都放在一个subStr中,方便查看我们的结果
    tx1.setText(sb.toString());
    // 如果传感器不为空,那么我们就可添加一个监听,获取传感器的温度情况
    if (mTempSensor != null) {
        sm.registerListener(mSensorEventListener, mTempSensor
                , SensorManager.SENSOR_DELAY_GAME);
    // 温度传感器的监听器
    private final SensorEventListener mSensorEventListener = new SensorEventListener() {
      @Override
      public void onSensorChanged(SensorEvent event) {
          if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH) {
              if (event.sensor.getStringType().toUpperCase().indexOf("TEMP") > 0) {
                  /*温度传感器返回当前的温度,单位是摄氏度(°C)。*/
                  float temperature = event.values[0];
                  Log.e("temperature: ", temperature);
                  sm.unregisterListener(mSensorEventListener, mTempSensor);
      @Override
      public void onAccuracyChanged(Sensor sensor, int accuracy) {
  };

到此,就是通过传感器获取手机大致的温度。
但是在实际的开发调试过程中,很多手机是没有温度传感器的。那此时,就只有通过直接读取Cpu信息来获取手机的CPU温度了。

读取CPU内核信息获取温度:

读取手机CPU信息获取温度,我们需要获取的是手机CPU内核(thermal)的信息,而这个信息是存储在手机的 /sys 节点目录中的。有开发经验的朋友就知道,读取手机缓存中的文件是需要有权限的。而这个权限是需要 su 权限,也就是root权限。那换而言之,一般的手机是读取不到这个节点信息的。这里也是一个坑,需要注意。

读取路径: /sys/class/thermal/thermal_zone* 这里的 * 对应了手机的内核文件夹编号,例如某些手机的 * 可能有17个(0~16),但是需要注意的是 并不是所有的文件夹都是存储CPU内核的信息

那怎么判断哪个文件夹目录使我们需要获取的信息目录呢?

这里介绍一个方式:
我们都知道,现在手机的处理器要不是高通(Qualcomm),要不是联发科(MTK),当然还有其他的类型,但是市场主流是上边的两个。而这两个CPU内核都有固定的名字。
我们可以通过 cat 命令读取 /sys/class/thermal/thermal_zone*/type 获取到的字符串来判断是哪种类型的CPU。也可以判断该目录是否是CPU内核信息。

CPU厂商

对应的内核前缀

MTK

mtktscpu

Qualcomm

tsens_tz_sensor

Exynos(三星)

exynos

不要问我怎么来的,多使用ADB真机调试,使用以上的命令你也可以总结出来这些东西。



ADB SHELL THERMAL.png


通过for循环,遍历 thermal_zone cat type 出来的信息判断是否包含了以上两种CPU的关键字,则可以判断该目录是否保存了CPU内核信息。
只要判断了那些目录是属于内核信息的,获取温度就手到拈来。只需要 cat 另一个参数 temp 那么输出的信息就是我们需要的温度啦。

奏是这么简单。



image.png


还有一点需要注意的是,某些手机返回回来的参数(很多情况下是thermal_zone0)是精确到小数点后3位,但是没有小数点,例如45300,其实是45.3℃。这里需要转换。

Android应用使用ADB命令,这个就不用再提了吧,使用 FileInputStream InputStreamReader BufferedReader 来读取应该大家都会。如果不会百度下也是很快就能搞定的。

对于没有权限的手机:

现在市面上有很多安全清理软件都是有CPU温度监控的模块,而他们获取的温度,可能是使用batteryTemp代替了CPU真正的温度。而获取电池温度的方式,只需要使用电池广播就行了。

最后代码就不贴了,原理就是这样。希望能帮助到有需要的朋友。老是复制粘贴哪有自己码代码有意思是吧。



作者:芥末末的沫


获取cpu温度

要获取cpu的温度可以通过汇编指令来读取,这里以intel cpu为例,用rdmsr指令读取IA32_THERM_STATUS_MSR(0x019C)的值,然后用TjunctionMax 减去这个值就是当前cpu的温度,对于一般的intel cpu 的TjunctionMax值是固定的,比如我的cpu是Intel Core 2 (45nm),在官网上可以查到其值是100摄氏度,到后面的intel cpu专门有个寄存器IA32_TEMPERATURE_TARGET(0x01A2)保存TjunctionMax的值,可以通过rdmsr指令读取。但是rdmsr指令只能在Ring0层运行,在运用层是执行 Read More

java虚拟机 java虚拟机教程

前言在本文将深入讨论 Java 虚拟机相关核心知识参考书籍:《深入理解 Java 虚拟机》周志明,学习课程:【炼数成金】深入 JVM 内核—原理、诊断与优化【龙果学院】深入理解 Java 虚拟机( JVM 性能调优+内存模型+虚拟机原理)【尚学堂】白鹤翔 JVM 虚拟机优化核心知识JVM 基本结构 类加载器执行引擎运行时数据区本地接口Class Files -> ClassLo