Leave me alone.

Python学习(01)

Posted on By gtahub

Python简介

Python 是一个高层次的结合了解释性、编译性、互动性和面向对象的脚本语言。

Python 的设计具有很强的可读性,相比其他语言经常使用英文关键字,其他语言的一些标点符号,它具有比其他语言更有特色语法结构。

  • Python 是一种解释型语言: 这意味着开发过程中没有了编译这个环节。类似于PHP和Perl语言。
  • Python 是交互式语言: 这意味着,您可以在一个 Python 提示符 »> 后直接执行代码。
  • Python 是面向对象语言: 这意味着Python支持面向对象的风格或代码封装在对象的编程技术。
  • Python 是初学者的语言:Python 对初级程序员而言,是一种伟大的语言,它支持广泛的应用程序开发,从简单的文字处理到 WWW 浏览器再到游戏。

Python特点

  • 1.易于学习:Python有相对较少的关键字,结构简单,和一个明确定义的语法,学习起来更加简单。
  • 2.易于阅读:Python代码定义的更清晰。
  • 3.易于维护:Python的成功在于它的源代码是相当容易维护的。
  • 4.一个广泛的标准库:Python的最大的优势之一是丰富的库,跨平台的,在UNIX,Windows和Macintosh兼容很好。
  • 5.互动模式:互动模式的支持,您可以从终端输入执行代码并获得结果的语言,互动的测试和调试代码片断。
  • 6.可移植:基于其开放源代码的特性,Python已经被移植(也就是使其工作)到许多平台。
  • 7.可扩展:如果你需要一段运行很快的关键代码,或者是想要编写一些不愿开放的算法,你可以使用C或C++完成那部分程序,然后从你的Python程序中调用。
  • 8.数据库:Python提供所有主要的商业数据库的接口。
  • 9.GUI编程:Python支持GUI可以创建和移植到许多系统调用。
  • 10.可嵌入: 你可以将Python嵌入到C/C++程序,让你的程序的用户获得”脚本化”的能力。

Python下载

下载地址:https://www.python.org/downloads/

# 查看python3版本
python3 -V
Python 3.8.9

PyCharm下载

下载地址:https://www.jetbrains.com/pycharm/download/

Python中的输出函数

print()函数

可以直接使用的函数叫print(),直接将展示东西输出在控制台

  • 向计算机发出打印指令
  • 把代码编译成计算机能听懂的机器语言
  • 做出相应的执行,在控制台上输出结果
print('hello world')

print()函数的使用

print()函数可以输出哪些内容

  • print()函数输出的内容可以是数字
  • print()函数输出的内容可以是字符串
  • print()函数输出的内容可以是含有运算符的表达式

print()函数可以将内容输出的目的地

  • 显示器
  • 文件

print()函数的输出形式

  • 换行
  • 不换行

代码

# 输出数字
print(520)
print(98.5)

# 输出字符串
# 单引号
print('hello world')
# 双引号
print("hello world")
# 三引号
print("""hello world""")

# 含有运算符的表达式
print(3 + 1)

# 将数据输出到文件中
_file = open('./text.txt', 'a+')
# a+:如果文件不存在就创建,存在就在文件内容的后面继续追加
print('hello world', file=_file)
_file.close()

# 不进行换行输出(输出内容在一行当中)
print('hello', 'world', 'Python')

转义字符

什么是转义字符

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

为什么需要转义字符

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

  • 反斜杠:\\
  • 单引号:\’
  • 双引号:\”

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

使用转义字符

  • 换行\n newline光标移动到下一行到开头
  • 回车\r: return光标移动到本行的开头
  • 水平制表符\t:tab键,光标移动到下一组4个空格的开始处
  • 退格\b:键盘上的backspace键,回退一个字符

代码

# 转义字符
# \ + 转义功能的首字母 n:newline的首字符表示换行
print('hello\nworld')
# 制表符
print('hello\tworld')
print('helloooo\tworld')
# 回车:world将hello进行了覆盖
print('hello\rworld')
# \b是退一个格,将o退没了
print('hello\bworld')

# 反斜杠转义
print('http:\\\\www.baidu.com')
print('你好:\'hello\'')

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

二进制与字符编码

代码

# 十进制unicode
print(ord('乘'))

# 十进制转二进制
print(bin(ord('乘')))

# 二进制
print(chr(0b100111001011000))

Python中的标识符和保留字

保留字

一些单词被赋予了特定的意义,这些单词在你给任何对象起名字的时候都不能用

import keyword
print(keyword.kwlist)

命名规则

  • 变量、函数、类、模块和其他对象起的名字叫做标识符

  • 规则:

    • 字母、数字、下划线_
    • 不能以数字开头
    • 不能是保留字
    • 严格区分大小写
    • 变量名一般多个单词之前用下划线分割,类名用大驼峰命名法

变量

变量是内存中一个带标签的盒子

# name变量名
# =复制运算符
# myname值
name = 'my name'

变量由三部分组成

  • 标识:标识对象所存储的内存地址,使用内置函数id(obj)来获取
  • 类型:表示的是对象的数据类型,使用内置函数type(obj)来获取
  • 值:表示对象所存储的具体数据,使用print(obj)可以将值进行打印输出

代码

name = 'my name'
print('标识', id(name))
print('类型', type(name))
print('值', name)

变量的定义和使用

当多次赋值之后,变量名会指向新的空间

name = 'my name'
print('标识', id(name))
print('类型', type(name))
print('值', name)

print('---------')
name = 'new name'
print('标识', id(name))
print('类型', type(name))
print('值', name)

# 第一个数据5792没有被使用变成了内存垃圾
# 标识 4343655792
# 类型 <class 'str'>
# 值 my name
# ---------
# 标识 4343632112
# 类型 <class 'str'>
# 值 new name

数据类型

常用的数据类型

  • 整数类型int:98
  • 浮点数类型float:3.14159
  • 布尔类型bool:True,False
  • 字符串类型str:’人生苦短,我用Python’

整数类型

整数类型

  • 英文为integer,简写为int,可以表示整数、负数和零
  • 整数的不同进制表示方式
    • 十进制:默认的进制
    • 二进制:以0b开头
    • 八进制:以0o开头
    • 十六进制:0x开头
进制 基本数 逢几进一 表示形式
十进制 0,1,2,3,4,5,6,7,8,9 10 118
二进制 0,1 2 0b1110110
八进制 0,1,2,3,4,5,6,7 8 0o166
十六进制 0,1,2,3,4,5,6,7,8,9,A,B,C,D,E,F 16 0x76

代码

# 整数类型
# 可以表示,正数,负数,0
n1=90
n2=-76
n3=0
print(n1, type(n1))
print(n2, type(n2))
print(n3, type(n3))

# 整数可以表示为二进制,十进制,八进制,十六进制
print('十进制', 118)
print('二进制', 0b1110110)
print('八进制', 0o166)
print('十六进制', 0x76)

浮点类型

浮点类型

  • 浮点数由整数部分和小数部分组成
  • 浮点数存储不精确性
    • 使用浮点数进行计算时,可能会出现小数位数不确定的情况
    • 解决方案:导入模块decimal

代码

from decimal import Decimal

a=3.14159
print(a, type(a))

n1=1.1
n2=2.2
print(n1 + n2)

print(Decimal(n1.__str__()) + Decimal(n2.__str__()))

布尔类型

布尔类型

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

代码

f1 = True
f2 = False

print(f1, type(f1))
print(f2, type(f2))

# 布尔值可以转成整数计算
print(f1 + 1)  # 1 + 1
print(f2 + 1)  # 0 + 1

字符串类型

字符串类型

  • 字符串又被称为不可变的字符序列
  • 可以使用单引号' '双引号" "三引号‘’‘ ’‘’""" """来定义
  • 单引号和双引号定义的字符串必须在一行
  • 三引号定义的字符串可以分布在连续的多行

代码

str1 = '人生苦短,我用Python'
str2 = "人生苦短,我用Python"
str3 = """人生苦短,我用Python"""

str4='''人生苦短,我用Python'''

print(str1, type(str1))
print(str2, type(str2))
print(str3, type(str3))
print(str4, type(str4))

数据类型转换

为什么需要数据类型转换

  • 需要将不同数据类型的数据拼接在一起

数据类型转换的函数

函数名 作用 注意事项 举例
str() 将其他数据类型转成字符串 也可用引号转换 str(123)
‘123’
int() 将其他数据类型转成整数 1.文字类和小数类字符串,无法转化成整数
2.浮点数转化成整数,抹零取整
int(‘123’)
int(9.8)
float() 将其他数据类型转成浮点数 1.文字类无法转成小数
2.整数转成浮点数,末尾为.0
float(‘9.9’)
float(9)

代码

name = '张三'
age = 20

print(type(name), type(age))  # 说明name与age的数据类型不相同
# print('我叫' + name + '今年,' + age + '岁')  # 将str类型与int类型进行连接时,报错,解决方案,类型转换
print('我叫' + name + '今年,' + str(age) + '岁')

print('-------------str()将其他类型转成str类型---------------')
a = 10
b = 198.8
c = False
print(type(a), type(b), type(c))
print(str(a), str(b), str(c))
print(type(str(a)), type(str(b)), type(str(c)))

print('-------------int()将其他类型转成int类型---------------')
s1 = '128'
s2 = '76.77'
s3 = 'hello'
f1 = 98.7
ff = True

print(type(s1), type(s2), type(s3), type(f1), type(ff))
print(int(s1), type(s1))  # 将str类型转成int类型,字符串为数字串
# print(int(s2), type(s2))  # 将str类型转成int类型,报错,因为字符串为小数字符串
# print(int(s3), type(s3))  # 将str类型转成int类型时,字符串必须为数字字符串(整数),非数字串不允许转换
print(int(f1), type(f1))  # float转成int类型,截取整数部分,舍掉小数部分
print(int(ff), type(ff))

print('-------------float()将其他类型转成float类型---------------')

s1 = '128.98'
s2 = '76'
s3 = 'hello'
ff = True
i = 98

print(type(s1), type(s2), type(s3), type(ff), type(i))
print(float(s1), type(float(s1)))
print(float(s2), type(float(s2)))
# print(float(s3), type(float(s3)))  # 字符串中的数据如果是非数字字符串,则不允许转换
print(float(ff), type(float(ff)))
print(float(i), type(float(i)))

Python中的注释

注释

  • 在代码中对代码的功能进行解释说明的标注性文字,可以提高代码的可读性
  • 注释的内容会被Python解释器忽略
  • 通常包括三种类型的注释
    • 单行注释:以#开头,直到换行结束
    • 多行注释:并没有单独的多行注释标记,将一对三引号之间的代码成为多行注释
    • 中文编码声明注释:在文件开头加上中文声明注释,用以指定源码文件的编码格式# coding:gbk

代码

# coding:gbk

# 输入功能(单行注释)
print('hello')

'''
多行注释
'''

Python的输入函数input()

input函数的介绍

input函数

  • 作用:接收来自用户的输入
  • 返回值类型:输入值的类型为str
  • 值的存储:使用=对输入的值进行存储

input函数的基本使用

# present:变量
# =:赋值运算符,将输入函数的结果赋值给变量
# input()是一个输入函数,需要输入回答
present = input('输入内容')

代码

present = input('输入内容')
print(present, type(present))

a = int(input('输入一个加数'))
b = int(input('输入另一个加数'))

print(type(a), type(b))
print(a + b)

Python中的运算符

Python中常用的运算符

常用运算符

  • 算术运算符
    • 标准算术运算符
      • 加(+)、减(-)、乘(*)、除(/)、整除(//)
    • 取余运算符
      • %
    • 幂运算符
      • **
  • 赋值运算符
  • 比较运算符
  • 布尔运算符
  • 位运算符

常用运算符

标准算术运算符

运算符 表示 例子 结果
+ 1+1 2
- 1-1 0
* 2*4 8
/ 1/2 0.5
% 取余(一正一负要公式)
余数=被除数-除数*商
9%4
9%-4
9-(-4)*)(-3)
1
-3
** 幂运算 2**3 2³=8
// 整数(一正一负向下取整) 11//2
9//4
-9//4
5
-3
-3
代码
print(1 + 1)  # 加法运算
print(1 - 1)  # 减法运算
print(2 * 4)  # 乘法运算
print(1 / 2)  # 除法运算
print(11 / 2)  # 除法运算
print(11 // 2)  # 5 整除运算
print(11 % 2)  # 取余运算
print(2 ** 2)  # 表示2的2次方
print(2 ** 3)  # 表示2的三次方

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

print(9 // -4) # -3
print(-9 // 4) # -3 一正一负的整数公式,向下取整

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

赋值运算符

=

  • 执行顺序:从右到左
  • 支持链式赋值:a=b=c=20
  • 支持参数赋值:+=、-=、*=、/=、//=、%=
  • 支持系列解包赋值:a,b,c=20,30,40
代码
a = 3 + 4
print(a)

a = b = c = 20
print(a, id(a))
print(b, id(b))
print(c, id(c))
print('-------支持参数赋值-------')
a = 20
a += 30  # 相当于a=a+30
print(a)
a -= 10  # 相当于a=a-10
print(a)
a *= 2
print(a)
a /= 3  # 相当于a=a*2
print(a)
print(type(a))  # float
a //= 2
print(a)
a %= 3
print(a)

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

# a,b=20,30,40 报错,因为左右变量的个数和值的个数不对应
print('-------------交换两个变量的值--------------')
a, b = 10, 20
print('交换之前:', a, b)
# 交换
a, b = b, a
print('交换之后:', a, b)

比较运算符

对变量或表达式对结果进行大小、真假等比较

  • >,<,>=,<=,!=
  • ==:对象value的比较
  • is,is not:对象的id的比较
代码
a, b = 10, 20
print('a>b', a > b)  # False
print('a<b', a < b)  # True
print('a<=b', a <= b)  # True
print('a>=b', a >= b)  # False
print('a==b', a == b)  # False
print('a!=b', a != b)  # True

'''
    一个=称为赋值运算符,==称为比较运算符
    一个变量由三部分组成,标识,类型,值
    == 比较的是值,比较对象的标识使用is
'''
a = 10
b = 10
print(a == b)  # True 说明a与b的value相等
print(a is b)  # True 说明a与b的id标识相等

list1 = [11, 22, 33, 44]
list2 = [11, 22, 33, 44]
print(list1 == list2)  # value True
print(list1 is list2)  # id False
print(id(list1))
print(id(list2))

print(a is not b)  # False
print(list1 is not list2)  # True

布尔运算符

对于布尔值之间的运算

  • and:两个运算数都为True时,运算结果才为True
  • or:只要有一个运算数为True,运算结果就为True
  • not:如果运算数为True,运算结果为False
  • in:
  • not in
代码
a, b = 1, 2
print('-------------and并且-----------------')
print(a == 1 and b == 2)  # True
print(a == 1 and b < 2)  # False
print(a != 1 and b == 2)  # False
print(a != 1 and b != 2)  # False

print('-------------or或者-----------------')
print(a == 1 or b == 2)  # True
print(a == 1 or b < 2)  # True
print(a != 1 or b == 2)  # True
print(a != 1 or b != 2)  # False

print('-------------not对bool类型操作数取反-----------------')
f = True
f2 = False
print(not f)
print(not f2)

print('-------------in与not in-----------------')
s='helloworld'
print('w' in s)
print('k' in s)
print('w' not in s)
print('k' not in s)

位运算符

将数据转成二进制进行计算

  • 位与&:对应数位都是1,结果数位才是1,否则为0
  • 位或|:对应数位都是0,结果数位才是0,否则为1
  • 左移位运算符«:高位溢出舍弃,低位补0
  • 右移位运算符»:低位溢出舍弃,高位补0
代码
# 4的二进制:0000 0100
# 8的二进制:0000 1000
# 对应位数都为1结果才是1:0000 0000
print(bin(4))
print(bin(8))
# 按位与&,同为1时结果才为1
print(4 & 8)

# 对应位数都是0,结果位数才是0:0000 1100
print(0b00001100)
# 按位或|,同为0时结果为0
print(4 | 8)


# 左移位:左高位截断,右低位补0
# 0000 1000,一位,相当于乘以2
print(0b00001000)
# 向左移动1位(移动1个位置)
print(4 << 1)
# 0001 0000,二位,相当于乘以4
print(0b00010000)
# 向左移动2位(移动2个位置)
print(4 << 2)

# 右移位:左高位补0,右低位截断
# 0000 0010,一位,相当于除以2
print(0b00000010)
# 向右移动1位(移动1个位置)
print(4 >> 1)
# 0000 0001,二位,相当于除以4
print(0b00000001)
# 向右移动2位(移动2个位置)
print(4 >> 2)


运算符的优先级

  • ():括号优先级最高
  • 算术运算符
  • 位运算符
  • 比较运算符
  • 布尔运算符
  • 赋值运算符

对象的布尔值

Python一切皆对象,所有对象都有一个布尔值

  • 获取对象的布尔值:使用内置函数bool()

  • 以下对象的布尔值为False

    • False
    • 数值0
    • None
    • 空字符串
    • 空列表
    • 空元组
    • 空字典
    • 空集合

代码

# 对象的布尔值
print(bool(False))  # False
print(bool(0))  # False
print(bool(0.0))  # False
print(bool(None))  # None
print(bool(''))  # 空字符
print(bool(""))  # 空字符
print(bool([]))  # 空列表
print(bool(list()))  # 空列表
print(bool(()))  # 空元组
print(bool(tuple()))  # 空元组
print(bool({}))  # 空字典
print(bool(dict()))  # 空字典
print(bool(set()))  # 空集合

程序的组织结构

1996年,计算机科学家证明了这样的事实:任何简单或复杂的算法都可以由顺序结构、选择结构和循环结构这三种基本结构组合而成

计算机的流程控制

  • 顺序结构
  • 选择结构
  • 循环结构

顺序结构

程序从上到下顺序地执行代码,中间没有任何的判断和跳转,直到程序结束

选择结构

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

单分支结构

语法结构
if 条件表达式:
		条件执行体
代码
money = 1000
s = int(input('输入取款金额'))
# 判断余额是否充足
if money >= s:
    money -= s
    print('取款成功,余额为:', money)

双分支结构

语法结构
if 条件表达式:
  	条件执行体1
else:
  	条件执行体2
代码
# 双分支结构if...else,二选一执行
'''
从键盘录入一个整数,编写程序让计算机判断是奇数还是偶数
'''
num = int(input('输入一个整数'))

# 条件判断
if num % 2 == 0:
    print(num, '是偶数')
else:
    print(num, '是奇数')

多分支结构

语法结构
if 条件表达式:
  	条件执行体1
elif 条件表达式2:
  	条件执行体2
elif 条件表达式N:
  	条件执行体N
else:
  	条件执行体N+1
代码

'''
多分支结构,多选一执行
从键盘录入一个整数 成绩

90-100 A
80-89 B
70-79 C
60-69 D
0-59 E
小于0或大于100 为非法数据
'''

score = int(input('输入一个成绩:'))
# 判断
if score >= 90 and score <= 100:
    print('A')
elif score >= 80 and score <= 89:
    print('B')
elif score >= 70 and score <= 79:
    print('C')
elif score >= 60 and score <= 69:
    print('D')
elif score >= 0 and score <= 59:
    print('E')
else:
    print('成绩有误,不在成绩的有效范围')

'-----------------写法二------------------'

if 90 <= score <= 100:
    print('A')
elif 80 <= score <= 89:
    print('B')
elif 70 <= score <= 79:
    print('C')
elif 60 <= score <= 69:
    print('D')
elif 0 <= score <= 59:
    print('E')
else:
    print('成绩有误,不在成绩的有效范围')

嵌套if

语法结构
if 条件表达式:
  	if 内层条件表达式:
      	内层条件执行体1
     else:
        内层条件执行体2
else:
  	条件执行体
代码
'''
会员 >= 200 8折
    >= 100 9折
    不打折
非会员 >=200 9.5折
    不打折
'''
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)

条件表达式

条件表达式是if…else的简写

语法结构

x if 判断条件 else y

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

代码
'''
从键盘录入两个整数,比较大小
'''

num_a = int(input('第一个整数'))
num_b = int(input('第二个整数'))
# 比较大小
'''
if num_a >= num_b:
    print(num_a, '大于等于', num_b)
else:
    print(num_a, '小于', num_b)
'''

print('使用条件表达式进行比较')
print(num_a.__str__() + '大于等于' + num_b.__str__() if num_a >= num_b else num_a.__str__() + '小于' + num_b.__str__())

pass语句

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

什么时候使用

先搭建语法结构,还没想好代码怎么写的时候

哪些语句一起使用

  • if语句的条件执行体
  • for-in语句的循环体
  • 定义函数时的函数体

代码

# pass语句,什么都不做,只是一个占位符,用到需要写语句的地方
answer = input('会员y/n')

# 判断是否是会员
if answer == 'y':
    pass
else:
    pass

range函数

range()函数

  • 用于生成一个整数序列
  • 创建range对象的三种方式
    • range(stop):创建一个(0,stop)之间的整数序列,步长为1
    • range(start,stop):创建一个(start,stop)之间的整数序列,步长为1
    • range(start,stop,step):创建一个(start,stop)之间的整数序列,步长为1
  • 返回值是一个迭代器对象
  • range类型的优点:不管range对象表示的整数序列有多长,所以range对象占用的内存空间都是相同的,因为仅仅需要存储start,stop和step,只有当用到range对象时,才会去计算序列中的相关元素
  • in与not in判断整数序列中是否存在(不存在)指定的整数

代码

# range()的三种创建方式

'''第一种,只有一个参数(小括号中只给了一个数)'''
r = range(10)  # [0, 1, 2, 3, 4, 5, 6, 7, 8, 9],默认从0开始,默认相差1称为步长
print(r)  # range(0, 10)
print(list(r))  # 用于查看range对象中的整数序列,list是列表的意思

'''第二种,给了两个参数(小括号中给了两个数)'''
r = range(1, 10)  # 指定了起始值,从1开始,到10结束(不包含10),默认步长为1
print(list(r))  # [1, 2, 3, 4, 5, 6, 7, 8, 9]

'''第三种,给了三个参数(小括号中给了三个数)'''
r = range(1, 10, 2)
print(list(r))  # [1, 3, 5, 7, 9]

'''判断指定的整数 在序列中是否存在in, not in'''
print(10 in r)
print(9 in r)

print(10 not in r)
print(9 not in r)

print(range(1, 20, 1))  # [1...19]
print(range(1, 101, 1))  # [1...100]

循环结构

反复做同一件事情的情况,称为循环,循环的分类

  • while
  • for-in

while

语法结构
while 条件表达式
			条件执行体循环体

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

  • if判断一次,条件为True执行一次
  • while是判断N+1次,条件为True执行N次
while循环的执行流程

四步循环法

  • 初始化变量
  • 条件判断
  • 条件执行体(循环体)
  • 改变变量
代码

a = 1
# 判断条件表达式
while a < 10:
    # 执行条件执行体
    print(a)
    a += 1

for-in

for-in循环

  • in表达从(字符串、序列等)中依次取值,又称为遍历
  • for-in遍历等对象必须是可迭代对象
语法结构
for 变量 in 可迭代对象:
  		循环体

循环体内不需要访问自定义变量,可以将自定义变量替代为下划线

代码

# 遍历字符串
for item in 'Python':
    print(item)

# range()产生一个整数序列,也是一个可迭代对象
for i in range(10):
    print(i)

# 如果在循环体中不需要使用到自定义遍历,可将自定义变量写为"_"
for _ in range(5):
    print('人生苦短,我用Python')

# 使用for循环,计算1到100之间到偶数和
sum = 0  # 用于存储偶数和
for i in range(1, 101):
    if not i % 2:
        sum += i
print(sum)

流程控制语句break

break语句

  • 用于结束循环结构,通常与分支结构if一起使用

代码


'''从键盘录入密码,最多录入三次,如果正确就结束循环'''
for item in range(3):
    pwd = input('输入密码:')
    if pwd == '8888':
        print('正确')
        break
    else:
        print('不正确')

a = 0
while a < 3:
    pwd = input('输入密码:')
    if pwd == '8888':
        print('正确')
        break
    else:
        print('不正确')
    a += 1

流程控制语句continue

continue语句

  • 用于结束当前循环,进入下一次循环,通常与分支结构中的if一起使用

代码


'''
要求输出1到50之间5的倍数,5,10,15,20,25...
5的倍数的共同点:和5的余数为0的数字都是5的倍数
'''

for item in range(1, 51):
    # 模5等于0取反
    if not item % 5:
        print(item)

print('--------使用continue----------')
for item in range(1,51):
    if item % 5:
        continue
    print(item)

else语句

else语句配合使用的三种情况

  • if…else:if条件表达式不成立时执行else
  • 循环的正常执行次数执行完,没有碰到break时执行else
    • while…else
    • for…else

代码


for i in range(3):
    pwd = input('输入密码')
    if pwd == '8888':
        print('密码正确')
        break
    else:
        print('不正确')
else:
    print('三次密码均输入错误')

a = 0
while a < 3:
    pwd = input('输入密码:')
    if pwd == '8888':
        print('正确')
        break
    else:
        print('不正确')
    a += 1
else:
    print('三次密码均输入错误')

列表

变量可以存储一个元素,而列表是一个大容器可以存储N多个元素,程序可以方便地对这些数据进行整体操作,列表相当于其他语言中的数组

       
索引 -3 -2 -1
数据 hello world 123
索引 0 1 2

代码

a = 10  # 变量存储的是一个对象的引用
lst = ['hello', 'world', 98]
print(id(lst))
print(type(lst))
print(lst)

列表的创建

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

列表的创建方式

  • 使用中括号
  • 调用内置函数list()

代码

'''创建列表的第一种方式,使用[]'''
lst = ['hello', 'world', 98]

'''创建列表的第二种方式,使用内置函数list()'''
lst2 = list(['hello', 'world', 98])

列表的特点

列表的特点

  • 列表元素按顺序有序排序
  • 索引映射唯一一个数据
  • 列表可以存储重复数据
  • 任意数据类型混存
  • 根据需要动态分配和回收内存

代码

'''创建列表的第一种方式,使用[]'''
lst = ['hello', 'world', 98, 'hello']
print(lst)
print(lst[0], lst[-4])

'''创建列表的第二种方式,使用内置函数list()'''
lst2 = list(['hello', 'world', 98])

列表的查询操作

获取列表中指定元素的索引

index()

  • 如果查询的列表中存在N个相同元素,只返回相同元素中的第一个元素的索引
  • 如果查询的元素在列表中不存在,则会抛出ValueError
  • 还可以在指定的start和stop之间进行查找

获取列表中的单个元素

获取单个元素

  • 正向索引从0到N-1 举例:lst[0]
  • 逆向索引从-N到-1 举例:lst[-N]
  • 指定索引不存在,则抛出IndexError

代码


lst = ['hello', 'world', 98, 'hello']
print(lst.index('hello'))  # 如果列表中有相同元素只返回列表中相同元素的第一个元素的索引
# print(lst.index('Python')) # ValueError: 'Python' is not in list

# lst.index('hello', 1, 3) # ValueError: 'hello' is not in list 从下标为1的索引查找,到3结束,不包括3

print(lst.index('hello', 1, 4))

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

# 获取索引为10的元素
# print(lst[10]) # IndexError: list index out of range

获取列表中的多个元素

语法格式
列表[start:stop:step]
切片操作
  • 切片的结果:原列表片段的拷贝
  • 切片的范围:[start,stop]
  • step默认为1:简写为[start:stop]
  • step为正数
    • [:stop:step]:切片的第一个元素默认是列表的第一个元素
    • [start::step]:切片的最后一个元素默认是列表的最后一个元素
    • 从start开始往后计算切片
  • step为负数
    • [:stop:step]:切片的第一个元素默认是列表的最后一个元素
    • [start::step]:切片的最后一个元素默认是列表的第一个元素
    • 从start开始往前计算切片
代码
lst = [10, 20, 30, 40, 50, 60, 70, 80]
# start=1,stop=6,step=1
print(lst[1:6:1])
print('原列表', id(lst))
lst2 = lst[1:6:1]
print('切的片段:', id(lst2))

print(lst[1:6])  # 默认步长为1
print(lst[1:6:])
# start=1,stop=6,step=2
print(lst[1:6:2])
# stop=6,step=2, start采用默认
print(lst[:6:2])
# start=1,step=2,stop采用默认
print(lst[1::2])

print('---------step步长为负数的情况----------')
print('原列表', lst)
print(lst[::-1])
# start=7,stop 省略 step=-1
print(lst[7::-1])
# start=6,stop=0,step=-2
print(lst[6:0:-2])

print('---------列表查询操作----------')
lst = [10, 20, 'python', 'hello']
print(10 in lst)  # True
print(100 in lst)  # False
print(10 not in lst)  # False
print(100 not in lst)  # True
for item in lst:
    print(item)

列表元素的增加操作

方法 操作描述
append() 在列表的末尾添加一个元素
extend() 在列表的末尾至少添加一个元素
insert() 在列表的任意位置添加一个元素
切片 在列表的任意位置添加至少一个元素

代码

# 向列表的末尾添加一个元素

lst = [10, 20, 30]
print('添加元素之前', lst, id(lst))
lst.append(100)
print('添加元素之后', lst, id(lst))
lst2 = ['hello', 'world']
# lst.append(lst2) # 将lst2作为一个元素添加到列表到末尾
print(lst)

# 向列表到末尾一次性添加多个元素
lst.extend(lst2)
print(lst)

# 在任意位置上添加一个元素
lst.insert(1,90)
print(lst)


lst3=[True,False,'hello']
# 在任意到位置上添加N多个元素,切掉的部分用新列表去替换
lst[1:]=lst3
print(lst)

列表元素的删除操作

方法 操作描述
remove() 一次删除一个元素。重复元素只删除第一个。元素不存在抛出ValueError
pop() 删除一个指定索引位置上的元素。指定索引不存在抛出IndexError。不指定索引,删除列表中最后一个元素
切片 一次至少删除一个元素
clear() 清空列表
del 删除列表

代码

lst = [10, 20, 30, 40, 50, 60, 30]
lst.remove(30)  # 从列表中移除一个元素,如果有重复元素只移除第一个元素
print(lst)

# lst.remove(100) # ValueError: list.remove(x): x not in list

# pop()根据索引移除元素
lst.pop(1)
print(lst)
# lst.pop(5) # IndexError: pop index out of range 如果指定的索引不存在,将抛出异常

lst.pop()  # 如果不指定参数(索引),将删除列表中的最后一个元素
print(lst)

'-------切片操作-删除至少一个元素,产生一个新的列表对象-----'
new_lst = lst[1:3]
print('原列表', lst)
print('切片后的列表', new_lst)

'''不产生新的列表对象,而是删除原列表中的内容'''
# 空列表替代
lst[1:3] = []
print(lst)


'''清除列表中的所有元素'''
lst.clear()
print(lst)

'''del语句将列表对象删除'''
del lst
# print(lst) # NameError: name 'lst' is not defined

列表元素的修改操作

  • 为指定索引的元素赋予一个新值
  • 为指定的切片赋予一个新值

代码

lst = [10, 20, 30, 40]
# 一次修改一个值
lst[2] = 100
print(lst)
lst[1:3] = [300, 400, 500, 600]
print(lst)

列表元素的排序操作

常见的两种方式

  • 调用sort()方法,列表中的所有元素按照从小到大的顺序进行排序,可以指定reverse=True,进行降序排序
  • 调用内置函数sorted(),可以指定reverse=True,进行降序排序,原列表不发生改变

代码

lst = [20, 40, 10, 98, 54]
print('排序前的列表', lst, id(lst))
# 开始排序,调用列表对象的sort方法,升序排序
lst.sort()
print('排序后的列表', lst, id(lst))

# 通过指定关键字参数,将列表中的元素进行降序排序
lst.sort(reverse=True)  # reverse=True表示降序排序,reverse=False就是升序排序
print(lst)
lst.sort(reverse=False)
print(lst)

print('---------使用内置函数sorted()对列表进行排序,将产生一个新的列表对象---------')
lst = [20, 40, 10, 98, 54]
print('原列表', lst, id(lst))
# 开始排序
new_list = sorted(lst)
print(lst, id(lst))
print(new_list, id(new_list))
# 指定关键字参数,实现列表元素的降序排序
desc_list = sorted(lst, reverse=True)
print(desc_list)

列表生成式

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

语法格式

# i*i:表示列表元素的表达式
# i:自定义变量
# range(1,10):可迭代对象
[ i*i for i in range(1,10) ]

注意事项:表示列表元素的表达式中通常包含自定义变量

代码

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

'''列表中的元素的值为2,4,6,8,10'''
lst2 = [i * 2 for i in range(1, 11)]
print(lst2)

字典

什么是字典

  • Python内置的数据结构之一,与列表一样是一个可变序列
  • 以键值对的方式存储数据,字典是一个无序的序列

语法格式

# scores:字典名
# {}:花括号
scores = { '张三': 100,... }

字典的实现原理

字典的实现原理与查字典类似,查字典是先根据部首或拼音查找对应的页码,Python中的字典是根据key查找value所在的位置

字典的创建

  • 使用花括号{}
  • 使用内置函数dict()

代码

'''字典的创建方式'''
'''使用{}创建字典'''
scores = {
    '张三': 100,
    '李四': 98,
    '王五': 45
}
print(scores)
print(type(scores))

'''创建dict()'''
student = dict(name='jack', age=20)
print(student)

'''空字典'''
d = {}
print(d)

字典的常用操作

字典中元素的获取

  • []:举例scores[‘张三’]
  • get()方法:举例scores.get(‘张三’)
  • []取值与get()取值的区别
    • []如果字典中不存在指定的key,抛出keyError异常
    • get()方法取值,如果字典中不存在指定的key,并不会抛出KeyError而是返回None,可以通过参数设置默认的value,以便指定的key不存在时返回
代码
'''获取字典的元素'''
scores = {'张三': 100, '李四': 98, '王五': 45}
'''第一种方式,使用[]'''
print(scores['张三'])
# print(scores['陈六']) # KeyError: '陈六'

'''第二种方式,使用get()方法'''
print(scores.get('张三'))
print(scores.get('陈六'))  # None
print(scores.get('麻七', 99))  # 99是在查找'麻七'所对应对key不存在时,提供对一个默认值

key的判断

  • in:指定key在字典中存在返回True
  • not in:指定key在字典中不存在返回True

字典的删除

语法

del scores[key]

字典元素的新增

语法

scores[key]=90
代码
'''key的判断'''
scores = {'张三': 100, '李四': 98, '王五': 45}
print('张三' in scores)
print('张三' not in scores)

del scores['张三']  # 删除指定的key-value对
# scores.clear()  # 清空字典的元素
print(scores)

scores['陈六'] = 98  # 新增元素
print(scores)

scores['陈六'] = 100  # 修改元素
print(scores)

获取字典视图

  • keys():获取字典中所有key
  • values():获取字典中所有value
  • items():获取字典中所有key,value对
代码
scores = {'张三': 100, '李四': 98, '王五': 45}

# 获取所有key
keys = scores.keys()
print(keys)
print(type(keys))
print(list(keys))

# 获取所有的value
values = scores.values()
print(values)
print(type(values))
print(list(values))

# 获取所有的键值对
items = scores.items()
print(items)
print(type(items))
print(list(items)) # 转换之后的列表元素是由元组组成

字典元素的遍历

语法

for item in scores:
  	print(item)
代码
scores = {'张三': 100, '李四': 98, '王五': 45}
# 字典元素的遍历
for item in scores:
    print(item, scores[item], scores.get(item))

字典的特点

  • 字典中的所有元素都是一个key-value对,key不允许重复,value可以重复
  • 字典中的元素是无序的
  • 字典中的key必须是不可变对象
  • 字典也可以根据需要动态地伸缩
  • 字典会浪费较大的内存,是一种使用空间换时间的数据结构
代码
d = {'name': '张三', 'name': '李四'}  # key不允许重复
print(d)

d = {'name': '张三', 'nickname': '张三'}  # value可以重复
print(d)

lst = [10, 20, 30]
lst.insert(1, 100)
print(lst)

d = {lst: 100}  # TypeError: unhashable type: 'list'
print(d)

字典生成式

内置函数zip()

  • 用于将可迭代的对象作为参数,将对象中对应的元素打包成一个元组,然后返回由这些元组组成的列表

语法格式

# item.upper():表示字典key的表达式
# price:表示字典value的表达式
# item,price:自定义表示key,value的变量
# zip(items,prices):可迭代对象
{ item.upper(): price for item,price in zip(items,prices) }
代码
items = ['f', 'b', 'o']
prices = [96, 78, 85, 100, 120]

# 进行压缩的过程中,会以元素少的列表为基准
d = {item.upper(): price for item, price in zip(items, prices)}
print(d)

元组

什么是元组

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

  • 不可变序列与可变序列
    • 不可变序列:字符串、元组
      • 不可变序列:没有增、删、改的操作
    • 可变序列:列表、字典
      • 可变序列:可以对序列执行增、删、改操作,对象地址不发生更改

语法格式

t = ('Python', 'hello')

元组的创建方式

  • 小括号():(value)
  • 内置函数tuple:tuple((value))
  • 只包含一个元素的元组需要使用逗号和小括号:t=(10,)

代码

'''元组的创建方式'''

'''第一种创建方式,使用()'''
t = ('Python', 'world', 98)
print(t)
print(type(t))
print(type(t) is tuple)

t2 = 'Python', 'world', 98  # 省略了小括号
print(t2)
print(type(t2))

t3 = ('Python',)  # 如果元组中只有一个元素,逗号不能省
print(t3)
print(type(t3))

'''第二种创建方式,使用内置函数tuple()'''
t1 = tuple(('Python', 'world', 98))
print(t1)
print(type(t1))

'''空元组的创建方式'''
t4 = ()
t5 = tuple()

'''空列表的创建方式'''
lst = []
lst1 = list()

'''空字典的创建方式'''
d = {}
d2 = dict()

print('空列表', lst, lst1)
print('空字典', d, d2)
print('空元组', t4, t5)

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

  • 在多任务环境下,同时操作对象时不需要加锁
  • 因此,在程序中尽量使用不可变序列
  • 注意事项:元组中存储的是对象的引用
    • 如果元组中对象本身是不可变对象,则不能再引用其他对象
    • 如果元组中的对象是可变对象,则可变对象的引用不允许改变,但数据可以改变

元组的遍历

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

代码

'''元组的遍历'''

t = ('Python', 'world', 98)
'''第一种获取元组元素的方式,使用索引'''
print(t[0])
print(t[1])
print(t[2])
'''遍历元组'''
for item in t:
    print(item)

集合

什么是集合

  • Python语言提供的内置数据结构
  • 与列表、字典一样都属于可变类型的序列
  • 集合是没有value的字典

集合的创建方式

  • 直接{}:s={‘Python’, ‘hello’, 90}
  • 使用内置函数set():s=set(range(6))

代码

# 集合的创建方式
'''第一种创建方式使用{}'''
s = {2, 3, 4, 5, 5, 6, 7, 7}  # 集合中的元素不允许重复
print(s)

'''第二种创建方式'''
s1 = set(range(6))
print(s1, type(s1))

s2 = set([1, 2, 3, 4, 5, 5, 6, 6])
print(s2, type(s2))

s3 = set((1, 2, 4, 4, 5, 65))  # 集合中的元素是无序的
print(s3, type(s3))

s4 = set('python')
print(s4, type(s4))

s5 = set({12, 4, 34, 55, 66})
print(s5, type(s5))

# 定义一个空集合
s6 = {}  # dict字典类型
print(type(s6))

s7 = set()
print(type(s7))

集合相关操作

集合元素的判断操作

  • in 或 not in

集合元素的新增操作

  • 调用add()方法,一次添加一个元素
  • 调用update()方法,至少添加一个元素

集合元素的删除操作

  • 调用remove()方法,一次删除一个指定元素,如果指定的元素不存在抛出KeyError
  • 调用discard()方法,一次删除一个指定元素,如果指定的元素不存在不抛出异常
  • 调用pop()方法,一次只删除一个任意元素
  • 调用clear()方法,清空集合

代码

# 集合的相关操作
s = {10, 20, 30, 405, 60}
print(10 in s)  # True
print(100 in s)  # False
print(10 not in s)  # False
print(100 not in s)  # True

''' 集合元素的新增操作'''
s.add(80) # add一次添加一个元素
print(s)

s.update({200,400,300}) # 一次至少添加一个元素
print(s)

s.update([100,99,8])
s.update((78,64,56))
print(s)


'''集合元素的删除操作'''
s.remove(100)
print(s)

# s.remove(500) # KeyError: 500
s.discard(500)
print(s)

s.pop()
print(s)

s.clear()
print(s)

集合间的关系

两个集合是否相等

  • 使用运算符==或!=进行判断

一个集合是否是另一个集合的子集

  • 可以调用方式issubset进行判断

一个集合是否是另一个集合的超集

  • 可以调用方法issuperset进行判断

两个集合是否没有交集

  • 可以调用方法isdisjoint进行判断

代码

'''两个集合是否相等(元素相同,就相等)'''

s = {10, 20, 30, 40}
s2 = {30, 40, 20, 10}
print(s == s2)  # True
print(s != s2)  # False

''' 一个集合是否是另一个集合的子集'''
s1 = {10, 20, 30, 40, 50, 60}
s2 = {10, 20, 30, 40}
s3 = {10, 20, 90}
print(s2.issubset(s1))  # True
print(s3.issubset(s1))  # False

'''一个集合是否是另一个集合的超集'''
print(s1.issuperset(s2))  # True
print(s1.issuperset(s3))  # False

'''两个集合是否没有交集'''
print(s2.isdisjoint(s3))  # False 有交集
s4 = {100, 200, 300}

print(s2.isdisjoint(s4))  # True 没有交集

集合的数学操作

# 集合的数学操作

# 交集
s1={1,2,3,4}
s2={2,3,4,5,6}
print(s1.intersection(s2))
print(s1 & s2) # intersection()与&等价,交集操作

# 并集操作
print(s1.union(s2))
print(s1 | s2) # union与 | 等价,并集操作
print(s1)
print(s2)

# 差集操作
print(s1.difference(s2))
print(s1 - s2)

# 对称差集
print(s1.symmetric_difference(s2))
print(s1 ^ s2)

集合生成式

用于生成集合的公式

# i*i表示集合元素的表达式
# i自定义变量
# range(1,10)可迭代对象
{ i*i for i in range(1,10) }
  • 将{}修改为[]就是列表生成式
  • 没有元组生成式

代码

# 列表生成式
lst = [i * i for i in range(6)]
print(lst)

# 集合生成式
s = {i * i for i in range(6)}
print(s)

列表、字典、元组、集合总结

数据结构 是否可变 是否重复 是否有序 定义符号
列表(list) 可变 可重复 有序 []
元组(tuple) 不可变 可重复 有序 ()
字典(dict) 可变 key不可重复,value可重复 无序 {key:value}
集合(set) 可变 不可重复 无序 {}

字符串的驻留机制

字符串

在Python中字符串是基本数据类型,是一个不可变的字符序列

什么是字符串驻留机制

仅保存一份相同且不可变字符串的方法,不同的值被存放在字符串的驻留池中,Python的驻留机制对相同的字符串只保留一份拷贝,后续创建相同字符串时,不会开辟新空间,而是把字符串的地址赋给新创建的变量

驻留机制的几种情况(交互模式)

  • 字符串的长度为0或1时
  • 符合标识符的字符串:含有字母数字下划线的字符串称为符合标识符的字符串
  • 字符串只在编译时进行驻留,而非运行时
  • [-5,256]之间的整数数字
  • sys中的intern方法强制2个字符串指向同一个对象
  • PyCharm对字符串进行了优化处理

代码

# 字符串的驻留机制
a = 'Python'
b = "Python"
c = '''Python'''
print(a, id(a))
print(b, id(b))
print(c, id(c))

'''字符串长度为0或1'''
s1 = ''
s2 = ''
print(s1 is s2)  # True
s1 = '%'
s2 = '%'
print(s1 is s2)  # True

'''符合标识符的字符串'''
s1 = 'abc%'
s2 = 'abc%'
print(s1 == s2)  # True
print(s1 is s2)  # False

'''字符串只在编译时进行驻留,而非运行时'''
a = 'abc'
# b的值在运行之前就已经连接完成
b = 'ab' + 'c'
# c的值是在运行时使用join对列表进行连接
c = ''.join(['ab', 'c'])
print(a is b)  # True
print(a is c)  # False

'''[-5,256]之间的整数数字'''
a = -5
b = -5
print(a is b)  # True
a = -6
b = -6
print(a is b)  # False

import sys

a='abc%'
b='abc%'
'''使用sys中对intern方法强制驻留'''
a = sys.intern(b)
print(a is b) # True

字符串驻留机制对优缺点

  • 当需要值相同的字符串时,可以直接从字符串池里拿来使用,避免频繁的创建和销毁,提升效率和节约内存,因此拼接字符串和修改字符串时会比较影响性能的
  • 在需要进行字符串拼接时建议使用str类型的join方法,而非+,因为join方法先计算出所有字符串的长度,然后再拷贝,只new一次对象,效率要比“+”效率高

字符串的常用操作

字符串的查询操作的方法

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

代码

# 字符串的查询操作
s = 'hello,hello'
print(s.index('lo'))  # 3
print(s.find('lo'))  # 3
print(s.rindex('lo'))  # 9
print(s.rfind('lo'))  # 9

# print(s.index('k')) # ValueError: substring not found
print(s.find('k'))  # -1
# s.rindex('k')
print(s.rfind('k'))

字符串的大小写转换操作的方法

方法名称 作用
upper() 把字符串中所有字符都转成大写字母
lower() 把字符串中所有字符都转成小写字母
swapcase() 把字符串中所有大小字母转换为小写字母,把所有小写字母转成大写字母
capitalize() 把第一个字符转换为大写,把其他字符转换为小写
title() 把每个单词的第一个字符转换为大写,把每个单词的剩余字符转换为小写

代码

# 字符串中的大小写转换的方法

s='hello,python'
a=s.upper() # 转成大写之后,会产生一个新的字符串对象
print(a, id(a))
print(s, id(s))

b=s.lower()
print(b, id(b))
print(s, id(s))

print(b == s)
print(b is s)

s2='hello,Python'
print(s2.swapcase())

print(s2.title())

print(s2.capitalize())

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

方法名称 作用
center() 居中对齐,第1个参数指定宽度,第2个参数指定填充符,第2个参数是可选的,默认是空格,如果设置宽度小于实际宽度则返回原字符串
ljust() 左对齐,第1个参数指定宽度,第2个参数指定填充符,第2个参数是可选的,默认是空格,如果设置宽度小于实际宽度则返回原字符串
rjust() 右对齐,第1个参数指定宽度,第2个参数指定填充符,第2个参数是可选的,默认是空格如果设置宽度小于实际宽度则返回原字符串
zfill() 右对齐,左边用0填充,该方法只接收一个参数,用于指定字符串的宽度,如果指定的宽度小于等于字符串的长度,返回字符串本身

代码

s='hello,Python'
'''居中对齐'''
print(s.center(20, '*'))

'''左对齐'''
print(s.ljust(20, '*'))
print(s.ljust(10))
print(s.ljust(20))

'''右对齐'''
print(s.rjust(20, '*'))
print(s.rjust(20))
print(s.rjust(10))

'''右对齐,使用0进行填充'''
print(s.zfill(20))
print(s.zfill(10))
print('-8910'.zfill(8))

字符串拆分操作的方法

方法名称 作用
split() 从字符串的左边开始拆分,默认的拆分字符是空格,返回的值是一个列表。通过参数sep指定拆分字符串的分隔符。通过参数maxsplit指定拆分字符串时的最大拆分次数,在经过最大拆分之后,剩余的字符串会单独作为一部分
rsplit() 从字符串的右边开始拆分,默认的拆分字符是空格,返回的值是一个列表。以通过参数sep指定拆分字符串的分隔符。通过maxsplit指定拆分字符串时最大拆分次数,在经过最大拆分之后,剩余的字符串会单独作为一部分

代码


s='hello world Python'
lst=s.split()
print(lst)

s1='hello|world|Python'
print(s1.split(sep='|',maxsplit=1))
print('---------------')
'''rsplit()从右侧开始拆分'''
print(s.rsplit())
print(s1.rsplit('|'))
print(s1.rsplit(sep='|', maxsplit=1))

判断字符串操作的方法

方法名称 作用
isidentifier() 判断指定的字符串是不是合法的标识符
isspace() 判断指定的字符串是否全部由空白字符组成(回车、换行、水平制表符)
isalpha() 判断指定的字符串是否全部由字母组成
isdecimal() 判断指定字符串是否全部由十进制的数字组成
isnumeric() 判断指定的字符串是否全部由数字组成
isalnum() 判断指定字符串是否全部由字母和数字组成

代码

s='hello,python'
print(s.isidentifier())
print('hello'.isidentifier())
print('张三_'.isidentifier())
print('张三_123'.isidentifier())

print('\t'.isspace())

print('abc'.isalpha())
print('张三'.isalpha()) # 拼音
print('张三1'.isalpha())

print('123'.isdecimal())
print('123四'.isdecimal())
print('Ⅱ'.isdecimal())

print('123'.isnumeric())
print('123四'.isnumeric())
print('Ⅱ'.isnumeric())

print('abc1'.isalnum())
print('张三123'.isalnum())
print('abc!'.isalnum())

字符串操作的其他方法

功能 方法名称  
字符串替换 replace() 第1个参数指定被替换的字符串,第2个参数指定替换字符串的字符串,该方法返回替换后得到的字符串,替换前的字符串不发生变化,调用该方法时可以通过第3个参数指定最大替换次数
字符串的合并 join() 将列表或元组中的字符串合并成一个字符串

代码

s = 'hello,Python'
print(s.replace('Python', 'java'))
s1='hello,Python,Python,Python'
print(s1.replace('Python', 'java', 2))

lst = ['hello','java','Python']
print('|'.join(lst))
print(''.join(lst))

t=('hello','Java','Python')
print(''.join(t))


print('*'.join('Python'))

字符串的比较操作

  • 运算符:>,>=,<,<=,==,!=
  • 比较规则:首先比较两个字符串中的第一个字符,如果相等则继续比较下一个字符,依次比较下去,直到两个字符串中的字符不相等时,其比较结果就是两个字符串的比较结果,两个字符串中的所有后续字符将不再被比较
  • 比较原理:两个字符进行比较时,比较的是其ordinal value(原始值),调用内置函数ord可以得到指定字符的ordinal value。与内置函数ord对应的是内置函数chr,调用内置函数chr时指定ordinal value可以得到其对应的字符

代码

print('apple' > 'app')
print('apple' > 'banana')

print(ord('a'), ord('b'))
print(ord('杨'))

print(chr(97), chr(98))
print(chr(26472))

'''
==与is的区别
==比较的是value
is比较的是id是否相等
'''

a=b='Python'
c='Python'
print(a == b)
print(b == c)

# a,b,c指向的是同一块内存空间
print(a is b)
print(a is c)

字符串的切片操作

字符串是不可变类型

  • 不具备增、删、改等操作
  • 切片操作将产生新的对象

代码

s = 'hello,Python'

s1 = s[:5]  # 由于没有指定起始位置,所以从0开始切
s2 = s[6:]  # 由于没有指定结束位置,所以切到字符串的最后一个元素
s3 = '!'

new_str = s1 + s3 + s2

print(s1)
print(s2)
print(new_str)

print(id(s))
print(id(s1))
print(id(s2))
print(id(s3))
print(id(new_str))

print('切片[start:end:step]')
print(s[1:5:1])  # 从1开始截到5(不包括5),步长为1
print(s[::2])  # 默认从0开始,没有写结束,默认到字符串到最后一个元素,步长为2,两个元素之间到索引间隔为2
print(s[::-1])  # 默认从字符串到最后一个元素开始,到字符串到第一个元素结束,因为步长为负数
print(s[-6::1])  # 从索引为-6开始,到字符串到最后一个元素结束,步长为1

格式化字符串

  • %作占位符
  • {}作占位符

代码

# 格式化字符串

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

# {}占位符
print('我叫{0},今年{1}岁'.format(name, age))

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

# 字符宽度
print('%10d' % 99)  # 10表示的是宽度
print('%.3f' % 3.1415926)  # .3表示小数点后三位

# 同时表示宽度和精度
print('%10.3f' % 3.1415926)  # 总宽度为10,小数点后3位

# {}格式化
print('{0:.3}'.format(3.1415926))  # .3表示一共是3位数
print('{:.3f}'.format(3.1415926))  # .3f表示3位小数
print('{:10.3f}'.format(3.1415926))  # .3f表示3位小数,10表示宽度,同时设置宽度和精度

字符串的编码转换

编码和解码的方式

  • 编码:将字符串转换为二进制数据(bytes)
  • 解码:将bytes类型的数据转换成字符串类型

代码


# byte代表一个二进制数据(字节类型的数据)

s = '天涯共此时'
byte=s.encode(encoding='GBK') # 编码
print(byte)
print(byte.decode(encoding='GBK')) # 解码

byte = s.encode(encoding='UTF-8')
print(byte.decode(encoding='UTF-8'))

函数

函数的创建和调用

什么是函数

函数就是执行特定任务和完成特定功能的一段代码

为什么需要函数

  • 复用代码
  • 隐藏实现细节
  • 提高可维护性
  • 提高可读性便于调试

函数的创建

语法

def 函数名([输入参数]):
  	函数体
    [return xxx]

代码

def calc(a, b):
    c = a + b
    return c

result = calc(10, 20)
print(result)

函数的参数传递

  • 位置实参
    • 根据形参对应的位置进行实参传递
  • 关键字实参
    • 根据形参名称进行实参传递

代码

def calc(a, b):  # a,b称为形式参数,简称形参,形参的位置是在函数的定义处
    c = a + b
    return c


result = calc(10, 20)  # 10,20称为实际参数的值,简称实参,实参的位置是函数的调用处
print(result)

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

'''
在函数调用过程中,进行参数的传递
如果是不可变对象,在函数体的修改不会影响实参的值 arg1修改为100,不会影响n1的值
如果是可变对象,在函数体的修改会影响到实参的值 arg2修改.append(10),会影响到n2点值
'''

函数定义默认值参数

函数定义时,给形参设置默认值,只有与默认值不符的时候才需要传递实参

代码

def fun(a, b=10):  # b称为默认值参数
    print(a, b)


# 函数的调用
fun(100)
fun(20, 30)

print('hello', end='\t')
print('world')

函数的参数定义

个数可变的位置参数

  • 定义函数时,可能无法事先确定传递的位置实参的个数时,使用可变的位置参数
  • 使用*定义个数可变的位置形参
  • 结果为一个元组

个数可变的关键字形参

  • 定义函数时,无法实现确定传递的关键字是实参的个数时,使用可变的关键字形参
  • 使用**定义个数可变的关键字形参
  • 结果为一个字典

代码

# 函数定义时的可变的位置参数
def fun(*args):
    print(args)
    # print(args[0])


fun(10)
fun(10, 20, 30)
fun(10, 20, 30, 40)


# 函数定义时的可变的关键字参数
def fun1(**args):
    print(args)


fun1(a=10)
fun1(a=10, b=30, c=40)

'''
def fun2(*args, *a):
    pass
    以上代码,程序会报错,可变的位置参数,只能是一个
def fun2(**arg, **a):
    pass
    以上代码,程序会报错,个数可变的关键字参数,只能是一个
'''

def fun2(*args, **args2):
    pass

'''
def fun3(**args1, *args2):
    pass
    在一个函数的定义过程中,既有个数可变的关键字形参,也有个数可变的位置形参
    要求,个数可变的位置形参,放在个数可变的关键字形参之前
'''

函数的参数总结

参数的类型 函数的定义 函数的调用 备注
位置实参
将序列中的每个元素都转换为位置实参
  使用*
关键字实参
将字典中的每个键值对都转换为关键字实参
  使用**
默认值形参    
关键字形参   使用*
个数可变的位置形参   使用*
个数可变的关键字形参   使用**

代码

# a,b,c在函数的定义处,所以是形式参数
def fun(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, c=300, b=200)

# 将字典参数传入
dic = {'a': 111, 'b': 222, 'c': 333}
# 在函数调用时,将字典中的键值对都转换为关键字实参传入
fun(**dic)
代码2
'''----------默认值形参------------'''
# b是在函数的定义处,所以b是形参,而且进行了赋值,所以b称为默认值形参
def fun(a, b=10):
    print('a=', a)
    print('b=', b)


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


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


fun2(10, 20, 30)
fun3(a=11, b=22, c=33)

# 从*之后的参数,在函数调用时,只能采用关键字参数传递
def fun4(a, b,*, c, d):
    print('a=', a)
    print('b=', b)
    print('c=', c)
    print('d=', d)

# 位置实参传递
# fun4(10,20,30,40)
# 关键字实参传递
fun4(a=10,b=2,c=30,d=40)
# 前两个参数,采用的是位置实参传递,而c,d采用的是关键字实参传递
fun4(10,20,c=30,d=40)
# c,d只能采用关键字传递

'''函数定义时的形参的顺序问题'''
def fun5(a,b,*,c,d,**args):
    pass

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

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

变量的作用域

  • 程序代码能访问该变量的区域
  • 根据变量的有效范围可分为
    • 局部变量
      • 在函数内定义并使用的变量,只在函数内部有效,局部变量使用global声明,这个变量就会变成全局变量
    • 全局变量
      • 函数体外定义的变量,可作用于函数内外

代码


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


# 因为a,c超出了起作用的范围(超出了作用域)
# print(c)
# print(a)

# name的作用范围为函数内部和外部都可以使用--称为全局变量
name = '张三'
print(name)

def fun2():
    print(name)

fun2()

def fun3():
    # 函数内部定义的变量,局部变量,使用global声明,这个变量实际上就变成了全局变量
    global age
    age = 20
    print(age)

fun3()
print(age)

递归函数

什么是递归函数

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

递归的组成部分

  • 递归调用与递归终止条件

递归的调用过程

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

递归的优缺点

  • 缺点:占用内存多,效率低下
  • 优点:思路和代码简单

代码


def fac(n):
    if n == 1:
        return 1
    else:
        return n * fac(n - 1)


print(fac(6))

Bug

Bug的常见类型

  • 粗心导致的语法错误SyntaxError

  • age = input('输入年龄')
    # str类型与整数比较
    if age >= 18:
      print()
    
  • # 未声明变量直接使用
    # 循环没有终止条件
    while i<10:
      print(i)
    

异常处理机制

  • Python提供了异常处理机制,可以在异常出现时即时捕获,然后内部“消化”,让程序继续运行
  • 多个except结构
    • 捕获异常的顺序按照先子类后父类的顺序,为了避免遗漏可能出现的异常,可以在最后增加BaseException
try:
    a = int(input('第一个整数'))
    b = int(input('第二个整数'))
    result = a / b
    print('结果为:', result)
except ZeroDivisionError:
    print('除数不允许为0')
except ValueError:
    print('只能输入数字')

print('程序结束')

try…except…else结构

如果try块中没有抛出异常,则执行else块,如果try中抛出异常,则执行except块

代码
try:
    a = int(input('第一个整数'))
    b = int(input('第二个整数'))
    result = a / b
    print('结果为:', result)
except BaseException as e:
    print('出错了', e)
else:
    print('计算结果为:', result)

try…except..else…finally结构

finally块无论是否发生异常都会被执行,常用来释放try块中申请的资源

代码
try:
    a = int(input('第一个整数'))
    b = int(input('第二个整数'))
    result = a / b
    print('结果为:', result)
except BaseException as e:
    print('出错了', e)
else:
    print('计算结果为:', result)
finally:
    print('谢谢使用')

Python常见的异常类型

异常类型 描述
ZeroDivisionError 除(或取模)零(所有数据类型)
IndexError 序列中没有此序列(index)
KeyError 映射中没有这个键
NameError 未声明/初始化对象(没有属性)
SyntaxError Python语法错误
ValueError 传入无效的参数

trackback模块

import traceback

try:
    print('------------')
    num = 10 / 0
except:
    traceback.print_exc()

编程思想

两大思想

  面向过程 面向对象
区别 事物比较简单,可以用线性的思维去解决 事物比较复杂,使用简单的线性思维无法解决
共同点 面向过程和面向对象都是解决实际问题的一种思维方式  
  二者相辅相成,并不是对立的。解决复杂问题,通过面向对象方式便于我们从宏观上把握事物之间复杂的关系、方便我们分析整个系统;具体到微观操作,仍然使用面向过程方式来处理  

类与对象

  • 类:类是多个类似事物组成的群体的统称。能够帮助我们快速理解和判断事物的性质
  • 数据类型
    • 不同的数据类型属于不同的类
    • 使用内置函数type查看数据类型
  • 对象
    • 100,99都是int类之下包含的相似的不同个例,这些个例被称为实例或对象

类的创建

语法

class Student:
  pass

类的组成

  • 类属性
  • 实例方法
  • 静态方法
  • 类方法

代码



# Student为类的名称(类名)由一个或多个单词组成,每个单词的首字母大写,其余小写
class Student:
    native_pace = '吉林' # 直接写在类里的变量,称为类属性

    def __init__(self, name, age):
        # self.name 称为实例属性,进行类一个赋值的操作,将局部变量的name的值赋给实例属性
        self.name = name
        self.age = age

    # 实例方法
    def eat(self):
        print('学生在吃饭...')

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

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


# 在类之外定义的称为函数,在类之内定义的称为方法
def drink():
    print('喝水')

print(id(Student))
print(type(Student))
print(Student)

对象的创建

  • 对象的创建又称为类的实例化
  • 语法:实例名=类名()

代码



# Student为类的名称(类名)由一个或多个单词组成,每个单词的首字母大写,其余小写
class Student:
    native_pace = '吉林' # 直接写在类里的变量,称为类属性

    def __init__(self, name, age):
        # self.name 称为实例属性,进行类一个赋值的操作,将局部变量的name的值赋给实例属性
        self.name = name
        self.age = age

    # 实例方法
    def eat(self):
        print('学生在吃饭...')

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

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


# 在类之外定义的称为函数,在类之内定义的称为方法
def drink():
    print('喝水')

print(id(Student))
print(type(Student))
print(Student)
print('-----------------')
# 创建Student对象
stu1 = Student('张三',20)
stu1.eat()
print(stu1.name)
print(stu1.age)
print(id(stu1))
print(type(stu1))
print(stu1)

print('-----------------')
Student.eat(stu1)

类属性、类方法、静态方法

  • 类属性:类中方法外的变量称为类属性,被该类的所有对象所共享
  • 类方法:使用@classmethod修饰的方法,使用类名直接访问的方法
  • 静态方法:使用@staticmethod修饰的方法,使用类名直接访问的方法

代码



# Student为类的名称(类名)由一个或多个单词组成,每个单词的首字母大写,其余小写
class Student:
    native_pace = '吉林' # 直接写在类里的变量,称为类属性

    def __init__(self, name, age):
        # self.name 称为实例属性,进行类一个赋值的操作,将局部变量的name的值赋给实例属性
        self.name = name
        self.age = age

    # 实例方法
    def eat(self):
        print('学生在吃饭...')

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

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


print(Student.native_pace)
stu1=Student('张三',20)
stu2=Student('李四',20)
print(stu1.native_pace)
print(stu2.native_pace)
Student.native_pace = '天津'

print(stu1.native_pace)
print(stu2.native_pace)

print('类方法的使用方式')
Student.cm()
print('静态方法的使用方式')
Student.method()

动态绑定属性和方法

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

代码

class Student:

    def __init__(self, name, age):
        self.name = name
        self.age = age

    def eat(self):
        print(self.name + '在吃饭...')


stu1 = Student('张三', 20)
stu2 = Student('李四', 20)
print(id(stu1))
print(id(stu2))
print('为stu2动态绑定属性')
stu2.gender = '女'
print(stu1.name, stu1.age)
print(stu2.name, stu2.age, stu2.gender)

stu1.eat()
stu2.eat()


def show():
    print('调用了show')


stu1.show = show
stu1.show()

# stu2.show() # 因为stu2没有绑定show方法

封装

面向对象的三大特征

  • 封装:提高程序的安全性
    • 将数据(属性)和行为(方法)包装到类对象中。在方法内部对属性进行操作,在类对象外部调用方法。这样,无需关心方法内部对具体实现细节,从而隔离了复杂度
    • 在Python中没有专门的修饰符用于属性的私有,如果该属性不希望在类对象外部被访问,前边使用两个“_”
  • 继承:提高代码的复用性
  • 多态:提高程序的可扩展性和可维护性

代码

class Student:

    def __init__(self, name, age):
        self.name = name
        self.__age = age # 年龄不希望在类的外部被使用,所以加类两个__

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

stu = Student('张三', 20)
stu.show()
print(stu.name)
# print(stu.__age)

print(dir(stu))
print(stu._Student__age)

继承

  • 语法格式

    class 子类(父类1, 父类2...):
      pass
    
  • 如果一个类没有继承任何类,则默认继承object
  • Python支持多继承
  • 定义子类时,必须在其构造函数中调用父类的构造函数

代码


class Person:
    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_no):
        super().__init__(name,age)
        self.stu_no = stu_no

class Teacher(Person):
    def __init__(self, name,age,teachofyear):
        super().__init__(name,age)
        self.teachofyear=teachofyear

stu = Student('张三',20,'1001')
teacher=Teacher('李四',34,10)

stu.info()
teacher.info()

'''多继承'''
class A:
    pass

class B:
    pass

class C(A,B):
    pass

方法重写

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

代码


class Person:
    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_no):
        super().__init__(name,age)
        self.stu_no = stu_no

    def info(self):
        super().info()
        print(self.stu_no)

class Teacher(Person):
    def __init__(self, name,age,teachofyear):
        super().__init__(name,age)
        self.teachofyear=teachofyear

    def info(self):
        super().info()
        print(self.teachofyear)

stu = Student('张三',20,'1001')
teacher=Teacher('李四',34,10)

stu.info()
teacher.info()


object类

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

多态

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

代码


class Animal:
    def eat(self):
        print('动物会吃')
class Dog(Animal):
    def eat(self):
        print('狗会吃')
class Pig(Animal):
    def eat(self):
        print('猪会吃')

class Person:
    def eat(self):
        print('人会吃')

def fun(o):
    o.eat()

fun(Animal())
fun(Dog())
fun(Pig())
fun(Person())

静态语言与动态语言

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

  • 静态语言实现多态的三个必要条件
    • 继承
    • 方法重写
    • 父类引用指向子类对象
  • 动态语言的多态崇尚类型,不需要关心对象是什么类型,只关心对象的行为

特殊方法和特殊属性

  名称 描述
特殊属性 __dict__ 获得类对象或实例对象所绑定的所以属性和方法的字典
特殊方法 __len__() 通过重写__len__()方法,让内置函数len()的参数可以是自定义类型
  __add__() 通过重写__add__()方法,可使用自定义对象具有’+’功能
  __new__() 用于创建对象
  __init__() 对创建的对象进行初始化

代码

class A:
    pass


class B:
    pass


class C(A, B):
    def __init__(self, name):
        self.name = name


class D(A):
    pass


x = C('xxxx')  # x是C类型的一个实例对象
print(x.__dict__)  # 实例对象的属性字典
print(C.__dict__)

print(x.__class__)  # 对象所属的类
print(C.__bases__)  # C类的父类类型的元素
print(C.__base__)  # 类的基类(最近的继承)
print(C.__mro__)  # 类的层次结构
print(A.__subclasses__())  # 子类的列表

类的浅拷贝与深拷贝

  • 变量的赋值操作
    • 只是形成两个变量,实际上还是指向同一个对象
  • 浅拷贝
    • Python拷贝一般都是浅拷贝,拷贝时,对象包含的子对象内容不拷贝,因为,源对象与拷贝对象会引用同一个子对象
  • 深拷贝
    • 使用copy模块的deepcopy函数,递归拷贝对象中包含的子对象,源对象和拷贝对象所有的子对象也不相同

代码


class CPU:
    pass

class Disk:
    pass

class Computer:
    def __init__(self, cpu, disk):
        self.cpu = cpu
        self.disk = disk

# 变量的赋值
cpu1=CPU()
cpu2=cpu1
print(cpu1, id(cpu1))
print(cpu2, id(cpu2))

# 类的浅拷贝
disk = Disk()
computer= Computer(cpu1, disk)

# 浅拷贝
import copy
computer2 = copy.copy(computer)
print(computer, computer.cpu, computer.disk)
print(computer2, computer2.cpu, computer2.disk)

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

模块

  • 函数与模块的关系
    • 一个模块中可以包含N多个函数
  • 在Python中一个扩展名为.py的文件就是一个模块
  • 使用模块的好处
    • 方便其他程序和脚本的导入并使用
    • 避免函数名和变量名冲突
    • 提高代码的可维护性
    • 提高代码的可重用性

自定义模块

  • 创建模块

    • 新建一个.py文件,名称不要与Python自带的标准模块名称相同
  • 导入模块

    import 模块名称 [as 别名]
    from 模块名称 import 函数/变量/
    

代码


import math

print(id(math))
print(type(math))
print(math)
print(math.pi)

print(dir(math))
# 2的三次方
print(math.pow(2, 3), type(math.pow(2, 3)))
print(math.ceil(9.001))
print(math.floor(9.999))

from math import pi
from math import pow

print(pi)
print(pow(2, 3))

以主程序形式运行

在每个模块的定义中都包括一个记录模块名称的变量__name__,程序可以检查该变量,以确定在哪个模块中执行。如果一个模块不是被导入到其他程序中执行,那么它可能在解释器的顶级模块中执行。顶级模块的__name__变量的值为__main__

代码

```calc.py
def add(a,b):
    return a+b


if __name__ == '__main__':
    print(add(10, 20)) # 只有运行calc时,才会执行运算
    
```calc2.py

import calc

print(calc.add(100,200))

Python中的包

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

  • 作用

    • 代码规范
    • 避免模块名称冲突
  • 包与目录的区别

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

    import 包名.模块名
    

Python中常用的内置模块

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

代码


import sys
import time
import urllib.request

print(sys.getsizeof(24))
print(sys.getsizeof(45))
print(sys.getsizeof(True))
print(sys.getsizeof(False))
print(time.time())
print(time.localtime(time.time()))
print(urllib.request.urlopen('http://www.baidu.com').read())

第三方模块的安装及使用

  • 第三方模块的安装

    pip install 模块名
    
  • 第三方模块的使用

    import 模块名
    

代码

import schedule
import time

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

schedule.every(3).seconds.do(job)

while True:
    schedule.run_pending()
    time.sleep(1)

文件读写

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

  • 语法规则

    file = open(filename, [,mode,encoding])
    # 打开模式默认只读
    # 默认文本文件中字符的编码格式为gbk
    

读取文件

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

常用的文件打开模式

文件的类型

  • 按文件中数据的组织形式,文件分为两大类
    • 文本文件:存储的是普通“字符”文本,默认为unicode字符集,可以使用记事本程序打开
    • 二进制文件:把数据内容用”字节“进行存储,无法用记事本打开,必须使用专用的软件打开,举例:mp3音频文件,jpg图片,.doc文档等
打开模式 描述
r 以只读模式打开文件,文件的指针将会放在文件的开头
w 以只写模式打开文件,如果文件不存在则创建,如果文件存在,则覆盖原有内容,文件指针在文件的开头
a 以追加模式打开文件,如果文件不存在则创建,文件指针在文件开头,如果文件存在,则在文件末尾追加内容,文件指针在原文件末尾
b 以二进制方式打开文件,不能单独使用,需要与其他模式一起使用,rb,或者wb
+ 以读写方式打开文件,不能单独使用,需要与其他模式一起使用,a+

代码

'''追加写入'''
file = open('b.txt', 'a')
file.write('p')
file.close()

'''复制文件操作'''
src_file = open('logo.png', 'rb')
target_file = open('copylogo.png', 'wb')

target_file.write(src_file.read())
target_file.close()
src_file.close()

文件对象的常用方法

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

with语句

上下文管理器

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

  • 实现了__enter__()方法和__exit__()方法,离开运行时上下文,自动调用上下文管理器的特殊方法__exit__()

代码

'''读取文件内容'''
with open('a.txt', 'r') as file:
    print(file.read())

'''自定义上下文管理器'''
class ContentMgr:
    def __enter__(self):
        print('enter execute')
        return self

    def __exit__(self, exc_type, exc_val, exc_tb):
        print('exit execute')

    def show(self):
        print('show execute')


with ContentMgr() as file:
    file.show()

'''复制文件操作'''
with open('logo.png', 'rb') as src_file:
    with open('copylogo.png', 'wb') as target_file:
        target_file.write(src_file.read())

目录操作

  • os模块是Python内置的与操作系统功能和文件系统相关的模块,该模块中的语句的执行结果通常与操作系统有关,在不同的操作系统上运行,得到的结果可能不一样
  • os模块与os.path模块用于对目录或文件进行操作

os模块操作目录相关函数

函数 说明
getcwd() 返回当前的工作目录
listdir(path) 返回指定路径下的文件和目录信息
mkdir(path,[,mode]) 创建目录
makedirs(path1/path2,…[,mode]) 创建多级目录
rmdir(path) 删除目录
removedirs(path1/path2……) 删除多级目录
chdir(path) 将path设置为当前工作目录

代码


import os

print(os.getcwd())
lst = os.listdir()
print(lst)

# os.mkdir('directory')
# os.makedirs('a/b/c')

# os.rmdir('a')
# os.removedirs('a/b/c')

os.chdir('../')
print(os.getcwd())

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

函数 说明
abspath(path) 用于获取文件或目录的绝对路径
exists(path) 用于判断文件或目录是否存在,如果存在返回True,否则返回False
join(path,name) 将目录与目录或者文件名拼接起来
splitext() 分离文件名和扩展名
basename(path) 从一个目录中提取文件名
dirname(path) 从一个路径中提取文件路径,不包括文件名
isdir(path) 用于判断是否为路径

代码


import os.path

print(os.path.abspath('demo1.py'))
print(os.path.exists('demo1.py'), os.path.exists('demo20.py'))
print(os.path.join('/home', 'demo1.py'))
print(os.path.split(os.path.abspath('demo1.py')))
print(os.path.splitext('demo1.py'))
print(os.path.basename(os.path.abspath('demo1.py')))
print(os.path.dirname(os.path.abspath('demo1.py')))
print(os.path.isdir(os.path.abspath('demo1.py')))

print('--------------递归读取文件---------------')
path = os.getcwd()
lst_files = os.walk(path)
for dirpath,dirname,filename in lst_files:
    print(dirpath)
    print(dirname)
    print(filename)
    print('------------')
    for dir in dirname:
        print(os.path.join(dirpath, dir))
    for file in filename:
        print(os.path.join(dirpath, file))

参考教程

-