在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两个数组必须以成员变量的形式进行声明,而不能是局部变量。