一、函数定义
二、作用:提高代码的可重用性和可维护性(代码层次结构更清晰)。
三、函数返回值:
四、实例
# 定义根据月份,判断季节的方法.
def get_season(month):
if month < 1 or month > 12:
return "输入有误"
if month <= 3:
return "春天"
if month <= 6:
return "夏天"
if month <= 9:
return "秋天"
return "冬天"
print(get_season(5))
print(get_season(15))
# 练习:定义检查列表是否具有相同元素的方法
def is_repeating(list_target):
for r in range(len(list_target) - 1):
for c in range(r + 1, len(list_target)):
if list_target[r] == list_target[c]:
return True # 退出方法(可以退出两层循环)
return False
re01 = is_repeating([3,44,5])
re01 = is_repeating([3,44,5,44])
print(re01)
#练习:定义在控制台中显示直角三角形的函数
def print_triangle(height,char):
"""
打印三角型
:param height: 三角形高度
:param char: 组成三角型的字符
:return:
"""
for r in range(height):
for c in range(r+1):
print(char, end="")
print()
print_triangle(8,"*")
print_triangle(5,"#")
print_triangle(4,"?")
# 练习:定义在控制台中显示矩形的函数
def print_rect(r_count,c_count,char):
for r in range(r_count):# 0 1 2
for c in range(c_count):#01234 01234 01234
print(char,end = "") # 在一行输出
print() # 换行
print_rect(2,3,"*")
print_rect(5,2,"#")
# 参数:变量
# 函数调用者 告诉 函数定义者的信息
# 1. 参照下列代码,定义判断是否是奇数的方法.
# number = int(input("请输入整数:"))
# if number % 2 == 1:
# print("奇数")
# else:
# print("偶数")
def is_uneven(number):
# 方法1:传统方法
# if number % 2 == 1:
# return True
# else:
# return False
# 方法2:条件表达式
# return True if number % 2 == 1 else False
# 方法3:bool表达式
return number % 2 == 1
re = is_uneven(6)
print(re)
# 2.参照下列代码,定义根据月份返回天数的方法.
# 要求:考虑2月,如果是闰年返回29天,否则返回28天.
# month = int(input("请输入月份:"))
# if month < 1 or month > 12:
# print("输入有误")
# elif month == 2:
# print("28天")
# elif month == 4 or month == 6 or month == 9 or month == 11:
# print("30天")
# else:
# print("31天")
# 向外返回的结果return,其类型应该统一.
# 方法1:传统方法:
# def get_day_by_month(year,month):
# if month < 1 or month > 12:
# return 0
# elif month == 2:
# if year % 4 == 0 and year % 100 != 0 or year % 400 ==0:
# return 29
# else:
# return 28
# elif month == 4 or month == 6 or month == 9 or month == 11:
# return 30
# else:
# return 31
# 方法2:定义函数嵌套法+条件表达式精简代码
def is_leap_year(year):
return year % 4 == 0 and year % 100 != 0 or year % 400 == 0
def get_day_by_month(year, month):
if month < 1 or month > 12:
return 0
if month == 2:
return 29 if is_leap_year(year) else 28
if month in (4, 6, 9, 11):
return 30
return 31
day = get_day_by_month(2019,13)
print(day)
# 3.参照下列代码,定义获取最小值方法.
# min = list01[0]
# for i in range(1, len(list01)):
# # 发现更大的,则替换假设的.
# if min > list01[i]:
# min = list01[i]
# print(min)
def get_min(list_target):
min = list_target[0]
for i in range(1, len(list_target)):
if min > list_target[i]:
min = list_target[i]
return min
min = get_min([2, 3, 45, 1, 9, 3])
print(min)
# 4. 定义函数,判断字符串中存在的中文字符数量.
# 中文编码范围:0x4E00 ord(字符) 0x9FA5
def get_chinese_char_count(str_target):
count = 0
for item in str_target:
if 0x4E00 <= ord(item) <= 0x9FA5:
count += 1
return count
count = get_chinese_char_count("a你sid你f好ln")
print(count)
# 5. 扩展练习(定义函数,返回指定范围内的素数)
# 例如:1--100
# 方法1:传统方法
# def get_prime(bengin,end):
# list_prime = []
# for number in range(bengin,end+1):
# if number < 2:
# pass
# else:
# for i in range(2, number):
# if number % i == 0:
# break # 如果有结论了,就不需要在和后面的数字比较了
# else:
# list_prime.append(number)
# return list_prime
# 方法2:自定义函数相互之间嵌套法
def get_prime(bengin, end):
"""
生成指定范围内的所有素数
:param bengin: 开始
:param end: 结束
:return: 范围内的所有素数
"""
list_prime = []
for number in range(bengin, end + 1):
if is_prime(number):
list_prime.append(number)
return list_prime
def is_prime(number):
"""
判断是否为素数
:param number: 需要判断的数
:return: 是不是素数
"""
if number < 2:
return False # 不是素数
# 判断能否被中间的数字整除
for i in range(2, number):
if number % i == 0:
return False # 不是素数
return True # 是素数
primes = get_prime(5, 100)
print(primes)
primes = get_prime(-5, 50)
print(primes)
五、函数重构代码思想
案例:未重构前代码如下
shang_pin_info = {
101: {"name": "屠龙刀", "price": 10000},
102: {"name": "倚天剑", "price": 10000},
103: {"name": "九阴白骨爪", "price": 8000},
104: {"name": "九阳神功", "price": 9000},
105: {"name": "降龙十八掌", "price": 8000},
106: {"name": "乾坤大挪移", "price": 10000}
}
ding_dan = []
def gou_wu():
"""
购物
:return:
"""
while True:
item = input("1键购买,2键结算。")
if item == "1":
for key, value in shang_pin_info.items():
print("编号:%d,名称:%s,单价:%d。" % (key, value["name"], value["price"]))
while True:
cid = int(input("请输入商品编号:"))
if cid in shang_pin_info:
break
else:
print("该商品不存在")
count = int(input("请输入购买数量:"))
ding_dan.append({"cid": cid, "count": count})
print("添加到购物车。")
elif item == "2":
zong_jia = 0
for item in ding_dan:
shang_pin = shang_pin_info[item["cid"]]
print("商品:%s,单价:%d,数量:%d." % (shang_pin["name"], shang_pin["price"], item["count"]))
zong_jia += shang_pin["price"] * item["count"]
while True:
qian = float(input("总价%d元,请输入金额:" % zong_jia))
if qian >= zong_jia:
print("购买成功,找回:%d元。" % (qian - zong_jia))
ding_dan.clear()
break
else:
print("金额不足.")
gou_wu()
重构后代码如下:
dict_commodity_info = {
101: {"name": "屠龙刀", "price": 10000},
102: {"name": "倚天剑", "price": 10000},
103: {"name": "九阴白骨爪", "price": 8000},
104: {"name": "九阳神功", "price": 9000},
105: {"name": "降龙十八掌", "price": 8000},
106: {"name": "乾坤大挪移", "price": 10000}
}
list_order = []
def select_menu():
"""
菜单选择
"""
while True:
item = input("1键购买,2键结算。")
if item == "1":
buying()
elif item == "2":
settlement()
def settlement():
"""
结算
:return:
"""
print_order()
total_price = get_total_price()
pay(total_price)
def pay(total_price):
"""
支付
:param total_price: 需要支付的价格
:return:
"""
while True:
money = float(input("总价%d元,请输入金额:" % total_price))
if money >= total_price:
print("购买成功,找回:%d元。" % (money - total_price))
list_order.clear()
break
else:
print("金额不足.")
def print_order():
"""
打印订单
:return:
"""
for item in list_order:
commodity = dict_commodity_info[item["cid"]]
print("商品:%s,单价:%d,数量:%d." % (commodity["name"], commodity["price"], item["count"]))
def get_total_price():
"""
计算总价
:return:
"""
total_price = 0
for item in list_order:
commodity = dict_commodity_info[item["cid"]]
total_price += commodity["price"] * item["count"]
return total_price
def buying():
"""
购买
:return:
"""
print_commodity()
order = create_order()
list_order.append(order)
print("添加到购物车。")
def create_order():
"""
创建订单
:return:
"""
while True:
cid = int(input("请输入商品编号:"))
if cid in dict_commodity_info:
break
else:
print("该商品不存在")
count = int(input("请输入购买数量:"))
return {"cid": cid, "count": count}
def print_commodity():
"""
打印商品
:return:
"""
for key, value in dict_commodity_info.items():
print("编号:%d,名称:%s,单价:%d。" % (key, value["name"], value["price"]))
# 程序入口
select_menu()
注:虽然代码量变大了,但是在项目编程时,逻辑更清晰,维护更便捷
六、函数内存图示例(即可变/不可变类型在传参时的区别):
示例1:
"""
内存图1.0
不可变对象传参
"""
def fun01(num01):
num01 = 2
print("num01:" + str(num01))# 2
number01 = 1
# 调用方法,在内存中开辟空间(栈帧)
# 栈帧中定义该方法内部创建的变量
# 方法执行完毕后,栈帧立即释放.
fun01(number01)
print("number01:" + str(number01))# 1
示例2:
"""
内存图2.0
可变对象传参
"""
def fun01(list_target):
list_target[0] = 2
print("list_target[0]:" + str(list_target[0]))
list_number = [1,2]
fun01(list_number)
print("list_number:" + str(list_number[0]))
示例3
def fun01(a,b,c):
a = 100
# 修改的是列表对象,
b[0] = 200
# 修改的是栈帧中的变量
c = 300
num01 = 1
list01 = [2]
list02 = [3]
fun01(num01,list01,list02)
print(num01)#? 1
print(list01)#? [200]
print(list02)#? [3]
七、2048核心算法:
""" 2048 核心算法:降維 """ # 练习1:定义函数,将零元素移动到末尾 # [2,0,2,0] --> [2,2,0,0] # [0,2,2,0] --> [2,2,0,0] # [0,4,2,4] --> [4,2,4,0] """ 方案1 def zero_to_end(list_target): # 1.将传入的列表中非零元素,拷贝到新列表中. # [2, 0, 2, 0] --> [2,2] --> [2,2,0,0] # [0, 4, 2, 4] -->[4, 2, 4] -->[4, 2, 4 ,0] # 2. 根据为零元素的数量,在新列表中添加零元素 # [2, 0, 2, 0] --> [2,2] new_list = [item for item in list_target if item != 0] # [2, 2] --> [2,2,0,0] new_list += [0] * list_target.count(0) # 3. 将新列表中元素赋值给传入的列表 list_target[:] = new_list """ # 方案2 def zero_to_end(list_target): # 从后往前判断,如果零元素,则删除,在末尾追加零元素 # [2, 0, 2, 0] --> [2, 2] --> [2, 2,0,0] for i in range(len(list_target) - 1, -1, -1): if list_target[i] == 0: del list_target[i] list_target.append(0) # list01 = [2, 0, 2, 0] # zero_to_end(list01) # print(list01) # 练习2:定义合并函数 # [2,2,0,0] --> [4,0,0,0] # [2,0,2,0] --> [4,0,0,0] # [2,0,0,2] --> [4,0,0,0] # [2,2,2,0] --> [4,2,0,0] def merge(list_target): # [2,0,2,0] --> [2,2,0,0] [2,2,2,0] zero_to_end(list_target) # [2,2,0,0] --> [4,0,0,0] [4,0,2,0] for i in range(len(list_target) -1): # 相邻且相同 if list_target[i] == list_target[i+1]: list_target[i] += list_target[i+1] list_target[i+1] = 0 zero_to_end(list_target)# [4,0,2,0] --> [4,2,0,0] # list01 = [2,2,2,0] # merge(list01) # print(list01) # 练习3:将二维列表,以表格的格式显示在控制台中 list01 = [ [2,0,0,2], [2,2,0,0], [2,0,4,4], [4,0,0,2], ] def print_map(map): for r in range(len(map)): for c in range(len(map[r])): print(map[r][c],end = " ") print() # print_map(list01) #练习4:定义向左移动函数.11:50 """ [2,0,0,2] [4,0,0,0] [2,2,0,0] [4,0,0,0] [2,0,4,4] [2,8,0,0] [4,0,0,2] [4,2,0,0] """ def move_left(map): # 获取第行 for r in range(len(map)): # 从左往右获取行 # 交给merge进行合并 merge(map[r]) def move_right(map): # 获取第行 for r in range(len(map)): # 从右往左获取行 # 交给merge进行合并 list_merge = map[r][::-1] merge(list_merge) map[r][::-1] = list_merge # 作业1:定义向上移动函数 # 从上往下获取列 # 交给合并方法 # 还给原列 # 作业2:定义向下移动函数 # 从下往上获取列 # 交给合并方法 # 还给原列 def move_up(list_target): list_merge = [] for c in range(len(list_target[0])): for r in range(len(list_target)): list_merge.append(list_target[r][c]) merge(list_merge) for r in range(len(list_target)): list_target[r][c] = list_merge[r] list_merge = [] def move_down(list_target): list_merge = [] for c in range(len(list_target[0])): for r in range(len(list_target) - 1, -1, -1): list_merge.append(list_target[r][c]) merge(list_merge) for r in range(len(list_target)): list_target[r][c] = list_merge[3 - r] list_merge = [] # move_left(list01) # move_right(list01) # move_up(list01) move_down(list01) print_map(list01)