1:block的循环引用问题最主要记住两点:

如果【block内部】使用【外部声明的强引用】访问【对象A】, 那么【block内部】会自动产生一个【强引用】指向【对象A】

如果【block内部】使用【外部声明的弱引用】访问【对象A】, 那么【block内部】会自动产生一个【弱引用】指向【对象A】

2:

#import "ViewController.h"
#import "XMGPerson.h" @interface ViewController ()
@property (nonatomic, copy) int (^sumBlock)(int, int);
@property (nonatomic, assign) int a;
@end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad]; [self test5];
/*
如果【block内部】使用【外部声明的强引用】访问【对象A】, 那么【block内部】会自动产生一个【强引用】指向【对象A】
如果【block内部】使用【外部声明的弱引用】访问【对象A】, 那么【block内部】会自动产生一个【弱引用】指向【对象A】
*/
} /**
* 不可行
*/
- (void)test6
{
XMGPerson *p = [[XMGPerson alloc] init]; __weak XMGPerson *weakP = p; p.name = @"Jack"; p.block = ^{ // block1
XMGPerson *strongP = weakP;
NSLog(@"block1 -- %@", strongP.name); dispatch_time_t when = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3.0 * NSEC_PER_SEC));
dispatch_after(when, dispatch_get_main_queue(), ^{ // block2
NSLog(@"block2 -- %@", weakP.name);
});
}; p.block();
} /**
* 可行
*/
- (void)test5
{
XMGPerson *p = [[XMGPerson alloc] init]; __weak XMGPerson *weakP = p; p.name = @"Jack"; p.block = ^{ // block1
NSLog(@"beign-------");
XMGPerson *strongP = weakP; dispatch_time_t when = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3.0 * NSEC_PER_SEC));
dispatch_after(when, dispatch_get_main_queue(), ^{ // block2 // NSLog(@"block2 -- %@", weakP.name);
NSLog(@"block2 -- %@", strongP.name); dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ // block3
NSLog(@"block3 -- %@", strongP.name);
}); });
}; p.block();
} /**
* 不可行
*/
- (void)test4
{
XMGPerson *p = [[XMGPerson alloc] init]; p.name = @"Jack"; p.block = ^{ // block1
NSLog(@"beign-------"); dispatch_time_t when = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3.0 * NSEC_PER_SEC));
dispatch_after(when, dispatch_get_main_queue(), ^{ // block2
NSLog(@"after-----------%@", p.name);
});
}; p.block();
} /**
* 不可行
*/
- (void)test3
{
XMGPerson *p = [[XMGPerson alloc] init];
__weak XMGPerson *weakP = p; p.name = @"Jack"; p.block = ^{ // block1
NSLog(@"beign-------");
XMGPerson *strongP = weakP; dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3.0 * NSEC_PER_SEC)), dispatch_get_main_queue(),
^{ // block2
NSLog(@"after-----------%@", strongP.name);
});
}; p.block();
} /**
* 可行
*/
- (void)test2
{
XMGPerson *p = [[XMGPerson alloc] init]; __weak XMGPerson *weakP = p; p.name = @"Jack";
p.block = ^{ // block1
NSLog(@"beign-------"); XMGPerson *strongP = weakP; dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3.0 * NSEC_PER_SEC)), dispatch_get_main_queue(),
^{ // block2
NSLog(@"after-----------%@", strongP.name);
});
}; p.block();
} - (void)test1
{
XMGPerson *p = [[XMGPerson alloc] init]; __weak XMGPerson *weakP = p; p.name = @"Jack";
p.block = ^{
NSLog(@"-----------%@", weakP.name);
};
} #pragma mark - 其他
- (void)test:(int (^)(int, int))sumBlock
{ } - (void)run:(int)a
{ } - (void)testBlock
{
[self test:^(int a, int b){
return a + b;
}]; void (^block)() = ^{
NSLog(@"-------");
};
block(); int (^sumBlock)(int, int) = ^(int num1, int num2){
return num1 + num2;
};
sumBlock(, ); int a = ; /*
返回值类型 (^block的变量名)(形参类型列表) = ^(形参列表) {
// block代码
};
block的变量名(实参列表);
*/
} @end

总结:1:block的定义:1:block以属性定义的时候,用copy修饰,

@property (nonatomic, copy) void (^block)();无参数无返回值,因为block默认是在堆里,需要将堆中的block,copy到栈中才能使用  2:block的非属性定义:1:无参数无返回值:

其中 void (^block)() 表示声明一个名称为block的block,block的类型表示为:void (^)(),右边的为block任务的代码块,
    block();表示调用block,右边为左边的block赋值
  void (^block)() =  ^{
NSLog(@"-------");
};
block();

2:有参数有返回值的block:左边int (^sumBlock)(int, int)定义一个有参数有返回值的block:sumBlock,而右边的block给左边的block赋值,写法:参数写在^(int a,int b)其中ab为block的参数,返回值写在block的代码块中。
 sumBlock(10, 10);调用block,并未block传递参数
 ^(int num1, int num2){
return num1 + num2;
};
 
  int (^sumBlock)(int, int) = ^(int num1, int num2){
return num1 + num2;
};
sumBlock(10, 10);
3:block作为参数传递:1: block作为参数传递 - (void)test:(int (^)(int, int))sumBlock,实现该方法,调用block传递参数 2:在外部调用test方法,其中ab就为传递的参数,

[self test:^(int a, int b){

NSLog(@"%d",a+b);

return a + b;

}];

- (void)test:(int (^)(int, int))sumBlock

{

sumBlock(2,1);

}

4:1:block的循环引用:XMGPerson alloc init 之后在内存中产生一个对象,name 和 block是该对象的属性(因为用copy修饰,也就相当于强引用),则该对象会对这两个属性分别有一个强引用,创建完对象之后,有一个强指针p指向该对象(指针里存放的是对象的地址,变量创建之后无论是全局变量还是局部变量都会有一个强引用),__weak那部分代码是将强指针p的地址复制给一个弱指针weakSelf,此时弱指针weakSelf会有一个弱引用指向person对象,p.name = @"JACK";此时name属性会对jack有一个强引用,block会对赋值的block有一个强引用。此时不会立即调用block代码块中的任务,但是此时会检测block代码块中有无引用外部变量(即使是外部无引用,在after中引用了,也算block中引用了外部变量,此时会对访问的该对象有一个强引用或是弱引用,代码块中有引用weakP,所以会产生一个弱引用),再执行代码调用block,执行block代码块中的任务,将一个弱引用weakP指针赋值给一个强引用,此时block代码块中的strongP会对person对象有一个强引用,在执行gcd延迟函数,此时会产生延迟函数的block块,会检测延迟函数中有无引用外界的变量,有引用了strongP,则会产生一个强引用指向person对象,此时的代码块不立即执行(但是系统会有一个强指针指向该代码块,如图2),此时会执行到方法结束,强指针p销毁,弱指针weakp销毁,strongP执行完销毁,但此时block2还指向person对象,所以person对象不会销毁,能打印出name  2:像类似于GCD的延迟函数在block中,想要保住对象使其不备销毁,就在block中定义一个强引用指向就可以了


 

iOS开发Block的介绍以及Block的循环引用问题的更多相关文章

  1. iOS开发UI篇—无限轮播(循环利用)

    iOS开发UI篇—无限轮播(循环利用) 一.无限轮播  1.简单说明 在开发中常需要对广告或者是一些图片进行自动的轮播,也就是所谓的无限滚动. 在开发的时候,我们通常的做法是使用一个UIScrollV ...

  2. iOS开发UI篇—无限轮播(循环展示)

    iOS开发UI篇—无限轮播(循环展示) 一.简单说明 之前的程序还存在一个问题,那就是不能循环展示,因为plist文件中只有五个数组,因此第一个和最后一个之后就没有了,下面介绍处理这种循环展示问题的小 ...

  3. iOS沙盒机制介绍,Block 的介绍

    一.iOS沙盒机制介绍 (转载) 1)概念:每个ios应用都有自己的应用沙盒,应用沙盒就是文件系统目录,与其他应用放入文件 系统隔离,ios系统不允许访问 其他应用的应用沙盒,但在ios8中已经开放访 ...

  4. iOS开发——OC基础-ARC、BLOCK、协议

    一.ARC ARC 是一种编译器特性!而不是IOS运行时特性,和JAVA中得垃圾回收机制完全不一样ARC是自iOS 5之后增加的新特性,完全消除了手动管理内存的烦琐,编译器会自动在适当的地方插入适当的 ...

  5. ios开发--GCD使用介绍:4-延迟执行操作

    在开发过程中,我们有时会希望把一些操作封装起来延迟一段时间后再执行.iOS开发中,有两种常用的方法可以实现延迟执行,一种是使用GCD,另外一种是使用NSRunLoop类中提供的方法. 1.使用GCD实 ...

  6. [翻译] iOS开发工具的介绍(第一部分)

    IOS DEVELOPMENT TIPS & TRICKS - PART I http://blog.trifork.com/2013/12/19/ios-development-tips-t ...

  7. ios开发杂项(基础性介绍等)

    IOS Xcode开发中的文件后缀名区别m,mm,cpp,h .h :头文件.头文件包含类,类型,函数和常数的声明. .m :源代码文件.这是典型的源代码文件扩展名,可以包含Objective-C和C ...

  8. iOS开发-UINavigationController简单介绍

    导航条或者说导航栏目现在在App中基本上也算是标配,类似于父子级别的味道在里面,UINavigationController就是负责简化这一实现功能的,属于iOS开发中比较常用的一种容器View co ...

  9. iOS Block详细介绍(block实现)

    Block的实现 数据结构定义 block的数据结构定义如下图 对应的结构体定义如下: struct Block_descriptor { unsigned long int reserved; un ...

随机推荐

  1. VNC Server模拟攻击实战

    VNC目前已被广泛应用的一个远程控制程序,很多攻击者对VNC的攻击技术研究热情是高涨的,丝毫不亚于对Windows的远程桌面(3389).PcAnywhere的攻击研究.从最开始爆发出来的VNC的低版 ...

  2. Android 用Socket实现PC和手机的文件传输

    PC服务器端代码: /* * PC与<a href="http://lib.csdn.net/base/android" class='replace_word' title ...

  3. BZOJ3626: [LNOI2014]LCA(树链剖分+线段树)

    Description 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深度定义为这个节点到根的距离+1.设dep[i]表示点i的深度,LCA(i,j)表示i与j的最近公共祖先.有q ...

  4. hibernate中的事务管理是怎么概念?

    1.JDBC事务 JDBC 事务是用 Connection 对象控制的.JDBC Connection 接口( java.sql.Connection )提供了两种事务模式:自动提交和手工提交. ja ...

  5. hysbz 2243 染色(树链剖分)

    题目链接:hysbz 2243 染色 题目大意:略. 解题思路:树链剖分+线段树的区间合并,可是区间合并比較简单,节点仅仅要记录左右端点的颜色就可以. #include <cstdio> ...

  6. php设置http请求头信息和响应头信息

    php设置http请求头信息和响应头信息 设置请求服务器的头信息可以用fsockopen,curl组件,header函数只能用来设置客户端响应的头信息,不能设置服务器的头信息. 例子;  一.head ...

  7. java list 容器的ConcurrentModificationException

    java中的很多容器在遍历的同时进行修改里面的元素都会ConcurrentModificationException,包括多线程情况和单线程的情况.多线程的情况就用说了,单线程出现这个异常一般是遍历( ...

  8. 6.在单机上实现ZooKeeper伪机群/伪集群部署

    转自:https://blog.csdn.net/poechant/article/details/6633923

  9. 1.3 Python基础知识 - 用户交互及传递参数

    一.用户交互 用户交互方面,每种开发语言都有不同的方式,例如shell语言用的是,“read -p "What is  your name ? " ”.python中是什么样子的呢 ...

  10. (转)linux screen 命令详解

    转自:http://www.cnblogs.com/mchina/archive/2013/01/30/2880680.html 一.背景 系统管理员经常需要SSH 或者telent 远程登录到Lin ...