Android传感器完全解析——电子罗盘app实现

2024-02-15 09:50

本文主要是介绍Android传感器完全解析——电子罗盘app实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

什么是传感器

维基百科是这样定义的:传感器是一种物理装置或生物器官,能够探测、感受外界的信号、物理条件(如光、热、湿度)或化学组成(如烟雾),并将探知的信息传递给其他装置或器官。


常用传感器介绍与用法

Android平台支持三个大类的传感器

  • Motion sensors(运动传感器)

这些传感器测量加速力,并沿三个轴的旋转力。此类别包括加速度计,重力感应器, 陀螺仪和旋转矢量传感器。

  • Environmental sensors (环境传感器)

这些传感器测量各种环境参数,例如环境空气温度和压力,照明和湿度。此类别包括气压计,光度计,和温度计。

  • Position sensors (位置传感器)

这些传感器测量设备的物理位置。这个类别包括方向传感器和磁力计。


传感器实现流程

第一步:得到SensorManager

SensorManager mSensorManager = (SensorManager) mContext
.getSystemService(Context.SENSOR_SERVICE);

第二步:注册传感器

Sensor sensor = mSensorManager
.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
if (null != sensor)
mSensorManager.registerListener(this, sensor,
SensorManager.SENSOR_DELAY_NORMAL);

registerListener这个方法有三个参数。

  • 第一个参数是传感器数据变化的监听器

我们需要去实现SensorEventListener接口,他里面有两个回调方法,

@Override
public void onSensorChanged(SensorEvent event) {
//当传感器的数值发生变化时调用
}
@Override
public void onAccuracyChanged(Sensor sensor, int accuracy) {
//传感器的精度发生变化时调用
}

onSensorChanged方法只有一个SensorEvent类型的参数event,其中SensorEvent类有一个values变量非常重要,该变量的类型是float[]。但该变量最多只有3个元素,而且根据传感器的不同,values变量中元素所代表的含义也不同。
关于values值的详细含义请看参考文章!!!

  • 第二个参数是我们需要监听的传感器
Sensor sensor = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);

Sensor.TYPE_ACCELEROMETER则是Android设定传感器类型,这里是指加速度传感器,

  • 第三个参数是传感器数据更新数据的速度

有以下四个值可选,他们的速度是递增的
SENSOR_DELAY_UI
SENSOR_DELAY_NORMAL
SENSOR_DELAY_GAME
SENSOR_DELAY_FASTEST

传感器的注销

//注销所有传感器对象
public voidunregisterListener(SensorEventListener listener)
//注销指定的传感器对象
public voidunregisterListener(SensorEventListener listener, Sensor sensor)

sensor的获取依旧是通过SensorManager.getDefaultSensor()方法。

获得手机支持的所有传感器

Listsensors = sensorManager.getSensorList(Sensor.TYPE_ALL);

Android传感器类型表

  • 加速度传感器:TYPE_ACCELEROMETER

以m/s2测量它设备所有三个物理轴线方向(x,y,和z)加速度。

  • 周围温度传感器:TYPE_AMBIENT_TEMPERATURE

检测周围空气温度。

  • 重力传感器:TYPE_GRAVITY

测量重力

  • 陀螺仪传感器:TYPE_GYROSCOPE

以rad/s测量设备三个物理轴线方向(x,y,和z)。旋转速度。

  • 光照传感器:TYPE_LIGHT

以lx测量周围的光线级别。

  • 线性加速度传感器:TYPE_LINEAR_ACCELERATION

检测沿着一个轴向的加速度。

  • 磁力传感器:TYPE_MAGNETIC_FIELD

测量周围的三个物理轴线方向的磁场。

  • 方向传感器: TYPE_ORIENTATION

测量设备所有三个物理轴线方向(x,y和x)的旋转角度。

  • 压力传感器:TYPE_PRESSURE

测量周围空气气压

  • 接近传感器:TYPE_PROXIMITY

检测物体与手机的距离

  • 相对湿度传感器:TYPE_RELATIVE_HUMIDITY

检测周围空气相对湿度

  • 旋转矢量传感器:TYPE_ROTATION_VECTOR

用于检测运动和检测旋转。

  • 温度传感器: TYPE_TEMPERATURE

检测设备的温度


传感器使用实践

这里以方向传感器为例

方向传感器的获取方式

Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ORIENTATION);

上面这个,对,已经被google弃用了,了解就好。

Android中的坐标系
这里写图片描述
自己画的有点丑,将就着看吧,Z轴默认垂直于地面,所谓获取的三个Values数组即对应手机与Z,Y,X形成的夹角,后面会说明,

前面说了,TYPE_ORIENTATION已被弃用,那么最新的方向传感器是如何做的呢?

事实上,Android 获取手机旋转的方向和角度是通过加速度传感器和地磁传感器共同计算得出的

OK,我们这时候是需要同时使用两个传感器的,看代码

Sensor accelerometerSensor = sensorManager.getDefaultSensor(Sensor.
TYPE_ACCELEROMETER);
Sensor magneticSensor = sensorManager.getDefaultSensor(Sensor.
TYPE_MAGNETIC_FIELD);
sensorManager.registerListener(listener, accelerometerSensor,
SensorManager.SENSOR_DELAY_GAME);
sensorManager.registerListener(listener, magneticSensor,
SensorManager.SENSOR_DELAY_GAME);

同时使用了加速度传感器和地磁传感器

获取旋转矩阵数组R

SensorManager.getRotationMatrix(R, null, accelerometerValues, magneticValues);

获取手机旋转数据

SensorManager.getOrientation(R, values);

values 是一个长度为 3 的 float 数组,手机在各个方向上的旋转数据都会被存放到这个数组当中。

对应关系:
values[0]->Z轴、values[1]->X轴、values[2]->Y轴

values[0]的取值范围是-180到180 度,其中±180 度表示正南方向,0 度表示正北方向,-90 度表示正西方向,90 度表示正东方向,如图
这里写图片描述
所谓,实践是检验真理的唯一标准,这是我检测后自行画的,大家看一下就明白该怎么根据获取到的角度来做对应的处理了

一个完整的方向传感器封装类

 public  class DirectionSensorUtils implements SensorEventListener {
private SensorManager sensorManager;
float[] accelerometerValues = new float[3];
float[] magneticValues = new float[3];
float lastRotateDegree;
private ImageView compassImg;//指南针背景图
public DirectionSensorUtils(Context context , ImageView compassImg) {
sensorManager = (SensorManager) context.getSystemService(Context.SENSOR_SERVICE);
this.compassImg = compassImg;
}
//注册传感器
public void registerSensor(){
Sensor accelerometerSensor = sensorManager.getDefaultSensor(Sensor.
TYPE_ACCELEROMETER);
Sensor magneticSensor = sensorManager.getDefaultSensor(Sensor.
TYPE_MAGNETIC_FIELD);
sensorManager.registerListener(this, accelerometerSensor,
SensorManager.SENSOR_DELAY_GAME);
sensorManager.registerListener(this, magneticSensor,
SensorManager.SENSOR_DELAY_GAME);
}
//解除传感器注册
public void unregisterSensor(){
if (sensorManager != null) {
sensorManager.unregisterListener(this);
}
}
@Override
public void onSensorChanged(SensorEvent event) {
// 判断当前是加速度传感器还是地磁传感器
if (event.sensor.getType() == Sensor.TYPE_ACCELEROMETER) {
// 通过clone()获取不同的values引用
accelerometerValues = event.values.clone();
} else if (event.sensor.getType() == Sensor.TYPE_MAGNETIC_FIELD) {
magneticValues = event.values.clone();
}
//获取地磁与加速度传感器组合的旋转矩阵
float[] R = new float[9];
float[] values = new float[3];
SensorManager.getRotationMatrix(R, null, accelerometerValues,
magneticValues);
SensorManager.getOrientation(R, values);
//values[0]->Z轴、values[1]->X轴、values[2]->Y轴
//使用前请进行转换,因为获取到的值是弧度,示例如下
//        Math.toDegrees(values[0]);
//        Math.toDegrees(values[1]);
//        Math.toDegrees(values[2]);
handleEvent(values);
}
public void handleEvent(float[] values){
// 这里实现了一个指南针
float rotateDegree = -(float) Math.toDegrees(values[0]);
if (Math.abs(rotateDegree - lastRotateDegree) > 1) {
RotateAnimation animation = new RotateAnimation
(lastRotateDegree, rotateDegree, Animation.RELATIVE_TO_SELF, 0.5f, Animation.
RELATIVE_TO_SELF, 0.5f);
animation.setFillAfter(true);
compassImg.startAnimation(animation);
lastRotateDegree = rotateDegree;
}
}
@Override
public void onAccuracyChanged(Sensor sensor, int accuracy) {
}
}

OK,还有其他很多传感器,但使用上大体一致,这里将光照、加速度、方向传感器封住为了一个工具类,地址:https://github.com/walkthehorizon/UtilsSet/blob/master/UtilsLibrary/SensorUtils.java,如果对你有帮助欢迎fork、star以及issue,谢谢

推荐阅读:
Android在方向锁定时获取屏幕旋转角度的另一种方式

本文最新更新日期___2016.09.26

这篇关于Android传感器完全解析——电子罗盘app实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/711058

相关文章

C#高效实现在Word文档中自动化创建图表的可视化方案

《C#高效实现在Word文档中自动化创建图表的可视化方案》本文将深入探讨如何利用C#,结合一款功能强大的第三方库,实现在Word文档中自动化创建图表,为你的数据呈现和报告生成提供一套实用且高效的解决方... 目录Word文档图表自动化:为什么选择C#?从零开始:C#实现Word文档图表的基本步骤深度优化:C

Mybatis的mapper文件中#和$的区别示例解析

《Mybatis的mapper文件中#和$的区别示例解析》MyBatis的mapper文件中,#{}和${}是两种参数占位符,核心差异在于参数解析方式、SQL注入风险、适用场景,以下从底层原理、使用场... 目录MyBATis 中 mapper 文件里 #{} 与 ${} 的核心区别一、核心区别对比表二、底

nginx跨域访问配置的几种方法实现

《nginx跨域访问配置的几种方法实现》本文详细介绍了Nginx跨域配置方法,包括基本配置、只允许指定域名、携带Cookie的跨域、动态设置允许的Origin、支持不同路径的跨域控制、静态资源跨域以及... 目录一、基本跨域配置二、只允许指定域名跨域三、完整示例四、配置后重载 nginx五、注意事项六、支持

Qt实现对Word网页的读取功能

《Qt实现对Word网页的读取功能》文章介绍了几种在Qt中实现Word文档(.docx/.doc)读写功能的方法,包括基于QAxObject的COM接口调用、DOCX模板替换及跨平台解决方案,重点讨论... 目录1. 核心实现方式2. 基于QAxObject的COM接口调用(Windows专用)2.1 环境

MySQL查看表的历史SQL的几种实现方法

《MySQL查看表的历史SQL的几种实现方法》:本文主要介绍多种查看MySQL表历史SQL的方法,包括通用查询日志、慢查询日志、performance_schema、binlog、第三方工具等,并... 目录mysql 查看某张表的历史SQL1.查看MySQL通用查询日志(需提前开启)2.查看慢查询日志3.

Java实现字符串大小写转换的常用方法

《Java实现字符串大小写转换的常用方法》在Java中,字符串大小写转换是文本处理的核心操作之一,Java提供了多种灵活的方式来实现大小写转换,适用于不同场景和需求,本文将全面解析大小写转换的各种方法... 目录前言核心转换方法1.String类的基础方法2. 考虑区域设置的转换3. 字符级别的转换高级转换

使用Python实现局域网远程监控电脑屏幕的方法

《使用Python实现局域网远程监控电脑屏幕的方法》文章介绍了两种使用Python在局域网内实现远程监控电脑屏幕的方法,方法一使用mss和socket,方法二使用PyAutoGUI和Flask,每种方... 目录方法一:使用mss和socket实现屏幕共享服务端(被监控端)客户端(监控端)方法二:使用PyA

Agent开发核心技术解析以及现代Agent架构设计

《Agent开发核心技术解析以及现代Agent架构设计》在人工智能领域,Agent并非一个全新的概念,但在大模型时代,它被赋予了全新的生命力,简单来说,Agent是一个能够自主感知环境、理解任务、制定... 目录一、回归本源:到底什么是Agent?二、核心链路拆解:Agent的"大脑"与"四肢"1. 规划模

MyBatis-Plus逻辑删除实现过程

《MyBatis-Plus逻辑删除实现过程》本文介绍了MyBatis-Plus如何实现逻辑删除功能,包括自动填充字段、配置与实现步骤、常见应用场景,并展示了如何使用remove方法进行逻辑删除,逻辑删... 目录1. 逻辑删除的必要性编程1.1 逻辑删除的定义1.2 逻辑删php除的优点1.3 适用场景2.

C#借助Spire.XLS for .NET实现在Excel中添加文档属性

《C#借助Spire.XLSfor.NET实现在Excel中添加文档属性》在日常的数据处理和项目管理中,Excel文档扮演着举足轻重的角色,本文将深入探讨如何在C#中借助强大的第三方库Spire.... 目录为什么需要程序化添加Excel文档属性使用Spire.XLS for .NET库实现文档属性管理Sp