zoukankan      html  css  js  c++  java
  • Python全栈开发【基础四】

    本节内容:

    • 匿名函数(lambda)
    • 函数式编程(map,filter,reduce)
    • 文件处理
    • 迭代器
    • 三元表达式
    • 列表解析与生成器表达式
    • 生成器

    匿名函数 

    lambda表达式:对于简单的函数,存在一种简便的表示方式,即lambda表达式 

    1 #这段代码
    2 def calc(n):
    3     return n**n
    4 print(calc(10))
    5  
    6 #换成匿名函数
    7 calc = lambda n:n**n
    8 print(calc(10))

    匿名函数主要是和其它函数搭配使用

    举例:

     1 #########次方函数举例############
     2 res = map(lambda x:x**2,[1,5,7,4,8])
     3 for i in res:
     4     print(i)
     5  
     6 输出
     7 1
     8 25
     9 49
    10 16
    11 64

    高阶函数、函数式编程

    高阶函数

    满足俩个特性任意一个即为高阶函数

    1.函数的传入参数是一个函数名

    2.函数的返回值是一个函数名

    一、map函数

     1 ###############map函数################
     2 array=[1,3,4,71,2]
     3 
     4 ret=[]
     5 for i in array:
     6     ret.append(i**2)
     7 print(ret)
     8 
     9 #如果我们有一万个列表,那么你只能把上面的逻辑定义成函数
    10 def map_test(array):
    11     ret=[]
    12     for i in array:
    13         ret.append(i**2)
    14     return ret
    15 
    16 print(map_test(array))
    17 
    18 #如果我们的需求变了,不是把列表中每个元素都平方,还有加1,减一,那么可以这样
    19 def add_num(x):
    20     return x+1
    21 def map_test(func,array):
    22     ret=[]
    23     for i in array:
    24         ret.append(func(i))
    25     return ret
    26 
    27 print(map_test(add_num,array))
    28 #可以使用匿名函数
    29 print(map_test(lambda x:x-1,array))
    30 
    31 
    32 #上面就是map函数的功能,map得到的结果是可迭代对象
    33 print(map(lambda x:x-1,range(5)))

    二、filter函数

     真假值判断,如果为真就留下,为假则舍弃

    1 filter(lambda x:x>22, [55,11,22,33,])

    print(list(filter(lambda x:x>22, [55,11,22,33,])))

    结果:[55, 33]

    三、reduce函数

    对可迭代对象中的指合并操作

    reduce()函数也是Python内置的一个高阶函数。

    reduce()函数接收的参数和 map()类似,一个函数 f,一个list,但行为和 map()不同,reduce()传入的函数 f 必须接收两个参数,reduce()对list的每个元素反复调用函数f,并返回最终结果值。

    例如,编写一个f函数,接收x和y,返回x和y的和:

    def f(x, y):
        return x + y

    调用 reduce(f, [1, 3, 5, 7, 9])时,reduce函数将做如下计算:

    先计算头两个元素:f(1, 3),结果为4;
    再把结果和第3个元素计算:f(4, 5),结果为9;
    再把结果和第4个元素计算:f(9, 7),结果为16;
    再把结果和第5个元素计算:f(16, 9),结果为25;
    由于没有更多的元素了,计算结束,返回结果25。

    上述计算实际上是对 list 的所有元素求和。虽然Python内置了求和函数sum(),但是,利用reduce()求和也很简单。

    reduce()还可以接收第3个可选参数,作为计算的初始值。如果把初始值设为100,计算:

    reduce(f, [1, 3, 5, 7, 9], 100)

    结果将变为125,因为第一轮计算是:

    计算初始值和第一个元素:f(100, 1),结果为101

     1 from functools import reduce
     2 #合并,得一个合并的结果
     3 array_test=[1,2,3,4,5,6,7]
     4 array=range(100)
     5 
     6 #报错啊,res没有指定初始值
     7 def reduce_test(func,array):
     8     l=list(array)
     9     for i in l:
    10         res=func(res,i)
    11     return res
    12 
    13 # print(reduce_test(lambda x,y:x+y,array))
    14 
    15 #可以从列表左边弹出第一个值
    16 def reduce_test(func,array):
    17     l=list(array)
    18     res=l.pop(0)
    19     for i in l:
    20         res=func(res,i)
    21     return res
    22 
    23 print(reduce_test(lambda x,y:x+y,array))
    24 
    25 #我们应该支持用户自己传入初始值
    26 def reduce_test(func,array,init=None):
    27     l=list(array)
    28     if init is None:
    29         res=l.pop(0)
    30     else:
    31         res=init
    32     for i in l:
    33         res=func(res,i)
    34     return res
    35 
    36 print(reduce_test(lambda x,y:x+y,array))
    37 print(reduce_test(lambda x,y:x+y,array,50))

    文件处理

    open函数,该函数用于文件处理

    文件句柄 = open('文件路径','模式')

    打开文件时,需要指定文件路径和以何等方式打开文件,打开后,即可获取该文件句柄,日后通过此文件句柄对该文件操作。

    打开文件的模式有:

    • r ,只读模式【默认模式,文件必须存在,不存在则抛出异常】
    • w,只写模式【不可读;不存在则创建;存在则清空内容】
    • x, 只写模式【不可读;不存在则创建,存在则报错】
    • a, 追加模式【可读;   不存在则创建;存在则只追加内容】

    "+" 表示可以同时读写某个文件

    • r+, 读写【可读,可写】
    • w+,写读【可读,可写】
    • x+ ,写读【可读,可写】
    • a+, 写读【可读,可写】

     "b"表示以字节的方式操作

    • rb  或 r+b
    • wb 或 w+b
    • xb 或 w+b
    • ab 或 a+b
    •  1 f = open ("1.txt","rb")
       2 data = f.read()
       3 print(data)
       4 print(data.decode("utf8"))
       5 
       6 f = open ("1.txt","wb")
       7 f.write('ocean
      '.encode('utf8'))
       8 
       9 
      10 f = open ("1.txt","ab")
      11 f.write('ocean
      '.encode('utf8'))

     注:以b方式打开时,读取到的内容是字节类型,写入时也需要提供字节类型,不能指定编码

      1 ####### r 读 #######
      2 
      3 f = open('test.log','r',encoding='utf-8')
      4 
      5 a = f.read()
      6 
      7 print(a)
      8 
      9  
     10 
     11 ###### w 写(会先清空!!!) ######
     12 
     13 f = open('test.log','w',encoding='utf-8')
     14 
     15 a = f.write('ocean')
     16 
     17 print(a)    #返回字符
     18 
     19  
     20 
     21 ####### a 追加(指针会先移动到最后) ########
     22 
     23 f = open('test.log','a',encoding='utf-8')
     24 
     25 a = f.write('good
    boy')
     26 
     27 print(a)    #返回字符
     28 
     29  
     30 
     31 ####### 读写 r+ ########
     32 
     33 f = open('test.log','r+',encoding='utf-8')
     34 
     35 a = f.read()
     36 
     37 print(a)
     38 
     39 f.write('ocean')
     40 
     41  
     42 
     43 ##### 写读 w+(会先清空!!!) ######
     44 
     45 f = open('test.log','w+',encoding='utf-8')
     46 
     47 a = f.read()
     48 
     49 print(a)
     50 
     51 f.write('ocean')
     52 
     53  
     54 
     55 ######## 写读 a+(指针先移到最后) #########
     56 
     57 f = open('test.log','a+',encoding='utf-8')
     58 
     59 f.seek(0)   #指针位置调为0
     60 
     61 a = f.read()
     62 
     63 print(a)
     64 
     65 b = f.write('ocean')
     66 
     67 print(b)
     68 
     69  
     70 
     71 ####### rb #########
     72 #字节只读方式
     73 f = open('test.log','rb')
     74 
     75 a = f.read()
     76 
     77 print(str(a,encoding='utf-8'))
     78 
     79  
     80 
     81 # ######## ab #########
     82 
     83 f = open('test.log','ab')
     84 
     85 f.write(bytes('老男人
    ',encoding='utf-8'))
     86 
     87 f.write(b'oldman')
     88 
     89 #结果:
     90 #老男人
     91 #oldman
     92  
     93 
     94 ##### 关闭文件 ######
     95 
     96 f.close()
     97 
     98  
     99 
    100 ##### 内存刷到硬盘 #####
    101 
    102 f.flush()
    103 
    104  
    105 
    106 ##### 获取指针位置 #####
    107 
    108 f.tell()
    109 
    110  
    111 
    112 ##### 指定文件中指针位置 #####
    113 
    114 f.seek(0)
    115 
    116  
    117 ###### 读取全部内容(如果设置了size,就读取size字节) ######
    118 
    119 f.read()
    120 
    121 f.read(9)
    122 
    123  
    124 
    125 ###### 读取一行 #####
    126 
    127 f.readline()
    128 
    129  
    130 
    131 ##### 读到的每一行内容作为列表的一个元素 #####
    132 
    133 f.readlines()

     

    注意:read(3)代表读取3个字符,其余的文件内光标移动都是以字节为单位如seek,tell,read,truncate

    迭代器

     一、什么是迭代器协议?

    1.迭代器协议是指:对象必须提供一个next方法,执行该方法要么返回迭代中的下一项,要么就引起一个StopIteration异常,以终止迭代 (只能往后走不能往前退

    2.可迭代对象:实现了迭代器协议的对象(如何实现:对象内部定义一个__iter__()方法

    3.协议是一种约定,可迭代对象实现了迭代器协议,python的内部工具(如for循环,sum,min,max函数等)使用迭代器协议访问对象。

    二、for循环本质:循环所有对象,全都是使用迭代器协议。

    (字符串,列表,元组,字典,集合,文件对象)这些都不是可迭代对象,只不过在for循环式,调用了他们内部的__iter__方法,把他们变成了可迭代对象

    然后for循环调用可迭代对象的__next__方法去取值,而且for循环会捕捉StopIteration异常,以终止迭代。

     1 l=['a','b','c']
     2 #一:下标访问方式
     3 print(l[0])
     4 print(l[1])
     5 print(l[2])
     6 # print(l[3])#超出边界报错:IndexError
     7 
     8 #二:遵循迭代器协议访问方式
     9 diedai_l=l.__iter__()
    10 print(diedai_l.__next__())
    11 print(diedai_l.__next__())
    12 print(diedai_l.__next__())
    13 # print(diedai_l.__next__())#超出边界报错:StopIteration
    14 
    15 #三:for循环访问方式
    16 #for循环l本质就是遵循迭代器协议的访问方式,先调用diedai_l=l.__iter__()方法,或者直接diedai_l=iter(l),然后依次执行diedai_l.next(),直到for循环捕捉到StopIteration终止循环
    17 #for循环所有对象的本质都是一样的原理
    18 
    19 for i in l:#diedai_l=l.__iter__()
    20     print(i) #i=diedai_l.next()
    21 
    22 #四:用while去模拟for循环做的事情
    23 diedai_l=l.__iter__()
    24 while True:
    25     try:
    26         print(diedai_l.__next__())
    27     except StopIteration:
    28         print('迭代完毕了,循环终止了')
    29         break

    三元表达式

    三元运算又称三元运算(三目运算),是对简单的条件语句的缩写。

    1 #result = 值1 if 条件 else 值2
    2   
    3 # 如果条件成立,那么将 “值1” 赋值给result变量,否则,将“值2”赋值给result变量
    4 
    5 #三元表达式
    6 name='marry'
    7 name='jack'
    8 res='woman' if name == 'marry' else 'man'
    9 print(res)

     列表解析与生成器表达式


    总结: 1
    egg_list=['鸡蛋%s' %i for i in range(10)] #列表解析 2 3 4 #列表解析直接生成列表,占用内存空间,用生成器表达式可避免浪费内存 5 6 laomuji=('鸡蛋%s' %i for i in range(10))#生成器表达式 7 print(laomuji) 8 print(next(laomuji)) #next本质就是调用__next__ 9 print(laomuji.__next__()) 10 print(next(laomuji))

    1.把列表解析的[]换成()得到的就是生成器表达式

    2.列表解析与生成器表达式都是一种便利的编程方式,只不过生成器表达式更节省内存

    3.Python不但使用迭代器协议,让for循环变得更加通用。大部分内置函数,也是使用迭代器协议访问对象的。例如, sum函数是Python的内置函数,该函数使用迭代器协议访问对象,而生成器实现了迭代器协议,所以,我们可以直接这样计算一系列值的和:

    1 sum(x ** 2 for x in xrange(4))

    而不用多此一举的先构造一个列表:

    1 sum([x ** 2 for x in xrange(4)])

    生成器

    一、什么是生成器? 

    可以理解为一种数据类型,这种数据类型自动实现了迭代器协议(其他的数据类型需要调用自己内置的__iter__方法),所以生成器就是可迭代对象

    二、生成器分类及在python中的表现形式:(Python有两种不同的方式提供生成器)

    1.生成器函数:常规函数定义,但是,使用yield语句而不是return语句返回结果。yield语句一次返回一个结果,在每个结果中间,挂起函数的状态,以便下次重它离开的地方继续执行

    2.生成器表达式:类似于列表推导,但是,生成器返回按需产生结果的一个对象,而不是一次构建一个结果列表

    三、为何使用生成器之生成器的优点

    Python使用生成器对延迟操作提供了支持。所谓延迟操作,是指在需要的时候才产生结果,而不是立即产生结果。这也是生成器的主要好处。

    四、生成器小结:

    1.是可迭代对象

    2.实现了延迟计算,省内存

    3.生成器本质和其他的数据类型一样,都是实现了迭代器协议,只不过生成器附加了一个延迟计算省内存的好处,其余的可迭代对象没有这点好处。

     1 def lay_eggs(num):
     2     egg_list=[]
     3     for egg in range(num):
     4         egg_list.append('蛋%s' %egg)
     5     return egg_list
     6 
     7 yikuangdan=lay_eggs(10) #我们拿到的是蛋
     8 print(yikuangdan)
     9 
    10 
    11 def lay_eggs(num):
    12     for egg in range(num):
    13         res='蛋%s' %egg
    14         yield res
    15         print('下完一个蛋')
    16 
    17 laomuji=lay_eggs(10)#我们拿到的是一只母鸡
    18 print(laomuji)
    19 print(laomuji.__next__())
    20 print(laomuji.__next__())
    21 print(laomuji.__next__())
    22 egg_l=list(laomuji)
    23 print(egg_l)
    24 #演示只能往后不能往前
    25 #演示蛋下完了,母鸡就死了

     

     

  • 相关阅读:
    Java后端工程师的学习技术栈
    ltp 分析 fail testcase
    程序员这个职业需要具备的素养
    你真的愿意到了50岁还要做编程吗?
    程序员的学习和积累
    程序员写博客的缘由
    VS2010生成文件
    从菜鸟到专家的五步编程语言学习法
    程序设计的18大原则
    怎样的代码才算是干净的代码?
  • 原文地址:https://www.cnblogs.com/mocean/p/6133376.html
Copyright © 2011-2022 走看看