Python编程学习-基础笔记03

五、复习

5.1 图书管理系统

在这个章节里,所写的是基于线性脚本,纯代码堆砌,随着学习的深入,会在后面章节,通过模块化的方式来优化。

'''
图书管理系统:
    至少5本书
    library = [{'bookname':xxx,'Author':xxx,'price':xxx,'number':xx},{},{}]
要求:
1,借书
    不存在
    存在 -->
        判断数量:number 多本的情况
        可能不同作者: 根据作者选择要借的书
        借书成功后:修改库存数量
2,还书:
    输入书名和作者 -->判断是否存在 --> 检查书名和作者,还书成功,修改库存数量number
3,查询(可以根据书名或作者查询)
    1,输入1 或 2 来选择按书名或作者查询
    2,不区分:
        search = input('书名或作者'), 比较2次
        if search in namelist:
            pass
        elif search in authorlist:
            pass
4,查看所有
5,退出
    提示读者借到或归还的书数目!
'''
#定义一个数据库来存放图书
library = [
    {'bookname':'红楼梦','Author':'曹雪芹','price':88,'number':10},
    {'bookname':'废艺斋集稿','Author':'曹雪芹','price':28,'number':5},
    {'bookname':'世界是平的','Author':'托马斯·弗里德曼','price':28,'number':5},
    {'bookname':'三国演义','Author':'罗贯中','price':68,'number':10},
    {'bookname':'儒林外史','Author':'吴敬梓','price':38,'number':0},
    {'bookname':'西游记','Author':'吴承恩','price':68,'number':6},
    {'bookname':'西厢记','Author':'王实甫','price':38,'number':4},
    {'bookname':'百年孤独','Author':'加西亚·马尔克斯','price':48,'number':3},
    {'bookname':'简.爱','Author':'夏洛蒂·勃朗特','price':28,'number':3},
    {'bookname':'封神榜','Author':'许仲琳','price':58,'number':0},
    {'bookname':'Python入门到精通','Author':'Forrest','price':58,'number':3},
    {'bookname':'Python入门到精通','Author':'Jessica','price':38,'number':3},
    {'bookname':'Python入门到精通','Author':'Jim','price':38,'number':3},
    {'bookname':'水浒传','Author':'施耐庵','price':78,'number':5}
]
print('*************欢迎进入天大图书管理系统*************')
#获取书名并将将其放入一个列表
bookname_list = []
for i in library:
    bookname_list.append(i['bookname'])
#获取作者名并将将其放入一个列表
author_list = []
for i in library:
    author_list.append(i['Author'])
#容器接收本次借阅的图书
borrowed_booklist = []
#容器接收本次归还的图书
returned_booklist = []
#定义循环控制变量
flag = True
while flag:
    print('\n1,借书 \n2,还书 \n3,查询(可以根据书名或作者查询)\n4,查看所有书籍 \n5,退出')
    oper = input('请输入你要使用的功能编号:')
    #借书
    if oper == '1':
        #输入你要借阅的书名
        book = input('请输入你要借阅的书名:').strip()
        #判断图书是否存在系统,不存在即返回提示
        if book in bookname_list:
            #将检索到是书的数量累加给num
            num = 0
            #先定义个空的作者集合,添加书的作者,如果同一书名对应多位作者,这里可供选择
            author_sublist = []
            #定义一个新的图书列表
            sub_library = []
            #遍历图书列表,添加数量和作者到变量
            for i in library:
                #找到书名后,累加数量和作者
                if i['bookname'] == book:
                    #将搜索到的书添加到新的列表
                    sub_library.append(i)
                    #将作者名直接追加到列表
                    author_sublist.append(i['Author'])
                    #书的数量累加
                    num += i['number']
            #判断图书存在且数量大于0
            if num > 0:
                # 判断书名相同,但是作者不同
                if len(author_sublist) >= 2:
                    print('你打算借阅的图书有不同的作者,请选择你需要借阅的作者名:')
                    #列出不同的作者,供读者选择
                    for j in range(len(author_sublist)):
                        print(f'{j + 1}:{author_sublist[j]}')
                    # 读者选择作者,去掉空格
                    choice = input('请选择作者:').strip()
                    #选择之后回到数据库查询
                    for k in library:
                        #判断书名和作者名都满足要求
                        if k['bookname'] == book and k['Author'] == choice:
                            #读者选择是否借阅
                            select = input('请确认是否借阅:是Y/否N')
                            # 确认借书
                            if select.lower() == 'y':
                                #确认借阅,先减库存
                                k['number'] -= 1
                                #将借到的书放入借书车
                                borrowed_booklist.append(k)
                                #给读者返回提示信息和剩余库存量
                                print(f"你已成功借阅图书:<<{book}>>,作者:{k['Author']}")
                                print(f"该图书:<<{book}>>,作者:{k['Author']},还剩下库存量{k['number']}本")
                            # 放弃借书
                            elif select.lower() == 'n':
                                print('放弃借阅图书!')
                else:
                    print(f"你打算借阅的图书<<{book}>>作者是:{author_sublist[0]},请确认是否借阅!")
                    for k in library:
                        #判断是否存在多名作者
                        if k['bookname'] == book:
                            #读者选择是否借阅
                            select = input('请确认是否借阅:是Y/否N')
                            # 确认借书
                            if select.lower() == 'y':
                                #确认借阅,先减库存
                                k['number'] -= 1
                                #将借到的书放入借书车
                                borrowed_booklist.append(k)
                                #给读者返回提示信息和剩余库存量
                                print(f"你已成功借阅图书:<<{book}>>,作者:{k['Author']}")
                                print(f"该图书:<<{book}>>,作者:{k['Author']},还剩下库存量{k['number']}本")
                            # 放弃借书
                            elif select.lower() == 'n':
                                print('放弃借阅图书!')
            else:
                print(f'对不起,你要借阅的图书:<<{book}>>没有库存,请选择借阅其他图书!')

        else:
            print(f'对不起,检索不到你要借阅的图书:<<{book}>>,请重新输入!')
    #还书
    elif oper == '2':
        # 输入你要归还的书名和作者名
        book = input('请输入你要归还的书名:').strip()
        author = input('请输入你要归还的作者名:').strip()
        #判断图书是否存在系统,不存在即返回提示
        if (book in bookname_list) and (author in author_list):
            for k in library:
                if k['bookname'] == book and k['Author'] == author:
                    # 读者选择是否归还
                    select = input('请确认是否归还图书:是Y/否N')
                    # 确认归还
                    if select.lower() == 'y':
                        # 确认归还,加库存量
                        k['number'] += 1
                        # 将归还的书放入还书列表
                        returned_booklist.append(k)
                        # 给读者返回提示信息和剩余库存量
                        print(f"你已成功归还图书:<<{book}>>,作者:{k['Author']}")
                        print(f"该图书:<<{book}>>,作者:{k['Author']},当前库存量{k['number']}本")
                    # 放弃借书
                    elif select.lower() == 'n':
                        print('放弃归还图书!')

        else:
            #提示错误信息
            print(f"请检查你输入的书名:<<{book}>>或作者:{author}是否正确!!")
    #查询
    elif oper == '3':
        #输入查询内容
        search = input('请输入要查询的书名或作者:').strip()
        #判断如果读者输入的是书名
        if search in bookname_list:
            print('为你检索到如下信息:')
            #遍历数据库查找结果
            for k in library:
                if k['bookname'] == search:
                    #返回检索信息
                    print(f"书名:《{k['bookname']}》,作者:{k['Author']},价格:{k['price']},库存:{k['number']}本")
        # 判断如果读者输入的是作者名
        elif search in author_list:
            print('为你检索到如下信息:')
            #遍历数据库查找结果
            for k in library:
                if k['Author'] == search:
                    #返回检索信息
                    print(f"书名:《{k['bookname']}》,作者:{k['Author']},价格:{k['price']},库存:{k['number']}本")
        else:
            #提示错误信息
            print(f"未检索到你查询的内容,请检查你输入关键字'{search}'是否正确!!")
    #查看所有
    elif oper == '4':
        #提示信息
        print('当前数据库所有图书列表如下:')
        for k in library:
            print(f"书名:《{k['bookname']}》,作者:{k['Author']},价格:{k['price']},库存:{k['number']}本")
    elif oper == '5':
        # 读者确认是否退出系统
        select = input('请确认是否退出系统:是Y/否N\n')
        # 确认退出
        if select.lower() == 'y':
            #判断已借到书
            if len(borrowed_booklist) > 0:
                #提示读者本次借阅或归还的图书
                print(f"你本次共借阅{len(borrowed_booklist)}本图书!")
            elif len(returned_booklist) > 0:
                #提示读者本次借阅或归还的图书
                print(f"你本次共归还{len(returned_booklist)}本图书!")
            else:
                print('你本次没有借或还书的操作!')
            # 给读者提示信息
            print(f">>>>>>>>正在退出系统,欢迎再次光临>>>>>>>>")
            flag = False
        # 放弃退出
        elif select.lower() == 'n':
            print('请继续浏览!')
    else:
        #提示错误信息
        print('输入格式有误,请重新输入!')

5.2 列表推导

''''
列表推导式:最终得到一个列表
格式1:[i for i in 可迭代的]
格式2:[i for i in 可迭代的 if 条件]
格式3:[结果1 if 条件 esle 结果2 for i in 可迭代的 ]
'''
#得到一个列表
list1 =[]
for i in range(1,21):
    list1.append(i)
print(list1) #[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20]

#list2 == list1,先循环,再赋值给i,加到外层的列表
list2 = [i for i in range(1,21)]
print(list2)

list2 = [i+2 for i in range(1,10)]
print(list2) #[3, 4, 5, 6, 7, 8, 9, 10, 11]

#格式2:[i for i in 可迭代的 if 条件]
list2 = [i for i in range(1,21) if i%2 == 0]
print(list2) #[2, 4, 6, 8, 10, 12, 14, 16, 18, 20]

#提取
list3 = ['hello','55','world','88','beauty','love']
#取出里面的单词
list4 = [word for word in list3 if word.isalpha()]
print(list4) #['hello', 'world', 'beauty', 'love']

list4 = [word.title() if word.startswith('h') else word.upper() for word in list3 ]
print(list4) #['Hello', '55', 'WORLD', '88', 'BEAUTY', 'LOVE']

#1-20,按3位截取
a = [x for x in range(1,21)]
b = [a[i:i+3] for i in range(0,len(a),3)]
print(b) #[[1, 2, 3], [4, 5, 6], [7, 8, 9], [10, 11, 12], [13, 14, 15], [16, 17, 18], [19, 20]]

5.3 集合和字典推导式

#集合推导式,{} 类似于列表推导式,在列表推导式的基础上增加了去重的功能

list = [1,3,2,5,9,8,5,6,3,2,7,6,5]

set1 = {x for x in list}
print(set1) #{1, 2, 3, 5, 6, 7, 8, 9}

set1 = {x for x in list if x > 5}
print(set1) #{8, 9, 6, 7}

set1 = {x-2 for x in list if x > 5}
print(set1) #{4, 5, 6, 7}

#字典推导式
dict1 = {'a':'A','b':'B','c':'C','d':'C',}
#key 和value对换,key是唯一的
new_dict = {value:key for key,value in dict1.items()}
print(new_dict) #{'A': 'a', 'B': 'b', 'C': 'd'}

5.4 生成器

5.4.1 生成器基础

通过列表生成式(列表推导式),我们可以直接创建一个列表,但是,收到内存现在,列表的容量是有限的而且,创建一个包含100万个元素的列表,不仅占用很大的内存空间,如果我们仅仅需要访问到前面几个元素,那后面大多数元素占用的空间都白白浪费。所以,如果列表元素可以按照某种算法推算出来,那我们是否可以在循环的过程中不断推算出后续的元素呢?这样就不必创建完整的list,从而节省大量的空间。在python中,这种一边循环一边计算的机制,成为生成器:Generator。

'''
得到生成器的方式:
    1,通过列表推导式得到生成器

'''

#[0, 3, 6, 9, 12, 15, 18, 21, 24, 27]
new_list = [x*3 for x in range(10)]
print(new_list)
#得到生成器
g = (x*3 for x in range(5))
print(type(g)) #<class 'generator'>
#方式一,通过调用__next__ 方式得到元素
print(g.__next__())
print(g.__next__())
print(g.__next__())
#方式二,next(生成器对象) 系统内置函数,没调用一次next,则会产生一个元素
print(next(g))
print(next(g))
print(next(g)) #StopIteration 超出生成器的范围,再调用就会抛出StopIteration异常

# 通过循环来处理生成器异常
# 得到生成器
g = (x*3 for x in range(10))

while True:
    try:
        e = next(g)
        print(e)
    except:
        print('没有更多的元素了!')
        break

5.4.2 借助函数完成生成器

#定义生成器的方式二:借助函数完成
#只要函数中出现了yield 关键字,说明函数就不是函数啦,变成生成器啦
'''
步骤:
    1,定义一个函数,函数中使用yield 关键字
    2,调用函数,接收调用的结果
    3,得到的结果就算是生成器
    4,借助next() 或__next__
'''
 def func():
     n = 0
     while True:
         n += 1
         yield n
#斐波那契数列
def fib(length):
    a,b = 0,1
    n = 0
    while n < length:
        # print(b)
        yield b #return b + pause
        a,b = b, a+b
        n += 1
    return '没有更多元素了!'  #StopIteration: 没有更多元素了!

#调用
g = fib(8)

print(next(g))

5.4.3 生成器应用

#定义生成器的方式二:借助函数完成
'''
生成器方法:
    1,next() 或__next__ 获取下一个元素
    2,send(value): 向每次生成器调用中传值
        注意:第一次调用,必须先传None
'''
def gen():
    n = 0
    while n < 5:
        temp = yield n
        print('temp:',temp)
        for x in range(temp):
            print('---->',x)
        n += 1
    return '没有更多元素!'
#调用
g = gen()
r = g.send(None)
print(r)
n1 = g.send(2)
print('n1:',n1)
n2 = g.send(3)
print('n2:',n2)

#结果分析
'''
0  --》 第一次,循环开始,yield 返出 n =0,此时被r = g.send(None) 接收
temp: 2 --> temp 接收外部send 的值 2
----> 0  --》 for循环2次
----> 1
n1: 1   --》 第二次循环,n1 接收到yield 返出的值 1
temp: 3 temp: 2 --> temp 接收外部send 的值 3
----> 0  --》 for循环3次
----> 1
----> 2
n2: 2 --》 第三次循环,n2 接收到yield 返出的值 2, 没有后续的值,yield处于暂停,然后整个程序退出
'''

5.4.3 生成器应用之多任务

'''
进程 --> 线程 --> 协程
生成器:generator
1,定义生成器的方式
    通过列表推导式
    借助函数 + yield
2,产生元素:
    next() 每次调用都会产生一个新元素,元素产生完毕再调用就会抛异常
    生成器自己的方法:
        g.__next__()
        g.send(value)
'''
#生成器应用--》 多任务
def task1(n):
    for i in range(n):
        print(f'正在做第{i}个任务')
        yield

def task2(n):
    for i in range(n):
        print(f'正在听第{i}首歌')
        yield
g1 = task1(5)
g2 = task2(5)

while True:
    try: #交替执行任务
        next(g1)
        next(g2)
    except:
        print('任务完成')
        break

5.5 迭代器

可迭代对象:生成器、元组、列表、字典、集合、字符串
如何判断一个对象是否可迭代?
迭代是访问集合元素的一种方式,迭代器是一个可以记住遍历遍历的位置的对象。迭代器对象从集合的第一个元素开始访问,直到所以元素访问完成迭代器只能往前,不能后退。
可以被next()函数调用,并不断返回下一个值的对象称为迭代器:iterator
可迭代的 是不是肯定是 迭代器?
生成器是可迭代的,也是迭代器,
list是可迭代的,但不是迭代器
list --> iter(list) --> 迭代器
生成器与迭代器 的关系:
生成器是迭代器的子集,它不需要借助函数,即可成为一个迭代器
列表、元组、字典、集合和字符串是可迭代的,需要借助iter()函数,将其变为迭代器

from collections.abc import Iterable
list1 = [1,2,5,3,4,6,2,1]
#判断是否是可迭代对象
f = isinstance(list1,Iterable)
print(f) #True
g = [x+1 for x in range(10)]
f = isinstance(g,Iterable)
print(f) #True
#可迭代对象转位迭代器
list1 = [1,2,3,4,5]
list1 = iter(list1) #通过iteration
print(next(list1)) #1
posted @ 2022-06-22 11:52  逆流的鱼2016  阅读(47)  评论(0编辑  收藏  举报