您好,歡迎來電子發(fā)燒友網! ,新用戶?[免費注冊]

您的位置:電子發(fā)燒友網>源碼下載>java源碼下載>

Runloop是怎樣進行線程?;?/h1>
大?。?/span>0.3 MB 人氣: 2017-09-26 需要積分:1

  AFN 中的實現(xiàn)

  在舊版本的AFN 中使用了 NSURLConnection 來發(fā)起并處理網絡連接。

  AFN 的做法是把網絡請求的發(fā)起和解析都放在同一個子線程中進行,子線程默認不開啟 runloop,它會向一個 C語言程序那樣在運行完所有代碼后退出線程。

  而網絡請求是異步的,這導致獲取到請求數(shù)據(jù)時,線程已經退出,代理方法沒有機會執(zhí)行。

  因此,AFN 的做法是使用一個 runloop 來保證線程不死,也就是下面這段被講爛了的代碼:

  + (void)networkRequestThreadEntryPoint:(id)__unused object {

  @autoreleasepool {

 ?。郏跱SThread currentThread] setName:@“AFNetworking”];

  NSRunLoop *runLoop = [NSRunLoop currentRunLoop];

 ?。踨unLoop addPort:[NSMachPort port] forMode:NSDefaultRunLoopMode];

  [runLoop run];

  }

  }

  稍微結合一下上下文,看看這個方法在哪里被調用:

  + (NSThread *)networkRequestThread {

  static NSThread *_networkRequestThread = nil;

  static dispatch_once_t oncePredicate;

  dispatch_once(&oncePredicate, ^{

  _networkRequestThread = [[NSThread alloc] initWithTarget:self selector:@selector(networkRequestThreadEntryPoint:) object:nil];

  [_networkRequestThread start];

  });

  return _networkRequestThread;

  }

  似乎這種寫法提供了一種思路:“如果需要在子線程中異步執(zhí)行操作,可以利用 runloop 進行線程保活”。但準確的來說,AFN 的這種寫法并不能實現(xiàn)我們的需求,它只是在 AFN 這個特殊場景下可以工作。

  NSThread 與內存泄漏

  這種寫法的第一個問題就是存在內存泄漏。我們構造以下用例,把 AFN 的線程創(chuàng)建放在一個循環(huán)里:

  - (void)memoryTest {

  for (int i = 0; i 《 100000; ++i) {

  NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(run) object:nil];

 ?。踭hread start];

  }

  }

  - (void)run {

  @autoreleasepool {

  NSLog(@“current thread = %@”, [NSThread currentThread]);

  NSRunLoop *runLoop = [NSRunLoop currentRunLoop];

  if (!self.emptyPort) {

  self.emptyPort = [NSMachPort port];

  }

 ?。踨unLoop addPort:self.emptyPort forMode:NSDefaultRunLoopMode];

 ?。踨unLoop run];

  }

  }

  奇怪的事情出現(xiàn)了,盡管是在 ARC 環(huán)境下,內存依然不停的上漲。如果我們把 run 方法中和 runloop 相關的代碼刪除則不會出現(xiàn)上述問題,顯然,開啟 runloop 導致了內存泄漏,也就是 thread 對象無法釋放。

非常好我支持^.^

(1) 100%

不好我反對

(0) 0%

      發(fā)表評論

      用戶評論
      評價:好評中評差評

      發(fā)表評論,獲取積分! 請遵守相關規(guī)定!

      ?