来说说CoreMotion

关于CoreMotion

Core Motion可以让开发者从各个内置传感器那里获取未经修改的传感数据,并观测或响应设备各种运动和角度变化。这些传感器包括陀螺仪、加速器和磁力仪(罗盘)。

CoreMotion负责处理的数据

它包含几种类型的数据,这些类都是CMLogItem类的子类:

  • 加速度数据 :CMAccelerometerData
typedef struct {
   double x;
   double y;
   double z;
} CMAcceleration;
@interface CMAccelerometerData : CMLogItem
{
@private
   id _internal;
}
//加速度的数据对象
@property(readonly, nonatomic) CMAcceleration acceleration;
@end
  • 螺旋仪数据 :CMGyroData
typedef struct {
   double x;
   double y;
   double z; 
} CMRotationRate;
@interface CMGyroData : CMLogItem
{
@private
   id _internal;
}
//螺旋仪数据对象
@property(readonly, nonatomic) CMRotationRate rotationRate;
@end
  • 磁感应数据 :magnetometerData
typedef struct {
   double x;
   double y;
   double z;
} CMMagneticField;
@interface CMMagnetometerData : CMLogItem
{
@private
   id _internal;
}
//磁力对象
@property(readonly, nonatomic) CMMagneticField magneticField;
@end
  • 前三种数据通过复杂运算得到的设备的运动数据 :CMDeviceMotion
@interface CMDeviceMotion : CMLogItem
{
@private
   id _internal;
}
//设备的状态对象
@property(readonly, nonatomic) CMAttitude *attitude;
//设备的角速度
@property(readonly, nonatomic) CMRotationRate rotationRate;
//设备的重力加速度
@property(readonly, nonatomic) CMAcceleration gravity;
//用户嫁给设备的加速度 设备的总加速度为重力加速度叫上用户给的加速度
@property(readonly, nonatomic) CMAcceleration userAcceleration;
//设备的磁场矢量对象
@property(readonly, nonatomic) CMCalibratedMagneticField magneticField NS_AVAILABLE(NA,5_0);

我们来看下 attitude 这个对象, 它又封装了许多设备的状态属性

@interface CMAttitude : NSObject 
{
@private
   id _internal;
}
//设备的欧拉角roll
@property(readonly, nonatomic) double roll;
//设备的欧拉角pitch
@property(readonly, nonatomic) double pitch;
//设备的欧拉角yaw
@property(readonly, nonatomic) double yaw;
//设备状态的旋转矩阵
@property(readonly, nonatomic) CMRotationMatrix rotationMatrix;
//设备状态的四元数
@property(readonly, nonatomic) CMQuaternion quaternion;
@end

CoreMotion的使用

CoreMotion中获取数据主要是两种方式:

  • 一种是Push,就是你提供一个线程管理器NSOperationQueue,再提供一个Block,这样,CoreMotion自动在每一个采样数据到来的时候回调这个Block,进行处理。在这中情况下,block中的操作会在你自己的主线程内执行。

  • 一种是 Pull,在这个方式里,你必须主动去像CMMotionManager要数据,这个数据就是最近一次的采样数据。你不去要,CMMotionManager就不会给你。当然,在这种情况下,CoreMotion所有的操作都在自己的后台线程中进行,不会有任何干扰你当前线程的行为。

有两点需要注意:第一就是需要检测是否可用,第二就是在不需要的时候,停止更新数据。苹果建议在使用CoreMotionManager的时候采用单例模式。

我们以CMAccelerometerData为例,其他方式都是一样的

pull获取方式

if ([_motionManager isAccelerometerAvailable]) {    
   // 设置加速计采样频率
   [_motionManager setAccelerometerUpdateInterval:1 / 40.0];
   [_motionManager startAccelerometerUpdates];
}else {
   NSLog(@"设备不支持加速计");
}

//可以写个定时器去定时主动获取数据
//获取数据
CMAcceleration acceleration=_motionManager.accelerometerData.acceleration;
NSLog(@"%f---%f---%f",acceleration.x,acceleration.y,acceleration.z);

push获取方式

//判断加速计是否可用
if([_motionManager isAccelerometerAvailable]) {
   // 设置加速计频率
   [_motionManager setAccelerometerUpdateInterval:1 / 40.0];
   //开始采样数据
   [_motionManager startAccelerometerUpdatesToQueue:[NSOperationQueue mainQueue] withHandler:^(CMAccelerometerData *accelerometerData, NSError *error) {
       NSLog(@"%f---%f",accelerometerData.acceleration.x,accelerometerData.acceleration.y);
   }];
} else{
   NSLog(@"设备不支持加速计");
}

我们再列举几个它的其他属性和方法:

@interface CMMotionManager : NSObject
{
@private
   id _internal;
}
//设置加速度传感器更新帧率
@property(assign, nonatomic) NSTimeInterval accelerometerUpdateInterval __TVOS_PROHIBITED;
//加速度传感器是否可用
@property(readonly, nonatomic, getter=isAccelerometerAvailable) BOOL accelerometerAvailable __TVOS_PROHIBITED;
//加速度传感器是否激活
@property(readonly, nonatomic, getter=isAccelerometerActive) BOOL accelerometerActive __TVOS_PROHIBITED;
//加速度传感器数据对象
@property(readonly, nullable) CMAccelerometerData *accelerometerData __TVOS_PROHIBITED;
//pull方式开始更新加速度数据
- (void)startAccelerometerUpdates __TVOS_PROHIBITED;
//push方式更新加速度数据
- (void)startAccelerometerUpdatesToQueue:(NSOperationQueue *)queue withHandler:(CMAccelerometerHandler)handler __TVOS_PROHIBITED;
//停止更新加速度数据
- (void)stopAccelerometerUpdates __TVOS_PROHIBITED;
//设备状态更新帧率
@property(assign, nonatomic) NSTimeInterval deviceMotionUpdateInterval __TVOS_PROHIBITED;
//参考器枚举
+ (CMAttitudeReferenceFrame)availableAttitudeReferenceFrames NS_AVAILABLE(NA,5_0) __TVOS_PROHIBITED;
@property(readonly, nonatomic) CMAttitudeReferenceFrame attitudeReferenceFrame NS_AVAILABLE(NA,5_0) __TVOS_PROHIBITED;
//设备运动信息是否可用
@property(readonly, nonatomic, getter=isDeviceMotionAvailable) BOOL deviceMotionAvailable __TVOS_PROHIBITED;
//设备运动信息是否激活
@property(readonly, nonatomic, getter=isDeviceMotionActive) BOOL deviceMotionActive __TVOS_PROHIBITED;
//设备运动信息对象
@property(readonly, nullable) CMDeviceMotion *deviceMotion __TVOS_PROHIBITED;
//pull方式开始刷新运动信息
- (void)startDeviceMotionUpdates __TVOS_PROHIBITED;
//push方式开始刷新运动信息
- (void)startDeviceMotionUpdatesToQueue:(NSOperationQueue *)queue withHandler:(CMDeviceMotionHandler)handler __TVOS_PROHIBITED;
//使用某个参考系
- (void)startDeviceMotionUpdatesUsingReferenceFrame:(CMAttitudeReferenceFrame)referenceFrame NS_AVAILABLE(NA,5_0) __TVOS_PROHIBITED;
//push方式开始刷新设备运动信息
- (void)startDeviceMotionUpdatesUsingReferenceFrame:(CMAttitudeReferenceFrame)referenceFrame toQueue:(NSOperationQueue *)queue withHandler:(CMDeviceMotionHandler)handler NS_AVAILABLE(NA,5_0) __TVOS_PROHIBITED;
//停止刷新设备运动信息
- (void)stopDeviceMotionUpdates __TVOS_PROHIBITED;

关于CMDeviceMotion

我们都知道,设备静止时受到的地球引力为1g,1g是物体在地球的海平面上受到的下拉力(9.8米/秒²)。假如设备从高处掉落,其加速计测量到的加速度将为0g。假如设备水平放在桌面上,则加速计测量出的加速度为1g,且方向朝上。

加速计测量3个轴(x、y和z)上的值,如图所示:

来说说CoreMotion_第1张图片

这个轴在方向上有些不同于传统坐标轴,考虑以下实际情况:

1g重力的分布情况是:y=-1.0

来说说CoreMotion_第2张图片

1g重力的分布情况是:x=1.0

来说说CoreMotion_第3张图片

1g重力的分布情况是:z=-1.0

来说说CoreMotion_第4张图片

1g重力的分布情况是:x= 0.707,y=-0.707

来说说CoreMotion_第5张图片

1g重力的分布情况是:y=-0.707,z= 0.707

来说说CoreMotion_第6张图片

仅当设备的朝向相对于重力的方向发生变化时,加速计才能检测到;要同时检测设备的朝向和运动数据,就需要用到陀螺仪了。当查询设备的陀螺仪时,它将报告设备绕x, y, z轴的旋转速度,单位为弧度/秒;2弧度相当于一整圈,因此陀螺仪返回读数2表示设备绕相应的轴每秒转一圈。

有两种方式访问设备的朝向和运动数据,一种是通过UIDevice请求朝向通知,另一种是利用框架Core Motion定期地直接访问加速计和陀螺仪数据。

  • 通过UIDevice请求朝向通知

虽然可直接查询加速计并使用它返回的值判断设备的朝向,但Apple为开发人员简化了这项工作。单例UIDevice表示当前设备,它包含方法beginGeneratingDeviceOrientationNotifications,该方法命令iOS将朝向通知发送到通知中心(NSNotificationCenter)。启动通知后,就可以注册一个NSNotificationCenter实例,以便设备的朝向发生变化时自动调用指定的方法。

通过访问UIDevice的属性orientation来获得设备当前朝向,该属性的类型为枚举值

UIDeviceOrientation,有6个预定义值:
UIDeviceOrientationFaceUp — 设备正面朝上
UIDeviceOrientationFaceDown — 设备正面朝下
UIDeviceOrientationPortrait — 纵向(Home键在下)
UIDeviceOrientationPortraitUpsideDown — 纵向倒转(Home键在上)
UIDeviceOrientationLandscapeLeft — Home键在左边的横向
UIDeviceOrientationLandscapeRight — Home键在右边的横向

- (void)viewDidLoad
{
   [[UIDevice currentDevice] beginGeneratingDeviceOrientationNotifications];

   [[NSNotificationCenter defaultCenter] addObserver:selfselector:@selector(orientationChanged:) name:@"UIDeviceOrientationDidChangeNotification"object:nil];

   [super viewDidLoad];
}

- (void)orientationChanged:(NSNotification *)notification
{
  UIDeviceOrientation orientation = [[UIDevice currentDevice] orientation];
  NSLog(@"当前朝向枚举数字值:%d",orientation);

  switch (orientation) {
    case UIDeviceOrientationPortrait:
      self.lblOriention.text = @"Portrait";
      break;
    case UIDeviceOrientationPortraitUpsideDown:
      self.lblOriention.text = @"Portrait Upside Down";
      break;
    case UIDeviceOrientationLandscapeLeft:
      self.lblOriention.text = @"Landscape Left";
      break;
    case UIDeviceOrientationLandscapeRight:
      self.lblOriention.text = @"Landscape Right";
      break;
    case UIDeviceOrientationFaceUp:
      self.lblOriention.text = @"Face Up";
      break;
    case UIDeviceOrientationFaceDown:
      self.lblOriention.text = @"Face Down";
      break;
    default:
      self.lblOriention.text = @"Unknown";
      break;
   }
}
  • 使用Core Motion读取加速计和陀螺仪数据

利用UIDevice只能判断极端朝向,应用程序经常要获悉这些朝向之间的过渡状态,如设备处于某个倾斜位置。Core Motion运动管理器让您能够指定从加速计和陀螺仪那里接收更新的频率(单位为秒),还让您能够直接指定一个处理程序块(handle block),每当更新就绪时都将执行该处理程序块。

实际加速度在Core Motion里被分解成了两部分:Gravity和UserAcceleration。Gravity代表重力1g在设备的分布情况,UserAcceleration代表设备运动中的加速度分布情况。将这两者相加就等于实际加速度。Gravity的三个轴所受的重力加起来始终等于1g,而UserAcceleration取决于单位时间内动作的幅度大小。

CMRotationRate的X,Y,Z分别代表三个轴上的旋转速率,单位为弧度/秒。旋转速度为1弧度/秒,意味着设备每秒旋转半圈。这里复习一下弧度与角度的转换:

1角度 = π/180 弧度

1弧度 = 180/π角度

360角度 = 360 * π/180 = 2π弧度 = 一整圈

CMAttitude的三个属性Yaw,Pitch和Roll分别代表左右摆动、俯仰以及滚动。可以将设备想象成一架飞机,下面的gif图演示了各种运动状态:

Yaw的运动状态:

来说说CoreMotion_第7张图片
yaw.gif

Pitch的运动状态:

来说说CoreMotion_第8张图片
pitch.gif

Roll的运动状态:

来说说CoreMotion_第9张图片
roll.gif

实例

下面的代码实现了这样一个界面,通过CMMotionManager返回了设备的各个状态值:

来说说CoreMotion_第10张图片
#import "ViewController.h"


@interface ViewController ()

@property (strong, nonatomic) IBOutlet UILabel *lblYaw;
@property (strong, nonatomic) IBOutlet UILabel *lblPitch;
@property (strong, nonatomic) IBOutlet UILabel *lblRoll;

@property (strong, nonatomic) IBOutlet UILabel *lblAccelerometerX;
@property (strong, nonatomic) IBOutlet UILabel *lblAccelerometerY;
@property (strong, nonatomic) IBOutlet UILabel *lblAccelerometerZ;

@property (strong, nonatomic) IBOutlet UILabel *lblGravityX;
@property (strong, nonatomic) IBOutlet UILabel *lblGravityY;
@property (strong, nonatomic) IBOutlet UILabel *lblGravityZ;

@property (strong, nonatomic) IBOutlet UILabel *lblRotationRateX;
@property (strong, nonatomic) IBOutlet UILabel *lblRotationRateY;
@property (strong, nonatomic) IBOutlet UILabel *lblRotationRateZ;

@property (strong, nonatomic) CMMotionManager *motionManager;

- (IBAction)motionSwitchHandler:(id)sender;

@end

@implementation ViewController

- (void)viewDidLoad
{
   [super viewDidLoad];

   self.motionManager = [[CMMotionManager alloc] init];
   self.motionManager.deviceMotionUpdateInterval = 1.0f/10.0f; //1秒10次
}

- (void)controlHardware
{
   [self.motionManager startDeviceMotionUpdatesToQueue:[NSOperationQueue currentQueue] withHandler:^(CMDeviceMotion *motion, NSError *error) {
       //Acceleration
       if(fabs(motion.userAcceleration.x)>1.3f)
           self.lblAccelerometerX.text = [NSString stringWithFormat:@"%.2f",motion.userAcceleration.x];
       if(fabs(motion.userAcceleration.y)>1.3f)
           self.lblAccelerometerY.text = [NSString stringWithFormat:@"%.2f",motion.userAcceleration.y];
       if(fabs(motion.userAcceleration.z)>1.3f)
           self.lblAccelerometerZ.text = [NSString stringWithFormat:@"%.2f",motion.userAcceleration.z];
       //Gravity
       self.lblGravityX.text = [NSString stringWithFormat:@"%.2f",motion.gravity.x];
       self.lblGravityY.text = [NSString stringWithFormat:@"%.2f",motion.gravity.y];
       self.lblGravityZ.text = [NSString stringWithFormat:@"%.2f",motion.gravity.z];
       //yaw,pitch,roll
       self.lblYaw.text = [NSString stringWithFormat:@"%.2f",motion.attitude.yaw];
       self.lblPitch.text = [NSString stringWithFormat:@"%.2f",motion.attitude.pitch];
       self.lblRoll.text = [NSString stringWithFormat:@"%.2f",motion.attitude.roll];
       //Gyroscope's rotationRate(CMRotationRate)
       self.lblRotationRateX.text = [NSString stringWithFormat:@"%.2f",motion.rotationRate.x];
       self.lblRotationRateY.text = [NSString stringWithFormat:@"%.2f",motion.rotationRate.y];
       self.lblRotationRateZ.text = [NSString stringWithFormat:@"%.2f",motion.rotationRate.z];
   }];
}

- (IBAction)motionSwitchHandler:(id)sender
{
   UISwitch *motionSwitch = (UISwitch *)sender;
   if(motionSwitch.on)
   {
       [self controlHardware];
   }
   else
   {
       [self.motionManager stopDeviceMotionUpdates];
   }
}

- (void)didReceiveMemoryWarning
{
   [super didReceiveMemoryWarning];
   // Dispose of any resources that can be recreated.
}
@end

你可能感兴趣的:(来说说CoreMotion)