简介
坐标系 x轴:从左到右 y轴:从下到上 z轴:从内到外 这个坐标系与Android 2D API中的不同,传感器中的返回值都以此坐标系为准。
SENSOR_TYPE_ACCELEROMETER 1 //加速度 SENSOR_TYPE_MAGNETIC_FIELD 2 //磁力 SENSOR_TYPE_ORIENTATION 3 //方向 SENSOR_TYPE_GYROSCOPE 4 //陀螺仪 SENSOR_TYPE_LIGHT 5 //光线感应 SENSOR_TYPE_PRESSURE 6 //压力 SENSOR_TYPE_TEMPERATURE 7 //温度 SENSOR_TYPE_PROXIMITY 8 //接近 SENSOR_TYPE_GRAVITY 9 //重力 SENSOR_TYPE_LINEAR_ACCELERATION 10//线性加速度 SENSOR_TYPE_ROTATION_VECTOR 11//旋转矢量
API概况 sensor相关API被放到了android.hardware包下,主要使用的类有Sensor、SensorEvent、SensorManager及SensorEventListener接口。 SensorManager顺其自然的担任起管理的工作,负责注册监听某Sensor的状态;Sensor的数据通过SensorEvent返回。
- Sensor: 表示传感器的类,它保存有传感器名称,厂商,版本,精确度等信息
- SensorEvent:表示传感器事件,它可以保存传感器的值,传感器类型,时间戳等信息
- SensorEventListener:用于接收传感器来自SensorManager的通知,当传感器发生变化时,它包含两个回调函数
- SensorManager:SensorManager让你可以访问手机的全部传感器
- SensorListener:已废除
延迟时间的精密度参数如下: SensorManager.SENSOR_DELAY_FASTEST 0ms SensorManager.SENSOR_DELAY_GAME 20ms SensorManager.SENSOR_DELAY_UI 60ms SensorManager.SENSOR_DELAY_NORMAL 200ms 因为感应检测Sensor的服务是否频繁和快慢都与电池参量的消耗有关,同时也会影响处理的效率,所以兼顾到消耗电池和处理效率的平衡,需要根据应用系统的需求来做适当的设置。
加速度
加速度传感器的背景 这里的加速度特指重力加速度,所以在【静止时】重力传感器的返回值与加速度传感器值相同。 地表上静止物体的重力加速度约为9.8 m/s^2. 借用SensorManager中的常量: public static final float STANDARD_GRAVITY = 9.80665F; 我们可以借助三轴上的值来确定设备的状态,比如:
- 将手机平放在桌面上,x轴默认为0,y轴默认0,z轴默认9.81。
- 将手机朝下放在桌面上,z轴为-9.81。
- 将手机向左倾斜,x轴为正值;当x轴的值接近重力加速度时,说明设备的左边朝下。
- 将手机向右倾斜,x轴为负值;当x轴的值接近负的g值时,说明设备的右边朝下。
- 将手机向上倾斜,y轴为负值;当y轴的值接近负的g值时,说明设备的上边朝下。
- 将手机向下倾斜,y轴为正值;当y轴的值接近g值时,说明设备的下边超下。
地磁
磁场传感器主要读取的是磁场的变化,通过该传感器便可开发出指南针、罗盘等磁场应用。 该传感器读取的数据同样是空间坐标系三个方向的磁场值,其数据单位为?T。 磁场传感器可以用来检测磁场大小,和加速度传感器一样,有x、y、z轴三个方向,单位为uT(microteslas),即微特斯拉。 磁场传感器也称为compass(指南针),在uses-feature中使用Android.hardware.sensor.compass作为其名字。 可以拿着手机到处测测,在电器附近不同位置,值还是相差巨大的。 不过单看磁场数值其实也看不出所以然。
方向
安卓平台提供了2个传感器用于让我们判断设备的位置,分别是【地磁场传感器】和【方向传感器】。关于Orientation Sensor在官方文档中的概述里有这样一句话: The orientation sensor is software-based and derives its data from the accelerometer and the geomagnetic field sensor. 方向传感器是基于软件的,并且它的数据是通过【加速度传感器】和【磁场传感器】共同获得的。
- 第一个元素azimuth,【z轴旋转角度】,手机由水平正北放置时开始顺时针旋转,z的值变化情况为0~360/0;表示指向地心的【方位角】
- 第二个元素pitch,【x轴旋转角度】,手机由水平正北放置时开始顺时针旋转,x的值变化情况为0~-180/180~0;表示前后旋转的【仰俯角】
- 第三个元素roll,【y轴旋转角度】,手机由水平正北放置时开始顺时针旋转,y的值变化情况为0~90~0~-90~0;表示左右旋转的【翻转角】
一定要清楚,上面的值都是【旋转】角度,上面的总结是没有错的,如果你觉得错了,那就是没有理解【旋转】的意思。 当手机顶部指向正北方时,方向值为0;顶部指向正东方时,方向值为90;顶部指向正南方时,方向值为180;顶部指向正西方时,方向值为270。
磁场+加速度代替方向传感器
在最新版的SDK中,使用Orientation传感器会看到这么一句话“This constant is deprecated. use SensorManager.getOrientation() instead. ” 即这种方式已过期,不建议使用!Google建议我们在应用程序中使用SensorManager.getOrientation()来获得原始数据。 public static float[] getOrientation (float[] R, float[] values)
- 第一个参数是R[] 是一个旋转矩阵,用来保存磁场和加速度的数据,可以理解为这个函数的传入值,通过它这个函数给你求出方位角。
- 第二个参数就是这个函数的输出了,他有函数自动为我们填充,这就是我们想要的。
- values[0] :方向角,但用(磁场+加速度)得到的数据范围是(-180~180),也就是说,0表示正北,90表示正东,180/-180表示正南,-90表示正西。而直接通过方向感应器数据范围是(0~359)360/0表示正北,90表示正东,180表示正南,270表示正西。
- values[1] pitch 倾斜角,即由静止状态开始,前后翻转,手机顶部往上抬起(0~-90),手机尾部往上抬起(0~90)
- values[2] roll 旋转角,即由静止状态开始,左右翻转,手机左侧抬起(0~90),手机右侧抬起(0~-90)
- 第一个就是我们需要填充的R数组,大小是9
- 第二个是一个转换矩阵,将磁场数据转换进实际的重力坐标中,一般默认情况下可以设置为null
- 第三个是一个大小为3的数组,表示从加速度感应器获取来的数据,在onSensorChanged中
- 第四个是一个大小为3的数组,表示从磁场感应器获取来的数据,在onSensorChanged中
加速度示例
public class AccelerometerActivity extends ListActivity implements SensorEventListener { private TextView tv_info; private SensorManager sm;//传感器管理器 private Vibrator vibrator;//震动 private long lastTime = System.currentTimeMillis(); private static final int UPTATE_INTERVAL_TIME = 3500;// 两次检测的时间间隔 private static final float MEDUMVALUE = SensorManager.STANDARD_GRAVITY + 8.5f;//标准值为9.80665 private static final float SENSEVALUE = SensorManager.STANDARD_GRAVITY - 0.5f; protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); String[] array = { "注册,摇一摇,检测手机屏幕方向", "取消注册", }; tv_info = new TextView(this); tv_info.setTextColor(Color.BLUE); tv_info.setTextSize(TypedValue.COMPLEX_UNIT_SP, 16); tv_info.setPadding(20, 10, 20, 10); getListView().addFooterView(tv_info); setListAdapter(new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, new ArrayList<String>(Arrays.asList(array)))); sm = (SensorManager) getSystemService(SENSOR_SERVICE); vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);//权限【android.permission.VIBRATE】 } protected void onResume() { super.onResume(); if (sm != null) sm.registerListener(this, sm.getDefaultSensor(Sensor.TYPE_ACCELEROMETER), SensorManager.SENSOR_DELAY_NORMAL); } protected void onPause() {//保证在不需要使用传感器的时候禁用传感器 super.onPause(); if (sm != null) sm.unregisterListener(this); } @Override protected void onListItemClick(ListView l, View v, int position, long id) { switch (position) { case 0: if (sm != null) sm.registerListener(this, sm.getDefaultSensor(Sensor.TYPE_ACCELEROMETER), SensorManager.SENSOR_DELAY_NORMAL);//设置获取传感器信息的频率 break; case 1://Accelerometer 加速度传感器,32 if (sm != null) sm.unregisterListener(this); break; } } @Override public void onSensorChanged(SensorEvent event) {//在感应检测到Sensor的值有变化时会被调用到 //实时检测,震动 if (Math.abs(event.values[0]) > MEDUMVALUE || Math.abs(event.values[1]) > MEDUMVALUE || Math.abs(event.values[2]) > MEDUMVALUE) vibrator.vibrate(200); //抽样检测 if (System.currentTimeMillis() - lastTime < UPTATE_INTERVAL_TIME) return; lastTime = System.currentTimeMillis();// 现在的时间变成last时间 tv_info.setText("传感器类型 " + event.sensor.getName() + "\n时间戳 " + event.timestamp + "\n精度 " + event.accuracy + // "\nx轴方向的值,右侧向上时为正 " + event.values[0] + "\ny轴方向的值,前侧向上时为正 " + event.values[1] + "\nz轴方向的值,屏幕向上时为正 " + event.values[2]); //检测手机屏幕方向 if (event.values[0] > SENSEVALUE) Toast.makeText(this, "屏幕朝左,重力指向设备左边", Toast.LENGTH_SHORT).show(); else if (event.values[0] < -SENSEVALUE) Toast.makeText(this, "屏幕朝右,重力指向设备右边", Toast.LENGTH_SHORT).show(); else if (event.values[1] > SENSEVALUE) Toast.makeText(this, "屏幕朝前,重力指向设备下边", Toast.LENGTH_SHORT).show(); else if (event.values[1] < -SENSEVALUE) Toast.makeText(this, "屏幕朝后,重力指向设备上边", Toast.LENGTH_SHORT).show(); else if (event.values[2] > SENSEVALUE) Toast.makeText(this, "屏幕朝上", Toast.LENGTH_SHORT).show(); else if (event.values[2] < -SENSEVALUE) Toast.makeText(this, "屏幕朝下", Toast.LENGTH_SHORT).show(); } @Override public void onAccuracyChanged(Sensor paramSensor, int paramInt) {//在感应检测到Sensor的精密度有变化时被调用到 } }
方向示例
public class OrientationActivity2 extends Activity implements SensorEventListener { private TextView tv_info; private TextView tv_orientation; private ImageView iv; private SensorManager sm;//传感器管理器 private float[] accelValues = new float[3]; private float[] magValues = new float[3]; private long lastTime = System.currentTimeMillis(); private static final int UPTATE_INTERVAL_TIME = 500;// 两次检测的时间间隔 private float lastRotateDegree; protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); tv_info = (TextView) findViewById(R.id.tv_info); tv_orientation = (TextView) findViewById(R.id.tv_orientation); iv = (ImageView) findViewById(R.id.iv); sm = (SensorManager) getSystemService(SENSOR_SERVICE); } protected void onResume() { super.onResume(); if (sm != null) { sm.registerListener(this, sm.getDefaultSensor(Sensor.TYPE_MAGNETIC_FIELD), SensorManager.SENSOR_DELAY_NORMAL); sm.registerListener(this, sm.getDefaultSensor(Sensor.TYPE_ACCELEROMETER), SensorManager.SENSOR_DELAY_NORMAL); } } protected void onPause() {//保证在不需要使用传感器的时候禁用传感器 super.onPause(); if (sm != null) sm.unregisterListener(this); } @Override public void onSensorChanged(SensorEvent event) { switch (event.sensor.getType()) { case Sensor.TYPE_ACCELEROMETER: accelValues = event.values; break; case Sensor.TYPE_MAGNETIC_FIELD: magValues = event.values; break; } tv_info.setText("传感器类型 " + event.sensor.getName() + "\n获取到的值\n" + event.values[0] + "\n" + event.values[1] + "\n" + event.values[2]); if (System.currentTimeMillis() - lastTime < UPTATE_INTERVAL_TIME) return; lastTime = System.currentTimeMillis();// 现在的时间变成last时间 calculateOrientation(); } @Override public void onAccuracyChanged(Sensor paramSensor, int paramInt) { }
private void calculateOrientation() { float[] R = new float[9];//旋转数组 float[] values = new float[3];//模拟方向传感器的数据 //要填充的旋转数组;将磁场数据转换进实际的重力坐标中,一般默认情况下可以设置为null;加速度传感器数据;地磁传感器数据 SensorManager.getRotationMatrix(R, null, accelValues, magValues); SensorManager.getOrientation(R, values); //将弧度转化为角度后输出 tv_orientation.setText("角度\n"); for (float value : values) { value = (float) Math.toDegrees(value); tv_orientation.append(value + "\n"); } float value = -(float) Math.toDegrees(values[0]); if (Math.abs(value - lastRotateDegree) > 1) { //旋转补间动画 RotateAnimation animation = new RotateAnimation(lastRotateDegree, value, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f); animation.setFillAfter(true); iv.startAnimation(animation); lastRotateDegree = value; } value = -value; if (value >= -10 && value < 10) { tv_orientation.append("正北"); } else if (value >= 10 && value < 80) { tv_orientation.append("东北"); } else if (value >= 80 && value <= 100) { tv_orientation.append("正东"); } else if (value >= 100 && value < 170) { tv_orientation.append("东南"); } else if ((value >= 170 && value <= 180) || (value) >= -180 && value < -170) { tv_orientation.append("正南"); } else if (value >= -170 && value < -100) { tv_orientation.append("西南"); } else if (value >= -100 && value < -80) { tv_orientation.append("正西"); } else if (value >= -80 && value < -10) { tv_orientation.append("西北"); } } }
附件列表
时间: 2024-10-22 17:02:59