iOS中 扫描二维码/生成二维码详解
最近大家总是问我有没有关于二维码的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中 扫描二维码/生成二维码详解相关推荐
- ffmpeg转码生成的m3u8格式详解
目录 一.ffmpeg转码生成的单码率和多码率的文件列表 二.Master Playlist主列表(name.m3u8) 1. 单码率(video_single_bitrate.m3u8 ) 2. 多 ...
- iOS中 视频直播功能-流媒体的使用(详解)韩俊强的CSDN博客
上一篇博客:(流媒体实现视频播放和下载功能):http://blog.csdn.net/qq_31810357/article/details/50574914 最近视频直播功能比较火,处于需求,研究 ...
- python可以使用二维元组吗_python中读入二维csv格式的表格方法详解(以元组/列表形式表示)...
怎么去读取一个没有表头的二维csv文件(如下图所示)? 并以元组的形式表现数据: ((1.0, 0.0, 3.0, 180.0), (2.0, 0.0, 2.0, 180.0), (3.0, 0.0, ...
- python二维元组_python中读入二维csv格式的表格方法详解(以元组/列表形式表示)
如何去读取一个没有表头的二维csv文件(如下图所示)? 并以元组的形式表现数据: ((1.0, 0.0, 3.0, 180.0), (2.0, 0.0, 2.0, 180.0), (3.0, 0.0, ...
- python元组读取到列表_python中读入二维csv格式的表格方法详解(以元组/列表形式表示)...
如何去读取一个没有表头的二维csv文件(如下图所示)? 并以元组的形式表现数据: ((1.0, 0.0, 3.0, 180.0), (2.0, 0.0, 2.0, 180.0), (3.0, 0.0, ...
- python二元多次函数拟合_对python实现二维函数高次拟合的示例详解
在参加"数据挖掘"比赛中遇到了关于函数高次拟合的问题,然后就整理了一下源码,以便后期的学习与改进. 在本次"数据挖掘"比赛中感觉收获最大的还是对于神经网络的认识 ...
- python画二维散点图-基于python 二维数组及画图的实例详解
1.二维数组取值 注:不管是二维数组,还是一维数组,数组里的数据类型要一模一样,即若是数值型,全为数值型 #二维数组 import numpy as np list1=[[1.73,1.68,1.71 ...
- python画二维数组散点图_基于python二维数组及画图的实例详解
基于python二维数组及画图的实例详解 下面小编就为大家分享一篇基于python 二维数组及画图的实例详解,具有很好的参考价值,希望对大家有所帮助.一起跟随小编过来看看吧 1.二维数组取值 注:不管 ...
- 【H264/AVC 句法和语义详解】(二):h264码流格式与NALU详解一
上一篇中,我们站在句法元素(或称语法元素)的角度,介绍了H.264的句法和语义,和句法元素的分层结构.在这篇中,我们更进一步,从比特的角度出发,来探索h264码流的组成.通过这篇的学习,我们会初步具备 ...
- 深度学习之图像分类(二十八)-- Sparse-MLP(MoE)网络详解
深度学习之图像分类(二十八)Sparse-MLP(MoE)网络详解 目录 深度学习之图像分类(二十八)Sparse-MLP(MoE)网络详解 1. 前言 2. Mixture of Experts 2 ...
最新文章
- 程序分析的一些概念总结
- 乐高小颗粒履带机器人_玩转乐高大颗粒积木(四十二)——乐高不倒翁、轮式是挖掘机、大颗粒皮筋车...
- spring aop execution表达式
- Intellij IDEA中安装使用PlantUML画时序图、类图等
- html画布 缩放的正方形,html5-canvas – 在动画HTML5画布中缩放和平移
- angularJS的$http.post请求,.net后台接收不到参数值的解决方案
- 两种极端情况的案例:N+1次查询和笛卡尔积
- python爬取去哪网数据_python最强的代理池,突破IP的封锁爬取海量数据(送项目源码)...
- Python机器学习:决策树003使用信息熵寻找最优划分
- 《规范敏捷交付:企业级敏捷软件交付的方法与实践》——3.11 观点总结
- 操作系统—基本知识(细致版)
- javascript动画系列 —— 切换图片(原生)
- C++中的cin cout
- Linux上查到目前透过FTP进来的使用者
- 马尔科夫决策过程(MDP) : BlackJack (MC-Off Policy)
- SEM测试样品减薄及表面复型
- c语言main的作用是什么,c语言main是什么意思-与非网
- Traffic Shifting
- PHP微信公众号,获取用户地理位置 定位 经纬度
- Hello, Fedora.