在Android中传感器的使用场景比其他传感器更为广泛,因为它能够准确地判断出手机在各个方向的旋转角度,利用这些角度可以编写出指南针、地平仪等有用的工具。
方向传感器的使用相较于其他的传感器有所不同。虽然在API中有TYPE_ORIENTATION常量,可以像得到加速度传感器那样得到方向传感器Sensor.TYPE_ORIENTATION,但是这样做的话,在新版的SDK中就会提示"这种方式已经过期,不建议使用!"官方推荐我们在应用程序中使用磁场域和加速度传感器结合SensorManager.setOrientation()来获得原始数据。下面通过实例来说明如何使用方向传感器。
**MainActivity.java**
package com.sdutacm.sensor;
import android.content.Context;
import android.hardware.Sensor;
import android.hardware.SensorEvent;
import android.hardware.SensorEventListener;
import android.hardware.SensorManager;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.widget.Toast;
public class MainActivity extends AppCompatActivity implements SensorEventListener{
private SensorManager sensorManager;
private Sensor accelerometerSensor;
private Sensor magneticFieldSensor;
private float[] accelerometerValues = new float[3];
private float[] magneticValues = new float[3];
//旋转矩阵,用来保存磁场和加速度的数据
private float[] r = new float[9];
//模拟方向传感器的数据(原始数据为弧度)
private float[] values = new float[3];
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
getSensorManager();
}
@Override
public void onSensorChanged(SensorEvent event) {
if(event.sensor.getType() == Sensor.TYPE_ACCELEROMETER){
//这里是对象,需要克隆一份,否则共用一份数据
accelerometerValues = event.values.clone();
}else if(event.sensor.getType() == Sensor.TYPE_MAGNETIC_FIELD){
//这里是对象,需要克隆一份,否则共用一份数据
magneticValues = event.values.clone();
}
/**
* 填充旋转数组r
* r:要填充的旋转数组
* I:将磁场数据转换成实际的重力坐标中,一般可以设置为null
* gravity:加速度传感器数据
* geomagnetic:地磁传感器数据
*/
SensorManager.getRotationMatrix(r,null,accelerometerValues,magneticValues);
/**
* R:旋转数组
* values :模拟方向传感器的数据
*/
sensorManager.getOrientation(r,values);
float degree = (float) Math.toDegrees(values[0]);
Log.d("指南针","当前手机角度为:"+degree);
Toast.makeText(this,"当前手机角度为:"+degree,Toast.LENGTH_SHORT).show();
//Sensor 发生变化时,在次通过event.values获取数据
// float x = event.values[0];
// float y = event.values[0];
// float z = event.values[0];
// float light = event.values[0];
// Log.d("Light","当前的光线强度为:"+light+"勒克斯");
// Toast.makeText(this,"当前的光线强度为:"+light+"勒克斯",Toast.LENGTH_SHORT).show();
// if(x>20 || y>20 || z>20){
// Toast.makeText(this,"欢迎使用摇一摇",Toast.LENGTH_SHORT).show();
// }
}
@Override
public void onAccuracyChanged(Sensor sensor, int accuracy) {
//注册的Sensor精度发生变化时,在此处处理
}
public void getSensorManager() {
sensorManager = (SensorManager) getSystemService(Context.SENSOR_SERVICE);
/**
* 传入的参数决定传感器的类型
* Senor.TYPE_ACCELEROMETER: 加速度传感器
* Senor.TYPE_LIGHT:光照传感器
* Senor.TYPE_GRAVITY:重力传感器
* SenorManager.getOrientation(); //方向传感器
*/
accelerometerSensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
magneticFieldSensor = sensorManager.getDefaultSensor(Sensor.TYPE_MAGNETIC_FIELD);
}
@Override
protected void onResume() {
super.onResume();
if(sensorManager != null){
//一般在Resume方法中注册
/**
* 第三个参数决定传感器信息更新速度
* SensorManager.SENSOR_DELAY_NORMAL:一般
* SENSOR_DELAY_FASTEST:最快
* SENSOR_DELAY_GAME:比较快,适合游戏
* SENSOR_DELAY_UI:慢
*/
sensorManager.registerListener(this,accelerometerSensor,SensorManager.SENSOR_DELAY_NORMAL);
sensorManager.registerListener(this,magneticFieldSensor,SensorManager.SENSOR_DELAY_NORMAL);
}
}
@Override
protected void onPause() {
super.onPause();
if(sensorManager != null){
//解除注册
sensorManager.unregisterListener(this,accelerometerSensor);
sensorManager.unregisterListener(this,magneticFieldSensor);
}
}
}
简单的说,想要获取方向信息,需要使用加速度传感器和磁场域,然后分别获取他们的结果,然后使用SensorManager.getRotationMatrix(r,null,accelerometerValues,magneticValues)结合给r赋值再使用SensorManager.getOrientation(r,values)从r数组中解析出方向信息,并赋值给values数组。values数组中的第一个值就代表手机当前的角度。
需要注意的是,加速度传感器和磁场域赋值的时候一定要调用values数组的clone()方法,不然,accelerometerValues和magneticValues会指向同一个引用.另外,accelerometerValues和magneticValues两个数组必须以成员变量的形式进行声明,而不能是局部变量。