1. 陀螺仪基础:从传感器数据到三维旋转
在 Android 里,我们通过 SensorManager 来和陀螺仪打交道。第一步是获取服务,拿到 SensorManager 后,查询设备上有没有陀螺仪(Sensor.TYPE_GYROSCOPE)。现在绝大多数手机都有,但稳妥起见,检查一下总是好的。接下来注册一个监听器,告诉系统接收旋转数据。这里有个关键参数叫采样延迟,比如 SENSOR_DELAY_GAME,它决定了数据更新的频率。游戏场景下需要高频率(约 50Hz),而像屏幕旋转这种应用,用 SENSOR_DELAY_UI(约 15Hz)就足够了,能省不少电。
数据会在 onSensorChanged 回调里拿到一个 SensorEvent 对象。它的 values 数组里装着数据:values[0]、values[1]、values[2] 分别对应绕 X、Y、Z 轴的角速度,单位是弧度/秒。这里有个右手定则需要记住:伸出你的右手,拇指指向某个轴的正方向,其余四指弯曲的方向就是正旋转的方向。举个例子,手机平放在桌上,屏幕朝上。此时 Z 轴垂直屏幕向上。如果你逆时针旋转手机,那么 Z 轴会得到一个正的角速度值;顺时针转,值就是负的。X 轴和 Y 轴同理。理解这三个数值的物理意义,是使用陀螺仪的第一步。
2. 数据处理核心:从角速度到实际角度
拿到原始的角速度数据,我们通常不会直接使用,因为它描述的是'瞬时速度'。我们更关心的是'总共转了多少度'。这就好比知道汽车每一秒的速度,想知道它开了多远,就需要把速度乘以时间,然后累加起来。对陀螺仪数据来说,这个过程叫积分。
积分在代码里实现就是一个简单的累加过程。关键是要利用好 SensorEvent 提供的 timestamp 字段,这个时间戳是纳秒精度的。我们需要计算两次传感器数据回调之间的时间差 dT(单位转换成秒),然后用角速度乘以这个时间差,得到这一小段时间内转过的角度(弧度),最后累加到总角度上。
下面给出一个经典的积分代码片段:
private static final float NS2S = 1.0f / 1000000000.0f; // 纳秒到秒的转换系数
private float timestamp;
private float[] angle = new float[3]; // 用于累积 X,Y,Z 轴旋转角度的数组
@Override
public void onSensorChanged(SensorEvent event) {
if (timestamp != 0) {
// 计算与上一次回调的时间差(秒)
final (event.timestamp - timestamp) * NS2S;
angle[] += event.values[] * dT;
angle[] += event.values[] * dT;
angle[] += event.values[] * dT;
timestamp = event.timestamp;
} {
timestamp = event.timestamp;
}
}

