函數參數,返回值,遞歸函數

来源:https://www.cnblogs.com/JcrLive/archive/2020/01/27/12235369.html
-Advertisement-
Play Games

函數進階 目標 函數參數和返回值的作用 函數的返回值 進階 函數的參數 進階 遞歸函數 01. 函數參數和返回值的作用 函數根據 有沒有參數 以及 有沒有返回值 ,可以 相互組合 ,一共有 4 種 組合形式 1. 無參數,無返回值 2. 無參數,有返回值 3. 有參數,無返回值 4. 有參數,有返回 ...


函數進階

目標

  • 函數參數和返回值的作用
  • 函數的返回值 進階
  • 函數的參數 進階
  • 遞歸函數

01. 函數參數和返回值的作用

函數根據 有沒有參數 以及 有沒有返回值,可以 相互組合,一共有 4 種 組合形式

  1. 無參數,無返回值
  2. 無參數,有返回值
  3. 有參數,無返回值
  4. 有參數,有返回值

定義函數時,是否接收參數,或者是否返回結果,是根據 實際的功能需求 來決定的!

  1. 如果函數 內部處理的數據不確定,就可以將外界的數據以參數傳遞到函數內部
  2. 如果希望一個函數 執行完成後,向外界彙報執行結果,就可以增加函數的返回值

1.1 無參數,無返回值

此類函數,不接收參數,也沒有返回值,應用場景如下:

  1. 只是單純地做一件事情,例如 顯示菜單
  2. 在函數內部 針對全局變數進行操作,例如:新建名片,最終結果 記錄在全局變數

註意:

  • 如果全局變數的數據類型是一個 可變類型,在函數內部可以使用 方法 修改全局變數的內容 —— 變數的引用不會改變
  • 在函數內部,使用賦值語句 才會 修改變數的引用

1.2 無參數,有返回值

此類函數,不接收參數,但是有返回值,應用場景如下:

  • 採集數據,例如 溫度計,返回結果就是當前的溫度,而不需要傳遞任何的參數

1.3 有參數,無返回值

此類函數,接收參數,沒有返回值,應用場景如下:

  • 函數內部的代碼保持不變,針對 不同的參數 處理 不同的數據
  • 例如 名片管理系統 針對 找到的名片修改刪除 操作

1.4 有參數,有返回值

此類函數,接收參數,同時有返回值,應用場景如下:

  • 函數內部的代碼保持不變,針對 不同的參數 處理 不同的數據,並且 返回期望的處理結果
  • 例如 名片管理系統 使用 字典預設值提示信息 提示用戶輸入內容
    • 如果輸入,返回輸入內容
    • 如果沒有輸入,返回字典預設值

02. 函數的返回值 進階

  • 在程式開發中,有時候,會希望 一個函數執行結束後,告訴調用者一個結果,以便調用者針對具體的結果做後續的處理
  • 返回值 是函數 完成工作後,最後 給調用者的 一個結果
  • 在函數中使用 return 關鍵字可以返回結果
  • 調用函數一方,可以 使用變數接收 函數的返回結果

問題:一個函數執行後能否返回多個結果?

示例 —— 溫度和濕度測量

  • 假設要開發一個函數能夠同時返回當前的溫度和濕度
  • 先完成返回溫度的功能如下:
def measure():
    """返回當前的溫度"""
    
    print("開始測量...")
    temp = 39
    print("測量結束...")
    
    return temp

result = measure()
print(result)
  • 在利用 元組 在返回溫度的同時,也能夠返回 濕度
  • 改造如下:
def measure():
    """返回當前的溫度"""

    print("開始測量...")
    temp = 39
    wetness = 10
    print("測量結束...")

    return (temp, wetness)

提示:如果一個函數返回的是元組,括弧可以省略

技巧

  • Python 中,可以 將一個元組 使用 賦值語句 同時賦值給 多個變數
  • 註意:變數的數量需要和元組中的元素數量保持一致
result = temp, wetness = measure()

面試題 —— 交換兩個數字

題目要求

  1. 有兩個整數變數 a = 6, b = 100
  2. 不使用其他變數,交換兩個變數的值

解法 1 —— 使用其他變數

# 解法 1 - 使用臨時變數
c = b
b = a
a = c

解法 2 —— 不使用臨時變數

# 解法 2 - 不使用臨時變數
a = a + b
b = a - b
a = a - b

解法 3 —— Python 專有,利用元組

a, b = b, a

03. 函數的參數 進階

3.1. 不可變和可變的參數

問題 1:在函數內部,針對參數使用 賦值語句,會不會影響調用函數時傳遞的 實參變數? —— 不會!

  • 無論傳遞的參數是 可變 還是 不可變
    • 只要 針對參數 使用 賦值語句,會在 函數內部 修改 局部變數的引用不會影響到 外部變數的引用
def demo(num, num_list):

    print("函數內部")

    # 賦值語句
    num = 200
    num_list = [1, 2, 3]

    print(num)
    print(num_list)

    print("函數代碼完成")


gl_num = 99
gl_list = [4, 5, 6]
demo(gl_num, gl_list)
print(gl_num)
print(gl_list)
    

問題 2:如果傳遞的參數是 可變類型,在函數內部,使用 方法 修改了數據的內容,同樣會影響到外部的數據

def mutable(num_list):

    # num_list = [1, 2, 3]
    num_list.extend([1, 2, 3])
    
    print(num_list)

gl_list = [6, 7, 8]
mutable(gl_list)
print(gl_list)

面試題 —— +=

  • python 中,列表變數調用 += 本質上是在執行列表變數的 extend 方法,不會修改變數的引用
def demo(num, num_list):

    print("函數內部代碼")

    # num = num + num
    num += num
    # num_list.extend(num_list) 由於是調用方法,所以不會修改變數的引用
    # 函數執行結束後,外部數據同樣會發生變化
    num_list += num_list

    print(num)
    print(num_list)
    print("函數代碼完成")


gl_num = 9
gl_list = [1, 2, 3]
demo(gl_num, gl_list)
print(gl_num)
print(gl_list)

3.2 預設參數

  • 定義函數時,可以給 某個參數 指定一個預設值,具有預設值的參數就叫做 預設參數
  • 調用函數時,如果沒有傳入 預設參數 的值,則在函數內部使用定義函數時指定的 參數預設值
  • 函數的預設參數,將常見的值設置為參數的預設值,從而 簡化函數的調用
  • 例如:對列表排序的方法
gl_num_list = [6, 3, 9]

# 預設就是升序排序,因為這種應用需求更多
gl_num_list.sort()
print(gl_num_list)

# 只有當需要降序排序時,才需要傳遞 `reverse` 參數
gl_num_list.sort(reverse=True)
print(gl_num_list)

指定函數的預設參數

  • 在參數後使用賦值語句,可以指定參數的預設值
def print_info(name, gender=True):

    gender_text = "男生"
    if not gender:
        gender_text = "女生"

    print("%s 是 %s" % (name, gender_text))

提示

  1. 預設參數,需要使用 最常見的值 作為預設值!
  2. 如果一個參數的值 不能確定,則不應該設置預設值,具體的數值在調用函數時,由外界傳遞!

預設參數的註意事項

1) 預設參數的定義位置
  • 必須保證 帶有預設值的預設參數 在參數列表末尾
  • 所以,以下定義是錯誤的!
def print_info(name, gender=True, title):
2) 調用帶有多個預設參數的函數
  • 調用函數時,如果有 多個預設參數需要指定參數名,這樣解釋器才能夠知道參數的對應關係!
def print_info(name, title="", gender=True):
    """

    :param title: 職位
    :param name: 班上同學的姓名
    :param gender: True 男生 False 女生
    """

    gender_text = "男生"

    if not gender:
        gender_text = "女生"

    print("%s%s 是 %s" % (title, name, gender_text))


# 提示:在指定預設參數的預設值時,應該使用最常見的值作為預設值!
print_info("小明")
print_info("老王", title="班長")
print_info("小美", gender=False)

3.3 多值參數(知道)

定義支持多值參數的函數

  • 有時可能需要 一個函數 能夠處理的參數 個數 是不確定的,這個時候,就可以使用 多值參數
  • python 中有 兩種 多值參數:
    • 參數名前增加 一個 * 可以接收 元組
    • 參數名前增加 兩個 * 可以接收 字典
  • 一般在給多值參數命名時,習慣使用以下兩個名字
    • *args —— 存放 元組 參數,前面有一個 *
    • **kwargs —— 存放 字典 參數,前面有兩個 *
  • argsarguments 的縮寫,有變數的含義
  • kwkeyword 的縮寫,kwargs 可以記憶 鍵值對參數
def demo(num, *args, **kwargs):

    print(num)
    print(args)
    print(kwargs)


demo(1, 2, 3, 4, 5, name="小明", age=18, gender=True)

提示:多值參數 的應用會經常出現在網路上一些大牛開發的框架中,知道多值參數,有利於我們能夠讀懂大牛的代碼

多值參數案例 —— 計算任意多個數字的和

需求

  1. 定義一個函數 sum_numbers,可以接收的 任意多個整數
  2. 功能要求:將傳遞的 所有數字累加 並且返回累加結果
def sum_numbers(*args):

    num = 0
    # 遍歷 args 元組順序求和
    for n in args:
        num += n

    return num

print(sum_numbers(1, 2, 3))

元組和字典的拆包(知道)

  • 在調用帶有多值參數的函數時,如果希望:
    • 將一個 元組變數,直接傳遞給 args
    • 將一個 字典變數,直接傳遞給 kwargs
  • 就可以使用 拆包,簡化參數的傳遞,拆包 的方式是:
    • 元組變數前,增加 一個 *
    • 字典變數前,增加 兩個 *
def demo(*args, **kwargs):

    print(args)
    print(kwargs)


# 需要將一個元組變數/字典變數傳遞給函數對應的參數
gl_nums = (1, 2, 3)
gl_xiaoming = {"name": "小明", "age": 18}

# 會把 num_tuple 和 xiaoming 作為元組傳遞個 args
# demo(gl_nums, gl_xiaoming)
demo(*gl_nums, **gl_xiaoming)

04. 函數的遞歸

函數調用自身的 編程技巧 稱為遞歸

4.1 遞歸函數的特點

特點

  • 一個函數 內部 調用自己
    • 函數內部可以調用其他函數,當然在函數內部也可以調用自己

代碼特點

  1. 函數內部的 代碼 是相同的,只是針對 參數 不同,處理的結果不同
  2. 參數滿足一個條件 時,函數不再執行
    • 這個非常重要,通常被稱為遞歸的出口,否則 會出現死迴圈

示例代碼

def sum_numbers(num):

    print(num)
    
    # 遞歸的出口很重要,否則會出現死迴圈
    if num == 1:
        return

    sum_numbers(num - 1)
    
sum_numbers(3)

4.2 遞歸案例 —— 計算數字累加

需求

  1. 定義一個函數 sum_numbers
  2. 能夠接收一個 num 的整數參數
  3. 計算 1 + 2 + ... num 的結果
def sum_numbers(num):

    if num == 1:
        return 1
    
    # 假設 sum_numbers 能夠完成 num - 1 的累加
    temp = sum_numbers(num - 1)

    # 函數內部的核心演算法就是 兩個數字的相加
    return num + temp

print(sum_numbers(2))

提示:遞歸是一個 編程技巧,初次接觸遞歸會感覺有些吃力!在處理 不確定的迴圈條件時,格外的有用,例如:遍歷整個文件目錄的結構


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

-Advertisement-
Play Games
更多相關文章
  • ![](https://img2018.cnblogs.com/blog/1853166/202001/1853166-20200126143013655-154923141.png) ![](https://img2018.cnblogs.com/blog/1853166/202001/18531... ...
  • 滿足最小分組的長度為其他分組的整數倍 ...
  • ![](https://img2018.cnblogs.com/blog/1853166/202001/1853166-20200126115527094-488420466.png)![](https://img2018.cnblogs.com/blog/1853166/202001/185316... ...
  • jQuery 滑動方法 通過 jQuery,您可以在元素上創建滑動效果。 jQuery 擁有以下滑動方法: slideDown() slideUp() slideToggle() jQuery slideDown() 方法 jQuery slideDown() 方法用於向下滑動元素。 語法: $(s ...
  • javascript export default (str) = { // 建立數據結構,堆棧,保存數據 let r = [] // 給定任意子輸入都返回第一個符合條件的子串 let match = (str) = { let j = str.match(/^(0+|1+)/)[0] let o ...
  • 第一種 第二種 第三種 第四種 ...
  • 互聯網那些事之數據丟失 本系列故事的所有案例和解決方案只是筆者以前在互聯網工作期間的一些事例,僅供大家參考,實際操作應該根據業務和項目情況設計,歡迎大家留言提出寶貴的意見 背景 小王和小明分別維護分散式系統中A、b兩個服務,有一個場景是 A服務會向B服務通過MQ發送事件並且推送用戶信息,然後B服務保 ...
  • 面向對象編程 —— Object Oriented Programming 簡寫 OOP 01. 面向對象基本概念 我們之前學習的編程方式就是 面向過程 的 面相過程 和 面相對象,是兩種不同的 編程方式 對比 面向過程 的特點,可以更好地瞭解什麼是 面向對象 1.1 過程和函數(科普) 過程 是早 ...
一周排行
    -Advertisement-
    Play Games
  • C#TMS系統代碼-基礎頁面BaseCity學習 本人純新手,剛進公司跟領導報道,我說我是java全棧,他問我會不會C#,我說大學學過,他說這個TMS系統就給你來管了。外包已經把代碼給我了,這幾天先把增刪改查的代碼背一下,說不定後面就要趕鴨子上架了 Service頁面 //using => impo ...
  • 委托與事件 委托 委托的定義 委托是C#中的一種類型,用於存儲對方法的引用。它允許將方法作為參數傳遞給其他方法,實現回調、事件處理和動態調用等功能。通俗來講,就是委托包含方法的記憶體地址,方法匹配與委托相同的簽名,因此通過使用正確的參數類型來調用方法。 委托的特性 引用方法:委托允許存儲對方法的引用, ...
  • 前言 這幾天閑來沒事看看ABP vNext的文檔和源碼,關於關於依賴註入(屬性註入)這塊兒產生了興趣。 我們都知道。Volo.ABP 依賴註入容器使用了第三方組件Autofac實現的。有三種註入方式,構造函數註入和方法註入和屬性註入。 ABP的屬性註入原則參考如下: 這時候我就開始疑惑了,因為我知道 ...
  • C#TMS系統代碼-業務頁面ShippingNotice學習 學一個業務頁面,ok,領導開完會就被裁掉了,很突然啊,他收拾東西的時候我還以為他要旅游提前請假了,還在尋思為什麼回家連自己買的幾箱飲料都要叫跑腿帶走,怕被偷嗎?還好我在他開會之前拿了兩瓶芬達 感覺感覺前面的BaseCity差不太多,這邊的 ...
  • 概述:在C#中,通過`Expression`類、`AndAlso`和`OrElse`方法可組合兩個`Expression<Func<T, bool>>`,實現多條件動態查詢。通過創建表達式樹,可輕鬆構建複雜的查詢條件。 在C#中,可以使用AndAlso和OrElse方法組合兩個Expression< ...
  • 閑來無聊在我的Biwen.QuickApi中實現一下極簡的事件匯流排,其實代碼還是蠻簡單的,對於初學者可能有些幫助 就貼出來,有什麼不足的地方也歡迎板磚交流~ 首先定義一個事件約定的空介面 public interface IEvent{} 然後定義事件訂閱者介面 public interface I ...
  • 1. 案例 成某三甲醫預約系統, 該項目在2024年初進行上線測試,在正常運行了兩天後,業務系統報錯:The connection pool has been exhausted, either raise MaxPoolSize (currently 800) or Timeout (curren ...
  • 背景 我們有些工具在 Web 版中已經有了很好的實踐,而在 WPF 中重新開發也是一種費時費力的操作,那麼直接集成則是最省事省力的方法了。 思路解釋 為什麼要使用 WPF?莫問為什麼,老 C# 開發的堅持,另外因為 Windows 上已經裝了 Webview2/edge 整體打包比 electron ...
  • EDP是一套集組織架構,許可權框架【功能許可權,操作許可權,數據訪問許可權,WebApi許可權】,自動化日誌,動態Interface,WebApi管理等基礎功能於一體的,基於.net的企業應用開發框架。通過友好的編碼方式實現數據行、列許可權的管控。 ...
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...