Python基础知识点总结

简介: 本文包括python基本知识:简单数据结构,数据结构类型(可变:列表,字典,集合,不可变:数值类型,字符串,元组),分支循环和控制流程,类和函数,文件处理和异常等等。

Python基础知识点总结

在这里插入图片描述


一、开发环境搭建

在这里插入图片描述


二、基本语法元素

2.1 程序的格式框架

  程序的格式框架,即段落格式,是Python语法的一部分,可以提高代码的可读性和可维护性。

2.1.1 缩进

  Python语言采用严格的缩进来表示程序逻辑。也就是我们所说的Python程序间的包含与层次关系。一般代码不要求缩进,顶行编写且不留空白。在if、while、for、def、class等保留字所在完整语句后通过英文的“:”结尾并在之后行进行缩进,表明后续代码与紧邻无缩进语句的所属关系。

在这里插入图片描述

  缩进可以用Tab键实现,也可以用多个空格实现(一般是4个空格),但两者不能混用。建议采用4个空格方式书写代码。

2.1.2 注释

  注释是代码中的辅助性文字,会被编译器或解释器略去,不被计算机执行,一般用于程序员对代码的说明。Python语言中使用“#”表示一行注释的开始。注释可以在一行中任意位置通过“#”开始,其后面的本行内容被当作注释,而之前的内容依然是Python执行程序的一部分。

注释的作用:

  • 注释一般用于在代码中标明作者和版权信息
  • 解释代码原理和用途
  • 通过注释单行代码辅助程序调试。
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : 1071505897@qq.com
# @Time  : 2022/8/2 13:40

# 单行注释

'''多行注释'''

"""多行注释"""

2.1.3 续行符

  Python程序是逐行编写的,每行代码长度并无限制,但从程序员角度,单行代码太长并不利于阅读。这个时候就可以使用续行符将单行代码分割为多行表达。

  Python中的续行符为“\”。续行符之后不允许再存在空格,即续行符后直接换行。

print("云南的城市有{}\n{}\n{}\n{}".format('昆明',\
                        '曲靖',\
                        '大理',\
                        '丽江'))

2.2 语法元素的名称

  Python语言的基本单位单词,少部分单词是Python语言规定的,被称为保留字。大部分单词是用户自己定义的,通过命名过程形成了变量或函数,用来代表数据或代码,称为标识符。

2.2.1 变量

  变量是保存和表示数据值的一种语法元素,变量的值是可以改变的,通过赋值运算符“=”方式被修改。Python语言中,变量可以随时命名、随时赋值、随时使用。

由三部分组成:

"""
标识:     id
类型:     type
值:       value  
"""

多次赋值后将指向新的空间

name = 'hello'  # 第一次赋值
print(id(name))  # 标识
# 2026989330544
name = 'world'  # 第二次赋值
print(id(name))
# 2026989317168

2.2.2 保留字

  保留字也称keyword关键字,被编程语言内部定义并保留使用的,每种程序设计语言都有一套保留字,保留字一般用来构成程序的整体框架,Python3.x中一共有35个保留字。

import keyword  # 导入关键字模块

print(keyword.kwlist)  # 调用keyword模块的kwlist方法

Python3.x中的35个保留字

and    as    assert    async    await    break    
class continue    def    del    elif    else    
except    False finally    for    from 
global    if import    in is lambda    None    
nonlocal    not    or    pass raise    return    
True    try     while     with     yield

  Python二级考试涉及到的保留字一共有22个。选学5个:None、finally、lambda、pass、with。
Python中的保留字也是大小写敏感的。举例:True为保留字,而true则不是保留字。

2.2.3 标识符

  标识符可以简单的理解为一个名字,主要用来标识变量、函数、类、模块和其他对象的名称。

标识符的命名规则

  • 字母、数字、下划线
  • 不能以数字开头
  • 不能是Python中的保留字
  • 只允许使用ISO-Latin(ISO-8859-1)字符集中的A-Za-z
  • 允许使用中文,但不建议
  • 注意:标识符对大小写敏感,nameName是两个不同的名字。

2.3 数据类型

2.3.1 二进制与字符编码

  二进制是一套计数方法,每个位置上的数有 2 种可能(0 - 1);二进制是计算机的执行语言,但是早在计算机出现前就存在这套计数方法,最早可追溯到古埃及。在日常生活中,我们使用的是十进制,每个位置上的数有 10 种可能(0 - 9)。
在这里插入图片描述
  早期的程序员爸爸为了让计算机能够认识我,将我能够认识的符号和数字对应好,然后做成一张表叫ASCII表,告诉计算机某种符号你应该使用哪个整数表示,A使用了8个位(置)才能装得下我,在计算机中他们叫一个字节。

ASCII码表

ASCII值 控制字符 ASCII值 控制字符 ASCII值 控制字符 ASCII值 控制字符
0 NUT 32 (space) 64 @ 96
1 SOH 33 65 A 97 a
2 STX 34 66 B 98 b
3 ETX 35 # 67 C 99 c
4 EOT 36 $ 68 D 100 d
5 ENQ 37 % 69 E 101 e
6 ACK 38 & 70 F 102 f
7 BEL 39 , 71 G 103 g
8 BS 40 ( 72 H 104 h
9 HT 41 ) 73 I 105 i
10 LF 42 * 74 J 106 j
11 VT 43 + 75 K 107 k
12 FF 44 , 76 L 108 l
13 CR 45 - 77 M 109 m
14 SO 46 . 78 N 110 n
15 SI 47 / 79 O 111 o
16 DLE 48 0 80 P 112 p
17 DCI 49 1 81 Q 113 q
18 DC2 50 2 82 R 114 r
19 DC3 51 3 83 S 115 s
20 DC4 52 4 84 T 116 t
21 NAK 53 5 85 U 117 u
22 SYN 54 6 86 V 118 v
23 TB 55 7 87 W 119 w
24 CAN 56 8 88 X 120 x
25 EM 57 9 89 Y 121 y
26 SUB 58 : 90 Z 122 z
27 ESC 59 ; 91 [ 123 {
28 FS 60 < 92 / 124 |
29 GS 61 = 93 ] 125 }
30 RS 62 > 94 ^ 126 ~
31 US 63 ? 95 127 DEL

ASCII码表字符解释

NUL 空 VT 垂直制表 SYN 空转同步
SOH 标题开始 FF 换页键 ETB 信息组传送结束
STX 正文开始 CR 回车 CAN 作废
ETX 正文结束 SO 移位输出 EM 纸尽
EOY 传输结束 SI 移位输入 SUB 代替
ENQ 请求 DLE 空格 ESC 换码
ACK 收到通知 DC1 设备控制1 FS 文字分隔符
BEL 报警 DC2 设备控制2 GS 组分隔符
BS 退一格 DC3 设备控制3 RS 记录分隔符
HT 水平制表符 DC4 设备控制4 US 单元分隔符
LF 换行 NAK 拒绝 DEL 删除

2.3.2 数值类型

Python提供的3种数值类型:
整数类型:与数学中的整数一致,包含正、负、0。一个整数有二进制、八进制、十进制、十六进制4种表示方式。

进制 基本数 逢几进一 表示形式
十进制 0,1,2,3,4,5,6,7,8,9 10 520
二进制 0,1 2 0b11010
八进制 0,1,2,3,4,5,6,7 8 0o520
十六进制 0,1,2,3,4,5,6,7,8,9,A,B,C,D,E,F 16 0xAF12546

浮点数类型:与数学中的小数一致,没有取值范围限制,可正、可负。有两种表示形式,一种是小数点的形式,另外一种是科学计数法。浮点数只有十进制形式。

复数:与数学中的复数一致,采用a+bj的形式表示,存在实部和虚部。

2.3.3 字符串类型

  计算机经常处理文本信息,文本信息在程序中使用字符串类型表示。在Python中使用单引号或双引号括起来的一个或多个字符来表示。单引号和双引号的作用相同。

# 字符串类型   被称为不可变的字符序列
print('我用python')
print("我用python")
print('''我用python''')

print(type('我用python'))
print(type("我用python"))
print(type('''我用python'''))

字符序列有两种序号体系:
正向递增序号: 有效范围为[0,N-1],其中N为字符串中字符的个数。最左侧的字符序号为0,依次向右递增,最右侧字符序号为N-1。

反向递减序号:有效范围为[-1,-N],其中N为字符串中字符的个数。最右侧的字符序与为-1,依次向左递减,最左侧的字符序号为-N。
在这里插入图片描述
两种索引可以同时使用,以下代码为对单个字符进行的索引。

s = 'hello word'
print(s[0])  # h
print(s[-10])  # h

还可以采用[N:M]格式获取字符串的子串,这种操作被称为切片操作。[N:M]获取字符串中从NM(但不包含M)的连续的子字符串。NM都表示的是索引序号,可以混合使用正向递增序号和反向递减序号。

s = 'hello word'
print(s[0:5])  # hello
print(s[0:-5])  # hello

通过Python默认提供的len()函数获取字符串的长度,一个中文字符和一个英文字符的长度都记为1。

print(len(s))  # 10

2.4 程序的语句元素

2.4.1 表达式

产生或计算新数据值的代码片段称为表达式。类似于数学中的公式,一般由数据和操作符构成。

2.4.2 赋值语句

对变量进行赋值的一行代码被称为赋值语句。在Python中使用一个“=”表示“赋值”,即将等号右侧表达式计算后的结果值赋给左侧的变量。

基本赋值语句语法
变量 = 表达式

同步赋值语句就是同时给多个变量进行赋值。
同步赋值语句语法:

变量1,变量2,......变量N = 表达式1,表达式2,......,表达式N

同步赋值语句应用最多的就是互换变量的值,即交换两个变量的值。

2.4.3 引用

Python程序会经常使用到当前程序之外已有的功能代码,这个过程叫引用。

Python语言中使用import这个保留字引用当前程序以外的功能库。
import <功能库名称>
引用功能库之后使用 功能库.函数名()的方式调用基本功能,这种方式简称为“A.B()”方式。

2.4.4 其他语句

除了赋值语句之外,Python还有分支语句和循环语句。先简单介绍一下使用,后面后详细讲解。

分支语句:根据判断条件选择程序执行路径。一般包含单分支结构、双分支结构和多分支结构。
单分支语法结构

if   条件:
   语句块

任何能够产生True或False的语句都可以作为条件,当条件为True时,则执行语句块中的内容。

双分支语法结构

if   条件:
     语句块1
 else:
     语句块2

当条件为True时,执行语句块1,当条件为False时,执行语句块2。其中if...else 都是保留字。

循环结构与分支结构一样都是程序控制语句,它的作用是根据判断条件确定一个程序是否再执行一次或者多次。

条件循环的语法结构

while  条件:
     语句块1
语句块2

当条件为True时,执行语句块1,然后再次判断条件,当条件为False时,退出循环,执行语句块2。


2.5 基本输入输出函数

2.5.1 input( )函数

  input()函数是从控制台获得用户的一行输入,无论用户输入什么内容,input()函数都以字符串类型返回。input()函数可以包含一些提示性文字,用来提示用户。

input语法格式:
变量=input('提示性文字')
  无论用户输入的是数字还是字符,input()函数统一按照字符串类型输出,为了后续能够操作用户输入的信息,需要将输入指定一个变量进行存储。

  input()函数的提示性文字不是必须的,可写可不写。

2.5.2 eval( )函数

  eval(s)函数将去掉字符串s最外侧的引号,并按照Python语句方式执行去掉引号后的字符内容。

eavl语法格式:
变量 = eval(字符串)

变量用来保存对字符串内存进行Python运算的结果。

  eval()函数去掉了字符串’1.2’最外侧的引号,结果赋值给a,所以a的类型为float类型。eval()函数去掉了字符串’1.2+3.4’最外侧的引号,对其内容当作Python语句进行运算,运算的结果为4.6,保存到变量a中,所以a的类型是float类型。

  当eval()函数处理字符串’python’时,字符串去掉了两边的引号,Python语句将其解释为一个变量,由于这个变量在之前别有定义,因此解释器报错。如果定义变量python并赋值为123,再运行这个语句时就没有问题,如果输出为123

eval()函数经常和input()函数一起使用,用来获取用户输入的数字。

eval()input()函数一起使用的语法:
变量=eval(input(提示性文字))

  用户输入的数字、包括小数和负数,input()解析为字符串,再由eval()去掉字符串引号,将被直接解析为数字保存到变量中。

2.5.3 print()函数

print()函数用于输出运算结果。

def print(self, *args, sep=' ', end='\n', file=None): # known special case of print
    """
    print(value, ..., sep=' ', end='\n', file=sys.stdout, flush=False)
    
    Prints the values to a stream, or to sys.stdout by default.
    Optional keyword arguments:
    file:  a file-like object (stream); defaults to the current sys.stdout.
    sep:   string inserted between values, default a space.
    end:   string appended after the last value, default a newline.
    flush: whether to forcibly flush the stream.
    """
    pass

sep=' '分隔符,默认为一个空格
end='\n'结束符,默认以换行结束

print()函数的三种用法:

  • 仅用于输出字符串或单个变量

print(待输出字符串或变量)

对于字符串,print()函数输出后将出掉两侧引号,对于其它类型,则会直接输出。

print()输出字符串表示时,字符串统一使用单引号表达。在[]中的字符串都使用了双引号,但是在被print()函数打印输出时,输出结果都采用了单引号。

  • 用于输出一个或多个变量 ,输出后的各变量之间用一个空格分隔。

print(变量1,变量2,......,变量N)

  • 用于混合输出字符串与变量的值,其中输出字符串模板中使用{}表示一个槽位,每个槽位对应.format()中的一个变量。

print(输出字符串模板.format(变量1,变量2,......,变量N))

print()函数输出的输出内容

'''可以输出数字   int   '''
print(2)  # 2

'''可以输出字符串   str   '''
print('hello')  # hello

'''输出含有运算符的表达式  '''
print(2+3)  # 5

print()函数输出的目的地

'''将内容输出到显示器'''
print('hello,world')  # hello,world

'''将内容输出到文件'''
# 注意:所指定的盘符存在;使用file = fp;
fp = open('E:\\text.txt', 'a')
print('hello,world!', file=fp)
fp.close()

‘整数{}和整数{}的差是:{}’是输出字符串模板,也就是混合字符串输出的样式,大括号{}表示一个槽位,括号中的内容由后面紧跟着的format()方法中的参数按顺序填充。

2.6转义字符与原字符

2.6.1 转义字符

就是反斜杠+想要实现的转义功能首字母。

为什么需要转义字符?
当字符串中包含反斜杠、单引号和双引号等有特殊用途的字符时,必须使用反斜杠对这些字符进行转义(转换一个含义)。

  • 反斜杠 :\\
  • 单引号:\'
  • 双引号: \“
print('\\')  # \
print('\'')  # '
print('\"')  # "

当字符串中包含换行、回车,水平制表符或退格等无法直接表示的特殊字符时,也可以使用转义字符当字符串中包含换行、回车,水平制表符或退格等无法直接表示的特殊字符时,也可以使用转义字符。

  • 换行:\n
  • 回车:\r
  • 水平制表符 : \t
  • 退格 :\b
# 回车\r
print('hello\rhelloworld')   # 将hello覆盖

# 换行\n
print('hello\nworld')

# 退格\b
print('helloo\bworld')

# 水平制表\t
print('hello\tworld\t你好\t好')

2.6.2 原字符

原字符:不希望字符串中的转义字符起作用,就使用原字符,就是在字符创之前加上

注意事项,最后一个字符不能是反斜线

# 原字符:不希望字符串中的转义字符起作用,就使用原字符,就是在字符创之前加上 r 或 R
# 注意事项,最后一个字符不能是反斜线
print(r'hello \n word')  # hello \n word

三、运算符

3.1 数值运算操作符

Python提供了9个基本的数值运算操作符。

操作符及运算 描述
x + y x 与 y 之和
x - y x 与 y 之差
x * y x 与 y 之积
x / y x 与 y 之商,产生结果为浮点数
x // y x 与 y 之整数商
x % y x 与 y 之余商,也称之为模运算
- x x 的负数
+ x x 本身
x**y x 的 y 次幂

3.2 数值运算函数

所谓函数表现为对参数的特定运算。

Python解释器自身提供了一些预先编写好的函数称为内置函数,其中与数值运算相关的函数如下:

序号 函数 描述
1 abs(x) x的绝对值
2 divmod(x, y) (x//y, x%y)输出元组形式的商与余数
3 pow(x, y)pow(x, y, z) x**y或(x**y)%z,幂运算
4 round(x)round(x, d) x四舍五入保留d位小数,d省去则为四舍五入的整数
5 max(x1, x2,...,xn) n个数中的最大值
6 min(x1, x2,...,xn) n个数中的最小值

3.3 算数运算符

'''
算数运算符:
    加 +
    减 -
    乘 *
    除 /
    整除 //
    取余 %
    幂 **
'''

print(-9 // -4)  # 2
print(9 // 4)  # 2

print(-9 // 4)  # -3 一正一负整除公式,商取下整
print(9 // -4)  # -3

print(9 % -4)  
# -3  公式  余数=被除数-除数*商  9-(-4)*(-3)= -3
print(-9 % 4)  
# 3      -9 - 4*(-3)= 3

# 规律:规律就是  取余数是负数 结果就是负数

3.4 赋值运算符

赋值运算符:运算顺序从右到左

'''
赋值运算符:运算顺序从右到左
=   a=10
+=  a+=b    a=a+b
-=  a-=b    a=a-b
*=  a*b     a=ab
/=  a/=b    a=a/b
//= a//=b   a=a//b
%   a%=b    a=a%b
'''

链式赋值,id相同

'''链式赋值,id相同'''
a = b = c = 20
print(a, type(a), id(a))
print(b, type(b), id(b))
print(c, type(c), id(c))
print('-------------------------')


output:
20 <class 'int'> 140725395003648
20 <class 'int'> 140725395003648
20 <class 'int'> 140725395003648
-------------------------

参数赋值

'''参数赋值'''
a = 20
a += 30  # a = a+30
print(a)  # 50

系列解包赋值

'''解包赋值'''
a, b, c = 10, 20, 30
print(a, id(a))
print(b, id(b))
print(c, id(c))

print('---------交换两个变量的值-----------')
a, b = 10, 20
print(a, b)
a, b = b, a
print(a, b)


output:
10 140725395003328
20 140725395003648
30 140725395003968
---------交换两个变量的值-----------
10 20
20 10

在这里插入图片描述

3.5 布尔运算符

布尔类型 用来表示真或假

  • True 表示真
  • False表示假
  • 布尔值可以转化为整数
    • True:1
    • False:0

Python中一切皆对象,所有对象都有一个布尔值,获取对象布尔值使用内置函数bool()

# Python中一切皆对象,所有对象都有一个布尔值
# 获取对象布尔值使用内置函数bool()
'''
以下对象布尔值为False:
    False
    数值0
    None
    空字符串 ''
    空列表 []
    空字典 {}
    空元组 ()
    空集合 {}
'''
print(bool(False))
print(bool(0))
print(bool(0.0))
print(bool(None))
print(bool(''))
print(bool([]))  # 空列表
print(bool(list()))

print(bool({}))  # 空字典
print(bool(dict()))

print(bool(()))  # 空元组
print(bool(tuple()))

print(bool(set()))  # 空集合
print('-------------以上对象bool值均为False------------')

3.6 比较运算符

比较运算符: 比较的值为TrueFalse

''''
   >    大于
   <    小于
  ==      等于  比较的是值:value
  !=      不等于
  >=    大于等于
  <=    小于等于 
 
 对象id的比较 is   is not 
 '''

3.7 位运算 & |

'''
位运算符:

位与 & 对应位数都是1,结果才为1,否则为0

位或 | 对应位数都是0,结果才为0

左移运算符   <<  高位溢出,低位补0
右移运算符   >>  低位溢出,高位补0 
'''

3.8 逻辑运算符:and or not

TrueFalse 二元数值构成了基本的真假逻辑,因此诞生了逻辑运算。

运算符 描述
and x and y,与操作
or x or y,或操作
not not x,非操作
'''
and 且      全真为真
or  或      有真则真
not 非      对bool类型操作数取反  
      
in  在...里       
not in 不在...里
'''

3.9 运算符的优先级

指数运算运算(最高优先级)      **
算数运算       *       /       //     %       +     -
位运算         <<      >>      &     |
比较运算符      >     <       >=      <=
等于运算符       ==      !=
身份运算符       is      is not
成员运算符       in      not in
逻辑运算符       and     or     not

四、流程控制

4.1 顺序结构

程序按照线性顺序依次执行的一种运行方式。

例:把大象装冰箱分几步:

# 把大象装冰箱分几步
print('--------程序开始--------')
print('1.把冰箱门打开')
print('2.把大象装进冰箱')
print('3.关上冰箱门')
print('--------程序结束--------')

4.2 分支结构(选择结构)

  程序根据判断条件的布尔值选择性地执行部分代码明确的让计算机知道在什么条件下,该去做什么

4.2.1 单分支 if

中文语义:如果......就......

在这里插入图片描述
语法结构:

if 条件表达式:
    条件执行体

if条件语句:判断一次,条件为True时执行一次

# if条件语句 判断一次 条件为True时执行一次

money = 100
s = int(input('请输入取款金额:'))
if s <= 100:
    money = money - s
    print('取款成功,余额为', money)

4.2.2 双分支 if...else

中文语义:如果......不满足......就......
在这里插入图片描述
语法结构:

if 条件表达式:
  条件执行体
else:
  条件执行体2

双分支结构if...else,二选一执行

'''双分支结构if...else,二选一执行'''

# 从键盘输入一个整数,编写程序让计算机判断是奇数还是偶数
num = int(input('请输入一个整数:'))
if num % 2 == 0:
    print(num, '是偶数')
else:
    print(num, '是奇数')

4.2.3 多分支 if...elif..else

中文语义:如果......就......如果......不满足......就......
在这里插入图片描述

语法结构:

if 条件表达式:
   条件执行体
elif   条件表达式:
   条件执行体
elif   条件表达式:
  条件执行体
elif   条件表达式:
  条件执行体
else:
  条件执行体
例:
从键盘录入一个整数成绩:
90-100 : A
80-89 : B
70-79 : C
60-69 : D
0-59 :E
小于0或大于100 不及格

代码:

num = int(input('从键盘输入一个成绩:'))
if 90 <= num <= 100:
    print('A')
elif 80 <= num <= 89:
    print('B')
elif 70 <= num <= 79:
    print('C')
elif 60 <= num <= 69:
    print('D')
elif 0 <= num <= 59:
    print('E')
else:
    print('不及格')

4.2.4 嵌套if的使用

语法结构

if 条件表达式1:
    if 内层条件表达式:
        内层执行体1
    else:
    内层执行体2
else:
    条件执行体

在这里插入图片描述
例:商场消费

会员:
    >=200    8折
    >=100    9折
    <100     不打折
非会员: 
    >=200    9.5折
    <100     不打折

代码:

answer = input('您是会员吗?y/n')
money = float(input('请输入您的购物金额:'))
# 外层判断是否是会员
if answer == 'y':  # 会员
    if money >= 200:
        print('打8折,付款金额为:', money*0.8)
    elif money >= 100:
        print('打9折,付款金额为:', money*0.9)
    else:
        print('不打折,付款金额为:', money)
else:  # 非会员
    if money >= 200:
        print('打9.5折,付款金额为:', money*0.95)
    else:
        print('不打折,付款金额为:', money)

4.2.5 三元表达式

语法结构x if 判断条件 else y

运算规则:如果判断条件的布尔值为True,条件表达式的返回值为x,否则条件表达式的返回值为False

# 条件表达式   x if 判断条件 else y
a = int(input('请输入第一个整数:'))
b = int(input('请输入第二个整数:'))

'''if a >= b:
    print(a, '>=', b)
else:
    print(a, '<', b)'''


# 条件表达式
print(str(a) + '小于' + str(b)) if a < b else (str(a) + '大于' + str(b))

4.2.6 pass语句

pass语句什么都不做,只是一个占位符,用在语法上需要语句的地方

def fun(a, b):
    pass

什么时候使用pass语句:先搭建语法结构,还没想好代码怎么写的时候 。

pass语句与哪些语句一起使用:

  1. if语句的条件执行体
  2. for-in语句的循环体
  3. 定义函数def时的函数体
a = 1
b = 2

if a < b:
    pass


for i in range(10):
    pass


def fun(a, b):
    pass

4.3循环结构

  反复做同一件事情的情况,称为循环,Python语言的循环结构包括两种:遍历循环无限循环。遍历循环使用保留字forin依次提取遍历结构各元素进行处理;无限循环使用保留字while根据判断条件执行程序
在这里插入图片描述

4.3.1 range()函数的使用

range():用于生成一个整数序列

创建range对象的三种方式
在这里插入图片描述
代码:

# 第一种创建方式  range(stop)
range(10)
print(range(10))  # range(0,10) 返回值是一个迭代器对象 默认从0开始,默认步长为1
print(list(range(10)))  # 用于查看range对象中的整数序列 list-->列表

# 第二种创建方式 range(start, stop)
print(range(0, 10))

# 第三种创建方式 range(start,stop,step)
print(range(0, 10, 1))

range()函数的返回值是一个迭代器对象

range类型的优点 :不管range对象表示的整数序列有多长,所有range对象占用的内存空间都是相同的,因为仅仅需要存储start,stop和step,只有当用到range对象时,才会去计算序列中的相关元素

'''内置函数 range(start(开始)未指定从0开始, stop(结束), step(步长)未指定步长为1)'''
r = range(1, 10, 2)
print(r)
print(list(r))  # 列表 [1, 3, 5, 7, 9]
print(tuple(r))  # 元组 (1, 3, 5, 7, 9)
print(set(r))  # 字典 {1, 3, 5, 7, 9}

innot in 判断整数序列中是否存在(不存在)指定的整数

'''判断指定整数在序列中是否存在 in, not in'''
print(3 in r)  # True
print(4 in r)  # False
print(3 not in r)  # False
print(4 not in r)  # True

for value in range(10):
    print(value, end=' ')

4.3.2 遍历循环for-in

for-in循环

  • in表达从(字符串、序列等)中依次取值,又称为遍历
  • for-in遍历的对象必须是可迭代对象

for-in的语法结构

for  自定义的变量  in 可迭代对象:
      循环体

for-in的执行图
在这里插入图片描述

# for in循环
# in 表达是依次 从 (字符串,序列等) 中依次取值,又称遍历
# for in遍历对象必须是可迭代对象

'''for in 语法结构'''
# for 自定义变量 in 可迭代对象:
#    循环体

for i in 'hello':
    print(i, end=' ')
print('')
for i in range(10):
    print(i, end=' ')
print('')
# 在循环体中如果不需要使用到自定义变量,可将自定义变量写为下划线('_')
for _ in range(3):   # 3表示次数
    print('hello')

使用for循环 计算1-100之间的偶数和

# 使用for循环 计算1-100之间的偶数和
sum = 0
for item in range(101):
    if item % 2 == 0:
        sum += item
print('1-100之间的偶数和为:', sum)

输出100-999之间的水仙花数

'''输出100-999之间的水仙花数  这个数的个位上的三次方 + 十位上的三次方 + 百位上的三次方  = 这个数'''
for item in range(100, 1000):
    ge = item % 10
    shi = item % 100 // 10  # item//10%10
    bai = item // 100
    if item == ge**3+shi**3+bai**3:
        print(item)

4.3.3无限循环while

语法结构

while  条件表达式:
    条件执行体(循环体)

四步循环法:

  1. 初始化变量
  2. 条件判断
  3. 条件执行体(循环体)
  4. 改变变量
  5. 打印输出

计算0-4之间的累加和

a = 0  # 1.初始化变量
s = 0
while a <= 4:  # 2.条件判断
    s += a  # 3.条件执行体
    a += 1  # 4.改变变量
    print(s)  # 5.打印输出

选择结构的if与循环结构while的区别

  • If是判断一次,条件为True执行一行
  • while是判断N+1次,条件为True执行N

4.3.4循环控制break和continue

循环结构有两个辅助循环控制的保留字:breakcontinue

  • break语句

break用来跳出最内层forwhile循环,脱离该循环后程序继续执行循环后续代码,通常与 分支结构 if 一起使用。

例:从键盘录入密码,最多录入三次,如果正确就结束循环

# for 循环
for _ in range(3):  # range()函数可以用作for循环的循环次数 3 表示for执行了3次
    pwd = '888'
    a = input('请输入密码:')
    if a == pwd:
        print('密码正确')
        break
    else:
        print('密码错误,请重新输入密码')

# while 循环,四步循环法
a = 0
while a < 3:
    '''条件执行体(循环体)'''
    b = '888'
    pwd = input('请输入密码:')
    if pwd == b:
        print('密码正确')
        break
    else:
        print('密码错误')
    a += 1
  • continue语句

continue语句用来结束当次循环,即跳出循环体中下面尚未执行的语句,但不跳出当前循环,进入下一次循环,通常与分支结构 if 一起使用

例:要求输出1-50-之间所有5的倍数

# for 循环
for i in range(1, 51):
    if i % 5 == 0:
        print(i)

# while 循环
a = 1
while a <= 50:
    if a % 5 == 0:
        print(a)
    a += 1

# continue 语句
for item in range(1, 51):
    if item % 5 != 0:
        continue
    print(item)

4.3.5嵌套循环

循环结构中又嵌套了另外的完整的循环结构,其中内层循环做为外层循环的循环体执行。

例1:打印一个三行四列矩形

for _ in range(3):
    for _ in range(4):
        print('-', end='\t')
    print()

例2:输出乘法口诀表

for i in range(1, 10):
    for j in range(1, i+1):
        print(i, 'x', j, '=', i*j, end='\t')
    print()

在这里插入图片描述


五、组合数据类型

5.1 列表类型

5.1.1 列表的定义

列表类型是包含0个或多个元素的有序序列,属于序列类型。列表可以进行元素的增加、删除、替换、查找等操作。列表没有长度限制,无素类型可以不同,不需要预定长度。

列表类型用中括号[]表示,也可以通过list(x)函数将集合或字符串类型转换成列表类型。list()函数可生成空列表。

列表需要使用中括号[],元素之间使用英文的逗号进行分隔

列表示意图:
在这里插入图片描述
列表创建方式

  1. 使用[]方括号
lst = ['hello', 'world', 'java']
print(lst)
  1. 使用内置函数list()
lst2 = list(['hello', 'world', 'java'] + ['hello', 'world', 'java'])
print(lst2)
  1. 列表生成式
lst = [i for i in range(1, 11)]
print(lst)

5.1.2 列表的特点

  1. 列表元素按顺序有序排列
  2. 索引映射唯一一个数据
  3. 列表可以存储重复数据
  4. 任意数据类型混存
  5. 根据需要动态分配内存空间

5.1.3 列表的索引

索引是列表的基本操作,用于获得列表第一个元素。该操作沿用序列类型的索引方式,即正向递增序号和反向递减序号,使用中括号作为索引操作符,索引序号不能超过列表的元素范围,否则会产生 IndexErrorr的错误。

获取列表中指定元素索引

语法:列表或列表变量.index('要获取索引的元素',起始位置,结束位置)

如果列表中含有相同元素,只返回相同元素的第一个元素的索引。

# 如果列表中含有相同元素,只返回相同元素的第一个元素的索引

lst = ['hello', 'world', 'java', 'hello', 'world', 'java']
print(lst.index('java'))  # 2
print(lst.index('java', 3, 6))  # 5

可以使用遍历循环对列表类型的元素进行遍历操作。

语法:for 迭代变量 in 列表名:

# 列表元素的遍历

lst = [10, 20, 30, 40, 50, 60, 70, 80]
for i in lst:
    print(i)

output:
10
20
30
40
50
60
70
80

5.1.4 列表的切片

切片是列表的基本操作,用于获得列表的一个片段,即获得零个或多个元素。切片后的结果也是列表类型。
切片有两种使用方式:

  • 列表或列表变量[N:M]
  • 列表或列表变量[N:M:K]

根据索引获取元素值

lst = ['hello', 'world', 'java', 'hello', 'world', 'java']

'''获取单个元素'''
# 获取列表索引为 2的元素
print(lst[2])  # java

# 获取列表索引为-3的元素
print(lst[-3])  # hello

'''获取多个元素   返回值为一个列表'''
# lst = [start(默认0开始) : stop(默认最后) : step(默认步长1)]
# step 为负数时相反

print(lst[0:5:1])
# ['hello', 'world', 'java', 'hello', 'world']

5.1.5 列表操作函数

操作函数 描述
len(ls) 列表ls的元素个数(长度)
min(ls) 列表ls的最小元素
max(ls) 列表ls的最大元素
list(x) x转换为列表类型
# -*- coding: utf-8 -*-
# @File  : 列表操作函数.py
# @author: Flyme awei 
# @email : 1071505897@qq.com
# @Time  : 2022/8/8 0:15

ls = [1, 3, 5, 7, 9]
print(len(ls))
print(max(ls))
print(min(ls))
x = (1, 3, 5, 7, 9)
print(list(x))

在这里插入图片描述

5.1.6 列表操作方法

列表类型的一些常用操作方法:

方法 描述
ls.append(x) 在列表ls末尾添加一个元素x
ls.insert(i, x) 在列表ls的第i个位置增加元素x
ls.clear() 删除列表ls所有元素
ls.pop(i) 将列表ls的第i个元素取出并从ls中删除该元素
ls.remove(x) 将列表中出现的第一个元素x删除
ls.reverse() 将列表ls中的元素反转
ls.copy() 生成一个新列表,复制ls中的所有元素

5.1.7 列表元素的增加操作

方法 描述
ls.append(x) 在列表ls末尾添加一个元素x
ls.insert(i, x) 在列表ls的第i个位置增加元素x
ls.extend() 在列表ls末尾至少添加一个元素
切片 在列表任意位置至少添加一个元素
lst = [10, 20, 30]
lst1 = ['hello', 'world']
print(lst)
print("添加元素之前id:", id(lst))

# 1. append(要添加的元素)  在列表末尾添加一个元素
lst.append(100)
print(lst)
print("添加元素之后id:", id(lst))

# 2. extend() 在列表末尾至少添加一个元素
lst.extend(lst1)
print(lst)

# 3. insert(索引, 要添加的对象)在任意(指定索引)位置添加一个元素
lst.insert(4, 'python')
print(lst)

# 4.切片 在任意位置添加至少一个元素
# lst[要切的位置索引 : 结束位置 : 步长默认为以1]=要添加的列表
lst[1:5:3] = lst1
print(lst)

在这里插入图片描述

5.1.8 列表元素的删除操作

方法 描述
ls.remove(x) 将列表中出现的第一个元素x删除
pop() 删除一个指定索引位置上的元素
切片 一次至少删除一个元素
clear() 清空列表
del 删除列表

代码:

lst = [10, 20, 30, 20, 40, 50, 60, 70, 80, 90]
# 1. remove(要移除的元素)  从列表中移除一个元素,重复只删一个,元素不存抛异常
lst.remove(20)
print(lst)

# 2. pop(索引)  删除一个指定索引上的元素,不指定索引删除最后一个元素,指定索引不存在抛异常
lst.pop(7)
print(lst)
lst.pop()
print(lst)

# 3. 切片操作 删除至少一个元素,将产生一个新的列表对象
new_lst = lst[1:4]  # 索引 1,2,3,产生一个新列表
print(new_lst)
'''不产生新的列表对象,而是删除原列表中的内容'''
lst[1:6] = []
print(lst)

# 4. clear 清空列表
lst.clear()
print(lst)

# 5. del  删除列表
del lst
# print(lst)

运行结果:
在这里插入图片描述

5.1.9 列表元素的修改操作

# 为指定索引的元素赋予一个新值
lst = [10, 20, 30, 40]
lst[2] = 100
print(lst)

# 为指定的切片赋予一个新值
lst[1:2] = [100, 200, 300]
print(lst)

在这里插入图片描述

5.1.10 列表元素的排序操作

sort()方法

class list(object):
    def sort(self, *args, **kwargs): # real signature unknown
        """
        Sort the list in ascending order and return None.
        
        The sort is in-place (i.e. the list itself is modified) and stable (i.e. the
        order of two equal elements is maintained).
        
        If a key function is given, apply it once to each list item and sort them,
        ascending or descending, according to their function values.
        
        The reverse flag can be set to sort in descending order.
        """
        pass
按升序对列表进行排序,返回 None
排序是原地的(即列表本身被修改)和稳定的(即
保持两个相等元素的顺序)。
如果给出了一个键函数,对每个列表项应用一次,并对它们进行排序,
根据它们的函数值升序或降序。
反向标志可以设置为降序排序。

内置函数sorted()

def sorted(*args, **kwargs): # real signature unknown
    """
    Return a new list containing all items from the iterable in ascending order.
    
    A custom key function can be supplied to customize the sort order, and the
    reverse flag can be set to request the result in descending order.
    """
    pass
返回一个新列表,其中包含可迭代对象中的所有项,按升序排列。
可以提供自定义键函数来自定义排序顺序
可以设置 reverse标志,以降序请求结果。
# 1.调用 sort()方法,sort()方法默认是 升序 排序,在调用该方法后,列表中的元素会按照从小到大的顺序排列
lst = [23, 46, 67, 12, 8, 27]
lst.sort()
print(lst)

# 2.如果我们想让其降序 排序,则可以通过指定关键字参数reverse
lst = [20, 40, 10, 98, 54]
print('排序前的列表', lst)
lst.sort(reverse=True)
print('排序后的列表', lst)

# 升序
lst.sort(reverse=False)
print('排序后的列表', lst)

# 使用内置函数 sorted() 对列表进行排序,将产生一个新的对象
lst = [20, 40, 10, 98, 54]
print(sorted(lst))  # 未指定升序
print(sorted(lst, reverse=True))  # 指定降序

在这里插入图片描述

5.1.11 列表生成式

列表生成式: 简称列表生成的公式。

语法格式:
lst = [列表元素表达式 for 自定义变量 in 可迭代对象]
例:lst = [i for i in range(1, 11)]

lst = [i for i in range(1, 11)]
print(lst)

在这里插入图片描述


5.2 字典

5.2.1 什么是字典

字典: Python内置的数据结构之一,与列表一样是一个 可变序列,以 键值对的方式存储数据,字典是一个 无序的序列Python语言中的字典使用大括号 { }建立,每个元素是一个键值对。

键值对”是组织数据的一种重要方式,广泛应用在当代大型信息系统中,如Web系统。键值对的基本思想是将“值”信息关联一个“键”信息,进而通过键信息找对应的值信息,这个过程叫映射。Python语言中通过字典类型实现映射。

5.2.2 字典示意图

在这里插入图片描述

5.2.3 字典的原理

字典的原理: Python中的字典是根据 key查找 value所在的位置

5.2.4 创建字典的方式

python中创建一个字典对象,常用的方式有两种。

  1. 第一种方式,使用花括号{ }
'''
第一种方式,使用花括号{},语法如下所示
使用{}创建字典
scores = {'张三': 100, '李四': 98, '王五': 45}
'''
scores = {'张三': 29, '李四': 10, '王五': 40}
print(scores)  # {'张三': 29, '李四': 10, '王五': 40}
print(type(scores))  # <class 'dict'>
  1. 第二种方式,使用内置函数dict()
'''第二种方式,使用内置函数dict()。dict即dictionary(字典)的缩写,语法如下所示。
字典名 = dict(键1=值1, 键2=值2)
'''
dic = dict(name='张三', age=20)
print(dic)  # {'name': '张三', 'age': 20}
  1. 创建空字典
d = {}
print(d)  # {}

5.2.5 字典元素的获取

根据键key获取字典的值value

# 1.使用 [] 获取
scores = {'张三': 29, '李四': 10, '王五': 40}
print('张三' in scores)  # True
print('张三' not in scores)  # False
print(scores['张三'])  # 29
 
# 2.使用 get()方法
print(scores.get('张三'))  # 29
print(scores.get('柽柳'))   # None
print(scores.get('麻子', 99))  # 99是在查找 麻子值的(value)不存在时的一个默认值

5.2.6 字典的操作函数

Python内置数据结构:字典dict()类源代码:

class dict(object):
    """
    dict() -> new empty dictionary
    dict(mapping) -> new dictionary initialized from a mapping object's
        (key, value) pairs
    dict(iterable) -> new dictionary initialized as if via:
        d = {}
        for k, v in iterable:
            d[k] = v
    dict(**kwargs) -> new dictionary initialized with the name=value pairs
        in the keyword argument list.  For example:  dict(one=1, two=2)
    """
    def clear(self): # real signature unknown; restored from __doc__
        """ D.clear() -> None.  Remove all items from D. """
        pass

    def copy(self): # real signature unknown; restored from __doc__
        """ D.copy() -> a shallow copy of D """
        pass

    @staticmethod # known case
    def fromkeys(*args, **kwargs): # real signature unknown
        """ Create a new dictionary with keys from iterable and values set to value. """
        pass

    def get(self, *args, **kwargs): # real signature unknown
        """ Return the value for key if key is in the dictionary, else default. """
        pass

    def items(self): # real signature unknown; restored from __doc__
        """ D.items() -> a set-like object providing a view on D's items """
        pass

    def keys(self): # real signature unknown; restored from __doc__
        """ D.keys() -> a set-like object providing a view on D's keys """
        pass

    def pop(self, k, d=None): # real signature unknown; restored from __doc__
        """
        D.pop(k[,d]) -> v, remove specified key and return the corresponding value.
        If key is not found, d is returned if given, otherwise KeyError is raised
        """
        pass

    def popitem(self, *args, **kwargs): # real signature unknown
        """
        Remove and return a (key, value) pair as a 2-tuple.
        
        Pairs are returned in LIFO (last-in, first-out) order.
        Raises KeyError if the dict is empty.
        """
        pass

    def setdefault(self, *args, **kwargs): # real signature unknown
        """
        Insert key with a value of default if key is not in the dictionary.
        
        Return the value for key if key is in the dictionary, else default.
        """
        pass

    def update(self, E=None, **F): # known special case of dict.update
        """
        D.update([E, ]**F) -> None.  Update D from dict/iterable E and F.
        If E is present and has a .keys() method, then does:  for k in E: D[k] = E[k]
        If E is present and lacks a .keys() method, then does:  for k, v in E: D[k] = v
        In either case, this is followed by: for k in F:  D[k] = F[k]
        """
        pass

    def values(self): # real signature unknown; restored from __doc__
        """ D.values() -> an object providing a view on D's values """
        pass

    def __contains__(self, *args, **kwargs): # real signature unknown
        """ True if the dictionary has the specified key, else False. """
        pass

    def __delitem__(self, *args, **kwargs): # real signature unknown
        """ Delete self[key]. """
        pass

    def __eq__(self, *args, **kwargs): # real signature unknown
        """ Return self==value. """
        pass

    def __getattribute__(self, *args, **kwargs): # real signature unknown
        """ Return getattr(self, name). """
        pass

    def __getitem__(self, y): # real signature unknown; restored from __doc__
        """ x.__getitem__(y) <==> x[y] """
        pass

    def __ge__(self, *args, **kwargs): # real signature unknown
        """ Return self>=value. """
        pass

    def __gt__(self, *args, **kwargs): # real signature unknown
        """ Return self>value. """
        pass

    def __init__(self, seq=None, **kwargs): # known special case of dict.__init__
        """
        dict() -> new empty dictionary
        dict(mapping) -> new dictionary initialized from a mapping object's
            (key, value) pairs
        dict(iterable) -> new dictionary initialized as if via:
            d = {}
            for k, v in iterable:
                d[k] = v
        dict(**kwargs) -> new dictionary initialized with the name=value pairs
            in the keyword argument list.  For example:  dict(one=1, two=2)
        # (copied from class doc)
        """
        pass

    def __iter__(self, *args, **kwargs): # real signature unknown
        """ Implement iter(self). """
        pass

    def __len__(self, *args, **kwargs): # real signature unknown
        """ Return len(self). """
        pass

    def __le__(self, *args, **kwargs): # real signature unknown
        """ Return self<=value. """
        pass

    def __lt__(self, *args, **kwargs): # real signature unknown
        """ Return self<value. """
        pass

    @staticmethod # known case of __new__
    def __new__(*args, **kwargs): # real signature unknown
        """ Create and return a new object.  See help(type) for accurate signature. """
        pass

    def __ne__(self, *args, **kwargs): # real signature unknown
        """ Return self!=value. """
        pass

    def __repr__(self, *args, **kwargs): # real signature unknown
        """ Return repr(self). """
        pass

    def __reversed__(self, *args, **kwargs): # real signature unknown
        """ Return a reverse iterator over the dict keys. """
        pass

    def __setitem__(self, *args, **kwargs): # real signature unknown
        """ Set self[key] to value. """
        pass

    def __sizeof__(self): # real signature unknown; restored from __doc__
        """ D.__sizeof__() -> size of D in memory, in bytes """
        pass

    __hash__ = None

字典类型的一些通用操作函数

操作函数 描述
dict() 生成一个字典
len(d) 字典d元素的个数(长度)
min(d) 字典d中键的最最小值
max(d) 字典d中键的最最大值

5.2.7 字典的操作方法

字典类型有一些操作方法,使用的语法格式为:
<字典对象名>.<方法名>(<方法参数>)

操作方法 描述
d.keys() 返回字典d所有键的信息
d.values() 返回字典d所有值的信息
d.items() 返回字典d所有键值对
d.get(key, default) 键存在则返回相应值,否则返回默认值default
d.pop(key, default) 键存在则返回相应值,同时删除键值对,否则返回默认值default
d.popitem() 随机从字典中取出一个兼职对,以元组(key, value)的形式返回,同时将该键值对从字典中删除。
d.clear() 删除所有的键值对,清空字典

5.2.8 字典元素的增删改操作

scores = {'张三': 29, '李四': 10, '王五': 40}

  • key的判断
# key的判断
scores = {'张三': 29, '李四': 10, '王五': 40}
print('张三' in scores)  # True
print('张三' not in scores)  # True
  • 字典的删除
# 字典元素的删除
del scores['张三']  # 删除指定的键值对
print(scores)  # {'李四': 10, '王五': 40}
  • 字典的清除
# 清空字典 clear()方法
scores.clear()
print(scores)  # {}
  • 字典元素的新增
# 新增元素
scores['陈六'] = 20
print(scores)  # {'陈六': 20}
  • 字典元素值的修改
# 修改value
scores['陈六'] = 100
print(scores)  # {'陈六': 100}

5.2.9 字典视图的获取

  • 获取所有的键, <字典对象名>.keys()方法 返回值为列表
# 获取所有的键 .keys()方法 返回值为列表
scores = {'张三': 29, '李四': 10, '王五': 40}
print(scores.keys())  # dict_keys(['张三', '李四', '王五'])
print(type(scores.keys()))  # <class 'dict_keys'>
print(list(scores.keys()))  # ['张三', '李四', '王五']
  • 获取所有的值,<字典对象名>.value()方法,返回值为列表
# 获取所有的值 <字典对象名>.value()方法 返回值为列表
dict_values = scores.values()
print(dict_values)  # dict_values([29, 10, 40])
print(type(dict_values))  # <class 'dict_values'>
print(list(dict_values))  # [29, 10, 40]
  • 获取所有的键值对,<字典对象名>.items()方法 返回值为元组
#  获取所有的键值对  返回值为元组
print(scores.items())
# dict_items([('张三', 29), ('李四', 10), ('王五', 40)])

5.2.10 字典元素的遍历

字典元素的遍历:使用for-in循环

# 字典元素的遍历
scores = {'张三': 29, '李四': 10, '王五': 40}
for item in scores:
    print(item, scores[item])

在这里插入图片描述

5.2.11 字典的特点

1.键 key 不可以重复,值 value可以重复

2.字典中元素是无序的

3.字典会浪费大的内存,是一种使用空间换时间的数据结构,但是查询速度快。

5.2.12 字典生成式

  • 内置函数zip()
用于将可迭代对象作为参数,将对象中对应的元素打包成一个元组,然后返回由这些元组组成的列表
  • 字典生成式:
    {key: value for key, value in zip(items,prices)}
items = ['Fruits', 'Books', 'Others']  # 键的列表
prices = [96, 78, 85]  # 值的列表
d = {items: prices for items, prices in zip(items, prices)}  # 两个列表生成一个字典
print(d)  # {'Fruits': 96, 'Books': 78, 'Others': 85}

# .upper()方法 键字母变大写
d = {items.upper(): prices for items, prices in zip(items, prices)}  # 两个列表生成一个字典
print(d)  # {'Fruits': 96, 'Books': 78, 'Others': 85}


items = ['Fruits', 'Books', 'Others']  # 键的列表
prices = [96, 78, 85]  # 值的列表
a = dict(zip(items, prices))
print(a)  # {'Fruits': 96, 'Books': 78, 'Others': 85}

在这里插入图片描述


5.3 元组

5.3.1 什么是元组

元组:Python内置的数据结构之一,是一个不可变序列。

不可变序列与可变序列

  • 不变可变序:字符串元组

不变可变序列:没有增、删,改的操作

# 修改后对象地址会发生改变
s = 'hello'
print(id(s))  # 2533879685808
s = s + 'world'
print(id(s))  # 2533879671984
print(s)  # helloworld
  • 可变序列:列表字典

可变序列:可以对序列执行增、删、改操作,对象地址不发生更改

# 可变序列:可以执行增删改操作,对象地址不发生改变
lst = [10, 20, 40]
print(id(lst))
lst.append(50)
print(id(lst))

在这里插入图片描述

5.3.2 元组的创建方式

  1. 直接小括号()
# 1.使用()
t = ('python', 'java')
print(t)  # ('python', 'java')
print(type(t))  # <class 'tuple'>
  1. 使用内置函数tuple()
# 2.使用内置函数tuple()
t2 = tuple(('java', 'python'))
print(t2)  # ('java', 'python')
print(type(t2))  # <class 'tuple'>
  1. 只包含一个元组的元素需要使用逗号,和小括号()
# 3.只包含一个元素,需要使用逗号和小括号(只有一个元素的时候必须加上,)
t3 = ('hello', )
print(t3)  # ('hello',)
print(type(t3))  # <class 'tuple'>
  1. 空元组创建方式
# 空元组创建方式
t4 = ()
t5 = tuple()
print('空元组', t4, t5)  # 空元组 () ()
print('空列表', [], list())  # 空列表 [] []
print('空字典', {}, dict())  # 空字典 {} {}
# print('空集合', {}, set())

为什么要将元组设计成不可变序列

在多任务环境下,同时操作对象时不需要加锁,因此,在程序中尽量使用不可变序列 。
注意事项:元组中存储的是对象的引用

  • 如果元组中对象本身不可对象,则不能再引用其它对象。
  • 如果元组中的对象是可变对象,则可变对象的引用不允许改变,但数据可以改变。

这样设计的原因是,元组的不可变性保证了数据的完整性,这样如果有多个地方都用到了元组,我们可以保证它的数据不会被改变。并且,相较于列表,元组的读取速度更快,占用的内存空间更小,并且可以作为字典的key去使用。

# 这样设计的原因是,元组的不可变性保证了数据的完整性,这样如果有多个地方都用到了元组,我们可以保证它的数据不会被改变。
# 并且,相较于列表,元组的读取速度更快,占用的内存空间更小,并且可以作为字典的key去使用。

t = (10, [20, 30], 40)
print(t)  # (10, [20, 30], 40)
print(type(t), id(t))  # <class 'tuple'> 2972759079744
print(t[0], type(t[0]), id(t[0]))  # 10 <class 'int'> 140726042761152
print(t[1], type(t[1]), id(t[1]))  # [20, 30] <class 'list'> 2972768483776
print(t[2], type(t[2]), id(t[2]))  # 40 <class 'int'> 140726042762112
t[1].append(100)
print(t, id(t))  # (10, [20, 30, 100], 40) 2972759079744

5.3.3 元组的遍历

元组是可迭代对象,所以可以使用for...in进行遍历

# 元组遍历
t = ('hello', 'java', 90)
print(t[0])
print(t[1])
print(t[2])

for item in t:
    print(item, end=' ')

在这里插入图片描述
列表类型覆盖了元组类型的所有主要功能。

5.4 集合

什么是集合

  1. Python内置数据结构之一
  2. 与列表,字典一样都属于可变序列
  3. 集合是没有value的字典
  4. 集合中的元素是无序的,元素不可以重复

5.4.1 集合操作符

操作符的运算 描述
S-T 返回一个新集合,包括在集合S中但不会在集合T
S&T 返回一个新集合,包括同时在集合ST中的元素
S^T 返回一个新集合,包括ST中的非共同元素
`S\ T` 返回一个新集合,包括集合ST中的所有元素

5.4.2 常用集合操作函数或方法

函数或方法 描述
S.add(x) 如果数据项x不在S中,将x增加到S
S.remove(x) 如果x集合在S中,移除该元素;不在则产生KeyError异常
S.clear() 移除S中的数据项
len(S) 返回集合S的元素个数
x in S 如果xs的元素,返回True;否则返回False
x not in S 如果xs的元素,返回True;否则返回False

5.4.3 集合的创建方式

  1. 使用 {}
s = {1, 2, 3, 4, 5, 6, 6}
print(s)
# {1, 2, 3, 4, 5, 6}
  1. 使用内置函数 set()
s1 = set(range(1, 7))
print(s1)
# {1, 2, 3, 4, 5, 6}
print(set('python'))
# {'p', 'h', 'n', 'o', 't', 'y'}

5.4.4 集合元素的判断操作

操作函数 描述
a in s 如果as的元素,返回True;否则返回False
a not in s 如果 a不是s的元素,返回True;否则返回False
# 判断元素是否在集合中
s = {10, 20, 30, 40, 50, 100}
print(10 in s)  # True
print(10 not in s)  # False

5.4.5 集合的新增操作

# 集合元素的新增操作
s = {10, 20, 30, 40, 50, 100}
s.add(80)  # 添加一个元素
print(s)  
# {100, 40, 10, 80, 50, 20, 30}
s.update({66, 77, 88})  # 添加多个元素(一次至少添加一个元素 )
print(s)
# {66, 100, 40, 10, 77, 80, 50, 20, 88, 30}
s.update((11, 22, 33))
print(s)
# {33, 66, 100, 40, 10, 11, 77, 80, 50, 20, 22, 88, 30}

5.4.6 集合的删除操作

# 集合元素的删除操作
#  1.调用remove方法,一次删除一个指定元素,元素不存在抛异常
s.remove(100)
print(s)
# {33, 66, 40, 10, 11, 77, 80, 50, 20, 22, 88, 30}

# 2.调用discard方法,一次删除一个指定元素,元素不存在不抛异常
s.discard(300)
print(s)
# {33, 66, 40, 10, 11, 77, 80, 50, 20, 22, 88, 30}

# 3.调用pop方法,一次只删除一个任意元素
s.pop()
print(s)
# {66, 40, 10, 11, 77, 80, 50, 20, 22, 88, 30}

# 4.调用clear方法,清空集合
s.clear()
print(s)
# set()

六、字符串

字符串又称为字符序列,根据字符串的内容多少分为 单行字符串多行字符串

单行字符串可以由一对单引号' '或一对双引号" "作为边界,单引号和双引号的作用相同。当使用单引号时双引号可以作为字符串的一部分,使用双引号时,单引号可以作为字符串的一部分。

多行字符串由一对三单引号''' '''或三双引号""" """作为边界来表示,二者作用相同。


6.1 字符串的驻留机制

6.1.1 字符串

字符串: Python基本数据类型:是一个不可变序列

6.1.2什么叫字符串的驻留机制

仅保存一份相同且不可变字符串的方法,不同的值会被保存在字符串的驻留池中。

Python的驻留机制会对相同的字符串只保留一份拷贝,后续创建相同的字符串时,不会开辟新的空间,而是把字符串的地址付给新创建的变量。

# -*- coding: utf-8 -*-
# @File  : demo24.py
# @author: Flyme awei
# @email : 1071505897@qq.com
# @Time  : 2022/8/11 16:07


a = 'python'
b = "python"
c = '''python'''
print(a, id(a))
print(b, id(b))
print(c, id(c))

在这里插入图片描述

6.1.3 字符串驻留机制的几种情况(交互式)

  1. 字符串的长度为1
  2. 符合标识符的字符串(只包含字母,数字,下划线)
  3. 字符串只在编译是进行驻留,而非运行时
  4. [-5,256]之间的整数数字
>>> s1 = ''
>>> s2 = ''
>>> s1 is s2
True
>>>
>>> s1 = 'a'
>>> s2 = 'a'
>>> s1 is s2
True
>>>
>>> s1 = 'abc_def'
>>> s2 = 'abc_def'
>>> s1 is s2
True
>>> s1 = 'abc%def'
>>> s2 = 'abc%def'
>>> s1 == s2
True
>>> s1 is s2
False
>>>
>>> a = 256
>>> b = 256
>>> a is b
True
>>> a = 257
>>> b = 257
>>> a is b
False
>>> a == b
True
>>> 

6.1.4 强制2个字符串指向同一个对象

sys中的 intern方法强制两个字符串指向同一个对象
'''sys中的intern方法强制两个字符串指向同一个对象'''
import sys
a = 'abc%'
b = 'abc%'
print(a is b)  # True
a = sys.intern(b)
print(id(a), id(b))  # 2989905230512 2989905230512

PyCharm对字符串进行了优化处理 。

6.1.5 字符串驻留机制的优缺点

  当需要值相同的字符串时,可以直接从字符串池里拿来使用,避免频繁的创建和销毁,提升效率和节约内存,因此拼接字符串和修改字符串是会比较影响性能的。

  在需要进行字符串拼接时建议使用 str类型的join方法,而非+ ,因为join()方法是先计算出所有字符中的长度,然后再拷贝,只new一次对象,效率要比"+"效率高 。


6.2 字符串类型的操作

6.2.1 字符串操作符

操作符 描述
+ x+y,连接两个字符串xy
* x*nn*x,复制n次字符串x
in x in s,如果xs的字串,返回True,否则返回False

6.2.2 字符串处理函数

函数 描述
len(x) 返回字符串x的长度,也可返回其它组合数据类型元素的个数
str(x) 返回任意类型x所对应的字符串形式
char(x) 返回Unicode编码x对应的单字符
ord(x) 返回x表示的Unicode编码
hex(x) 返回整数x对应十六进制的小写形式字符串
oct(x) 返回整数x对应八进制的小写形式字符串

6.2.3 字符串处理方法

方法 描述
s.lower() 字符串s全部转为小写
s.upper() 字符串s全部转为大写
s.split(sep=None) 返回一个列表,由s根据sep被分割的部分构成,省略sep默认以空格分割
s.count(sub) 返回字串sub出现的次数
s.replace(old, new) 返回字符串s的副本,所有old字串被替换为new
s.center(width, fillchar) 字符串居中函数,fillchar参数可选
s.strip(chars) 从字符串s中去掉咋其左侧和右侧chars中出现的字符
s.join(iter) iter变量的每一个元素增加一个s字符串

6.2.4 字符串的查询操作

方法名称 作用
index() 查找字串substr第一次出现的位置,如果查找的字串不存在,抛ValueError异常
rindex() 查找字串substr最后一次出现的位置,如果查找的字串不存在,抛ValueError异常
find() 查找字串substr第一次出现的位置,如果查找的字串不存在,返回-1
rfind() 查找字串substr最后一次出现的位置,如果查找的字串不存在,返回-1
'''
index()查找第一次出现的位置  抛异常
rindex()查找最后一次次出现的位置  抛异常

find()查找第一次出现的位置  不抛异常,返回值为-1
rfind()查找最后一次出现的位置  抛异常
'''
s = 'hello,hello'
print(s.index('o'))
print(s.rindex('o'))
print(s.find('lo'))
print(s.find('ui'))  # -1

在这里插入图片描述

6.2.5 字符串大小写转换操作

方法 作用
upper() 把所有的字符串转换为大写字母
lower() 把所有的字符串转换为小写字母
swapcase() 大写字符转换为小写字符,将小写字符转换为大写字符。
capitalize() 使第一个字符大写字母,其余字符为小写字母
title() 返回字符串的一个版本,其中每个单词都有标题。更具体地说,单词以大写字母开头,其余都以大写字母开头区分大小写的字符小写。
# 字符串的大小写转换
# 1.upper()把字符串中的所有字符转为大写
# 2.lower()把字符串中的所有字符都转换为小写
# 3.swap case() 大转小,小转大
# 4.capitalize()把第一个字符转为大写,其余字符转为小写
# 5.title()把字符串首字母转换为大写,把剩余的转换为小写

s = 'hellopython'
print(s.upper())  # 转大写
print(s.lower())  # 转换后id改变,会产生一个新的空间
print(s.swapcase())
print(s.capitalize())
print(s.title())

在这里插入图片描述

6.2.6 字符串内容对齐操作方法

方法 作用
center(width,'') 返回一个居中长度为width的字符串。使用指定的填充字符(默认为空格)填充。
ljust(width,' ') 返回长度为width的左对齐字符串。使用指定的填充字符(默认为空格)填充。
rjust(width,' ') 返回长度为width的右对齐字符串。使用指定的填充字符(默认为空格)填充。
zfill('int') 在左侧填充数字字符串,以填充给定宽度的字段。字符串永远不会被截断。
'''字符串的对其操作'''
# 1.center 居中对齐
s = 'hello,python'
print(s.center(100, '*'))

# 2.ljust 左对齐
print(s.ljust(100, '*'))

# 3.rjust 右对齐
print(s.rjust(100, '*'))

# 3.zfill 右对齐,左侧为0填充
print(s.zfill(100))

在这里插入图片描述

6.2.7 字符串的劈分操作

方法 作用
split() 返回字符串中的单词列表,使用sep作为分隔符字符串。sep用来分割字符串的分隔符。None(默认值)表示根据任何空格进行分割,并从结果中丢弃空字符串。maxsplit最大分割次数。-1(默认值)表示无限制。
rsplit() 返回字符串中的单词列表,使用sep作为分隔符字符串。sep用来分割字符串的分隔符。None(默认值)表示根据任何空格进行分割,并从结果中丢弃空字符串。maxsplit最大分割次数。-1(默认值)表示无限制。劈叉从绳子的末端开始,一直到前面。
# 字符串的劈分操作  split
# 1. split从字符串左侧开始分割,默认值为空格字符串,返回值是一个列表
#  以通过参数sep指定劈分字符串是劈分符
# 通过maxsplit指定劈分字符串的最大劈分次数

s = 'hello#world#python'
lst = s.split('#')
print(lst)
s1 = 'hello|world|python'
print(s1.split())
print(s1.split(sep='|'))
print(s1.split())
s1 = 'hello|world|python'
print(s1.split())
print(s1.split(sep='|', maxsplit=1))
# 以参数sep 指定劈分字符串是劈分符
print('-----------------------')

# 2.rsplit  从右侧开始劈分
print(s1.rsplit(sep='|', maxsplit=1))

在这里插入图片描述

6.2.8 判断字符串的方法

方法 作用
isidentifier() 判断字符串是合法标识符
isspace() 判断字符串是否全部由空字符串组成(回车,换行,水平制表)
isalpha() 判断是否全部由字符组成
isdecimal() 判断是否全部由十进制数字组成
isnumeric() 判断是否全部由数字组成
isalnum() 判断字符串是否全部由字母和数字组成
# 1. isidentifier 判断字符串是合法标识符
s = 'hello, python'
print('1.', s.isidentifier())  # False
print('2.', 'hello'.isidentifier())  # True

# 2. isspase 判断字符串是否全部由空字符串组成(回车,换行,水平制表)
print('   '.isspace())
print('-----------------------')

# 3. isalpha  判断是否全部由字符组成
print('fhaisdfh'.isalpha())

# 4. isnumeric 判断是否全部由数字组成
print('67867'.isnumeric())

# 5. isdecimal 判断是否全部由十进制数字组成
print('78'.isdecimal())

# 6. iszlnum 判断字符串是否全部由字母和数字组成
print('yut6786'.isalnum())

在这里插入图片描述

6.2.9 字符串的替换与合并操作

方法 作用
replace() 返回一个副本,其中所有出现的子字符串old都被new替换。count替换的最大次数。-1(默认值)表示替换所有匹配项。如果给出了可选参数count,则只出现第一个count更换。
join() 连接任意数量的字符串。调用其方法的字符串被插入到每个给定字符串之间。结果以新字符串的形式返回。例如: '.'.join(['ab', 'pq', 'rs']) -> 'ab.pq.rs'
# 1.字符串的替换  replace()
s = 'hello,Python'
print(s.replace('Python', 'java'))
s1 = 'hello,python,python,python'
print(s1.replace('python', 'java', 2))  
# 通过第三个参数指定最大替换次数

# 2.字符串合并 join() 将列表或元组中字符串合并成一个字符串
lst = ['hello', 'java', 'python']
print('|'.join(lst))
print(''.join(lst))

t = ('hello', 'java', 'python')
print(''.join(t))

在这里插入图片描述

6.2.10 字符串的比较运算

两个字符串进行比较时,比较的是 ordinal value (原始值) ,如果相等则继续比较下一个字符,知道两个字符串不相等
运算符 含义
> 大于
< 小于
== 等于
>= 大于等于
<= 小于等于
!= 不等于
# 字符窜的比较
# 运算符 > < == <= >= !=
print('apple' > 'app')  # True
print('apple' > 'banana')

'''调用内置函数ord可以得到指定字符的ordinal value '''
print(ord('a'), ord('b'))
print(ord('刘'), ord('张'))

'''== 与 is 的区别'''
# == 比较的是 value
# is 比较的是 id
a = b = 'pthon'
print(a is b)
print(a == b)

在这里插入图片描述

6.2.11 字符串的切片操作

对字符串中某个子串或区间的检索称为切片。

语法如下:

字符串或字符串变量[N:M]
切片获取字符串 NM(不包含 M)的子字符串,其中, NM为字符串的索引序号,可以混合使用正向递增序号和反向递减序号。切片要求 NM 都在字符串的索引区间,如果 N大于 M,则返回空字符串。如果 N缺失,则默认将 N设为 0;如果 M缺失,则默认表示到字符串结尾。
# 字符串的切片操作
# 字符串是不可变类型 不能进行 增 删 改 操作
# 切片将产生新的对象
s = 'hello,python'
# print(s[start : end : step])
print(s[3])
print(s[:3])
print(s[:3:2])

print(s[::-1])  # 默认从字符串最后一个语速开始切,到字符串第一个元素结束

在这里插入图片描述

6.2.12 格式化字符串

  • 为什么要格式化字符串
在字符串中整合变量是需要使用字符串的格式化方法。
字符串格式化用于解决字符串和变量同时输出的格式安排问题。
  • 格式化字符串的三种方式
  1. %作为占位符
  2. {}作为占位符''.format()
  3. f '我叫%s,今年%d岁了' %s 占了一个字符串 %d 占了一个整数

在这里插入图片描述
s = "python"

  • <填充><对齐><宽度>
符号 描述
{:25}.format(s) 默认左对齐
{:1}.format(s) 指定宽度为1,不足变量s的宽度,以实际变量宽度为准
{:^25}.format(s) 居中对齐
{:>25}.format(s) 右对齐
{:*^25}.format(s) 居中对齐,填充*
{:+^25}.format(s) 居中对齐填充+
{:^1}.format(s) 指定宽度为1,不足变量s的宽度,以实际变量宽度为准
  • <,><.精度><类型>,其中,逗号(,)用于显示数字类型的千分位分隔符。
符号 描述
"{:-^25,}".format(1234567890) '------1,234,567,890------'
"{0:-^25}".format(1234567890) '-------1234567890--------'
  • <.精度>有小数点(.)开头。
符号 描述
"{:2f}".format(12345.6788890) '12345.67'
"{:25.3f}".format(12345.67890) ' 12345.679'
"{:.5}".format("123456789") '12345'
"{:.15)".format('123456789') '123456789'
# 格式化字符串

name = '张三'
age = 20
print('我叫%s,今年%d岁了' % (name, age))  # % 作为占位符

print('我的名字是{0},我今年{1}岁了'.format(name, age))  # {} 作为占位符

print(f'我叫{name},今年{age}岁')

# 表示宽度 %10d
print('%10d' % 99)

# 表示小数点位数 %.nf  精度 :.n
print('%.3f' % 3.78234685)  # 三位小数

print('{:.3}'.format(3.34638567))  # 三位有效数字

在这里插入图片描述

6.2.13 数值与字符串类型转换函数

函数 描述
int(x) x转换为整数,x可以是浮点数或数字类字符串
folat(x) x转换为浮点数,x可以是整数或数字类字符串
str(x) x转换为整数,x可以是整数或浮点数

6.3 字符串编码转换

6.3.1 为什么要进行字符串编码转换

在这里插入图片描述

6.3.2 编码与解码的方式

编码s.encode()将字符串转换为二进制数据(bytes)
解码s.decode()bytes类型的数据转换成字符串类型

s = '人生苦短,我用Python!'
# 编码
print(s.encode(encoding='utf_8'))  # utf-8 一个中文占两个字节
print(s.encode(encoding='GBK'))  # GBK 一个中文占三个字节

# 解码
byte = s.encode(encoding='GBK')  # 编码
print(byte.decode(encoding='GBK'))  # 解码
# 编码和解码格式要相同

在这里插入图片描述


七、函数和代码复用

7.1 函数的定义

语法格式:

def  函数名 ([参数列表]):
    函数体
    return 返回值列表 

例:

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/14 17:02

def calc(a, b):  # 函数定义
    c = a + b
    return c

7.2 函数的调用

函数的定义也叫函数“声明”,定义后的函数不能直接运行,需要经过“调用”才能得到运行。

语法格式:

函数名(实际赋值参数列表)
# 函数创建
def calc(a, b):  # 函数定义
    c = a + b
    return c

'''
函数调用:
函数名(实际参数列表)'''
d = calc(10, 20)  # 函数调用
print(d)  # 30

7.3 函数的参数传递

函数的参数在定义时可以指定默认值,当函数被调用时,如果没人传入对应的参数时,则使用函数定义时的默认值替代。

语法格式

def  函数名(非可选参数列表,可选参数列表=默认值):
    函数体
    return 返回值列表 

7.3.1 形式参数与实际参数

  • 函数定义处的参数称为形式参数
  • 函数调用处的参数称为实际参数

7.3.2 位置传参与关键字传参

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/14 17:02

# 函数的参数传递
'''
1.位置传参 根据形参对应的位置进行参数传递
2.关键字实参 根据形参名称进行参数传递
'''

def calc(a, b):  # a b 为函数定义当处的形式参数
    c = a + b
    return c


calc(10, 20)  # 10 20 为函数调用处的实际参数

# 位置实参
print(calc(10, 20))  

# =左侧的变量的名称称为关键字参数
print(calc(b=10, a=20))  

在这里插入图片描述

7.3.3 可变对象与不可变对象的参数传递

在函数调用过程中,进行参的传递:

  • 如果是不可变对象,函数体的修改不会影响实参的值
  • 若果是可变对象,函数体的改变会影响到实参的值
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/14 17:02

'''
在函数调用过程中,进行参的传递:
如果是不可变对象,函数体的修改不会影响实参的值
若果是可变对象,函数体的改变会影响到实参的值
'''


def fun(arg1, arg2):
    print('arg1=', arg1)
    print('arg2=', arg2)
    arg1 = 100  # arg1 的修改为100不会影响n1的值
    arg2.append(200)  # are2 的修改会影响到n2的值
    print('arg1=', arg1)  # 100
    print('arg2=', arg2)  # [10, 20, 200]
    return arg1, arg2


a = 10
b = [10, 20]
print('a', a)  # 10
print('b', b)  # [10, 20]
x = fun(a, b)  # 位置传参 arg1,arg2时是函数定义处的形参,n1,n2是函数调用处的实参, 实参和形参的名称可以不一致
print(a)  # 10
print(b)  # [10, 20, 200]
print(x)  # (100, [10, 20, 200])

在这里插入图片描述

7.3.4 个数可变的位置、关键字参数

  • 个数可变的位置参数: *arges
参数定义时,可能无法事先确定传递的位置实参的个数,使用可变的位置参数
  • 个数可变的关键字形参:**kwargs
参数定义时,可能无法事先确定传递的位置实参的个数,使用可变的位置参数

在一个函数的定义过程中,既有个数可变的 *arges位置形参,又有个数可变的**kwargs关键字形参,要求把位置形参放在前面

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/14 17:02


# 个数可变的位置参数 *arges
# 参数定义时,可能无法事先确定传递的位置时参的个数,使用可变的位置参数

def fun(*args):  # 函数定义时,可变的位置参数
    print(args)  # 返回值为元组
    # print(args[0])  返回元组的第1个对象


fun(10)  # (10, )
fun(10, 20)  # (10, 20)
fun(11, 22, 33)  # (11, 22, 33)


# 个数可变的关键字形参
# 参数定义时,可能无法事先确定传递的位置是参的个数,使用可变的位置参数
def fun(**kwargs):
    print(kwargs)  # 返回值是 字典


fun(a=10)  # {'a': 10}
fun(a=10, b=20, c=30)  # {'a': 10, 'b': 20, 'c': 30}

'''
def fun2(*args,*a)
    pass
以上程序报错,个数可变的的位置参数值能有一个
def fun2(**kwargs)
    pass
个数可变的关键字参数也只能有一个
'''

# 在一个函数的定义过程中,既有个数可变的位置形参,又有个数可变的关键字形参,要求把位置形参放在前面
def fun(*args, **kwargs):
    pass
    

在这里插入图片描述

7.3.5 函数参数总结(一)

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/14 17:02


def fun(a, b, c):  # a,b,c在函数定义处,所以是形式参数
    print('a=', a)
    print('b=', b)
    print('c=', c)


# 函数调用
fun(10, 20, 30)  # 函数调用时的参数传递,称为位置传参
lst = [11, 22, 33]
fun(*lst)  # 函数调用,将列表中的每个元素都转换为位置实参传递

fun(a=100, b=200, c=300)  # 函数调用,所以是关键字实参
dic = {'a': 111, 'b': 222, 'c': 333}
print(fun(**dic))  # 函数调用时,将字典中的键值对都转换为关键字实参传递

在这里插入图片描述

7.3.6 函数参数总结(二)

def fun(a, b=10):  # a是形式参数

    print('b=', b)


def fun2(*args):  # 个数可变的位置形参
    print(args)


def fun3(**kwargs):  # 个数可变的关键字形参
    print(kwargs)


# 函数调用
fun(10)
fun(10, 20)
fun2(10, 20, 30)
fun3(a=10, b=20)
print('--------------------------')


def fun4(a, b, *, c, d):  # 需求 c,d 只能采用关键字实参传递
    print('a=', a)
    print('b=', b)
    print('c=', c)
    print('d=', d)


# fun4(10, 20, 30, 40)   # 位置实参传递
fun4(a=10, b=20, c=30, d=40)  # 关键字实参传递
print('--------------------------')
fun4(10, 20, c=30, d=40)  # 前面两个进行位置实参传递,后面两个采用关键字实参传递

''' 函数定义时的形参顺序问题'''


def fun5(a, b, *, c, d, **kwargs):
    pass


def fun6(*args, **kwargs):
    pass


def fun7(a, b=10, *args, **kwargs):
    pass

在这里插入图片描述


7.4 函数的返回值

   return语句用来结束函数并将程序返回到函数调用的位置继续执行。 return语句可以出现在函数中的任何部分,同时可以将 0个、 1个或多个函数运算的结果返回给函数被调用处的变量。

函数的返回值

  1. 如果没有返回值(函数执行完毕后,不需要给调用处提供数据),return可以省略不写
  2. 若果返回值为1个,直接返回类型原类型
  3. 如果返回值为个,返回结果为元组
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/14 17:02

'''
函数的返回值:
(1)如果没有返回值(函数执行完毕后,不需要给调用处提供数据),return可以省略不写
(2)若果返回值为1个,直接返回类型原类型
(2)如果返回值为多个,返回结果为元组
'''


def fun(lst):
    odd = []
    even = []
    for i in lst:
        if i % 2:
            odd.append(i)
        else:
            even.append(i)
    print(odd, even)
    return odd, even  # 没有返回值,可以省略  如果返回值为多个,返回结果为元组


lst1 = [12, 34, 45, 57, 67, 78, 89]
print(lst1)

# 函数调用
print(type(fun(lst1)))
'''函数定义时,是否需要返回值,视情况而定'''

在这里插入图片描述


7.5 变量的作用域

变量的作用域:程序代码能访问该变量的区域
  1. 局部变量:在函数内定义并使用的变量,只在函数内部有效,局部变量使用global声明,这个变量就会变成全局变量
  2. 全局变量:函数体外定义的变量,可作用于函数内外
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/14 17:02

# 变量的作用域:程序代码能访问该变量的区域
'''
1.局部变量 在函数内定义并使用的变量,只在函数内部有效,局部变量使用global声明,这个变量就会变成全局变量
2.全局变量 函数体外定义的变量,可作用于函数内外
'''


def fun(a, b):
    c = a+b  # c,在函数内定义的变称为局部变量  a,b为函数的形参,作用范围也在在函数内部,相当于局部变量
    print(c)


name = '阿伟'  # 函数外部定义的变量,全局变量,函数体内外都可以使用
print(name)  # 阿伟


def fun2(self):
    print(name)
# 函数调用


fun2(name)  # 阿伟


def fun3():  # 函数定义
    global age  # 函数内部定义的变量,局部变量  使用global声明,变量就变为全局变量
    print(age)  # 20


age = 20
fun3()  # 函数调用
print(age)  # 20

在这里插入图片描述


7.6 代码复用

  函数是程序的一种基本抽象方式,它将一系列代码组织起来通过命名供其他程序使用。函数封装的直接好处是代码复用,任何其他代码只要输入参数即可调用函数,从而避免相同功能的代码在被调用处重复编写。代码复用有另一个好处,当更新函数功能时,所有被调用处的功能都被更新。

  程序由一系列代码组成,如果代码是顺序但无组织的,不仅不利于阅读和理解,也很难进行升级和维护。当程序长度在百行以上,如果不划分模块,程序的可读性就已经很糟糕了。解决这一问题最好的方法是将一个程序分割成短小的程序段,每一段程序完成一个小的功能。使用函数对合理划分为功能模块,并基于模块设计程序是一种常用方法,被称为“模块化设计”。

  模块化设计是指函数的封装功能将程序划分成主程序、子程序和子程序间关系的表达。模块化设计是使用函数设计的思考方法, 以功能块为基本单位,一般有两个基本要求:

  • 紧耦合:尽可能合理划分功能块,功能块内部耦合紧密;
  • 松耦合:模块间关系尽可能简单,功能块之间耦合度低。

   使用函数只是模块化设计的必要非充分条件,根据计算需求合理划分函数十分重要。一般来说,完成特定功能或被经常复用的一组语句应该采用函数来封装,并尽可能减少函数间参数和返回值的数量。


7.7 递归函数

7.7.1 什么是递归函数

如果在一个函数的函数体内调用了该函数本身,这个函数就称为递归函数。

7.7.2 递归的组成部分

递归调用递归终止条件

7.7.3 递归的调用过程

每递归调用一次函数,都会在栈内存分配一个栈帧,每执行完一次函数,都会释放相应的空间。

7.7.4 递归的优缺点

缺点: 占用内存多,效率低下
优点: 思路和代码简单
  • 使用递归函数计算阶乘
# 1.使用递归函数计算阶乘
def fun(n):  # 阶乘函数
    if n == 1:
        return 1
    elif n == 2:
        return 2
    else:
        return n * fun(n-1)


print(fun(6))  # 720
  • 斐波那契数列
# 2.斐波那契数列 1 1 2 3 5 8 13 ...
def fib(n):  # 斐波那契函数
    if n == 1:
        return 1
    elif n == 2:
        return 1
    else:
        return fib(n - 1) + fib(n - 2)  # 求斐波那契数列的第 n 项


print(fib(6), end=' ')  # 8

八、异常处理机制

   Python程序一般对输入有一定要求,但当实际输入不满足程序要求时,可能会产生程序的运行错误。

8.1 Bub的由来及分类

8.1.1 Bug的由来

  世界上第一部万用计算机的进化版-马克2号( Mark II)
  1945年9月9日,下午三点。哈珀中尉正领着她的小组构造一个称为“ 马克二型”的计算机。这还不是一个完全的电子计算机,它使用了大量的继电器,一种电子机械装置。第二次世界大战还没有结束。哈珀的小组日以继夜地工作。机房是一间第一次世界大战时建造的老建筑。那是一个炎热的夏天,房间没有空调,所有窗户都敞开散热。

  突然,马克二型死机了。技术人员试了很多办法,最后定位到第70号继电器出错。哈珀观察这个出错的继电器,发现一只飞蛾躺在中间,已经被继电器打死。她小心地用摄子将蛾子夹出来,用透明胶布帖到“事件记录本”中,并注明“第一个发现虫子的实例。”

  从此以后,人们将计算机错误戏称为虫子(bug),而把找寻错误的工作称为(debug)。

在这里插入图片描述

8.1.2 Bug常见类型

  1. 粗心导致的错误
1.漏了末尾的冒号: if 循环语句、 else 语句等
2.缩进错误
3.英文符号写为中文符号
4.把字符串 str和数字凭借在一起
5. while循环没有定义变量
6.比较运算符 == 和赋值运算符 = 混用
age = int(input('请输入年龄:'))  # input函数返回值为str类型
if age > 18:
    print('成年了')
  1. 知识不熟悉导致的错误
1. index索引越界
2.列表方法 lst.append
# 索引越界
lst = [1, 2, 3, 4]
# print(lst[4])
print(lst[3])  # 4


# append()方法使用不熟练
lst = []
# lst = append(1, 2, 3)
lst.append(1)  # 每次只能添加一个元素
  1. 思路不清晰导致的错误
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 10:22

# 思路不清晰导致的问题 基础知识不牢固,练练练!!!
lst = [{'rating': [9.7, 2062397], 'id': '1292052', 'type': ['犯罪', '剧情'], 'title': '肖申克的救赎', 'actors': ['蒂姆·罗宾斯', '摩根·弗里曼']},
       {'rating': [9.6, 1528760], 'id': '1291546', 'type': ['剧情', '爱情', '同性'], 'title': '霸王别姬', 'actors': ['张国荣', '张丰毅', '巩俐', '葛优']},
       {'rating': [9.5, 1559181], 'id': '1292720', 'type': ['剧情', '爱情'], 'title': '阿甘正传', 'actors': ['汤姆·汉克斯', '罗宾·怀特 ']}
       ]

name = input('请输入你要查询的演员:')
for item in lst:  # 遍历列表lst  -->{}  item是一个又一个的字典
    act_lst = item['actors']  # 通过item字典中的键actors获取值演员列表
    for actor in act_lst:  # 遍历演员列表
        if name in actor:
            print(name, '出演了', item['title'])

在这里插入图片描述

  1. 被动掉坑引起的错误

除零异常

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 10:22

a = int(input('请输入第一个整数:'))
b = int(input('情书如第二个整数:'))
result = a/b  # 注意:0不可以做除数
print('输出结果为:', result)

在这里插入图片描述
解决办法

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 10:22

try:
    a = int(input('请输入第一个整数:'))
    b = int(input('请输如第二个整数:'))
    result = a / b  # 注意:0不可以做除数
    print("{}除{}等于{}".format(a, b, result))
except ZeroDivisionError as error:
    print(error)

在这里插入图片描述


8.2 异常处理

   Python提供了异常处理机制,可以在异常出现时即时捕获,然后内部“ 消化”,让程序继续运行

8.2.1 try-exception语句

try:
    ...  可能会出异常的代码
    ...
    ...
except xxx(异常类型):
    ...  异常处理代码
    ...
    ...
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 10:22

try:
    a = int(input('请输入第一个整数:'))
    b = int(input('请输如第二个整数:'))
    result = a / b  # 注意:0不可以做除数
    print('输出结果为:', result)
except ZeroDivisionError:
    print('对不起,除数不能为0')
print('程序结束')

在这里插入图片描述
在这里插入图片描述

8.2.2 多个except语句

try:
    ...  可能会出异常的代码
    ...
    ...
except Exception1:
    ...  异常处理代码
    ...
    ...
except Exception2:
    ...  异常处理代码
    ...
    ...
except BaseException:
    ...  异常处理代码
    ...
    ...
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 10:22

try:
    a = int(input('请输入第一个整数:'))
    b = int(input('请输如第二个整数:'))
    result = a / b  # 注意:0不可以做除数
    print('输出结果为:', result)
except ZeroDivisionError:
    print('对不起,除数不能为0')
except ValueError:
    print('对不起,只能输入数字串')
print('程序结束')

正常输出:
在这里插入图片描述
除零异常:ZeroDivisionError
在这里插入图片描述
值错误:ValueError
在这里插入图片描述

8.2.3 try-except-else结构

如果 try中没有抛出异常,则执行 else块;如果 try中抛异常,则执行 except块, else块不执行。
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 10:22

# 如果 try中没有抛出异常,则执行else块,如果try中抛异常,则执行except块
try:
    a = int(input('请输入第一个整数:'))
    b = int(input('情书如第二个整数:'))
    result = a / b  # 注意:0不可以做除数
except BaseException as e:
    print('出错了', e)
else:
    print('计算结果为', result)

在这里插入图片描述

8.2.4 try-except-else-finally结构

finally 无论是否发生异常都会被执行,常用来解释 try块中申请的资源
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 10:22

'''
try except else finally 结构
finally 无论是否发生异常都会被执行,能常用来解释try块中申请的资源
'''

try:
    a = int(input('请输入第一个整数:'))
    b = int(input('请输入第二个整数:'))
    result = a / b
except BaseException as e:
    print('出错了', e)
else:
    print('结果为:', result)
finally:
    print('谢谢你的使用')
print('程序结束')

在这里插入图片描述

8.2.5 traceback模块

使用 traceback模块打印异常信息
# -*- coding: utf-8 -*-
# @File  : 8.trackback模块.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 16:01


import traceback

try:
    print(520)
    print(1 / 0)
except:
    traceback.print_exc()

在这里插入图片描述


8.3 Python常见异常类型

异常 描述
ZeroDivisionError 除零异常
IndexError 列表中没有次索引(index) 注:索引从零开始
KeyError 映射中没有这个键 dic['key']
NaneError 未声明/初始化对象 (没有属性)
SyntaxError Python语法错误
ValueError 传入无效参数

8.4 PyCharm程序调试

先看一段代码:

# -*- coding: utf-8 -*-
# @File  : PyCharm程序的调试.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 16:46


i = 1
while i < 2:
    print(i)

在这里插入图片描述

8.4.1 断点

程序运行到此处,暂时挂起,停止执行。此时可以详细观察程序的运行情况,方便做出进一步的判断。

在这里插入图片描述

8.4.2 调试

进入调试视图的三种方式:

(1)单击工具栏上的按钮
在这里插入图片描述

(2)右键单击编辑区:点击:debug’模块名’
在这里插入图片描述

(3)快捷键:shift+F9


九、类和对象

9.1 两大编程思想

9.1.1 面向过程

面向过程:事物比较简单,可以用线性思维取解决( 宏观上

9.1.2 面向对象

面向对象:事物比较复杂,使用简单线性思维无法解决( 细节上

共同点:面向过程和面向对象都是解决实际问题的一种思维方式

二者相辅相成,并不是独立的,解决复杂问题,通过面向对象便于我们从宏观上把握事物之间复杂关系,方便我们分析整个系统;具体到微观操作,仍然使用面向过程来处理


9.2 类和对象

9.2.1 类

类(class):多个类似事物组成的群体统称,能够帮我们快速理解和判断事物的性质。

数据类型:
1.不同数据类型属于不同的类
2.使用内置函数type查看数据类型

9.2.2 对象

对象(object):类的具体化 Python中一切皆对象,创建对象称为类的实例化

9.2.3 类的创建

创建类的语法:

class 类名:
    pass

类的组成:

  1. 类属性 类中方法外的变量,被该类的所有对象所共享
  2. 实例方法 方法必须有第一个形参,通常叫做self
  3. 静态方法 使用@staticmethod修饰,使用类名直接访问的方法
  4. 类方法 使用@calassmethod修饰,使用类名直接访问的方法

类的创建:

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 21:24

# 类的创建
class Student:  # Student 类名 由一个或多个单词组成,每个单词首字母大写,其余字母小写  Student称为类对象
    native_pace = '天津'  # 类属性
    name = '李华'  # 直接写在类里的变量(类中方法外的变量),称为 类属性:类中方法外的变量,被该类的所有对象所共享
    age = 30  # 类属性

    def __init__(self, name, age):  # 初始化方法方法
        self.name = name  # self.name 称为实例属性,进行了一次赋值操作,将局部变量name的值赋给实例属性
        self.age = age

    def eat(self):  # 实例方法  方法必须有第一个形参,通常叫做 'self'
        print('学生在吃饭')

    @staticmethod  # 静态方法
    def se():  # 在静态方法中不能写 self
        print('静态方法')

    @classmethod  # 类方法
    def cm(cls):
        print('类方法')

# Python中一切皆对象,类Student也是对象
print(id(Student))  # 有内存空间
print(type(Student))  # 类型
print(Student)  # 值

在这里插入图片描述

9.2.4 对象的创建

对象的创建又称类的实例化
意义:有了实例 就可以调用类中的内容

语法:

实例名(对象名) = 类名()
# 创建对象
stu1 = Student('张三', 20)  # # 创建Student类的对象  stu1叫做实例对象
# 对象三要素
print(id(stu1))  # id标识
print(type(stu1))  # type类型
print(stu1)  # value值

9.2.5 方法调用

类属性,类方法,静态方法都是使用 类名.方法名调用
实例方法采用 对象名.方法名调用
# 调用方法
stu1.eat()  # 对象名.方法名  调用Student类中的eat方法
print(stu1.name)
print(stu1.age)
stu1.eat()  # 对象名.方法()   stu1-->实际上就是方法定义处的self  调用Student类中的eat方法

print('----------类属性的使用方式----------')
print(Student.name)
stu2 = Student('张三', 20)
stu3 = Student('李四', 20)
print(Student.native_pace)
Student.native_pace = '吉林'
print(stu2.native_pace)
print(stu3.native_pace)
print(stu2.name, stu2.age)
print(stu3.name, stu3.age)

print('----------类方法的使用方式----------')
Student.cm()

print('----------静态方法的使用方式----------')
Student.se()

在这里插入图片描述

9.2.6 为对象动态绑定属性和方法

动态绑定属性: 对象名.属性 = "值"
动态绑定方法: 对象名.方法名 = 方法名
# -*- coding: utf-8 -*-
# @File  : 动态绑定属性和方法.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 22:25


class Student:
    def __init__(self, name, age):  # 初始化方法
        self.name = name  # 将局部变量 name的值赋给实例属性self.name,为所有对象所共享
        self.age = age  # 局部变量 age的值赋给实例属性self.age

    def eat(self):  # 实例方法
        print("{}在吃饭".format(self.name))


# 创建对象
stu1 = Student('张三', 18)
stu2 = Student('李四', 20)

print(id(stu1))  # 2502200745408
print(id(stu2))  # 2502201262576

print("------------为对象stu1动态绑定性别属性----------")
stu1.gender = '女'
print(stu1.name, stu1.age, stu1.gender)  #张三 18 女
print(stu2.name, stu2.age)  # 李四 20

stu1.eat()
stu2.eat()


def fun():  # 在类外面定义函数fun
    print('定义在类外的称为函数')


print("-----------为stu1动态绑定方法-----------")
stu1.fun = fun
stu1.fun()
# stu2.fun()

在这里插入图片描述


十、面向对象三大特征

10.1封装

封装:将属性和方法包装到类对象中,在方法内部对属性进行操作,在类对象外部调用方法。

代码实现:

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


# 封装:提高程序的安全性
# 将属性和方法包装到类对象中
# 在方法内部对属性进行操作,在类对象外部调用方法

class Car:
    def __init__(self, brand):
        self.brand = brand  # 实例属性

    @staticmethod
    def start():  # 静态方法
        print('汽车已启动...')


car = Car('奥迪A8')
car.start()
print(car.brand)

在这里插入图片描述

如果不希望实例属性在类的外部被使用,可以在前面加上两个下划线" _"
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


class Student:
    def __init__(self, name, age):
        self.name = name
        self.__age = age  # 如果不希望实例属性在类的外部被使用,所以在前面加上两个下划线

    def show(self):
        return self.name, self.__age

    @staticmethod
    def eat():
        print('吃')


stu1 = Student('李华', 20)
stu1.show()  # 调用方法
print(dir(stu1))  # 查看对象可以用的属性
print('-------------')
print(stu1.name, stu1._Student__age)  # 在类外部通过_Student__age访问实例属性self.__age
stu1.eat()

在这里插入图片描述


10.2继承

继承:子类可以继承父类的属性和方法,提高代码的复用性。
如果一个对象没有继承任何类,则默认继承 object类

语法格式:

class 子类名(父类1,父类2,...):
    pass

代码实现:

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


class Person(object):
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def info(self):
        print(self.name, self.age)


class Student(Person):
    def __init__(self, name, age, stu_nb):
        super(Student, self).__init__(name, age)  # 继承父类的属性
        self.stu_nb = stu_nb  # 新增属性

    def __str__(self):
        return self.name, self.age, self.stu_nb


class Teach(Person):
    def __init__(self, name, age, teach_of_year):
        super(Teach, self).__init__(name, age)
        self.teach_of_year = teach_of_year


student = Student('张三', 20, '1001')  # 创建对象
teacher = Teach('杨老师', 34, 10)

student.info()
teacher.info()
print(student.__str__())
print(student.stu_nb)
print(teacher.teach_of_year)

在这里插入图片描述

10.2.1 方法重写

如果子类对继承自己父类的某个属性不满意,可以在子类对其(方法体)进行重新编写。

子类重写后的方法通过 super().方法名() 调用父类中被重写的方法。

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


# 如果子类对继承自己父类的某个属性不满意,可以在子类对其(方法体)进行重新编写
# 子类重写后的方法通过 super()...方法名() 调用父类中被重写的方法


class Person(object):
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def info(self):
        print(self.name, self.age)


class Student(Person):
    def __init__(self, name, age, stu_nb):
        super(Student, self).__init__(name, age)
        self.stu_nb = stu_nb

    def info(self):  # 方法重写
        super().info()  # 调用父类中方法
        print(f'学号:{self.stu_nb}')  # f''格式化字符串


class Teach(Person):
    def __init__(self, name, age, teach_of_year):
        super(Teach, self).__init__(name, age)
        self.teach_of_year = teach_of_year

    def info(self):  # 方法重写
        super().info()
        print('教龄{0}'.format(self.teach_of_year))  # 格式化字符串


student = Student('张三', 20, '1001')
teacher = Teach('杨老师', 34, 10)

student.info()
print('-----------------')
teacher.info()

在这里插入图片描述

10.2.2 object类

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


'''
object 类是所有类的父类,所有类都有object类的属性和方法
内置函数dir()可以查看指定对象所有属性
Object有一个__str__方法,用于返回一个对于”对象的描述
对应内置函数str()通常用于print()方法,帮我们查看对象的信息,所以经常会对__str__进行重写“'''


class Student(object):
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def __str__(self):  # 重写父类object中的方法
        return '我的名字是{0},今年{1}岁了'.format(self.name, self.age)


stu = Student('张三', 20)
print(dir(stu))  # 查看stu这个对象的所有属性和方法 从object类中继承的
print(stu)  # 默认调用__str__()这样的方法 输出:我的名字是张三,今年20岁了

print(type(stu))  # <class '__main__.Student'>  Student类型

在这里插入图片描述

10.2.3 多重继承

一个子类可以有多个“直接父类”,这样,就具备了“多个父类”的特点,通过类的特殊属性 __mro__ 可以查看类的组织结构。

定义子类时,必须在其构造函数中调用父类的构造函数

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


# 多继承
class A(object):
    pass


class B(object):
    pass


class C(A, B):
    pass

10.3多态

多态:多态就是具有多种形态,即便不知道一个变量所引用的对象是什么类型,仍然可以使用这个变量调用方法,在运行过程中根据变量所引用的对象类型,动态决定调用那个对象中的方法。

代码实现:

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


''' 
多态:多态就是具有多种形态,即便不知道一个变量所引用的对象是什么类型,仍然可以使用这个变量调用方法,在运行过程中根据变量所引用的对象类型,动态决定调用那个对象中的方法'''

# 动态语言多崇尚鸭子类型,当一只鸟走起来向鸭子,游起来以向鸭子,看起来也像鸭子,那么这只鸟及可以被称为鸭子


class Animal(object):
    def eat(self):
        print('动物会吃')


class Dog(Animal):
    def eat(self):
        print('够吃骨头')


class Cat(Animal):
    def eat(self):
        print('猫吃小鱼')


class Person:
    def eat(self):
        print('人吃五谷杂粮')


# 定义一个函数
def fun(fun1):
    fun1.eat()  # 调用对象的eat()方法


if __name__ == '__main__':
    # 开始调用函数
    fun(Animal())  # Cat继承了Animal Dog继承了Animal
    fun(Cat())  # Cat 和Dog是重写了父类中的eat方法,调用了自己重写后的内容
    fun(Dog())

    print('------------------')
    fun(Person())  # Person 没有继承关系 但是有eat方法,直接调用eat方法

在这里插入图片描述

动态语言与静态语言

Python是一门动态语言,可以在创建对象后动态的绑定属性和方法,

静态语言和动态语言关于多态的区别:

静态语言实现多态的三个必要条件( Java)
1. 继承
2. 方法重写
3. 父类引用指向子类对象
动态语言:( Python)
动态语言的多态崇尚 “ 鸭子类型“ 一只鸟走起来像鸭子,游起来像鸭子,那么这只鸟就可以被称为鸭子。在鸭子类型中,不需要关心对象是什么类型,到底是不是鸭子,只关心对象的 行为'''

10.4 类的特殊属性和方法

10.4.1 特殊属性

特殊属性 描述
__dict__ 获得类对象或实例对象所绑定的所有属性的方法的字典
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


# 特殊属性 __dict__ 获得类对象或实例对象所绑定的所有 属性 或 方法 的字典
class A:
    pass


class B:
    pass


class C(A, B):
    def __init__(self, name, age):
        # 实例属性
        self.name = name
        self.age = age


if __name__ == '__main__':

    # 创建C类的对象
    x = C('Jack', 20)  # x是C类的一个实例对象

    print(x.__dict__)  # 获得实例对象属性的字典
    print(C.__dict__)  # 获得类对象的属性和方法的字典
    print('-----------------')

    print(x.__class__)  # 输出对象所属的类
    print(C.__bases__)  # C类父类类型的元组  (<class '__main__.A'>, <class '__main__.B'>)
    print(C.__base__)  # 类的基类  离C类最近的父类
    print(C.__mro__)  # 查看类的层次结构
    print(A.__subclasses__())  # 子类的列表

在这里插入图片描述

10.4.2 特殊方法

特殊方法 描述
__len__() 通过重写 __len__()方法,让内置函数len()的参数可以是自定义类型
__add__() 通过重写__add__()方法,可以让自定义对象具有+的功能
__new__() 用于创建对象
__init__() 对创建的对象进行初始化

__len__()方法和 __add__() 方法

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


# 1.特殊方法  __add__()
# 通过重写 __add__()方法,可以使自定义对象具有 “+” 的功能
a = 20
b = 100
c = a + b  # 两个整数类型的对象的相加操作
d = a.__add__(b)
print(c)
print(d)


class Student:
    sex = '女'  # 类属性

    def __init__(self, name):  # 初始化方法
        self.name = name

    def __add__(self, other):  # 重写 __add__()方法 可以使自定义对象具有 “+” 的功能
        return self.name + other.name

    def __len__(self):  # 重写 __len__方法 让自定义函数len()的参数可以是自定义类型
        return len(self.name)


stu1 = Student('Jack')
stu2 = Student('李四')
s = stu1 + stu2  # 实现了两个对象的加法运算(因为在Student类中 编写__add__()特殊的方法)
print(s)

# 2.特殊方法  __len__()
# 通过重写__len__()方法,让自定义函数len()的参数可以是自定义类型
lst = [11, 22, 33, 44]
print(len(lst))  # len是内置函数,可以计算列表的一个长度
print(lst.__len__())  # 特殊方法
print(len(stu1))

在这里插入图片描述

__new__方法

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


class Person(object):
    def __new__(cls, *args, **kwargs):  # 创建对象
        print('__new__()方法被调用执行了,cls的id值为{0}'.format(id(cls)))
        obj = super().__new__(cls)  # 创建对象 obj
        print(f'创建对象(obj)的id值为:{id(obj)}')
        print(Person)  # <class '__main__.Person'>
        print(obj)  # <__main__.Person object at 0x000001C8B13D9CA0>
        return obj

    def __init__(self, name, age):  # 对对象的属性进行初始化
        print(f'__init__()被调用执行了,self的id值为{id(self)}')
        self.nane = name
        self.age = age


if __name__ == '__main__':
    print(f'object这个类对象的id为:{id(object)}')
    print(f'Person这个类对象的id为:{id(Person)}')

    # 创建Person类的实例对象
    p1 = Person('张三', 20)

    print(f'p1这个Person类的实例对象的id为{id(p1)}')

在这里插入图片描述

__init__方法

# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/15 23:27


class Person(object):
    def __new__(cls, *args, **kwargs):  # 创建对象
        print('__new__()方法被调用执行了,cls的id值为{0}'.format(id(cls)))
        obj = super().__new__(cls)  # 创建对象 obj
        print(f'创建对象(obj)的id值为:{id(obj)}')
        return obj

    def __init__(self, name, age):  # 对对象的属性进行初始化
        print(f'__init__()被调用执行了,self的id值为{id(self)}')
        self.nane = name
        self.age = age


print(f'object这个类对象的id为:{id(object)}')
print(f'Person这个类对象的id为:{id(Person)}')

# 创建Person类的实例对象
p1 = Person('张三', 20)
print(f'p1这个Person类的实例对象的id为{id(p1)}')

在这里插入图片描述


10.5 变量的赋值操作

只是多生成了一个变量,实际上还是指向同一个对象
# -*- coding: utf-8 -*-
# author : Flyme awei 
# 开发时间: 2022/7/1 15:32

class CPU:
    pass


class Disk:
    pass


class Computer:
    def __init__(self, cpu, disk):  # 给对象的实例属性进行初始化
        self.cpu = cpu
        self.disk = disk


# 变量的赋值
cp1 = Computer(cpu='CPU', disk='DISK')  # 创建CPU类的实例对象
cp2 = cp1  
# 变量的赋值,一个对象的实例采用两个变量存储,实际上还是指向一个对象
print(cp1, id(cp1))
print(cp2, id(cp2))

在这里插入图片描述

赋值(=),就是创建了对象的一个新的引用,修改其中任意一个变量都会影响到另一个。


10.6 对象的浅拷贝和深拷贝

10.6.1 浅拷贝

Python拷贝一般都是 浅拷贝,拷贝时,对象包含的子对象内容不拷贝,因此,源对象与拷贝对象会引用同一个子对象。
# -*- coding: utf-8 -*-
# author : Flyme awei 
# 开发时间: 2022/7/1 15:32

import copy


class CPU:
    pass


class Disk:
    pass


class Computer:
    def __init__(self, cpu, disk):  # 给对象的实例属性进行初始化
        self.cpu = cpu
        self.disk = disk


cpu = CPU()  # 创建一个 CPU 类的实例对象
disk = Disk()  # 创建一个Disk 类对象
computer = Computer(cpu, disk)  # 创建一个Computer类的实例对象

# 浅拷贝
print(cpu)
print(disk)
computer2 = copy.copy(computer)  # 子对象不拷贝
print(computer, computer.cpu, computer.disk)
print(computer2, computer2.cpu, computer2.disk)


# 类的浅拷贝:
# Python的拷贝一般都是浅拷贝,拷贝时,对象包含的子对象内容不拷贝
# 因此,源对象与拷贝对象会引用同一个子对象

在这里插入图片描述

**浅拷贝:创建一个新的对象,但它包含的是对原始对象中包含项的引用
(如果用引用的方式修改其中一个对象,另外一个也会修改改变)**

哪些是浅拷贝:

  1. 完全切片方法;
  2. 工厂函数,如list()
  3. copy模块的copy()函数。

10.6.2 深拷贝

使用 copy模块的 deepcopy函数,递归拷贝对象中包含的子对象,源对象和拷贝对象所有的子对象也不相同。
# -*- coding: utf-8 -*-
# author : Flyme awei 
# 开发时间: 2022/7/1 15:32

import copy


class CPU:
    pass


class Disk:
    pass


class Computer:
    def __init__(self, cpu, disk):  # 给对象的实例属性进行初始化
        self.cpu = cpu
        self.disk = disk


cpu = CPU()  # 创建一个 CPU 对象
disk = Disk()  # 创建一个硬盘类对象
computer = Computer(cpu, disk)  # 创建一个计算机类对象

# 深拷贝
computer1 = copy.deepcopy(computer)
print(computer, computer.cpu, computer.disk)
print(computer1, computer1.cpu, computer1.disk)

# 类的深拷贝
# 使用copy模块的deepcopy函数,递归拷贝对象中包含的子对象
# 源对象和拷贝对象所有的子对象也不同

在这里插入图片描述

深拷贝:创建一个新的对象,并且递归的复制它所包含的对象。

修改其中一个,另外一个不会改变。因此,新对象和原对象没有任何关联。
例如:{ copy模块的 deepcopy()函数}

十一、包和模块

11.1 Python中的模块

11.1.1 什么叫做模块

1.模块:英文名 modules

2.在Python中 一个拓展名为.py的文件就是一个模块

3.函数与模块的关系:

  • 包含语句
  • 包含类class
  • 一个模块中可以包含n多个函数def

4.使用模块的好处:

  • 方便其它程序和脚本的导入
  • 避免函数名和变量名冲突
  • 提高代码的可维护性
  • 提高代码的课重用性
# -*- coding: utf-8 -*-
# @File  : demo.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/16 11:58


def fun():  # 函数
    pass


class Student:  # 类
    a = 'hello'  # 类属性

    def __init__(self, name):  # 初始化方法
        self.name_ = name  # 实例属性

    def __str__(self):  # 实例方法
        pass

    @staticmethod
    def cm():
        print('静态方法')

    @classmethod
    def cs(cls):
        print('类方法')


a = 10  # 变量
b = 20
c = a + b  # 语句

11.1.2 导入模块

创建模块:

新建一个 .py文件,名称尽量不要与 python自带的标准模块名相同

导入模块:

import 模块名称 as 别名  # 导入模块中所有的
from 模块名称 import 函数/变量/类  # 导入模块的指定内容

代码实现:

# -*- coding: utf-8 -*-
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/16 11:58

import math  # 导入关于数学运算的模块
print(id(math))
print(type(math))  # module
print(math)
print('----------------')
print(math.pi)  # 3.14...
print(dir(math))
print(math.pow(2, 3))  # 幂 2**3=8
print(math.e)
print(math.ceil(9.435))  # 取上整
print(math.floor(9.35))  # 取下整
print(math.log(2, 4))

from math import pi  # 导入模块指定的内容
print(pi)

11.1.3 导入自定义模块

自定义模块calc

# -*- coding: utf-8 -*-
# @File  : calc.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/16 11:58


def add(a, b):
    return a+b


def div(a, b):
    return a/b

导入自定义模块 calc

import calc  # 导入自定义模块 calc
print(calc.add(10, 20))
print(calc.div(8, 2))


from calc import add  # 自定义导入calc模块的add方法
print(add(23, 53))

在这里插入图片描述

11.1.4 以主程序的方式运行

在每个模块的定义中都包含了一个记录模块名称的变量 __name__,程序可以检查该变量,以确定他们在哪个模块中执行。

如果一个模块不是被导入到其他程序中执行,那么它可能在解释器的顶级模块中执行。

顶级模块__name__变量的值为__main__

语句:

if __name__ = '__main__':
    pass

自定义模块calc2

# -*- coding: utf-8 -*-
# @File  : calc2.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/16 11:58


def add(a, b):
    return a+b


if __name__ == '__main__':
    print(add(10, 20))   # 只有当点击运行calc2模块时,才会执行运算

导入calc2模块

# 顶级模块__name__变量的值为__main__
import calc2    # 调用calc2的 add 模块


print(calc2.add(100, 200))

在这里插入图片描述


11.2 Python中的包

1.包是一个分层次的目录结构,它将一组功能相近的模块组织在一个目录下

2.包的作用:

  • 代码规范
  • 避免模块名称冲突

3.包与目录的区别:

  • 包含__init__.py文件的目录称为
  • 目录里通常不包含__init__.py文件

4.包的导入: import 包名.模块名

package1下面有module_A模块和module_B模块
在这里插入图片描述

module_A模块

# -*- coding: utf-8 -*-
# @File  : module_A.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/16 11:58

# 包的导入
a = 10

module_B模块

# -*- coding: utf-8 -*-
# @File  : module_B.py
# @author: Flyme awei
# @email : Flymeawei@163.com
# @Time  : 2022/8/16 11:58

b = 100

导入包package1

'''
包的导入 
import 包名.模块名
'''

import package1.module_A as ma  # 导入包package1中的module_A模块
print(ma.a)  # 10

导入带有包的模块注意事项

1.使用 import方式进行导入时,只能跟 包名模块名
import package1
import package1.module_A
import calc
2.使用 from...import 可以导入 模块函数变量
from package1 import module_A  # 导入模块module_A
from package1.module_A import a   # 导入变量a
print(a)  # 10

在这里插入图片描述


11.3 Python中常用的内置模块

模块名 描述
sys python解释器及环境相关的标准库
time 提供时间相关的各种函数标准库
os 提供访问操作系统服务功能的标准库
calendar 提供与日期相关的各种函数的标准库
urllib 用于读取来自网上(服务器)的数据标准库
json 用于使用JSON序列化和反序列化对象
re 用于在字符串中执行正则表达式匹配和替换
math 提供标准运算函数的标准库
decimal 用于进行精确控制运算精度、有效数位和四舍五入操作的十进制运算
logging 提供了灵活的记录事件、错误、警告和调试信息等日志信息的功能
'''
模块名                        描述
----------------------------------------------------------
1.sys           与python解释器及环境相关的标准库

2.time          提供时间相关的各种函数标准库

3.os            提供访问操作系统服务功能的标准库

4.calendar      提供与日期相关的各种函数的标准库

5.urllib        用于读取来自网上(服务器)的数据标准库

6.json          用于使用JSON序列化和反序列化对象

7.re            用于在字符串中执行正则表达式匹配和替换

8.math          提供标准运算函数的标准库

9.decimal       用于进行精确控制运算精度、有效数位和四舍五入操作的十进制运算

10.logging      提供了灵活的记录事件、错误、警告和调试信息等日志信息的功能
'''

11.3.1sys模块

与python解释器及环境相关的标准库

import sys  # 导入与python解释器和环境相关的标准库
print(sys.getsizeof(24))  # 字节
print(sys.getsizeof(54))
print(sys.getsizeof(True))
print(sys.getsizeof(False))

在这里插入图片描述

11.3.2 time模块

提供与时间相关各种函数的标准库

# time 提供与时间相关各种函数的标准库
import time
print(time.time())
print(time.localtime(time.time()))

在这里插入图片描述

11.3.3 os模块

提供访问操作系统服务功能的标准库

# 提供访问操作系统服务功能的标准库
import os

11.3.4 urllib模块

import urllib.request
print(urllib.request.urlopen('http://www.baidu.com').read())

在这里插入图片描述


11.4 第三方模块

11.4.1 第三方模块安装

pip install 模块名

11.4.2 第三方模块使用

import 模块名

import time
import schedule


def job():
    print('哈哈哈哈哈哈哈哈')


schedule.every(1).seconds.do(job)
while True:
    schedule.run_pending()
    time.sleep(1)

在这里插入图片描述


十二、文件操作

12.1 编码格式介绍

常见的字符编码格式

Python解释器使用的是 Unicode(内存)
Unicode 定长编码: 2个字节表示一个字符

.py文件放在磁盘上使用UTF—8储存(外存)
UTF-8 变长编码:UTF-8Unicode的实现,1-4个字节表示一个字符,英文1个字节,汉字3个字节。

新建demo1.py文件

# -*- coding: utf-8 -*-
# @File  : demo1.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/17 10:45

print("Hello World")

记事本打开:
在这里插入图片描述

# 不同的编码格式决定了占用磁盘空间的大小
# coding:UTF-8  # 中文编码注释
# encoding=gbk  # 编码格式

12.2 文件读写原理

12.2.1 文件的读写简称IO操作

输入input,输出output
在这里插入图片描述

12.2.2 文件读写流程

在这里插入图片描述

12.2.3 操作原理:

文件操作原理

  1. python操作文件
  2. 打开或新建文件open
  3. 读,写文件read、write
  4. 关闭资源file.close()

12.3 文件读写操作

内置函数open()创建文件对象

通过 IO流将磁盘文件中的内容与程序中的对象中的内容进行同步

语法规则

file = open(filename, mode='r', buffering=None, encoding=None)
"""
file  -->被创建的文件对象
open  -->创建文件对象的函数             
filename  -->要创建或打开的文件名称
mode  -->打开模式默认为只读
encoding  -->字符编码格式        
"""

代码演示:
a.txt文件
在这里插入图片描述

3.文件读写操作.py文件

# -*- coding: utf-8 -*-
# @File  : 3.文件读写操作.py
# @author: Flyme awei 
# @email : Flymeawei@163.com
# @Time  : 2022/8/17 10:45

file = open('a.txt', 'r')  # r 以只读模式打开文件,文件的指针将会放在文件的开头
print(file.readlines())  # 返回值为一个列表
file.close()

在这里插入图片描述


12.4 文件的打开开和关闭

12.4.1 文件的状态和操作过程

在这里插入图片描述

12.4.2 文件的打开模式

文件类型:按文件中数据的组织形式,文件分为以下两大类:

1.文本文件:存储的是普通“字符”文本,默认为Unicode字符集,可以使用记事本程序打开

2.二进制文件:把数据内容以字节进行存储,无法用记事本打开,必须使用专用的文软件打开,举例:MP3音频文件 JPG图片 .doc文档等

常用 的文件打开模式

打开模式 描述
r 以只读模式打开文件,文件的指针将会放在文件的开头
w 以只写方式打开文件,如果文件不存在就创建,如果文件存在则覆盖原有内容,文件指针放在开头
a 以追加模式打开文件,如果文件不存在则创建,文件指针在开头,如果文件存在,则在文件末尾追加内容,文件指针在原文末尾
b 以二进制方式打开文件,不能单独使用,需要与其它模式一起使用:rb或者wb
+ 以读写方式打开文件,不能单独使用,需要与其它模式一起使用,a+
'''
打开模式                        描述
---------------------------------------------------------------------------------------------------
 r          以只读模式打开文件,文件的指针将会放在文件的开头

 w          以只写方式打开文件,如果文件不存在就创建,如果文件存在则覆盖原有内容,文件指针放在开头

 a          以追加模式打开文件,如果文件不存在则创建,文件指针在开头,如果文件存在,则在文件末尾追加内容,文件指针在原文末尾

 b          以二进制方式打开文件,不能单独使用,需要与其它模式一起使用,rb,或者wb

 +          以读写方式打开文件,不能单独使用,需要与其它模式一起使用
'''

代码演示:

file = open('a.txt', 'r')  # 只读
print(file.readlines())
file.close()

file = open('b.txt', 'w')  # 只写
file.write('hello.python')
file.close()

file = open('c.txt', 'a')  # 追加
file.write('你好啊')
file.close()

file1 = open('logo.png', 'rb')  # 以二进制读
file2 = open('copy_logo.png', 'wb')  # 以二进制写
a = file1.read()
file2.write(a)  # 边读边写
file1.close()
file2.close()

file = open('c.txt', 'a+')  # 以读写方式打开
file.write('java')
file.close()

在这里插入图片描述

a.txt文件
在这里插入图片描述

b.txt文件
在这里插入图片描述

c.txt文件
在这里插入图片描述
logo.png文件
在这里插入图片描述
copy_logo.png文件
在这里插入图片描述

12.4.3 文件的关闭

file.close()


12.5 文件对象的常用方法

方法名 说明
read([size]) 从文件中读取size个字节或字符返回,若省略[size]则读到文件末尾
readline() 从文本中读取一行内容
readlines() 把文本文件中每一行都作为独立的字符串对象,并将这些对象放回列表返回
write(str) 将字符串str内容写进文件
writelines(s_list) 将字符串列表s_list写入文本文件,不添加换行符
seek(offset,whence) 把文件指针移到新的位置,offset表示相对whence的位置;offset:为正往结束方向移动, whence不同的值呆鸟不同的含义;0:从文件开头开始计算; 1:从文件当前位置开始计算
tell() 返回文件指针当前位置
flush() 缓冲区的内容写入文件,但不关闭文件
close() 把缓冲区内容写入文件,同事关闭文件,释放文件对象资源
"""
方法名                                      说明
----------------------------------------------------------------------------------------
1.read([size])               从文件中读取size个字节或字符返回,若省略[size]则读到文件末尾

2.readline()                 从文本中读取一行内容

3.readlines()                把文本文件中每一行都作为独立的字符串对象,并将这些对象放回列表返回

4.write(str)                 将字符串内容写进文件

5.writelines(s_list)         将字符串列表s_list写入文本文件,不添加换行符

6.seek(offset, whence)      把文件指针移到新的位置,offset表示相对whence的位置:
                             offset:为正往结束方向移动,
                             whence不同的值呆鸟不同的含义:
                             0:从文件开头开始计算 
                             1:从文件当前位置开始计算
                             2:从文件结尾开始计算

7.tell()                     返回文件指针当前位置

8.flush()                    把缓冲区的内容写入文件,但不关闭文件

9.close()                    把缓冲区内容写入文件,同事关闭文件,释放文件对象资源
"""

12.6 with上下文管理

with语句可以自动管理上下文资源,不论什么原因跳出 with块,都能确保文件正常关闭,以此来达到释放资源的目的

语句:

with open('logo.png', 'rb') as src_file: 
# open('logo.png', 'rb')称为上下文表达式
    src_file.read()  (with语句体)"""
with open('logo.png', 'rb') as src_file:
    print(src_file.read())  
    # 不用写 file.close
   open('logo.png', 'rb')称为 上下文表达式,同时创建一个运行时上下文,自动调用 __enter__方法,并将返回值赋给 src_file, 实现了特殊方法 __enter__(), __exit__(),遵守了上下文管理协议,这句表达式的对象就是上下文管理器,离开运行时上下文,自动调用上下文管理器的特殊方法
# MyContentMgr实现了特殊方法__enter__(), __exit__(),称为该类对象遵守了上下文管理器协议
# 该类对象的实例对象 MyContentMgr() 被称为上下文管理器
# 类对象MyContentMgr的实例对象 MyContentMgr() 被称为上下文管理器

class MyContentMgr(object):
    def __enter__(self):
        print('enter方法被调用执行了')
        return self

    def __exit__(self, exc_type, exc_val, exc_tb):
        print('exit方法被调用执行了')
        return self

    def show(self):
        print('show方法被调用执行了')


with MyContentMgr() as file:  # 相当于file = MyContentMgr()
    file.show()

在这里插入图片描述

with语句复制文件:

# with语句 上下文管理器
with open('logo.png', 'rb') as src_file:
    with open('copy2logo.png', 'wb') as target_file:
        target_file.write(src_file.read())
# 不需要手动写关闭文件过程

12.7 os模块

1. OS模块是 python内置的与 操作系统功能文件系统相关的模块,该模块中的语句执行结果通常与操作系统有关,在不同操作系统上运行,得到的结果可能不一样.

2.os模块于os.path模块用于对目录或文件进行操作

# 目录操作
# os 与操作系统相关的一个模块

import os

# os.system('notepad.exe')  # 打开记事本
# os.system('calc.exe')  # 打开计算器

# 直接打开可执行文件
os.startfile('C:\\Program Files\\Microsoft Office\\root\\Office16\\excel.exe')  # 打开 Excel程序

打开记事本:
os.system('notepad.exe')

打开计算器:
os.system('calc.exe')

12.7.1 os模块操作目录相关函数

os 模块操作目录相关函数

函数 说明
getcwd() 返回当前工作目录
listdir(path) 返回指定路径下的文件和目录信息
mkdir(path[,mode]) 创建目录
makedirs(path1/path2/...[,mode]) 创建多级目录
rmdir(path) 删除目录
removedirs(path1/path2...) 删除多级目录
chdir(path) path设置为当前工作目录
# os 模块操作目录相关函数
"""
函数                                            说明
----------------------------------------------------------------------------
1.getcwd()                                  返回当前工作目录

2.listdir(path)                             返回指定路径下的文件和目录信息

3.mkdir(path[,mode])                        创建目录

4.makedirs(path1/path2/...[,mode])          创建多级目录

5.rmdir(path)                               删除目录

6.removedirs(path1/path2...)                删除多级目录

7.chdir(path)                               将path设置为当前工作目录
"""

代码实现:

import os  # 导入与操作系统和文件系统相关的模块
print(os.getcwd())  # 返回当前目录

print(os.listdir('../chap15 文件'))  # 返回指定路径下的文件和目录信息

# os.mkdir('new dir')  # 创建目录

# os.makedirs('A/B/C')  # 创建多级内目录

# os.rmdir('new dir')  # 删除目录

# os.removedirs('A/B/C')  # 移除多级目录

os.chdir('/Python基础语法\\chap14 模块')  # 将path设置为当前工作目录
print(os.getcwd())  # 返回当前工作目录

在这里插入图片描述

12.7.2 os.path模块操作目录相关函数

os.path 模块操作目录相关函数

函数 说明
abspath(path) 用于获取文件目录的绝对路径
exists(path) 用于判断文件或目录是否存在,如果存在返回True,否则返回False
join(path,name) 将目录与目录或者文件名拼接起来
splitext() 分离文件名和拓展名
basename(path) 从一个目录中提取文件名
dirname(path) 从一个目录中提取文件名
isdir(path) 用于判断是否为路径
# os.path 模块操作目录相关函数
"""
函数                                  说明
----------------------------------------------------------------------
1.abspath(path)                 用于获取文件目录的绝对路径

2.exists(path)                  用于判断文件或目录是否存在,如果存在返回True,否则返回False

3.join(path,name)               将目录与目录或者文件名拼接起来

4.splitext()                    分离文件名和拓展名

5.basename(path)                从一个目录中提取文件名

6.dirname(path)                 从一个路径中提取文件路径,不包括文件名

7.isdir(path)                   用于判断是否为路径
"""

代码实现:

import os.path
print(os.path.abspath('demo2.py'))  # 获取绝对路径

print(os.path.exists('demo2.py'))  # 用于判断文件或目录是否存在,如果存在返回True,否则返回False

print(os.path.join('E:\\python', 'demo13.py'))  # 将目录与目录或者文件名拼接起来

print(os.path.split('/Python基础语法\\chap15 文件'))  # 将目录与文件进行拆分
print(os.path.splitext('demo2.py'))  # 分开文件和拓展名

print(os.path.basename('/Python基础语法\\chap15 文件'))

print(os.path.dirname('/Python基础语法\\chap15 文件'))

print(os.path.isdir('/Python基础语法\\chap15 文件'))

在这里插入图片描述

12.7.3 列出指定目录下的所有.py文件

代码实现:

# 列出指定目录下的所有.py文件
import os
path = os.getcwd()  # 返回当前工作目录
lst = os.listdir(path)  # 返回 path路径下的文件和目录信息
for filename in lst:
    if filename.endswith('.py'):  # 列出.py文件    endswith-->以什么结尾
        print(filename)

在这里插入图片描述

12.7.4 遍历指定目录下所有的文件以及目录

代码实现:os.walk(path)

import os

path = os.getcwd()
lst_files = os.walk(path)  # 遍历指定目录下所有的文件以及目录
print(lst_files)  # 返回值为一个生成器对象
for dir_path, dir_name, file_name in lst_files:
    # print(dir_path)
    # print(dir_name)
    # print(file_name)
    # print('----------------')
    for dir_ in dir_name:
        print(os.path.join(dir_path, dir_))
    print('-----------------------------')
    for file in file_name:
        print(os.path.join(dir_path, file))

相关文章
|
3月前
|
测试技术 API Python
【10月更文挑战第1天】python知识点100篇系列(13)-几种方法让你的电脑一直在工作
【10月更文挑战第1天】 本文介绍了如何通过Python自动操作鼠标或键盘使电脑保持活跃状态,避免自动息屏。提供了三种方法:1) 使用PyAutoGUI,通过安装pip工具并执行`pip install pyautogui`安装,利用`moveRel()`方法定时移动鼠标;2) 使用Pymouse,通过`pip install pyuserinput`安装,采用`move()`方法移动鼠标绝对位置;3) 使用PyKeyboard,同样需安装pyuserinput,模拟键盘操作。文中推荐使用PyAutoGUI,因其功能丰富且文档详尽。
|
5月前
|
Python
python知识点
【8月更文挑战第27天】python知识点
3408 2
WK
|
5月前
|
存储 机器学习/深度学习 JSON
Python入门知识点
Python入门覆盖历史、设计理念、变量、数据类型、控制结构等。了解Python的发展,掌握动态类型的灵活性,熟悉整数、浮点数、字符串等数据类型。学会if/else、for/while循环构建逻辑流程,使用def定义函数,lambda快速创建匿名函数。通过类实现面向对象编程,利用模块和包组织代码。掌握try-except处理异常,open()进行文件操作。利用标准库和第三方库增强功能,理解集合、字典、列表推导式的应用,深入魔法方法、递归、装饰器等高级特性,以及上下文管理器和字符串、列表、元组的操作技巧。
WK
47 0
|
2月前
|
缓存 Java 索引
[Python]知识点
本文主要介绍了Python的一些高级知识点和使用细节,包括pip的使用、内置函数、列表、元组、字典、集合、变量、Lambda表达式、面向对象编程、异常处理、模块及标准库等。文章适合有一定Python基础的读者,重点在于深入理解和掌握Python的高级特性。文中还提供了大量示例代码,帮助读者更好地理解和应用这些知识点。
47 1
[Python]知识点
|
3月前
|
安全 Linux 数据安全/隐私保护
python知识点100篇系列(15)-加密python源代码为pyd文件
【10月更文挑战第5天】为了保护Python源码不被查看,可将其编译成二进制文件(Windows下为.pyd,Linux下为.so)。以Python3.8为例,通过Cython工具,先写好Python代码并加入`# cython: language_level=3`指令,安装easycython库后,使用`easycython *.py`命令编译源文件,最终生成.pyd文件供直接导入使用。
120 3
python知识点100篇系列(15)-加密python源代码为pyd文件
|
3月前
|
网络协议 数据库连接 Python
python知识点100篇系列(17)-替换requests的python库httpx
【10月更文挑战第4天】Requests 是基于 Python 开发的 HTTP 库,使用简单,功能强大。然而,随着 Python 3.6 的发布,出现了 Requests 的替代品 —— httpx。httpx 继承了 Requests 的所有特性,并增加了对异步请求的支持,支持 HTTP/1.1 和 HTTP/2,能够发送同步和异步请求,适用于 WSGI 和 ASGI 应用。安装使用 httpx 需要 Python 3.6 及以上版本,异步请求则需要 Python 3.8 及以上。httpx 提供了 Client 和 AsyncClient,分别用于优化同步和异步请求的性能。
python知识点100篇系列(17)-替换requests的python库httpx
|
3月前
|
调度 Python
python知识点100篇系列(20)-python协程与异步编程asyncio
【10月更文挑战第8天】协程(Coroutine)是一种用户态内的上下文切换技术,通过单线程实现代码块间的切换执行。Python中实现协程的方法包括yield、asyncio模块及async/await关键字。其中,async/await结合asyncio模块可更便捷地编写和管理协程,支持异步IO操作,提高程序并发性能。协程函数、协程对象、Task对象等是其核心概念。
|
3月前
|
Python Windows
python知识点100篇系列(24)- 简单强大的日志记录器loguru
【10月更文挑战第11天】Loguru 是一个功能强大的日志记录库,支持日志滚动、压缩、定时删除、高亮和告警等功能。安装简单,使用方便,可通过 `pip install loguru` 快速安装。支持将日志输出到终端或文件,并提供丰富的配置选项,如按时间或大小滚动日志、压缩日志文件等。还支持与邮件通知模块结合,实现邮件告警功能。
python知识点100篇系列(24)- 简单强大的日志记录器loguru
|
3月前
|
自然语言处理 Python Windows
python知识点100篇系列(23)- 使用stylecloud生成词云
【10月更文挑战第10天】`stylecloud` 是 `wordcloud` 的优化版,支持使用 Font Awesome 图标自定义词云形状,操作更简便。本文介绍如何安装 `jieba` 和 `stylecloud` 库,并使用它们生成中文词云。通过 `jieba` 进行分词,再利用 `stylecloud` 的 `gen_stylecloud` 方法生成具有特定形状和颜色的词云图像。
python知识点100篇系列(23)- 使用stylecloud生成词云
|
3月前
|
Java Python
> python知识点100篇系列(19)-使用python下载文件的几种方式
【10月更文挑战第7天】本文介绍了使用Python下载文件的五种方法,包括使用requests、wget、线程池、urllib3和asyncio模块。每种方法适用于不同的场景,如单文件下载、多文件并发下载等,提供了丰富的选择。