• Python


    学廖雪峰老师Python教程笔记

    一、Python简介

    1、Python适合开发哪些类型的应用呢?

      首选是网络应用,包括网站、后台服务等等;

      其次是很多日常需要的小工具,包括系统管理员需要的脚本任务等等;

      另外就是把其他语言开发的程序再包装起来,方便使用。

    2、Python缺点

      运行速度慢、代码不能加密

    二、Python安装

    在Mac上安装Python

       目前,Python有两个版本,一个是2.x版本,一个是3.x版,这两个版本是不兼容的。

       如果你正在使用Mac,系统是OS X>=10.9,那么系统自带的Python版本是2.7。要安装最新的Python3.7,有两个方法:

        方法一:从Python官网下载Python 3.7的安装程序(网速慢的同学请移步国内镜像),双击运行并安装;

        方法二:如果安装了Homebrew,直接通过命令brew install python3安装即可。

    三、第一个Python程序

      1、安装好以后,输入Python回车进入Python的交互式命令行,100+200回车输出300,print('hello word')输出hello word

      也可以直接在命令行模式执行,Python hello.py执行Python脚本。

        2、直接运行py文件

        Windows上不可以像.exe文件那样直接运行.py文件。但是在Mac和Linux上是可以的,方法是在.py文件的第一行加上一个特殊的注释:

    #!/usr/bin/env python3
    
    print('hello, world')

    然后通过命令给hello.py以执行权限:

    chmod a+x hello.py

    就可以直接运行hello.py了,比如在Mac下运行:

    ./hello.py
    hello,word

    小结:1、用文本编辑器写Python程序,然后保存为后缀为.py的文件,就可以用Python直接运行这个程序了。

    2、Python的交互模式和直接运行.py文件有什么区别呢?

      直接输入Python进入交互模式,相当于启动了Python解释器,但是等待你一行一行的输入源代码,没输入一行就执行一行。

      直接运行.py文件相当于启动了Python解释器,然后一次性把.py文件的源代码给执行了,你是没有机会以交互的方式输入源代码的。

      用Python发程序,完全可以一边在文本编辑器里写代码,一边开一个交互式命令窗口,在写代码的过程中,把部分代码粘到命令行去验证,事半功倍!提前是得有个27‘的超大显示器!

    四、Python基础

    1、Python程序是大小写敏感的,如果写错了大小写,程序会报错。

    2、字符编码

    我们已经讲过了,字符串也是一种数据类型,但是,字符串比较特殊的是还有一个编码问题。

    因为计算机只能处理数字,如果要处理文本,就必须先把文本转换为数字才能处理。最早的计算机在设计时采用8个比特(bit)作为一个字节(byte),所以,一个字节能表示的最大的整数就是255(二进制11111111=十进制255),如果要表示更大的整数,就必须用更多的字节。比如两个字节可以表示的最大整数是65535,4个字节可以表示的最大整数是4294967295

    由于计算机是美国人发明的,因此,最早只有127个字符被编码到计算机里,也就是大小写英文字母、数字和一些符号,这个编码表被称为ASCII编码,比如大写字母A的编码是65,小写字母z的编码是122

    但是要处理中文显然一个字节是不够的,至少需要两个字节,而且还不能和ASCII编码冲突,所以,中国制定了GB2312编码,用来把中文编进去。

    你可以想得到的是,全世界有上百种语言,日本把日文编到Shift_JIS里,韩国把韩文编到Euc-kr里,各国有各国的标准,就会不可避免地出现冲突,结果就是,在多语言混合的文本中,显示出来会有乱码。

    char-encoding-problem

    因此,Unicode应运而生。Unicode把所有语言都统一到一套编码里,这样就不会再有乱码问题了。

    Unicode标准也在不断发展,但最常用的是用两个字节表示一个字符(如果要用到非常偏僻的字符,就需要4个字节)。现代操作系统和大多数编程语言都直接支持Unicode。

    现在,捋一捋ASCII编码和Unicode编码的区别:ASCII编码是1个字节,而Unicode编码通常是2个字节。

    字母A用ASCII编码是十进制的65,二进制的01000001

    字符0用ASCII编码是十进制的48,二进制的00110000,注意字符'0'和整数0是不同的;

    汉字已经超出了ASCII编码的范围,用Unicode编码是十进制的20013,二进制的01001110 00101101

    你可以猜测,如果把ASCII编码的A用Unicode编码,只需要在前面补0就可以,因此,A的Unicode编码是00000000 01000001

    新的问题又出现了:如果统一成Unicode编码,乱码问题从此消失了。但是,如果你写的文本基本上全部是英文的话,用Unicode编码比ASCII编码需要多一倍的存储空间,在存储和传输上就十分不划算。

    所以,本着节约的精神,又出现了把Unicode编码转化为“可变长编码”的UTF-8编码。UTF-8编码把一个Unicode字符根据不同的数字大小编码成1-6个字节,常用的英文字母被编码成1个字节,汉字通常是3个字节,只有很生僻的字符才会被编码成4-6个字节。如果你要传输的文本包含大量英文字符,用UTF-8编码就能节省空间:

    字符ASCIIUnicodeUTF-8
    A 01000001 00000000 01000001 01000001
    x 01001110 00101101 11100100 10111000 10101101

    从上面的表格还可以发现,UTF-8编码有一个额外的好处,就是ASCII编码实际上可以被看成是UTF-8编码的一部分,所以,大量只支持ASCII编码的历史遗留软件可以在UTF-8编码下继续工作。

    搞清楚了ASCII、Unicode和UTF-8的关系,我们就可以总结一下现在计算机系统通用的字符编码工作方式:

    在计算机内存中,统一使用Unicode编码,当需要保存到硬盘或者需要传输的时候,就转换为UTF-8编码。

    用记事本编辑的时候,从文件读取的UTF-8字符被转换为Unicode字符到内存里,编辑完成后,保存的时候再把Unicode转换为UTF-8保存到文件:

    浏览网页的时候,服务器会把动态生成的Unicode内容转换为UTF-8再传输到浏览器:

    所以你看到很多网页的源码上会有类似<meta charset="UTF-8" />的信息,表示该网页正是用的UTF-8编码。

    2、1个中文字符经过utf-8编码后通常会占用3个字节,而1个英文字符只占用1个字节。

    在操作字符串时,我们经常遇到str和bytes的互相转换。为了避免乱码问题,应当始终坚持使用utf-8编码对str和bytes进行转换。

    由于Python源代码也是一个文本文件,所以,当你的源代码中包含中文的时候,在保存源代码时,就需要务必指定保存为utf-8编码。当Python解释器读取源代码时,为了让他按utf-8编码读取,我们通常在文件开头写上这两行:

    #!/usr/bin/env python3
    # -*- coding: utf-8 -*-

    第一行注释是为了告诉Linux/OS X系统,这是一个Python可执行程序,Windows系统会忽略这个注释;

    第二行注释是为了告诉Python解释器,按照utf-8编码读取源代码,否则,你在源代码中写的中文输出可能会有乱码。

    申明了utf-8编码并不意味着你的.py文件就是utf-8编码,必须并且要确保文本编辑器正在使用utf-8 without BOM编码:

    如果.py文件本身使用utf-8编码,并且也申明了# -*- coding: utf-8 -*-,打开命令提示符测试就可以正常显示中文:

    3、格式化

    即字符替换:

    例:

    >>>'Hello, %s' % 'word'
    'Hello, word'
    >>>'Hi, %s, you have $%d.' % ('Michael', 1000000)
    'Hi, Michael, you have $1000000.'

    %s表示用字符串替换,%d表示用整数替换,%f表示用浮点数替换,有几个%?占位符,后面就跟几个变量或者值,顺序要对应好。如果只有一个%?,括号可以省略。

    例:

    #-*- coding:utf-8 -*-
    print('%2d-%02d' % (3,1))
    print('%.2f' % 3.1415926)

    如果你不太确定应该用什么,%s永远起作用,它会把任何数据类型转换为字符串:

    >>>'age: %s Gender: %s' % (25, True)
    'Age:25 Gender:True' 

    有些时候,字符串里面的%是一个普通字符怎么办?这个时候就需要转义,用%%来表示一个%

    >>> 'growth rate: %d %%' % 7
    'growth rate: 7 %'

    小结:

    Python 3的字符串使用Unicode,直接支持多语言。

    strbytes互相转换时,需要指定编码。最常用的编码是UTF-8。Python当然也支持其他编码方式,比如把Unicode编码成GB2312

    >>> '中文'.encode('gb2312')
    b'xd6xd0xcexc4'
    

    但这种方式纯属自找麻烦,如果没有特殊业务要求,请牢记仅使用UTF-8编码。

    格式化字符串的时候,可以用Python的交互式环境测试,方便快捷。

    4、使用list和tuple

    list

    1)创建列表:

    >>>classmates = ['gouzi',  'caihua',  'damao']
    >>>classmates
    ['gouzi',  'caihua',  'damao']

    2)获取list元素个数

    >>>len(classmates)
    3

    3)访问

    >>> classmates[0]
    'gouzi'
    >>> classmates[1]
    'caihua'
    >>> classmates[2]
    'damao'
    >>> classmates[3]
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    IndexError: list index out of range

    当索引超出了范围时,Python会报一个IndexError错误,所以,要确保索引不要越界,记得最后一个元素的索引是len(classmates) - 1

    如果要取最后一个元素,除了计算索引位置外,还可以用-1做索引,直接获取最后一个元素:

    >>> classmates[-1]
    'damao'

    以此类推,可以获取倒数第2个、倒数第3个:

    >>> classmates[-2]
    'caihua'
    >>> classmates[-3]
    'gouzi'
    >>> classmates[-4]
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    IndexError: list index out of range

    倒数第4个就越界了。

    4)、追加元素到末尾

    >>>classmates.append('erya')
    >>>classmates
    ['gouzi', 'caihua', 'damao', 'erya']

    也可以把元素插入到指定的位置,比如索引为1的位置:

    >>>classmates.insert(1, 'goudan')
    >>>classmates
    ['gouzi', 'goudan', 'caihua', 'damao', 'erya']

    5)、删除list末尾的元素,用pop()方法:

    >>>classmates.pop()
    'erya'
    >>>classmates
    ['gouzi', 'goudan', 'caihua', 'damao']

    要删除指定位置的元素,用pop(i)方法,其中i是索引位置:

    >>>classmates.pop(1)
    'goudan'
    >>>classmates
    ['gouzi', 'caihua', 'damao']

    6)、把某个元素替换成别的元素,可以直接赋值给对应的索引位置:

    >>>classmates[1] = 'yuanshuai'
    >>>classmates
    ['gouzi', 'yuanshuai', 'caihua']

    7)、list里面元素的数据类型也可以不同,比如:

    >>>L = ['Apple', 123, True]

    8)、list元素也可以是另外一个list,比如:

    >>> s = ['python', 'java', ['asp', 'php'], 'scheme']
    >>> len(s)
    4

    要拿到'php'可以写s[2][1],因此s可以看成是一个二维数组,类似的还有三维、四维...数组,不过很少用到。

    如果一个list中一个元素也没有,就是一个空的list,他的长度为0:

    tuple

    另一种有序列表叫元组:tuple。tuple和list非常类似,但是tuple一旦初始化就不能修改,比如同样是累出同学的名字:

    1)、创建

    >>>classmates = ('gouzi', 'caihua', 'erya')

    所以创建list使用[],创建tuple是使用()

    现在,classmates这个tuple不能变了,它也没有append(),insert()这样的方法。其他获取元素的方法和list是一样的,你可以正常地使用classmates[0],classmates[-1],但不能赋值成另外的元素。

    不可变的tuple有什么意义?因为tuple不可变,所以代码更安全。如果可能,能用tuple代替list就尽量用tuple。

    tuple的陷阱:当你定义一个tuple时,在定义的时候,tuple的元素就必须被确定下来,比如:

    >>> t = (1, 2)
    >>> t 
    (1, 2)

    如果定义一个空的tuple,可以写成():

    >>> t = ()
    >>> t
    ()

    但是要定义一个只有1个元素的tuple,如果你这么定义

    >>> t = (1)
    >>> t
    1

    定义的不是tuple,是1这个数!这是因为括号()既可以表示tuple,是1这个数!这是因为括号()既可以表示tuple,又可以表示数学公式中的小括号,这就产生了歧义,因此,Python规定,这种情况下,按小括号进行计算,计算结果自然是1。

    所以,只有1个元素的tuple定义事必须加一个逗号,来消除歧义:

    >>> t = (1, )
    >>> t
    (1, )

    python在显示只有一个元素的tuple时,也会加一个逗号,一面你误会成数学计算意义上的括号。

    最后来看一个‘可变的’tuple:

    >>> t = ('a', 'b', ['A', 'B'])
    >>> t[2][0] = 'X'
    >>> t[2][1] = 'Y'
    >>> t
    ('a', 'b', ['X', 'Y'])

    4、条件判断

    1)、小例子

    age = 20
    if age  >= 18:
        print('your age is', age)
        print('adult')
    elif age>=6:
      print('teenager')
    else:
      print('your age is', age)
      print('teenager')

    形式为:

    if <条件判断1>:
      <执行1>
    elif <条件判断2>:
      <执行2>
    elif <条件判断3>:
      <执行3>
    else:
      <执行4>

    if语句执行有个特点,他是从上往下判断,如果在某个判断上是true,把该判断对应的语句执行后,就忽略剩下的elif和else,所以,请测试并解释为什么下面的程序打印的是teenager:

    age = 20
    if age >= 6:
        print('teenager')
    elif age >= 18:
        print('adult')
    else:
        print('kid')

    if判断条件还可以简写,比如写:

    if x:
        print('True');

    只要x是非零数值、费控字符串、非空list等,就判断为True,否则为False。

    再议input

    最后看一个有问题的条件判断。很多同学会用input()读取用户的输入,这样可以自己输入,程序运行的更有意思:

    birth = input('birth:');
    if birth < 2000:
        print('00前')
    else:
        print('00后')

    输入1982,结果报错:

    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    TypeError: unorderable types: str() > int()

    这是因为input()返回数据类型是str,str不能直接和整数比较,必须先把str转换成整数。Python提供了int()函数来完成这件事情:

    s = input('birth:')
    birth = int(s)
    if birth < 2000;
        print('00前')
    else:
        print('00后')

    再次运行就可以得到正确的结果。但是,如果输入abc呢?又会得到一个错误信息:

    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    ValueError: invalid literal for int() with base 10: 'abc'

    原来int()函数发现一个字符串并不是合法的数字时就会报错,程序就退出了。

    如何检查并捕获程序运行期的错误呢?后面的错误和调试会讲到。

    5、循环

    Python有两种循环,一种是for...in,一种是while循环

    1)、for...in循环

    例:

    names = ['Michael', 'Bob', 'Tracy']
    for name in names :
        print(name)

    所以for x in ...循环就是把每个元素带入变量x,然后执行缩进快的语句。

    再比如我们想计算1-10的整数之和,可以用一个sum变量做累加:

    sum = 0
    for x in [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]:
           sum = num + x
    print(sum)

    如果要计算1-100的整数之和,从1写到100有点困难,幸好Python提供一个range()函数,可以生成一个整数序列,再通过list()函数可以转换为list。比如range(5)生成的序列是从0开始小于5的整数:

    >>>list(range(5))
    [0, 1, 2, 3, 4]

    range(101)就可以生成0-100的整数序列,计算如下:

    sum = 0
    for x in range(101):
        sum = sum + x
    print(sum)

    2)、while循环

    只要条件满足,就不断循环,条件不满足时退出循环。比如我们要计算100以内所有奇数之和,可以用while循环实现:

    sum = 0
    n = 99
    while n>0:
        sum = sum + n
        n = n -2
    print(sum)

    再循环内部变量n不断自减,知道变为-1时,不再满足while条件,魂环退出。

    3)break

    再循环中,break语句可以提前退出循环。例如,本来要循环打印1~100的数字:

    n = 1
    while n <= 100:
        print(n)
        n = n + 1
    print('END')

    上面打印的代码 可以打印出1~100.

    如果要提前结束循环,可以用break语句:

    n = 1
    while n<= 100
        if n > 10:
            break
        print(n)
        n = n + 1
    print('END')

    执行上面的代码可以看到,打印出1~10后,紧接着打印END,程序结束。

    可见break的作用是提前结束循环。

    continue

    再循环过程中,也可以通过continue语句,跳过当前的这次循环,直接开始下一次循环。

    n = 0
    while n < 10:
        n = n + 1
        print(n)

    上面的程序可以打印出1~10。但是,如果我们想只打印奇数,可以用continue语句跳过某些循环:

    n = 0
    while n < 10:
        n = n + 1
        if n % 2 == 0: # 如果n是偶数,执行continue语句
            continue # continue语句会直接继续下一轮循环,后续的print()语句不会执行
        print(n)

    执行上面的代码可以看到,打印的不再是1~10,而是1, 3, 5, 7, 9

    可见continue的作用是提前结束本轮循环,并直接开始下一轮循环。

    6、使用dict和set

    Python内置了字典:dict的支持,dict全称dictionary,在其他语言中也称为map,使用键-值(key-value)存储,具有极快的查找速度。

    假设要根据同学的名字查找对应的成绩,用dict实现,只需要一个“名字”-“成绩”的对照表,直接根据名字查找成绩,无论这个表有多大,查找速度都不会变慢。用Python写一个dict如下:

    >>>d = {'gouzi' : 29, 'dabai' : 26, 'cuihua' : 18}
    >>>d['gouzi']
    29

    为什么dict查找速度这么快?因为dict的实现原理和查字典是一样的。假设字段包含了1万个汉字,我们要查某一个字,一个办法是把字典从第一页往后翻,知道找到我们想要的字为止,这种方法就是在list中查找元素的方法,list越大,查找越慢。

    第二种方法是先在字典的索引里(比如部首表)查这个字对应的页码,然后直接翻到该页,找到这个字。无论找哪个字,这种查找速度都非常快,不会随着字典大小的增加而变慢。

    dict就是第二种实现方法,给定一个名字,比如‘gouzi’,dict在内部就可以直接计算出gouzi对应的存放成绩的“页码”,也就是29这个数字存放的内存地址,直接取出来,所以速度非常快。

    这总key-value存储方式,在放进去的时候,必须根据key算出value的存放位置,这样,去的时候才能根据key直接拿到value。

    把数据放入dict的方法,除了初始化时指定外,还可以通过key放入:

    >>>d['Adam'] = 67
    >>>d['Adam'] 
    67

    由于一个key只能对应一个value,所以,多次对一个key放入value,后面的值会把前面的值冲掉:

    >>> d['Jack'] = 90
    >>> d['Jack']
    90
    >>> d['Jack'] = 88
    >>> d['Jack']
    88

    如果key不存在,dict就会报错:

    >>> d['Thomas']
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    KeyError: 'Thomas'

    要避免key不存在的错误,有两种办法,一是通过in判断key是否存在:

    >>>'Tomas' in d
    False

    二是通过dict提供的get()方法,如果key不存在,可以返回None,或者自己指定的value:

    >>>d.get('dada')
    >>>d.get('dada', -1)
    -1

    注意:返回None的时候Python的交互环境不显示结果。

    要删除一个key,用pop(key)方法,对应的value也会从dict中删除:

    >>>d.pop('cuihua')
    12
    >>>d
    {'gouzi':29, 'libai': 101}

    请务必注意,dict内部存放的顺序和key放入的顺序和key放入的顺序是没有关系的。

    和list比较,dict有以下几个特点:

      1、查找和插入的速度极快,不会随着key的增加而变慢;

      2、需要占用大量的内存,内存浪费多。

    而list相反:

      1、查找和插入的时间随着元素的增加而增加;

      2、占用空间小,浪费内存很少。

    所以,dict使用空间来换取时间的一种方法。

    dict可以用在需要高速查找的很多地方,在Python代码中几乎无处不在,正确使用dict非常重要,需要牢记的而第一条就是dict的key必须是不可变对象。

    这是因为dict根据key来计算value的存储位置,如果每次计算相同的key得出的结果不同,那dict内部就完全混乱了。这个通过key来计算vlaue的存储位置,如果每次计算相同的key得出的结果不同,那dict内部就完全混乱了。这个通过key计算位置的额算法称为哈希算法。

    要保证hash的正确性,作为key的对象就不能变。在Python中,字符串、整数等都是不可变的,因此,可以放心地作为key。而list是可变的,就不能作为key:

     set

    set和dict类似,也是一组key的集合,但不存储value。由于key不能重复,所以,在set中,没有重复的key。

    要创建一个set,需要提供一个list作为输入集合:

    >>> s = set([1, 2, 3])
    >>> s
    {1, 2, 3}

    注意,传入的参数[1, 2, 3]是一个list,而显示的{1, 2, 3}只是告诉你这个set内部有1,2,3这3个元素,显示的顺序也不表示set是有序的。。

    重复元素在set中自动被过滤:

    >>> s = set([1, 1, 2, 2, 3, 3])
    >>> s
    {1, 2, 3}

    通过add(key)方法可以添加元素到set中,可以重复添加,但不会有效果:

    >>> s.add(4)
    >>> s
    {1, 2, 3, 4}
    >>> s.add(4)
    >>> s
    {1, 2, 3, 4}

    通过remove(key)方法可以删除元素:

    >>> s.remove(4)
    >>> s
    {1, 2, 3}

    set可以看成数学意义上的无序和无重复元素的集合,因此,两个set可以做数学意义上的交集、并集等操作:

    >>> s1 = set([1, 2, 3])
    >>> s2 = set([2, 3, 4])
    >>> s1 & s2
    {2, 3}
    >>> s1 | s2
    {1, 2, 3, 4}

    set和dict的唯一区别仅在于没有存储对应的value,但是,set的原理和dict一样,所以,同样不可以放入可变对象,因为无法判断两个可变对象是否相等,也就无法保证set内部“不会有重复元素”。试试把list放入set,看看是否会报错。

    再议不可变对象

    上面我们讲了,str是不变对象,而list是可变对象。

    对于可变对象,比如list,对list进行操作,list内部的内容是会变化的,比如:

    >>> a = ['c', 'b', 'a']
    >>> a.sort()
    >>> a
    ['a', 'b', 'c']

    而对于不可变对象,比如str,对str进行操作呢:

    >>> a = 'abc'
    >>> a.replace('a', 'A')
    'Abc'
    >>> a
    'abc'

    虽然字符串有个replace()方法,也确实变出了'Abc',但变量a最后仍是'abc',应该怎么理解呢?

    我们先把代码改成下面这样:

    >>> a = 'abc'
    >>> b = a.replace('a', 'A')
    >>> b
    'Abc'
    >>> a
    'abc'

    要始终牢记的是,a是变量,而'abc'才是字符串对象!有些时候,我们经常说,对象a的内容是'abc',但其实是指,a本身是一个变量,它指向的对象的内容才是'abc'

    ┌───┐                  ┌───────┐
    │ a │─────────────────>│ 'abc' │
    └───┘                  └───────┘
    

    当我们调用a.replace('a', 'A')时,实际上调用方法replace是作用在字符串对象'abc'上的,而这个方法虽然名字叫replace,但却没有改变字符串'abc'的内容。相反,replace方法创建了一个新字符串'Abc'并返回,如果我们用变量b指向该新字符串,就容易理解了,变量a仍指向原有的字符串'abc',但变量b却指向新字符串'Abc'了:

    ┌───┐                  ┌───────┐
    │ a │─────────────────>│ 'abc' │
    └───┘                  └───────┘
    ┌───┐                  ┌───────┐
    │ b │─────────────────>│ 'Abc' │
    └───┘                  └───────┘
    

    所以,对于不变对象来说,调用对象自身的任意方法,也不会改变该对象自身的内容。相反,这些方法会创建新的对象并返回,这样,就保证了不可变对象本身永远是不可变的。

    小结

    使用key-value存储结构的dict在Python中非常有用,选择不可变对象作为key很重要,最常用的key是字符串。

    tuple虽然是不变对象,但试试把(1, 2, 3)(1, [2, 3])放入dict或set中,并解释结果。

    五、定义函数

    在Python中,定义一个函数要使用def语句,依次写出函数名、括号、括号中的参数和冒号:,然后,在缩进块中编写函数体,函数的返回值用return语句返回。

    我们以自定义一个求绝对值的my_abs函数为例:

    
    
    # -*- coding: utf-8 -*-
    def my_abs(x):
        if x >= 0:
            return x
        else:
            return -x
    print(my_abs(-99))

    请注意,函数体内部的语句在执行时,一旦执行到return时,函数就执行完毕,并将结果返回。因此,函数内部通过条件判断和循环可以实现非常复杂的逻辑。

    如果没有return语句,函数执行完毕后也会返回结果,只是结果为Nonereturn None可以简写为return

    在Python交互环境中定义函数时,注意Python会出现...的提示。函数定义结束后需要按两次回车重新回到>>>提示符下:

    ┌────────────────────────────────────────────────────────┐
    │Command Prompt - python                           - □ x │
    ├────────────────────────────────────────────────────────┤
    │>>> def my_abs(x):                                      │
    │...     if x >= 0:                                      │
    │...         return x                                    │
    │...     else:                                           │
    │...         return -x                                   │
    │...                                                     │
    │>>> my_abs(-9)                                          │
    │9                                                       │
    │>>> _                                                   │
    │                                                        │
    │                                                        │
    └────────────────────────────────────────────────────────┘
    

    如果你已经把my_abs()的函数定义保存为abstest.py文件了,那么,可以在该文件的当前目录下启动Python解释器,用from abstest import my_abs来导入my_abs()函数,注意abstest是文件名(不含.py扩展名):

    ┌────────────────────────────────────────────────────────┐
    │Command Prompt - python                           - □ x │
    ├────────────────────────────────────────────────────────┤
    │>>> from abstest import my_abs                          │
    │>>> my_abs(-9)                                          │
    │9                                                       │
    │>>> _                                                   │
    │                                                        │
    │                                                        │
    │                                                        │
    │                                                        │
    │                                                        │
    │                                                        │
    │                                                        │
    └────────────────────────────────────────────────────────┘
    

    import的用法在后续模块一节中会详细介绍。

    空函数

    如果想定义一个什么事也不做的空函数,可以用pass语句:

    def nop():
        pass

    pass语句什么都不做,那有什么用?实际上pass可以用来作为占位符,比如现在还没想好怎么写函数的代码,就可以先放一个pass,让代码能运行起来。

    pass还可以用在其他语句里,比如:

    if age >= 18:
        pass

    缺少了pass,代码运行就会有语法错误。

    参数检查

    调用函数时,如果参数个数不对,Python解释器会自动检查出来,并抛出TypeError

    >>> my_abs(1, 2)
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    TypeError: my_abs() takes 1 positional argument but 2 were given

    但是如果参数类型不对,Python解释器就无法帮我们检查。试试my_abs和内置函数abs的差别:

    >>> my_abs('A')
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
      File "<stdin>", line 2, in my_abs
    TypeError: unorderable types: str() >= int()
    >>> abs('A')
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    TypeError: bad operand type for abs(): 'str'

    当传入了不恰当的参数时,内置函数abs会检查出参数错误,而我们定义的my_abs没有参数检查,会导致if语句出错,出错信息和abs不一样。所以,这个函数定义不够完善。

    让我们修改一下my_abs的定义,对参数类型做检查,只允许整数和浮点数类型的参数。数据类型检查可以用内置函数isinstance()实现:

    def my_abs(x):
        if not isinstance(x, (int, float)):
            raise TypeError('bad operand type')
        if x >= 0:
            return x
        else:
            return -x

    添加了参数检查后,如果传入错误的参数类型,函数就可以抛出一个错误:

    >>> my_abs('A')
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
      File "<stdin>", line 3, in my_abs
    TypeError: bad operand type

    错误和异常处理将在后续讲到。

    返回多个值

    函数可以返回多个值吗?答案是肯定的。

    比如在游戏中经常需要从一个点移动到另一个点,给出坐标、位移和角度,就可以计算出新的新的坐标:

    import math
    
    def move(x, y, step, angle=0):
        nx = x + step * math.cos(angle)
        ny = y - step * math.sin(angle)
        return nx, ny

    import math语句表示导入math包,并允许后续代码引用math包里的sincos等函数。

    然后,我们就可以同时获得返回值:

    >>> x, y = move(100, 100, 60, math.pi / 6)
    >>> print(x, y)
    151.96152422706632 70.0

    但其实这只是一种假象,Python函数返回的仍然是单一值:

    >>> r = move(100, 100, 60, math.pi / 6)
    >>> print(r)
    (151.96152422706632, 70.0)

    原来返回值是一个tuple!但是,在语法上,返回一个tuple可以省略括号,而多个变量可以同时接收一个tuple,按位置赋给对应的值,所以,Python的函数返回多值其实就是返回一个tuple,但写起来更方便。

    小结

    定义函数时,需要确定函数名和参数个数;

    如果有必要,可以先对参数的数据类型做检查;

    函数体内部可以用return随时返回函数结果;

    函数执行完毕也没有return语句时,自动return None

    函数可以同时返回多个值,但其实就是一个tuple。

    练习

    请定义一个函数quadratic(a, b, c),接收3个参数,返回一元二次方程:

    ax2 + bx + c = 0

    的两个解。

    提示:计算平方根可以调用math.sqrt()函数:

    >>> import math
    >>> math.sqrt(2)
    1.4142135623730951

  • 相关阅读:
    前言
    echarts踩坑---容器高度自适应
    vue中刷新页面时去闪烁,提升体验方法
    2018.11.7
    07-sel-express 框架快速搭建案例
    第三方包 vue-resource
    zepto.js-定制zepto步骤
    CSS-单位em 和 rem
    ES6-个人学习大纲
    响应式布局
  • 原文地址:https://www.cnblogs.com/bigclould/p/9921056.html
Copyright © 2020-2023  润新知