热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

【转】blocks实践指南

转自:http:blog.sina.com.cnsblog_67419c420100vmkt.html 1.为什么要使用blocks将一个blocks作为函数或者方法的参数传递,可

转自:http://blog.sina.com.cn/s/blog_67419c420100vmkt.html

 

1.为什么要使用blocks

将一个blocks作为函数或者方法的参数传递,可以使函数或者方法在恰当的点实现回调。以各种NSNotification类为例子,传统的方法是一个对象添加自己作为此通知的观察者,实现处理接到通知应该调用的方法(通过addObserver....)。

如下:



































- (void)viewDidLoad {

[super viewDidLoad];

[[NSNotificationCenter defaultCenter] addObserver:self

selector:@selector(keyboardWillShow:)

name:UIKeyboardWillShowNotification object:nil];

}


- (void)keyboardWillShow:(NSNotification *)notification {

// Notification-handling code goes here.

}

然而通过方法 addObserverForName:object:queue:usingBlock: 我们也可以实现对通知的监控。

实现如下:


























- (void)viewDidLoad {

[super viewDidLoad];

[[NSNotificationCenter defaultCenter] addObserverForName:UIKeyboardWillShowNotification

object:nil queue:[NSOperationQueue mainQueue] usingBlock:^(NSNotification *notif) {

// Notification-handling code goes here.

}];

}

2.系统框架API中的blocks

另一个使用blocks的明显动机就是系统框架中提供了很多方法和函数用blocks作为参数。一般都存在于:

completion handlers,notification handlers,error handlers,enumeration,view animation and transitions,sorting 中,下面讲详细介绍。

(1)completion and Error handlers

     completion handlers是回调,其允许当框架方法或者函数完成任务时,在进行一些处理。多数情况下,用completion handler去free state 或者更新用户界面。很多框架方法让我们用blocks实现completion handlers,而不是代理方法或者notification handlers。

    UIView类有很多关于animation和view transitions的方法有completion handler block 参数。下面是animateWithDuration:animations:completion: 方法的一个实现例子。

- (IBAction)animateView:(id)sender { CGRect cacheFrame = self.imageView.frame; [UIView animateWithDuration:1.5 animations:^{ CGRect newFrame = self.imageView.frame; newFrame.origin.y = newFrame.origin.y + 150.0; self.imageView.frame = newFrame; self.imageView.alpha = 0.2; } completion:^ (BOOL finished) { if (finished) { // Revert image view to original. sleep(3); self.imageView.frame = cacheFrame; self.imageView.alpha = 1.0; } }]; }

一些框架方法有error handlers,和completion handlers很相似。这些方法当其因为一些错误条件不能完成自己的任务的时候,调用这些error handlers blocks。我们典型的实现这些error handlers去通知用户出现错误。

(2)Notification handlers

   NSNotificationCenter的addObserverForName:object:queue:usingBlock:方法让我们去实现一个通知的handler去建立一个observation。下面是一个例子:
































- (void)applicationDidFinishLaunching:(NSNotification *)aNotification {

opQ = [[NSOperationQueue alloc] init];

[[NSNotificationCenter defaultCenter] addObserverForName:@"CustomOperationCompleted"

object:nil queue:opQ

usingBlock:^(NSNotification *notif) {

NSNumber *theNum = [notif.userInfo objectForKey:@"NumberOfItemsProcessed"];

NSLog(@"Number of items processed: %i", [theNum intValue]);

}];

}

上面的方法要求传入一个NSOperationQueue实例,因此应用程序可以指定执行block handler 的execution context。

(3)Enumeration

   Foundation framework的集合类(NSArray ,NSDictionary, NSSet 和 NSIndexSet)声明了一些方法,可以用blocks来枚举整个集合,其实等价与下代码段:














for (id item in collection) {

// Code to operate on each item in turn.

}

    大概有两种形式。第一种方法名字以enumerate开始,没有返回值。

    第二种方法 block参数之前的名字为passingTest,这种方法返回整数或者一个NSIndexSet对象。这种方法的block一般是对迭代的每一个元素进行一个test,当test通过时返回YES。返回整数或者一个标识通过test的对象的集合。

   下面的代码对NSArray分别做了实例。一个方法的block返回数组中每个string有特定前缀的集合。


















































NSString *area = @"Europe";

NSArray *timeZOneNames= [NSTimeZone knownTimeZoneNames];

NSMutableArray *areaArray = [NSMutableArray arrayWithCapacity:1];

NSIndexSet *areaIndexes = [timeZoneNames indexesOfObjectsWithOptions:NSEnumerationConcurrent

passingTest:^(id obj, NSUInteger idx, BOOL *stop) {

NSString *tmpStr = (NSString *)obj;

return [tmpStr hasPrefix:area];

}];


NSArray *tmpArray = [timeZoneNames objectsAtIndexes:areaIndexes];

[tmpArray enumerateObjectsWithOptions:NSEnumerationConcurrent|NSEnumerationReverse

usingBlock:^(id obj, NSUInteger idx, BOOL *stop) {

[areaArray addObject:[obj substringFromIndex:[area length]+1]];

}];

NSLog(@"Cities in %@ time zone:%@", area, areaArray);

    尽管NSString类不属于集合,但是它也有两个有block参数的方法,为enumerateSubstringsInRange:options:usingBlock:  enumerateLinesUsingBlock:。第一个方法的枚举单元可以是行,段,词,句,等等,但是第二个方法的就只是行。

下面的代码是使用block去匹配substring。









































NSString *musician = @"Beatles";

NSString *musicDates = [NSString stringWithContentsOfFile:

@"/usr/share/calendar/calendar.music"

encoding:NSASCIIStringEncoding error:NULL];

[musicDates enumerateSubstringsInRange:NSMakeRange(0, [musicDates length]-1)

options:NSStringEnumerationByLines

usingBlock:^(NSString *substring, NSRange substringRange, NSRange enclosingRange, BOOL *stop) {

NSRange found = [substring rangeOfString:musician];

if (found.location != NSNotFound) {

NSLog(@"%@", substring);

}

}];

(4)view animation 和 transitions

ios4.0之后UIView类引入了一些用于动画和view transitions的带block参数的方法。block参数分两类:

    block用于改变view的属性以完成动画。

    completion handlers。

下面的代码讲述了animateWithDuration:animations:completion:方法的用法。




















[UIView animateWithDuration:0.2 animations:^{

view.alpha = 0.0;

} completion:^(BOOL finished){

[view removeFromSuperview];

}];

这个动画就是让view逐渐消失,然后从父视图上移除。

  方法transitionWithView:duration:options:animations:completion:用于两个view的transitions,包括flips和curls。

下面的代码段实现了两个views 的flip transitions。























[UIView transitionWithView:containerView duration:0.2

options:UIViewAnimationOptionTransitionFlipFromLeft animations:^{

[fromView removeFromSuperview];

[containerView addSubview:toView]

}

completion:NULL];

(5)sorting

foundation framework声明了NSComparator类型用于比较两个元素。

typedef NSComparisonResult (^NSComparator)(id obj1, id obj2);

NSComparator是一个block类型,用于两个对象的比较,然后返回一个NSComparisonResult值。下面是一个例子:















































NSArray *stringsArray = [NSArray arrayWithObjects:

@"string 1",

@"String 21",

@"string 12",

@"String 11",

@"String 02", nil];

static NSStringCompareOptions comparisOnOptions= NSCaseInsensitiveSearch | NSNumericSearch |

NSWidthInsensitiveSearch | NSForcedOrderingSearch;

NSLocale *currentLocale = [NSLocale currentLocale];

NSComparator finderSort = ^(id string1, id string2) {

NSRange string1Range = NSMakeRange(0, [string1 length]);

return [string1 compare:string2 options:comparisonOptions range:string1Range locale:currentLocale];

};

NSLog(@"finderSort: %@", [stringsArray sortedArrayUsingComparator:finderSort]);

3.block 和concurrency

   block封装了一段代码,可以异步执行。基于这一事实,块可以用在GCD和NSOperationQueue上面。

   GCD函数dispatch_sync和函数dispatch_async的第二个参数就是用block。 



推荐阅读
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • CentOS7源码编译安装MySQL5.6
    2019独角兽企业重金招聘Python工程师标准一、先在cmake官网下个最新的cmake源码包cmake官网:https:www.cmake.org如此时最新 ... [详细]
  • 本文深入探讨了 Java 中的 Serializable 接口,解释了其实现机制、用途及注意事项,帮助开发者更好地理解和使用序列化功能。 ... [详细]
  • DNN Community 和 Professional 版本的主要差异
    本文详细解析了 DotNetNuke (DNN) 的两种主要版本:Community 和 Professional。通过对比两者的功能和附加组件,帮助用户选择最适合其需求的版本。 ... [详细]
  • 本文详细介绍了如何构建一个高效的UI管理系统,集中处理UI页面的打开、关闭、层级管理和页面跳转等问题。通过UIManager统一管理外部切换逻辑,实现功能逻辑分散化和代码复用,支持多人协作开发。 ... [详细]
  • 作为一名新手,您可能会在初次尝试使用Eclipse进行Struts开发时遇到一些挑战。本文将为您提供详细的指导和解决方案,帮助您克服常见的配置和操作难题。 ... [详细]
  • Explore how Matterverse is redefining the metaverse experience, creating immersive and meaningful virtual environments that foster genuine connections and economic opportunities. ... [详细]
  • 主要用了2个类来实现的,话不多说,直接看运行结果,然后在奉上源代码1.Index.javaimportjava.awt.Color;im ... [详细]
  • 本文详细介绍了 Dockerfile 的编写方法及其在网络配置中的应用,涵盖基础指令、镜像构建与发布流程,并深入探讨了 Docker 的默认网络、容器互联及自定义网络的实现。 ... [详细]
  • 使用 Azure Service Principal 和 Microsoft Graph API 获取 AAD 用户列表
    本文介绍了一段通用代码示例,该代码不仅能够操作 Azure Active Directory (AAD),还可以通过 Azure Service Principal 的授权访问和管理 Azure 订阅资源。Azure 的架构可以分为两个层级:AAD 和 Subscription。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
author-avatar
壹花壹浄土
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有