掉进悬崖的小白,学python时被拉进函数漩涡,参数类型灌入脑中,学会复用代码 DRY,局部变量和全局变量的冲突解决。猜数游戏,数字转换大写汉字,随机赋不重复值。

函数(function)

函数论:

广义:
为获得某种东西或达到某种目的而采取的手段与行为方式。

狭义:
函数是指由一系列的程序语句组成的代码块

特性:

1.带名字的代码段
2.函数是代码重复使用的重要手段,是重用的基本手段:
无论现实世界还是程序世界,都以函数来达到重用的目的

定义函数

定义语法:[]表示可选 <>表示必备
<函数名>([参数列表])<:>
# 待执行语句
# 如果有需要显式返回<return 标识符>或
用缩进块来表示函数体,是函数的管辖区块

函数可以出现的位置:
模块(moudle)、类(class)、函数(function)中

命名规范:
函数名、参数名全部小写,多语义用_连接,函数名、参数名要尽量有意义。

调用函数

函数的目的在于重用,所有的函数编写完成后,都处于等待调用状态,被调用后函数开始执行,直到函数返回(有无返回值均必须返回)

函数大多数会在其他函数内部或模块内被调用
注意:模块内调用时注意只能调用该行语句之前出现过的函数
原因:代码自上向下执行

调用的规则:

1.调用者和被调用者相互可见
采用直接调用的方式
示例:

def func1():
 print(“函数1”)
 passfunc1() # 调用函数func1

2.调用者和被调用者相互不可见
先import模块再调用:注意调用时带上模块.
示例:

be_call_functions.py:
 def be_call():
 return “test” # 此处返回
 call_functions.py:
 import be_call_functionsbe_call_functions.be_call()

3.特殊的调用方式
析构函数等魔法函数需要用特殊的方案调用
例如:
析构函数由系统自行调用等等
后续会逐步学到

返回值类型:

所有函数都需要返回
只是没写return的会自动返回None
函数有无返回值取决于函数的调用者是否需要返回值
注意:
函数遇到return立即退出并返回 so也常作为控制方法结束的手段
思考:了解
若返回值数量超过1个怎么办?
默认多返回:return a, b, c, d…
或者使用列表等其他容器返回
一般这样接收结果:
ret1, ret2, ret3 = func_return()

参数

参数分为两种类型:

形式参数:由函数编写者在预定义时限定
实际参数:任意的数据类型(为形参赋值)在函数的调用者中定义赋值,在实际调用(使用)函数的时候 传入函数的参数

声明和传参:
形式参数:

def area(radius): # radius即为形式参数
 return 3.14*radius**2

实际参数:
r = 5
print(area®) # r即为实际参数

// 传参的实质:用实际参数为形式参数赋值
radius = r

位置参数:固定位置的参数

def func_args(arg1, arg2, arg3):
 print(arg1, arg2, arg3)


调用时传实参时按位置传递:

func_args(1, 2, 3)


结果:1 2 3
也可以:
func_args(arg3=3, arg1=1, arg2=2)

默认参数:可以为参数设定默认值 调用时可以选择使用默认值而不传实参

def func_args(arg1=1, arg2=2, arg3=3):
 print(arg1, arg2, arg3)


调用时传实参:
func_args()
结果:1 2 3
也可以:

func_args(3, 2, 1)
 func_args(arg3=6, arg1=6, arg2=6)

命名关键字参数:*

以后的参数全部为显式命名传参的参数
注意:命名关键字参数和可变参数不能同时出现
def func_args(
, arg1, arg2, arg3):
print(arg1, arg2, arg3)
调用时传实参:必须显式命名传参
func_args(arg3=6, arg1=6, arg2=6)

可变参数:*args

可以选择传递0-n个参数
注意:可变参数无法和命名关键字参数同时出现

def func_args(*args):
 for i in range(len(args)):
 pri nt(args[i])


调用时传实参:

func_args()
 func_args(1)
 func_args(1, 2)
 func_args(1, 2, 3…,n)

参数的传递

参数传递需要明白的要点:
形参的修正对实参的影响程度

参数传递的方式:
形式参数 = 实际参数

传参规则:
根据形式参数类型的不同 对实际参数的影响B程度也不相同
A:不可变类型:字符串、数字、布尔、元组
B:列表、类、对象、集合、字典

函数内部形参修正对函数外实参的影响程度:
A:不影响实际参数(同上述A:不可变类型)
B:影响实际参数(同上述B:可变类型)

不可变类型传参:形参的修改不会传递给实参 推荐

def swap(x,y):
x, y = y, x # 形参x,形参y交换数据

x = 666 # 实参x 都叫x,但意义不同
y = 888 # 实参y 都叫y,但意义不同
swap(x, y) #参数传递 形参x = 实参x 形参y = 实参y
print(“实参x =”, x, “实参y=”, y)
结果:实参x = 666 实参y= 888

可变类型传参:形参的修正会传递给实参

def change_list(arg_list):
arg_list[1] = 666

my_list = [1, 2, 3, 4]
change_list(my_list)
print(my_list)
结果:[1, 666, 3, 4]

局部变量:

在函数内部声明并使用的数据量,随函数的启动而出生,随函数的退出而消亡
作用域在函数内
def fun(x):
y= 2
print( “乘法的运行结果:” , x * y)
print( “y的值是:”, y)
报错的原因是因为试图访问局部变量,但是访问的地方不在该变量y的作用域中

全局变量

在模块中直接声明使用的数据量,整个模块中可以使用
作用域在模块内
g_num=“我属于模块 在模块内的任意地方都可使用”

def visitor_g1():
 print(g_num)def visitor_g2():
 print(g_num)visitor_g1()
 visitor_g2()
 print(g_num)

冲突场景

1.函数内修改全局变量会被解释器认为新声明了一个局部变量

例:

num = 10
def set_num(in_num):
 num = in_num
 passset_num(111)
 print(num)


结果:10

解决办法:global关键字

num = 10
def set_num(in_num):
 global num
 num = in_num
 passset_num(111)
 print(num)


结果:111

2.在函数内新声明的局部变量和全局变量同名,但是又想使用全局变量

discount = 0.9 # 全场9折
def pay(money):
 discount = 0.8 # 折上折
 print(“请pay:”, money * discount * discount)
 passpay(10000)


结果:请pay: 6400.0

解决办法:globals()函数

discount = 0.9 # 全场9折
def pay(money):
 discount = 0.8 # 折上折
 print(“请pay:”, money * globals()[“discount”] * discount)
 passpay(10000)


结果:请pay: 7200.0

有趣练习题

##持续获取用户输入,直到用户输入的字符串中既有字母又有数字。

def task5():
    while 1:
        str1 = input("输入一个既有字母又有数字的消息:")
        # 做检测
        is_have_char = False
        is_have_num = False
        for i in range(len(str1)):
            if "0" <= str1[i] <= "9":
                is_have_num = True
            if "A" <= str1[i] <= "Z" or "a" <= str1[i] <= "z":
                is_have_char = True
            # 为了性能 提前中断
            if is_have_char and is_have_num:
                break
        else:
            continue
        break
    return str1


# print(task5())

有一个长度为62的字符列表,请为这个列表随机赋值(a-z A-Z 0-9)要求每个下标位置的字符都不重复。

def task6():
    chars = []
    while len(chars) != 62:
        # 随机出62个字符当中的一个
        # ord(字符)可以得到该字符的数字表示
        random_char_num = random.randint(ord("0"), ord("z"))
        # 48-57 65-90 97-122
        if ord("9") < random_char_num < ord("A") or ord("Z") < random_char_num < ord("a"):
            continue
        # chr(数字)->字符
        char = chr(random_char_num)
        if char not in chars:
            chars.append(char)
    return chars


# print(task6())

写一个函数将整型数字(int 不含0)转换为汉字大写格式。例:111=>壹佰壹拾壹

说明:

# 167891111
# 使用列表:
# 汉字列表:["零","壹","贰","叁","肆","伍","陆","柒","捌","玖"]
#            0    1    2    3    。。。。                 9
# "167891111"=>"1"->汉字[int("1")]
# 单位:["圆", "拾", "佰", "仟", "萬", "拾", "佰", "仟", "亿", "拾", "佰", "仟", "萬"]
#        0    1    2     3   4    5    6    7.....
# "167891111"
#  012345678
#  876543210:len-1-i:单位[len-1-i]

解题:

def task7(num):
    # 做数字位数的限定
    if len(str(num)) > 13:
        print("暂时不支持13位以上的转换!")
        return
    # 准备汉字和单位列表
    zhs = ["零", "壹", "贰", "叁", "肆", "伍", "陆", "柒", "捌", "玖"]
    units = ["圆", "拾", "佰", "仟", "萬", "拾", "佰", "仟", "亿", "拾", "佰", "仟", "萬"]
    # 设定转换后的字符串
    ret = ""
    num_str = str(num)
    for i in range(len(num_str)):
        ch = num_str[i]
        ret += zhs[int(ch)] + units[len(num_str) - 1 - i]
    return ret


# print(task7(123456789))

随机赋值,不重复操作

# 14.使用Random类给一个列表的所有元素赋随机初值(不重复)。
def task14():
    # 为10个位置的列表找到10个各不相同的随机数(1-10)
    # 为每一个位置找不重复的随机数
    # 定义一个列表
    # nums = [0] * 10
    # for i in range(len(nums)):
    #     while 1:
    #         # 为i位置找到和之前位置不同的随机数
    #         random_num = random.randint(1,10)
    #         if random_num not in nums:
    #             nums[i] = random_num
    #             break
    # nums = []
    # while len(nums) != 10:
    #     random_num = random.randint(1, 10)
    #     if random_num not in nums:
    #         nums.append(random_num)
    # return nums

    # 为数字找随机位置:洗牌算法
    # nums = [0] * 10
    # for i in range(1, 11):
    #     while 1:
    #         # i:1-10
    #         # 随机位置
    #         random_index = random.randint(0, len(nums) - 1)
    #         # 找到了一个随机位置
    #         # 判断该位置是否已经有数字了
    #         if nums[random_index] == 0:
    #             nums[random_index] = i
    #             break

    # 使用随机打乱方式获取不重复数据
    nums = list(range(1, 11))
    random.shuffle(nums)
    return nums


# print(task14())

猜数游戏

# 23.猜数游戏
# 1.随机出现一个数(范围自定义) 作为答案
# 2.提示用户输入并根据答案和用户输入的大小关系输出大了? 小了?
# 3.5次机会
# 4.可以重复玩
# 5.根据猜对的次数进行评价
# 6.无论对错 请告知答案
#
def repeat_play():  # 定义重复玩函数
    while True:
        guess_nums()
        an = input("是否退出游戏?(y/n)")
        if an == "y":
            return False


def guess_nums():  # 猜数游戏
    i = 0
    game_answer = random.randint(0, 100)
    while i < 5:
        user_answer = in_int()
        # print(game_answer)
        i += 1
        if user_answer == game_answer:
            break
        print(f"第{i}次你猜{user_answer}大了") if user_answer > game_answer else print(f"第{i}次你猜{user_answer}小了")
    print(f"本轮随机答案为:{game_answer}")
    print(f"{i}次猜对,欧皇附体!") if i == 0 else print(f"{i}次猜对,运气真好!") \
        if i == 1 else print(f"{i}次猜对,猜的不错!") if 4 > i > 1 else print(f"{i}次猜对,太及时了!") \
        if i == 4 else print("本次游戏未猜中,继续努力!")


# guess_nums()
# repeat_play()