Python学习笔记(水桶谜题代码学习)——应用*符号解包列表所有元素传递给函数用法

原文:http://inventwithpython.com/bigbookpython/project81.html

在这个小游戏中,有三个水桶,容量分别是3升、5升和8升,需要在其中一个水桶中收集正好四升水。规则是:

1、桶只能被清空、完全装满或倒入另一个桶中。

2、把A水桶中的水倒入B水桶,要么把B水桶倒满,要么A桶水量不足,水量不足则倒空

例如,A桶8 升满桶水,把A桶里面的水倒入 空的3升的C桶,会把C倒满有3升,A桶剩余5升。

再如,B桶5升满桶水,A桶8升空,把B桶水倒入A桶,B桶将全部倒入A桶,A桶成5升,B为空。

# ///
# 水桶谜题 
# 
#

import sys

print('''
***水桶谜题***
* 有三个水桶,容量分别是8升/5升/3升,需在其中一个水桶中收集正好四升水。
* 规则是:
* 1、桶只能被清空、完全装满或倒入另一个桶中。
* 2、把A桶水倒入B桶,要么把B桶倒满,要么A桶水量不足,水量不足则倒空。
* 3、例如:A桶有8升水,把A桶水倒入空的3升的C桶,会把C倒满,A桶剩余5升。
* 4、再如:B桶有5升水,A桶8升空,把B桶水倒入A桶,B桶将倒空,A桶成5升
''')

GOAL = 4         # 桶中装恰好4升水才能获胜
step_count = 0   # 记录玩家解决此问题所用的步数

# 每个桶的初始水量,使用一个字典记录
waterInBucket = {'8':0,'5':0,'3':0}

while True:   # 游戏主循环
    # 显示桶内当前水量
    print()
    print(f'请让任意一个水桶中的水正好有 {GOAL} L 。')
    print('当前3个水桶的水量如下:')
    
    waterDisplay = []    # 用于存储表示水或者剩余空间的字符串,打印此列表元素显示3个水桶的盛水状态

    # 8L 水桶
    for i in range(1,9):
        if waterInBucket['8'] < i:
            waterDisplay.append('      ')   # 表示空白
        else:
            waterDisplay.append('wwwwww')   # 表示加水

    # 5L 水桶
    for i in range(1,6):
        if waterInBucket['5'] < i:
            waterDisplay.append('      ')   # 表示空白
        else:
            waterDisplay.append('wwwwww')   # 表示加水
    
    # 3L 水桶
    for i in range(1,4):
        if waterInBucket['3'] < i:
            waterDisplay.append('      ')   # 表示空白
        else:
            waterDisplay.append('wwwwww')   # 表示加水

    # 显示3个桶的状体
    print('''
8|{7}|
7|{6}|
6|{5}|
5|{4}|  5|{12}|
4|{3}|  4|{11}|
3|{2}|  3|{10}|  3|{15}|
2|{1}|  2|{9}|  2|{14}|
1|{0}|  1|{8}|  1|{13}|
 +------+   +------+   +------+
   8L         5L         3L                                              
'''.format(*waterDisplay))    # 解包waterDisplay列表
    
    # 判断是否有满足目标水量的桶
    for water_amount in waterInBucket.values():
        if water_amount == GOAL:
            print(f'恭喜您完成目标!您解决此问题使用了: {step_count} 步。')
            sys.exit()  
    
    # 让玩家选择一个动作来处理一个桶
    print('你可以选择下面的动作:')
    print('  输入 F/f 键回车,来给一个桶加满水。')
    print('  输入 E/e 键回车,清空一个桶。')
    print('  输入 P/p 键回车,把一个桶的水倒入另一个桶(加完或者加满)。')
    print('  输入 Q/q 键回车,将退出系统。')

    while True:   # 不断询问,直到玩家输入有效的动作
        # 玩家输入的都为大写
        player_action = input('>').upper()   
        if player_action.startswith('Q'):
            print('再见,欢迎有空再玩!')
            sys.exit()
        
        if player_action in ('F','E','P'):
            break   # 跳出循环,让玩家选择接下来的动作

        # 玩家没有输入有效的选择,提醒并返回循环
        print('输入无效键,请输入 F,E,P, 或者Q键。')

    # 让玩家选择一个桶
    while True:         # 继续询问,直到玩家选择有效的桶
        print('请输入数字选择一个桶,:8, 5, 3,或者输入Q退出。')
        src_bucket_id = input('>').upper()

        # if src_bucket_id.startswith('Q'):
        #     print('谢谢玩这款游戏。')
        #     sys.exit()
        
        if src_bucket_id in ('8','5','3'):
            break   # 跳出循环,否则循环继续,让玩家继续输入有效的选择
    
    # 执行选择的动作
    if player_action == 'F':
        # 将所选水桶的水量设置为最大容量
        src_bucket_id_size = int(src_bucket_id)  
        waterInBucket[src_bucket_id] = src_bucket_id_size
        step_count += 1  # 玩家执行步数 +1

    elif player_action == 'E':   # 清空水桶
        waterInBucket[src_bucket_id] = 0   
        step_count += 1
        
    elif player_action == 'P':
        # 让玩家选择要倒入水的桶
        while True:  # 继续询问,直到玩家选择有效的桶
            print('请选择要倒入水的桶,输入水桶编号:8, 5 ,或者3')
            dst_bucket_id = input('>').upper()
            if dst_bucket_id in ('8','5','3'):
                break    # 玩家输入了一个有效的水桶退出循环,否则继续循环
        
        dst_bucket_id_size = int(dst_bucket_id)  # 最大容量
        dst_bucket_empty_space = dst_bucket_id_size - waterInBucket[dst_bucket_id]  # 要倒入水的水桶剩余最大容量
        water_in_src_bucket = waterInBucket[src_bucket_id]  # 倒出水的桶的水的容量
        amount_to_pour = min(dst_bucket_empty_space,water_in_src_bucket)  # 实际倒出的水的水量

        # 倒出水的水桶水量更新
        waterInBucket[src_bucket_id] -= amount_to_pour
    
        # 倒入水的水桶水量更新
        waterInBucket[dst_bucket_id] += amount_to_pour

        # 计数
        step_count += 1
    
    elif player_action == 'C':
        pass # 未来可扩展动作

两个知识点:

print('''
8|{7}|
7|{6}|
6|{5}|
5|{4}|  5|{12}|
4|{3}|  4|{11}|
3|{2}|  3|{10}|  3|{15}|
2|{1}|  2|{9}|  2|{14}|
1|{0}|  1|{8}|  1|{13}|
 +------+   +------+   +------+
   8L         5L         3L                                              
'''.format(*waterDisplay))    # 格式化字符串,解包waterDisplay列表,填入相应数值

(1)字符串格式化

上面print()函数打印表示3个水桶及水量的字符串,字符串模板中有16个变量,分别代表3个水桶的每一升水容量,如果是1升水是“wwwwww”(6个'w'),否则是"      "(6个空格),这16个变量取自变量waterDisplay(一个列表),使用format()函数格式化。

(2)元素解包:当你想将列表的每个元素传递给一个函数时,可以使用 *

def add(a, b):  
    return a + b  
  
numbers = [1, 2]  
result = add(*numbers)  # result 的值为 3

上面小游戏代码就是使用了*符号,解包waterDisplay列表,把列表中的所有元素传递给了format()函数,从而格式化3个水桶字符穿。

你可能感兴趣的:(学习,笔记,python)