最近大家总是问我有没有关于二维码的demo,为了满足大家的需求,特此研究了一番,希望能帮到大家!

指示根视图:

?
1
self.window.rootViewController = [[UINavigationController alloc]initWithRootViewController:[SecondViewController new ]];

生成二维码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
//  Created by 韩俊强 on 15/11/27.
//  Copyright (c) 2015年 韩俊强. All rights reserved.
//
# import "SecondViewController.h"
@interface SecondViewController ()
@property (nonatomic, strong) UITextField *tfCode;
@property (nonatomic, strong) UIButton *btnGenerate;
@property (nonatomic, strong) UIImageView *imageView;
@end
@implementation SecondViewController
- ( void )viewDidLoad {
     [ super viewDidLoad];
     CGSize windowSize = [UIScreen mainScreen].bounds.size;
     
     self.tfCode = [[UITextField alloc] initWithFrame:CGRectMake( 10 , 64 , windowSize.width- 100 , 40 )];
     [self.view addSubview:self.tfCode];
     self.tfCode.borderStyle = UITextBorderStyleRoundedRect;
     
     self.btnGenerate = [[UIButton alloc] initWithFrame:CGRectMake(windowSize.width- 100 , 64 , 90 , 40 )];
     [self.view addSubview:self.btnGenerate];
     [self.btnGenerate addTarget:self action: @selector (actionGenerate) forControlEvents:UIControlEventTouchUpInside];
     self.btnGenerate.backgroundColor = [UIColor lightGrayColor];
     [self.btnGenerate setTitle:@ "生成" forState:UIControlStateNormal];
     [self.btnGenerate setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
     
     self.imageView = [[UIImageView alloc] initWithFrame:CGRectMake( 0 , 0 , 300 , 300 )];
     [self.view addSubview:self.imageView];
     self.imageView.center = CGPointMake(windowSize.width/ 2 , windowSize.height/ 2 );
     
     self.tfCode.text = @ "http://www.baidu.com" ;
}
- ( void )actionGenerate
{
     NSString *text = self.tfCode.text;
     
     NSData *stringData = [text dataUsingEncoding: NSUTF8StringEncoding];
     
     //生成
     CIFilter *qrFilter = [CIFilter filterWithName:@ "CIQRCodeGenerator" ];
     [qrFilter setValue:stringData forKey:@ "inputMessage" ];
     [qrFilter setValue:@ "M" forKey:@ "inputCorrectionLevel" ];
     
     UIColor *onColor = [UIColor blackColor];
     UIColor *offColor = [UIColor whiteColor];
     
     //上色
     CIFilter *colorFilter = [CIFilter filterWithName:@ "CIFalseColor" keysAndValues:@ "inputImage" ,qrFilter.outputImage,@ "inputColor0" ,[CIColor colorWithCGColor:onColor.CGColor],@ "inputColor1" ,[CIColor colorWithCGColor:offColor.CGColor],nil];
     
     CIImage *qrImage = colorFilter.outputImage;
     
     //绘制
     CGSize size = CGSizeMake( 300 , 300 );
     CGImageRef cgImage = [[CIContext contextWithOptions:nil] createCGImage:qrImage fromRect:qrImage.extent];
     UIGraphicsBeginImageContext(size);
     CGContextRef context = UIGraphicsGetCurrentContext();
     CGContextSetInterpolationQuality(context, kCGInterpolationNone);
     CGContextScaleCTM(context, 1.0 , - 1.0 );
     CGContextDrawImage(context, CGContextGetClipBoundingBox(context), cgImage);
     UIImage *codeImage = UIGraphicsGetImageFromCurrentImageContext();
     UIGraphicsEndImageContext();
     
     CGImageRelease(cgImage);
     
     self.imageView.image = codeImage;
}

扫描二维码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
//  Created by 韩俊强 on 15/11/27.
//  Copyright (c) 2015年 韩俊强. All rights reserved.
//
# import "RootViewController.h"
# import
@interface RootViewController ()
@property (nonatomic, strong) UIView *scanRectView;
// 硬件设备
@property (strong, nonatomic) AVCaptureDevice            *device;
//输入设备
@property (strong, nonatomic) AVCaptureDeviceInput       *input;
//输出设备
@property (strong, nonatomic) AVCaptureMetadataOutput    *output;
//桥梁.连接输入和输出设备,
@property (strong, nonatomic) AVCaptureSession           *session;
@property (strong, nonatomic) AVCaptureVideoPreviewLayer *preview;
@end
@implementation RootViewController
- ( void )viewDidLoad {
     [ super viewDidLoad];
     CGSize windowSize = [UIScreen mainScreen].bounds.size;
     
     CGSize scanSize = CGSizeMake(windowSize.width* 3 / 4 , windowSize.width* 3 / 4 );
     CGRect scanRect = CGRectMake((windowSize.width-scanSize.width)/ 2 , (windowSize.height-scanSize.height)/ 2 , scanSize.width, scanSize.height);
     
     scanRect = CGRectMake(scanRect.origin.y/windowSize.height, scanRect.origin.x/windowSize.width, scanRect.size.height/windowSize.height,scanRect.size.width/windowSize.width);
     
     self.device = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];
     
     self.input = [AVCaptureDeviceInput deviceInputWithDevice:self.device error:nil];
     
     self.output = [[AVCaptureMetadataOutput alloc]init];
     [self.output setMetadataObjectsDelegate:self queue:dispatch_get_main_queue()];
     
     self.session = [[AVCaptureSession alloc]init];
     [self.session setSessionPreset:([UIScreen mainScreen].bounds.size.height< 500 )?AVCaptureSessionPreset640x480:AVCaptureSessionPresetHigh];
     [self.session addInput:self.input];
     [self.session addOutput:self.output];
     self.output.metadataObjectTypes=@[AVMetadataObjectTypeQRCode];
     self.output.rectOfInterest = scanRect;
     
     self.preview = [AVCaptureVideoPreviewLayer layerWithSession:self.session];
     self.preview.videoGravity = AVLayerVideoGravityResizeAspectFill;
     self.preview.frame = [UIScreen mainScreen].bounds;
     [self.view.layer insertSublayer:self.preview atIndex: 0 ];
     
     self.scanRectView = [UIView new ];
     [self.view addSubview:self.scanRectView];
     self.scanRectView.frame = CGRectMake( 0 , 0 , scanSize.width, scanSize.height);
     self.scanRectView.center = CGPointMake(CGRectGetMidX([UIScreen mainScreen].bounds), CGRectGetMidY([UIScreen mainScreen].bounds));
     self.scanRectView.layer.borderColor = [UIColor redColor].CGColor;
     self.scanRectView.layer.borderWidth = 1 ;
     
     
     //开始捕获
     [self.session startRunning];
     
}
- ( void )captureOutput:(AVCaptureOutput *)captureOutput didOutputMetadataObjects:(NSArray *)metadataObjects fromConnection:(AVCaptureConnection *)connection
{
     if ( (metadataObjects.count== 0 ) )
     {
         return ;
     }
     
     if (metadataObjects.count> 0 ) {
         
         [self.session stopRunning];
         
         AVMetadataMachineReadableCodeObject *metadataObject = metadataObjects.firstObject;
         //输出扫描字符串
         
         UIAlertView *alert = [[UIAlertView alloc] initWithTitle:metadataObject.stringValue message:@ "" delegate:self cancelButtonTitle:@ "ok" otherButtonTitles: nil];
         
         [alert show];
     }
}
- ( void )alertView:(UIAlertView *)alertView willDismissWithButtonIndex:(NSInteger)buttonIndex
{
     [self.session startRunning];
}</avcapturemetadataoutputobjectsdelegate,uialertviewdelegate></avfoundation>

最终效果:(由于扫描二维码无法展示效果,所以自己动手真机测试吧!)

IOS7之前,开发者进行扫码编程时,一般会借助第三方库。常用的是ZBarSDK,IOS7之后,系统的AVMetadataObject类中,为我们提供了解析二维码的接口。经过测试,使用原生API扫描和处理的效率非常高,远远高于第三方库。

一、使用方法示例

官方提供的接口非常简单,代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
@interface ViewController () //用于处理采集信息的代理
{
AVCaptureSession * session; //输入输出的中间桥梁
}
@end
@implementation ViewController
- ( void )viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
//获取摄像设备
AVCaptureDevice * device = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];
//创建输入流
AVCaptureDeviceInput * input = [AVCaptureDeviceInput deviceInputWithDevice:device error:nil];
//创建输出流
AVCaptureMetadataOutput * output = [[AVCaptureMetadataOutput alloc]init];
//设置代理 在主线程里刷新
[output setMetadataObjectsDelegate:self queue:dispatch_get_main_queue()];
//初始化链接对象
session = [[AVCaptureSession alloc]init];
//高质量采集率
[session setSessionPreset:AVCaptureSessionPresetHigh];
[session addInput:input];
[session addOutput:output];
//设置扫码支持的编码格式(如下设置条形码和二维码兼容)
output.metadataObjectTypes=@[AVMetadataObjectTypeQRCode,AVMetadataObjectTypeEAN13Code, AVMetadataObjectTypeEAN8Code, AVMetadataObjectTypeCode128Code];
AVCaptureVideoPreviewLayer * layer = [AVCaptureVideoPreviewLayer layerWithSession:session];
layer.videoGravity=AVLayerVideoGravityResizeAspectFill;
layer.frame=self.view.layer.bounds;
[self.view.layer insertSublayer:layer atIndex:0];
//开始捕获
[session startRunning];
}

之后我们的UI上已经可以看到摄像头捕获的内容,只要实现代理中的方法,就可以完成二维码条形码的扫描:

?
1
2
3
4
5
6
7
8
-( void )captureOutput:(AVCaptureOutput *)captureOutput didOutputMetadataObjects:(NSArray *)metadataObjects fromConnection:(AVCaptureConnection *)connection{
if (metadataObjects.count>0) {
//[session stopRunning];
AVMetadataMachineReadableCodeObject * metadataObject = [metadataObjects objectAtIndex : 0 ];
//输出扫描字符串
NSLog(@ "%@" ,metadataObject.stringValue);
}
}

二、一些优化

通过上面的代码测试,我们可以发现系统的解析处理效率是相当的高,IOS官方提供的API也确实非常强大,然而,我们可以做进一步的优化,将效率更加提高:

首先,AVCaptureMetadataOutput类中有一个这样的属性(在IOS7.0之后可用):

@property(nonatomic) CGRect rectOfInterest;

这个属性大致意思就是告诉系统它需要注意的区域,大部分APP的扫码UI中都会有一个框,提醒你将条形码放入那个区域,这个属性的作用就在这里,它可以设置一个范围,只处理在这个范围内捕获到的图像的信息。如此一来,可想而知,我们代码的效率又会得到很大的提高,在使用这个属性的时候。需要几点注意:

1、这个CGRect参数和普通的Rect范围不太一样,它的四个值的范围都是0-1,表示比例。

2、经过测试发现,这个参数里面的x对应的恰恰是距离左上角的垂直距离,y对应的是距离左上角的水平距离。

3、宽度和高度设置的情况也是类似。

3、举个例子如果我们想让扫描的处理区域是屏幕的下半部分,我们这样设置

?
1
output.rectOfInterest=CGRectMake(0.5,0,0.5, 1);

具体apple为什么要设计成这样,或者是这个参数我的用法那里不对,还需要了解的朋友给个指导。

转自:http://www.2cto.com/kf/201601/456033.html

另外有一个支持64为编译的二维码生成库下载地址:http://download.csdn.net/download/u011148716/8315981

如果只是生成的話,see
https://github.com/moky/slanissue-ios/blob/master/uikit_ext/S9QRCodeImage.h

用法:

UIImage *qrImage = [UIImage imageWithQRCode:@"你的文字" size:256];

iOS中 扫描二维码/生成二维码详解相关推荐

  1. ffmpeg转码生成的m3u8格式详解

    目录 一.ffmpeg转码生成的单码率和多码率的文件列表 二.Master Playlist主列表(name.m3u8) 1. 单码率(video_single_bitrate.m3u8 ) 2. 多 ...

  2. iOS中 视频直播功能-流媒体的使用(详解)韩俊强的CSDN博客

    上一篇博客:(流媒体实现视频播放和下载功能):http://blog.csdn.net/qq_31810357/article/details/50574914 最近视频直播功能比较火,处于需求,研究 ...

  3. python可以使用二维元组吗_python中读入二维csv格式的表格方法详解(以元组/列表形式表示)...

    怎么去读取一个没有表头的二维csv文件(如下图所示)? 并以元组的形式表现数据: ((1.0, 0.0, 3.0, 180.0), (2.0, 0.0, 2.0, 180.0), (3.0, 0.0, ...

  4. python二维元组_python中读入二维csv格式的表格方法详解(以元组/列表形式表示)

    如何去读取一个没有表头的二维csv文件(如下图所示)? 并以元组的形式表现数据: ((1.0, 0.0, 3.0, 180.0), (2.0, 0.0, 2.0, 180.0), (3.0, 0.0, ...

  5. python元组读取到列表_python中读入二维csv格式的表格方法详解(以元组/列表形式表示)...

    如何去读取一个没有表头的二维csv文件(如下图所示)? 并以元组的形式表现数据: ((1.0, 0.0, 3.0, 180.0), (2.0, 0.0, 2.0, 180.0), (3.0, 0.0, ...

  6. python二元多次函数拟合_对python实现二维函数高次拟合的示例详解

    在参加"数据挖掘"比赛中遇到了关于函数高次拟合的问题,然后就整理了一下源码,以便后期的学习与改进. 在本次"数据挖掘"比赛中感觉收获最大的还是对于神经网络的认识 ...

  7. python画二维散点图-基于python 二维数组及画图的实例详解

    1.二维数组取值 注:不管是二维数组,还是一维数组,数组里的数据类型要一模一样,即若是数值型,全为数值型 #二维数组 import numpy as np list1=[[1.73,1.68,1.71 ...

  8. python画二维数组散点图_基于python二维数组及画图的实例详解

    基于python二维数组及画图的实例详解 下面小编就为大家分享一篇基于python 二维数组及画图的实例详解,具有很好的参考价值,希望对大家有所帮助.一起跟随小编过来看看吧 1.二维数组取值 注:不管 ...

  9. 【H264/AVC 句法和语义详解】(二):h264码流格式与NALU详解一

    上一篇中,我们站在句法元素(或称语法元素)的角度,介绍了H.264的句法和语义,和句法元素的分层结构.在这篇中,我们更进一步,从比特的角度出发,来探索h264码流的组成.通过这篇的学习,我们会初步具备 ...

  10. 深度学习之图像分类(二十八)-- Sparse-MLP(MoE)网络详解

    深度学习之图像分类(二十八)Sparse-MLP(MoE)网络详解 目录 深度学习之图像分类(二十八)Sparse-MLP(MoE)网络详解 1. 前言 2. Mixture of Experts 2 ...

最新文章

  1. 程序分析的一些概念总结
  2. 乐高小颗粒履带机器人_玩转乐高大颗粒积木(四十二)——乐高不倒翁、轮式是挖掘机、大颗粒皮筋车...
  3. spring aop execution表达式
  4. Intellij IDEA中安装使用PlantUML画时序图、类图等
  5. html画布 缩放的正方形,html5-canvas – 在动画HTML5画布中缩放和平移
  6. angularJS的$http.post请求,.net后台接收不到参数值的解决方案
  7. 两种极端情况的案例:N+1次查询和笛卡尔积
  8. python爬取去哪网数据_python最强的代理池,突破IP的封锁爬取海量数据(送项目源码)...
  9. Python机器学习:决策树003使用信息熵寻找最优划分
  10. 《规范敏捷交付:企业级敏捷软件交付的方法与实践》——3.11 观点总结
  11. 操作系统—基本知识(细致版)
  12. javascript动画系列 —— 切换图片(原生)
  13. C++中的cin cout
  14. Linux上查到目前透过FTP进来的使用者
  15. 马尔科夫决策过程(MDP) : BlackJack (MC-Off Policy)
  16. SEM测试样品减薄及表面复型
  17. c语言main的作用是什么,c语言main是什么意思-与非网
  18. Traffic Shifting
  19. PHP微信公众号,获取用户地理位置 定位 经纬度
  20. Hello, Fedora.

热门文章

  1. CP 15 协处理器
  2. 如何用海思HI3516DV300/3518EV200推流H.264
  3. VUE强制渲染,强制更新
  4. prometheus +granfana监控告警
  5. 运行中的程序突然出现Killed原因
  6. uchar 和 uchar3的疑惑
  7. 前端HTML------负责网页结构
  8. 安装windows与Ubuntu双系统,并使用GRUB启动引导器
  9. C#:实体框架EF(entity framework)
  10. cufflinks suit