Objective-C之Block剖析

本文详细介绍了Block的概念、声明与使用方式,并探讨了Block的特性及其在页面间传参和延迟加载的应用。此外,还深入剖析了Block的变量截获、存储域以及如何避免循环引用等问题。

概念


Block就是一个实现的闭包(Closure),一个允许其访问常规范围之外变量的函数。

在程序语言中,闭包就是一种语法糖,它以很自然的形式,把我们的目的和我们的目的所涉及的资源全给自动打包在一起,以某种自然的、尽量不让人误解的方式让人来使用。

Block与函数指针很相似:

函数指针:  int            (*fp)        (int n)

               返回类型    变量名    参数列表

变量名:     int           (^blk)       (int n)

Block的声明、定义与使用

void (^printBlock)(NSString *s);    
printBlock = ^(NSString* str)    
{    
    NSLog(@"print:%@", str);    
};    
printBlock(@"hello world!"); 
可将Block的声明与定义放到一起:

int (^blk) (int) = ^(int addend)  
{  
    return addend+1;  
}; 
块常量:

^int (int addend)  
{  
    return addend+1;  
}
不带参数的Block常量:

^{  
    NSLog(@"hello");  
}; 
返回Block的函数:

int (^func()) (int) //第一个int表示Block的返回值,而非函数返回值  
{  
     return ^(int count) {return count + 1;};  
} 
更简明的写法:

typedef int (^blk_t) (int);  
blk_t func()  
{  
     ////////////////  
} 

Block的特性:

Block在被调用之后其实际主体(根据Block调用产生的结果进行的相关操作)才会被执行;

Block可像其他C语言类型一样使用。


注意点:

块常量与块变量的区别:

返回值方面:块变量在声明中返回值是必选项,无返回值声明则为void,块常量可省略返回值,因为可根据块表达式的主体推断出返回值类型。


块的应用:


1.在页面间传参(可实现delegate的功能)

需求:在ViewController中,点击Button,push到下一个页面NextViewController,在NextViewController的输入框TextField中输入一串字符,返回的时候,在ViewController的Label上面显示文字内容。

ViewController:

- (IBAction)btnClicked:(id)sender  
{  
    NextViewController *nextVC = [[NextViewController alloc] initWithNibName:@"NextViewController" bundle:nil];
    //定义block的主体,此时并未执行,而是在NextViewController中的按钮点击之后执行  
    nextVC.NextViewControllerBlock = ^(NSString *tfText){  
        [self resetLabel:tfText];  
    };  
    [self.navigationController pushViewController:nextVC animated:YES];  
}  
#pragma mark - NextViewControllerBlock method
- (void)resetLabel:(NSString *)textStr  
{  
    self.label.text = textStr;  
}

NextViewCOntroller:

@interface NextViewController : UIViewController
//声明block  
@property (nonatomic, copy) void (^NextViewControllerBlock)(NSString *tfText);  
@end 

- (IBAction)popBtnClicked:(id)sender {  
    if (self.NextViewControllerBlock) { 
//执行block  
self.NextViewControllerBlock(self.inputTF.text);
 }  
    [self.navigationController popViewControllerAnimated:YES];//关闭页面  
}  

2. 利用上文介绍的Block特性可实现延迟加载。


Block深度剖析


1. 截获变量值

int n = 0;  
charchar *fmt = "n = %d";  
void (^blk)(void) = ^{print(fmt, n);};  
n = 1;  
fmt = " value changed. n = %d";  
blk(); 
上述代码的执行结果为n = 0,而不是value changed. n = 1。
因为Block表达式会保存变量的当前值(的副本),这就是截获变量值。

所以,在Block里面也不能对外部的值进行修改,要想实现修改,需要外部的对变量用__block修饰。
但是C语言的变长数组(不是由常数表达式表示长度的数组)和含这种数组的C语言结构体不能被__block修饰。

像下面一样,在Block里面访问C语言数组也会出错:

const char text[] = "hello";  
void (^blk)(void) = ^{  
    printf("%c",text[0]);  
};
说明在Block里面不能引用数组的空间。解决:把 text 声明为指针类型。


2. Block存储域

存储在栈上的Block变量(及__block变量),在其作用域结束时,就会被废弃。所以要想在Block变量作用域外部使用它,必须要将其复制到堆上(堆上变量不会被自动释放)。

ARC有效时,大多数情况下,编译器会进行恰当判断,自动生成将Block变量从栈上复制到堆上。

编译器不能自动判断的情况为:向方法/函数的参数传递Block时。

以下方法不需要手动复制:

 [i] Cocoa框架的方法中带有usingBlock等时;

[ii] GCD的API。

策略:ARC有效时,手动对Block变量进行copy,不会有任何问题。


3.Block循环引用

若在Block中使用了由__strong修饰的对象,则当Block从栈复制到堆上时,该对象为Block持有,这样容易引起循环引用。

循环引用的一种情形:



循环引用的避免:


除了使用__weak解除循环引用之外,还可用__block,但必须要使Block代码执行,在Block代码中将引用的变量置为nil。

基于可靠性评估序贯蒙特卡洛模拟法的配电网可靠性评估研究(Matlab代码实现)内容概要:本文围绕“基于可靠性评估序贯蒙特卡洛模拟法的配电网可靠性评估研究”,介绍了利用Matlab代码实现配电网可靠性的仿真分析方法。重点采用序贯蒙特卡洛模拟法对配电网进行长时间段的状态抽样与统计,通过模拟系统元件的故障与修复过程,评估配电网的关键可靠性指标,如系统停电频率、停电持续时间、负荷点可靠性等。该方法能够有效处理复杂网络结构与设备时序特性,提升评估精度,适用于含分布式电源、电动汽车等新型负荷接入的现代配电网。文中提供了完整的Matlab实现代码与案例分析,便于复现和扩展应用。; 适合人群:具备电力系统基础知识和Matlab编程能力的高校研究生、科研人员及电力行业技术人员,尤其适合从事配电网规划、运行与可靠性分析相关工作的人员; 使用场景及目标:①掌握序贯蒙特卡洛模拟法在电力系统可靠性评估中的基本原理与实现流程;②学习如何通过Matlab构建配电网仿真模型并进行状态转移模拟;③应用于含新能源接入的复杂配电网可靠性定量评估与优化设计; 阅读建议:建议结合文中提供的Matlab代码逐段调试运行,理解状态抽样、故障判断、修复逻辑及指标统计的具体实现方式,同时可扩展至不同网络结构或加入更多不确定性因素进行深化研究。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值