一、文件操作
1.1 文件的基本操作
1.1.1 打开
打开文件操作步骤:
- 打开文件
- 读写等操作
- 关闭文件
用 open 函数,打开已经存在的文件,或者创建一个新文件,语法: open(name, mode)
name :是要打开的目标文件名的字符串(可以包含文件所在的具体路径)。mode :设置打开文件的模式(访问模式):只读、写入、追加等。
1.1.2 打开文件模式
模式 | 描述 |
---|
r | 以只读方式打开文件。文件的指针将会放在文件的开头。这是默认模式。 | rb | 以二进制格式打开一个文件用于只读。文件指针将会放在文件的开头。这是默认模式。 | r+ | 打开一个文件用于读写。文件指针将会放在文件的开头。 | rb+ | 以二进制格式打开一个文件用于读写。文件指针将会放在文件的开头。 | w | 打开一个文件只用于写入。如果该文件已存在则打开文件,并从开头开始编辑,即原有内容会被删除。如果该文件不存在,创建新文件。 | wb | 以二进制格式打开一个文件只用于写入。如果该文件已存在则打开文件,并从开头开始编辑,即原有内容会被删除。如果该文件不存在,创建新文件。 | w+ | 打开一个文件用于读写。如果该文件已存在则打开文件,并从开头开始编辑,即原有内容会被删除。如果该文件不存在,创建新文件。 | wb+ | 以二进制格式打开一个文件用于读写。如果该文件已存在则打开文件,并从开头开始编辑,即原有内容会被删除。如果该文件不存在,创建新文件。 | a | 打开一个文件用于追加。如果该文件已存在、文件指针将会放在文件的结尾。也就是说,新的内容将会被写入到已有内容之后。如果该文件不存在,创建新文件进行写入。 | ab | 以二进制格式打开一个文件用于追加。如果该文件已存在,文件指针将会放在文件的结尾。也就是说,新的内容将会被写入到已有内容之后。如果该文件不存在,创建新文件进行写入。 | a+ | 打开一个文件用于读写。如果该文件已存在,文件指针将会放在文件的结尾。文件打开时会是追加模式。如果该文件不存在,创建新文件用于读写。 | ab+ | 以二进制格式打开一个文件用于追加。如果该文件已存在,文件指针将会放在文件的结尾。如果该文件不存在,创建新文件用于读写。 |
注意:
- 基本模式有:
r w a ,其他的模式是在其基础上泛生而来。 - 模式中带有 b,代表二进制。模式中带有 +,代表可读可写。
示例1:
f = open('test.txt', 'r')
con = f.read()
print(con)
f.close()
f = open('text1.txt', 'w')
f.write('bbb')
f.close()
f = open('test.txt', 'a')
f.write('lalala')
f.close()
f = open('text.txt')
f.close()
示例2:
f = open('test.txt', 'w+')
f.write("haha")
con = f.read()
print(con)
f.close()
1.1.3 文件对象方法
1.1.3.1 写
语法: 文件对象.write( '内容')
f = open('test.txt', 'w')
f.write('hello world')
f.close()
注意:
- w 和 a 模式:如果文件不存在则创建该文件如果文件存在,w 模式先清空再写入,a 模式直接末尾追加。
- r 模式:如果文件不存在则报错。
1.1.3.2 读
-
设 test.txt 文件中内容如下: -
read() 语法: 文件对象.read( num) num 表示要从文件中读取的数据的长度(单位是字节),如果没有传入 num,那么就表示读取文件中所有的数据。 f = open('test.txt', 'r')
print(f.read())
print(f.read(4))
f.close()
结果如下: -
readlines() readlines 可以按照行的方式把整个文件中的内容进行一次性读取,并且返回的是一个列表,其中每一行的数据为一个元素。 f = open('test.txt', 'r')
con = f.readlines()
print(con)
f.close()
-
readline() readline 每次打印一行数据,文件指针会后移。 f = open('test.txt', 'r')
con = f.readline()
print(f'第??:{con}')
con = f.readline()
print(f'第二?:{con}')
con = f.readline()
print(f'第三?:{con}')
f.close()
结果如下:
1.1.2.3 seek()
作用: 用来移动文件指针
语法: 文件对象.seek(偏移量,起始位置)
起始位置: 0:文件开头?1:当前位置 ?2:文件结尾
f = open('test.txt', 'r+')
f.seek(2, 0)
con = f.read()
print(con)
f.close()
结果如下:
1.1.4 关闭
文件对象.close()
1.2 文件备份
备份文件步骤:
- 接收用户输入的文件名
- 规划备份文件名
- 备份文件写入数据
代码如下:
old_name = input('请输入要备份的文件名:')
index = old_name.rfind('.')
if index > 0:
postfix = old_name[index:]
new_name = old_name[:index] + '[备份]' + postfix
old_f = open(old_name, 'rb')
new_f = open(new_name, 'wb')
while True:
con = old_f.read(1024)
if len(con) == 0:
break
new_f.write(con)
old_f.close()
new_f.close()
1.3 文件和文件夹操作
在 Python 中文件和文件夹的操作要借助 os 模块里面的相关功能。
导入 os 模块:import os
1.3.1 文件/文件夹重命名
os.rename(目标文件名,新文件名)
1.3.2 删除文件
os.remove(目标文件名)
1.3.3 创建文件夹
os.mkdir(文件夹名字)
1.3.4 删除文件夹
os.rmdir(文件夹名字)
1.3.5 获取当前目录
os.getcwd()
1.3.6 改变默认目录
os.chdir(目录)
1.3.7 获取目录列表
os.listdir(?目录)
应用:批量重命名
import os
flag = 2
file_list = os.listdir()
print(file_list)
for i in file_list:
if flag == 1:
new_name = 'python-' + i
elif flag == 2:
num = len('python-')
new_name = i[num:]
os.rename(i, new_name)
二、面向对象基础
2.1 面向对象的实现方法
2.1.1 定义类
语法:
class 类名():
代码
......
注意: 类名遵循大驼峰命名习惯,即首字母大写
扩展
2.1.2 创建对象
语法: 对象名 = 类名()
注意: 创建对象的过程也叫实例化对象。
2.1.3 self
self 指的是调用该函数的对象。
示例:
class Washer():
def wash(self):
print("洗衣服")
print(self)
haier1 = Washer()
print(haier1)
haier1.wash()
haier2 = Washer()
print(haier2)
2.2 添加和获取对象属性
添加对象属性的方式有两种:类外面添加、类里面添加
-
类外面添加和获取对象属性 语法:对象名.属性名 -
类里面添加获取对象属性 语法:self.属性名
示例:
class Washer():
def __init__(self):
self.height = 300
def print_info(self):
print(f'洗衣机的宽度:{self.width}')
haier = Washer()
haier.width = 400
print(f'洗衣机的宽度:{haier.width}')
haier.print_info()
2.3 魔法方法
__xx__() 的函数叫做魔法方法,指的是具有特殊功能的函数。
2.3.1 init()
__init__() 方法的作用:初始化对象,分为带参数和不带参数两种。
-
不带参数: class Washer():
def __init__(self):
self.width = 400
self.height = 400
def print_info(self):
print(f'洗衣机的宽度:{self.width}')
print(f'洗衣机的宽度:{self.width}')
haier = Washer()
haier.print_info()
-
带参数: class Washer():
def __init__(self, width, height):
self.width = width
self.height = height
def print_info(self):
print(f'洗衣机的宽度:{self.width}')
print(f'洗衣机的宽度:{self.width}')
haier1 = Washer(100,200)
haier1.print_info()
注意︰
__init__() 方法,在创建一个对象时默认被调用,不需要手动调用__init__(self) 中的 self 参数,不需要开发者传递,python解释器会自动把当前的对象引用传递过去。
2.3.2 str()
当使用 print 输出对象的时候,默认打印对象的内存地址。如果类定义了__str__ 方法,那么就会打印从在这个方法中 return 的数据。
示例:
class Washer():
def __str__(self):
return '洗衣机类的解释说明'
haier1 = Washer()
print(haier1)
2.3.3 del()
当删除对象时,解释器也会默认调用 __del__() 方法。
class Washer():
def __del__(self):
print('删除')
haier = Washer()
del haier
三、面向对象继承
3.1 旧式类与新式类
3.2 单继承
class A(object):
def __init__(self):
self.num = 1
def info_print(self):
print(self.num)
class B(A):
pass
result = B()
result.info_print()
print(result.num)
3.3 多继承
多继承:一个类同时继承了多个父类。
class A(object):
def __init__(self):
self.num_a = 1
def info_print(self):
print(f'A类:{self.num_a}')
class B(object):
def __init__(self):
self.num_a = 2
def info_print(self):
print(f'B类:{self.num_a}')
class C(A, B):
pass
cc = C()
print(cc.num_a)
cc.info_print()
注意:
- 如果一个类继承多个父类,默认使用第一个父类的同名属性和方法
- 单继承中,子类默认继承父类的所有属性和方法
- 多继承中,子类继承第一个父类的所有属性和所有类的方法
- 两种继承方式子类都可以重写父类属性和方法
如:
class A(object):
def __init__(self):
self.num_a = 1
self.num_aa = 2
def info_print(self):
print(f'A类:{self.num_a}')
class B(object):
def __init__(self):
self.num_a = 3
self.num_b = 4
def info_print(self):
print(f'B类:{self.num_a}')
def info_print2(self):
print(f'B类:{self.num_a}')
class C(A, B):
pass
cc = C()
print(vars(cc))
print(cc.num_a)
print(cc.num_aa)
cc.info_print()
cc.info_print2()
3.4 子类重写父类同名方法和属性
如果一个类继承多个父类,子类会调用子类同名属性和方法。
class A(object):
def __init__(self):
self.num = 1
def info_print(self):
print(f'A父类:{self.num}')
class B(object):
def __init__(self):
self.num = 2
def info_print(self):
print(f'B父类:{self.num}')
class C(A, B):
def __init__(self):
self.num = 33
def info_print(self):
print(f'子类:{self.num}')
cc = C()
print(cc.num)
cc.info_print()
print(C.__mro__)
3.5 子类调用父类同名方法和属性
class A(object):
def __init__(self):
self.num = 11
def info_print(self):
print(f'A父类:{self.num}')
class B(object):
def __init__(self):
self.num = 22
def info_print(self):
print(f'B父类:{self.num}')
class A(object):
def __init__(self):
self.num = 11
def info_print(self):
print(f'A父类:{self.num}')
class B(object):
def __init__(self):
self.num = 22
def info_print(self):
print(f'B父类:{self.num}')
class C(A, B):
def __init__(self):
self.num = 33
def info_print(self):
self.__init__()
print(f'子类:{self.num}')
def info_a_print(self):
A.__init__(self)
A.info_print(self)
def info_b_print(self):
B.__init__(self)
B.info_print(self)
cc = C()
print(cc.num)
cc.info_a_print()
cc.info_b_print()
cc.info_print()
3.6 多层继承
class A(object):
def __init__(self):
self.num = 11
def info_print(self):
print(f'A父类:{self.num}')
class B(object):
def __init__(self):
self.num = 22
def info_print(self):
print(f'B父类:{self.num}')
class C(A, B):
def __init__(self):
self.num = 33
def info_print(self):
self.__init__()
print(f'子类:{self.num}')
def info_a_print(self):
A.__init__(self)
A.info_print(self)
def info_b_print(self):
B.__init__(self)
B.info_print(self)
class D(C):
pass
dd = D()
dd.info_print()
dd.info_a_print()
dd.info_b_print()
3.7 super()调用父类方法
使用super() 可以自动查找父类。调用顺序遵循 __mro__ 类属性的顺序。比较适合单继承使用。 super() 包括 有参数 和 无参数 两种方式。
-
B类继承A类,C类继承B类,一次性调用父类的同名属性和方法,代码如下:
class A(object):
def __init__(self):
self.num = 11
def info_print(self):
print(f'A父类:{self.num}')
class B(A):
def __init__(self):
self.num = 22
def info_print(self):
print(f'B父类:{self.num}')
super().__init__()
super().info_print()
class C(B):
def __init__(self):
self.num = 33
def info_print(self):
self.__init__()
print(f'子类:{self.num}')
def info_a_b_info(self):
super().__init__()
super().info_print()
cc = C()
cc.info_a_b_info()
-
C 类同时继承 A 类与 B 类,代码如下:
class A(object):
def __init__(self):
self.num = 11
def info_print(self):
print(f'A父类:{self.num}')
class B(object):
def __init__(self):
self.num = 22
def info_print(self):
print(f'B父类:{self.num}')
class C(A, B):
def __init__(self):
self.num = 33
def info_print(self):
self.__init__()
print(f'子类:{self.num}')
def info_a_b_info(self):
super().__init__()
super().info_print()
cc = C()
cc.info_a_b_info()
3.8 私有权限
??python中,实例属性和方法若设置为私有权限,则 该实例属性或实例方法不继承给子类,并且一般定义函数名 get_xx 用来获取私有属性,定义set_xx用来修改私有属性值。
设置私有权限的方法:在属性名和方法名前面加上两个下划线___。
class A(object):
def __init__(self):
self.num = 11
self.__num = 22
def info_print(self):
print(f'A父类:{self.num}')
self.__info_print()
def __info_print(self):
print(f'私有方法:{self.__num}')
def get_num(self):
return self.__num
def set_num(self):
self.__num = 500
class B(A):
pass
bb = B()
bb.info_print()
print(bb.get_num())
bb.set_num()
print(bb.get_num())
三、面向对象其他
3.1 面向对象三大特征
- 封装:将属性和方法书写到类的里面的操作即为封装,并且封装可以为属性和方法添加私有权限。
- 继承:
- 单继承中,子类默认继承父类的所有属性和方法
- 多继承中,子类继承第一个父类的所有属性和所有类的方法
- 两种继承方式子类都可以重写父类属性和方法。
- 多态:传入不同的对象,产生不同的结果。
3.2 多态
定义︰ 多态是一种使用对象的方式,子类重写父类方法,调用不同子类对象的相同父类方法,可以产生不同的结果。
class Dog(object):
def work(self):
print('指哪打哪...')
class ArmyDog(Dog):
def work(self):
print('追击敌人...')
class DrugDog(Dog):
def work(self):
print('追查毒品...')
class Person(object):
def work_with_dog(self, dog):
dog.work()
ad = ArmyDog()
dd = DrugDog()
person = Person()
person.work_with_dog(ad)
person.work_with_dog(dd)
3.3 类属性和实例属性
3.3.1 类属性
- 设置和访问类属性
类属性就是类对象所拥有的属性,它被该类的所有实例对象 所共有。类属性可以使用类对象或实例对象访问。 - 修改类属性
类属性只能通过类对象修改,不能通过实例对象修改。 如果通过实例对象修改类属性,表示的是创建了一个实例属性。
示例:
class Dog(object):
tooth = 10
dog = Dog()
print(Dog.tooth)
print(dog.tooth)
Dog.tooth = 20
print(Dog.tooth)
dog.tooth = 40
print(Dog.tooth)
print(dog.tooth)
3.3.2 实例属性
示例属性,也叫对象属性。其可通过类内类外都可修改和获取。
示例:
class Washer():
def __init__(self):
self.height = 300
def print_info(self):
print(f'洗衣机的宽度:{self.width}')
haier = Washer()
haier.width = 400
print(f'洗衣机的宽度:{haier.width}')
haier.print_info()
注意: 实例属性要求每个对象为其 单独开辟一份内存空间来记录数据,而类属性为全类所共有,仅占用一份内存,更加节省内存空间。
3.4 类方法和静态方法
3.4.1 类方法
类方法特点:
示例:
class Dog(object):
__tooth = 10
@classmethod
def get_tooth(cls):
return cls.__tooth
dog = Dog()
result = dog.get_tooth()
print(result)
使用场景:
- 类方法一般和类属性配合使用。
- 当方法中需要使用 类对象(如:访问私有类属性等)时,定义类方法
3.4.2 静态方法
静态方法特点:
- 需要通过装饰器
@staticmethod 来进行修饰,静态方法既不需要传递类对象也不需要传递实例对象(形参没有self/cls) - 静态方法也能够通过 实例对象 和 类对象 去访问。
示例:
class Dog(object):
@staticmethod
def info_print():
print('这是静态方法')
dog = Dog()
dog.info_print()
Dog.info_print()
使用场景:
- 当方法中 既不需要使用实例对象(如:实例对象,实例属性),也不需要使用类对象(如:类属性、类方法、创建实例等)时,定义静态方法。
- 取消不需要的参数传递,有利于减少不必要的内存占用和性能消耗
四、异常
4.1 异常语法
-
一般写法 try:
可能发生错误的代码
except :
如果出现异常执行的代码
-
捕获指定异常 try:
可能发生错误的代码
except 异常类型:
如果捕获到该异常类型执行的代码
-
捕获多个指定异常 try:
可能发生错误的代码
except (异常类型1, 异常类型2, ...):
如果捕获到该异常类型执行的代码
-
捕获异常描述信息 try:
可能发生错误的代码
except 异常类型 as result:
print(result)
如果捕获到该异常类型执行的代码
-
捕获所有异常 ★ try:
可能发生错误的代码
except Exception:
如果捕获到该异常类型执行的代码
Exception是所有程序异常类的父类。 -
异常的 else try:
可能发生错误的代码
except 异常类型:
如果捕获到该异常类型执行的代码
else:
print('else是没有异常的时候执行的代码')
-
异常的 finally ★ try:
可能发生错误的代码
except 异常类型:
如果捕获到该异常类型执行的代码
else:
print('else是没有异常的时候执行的代码')
finally:
print('finally是有无异常都要执行的代码')
示例:
try:
f = open('test.txt', 'r')
except Exception :
f = open('test.txt', 'w')
else:
print('没有异常执行的代码')
finally:
f.close()
4.2 自定义异常
Python中,抛出自定义异常的语法为raise异常类对象。
示例:密码长度不足,则报异常(用户输入密码,如果输入的长度不足5位,则报错,即抛出自定义异常,并捕获该异常)。
class ShortInputError(Exception):
def __init__(self, length, min_len):
self.length = length
self.min_len = min_len
def __str__(self):
return f'你输入的长度是{self.length},不能少于{self.min_len}个字符'
def main():
try:
con = input('请输入密码:')
if len(con) < 5:
raise ShortInputError(len(con), 5)
except Exception as result:
print(result)
else:
print('密码已经输入完成')
main()
运行结果:
五、模块和包
python 模块是一个 python 文件,导入模块就是导入 pthon 文件。
5.1 导入模块
5.1.1 导入方式
import 模块名 from 模块名 import 功能1, 功能2, ... from 模块名 import *
示例:
import math
print(math.sqrt(3))
from math import sqrt
print(sqrt(9))
from math import *
print(sqrt(9))
5.1.2 定义别名
5.2 自定义模块
自定义模块 my_module.py:
def testA(a, b):
print(a + b)
if __name__ == '__main__':
testA(4, 4)
调用自定义模块:
import my_module
my_module.testA(2,3)
注意: 如果使用 from .. import .. 或 from .. import * 导入多个模块的时候,且模块内有同名功能。当调用这个同名功能的时候,调用到的是 后导入的模块的功能。
5.3 模块定位顺序
当导入一个模块,Python解析器对模块位置的搜索顺序是:
- 当前目录
- 如果不在当前目录,Python则搜索在shell变量PYTHONPATH下的每个目录。
- 如果都找不到,Python会察看默认路径。UNIX下,默认路径一般为/usr/local/lib/pythonl/
示例:
import random
num = random.randint(0, 5)
print(num)
拓展:
import time
print(time)
time = 1
print(time)
5.4 all 列表
如果模块文件中有 __all__ 变量,当使用 from xxx import * 导入时,只能导入这个列表中的元素。
自定义模块 my_modeule2.py:
__all__ = ['testA']
def testA():
print('testA函数')
def testB():
print('testB函数')
测试代码:
from my_module_2 import *
testA()
5.5 包
如果一个文件夹中含有__init__.py 文件,那么这个文件夹就称之为包。
包是模块的一个集合,将模块以文件夹的方式集中在一起。
5.5.1 导包方式
-
方式一: import 包名.模块名
调用方式:包名.模块名.目标
-
方式二: from 包名 import *
调用方式:模块名.目标
注意: 必须在 __init__.py 文件中添加 __all__ = [] ,控制允许导入的模块列表。
包结构:
方式一示例:
my_module1.py:
def info_print1():
print('my_module1')
测试代码:
import mypackage.my_module1
mypackage.my_module1.info_print1()
方式二示例:
my_module2.py:
def info_print2():
print('my_module2')
__init__.py:
__all__ = ['my_module1']
测试代码:
from mypackage import *
my_module1.info_print1()
my_module1.info_print2()
5.5.2 注意事项
方式一在不引入 all 列表的时候方法也可以正常使用,但 pycharm 不会提示。
|