iOS 传感器集锦
https://www.jianshu.com/p/5fc26af852b6
传感器集锦:指纹识别、运动传感器、加速计、环境光感、距离传感器、磁力计、陀螺仪
一、指纹识别
应用:指纹解锁、指纹登录、指纹支付
苹果从iPhone5S开始,具有指纹识别技术,从iOS8.0之后苹果允许第三方 App 使用 Touch ID进行身份验证。
连续三次指纹识别错误后,会自动弹出密码框,通过Touch ID的密码进行身份验证,如果此时取消密码验证,再2次指纹识别失败后,也就是 3 + 2 = 5次指纹识别失败后,Touch ID功能被锁定,就只能通过密码来进行身份验证和解锁Touch ID 功能。
全部代码如下:
#import <LocalAuthentication/LocalAuthentication.h>
-(void)OnTouchIDBtn:(UIButton *)sender{
//判断设备是否支持Touch ID
if ([[UIDevice currentDevice].systemVersion floatValue] < 8.0) {
[self createAlterView:@"不支持指纹识别"];
return;
}else{
LAContext *ctx = [[LAContext alloc] init];
//设置 输入密码 按钮的标题
ctx.localizedFallbackTitle = @"验证登录密码";
//设置 取消 按钮的标题 iOS10之后
ctx.localizedCancelTitle = @"取消";
//检测指纹数据库更改 验证成功后返回一个NSData对象,否则返回nil
//ctx.evaluatedPolicyDomainState;
// 这个属性应该是类似于支付宝的指纹开启应用,如果你打开他解锁之后,按Home键返回桌面,再次进入支付宝是不需要录入指纹的。因为这个属性可以设置一个时间间隔,在时间间隔内是不需要再次录入。默认是0秒,最长可以设置5分钟
//ctx.touchIDAuthenticationAllowableReuseDuration = 5;
NSError * error;
_localizedReason = @"通过Home键验证已有手机指纹";
// 判断设备是否支持指纹识别
if ([ctx canEvaluatePolicy:LAPolicyDeviceOwnerAuthentication error:&error]) {
// 验证指纹是否匹配,需要弹出输入密码的弹框的话:iOS9之后用 LAPolicyDeviceOwnerAuthentication ; iOS9之前用LAPolicyDeviceOwnerAuthenticationWithBiometrics
[ctx evaluatePolicy:LAPolicyDeviceOwnerAuthentication localizedReason:_localizedReason reply:^(BOOL success, NSError * error) {
if (success) {
[self createAlterView:@"指纹验证成功"];
}else{
// 错误码 error.code
NSLog(@"指纹识别错误描述 %@",error.description);
// -1: 连续三次指纹识别错误
// -2: 在TouchID对话框中点击了取消按钮
// -3: 在TouchID对话框中点击了输入密码按钮
// -4: TouchID对话框被系统取消,例如按下Home或者电源键
// -8: 连续五次指纹识别错误,TouchID功能被锁定,下一次需要输入系统密码
NSString * message;
switch (error.code) {
case -1://LAErrorAuthenticationFailed
message = @"已经连续三次指纹识别错误了,请输入密码验证";
_localizedReason = @"指纹验证失败";
break;
case -2:
message = @"在TouchID对话框中点击了取消按钮";
return ;
break;
case -3:
message = @"在TouchID对话框中点击了输入密码按钮";
break;
case -4:
message = @"TouchID对话框被系统取消,例如按下Home或者电源键或者弹出密码框";
break;
case -8:
message = @"TouchID已经被锁定,请前往设置界面重新启用";
break;
default:
break;
}
[self createAlterView:message];
}
}];
}else{
if (error.code == -8) {
[self createAlterView:@"由于五次识别错误TouchID已经被锁定,请前往设置界面重新启用"];
}else{
[self createAlterView:@"TouchID没有设置指纹,请前往设置"];
}
}
}
}
- (void)createAlterView:(NSString *)message{
UIAlertController * vc = [UIAlertController alertControllerWithTitle:@"提示" message:message preferredStyle:UIAlertControllerStyleAlert];
[self presentViewController:vc animated:NO completion:^(void){
[NSThread sleepForTimeInterval:1.0];
[vc dismissViewControllerAnimated:YES completion:nil];
}];
}
二、运动传感器/加速计/陀螺仪
应用:水平位置测试、摇一摇、计步器、游戏、特效动画
加速计和运动传感器主要监测设备在X、Y、Z轴上的加速度 ,根据加速度数值,就可以判断出在各个方向上的作用力度,陀螺仪主要用来监测设备的旋转方向和角度。
这几个传感器都是依赖于苹果官方CoreMotion框架,用法都差不多,先判断各个传感器是否可用开启,然后设置各个传感器的采集频率,接着就开始采集数据,并返回采集到的运动信息参数:各个方向的重力加速度、旋转方向角度等等。
- 运动传感器的示例代码如下,其它的用法都差不多,只是相关的方法名称、属性名称和返回的参数类型和值不同,详情可以看demo,注释还算清晰;对于眼镜的绘制可以看下我之前的笔记:CALayer系列、CGContextRef、UIBezierPath、文本属性Attributes
self.motionManager= [[CMMotionManager alloc] init];
//判断设备运动传感器是否可用
if(!self.motionManager.isDeviceMotionAvailable){
NSLog(@"手机没有此功能,换肾吧");
}
//更新速率是100Hz
self.motionManager.deviceMotionUpdateInterval = 0.1;
//开始更新采集数据
//需要时采集数据
//[motionManager startDeviceMotionUpdates];
//实时获取数据
[self.motionManager startDeviceMotionUpdatesToQueue:[NSOperationQueue mainQueue] withHandler:^(CMDeviceMotion * _Nullable motion, NSError * _Nullable error) {
//获取X的值
double x = motion.gravity.x;
//手机水平位置测试
//判断y是否小于0,大于等于-1.0
if(motion.gravity.y < 0.0 && motion.gravity.y >= -1.0){
//设置旋转
[imageView1 setRotation:80 * motion.gravity.y];
}else if (motion.gravity.z * -1 > 0 && motion.gravity.z * -1 <= 1.0){
[imageView1 setRotation:80 - (80 * motion.gravity.z * -1)];
}
//X、Y方向上的夹角
double rotation = atan2(motion.gravity.x, motion.gravity.y) - M_PI;
NSLog(@"%.2f",rotation);
//图片始终保持垂直方向
imageView2.transform = CGAffineTransformMakeRotation(rotation);
}];
三、环境光感处理器
应用:常见的比如说根据环境的亮度去调整屏幕的亮度,在黑暗情况下,手机会自动调暗屏幕亮度,以防刺眼;iPhone 系统相机拍照时光线暗时会自动打开闪光灯;共享单车在黑暗的情况下扫码时检测到特别暗就自动提示打开闪光灯。
利用摄像头获取环境光感参数,通过摄像头采集环境参数,然后在代理方法中输出,光感越小,环境越暗,然后根据需要做相应的操作。
示例中是环境变暗后就自动提示是否打开闪光灯,打开之后,环境变亮后会自动提示是否关闭闪光灯。
// 1.获取硬件设备
AVCaptureDevice *device = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];
if (device == nil) {
NSLog(@"该换肾了");
return;
}
// 2.创建输入流
AVCaptureDeviceInput *input = [[AVCaptureDeviceInput alloc]initWithDevice:device error:nil];
// 3.创建设备输出流
AVCaptureVideoDataOutput *output = [[AVCaptureVideoDataOutput alloc] init];
[output setSampleBufferDelegate:self queue:dispatch_get_main_queue()];
// AVCaptureSession属性
_session = [[AVCaptureSession alloc]init];
// 设置为高质量采集率
[_session setSessionPreset:AVCaptureSessionPresetHigh];
// 添加会话输入和输出
if ([_session canAddInput:input]) {
[_session addInput:input];
}
if ([_session canAddOutput:output]) {
[_session addOutput:output];
}
// 9.启动会话
[_session startRunning];
#pragma mark- AVCaptureVideoDataOutputSampleBufferDelegate的方法
- (void)captureOutput:(AVCaptureOutput *)captureOutput didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer fromConnection:(AVCaptureConnection *)connection {
CFDictionaryRef metadataDict = CMCopyDictionaryOfAttachments(NULL,sampleBuffer, kCMAttachmentMode_ShouldPropagate);
NSDictionary *metadata = [[NSMutableDictionary alloc] initWithDictionary:(__bridge NSDictionary*)metadataDict];
CFRelease(metadataDict);
NSDictionary *exifMetadata = [[metadata objectForKey:(NSString *)kCGImagePropertyExifDictionary] mutableCopy];
float brightnessValue = [[exifMetadata objectForKey:(NSString *)kCGImagePropertyExifBrightnessValue] floatValue];
NSLog(@"环境光感 : %f",brightnessValue);
// 根据brightnessValue的值来打开和关闭闪光灯
AVCaptureDevice *device = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];
BOOL result = [device hasTorch];// 判断设备是否有闪光灯
if ((brightnessValue < 0) && result) {// 打开闪光灯
if(device.torchMode == AVCaptureTorchModeOn){
return;
}
UIAlertController * alertVC = [UIAlertController alertControllerWithTitle:@"提示" message:@"小主是否要打开闪光灯?" preferredStyle:UIAlertControllerStyleAlert];
UIAlertAction * openAction = [UIAlertAction actionWithTitle:@"打开" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
[device lockForConfiguration:nil];
[device setTorchMode: AVCaptureTorchModeOn];//开
[device unlockForConfiguration];
}];
[alertVC addAction:openAction];
UIAlertAction * cancleAction = [UIAlertAction actionWithTitle:@"取消" style:UIAlertActionStyleCancel handler:^(UIAlertAction * _Nonnull action) {}];
[alertVC addAction:cancleAction];
[self presentViewController:alertVC animated:NO completion:nil];
}else if((brightnessValue > 0) && result) {// 关闭闪光灯
if(device.torchMode == AVCaptureTorchModeOn){
UIAlertController * alertVC = [UIAlertController alertControllerWithTitle:@"提示" message:@"小主是否要关闭闪光灯?" preferredStyle:UIAlertControllerStyleAlert];
UIAlertAction * openAction = [UIAlertAction actionWithTitle:@"关闭" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
[device lockForConfiguration:nil];
[device setTorchMode: AVCaptureTorchModeOff];//关
[device unlockForConfiguration];
}];
[alertVC addAction:openAction];
UIAlertAction * cancleAction = [UIAlertAction actionWithTitle:@"取消" style:UIAlertActionStyleCancel handler:^(UIAlertAction * _Nonnull action) { }];
[alertVC addAction:cancleAction];
[self presentViewController:alertVC animated:NO completion:nil];
}
}
}
四、距离传感器
距离传感器: 感应是否有其他物体靠近屏幕,iPhone手机中内置了距离传感器,位置在手机的听筒附近,当我们在打电话或听微信语音的时候靠近听筒,手机的屏幕会自动熄灭,这就靠距离传感器来控制
首先打开距离传感器,然后添加通知UIDeviceProximityStateDidChangeNotification监听有物品靠近还是离开,从而做出操作,记得最后要关闭距离传感器,有始有终哦。
示例中是默认用扬声器播放音乐,当有物体(比如耳朵)靠近听筒附近时就切换听筒播放音乐,物体离开后就继续用扬声器播放音乐。
- (void)distanceSensor{
// 打开距离传感器
[UIDevice currentDevice].proximityMonitoringEnabled = YES;
// 通过通知监听有物品靠近还是离开
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(proximityStateDidChange:) name:UIDeviceProximityStateDidChangeNotification object:nil];
AVAudioSession *audioSession = [AVAudioSession sharedInstance];
//默认情况下扬声器播放
[audioSession setCategory:AVAudioSessionCategoryPlayback error:nil];
[audioSession setActive:YES error:nil];
NSString * path = [[NSBundle mainBundle] pathForResource:@"SeeYouAgain" ofType:@"mp3"];
if(path == nil){
return;
}
_play = [[AVPlayer alloc] initWithURL:[NSURL fileURLWithPath:path]];
[_play play];
}
- (void)proximityStateDidChange:(NSNotification *)note
{
if ([UIDevice currentDevice].proximityState) {
NSLog(@"有东西靠近");
//听筒播放
[[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayAndRecord error:nil];
} else {
NSLog(@"有物体离开");
//扬声器播放
[[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayback error:nil];
}
}
- (void)dealloc{
[_play pause];
_play = nil;
//关闭距离传感器
[UIDevice currentDevice].proximityMonitoringEnabled = NO;
[self removeObserver];
}
五、磁力计
请看我的上一篇博客:iOS仿系统指南针
传࿆送࿆之࿆门࿆ ——> 传感器集锦
温馨提示:请真机调试看效果。
如果需要跟我交流的话:
※ Github: https://github.com/wsl2ls
※ 个人博客:https://wsl2ls.github.io
※ 简书:https://www.jianshu.com/u/e15d1f644bea
作者:且行且珍惜_iOS
链接:https://www.jianshu.com/p/5fc26af852b6
來源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。
距离传感器
默认情况下,每一个应用程序距离传感器都是关闭状态
如果需要,需要通过代码将其打开
// 过期代码 [UIApplication sharedApplication].proximitySensingEnabled
[UIDevice currentDevice].proximityMonitoringEnabled = YES;
三 加速计信息获取##
UIAccelerometer方法,该方法已过期
- (void)viewDidLoad {
[super viewDidLoad];
// 1.获取单例对象
UIAccelerometer *acceleromter = [UIAccelerometer sharedAccelerometer];
// 2.设置代理
acceleromter.delegate = self;
// 3.设置采样间隔
acceleromter.updateInterval = 1.0 / 5;
}
#pragma mark - 实现UIAccelerometer的代理方法
- (void)accelerometer:(UIAccelerometer *)accelerometer didAccelerate:(UIAcceleration *)acceleration
{
NSLog(@"x:%f y:%f z:%f", acceleration.x, acceleration.y, acceleration.z);
}
四 CoreMotion方法##
Core Motion获取数据的两种方式
push:实时采集所有数据(采集频率高)
pull:在有需要的时候,再主动去采集数据
加速计信息获取(pull/push)
// 1.创建运动管理者对象
CMMotionManager *mgr = [[CMMotionManager alloc] init];
// 2.判断加速计是否可用
if (!self.mgr.isAccelerometerAvailable) {
NSLog(@"加速计不可用,请更换手机");
return;
}
// 3.设置采样间隔
self.mgr.accelerometerUpdateInterval = 1.0;
// 4.开始采样
[self.mgr startAccelerometerUpdatesToQueue:[NSOperationQueue mainQueue] withHandler:^(CMAccelerometerData * _Nullable accelerometerData, NSError * _Nullable error) {
if (error) {
NSLog(@"%@", error);
return;
}
// 获取加速计的值
CMAcceleration acceleration = accelerometerData.acceleration;
NSLog(@"x:%f y:%f z:%f", acceleration.x, acceleration.y, acceleration.z);
}];
// 开始采样
[self.mgr startAccelerometerUpdates];
1
2
陀螺仪信息获取(pull/push)
// push方式获取陀螺仪信息
// 1.判断陀螺仪是否可用
if (!self.mgr.isGyroAvailable) {
NSLog(@"设备小于iPhone4,或者陀螺仪损坏");
return;
}
// 2.设置采样
self.mgr.gyroUpdateInterval = 1.0 / 10;
// 3.开始采样
[self.mgr startGyroUpdatesToQueue:[NSOperationQueue mainQueue] withHandler:^(CMGyroData * _Nullable gyroData, NSError * _Nullable error) {
if (error) {
NSLog(@"%@", error);
return;
}
// 获取陀螺仪的信息
CMRotationRate rotationRate = gyroData.rotationRate;
NSLog(@"x:%f y:%f z:%f", rotationRate.x, rotationRate.y, rotationRate.z);
}];
// 开始采样
[self.mgr startGyroUpdates];
1
2
五 摇⼀摇功能##
监控摇一摇的方法
方法1:通过分析加速计数据来判断是否进行了摇一摇操作(比较复杂)
方法2:iOS自带的Shake监控API(非常简单)
判断摇一摇的步骤:实现3个摇一摇监听方法
- (void)motionBegan:(UIEventSubtype)motion withEvent:(UIEvent *)event /** 检测到摇动 */
- (void)motionCancelled:(UIEventSubtype)motion withEvent:(UIEvent *)event /** 摇动取消(被中断) */
- (void)motionEnded:(UIEventSubtype)motion withEvent:(UIEvent *)event /** 摇动结束 */
六 计步器##
CMStepCounter方法,已过期
// 1.判断计步器是否可用
if (![CMStepCounter isStepCountingAvailable]) {
NSLog(@"计步器不可用");
return;
}
// 2.开始计步
// 2.1.创建计步器
CMStepCounter *stepCounter = [[CMStepCounter alloc] init];
// 2.2.开始计步
// updateOn : 用户走了多少步之后,更新block
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
[stepCounter startStepCountingUpdatesToQueue:queue updateOn:5 withHandler:^(NSInteger numberOfSteps, NSDate * _Nonnull timestamp, NSError * _Nullable error) {
if (error) return;
NSString *stepString = [NSString stringWithFormat:@"您一共走了%ld步", numberOfSteps];
[self.stepLabel performSelectorOnMainThread:@selector(setText:) withObject:stepString waitUntilDone:YES];
}];
新方法CMPedometer
// 1.判断计步器是否可用
if (![CMPedometer isStepCountingAvailable]) {
return;
}
// 2.开始计步
// 2.1.创建计步对象
CMPedometer *pedometer = [[CMPedometer alloc] init];
// 2.2.开始计步
// FromDate : 从什么时间开始计步
NSDate *date = [NSDate date];
[self.pedometer startPedometerUpdatesFromDate:date withHandler:^(CMPedometerData * _Nullable pedometerData, NSError * _Nullable error) {
if (error) {
NSLog(@"%@", error);
return;
}
NSLog(@"您一共走了%@步", pedometerData.numberOfSteps);
}];
案例:计算7天一共走了多少步
NSDateFormatter *fmt = [[NSDateFormatter alloc] init];
fmt.dateFormat = @"yyyy-MM-dd";
NSDate *fromDate = [fmt dateFromString:@"2015-9-26"];
NSDate *toDate = [fmt dateFromString:@"2015-9-28"];
[self.pedometer queryPedometerDataFromDate:fromDate toDate:toDate withHandler:^(CMPedometerData * _Nullable pedometerData, NSError * _Nullable error) {
NSLog(@"%@", pedometerData.numberOfSteps);
}];
---------------------
作者:贱兔不二
来源:CSDN
原文:https://blog.csdn.net/ab20514/article/details/48846867
版权声明:本文为博主原创文章,转载请附上博文链接!
iOS 传感器集锦的更多相关文章
- iOS传感器集锦、飞机大战、开发调试工具、强制更新、Swift仿QQ空间头部等源码
iOS精选源码 飞机大作战 MUPhotoPreview -简单易用的图片浏览器 LLDebugTool是一款针对开发者和测试者的调试工具,它可以帮... 多个UIScrollView.UITable ...
- [iOS]技巧集锦:UICollectionView在旋转屏幕后Cell中的约束不起作用或自动布局失效
这似乎是iOS的一个BUG(ref: stackoverflow的大神们讲的) 解决方案 在继承自UITableViewCell的子类中的init方法中加入如下设置: self.contentView ...
- iOS面试集锦3
1.写一个NSString类的实现 + (id)initWithCString:(c*****t char *)nullTerminatedCString encoding:(NSStringEnco ...
- iOS 面试集锦2
4.写一个setter方法用于完成@property (nonatomic,retain)NSString *name,写一个setter方法用于完成@property(nonatomic,copy) ...
- iOS 面试集锦
是第一篇: 1.Difference between shallow copy and deep copy? 浅复制和深复制的区别? 答案:浅层复制:只复制指向对象的指针,而不复制引用对象本身. 深层 ...
- [iOS]技巧集锦:UITableView自定义Cell中的控件无法完全对齐Cell的左边界和右边界
这是个很诡异的问题,由于一些特殊需求,我的TableView的Cell的背景色是透明,其中的控件会有背景色,第一个控件和最后一个控件我都用IB自动设了约束,对齐Cell的左边界和右边界,但是自动约束很 ...
- [iOS]技巧集锦:UICollectionView内容下沉64像素原因和解决方案
现象 UICollectionView的内容在按Home键再回到APP时,会下沉64像素. 原因 页面有NavigationBar,正好是64像素,Controller勾选了Adjust Scroll ...
- IOS 问题集锦
1._ UIWebview 拦截URL的时候:_NSCFString containsString:]: unrecognized selector sent to instance 的解决方案 NS ...
- iOS 开发问题集锦(三)
iOS 开发问题集锦(三) 介于群里大部分童鞋都是新手,为了大家能够更好的提问,并且提的问题能更好的得到回答,下面写几点提问时的注意事项: 1.认真对待你的问题,在提问题前有过认真的思考: 2.先在 ...
随机推荐
- 转: OVER() 系列函数介绍
OVER(PARTITION BY)函数介绍 开窗函数 Oracle从8.1.6开始提供分析函数,分析函数用于计算基于组的某种聚合值,它和聚合函数的不同之处是:对于每个组返 ...
- 排序算法之折半插入排序的思想以及Java实现
1 基本思想 折半插入排序(binary insertion sort)的基本原理与直接插入排序相同,不同之处在于,确定当前记录在前面有序子数组中的位置时,直接插入排序是采用顺序查找的方法,而折半插入 ...
- MySql基本使用方法
一,基本概念 1, 常用的两种引擎: (1) InnoDB a,支持ACID,简单地说就是支持事务完整性.一致性: b,支持行锁,以及类似ORACLE的一 ...
- C# -- 使用反射(Reflect)获取dll文件中的类型并调用方法
使用反射(Reflect)获取dll文件中的类型并调用方法 需引用:System.Reflection; 1. 使用反射(Reflect)获取dll文件中的类型并调用方法(入门案例) static v ...
- Git版本管理工具常用命令说明
Git常用命令 $ touch README.md 创建一个README.md文件 $ git init 创建本地仓库(repository),将会在文件夹下创建一个 .git 文件夹,.git 文 ...
- MySQL高级知识(十五)——主从复制
前言:本章主要讲解MySQL主从复制的操作步骤.由于环境限制,主机使用Windows环境,从机使用用Linux环境.另外MySQL的版本最好一致,笔者采用的MySQL5.7.22版本,具体安装过程请查 ...
- Error response from daemon: --cluster-store and --cluster-advertise daemon configurations are incompatible with swarm mode
将docker worker node加入swarm集群时,出现以下错误 1.试验环境: centos7 2.报错翻译:--cluster-store和--cluster-advertise后台配置与 ...
- CSAPP:第一章计算机系统漫游
CSAPP:计算机系统漫游 关键点:上下文.程序运行.计算机系统抽象. 信息就是位+上下文一个程序的运行过程系统的硬件组成编译系统是如何工作的?一个程序的运行过程(c语言举例)计算机系统中的抽象 信息 ...
- 计划任务执行bat
@echo offtaskkill /f /t /im ControlKJmen.exetaskkill /f /t /im KJMen.exetaskkill /f /t /im DisplayLo ...
- zabbix 应用监控作业笔记 ansible-playbook
目录 目录结构 zabbix-web.yaml zabbix-backup.yaml zabbix-nfs.yaml zabbix-mysql.yaml zabbix-server.yaml zabb ...