成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

block 解析 - 內存

移動開發 iOS
block結構體相應的也有一個成員引用,這樣會增加對局部變量的 _para1引用,在Block銷毀的時候引用就釋放掉了

我們了解到了用__block修飾的變量,可以在block內部修改,__block變量其實對應一個結構體

  1. struct __Block_byref__para1_0 { 
  2.   void *__isa; 
  3. __Block_byref__para1_0 *__forwarding; 
  4.  int __flags; 
  5.  int __size; 
  6.  char *_para1; 
  7. }; 

block結構體相應的也有一個成員引用,這樣會增加對局部變量的 _para1引用,在Block銷毀的時候引用就釋放掉了

  1. struct __main1_block_impl_0 { 
  2.   struct __block_impl impl; 
  3.   struct __main1_block_desc_0* Desc; 
  4.   __Block_byref__para1_0 *_para1; // by ref 

block內部 成員變量 使用,可以在block內部修改,對應一個、多個成員變量參數,block結構體內部有一個成員引用

  1. struct __KDBlockTest__test3_block_impl_0 { 
  2.   struct __block_impl impl; 
  3.   struct __KDBlockTest__test3_block_desc_0* Desc; 
  4.   KDBlockTest *self; 

這樣會增加對self的引用,在Block銷毀的時候引用就釋放掉了

循環引用

在使用的時候需要注意循環引用,即某個對象有一個copy或者strong的 block成員屬性,這時block內部直接引用了 成員變量(全局變量) 或者self,這樣就產生了self持有 block成員,block成員又持有了self,就會導致循環引用。

我們看以下代碼(ARC):

  1. typedef void(^ActionTest)(void); 
  2. @interface KDBlockTest() 
  3.     __block NSString *_person2; 
  4.     ActionTest _action; 
  5. }
  1. @implementation KDBlockTest 
  2. #pragma mark - system 
  3. -(instancetype)init 
  4.     self=[super init]; 
  5.     if(self) 
  6.     { 
  7.         [self test3]; 
  8.     } 
  9.     return self; 
  10. -(void)dealloc 
  11.     NSLog(@"KDBlockTest dealloc"); 
  12.  
  13. #pragma mark - private 
  14. ////循環引用 
  15. -(void )test3 
  16.     _person2=@"person2"
  17.     _action= ^(void) { 
  18.         //block內賦值 
  19.         NSLog(@"excuteing _person2:%@,%p",_person2,_person2); 
  20.     }; 
  21.     _action(); 

這樣我們執行以下代碼:

  1. KDBlockTest *_test=[[KDBlockTest alloc]init]; 

通過調試發現沒有走到dealloc,這里不管成員變量 _person2 是否聲明 __block都沒有什么影響。

成員變量 這一篇已經詳細介紹了,對于block 使用 成員變量、self來說,block內部是直接強引用self的。也就是block持有了self,在這里bock又作為self的一個成員被持有,所以就形成了相互引用,導致無法釋放。

——weak

對于上面這種情況,我們引入了__weak解決,__weak不會增加對象的引用計數,而且當指向的內存銷毀后,__weak指針會自動置為nil。

我們對上面的代碼稍作修改

 

  1. -(void )test3 
  2.     __weak typeof(self) _weakSelf=self; 
  3.     _person2=@"person2"
  4.     NSLog(@"init:%@,%p,%p",_person2,_person2,&self); 
  5.     _action= ^(void) { 
  6.         //block內賦值 
  7.         NSLog(@"excuteing _person2:%@,%p,%p",_weakSelf.person2,_weakSelf.person2,&_weakSelf); 
  8.     }; 
  9.     _action(); 

輸出日志:

2014-07-29 13:38:30.872 Test[2642:60b] init:person2,0x5b980,0x27dae944
2014-07-29 13:38:30.875 Test[2642:60b] excuteing _person2:person2,0x5b980,0x1562ed44
2014-07-29 13:38:30.876 Test[2642:60b] KDBlockTest dealloc

從日志可以看出block內部使用 person2 、_weakSelf 和外面的 person2 、self 地址是一樣的,看來也是引用關系,既達到block內部修改變量的效果,又沒有對變量產生強引用。我們來看下轉換后的代碼:

block結構體的定義:

  1. struct __KDBlockTest__test3_block_impl_0 { 
  2.   struct __block_impl impl; 
  3.   struct __KDBlockTest__test3_block_desc_0* Desc; 
  4.   __weak typeof (self) _weakSelf; 
  5.   __KDBlockTest__test3_block_impl_0(void *fp, struct __KDBlockTest__test3_block_desc_0 *desc, __weak typeof (self) __weakSelf, int flags=0) : _weakSelf(__weakSelf) { 
  6.     impl.isa = &_NSConcreteStackBlock; 
  7.     impl.Flags = flags; 
  8.     impl.FuncPtr = fp; 
  9.     Desc = desc; 
  10.   } 
  11. }; 

重點就在這,使用_weak聲明的self,block結構體對應 也生成了一個_weak的self成員。我們在看下 我們的test3 方法:

  1. static void _I_KDBlockTest_test3(KDBlockTest * self, SEL _cmd) { 
  2.     __attribute__((objc_gc(weak))) typeof(self) _weakSelf=self; 
  3.     (*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2))=(NSString *)&__NSConstantStringImpl__var_folders_5l_2l25j3tn0wl_3zy1hpsq1rhc0000gp_T_KDBlockTest_d10f18_mi_1; 
  4.     NSLog((NSString *)&__NSConstantStringImpl__var_folders_5l_2l25j3tn0wl_3zy1hpsq1rhc0000gp_T_KDBlockTest_d10f18_mi_2,(*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2)),(*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2)),&self); 
  5.     (*(ActionTest *)((char *)self + OBJC_IVAR_$_KDBlockTest$_action))= (void (*)())&__KDBlockTest__test3_block_impl_0((void *)__KDBlockTest__test3_block_func_0, &__KDBlockTest__test3_block_desc_0_DATA, _weakSelf, 570425344); 
  6.     ((void (*)(__block_impl *))((__block_impl *)(*(ActionTest *)((char *)self + OBJC_IVAR_$_KDBlockTest$_action)))->FuncPtr)((__block_impl *)(*(ActionTest *)((char *)self + OBJC_IVAR_$_KDBlockTest$_action))); 

block初始化的時候把  _weakSelf的地址傳入,block內部對_weakSelf進行弱引用。在執行block的時候

  1. static void __KDBlockTest__test3_block_func_0(struct __KDBlockTest__test3_block_impl_0 *__cself) { 
  2.   __weak typeof (self) _weakSelf = __cself->_weakSelf; // bound by copy 
  3.  
  4.  
  5.         NSLog((NSString *)&__NSConstantStringImpl__var_folders_5l_2l25j3tn0wl_3zy1hpsq1rhc0000gp_T_KDBlockTest_d10f18_mi_3,((NSString *(*)(id, SEL))(void *)objc_msgSend)((id)_weakSelf, sel_registerName("person2")),((NSString *(*)(id, SEL))(void *)objc_msgSend)((id)_weakSelf, sel_registerName("person2")),&_weakSelf); 
  6.     } 

通過取得block結構體的 弱引用對象self 成員來訪問相對應的方法 person2 (給對象發消息)。

—weak變量

上面例子,我們稍作修改:

  1. (void )test3 
  2.     _person2=@"person2"
  3.     __weak typeof(_person2) _weakPerson2=_person2; 
  4.     NSLog(@"init:%@,%p,%p",_person2,_person2,&_person2); 
  5.     NSLog(@"init weak:%@,%p,%p",_weakPerson2,_weakPerson2,&_weakPerson2); 
  6.     _action= ^(void) { 
  7.         //block內賦值 
  8.     //_weakPerson2=@"person4";//error ,不能修改 
  9.  
  10.         NSLog(@"excuteing _person2:%@,%p,%p",_weakPerson2,_weakPerson2,&_weakPerson2); 
  11.     }; 
  12.     _person2=@"person22"
  13.     NSLog(@"before:%@,%p,%p",_person2,_person2,&_person2); 
  14.     NSLog(@"before weak:%@,%p,%p",_weakPerson2,_weakPerson2,&_weakPerson2); 
  15.     _action(); 
  16.     NSLog(@"after:%@,%p,%p",_person2,_person2,&_person2); 

輸出日志:

2014-07-29 15:29:33.472 Test[2719:60b] init:person2,0x5397c,0x16566db8
2014-07-29 15:29:33.475 Test[2719:60b] init weak:person2,0x5397c,0x27db693c
2014-07-29 15:29:33.476 Test[2719:60b] before:person22,0x539bc,0x16566db8
2014-07-29 15:29:33.477 Test[2719:60b] before weak:person2,0x5397c,0x27db693c
2014-07-29 15:29:33.479 Test[2719:60b] excuteing _person2:person2,0x5397c,0x165b5be4
2014-07-29 15:29:33.480 Test[2719:60b] after:person22,0x539bc,0x16566db8
2014-07-29 15:29:33.481 Test[2719:60b] KDBlockTest dealloc

從日志可以看出:

  1. 直接用__weak修飾符修飾_person2變量也可以,也可以避免循環引用,但是不可以在block內部修改外部 參數的值
  2. 在block外部修改變量指針指向,即把指針指向另外一塊內存,block內部無法更新到。

我們來看下轉換后的代碼:其實和不加__block的局部變量差不多,無非多了一個弱引用,不會對引用計數有影響。

 

  1. struct __KDBlockTest__test3_block_impl_0 { 
  2.   struct __block_impl impl; 
  3.   struct __KDBlockTest__test3_block_desc_0* Desc; 
  4.   __weak typeof (self->_person2) _weakPerson2; 
  5.   __KDBlockTest__test3_block_impl_0(void *fp, struct __KDBlockTest__test3_block_desc_0 *desc, __weak typeof (self->_person2) __weakPerson2, int flags=0) : _weakPerson2(__weakPerson2) { 
  6.     impl.isa = &_NSConcreteStackBlock; 
  7.     impl.Flags = flags; 
  8.     impl.FuncPtr = fp; 
  9.     Desc = desc; 
  10.   } 
  11. }; 
  1. static void _I_KDBlockTest_test3(KDBlockTest * self, SEL _cmd) { 
  2.  
  3.     (*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2))=(NSString *)&__NSConstantStringImpl__var_folders_5l_2l25j3tn0wl_3zy1hpsq1rhc0000gp_T_KDBlockTest_f32cef_mi_1; 
  4. //聲明_weak 變量 
  5.     __attribute__((objc_gc(weak))) typeof(_person2) _weakPerson2=(*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2)); 
  6.     NSLog((NSString *)&__NSConstantStringImpl__var_folders_5l_2l25j3tn0wl_3zy1hpsq1rhc0000gp_T_KDBlockTest_f32cef_mi_2,(*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2)),(*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2)),&(*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2))); 
  7.     NSLog((NSString *)&__NSConstantStringImpl__var_folders_5l_2l25j3tn0wl_3zy1hpsq1rhc0000gp_T_KDBlockTest_f32cef_mi_3,_weakPerson2,_weakPerson2,&_weakPerson2); 
  8. //初始化block 
  9.     (*(ActionTest *)((char *)self + OBJC_IVAR_$_KDBlockTest$_action))= (void (*)())&__KDBlockTest__test3_block_impl_0((void *)__KDBlockTest__test3_block_func_0, &__KDBlockTest__test3_block_desc_0_DATA, _weakPerson2, 570425344); 
  10.     (*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2))=(NSString *)&__NSConstantStringImpl__var_folders_5l_2l25j3tn0wl_3zy1hpsq1rhc0000gp_T_KDBlockTest_f32cef_mi_5; 
  11.     NSLog((NSString *)&__NSConstantStringImpl__var_folders_5l_2l25j3tn0wl_3zy1hpsq1rhc0000gp_T_KDBlockTest_f32cef_mi_6,(*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2)),(*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2)),&(*(NSString **)((char *)self + OBJC_IVAR_$_KDBlockTest$_person2))); 
  12.     NSLog((NSString *)&__NSConstantStringImpl__var_folders_5l_2l25j3tn0wl_3zy1hpsq1rhc0000gp_T_KDBlockTest_f32cef_mi_7,_weakPerson2,_weakPerson2,&_weakPerson2); 
  13.     ((void (*)(__block_impl *))((__block_impl *)(*(ActionTest *)((char *)self + OBJC_IVAR_$_KDBlockTest$_action)))->FuncPtr)((__block_impl *)(*(ActionTest *)((char *)self + OBJC_IVAR_$_KDBlockTest$_action))); 

在聲明 _weak變量的時候,生成了一個 弱引用的指針 指向 self的person2變量。在block初始化的時候,把弱引用指針指向的內容地址 傳遞給了block成員

__weak typeof (self->_person2) _weakPerson2;

block結構體內部通過 成員 _weakPerson2 直接弱引用了外部變量 person2的內容地址。這時候如果把person2指針指向另外一塊內存地址,那么肯定是同步不到block內部的,這個和 局部變量  大同小異。

總結:

  1. 聲明 __weak typeof(self) _weakSelf=self;  這樣block內部 生成一個成員 ,會對self弱引用,對于值類型、引用類型都可以修改,并且修改指針指向都可以同步到任何地方。
  2. 聲明 __weak typeof(_person2) _weakPerson2=_person2;  針對某個具體的成員變量使用weak修飾符,這樣可以避免循環引用,并且不能再block內部修改_weakPerson2。規則如下:
    • 對值類型的修改,如果block初始化后,對值類型修改,無法同步到block內部。
    • 對于引用類型的修改,如果block初始化后,修改指針指向,即指向另外一塊內存,這樣也是無法同步到block內部

       

      對于引用類型的修改,如果block初始化后,對指針指向的內存進行修改,即NSMutableArray add 、remove操作,這樣是可以用同步到block內部。

責任編輯:chenqingxiang 來源: cnblogs
相關推薦

2014-07-31 16:47:10

block

2013-07-19 13:16:26

iOS中BlockiOS開發學習內存管理

2015-03-30 11:18:50

內存管理Android

2010-09-25 14:12:50

Java內存分配

2010-09-16 09:13:09

CSS display

2016-03-21 10:31:25

Android內存泄露

2010-09-26 14:55:46

JVM內存監控

2010-09-25 12:54:24

JVM內存

2017-03-07 09:45:43

iOSBlock開發

2013-10-11 17:24:47

Linux運維內存管理

2021-10-15 08:51:09

Linux內存 Kmalloc

2020-12-23 13:14:00

LinuxLinux內存Swap

2013-07-19 12:52:50

iOS中BlockiOS開發學習

2021-03-30 10:50:18

Linux內存命令

2021-04-30 20:20:36

HugePages大內存頁系統

2016-10-09 14:41:40

Swift開發ARC

2016-03-07 09:09:35

blockios開發實踐

2011-05-05 18:28:18

2010-02-22 08:58:35

JVM內存模型垃圾收集

2023-10-12 19:41:55

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: avav在线看| 国产精品欧美精品 | 91精品导航| 一区在线视频 | 色欧美综合| 久久精品国产99国产精品 | av看片| 免费看黄视频网站 | 伊人二区| 成人99 | 特级黄一级播放 | 天天综合91 | 久久噜噜噜精品国产亚洲综合 | 欧美日韩电影在线 | 黄瓜av| www.色综合 | 欧美精品在线观看 | 国产精产国品一二三产区视频 | 成人免费小视频 | 欧美日韩精品 | 亚洲综合国产精品 | 久久成人午夜 | 久久成人一区 | 欧美a级成人淫片免费看 | 日韩视频91 | 成人免费大片黄在线播放 | 国产精品美女久久久久久免费 | 日韩av一区二区在线观看 | h片在线播放 | 毛片视频免费 | 欧美成年人视频在线观看 | 中文字幕91 | 97av视频在线 | 一级片片 | 国产福利在线免费观看 | 中文字幕在线第二页 | 日本精品一区 | 九九热在线视频 | 欧美精品一区二区三区在线播放 | 亚洲精品一区在线 | 国产精品成人一区二区 |