Swift函數和閉包

来源:http://www.cnblogs.com/evening015/archive/2016/04/03/5349089.html
-Advertisement-
Play Games

簡介:本文主要講:函數的定義,外部參數的用處,無返回類型的三種函數定義方式 閉包的定義,閉包的概念和用法,尾隨閉包的寫法,解除迴圈引用的方法 一、函數: 代碼實現 函數的定義 格式 func 函數名(行參列表) -> 返回值 {代碼實現} 調用 let result = 函數名(值1, 參數2: 值 ...


簡介:本文主要講:函數的定義,外部參數的用處,無返回類型的三種函數定義方式

閉包的定義,閉包的概念和用法,尾隨閉包的寫法,解除迴圈引用的方法

一、函數:

代碼實現

  • 函數的定義
    • 格式 func 函數名(行參列表) -> 返回值 {代碼實現}
    • 調用 let result = 函數名(值1, 參數2: 值2...)
func sum(a: Int, b: Int) -> Int {
    return a + b
}

let result = sum(10, b: 20)
  • 沒有返回值的函數,一共有三種寫法
    • 省略
    • ()
    • Void
func demo(str: String) -> Void {
    print(str)
}
func demo1(str: String) -> () {
    print(str)
}
func demo2(str: String) {
    print(str)
}

demo("hello")
demo1("hello world")
demo2("olleh")
  • 外部參數
    • 在形參名前再增加一個外部參數名,能夠方便調用人員更好地理解函數的語義
    • 格式:func 函數名(外部參數名 形式參數名: 形式參數類型) -> 返回值類型 { // 代碼實現 }
    • Swift 2.0 中,預設第一個參數名省略
func sum1(num1 a: Int, num2 b: Int) -> Int {
    return a + b
}

sum1(num1: 10, num2: 20)

二、閉包:

與 OC 中的 Block 類似,閉包主要用於非同步操作執行完成後的代碼回調,網路訪問結果以參數的形式傳遞給調用方

閉包的定義

  • 定義一個函數
//: 定義一個 sum 函數
func sum(num1 num1: Int, num2: Int) -> Int {
    return num1 + num2
}
sum(num1: 10, num2: 30)

//: 在 Swift 中函數本身就可以當作參數被定義和傳遞
let mySum = sum
let result = mySum(num1: 20, num2: 30)
  • 定義一個閉包
    • 閉包 = { (行參) -> 返回值 in // 代碼實現 }
    • in 用於區分函數定義和代碼實現
//: 閉包 = { (行參) -> 返回值 in // 代碼實現 }
let sumFunc = { (num1 x: Int, num2 y: Int) -> Int in
    return x + y
}
sumFunc(num1: 10, num2: 20)
  • 最簡單的閉包,如果沒有參數/返回值,則 參數/返回值/in 統統都可以省略
    • { 代碼實現 }
let demoFunc = {
    print("hello")
}

基本使用

GCD 非同步

  • 模擬在後臺線程載入數據
func loadData() {
    dispatch_async(dispatch_get_global_queue(0, 0), { () -> Void in
        print("耗時操作 \(NSThread .currentThread())")
    })
}
  • 尾隨閉包,如果閉包是最後一個參數,可以用以下寫法
  • 註意上下兩段代碼,} 的位置
func loadData() {
    dispatch_async(dispatch_get_global_queue(0, 0)) { () -> Void in
        print("耗時操作 \(NSThread .currentThread())")
    }
}
  • 閉包的簡寫,如果閉包中沒有參數和返回值,可以省略
func loadData() {
    dispatch_async(dispatch_get_global_queue(0, 0)) {
        print("耗時操作 \(NSThread .currentThread())")
    }
}

自定義閉包參數,實現主線程回調

  • 添加沒有參數,沒有返回值的閉包
override func viewDidLoad() {
    super.viewDidLoad()

    loadData {
        print("完成回調")
    }
}

// MARK: - 自定義閉包參數
func loadData(finished: ()->()) {

    dispatch_async(dispatch_get_global_queue(0, 0)) {
        print("耗時操作 \(NSThread.currentThread())")

        dispatch_sync(dispatch_get_main_queue()) {
            print("主線程回調 \(NSThread.currentThread())")

            // 執行回調
            finished()
        }
    }
}
  • 添加回調參數
override func viewDidLoad() {
    super.viewDidLoad()

    loadData4 { (html) -> () in
        print(html)
    }
}

/// 載入數據
/// 完成回調 - 傳入回調閉包,接收非同步執行的結果
func loadData4(finished: (html: String) -> ()) {

    dispatch_async(dispatch_get_global_queue(0, 0)) {
        print("載入數據 \(NSThread.currentThread())")

        dispatch_sync(dispatch_get_main_queue()) {
            print("完成回調 \(NSThread.currentThread())")

            finished(html: "<h1>hello world</h1>")
        }
    }
}


迴圈引用

  • 建立 NetworkTools 對象
class NetworkTools: NSObject {

    /// 載入數據
    ///
    /// - parameter finished: 完成回調
    func loadData(finished: () -> ()) {
        print("開始載入數據...")

        // ...
        finished()
    }

    deinit {
        print("網路工具 88")
    }
}
  • 實例化 NetworkTools 並且載入數據
class ViewController: UIViewController {

    var tools: NetworkTools?

    override func viewDidLoad() {
        super.viewDidLoad()

        tools = NetworkTools()
        tools?.loadData() {
            print("come here \(self.view)")
        }
    }

    /// 與 OC 中的 dealloc 類似,註意此函數沒有()
    deinit {
        print("控制器 88")
    }
}

運行不會形成迴圈引用,因為 loadData 執行完畢後,就會釋放對 self 的引用

  • 修改 NetworkTools,定義回調閉包屬性
/// 完成回調屬性
var finishedCallBack: (()->())?

/// 載入數據
///
/// - parameter finished: 完成回調
func loadData(finished: () -> ()) {

    self.finishedCallBack = finished

    print("開始載入數據...")

    // ...
    working()
}

func working() {
    finishedCallBack?()
}

deinit {
    print("網路工具 88")
}

運行測試,會出現迴圈引用

解除迴圈引用

  • 與 OC 類似的方法
/// 類似於 OC 的解除引用
func demo() {
    weak var weakSelf = self
    tools?.loadData() {
        print("\(weakSelf?.view)")
    }
}
  • Swift 推薦的方法
loadData { [weak self] in
    print("\(self?.view)")
}
  • 還可以
loadData { [unowned self] in
    print("\(self.view)")
}

閉包(Block) 的迴圈引用小結

  • Swift

    • [weak self]
      • self是可選項,如果self已經被釋放,則為nil
    • [unowned self]
      • self不是可選項,如果self已經被釋放,則出現野指針訪問
  • Objc

    • __weak typeof(self) weakSelf;
      • 如果self已經被釋放,則為nil
    • __unsafe_unretained typeof(self) weakSelf;
      • 如果self已經被釋放,則出現野指針訪問

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這裡體現出閉包的數據共用 這裡體現出閉包的局部變數升級為成員變數 (在拉姆達表達式生成的匿名類中不會不會隨著方法執行完後彈棧 而是隨著回調函數徹底執行完後才被回收) ...
  • 一、前言 之前半年時間感覺自己有點浮躁,導致停頓了半年多的時間沒有更新博客,今天重新開始記錄博文,希望自己可以找回初心,繼續沉澱。由於最近做的項目中用到SignalR技術,所以打算總結下Asp.net SignalR的相關內容,希望對剛接觸或者接觸不多的朋友有所幫助。今天的專題就是讓大家可以快速的上 ...
  • 今天有意的在博客園裡面搜索了一下 Z.ExtensionMethods 這個擴展類庫,確發現只搜到跟這個真正相關的才兩篇博文而已,我都點進去看了一下,也都只是提到而已,沒有專門介紹,才引起我寫這篇文檔。 一. Z.ExtensionMethods 介紹 Z.ExtensionMethods 是國外( ...
  • ...
  • 2016-04-03 實現對二維數組排序❖ 對購物車商品表格實現:按數量,按單價分別降序/升序排序。❖ 查閱參考手中, usort( )函數的說明。 要求效果圖如下: 註:此處下方實際應有四個按鈕,分別控制四種不同的排序,因為一些特殊的原因無法給出。 實現代碼如下: sort-cart.php: 這 ...
  • 環境:windows 7 64位;python2.7;IDE pycharm2016.1 功能: 批量下載百度貼吧某吧某頁的所有帖子中的所有圖片 使用方法: 1.安裝python2.7,安裝re模塊,安裝urllib2模塊 2.複製以下源代碼保存為tbImgiDownloader.py文件 3.打開 ...
  • 一、刪除首碼 '*' 1 #include<iostream> 2 #include<cstdio> 3 4 using namespace std; 5 6 //主函數 7 int main() 8 { 9 char chr[20],*b,*p; //字元串緩衝區;字元串頭指針;字元串臨時指針 1 ...
  • 在使用Json傳值並且使用@RequestBody註解的時候需要註意一些問題: 第一條容易理解,因為RequestBody就是request的inputStream,這個流在第一次使用該註解後會關閉,後面的都會報錯(stream closed)。 第二條如果沒有包含前臺傳來的欄位,就會報錯:Unre ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...