• (4)函数之三元表达式、列表推导式、生成器表达式、递归、匿名函数、内置函数


    一、三元表达式、列表推导式、生成器表达式


    1,三元表达式

    name = input('姓名>>: ')
    res = 'SB' if name == '菊花四' else 'NB'
    print(res)

    2,列表推导式

    # 1,示例
    egg_list=[]
    for i in range(10):
        egg_list.append('鸡蛋%s' %i)
    
    egg_list = ['鸡蛋%s' %i for i in range(10)]
    
    # 2,语法
    [expression for item1 in iterable1 if condition1
    for item2 in iterable2 if condition2
    ...
    for itemN in iterableN if conditionN
    ]
    类似于
    res = []
    for item1 in iterable1:
        if condition1:
            for item2 in iterable2:
                if condition2
                    ...
                    for itemN in iterableN:
                        if conditionN:
                            res.append(expression)
    
    # 3,优点:方便,改变了编程习惯,可称之为声明式编程

    3,生成器表达式

    # 1,把列表推导式的[]换成()就是生成器表达式
    
    # 2,示例:生一筐鸡蛋变成给你一只老母鸡,用的时候就下蛋,这也是生成器的特性
    
    >>> chicken = ('鸡蛋%s' %i for i in range(5))
    >>> chicken
    <generator object <genexpr> at 0x10143f200>
    >>> next(chicken)
    '鸡蛋0'
    >>> list(chicken)     # 因chicken可迭代,因而可以转成列表
    ['鸡蛋1', '鸡蛋2', '鸡蛋3', '鸡蛋4',]
    
    # 3,优点:省内存,一次只产生一个值在内存中

    4,练习题

    1)将names = ['zixi','oldsix_sb','oldfour','dalaoer']中的名字全部变大写。

    2)将names = ['zixi','oldsix_sb','oldfour','dalaoer']中以sb结尾的名字过滤掉,然后保存剩下的名字长度。

    3)求文件a.txt中最长的行的长度(长度按字符个数算,需要使用max函数)

    4)求文件a.txt中总共包含的字符个数?思考为何在第一次之后的n次sum求和得到的结果为0?(需要使用sum函数)

    5)思考题

    with open('a.txt') as f:
        g = (len(line) for line in f)
    print(sum(g))     # 为何报错?

    6)文件shopping.txt 内容如下:

    mac,20000,3
    lenovo,3000,10
    tesla,1000000,10
    chicken,200,1

    求总共花了多少钱?

    打印出所有商品的信息,格式为[{'name':'xxx','price':333,'count':3},...]

    求单价大于10000的商品信息,格式同上。

    # 题目一
    names = ['zixi','oldsix_sb','oldfour','dalaoer']
    names = [name.upper() for name in names]
    
    # 题目二
    names = ['zixi','oldsix_sb','oldfour','dalaoer']
    names = [len(name) for name in names if not name.endswith('sb')]
    
    # 题目三
    with open('a.txt',encoding='utf-8') as f:
        print(max(len(line) for line in f))
    
    # 题目四
    with open('a.txt', encoding='utf-8') as f:
        print(sum(len(line) for line in f))
        print(sum(len(line) for line in f))     # 求包换换行符在内的文件所有的字符数,为何得到的值为0?
        print(sum(len(line) for line in f))     # 求包换换行符在内的文件所有的字符数,为何得到的值为0?
    
    # 题目五(略)
    
    # 题目六:每次必须重新打开文件或seek到文件开头,因为迭代完一次就结束了
    with open('a.txt',encoding='utf-8') as f:
        info = [line.split() for line in f]
        cost = sum(float(unit_price)*int(count) for _,unit_price,count in info)
        print(cost)
    
    
    with open('a.txt',encoding='utf-8') as f:
        info = [{
            'name': line.split()[0],
            'price': float(line.split()[1]),
            'count': int(line.split()[2]),
        } for line in f]
        print(info)
    
    
    with open('a.txt',encoding='utf-8') as f:
        info = [{
            'name': line.split()[0],
            'price': float(line.split()[1]),
            'count': int(line.split()[2]),
        } for line in f if float(line.split()[1]) > 10000]
        print(info)
    result

    二、递归与二分法


    1,递归调用的定义

    递归调用是函数嵌套调用的一种特殊形式,函数在调用时,直接或间接调用了自身,就是递归调用。

    # 直接调用本身
    def f1():
        print('from f1')
        f1()
    f1()
    
    # 间接调用本身
    def f1():
        print('from f1')
        f2()
    
    def f2():
        print('from f2')
        f1()
    f1()
    
    # 调用函数会产生局部的名称空间,占用内存,因为上述这种调用会无需调用本身,python解释器的
    #内存管理机制为了防止其无限制占用内存,对函数的递归调用做了最大的层级限制
    
    # 可以修改递归最大深度
    import sys
    sys.getrecursionlimit()
    sys.setrecursionlimit(2000)
    
    def f1(n):
        print('from f1',n)
        f1(n+1)
    f1(1)
    
    # 虽然可以设置,但是因为不是尾递归,仍然要保存栈,内存大小一定,不可能无限递归,而且无限制地
    # 递归调用本身是毫无意义的,递归应该分为两个明确的阶段,回溯与递推
    详解

    2,递归调用应该分为两个明确的阶段:递推,回溯 

    # 1,递归调用应该包含两个明确的阶段:回溯,递推
    """
    回溯就是从外向里一层一层递归调用下去,
    回溯阶段必须要有一个明确地结束条件,每进入下一次递归时,问题的规模都应该有所减少(否则,单纯地重复调用自身是毫无意义的)
    
    递推就是从里向外一层一层结束递归
    """
    
    # 2,示例+图解。。。
    # salary(5)=salary(4)+300
    # salary(4)=salary(3)+300
    # salary(3)=salary(2)+300
    # salary(2)=salary(1)+300
    # salary(1)=100
    #
    # salary(n)=salary(n-1)+300     n>1
    # salary(1) =100                n=1
    
    def salary(n):
        if n == 1:
            return 100
        return salary(n-1)+300
    
    print(salary(5))

    3,python中的递归效率低且没有尾递归优化

    # python中的递归
    python中的递归效率低,需要在进入下一次递归时保留当前的状态,在其他语言中可以有解决方法:尾递归优化,即在函数的最后一步(而非最后一行)调用自己。
    但是python又没有尾递归,且对递归层级做了限制
    
    
    # 总结递归的使用:
    1,必须有一个明确的结束条件
    
    2,每次进入更深一层递归时,问题规模相比上次递归都应有所减少
    
    3,递归效率不高,递归层次过多会导致栈溢出(在计算机中,函数调用是通过栈(stack)这种数据结构实现的,每当进入一个函数调用,
    栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出)

    4,二分法

    想从一个按照从小到大排列的数字列表中找到指定的数字,遍历的效率太低,用二分法(算法的一种,算法是解决问题的方法)可以极大低缩小问题规模。

    l=[1,2,10,30,33,99,101,200,301,311,402,403,500,900,1000] 
    # 从小到大排列的数字列表
    
    def search(n,l):
        print(l)
        if len(l) == 0:
            print('not exists')
            return
        mid_index=len(l) // 2
        if n > l[mid_index]:
            # in the right
            l=l[mid_index+1:]
            search(n,l)
        elif n < l[mid_index]:
            # in the left
            l=l[:mid_index]
            search(n,l)
        else:
            print('find it')
    
    
    search(3,l)
    实现类似in的效果
    l = [1,2,10,30,33,99,101,200,301,402]
    
    def search(num,l,start=0,stop=len(l)-1):
        if start <= stop:
            mid = start+(stop-start)//2
            print('start:[%s] stop:[%s] mid:[%s] mid_val:[%s]' %(start,stop,mid,l[mid]))
            if num > l[mid]:
                start = mid+1
            elif num < l[mid]:
                stop = mid-1
            else:
                print('find it',mid)
                return
            search(num,l,start,stop)
        else:         # 如果stop > start则意味着列表实际上已经全部切完,即切为空
            print('not exists')
            return
    
    search(301,l)
    
        
    实现类似于l.index(30)的效果

    三、匿名函数


    1,什么是匿名函数?

    # 匿名就是没有名字
    def func(x,y,z=1):
        return x+y+z
    
    # 匿名
    lambda x,y,z = 1:x+y+z     
    # 与函数有相同的作用域,但是匿名意味着引用计数为0,使用一次就释放,除非让其有名字
    
    func=lambda x,y,z=1:x+y+z 
    func(1,2,3)
    # 让其有名字就没有意义

    2,有名字的函数与匿名函数的对比

    # 有名函数与匿名函数的对比
    有名函数:循环使用,保存了名字,通过名字就可以重复引用函数功能
    
    匿名函数:一次性使用,随时随时定义
    
    应用:max,min,sorted,map,reduce,filter

    四、内置函数


    注意:

    # 内置函数id()可以返回一个对象的身份,返回值为整数。这个整数通常对应与该对象在内存中的位置,
    # 但这与python的具体实现有关,不应该作为对身份的定义,即不够精准,最精准的还是以内存地址为准。
    # is运算符用于比较两个对象的身份,等号比较两个对象的值,内置函数type()则返回一个对象的类型。
    
    # 更多内置函数:https://docs.python.org/3/library/functions.html?highlight=built#ascii 

    # 字符串可以提供的参数 's' None
    >>> format('some string','s')
    'some string'
    >>> format('some string')
    'some string'
    
    # 整形数值可以提供的参数有 'b' 'c' 'd' 'o' 'x' 'X' 'n' None
    >>> format(3,'b')      # 转换成二进制
    '11'
    >>> format(97,'c')     # 转换unicode成字符
    'a'
    >>> format(11,'d')     # 转换成10进制
    '11'
    >>> format(11,'o')     # 转换成8进制
    '13'
    >>> format(11,'x')     # 转换成16进制 小写字母表示
    'b'
    >>> format(11,'X')     # 转换成16进制 大写字母表示
    'B'
    >>> format(11,'n')     # 和d一样
    '11'
    >>> format(11)         # 默认和d一样
    '11'
    
    # 浮点数可以提供的参数有 'e' 'E' 'f' 'F' 'g' 'G' 'n' '%' None
    >>> format(314159267,'e')          # 科学计数法,默认保留6位小数
    '3.141593e+08'
    >>> format(314159267,'0.2e')     # 科学计数法,指定保留2位小数
    '3.14e+08'
    >>> format(314159267,'0.2E')     # 科学计数法,指定保留2位小数,采用大写E表示
    '3.14E+08'
    >>> format(314159267,'f')         # 小数点计数法,默认保留6位小数
    '314159267.000000'
    >>> format(3.14159267000,'f')     # 小数点计数法,默认保留6位小数
    '3.141593'
    >>> format(3.14159267000,'0.8f')     # 小数点计数法,指定保留8位小数
    '3.14159267'
    >>> format(3.14159267000,'0.10f')     # 小数点计数法,指定保留10位小数
    '3.1415926700'
    >>> format(3.14e+1000000,'F')      # 小数点计数法,无穷大转换成大小字母
    'INF'
    
    # g的格式化比较特殊,假设p为格式中指定的保留小数位数,先尝试采用科学计数法格式化,得到幂指数exp,如果-4<=exp<p,则采用小数计数法,并保留p-1-exp位小数,否则按小数计数法计数,并按p-1保留小数位数
    >>> format(0.00003141566,'.1g')     # p=1,exp=-5 ==》 -4<=exp<p不成立,按科学计数法计数,保留0位小数点
    '3e-05'
    >>> format(0.00003141566,'.2g')     # p=1,exp=-5 ==》 -4<=exp<p不成立,按科学计数法计数,保留1位小数点
    '3.1e-05'
    >>> format(0.00003141566,'.3g')     # p=1,exp=-5 ==》 -4<=exp<p不成立,按科学计数法计数,保留2位小数点
    '3.14e-05'
    >>> format(0.00003141566,'.3G')     # p=1,exp=-5 ==》 -4<=exp<p不成立,按科学计数法计数,保留0位小数点,E使用大写
    '3.14E-05'
    >>> format(3.1415926777,'.1g')     # p=1,exp=0 ==》 -4<=exp<p成立,按小数计数法计数,保留0位小数点
    '3'
    >>> format(3.1415926777,'.2g')     # p=1,exp=0 ==》 -4<=exp<p成立,按小数计数法计数,保留1位小数点
    '3.1'
    >>> format(3.1415926777,'.3g')     # p=1,exp=0 ==》 -4<=exp<p成立,按小数计数法计数,保留2位小数点
    '3.14'
    >>> format(0.00003141566,'.1n')     # 和g相同
    '3e-05'
    >>> format(0.00003141566,'.3n')     # 和g相同
    '3.14e-05'
    >>> format(0.00003141566)         # 和g相同
    '3.141566e-05'
    
                
    format(了解即可)
    字典的运算:最小值,最大值,排序
    salaries={
        'zixi':3000,
        'oldsix':100000000,
        'oldfour':10000,
        'dalaoer':2000
    }
    
    迭代字典,取得是key,因而比较的是key的最大和最小值
    >>> max(salaries)
    'oldsix'
    >>> min(salaries)
    'dalaoer'
    
    可以取values,来比较
    >>> max(salaries.values())
    >>> min(salaries.values())
    但通常我们都是想取出,工资最高的那个人名,即比较的是salaries的值,得到的是键
    >>> max(salaries,key=lambda k:salary[k])
    'oldsix'
    >>> min(salaries,key=lambda k:salary[k])
    'dalaoer'
    
    
    
    也可以通过zip的方式实现
    salaries_and_names=zip(salaries.values(),salaries.keys())
    
    先比较值,值相同则比较键
    >>> max(salaries_and_names)
    (100000000, 'oldsix')
    
    
    salaries_and_names是迭代器,因而只能访问一次
    >>> min(salaries_and_names)
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    ValueError: min() arg is an empty sequence
    
    
    
    sorted(iterable,key=None,reverse=False)
    !!!lambda与内置函数结合使用!!!
    # 1,语法
    # eval(str,[,globasl[,locals]])
    # exec(str,[,globasl[,locals]])
    
    # 2,区别
    # 示例一:
    s='1+2+3'
    print(eval(s))     # eval用来执行表达式,并返回表达式执行的结果
    print(exec(s))     # exec用来执行语句,不会返回任何值
    '''
    6
    None
    '''
    
    # 示例二:
    print(eval('1+2+x',{'x':3},{'x':30}))     # 返回33
    print(exec('1+2+x',{'x':3},{'x':30}))     # 返回None
    
    # print(eval('for i in range(10):print(i)'))     # 语法错误,eval不能执行表达式
    print(exec('for i in range(10):print(i)'))
    eval与exec
    complie(str,filename,kind)
    # filename:用于追踪str来自于哪个文件,如果不想追踪就可以不定义
    # kind可以是:single代表一条语句,exec代表一组语句,eval代表一个表达式
    
    s = 'for i in range(10):print(i)'
    code = compile(s,'','exec')
    exec(code)
    
    
    s = '1+2+3'
    code = compile(s,'','eval')
    eval(code)
    complie(了解)

    五、阶段练习


    练习一:

    1,文件内容如下,标题为:姓名,性别,年纪,薪资:

    zixi male 18 3000
    oldsix male 38 30000
    oldfour female 28 20000
    dalaoer female 28 10000
    
    要求:
    从文件中取出每一条记录放入列表中,
    列表的每个元素都是{'name':'zixi','sex':'male','age':18,'salary':3000}的形式

    2,根据1得到的列表,取出薪资最高的人的信息。

    3,根据1得到的列表,取出最年轻的人的信息。

    4,根据1得到的列表,将每个人的信息中的名字映射成首字母大写的形式。

    5,根据1得到的列表,过滤掉名字以a开头的人的信息。

    6,使用递归打印斐波那契数列(前两个数的和得到第三个数,如:0 1 1 2 3 4 7...)。

    7,一个嵌套很多层的列表,如l=[1,2,[3,[4,5,6,[7,8,[9,10,[11,12,13,[14,15]]]]]]],用递归取出所有的值。

    # 题目一
    with open('db.txt') as f:
        items = (line.split() for line in f)
        info = [{'name':name,'sex':sex,'age':age,'salary':salary} 
              for name,sex,age,salary in items]
    
    print(info)
    
    
    # 题目二
    print(max(info,key=lambda dic:dic['salary']))
    
    
    # 题目三
    print(min(info,key=lambda dic:dic['age']))
    
    
    # 题目四
    info_new = map(lambda item:{'name':item['name'].capitalize(),
                                'sex':item['sex'],
                                'age':item['age'],
                                'salary':item['salary']},info)
    
    print(list(info_new))
    
    
    # 题目五
    g = filter(lambda item:item['name'].startswith('a'),info)
    print(list(g))
    
    
    # 题目六
    # 非递归
    def fib(n):
        a,b = 0,1
        while a < n:
            print(a,end=' ')
            a,b = b,a+b
        print()
    
    fib(10)
    
    # 递归
    def fib(a,b,stop):
        if a > stop:
            return
        print(a,end=' ')
        fib(b,a+b,stop)
    
    fib(0,1,10)
    
    
    # 题目七
    l = [1,2,[3,[4,5,6,[7,8,[9,10,[11,12,13,[14,15]]]]]]]
    
    def get(seq):
        for item in seq:
            if type(item) is list:
                get(item)
            else:
                print(item)
    get(l)
    result

    练习二:

    作业需求:
    
    模拟实现一个ATM + 购物商城程序
    
        1,额度 15000或自定义
        2,实现购物商城,买东西加入 购物车,调用信用卡接口结账
        3,可以提现,手续费5%
        4,每月22号出账单,每月10号为还款日,过期未还,按欠款总额 万分之5 每日计息
        5,支持多账户登录
        6,支持账户间转账
        7,记录每月日常消费流水
        8,提供还款接口
        9,ATM记录操作日志 
        10,提供管理接口,包括添加账户、用户额度,冻结账户等。。。
        11,用户认证用装饰器
    示例代码 https://github.com/triaquae/py3_training/tree/master/atm 
    
    简易流程图:https://www.processon.com/view/link/589eb841e4b0999184934329  
    result
  • 相关阅读:
    2018年全国多校算法寒假训练营练习比赛(第二场)F
    牛客练习赛2 A
    牛客练习赛1 C
    牛客练习赛1 B
    vb编程代码大全
    javascript编程代码笔记
    391.FANUC宏程序编程
    宏程序编程实例,简单易懂
    Java类与类之间的关系详细介绍
    C++虚继承时的构造函数的讲解
  • 原文地址:https://www.cnblogs.com/zoling7/p/13426632.html
Copyright © 2020-2023  润新知