模块
python内置模块之re模块
在python中要想使用正则必须借助于模块 re就是其中之一
"""基本操作方法"""
import re
# re.findall('正则表达式','待匹配的文本') # 根据正则匹配除所有符合条件的数据
res = re.findall('a', 'eva jason jackson')
print(res) # 结果是一个列表(要么有元素 要么空列表)
# res = re.search('正则表达式','待匹配的文本') # 根据正则匹配到一个符合条件的就结束
res = re.search('a','eva jason jackson')
print(res) # 结果对象
print(res.group()) # 正在的结果
if res:
print(res.group())
else:
print('不好意思,没有找到')
"""如果没有符合条件的数据 那么search返回None 并且使用group会直接报错"""
res = re.match('a','abac') # 根据正则从头开始匹配(文本内容必须在开头匹配上)
print(res)
print(res.group())
if res:
print(res.group())
else:
print('不好意思没有找到')
"""如果没有符合条件的数据 那么match返回None 并且使用group会直接报错"""
re模块其他方法
import re
# 先按'a'分割得到''和'bcd',再对''和'bcd'分别按b分割
res = re.split('[ab]','abcd')
print(res) # ['', '', 'cd']
# 类似字符串类型的replace方法
res = re.sub('\d','H','eva3jason4yuan4',1) # 替换正则匹配到的内容
res = re.sub('\d','H','eva3jason4yuan4') # 不写替换所有
print(res) # evaHjasonHyuanH
"""返回元组 并提示替换了几处"""
res = re.subn('\d','H','eva3jason4yuan4',1)
print(res)
res = re.subn('\d','H','eva3jason4yuan4',)
print(res)
import time
"""常用"""
regexp_obj = re.compile('\d+')
res = regexp_obj.search('123dsawd1234123sqde123123rsd13214r1ed123')
res1 = regexp_obj.match('ws12s1e12e3xdasd123e124r1')
res2 = regexp_obj.findall('12321dqqqdxase123dqe123de123dqe12')
print(res,res1,res2)
"""常用"""
res = re.finditer('\d+','qdwqdq65d12gfufgu7q6weh12gf3u67t5dqw1')
print([i.group() for i in res])
res = re.search('^[1-9](\d{14})(\d{2}[0-9x])?$','372861354972637834')
print(res)
print(res.group())
print(res.group(1))
print(res.group(2))
'''常用'''
findall针对分组优先展示 无名分组
res = re.findall("^[1-9]\d{14}(\d{2}[0-9x])?$",'372861354972637834')
print(res)
# 取消分组优先展示 无名分组
res1 = re.findall("^[1-9]\d{14}(?:\d{2}[0-9x])?$",'372861354972637834')
print(res1)
有名分组
res = re.search('^[1-9](?P<xxx>\d{14})(?P<ooo>\d{2}[0-9x])?$',"372861354972637834")
print(res)
print(res.group()) # 372861354972637834
print(res.group(1)) # 72861354972637
print(res.group('xxx')) # 72861354972637
print(res.group('ooo')) # 023
正则实战案例
import re
# 读取待匹配的数据
with open(r'a.txt','r',encoding='utf8') as f:
data = f.read()
# 利用正则匹配数据
# 分公司名称
res = re.findall('<h2>(.*?)</h2>',data)
print(res)
# 分公司地址
address_list = re.findall("<p class='mapIco'>(.*?)</p>",data)
print(address_list)
# 分公司邮箱
email_list = re.findall("<p class='mailIco'>(.*?)</p>", data)
print(email_list)
# 分公司电话
phone_list = re.findall("<p class='telIco'>(.*?)</p>",data)
print(phone_list)
res = zip(res,address_list,email_list,phone_list)
for data_tuple in res:
print("""
公司名称:%s
公司地址:%s
公司邮箱:%s
公司电话:%s
""" % (data_tuple[0],data_tuple[1],data_tuple[2],data_tuple[3])
collections模块
该模块内部提供了一些高阶的数据类型
# 1.namedtuple(具名元组)
from collections import namedtuple
"""
namedtuple('名称',[名字1,名字2,...])
namedtuple('名称','名字1 名字2 ...')
"""
point = namedtuple('坐标',['x','y'])
res = point(11,22)
print(res)
print(res.x)
print(res.y)
point = namedtuple('坐标','x y z')
res = point(11,22,33)
print(res)
print(res.x)
print(res.y)
print(res.z)
card = namedtuple('扑克','花色 点数')
card1 = card('♠','A')
card2 = card('♥','K')
print(card1)
print(card1.点数)
print(card1.花色)
# 2.队列模块
import queue # 内置队列模块:FIFO
# 初始化队列
q = queue.Queue()
# 往列队中添加元素
q.put('first')
q.put('second')
q.put('third')
# 从队列中获取元素
print(q.get())
print(q.get())
print(q.get())
print(q.get()) # 值没了会原地等待
# 3.双端队列
from collections import deque
q = deque([11,22,33])
q.append(44) # 从右边添加
q.appendleft(55) # 从左边添加
print(q.pop()) # 从左边取值
print(q.popleft()) # 从右边取值
# 4.有序字典
normal_dict = dict([('name','jason'),('pwd',123),('hobby','study')])
print(normal_dict)
from collections import OrderedDict
order_dict = OrderedDict([('name','jason'),('pwd',123),('hobby','study')])
print(order_dict)
OrderedDict([('name','jason'),('pwd',123),('hobby','study')])
order_dict['xxx'] = 111
print(order_dict)
OrderedDict([('name','jason'),('pwd',123),('hobby','study')])
normal_dict['yyy'] = 222
print(normal_dict)
# 5.默认值字典
from collections import defaultdict
values = [11,22,33,44,55,66,77,88,99,90]
my_dict = defaultdict(list)
for value in values:
if value>60:
my_dict['K1'].append(value)
else:
my_dict['k2'].append(value)
print(my_dict)
# 6.计数器
res = 'cbacbabcbacbacbabcbacbacb'
# 统计字符串中每个元素出现的次数
new_dict = {}
for i in res:
if i not in new_dict:
new_dict[i] = 1
else:
new_dict[i] +=1
print(new_dict)
from collections import Counter # 计数器
ret = Counter(res)
print(ret)
time模块
"""
时间的三种表现形式
1.时间戳(秒数)
2.结构化时间(一般是给机器看的)
3.格式化时间(一般是给人看的)
三种时间是可以相互转换的!!!
"""
time.sleep() # 原地阻塞指定的秒数
time.time() # 获取时间戳时间
import time
# 格式化时间
print(time.strftime('%Y-%m-%d')) # 2021-11-25
print(time.strftime('%Y-%m-%d %H:%M:%S')) # 2021-11-25 17:21:50
print(time.strftime('%Y-%m-%d %X')) # 2021-11-25 17:22:34
"""
更多时间相关符号 保存到容易查找到的位置即可
"""
print(time.localtime())
time.struct_time
tm_year=2021
tm_mon=11
tm_mday=25
tm_hour=17
tm_min=24
tm_sec=6
tm_wday=3
tm_yday=329
tm_isdst=0
print(time.time())
print(time.gmtime(11111111111))
print(time.localtime())
datetime模块
datetime
print(datetime.date.today()) # 2021-11-25
print(datetime.datetime.today()) # 2021-11-25 18:51:20.140301
"""date年月日 datetime年月日时分秒 time时分秒(MySQL django后期可以)"""
res = datetime.datetime.today()
print(res.year) # 2021
print(res.month) # 11
print(res.day) # 25
print(res.weekday()) # 获取星期(weekday星期是0-6) 0表示周一
print(res.isoweekday()) # 获取星期(weekday星期是1-7) 1表示周一
"""时间差(timedelta)"""
ctime = datetime.datetime.today()
time_tel = datetime.timedelta(days=3)
print(ctime) # 2021-11-25 18:56:46.571683
print(ctime - time_tel) # 2021-11-22 18:56:46.571683
print(ctime + time_tel) # 2021-11-28 18:56:46.571683
"""
日期对象 = 日期对象 +/- timedelta对象
timedelta对象 = 日期对象 +/- 日期对象
"""
ret = ctime + time_tel
print(ret - ctime) # 3 days, 0:00:00
print(ctime - ret) # -3 days, 0:00:00
# 小练习 计算举例今年离过生日还有多少天
birthday = datetime.date(2022, 11, 11)
now_date = datetime.date.today()
days = birthday - now_date
print('距离生日还有{}天'.format(days))
# UTC时间与我们东八区时间差 八个小时
print(datetime.datetime.now()) # 2021-11-25 19:03:22.680976
print(datetime.datetime.utcnow()) # 2021-11-25 11:03:22.680976
random模块
import random
print(random.random()) # 随机产生一个0-1之间的小数
print(random.randint(1, 6)) # 随机产生一个1-6之间的整数 掷骰子
print(random.uniform(1, 6)) # 随机产生一个1-6之间的小数
print(random.choice(['学习', '锻炼', '努力', '坚持', '勤奋'])) # 随机抽取一个
print(random.sample(['学习', '锻炼', '努力', '坚持', '勤奋'], 3)) # 随机抽取指定样本量
l = [2, 3, 4, 5, 6, 7, 8, 9, 10, 'J', 'Q', 'K', 'A']
random.shuffle(l) # 随机打乱容器类型中的诸多元素
print(l)
# 搜狗公司笔试题
# 随机验证码可以是由数字 小写字母 大写 字母 任意组合
# 编写能够产生5位数的随机验证码
'''ps:五位 每位都可能是三种情况之一'''
def get_code(n):
code = ''
for i in range(n):
num = str(random.randint(0,9))
upper = chr(random.randint(65,90))
lower = chr(random.randint(97,122))
temp = random.choice([num,upper,lower])
code += temp
return code
print(get_code(6),get_code(4),get_code(9))
os模块
# 与操作系统打交道
import os
# 1.创建单级目录(文件夹)
os.mkdir('学习资料')
# os.mkdir(r'学习资料\c老师视频作品') 报错
# 2.创造多级目录(文件夹)
os.makedirs(r'c老师视频合集\s老师2021合集\1080p高清')
# 3.删除空目录(文件夹)
os.rmdir(r'学习资料')
os.removedirs(r'c老师视频合集')
# 4.获取当前文件所在的路径(可以嵌套)
BASE_DIR = os.path.dirname(__file__)
# 5.路径拼接(******)能够自动识别不同的操作系统分隔符问题
movie_dir = os.path.join(BASE_DIR,'老师教学视频')
# 6.列举出指定路径下的文件名称(任意类型文件)
data_movin_list = os.listdir('老师教学视频')
while True:
for i,j in enumerate(data_movin_list):
print(i+1,j)
choice = input('请选择你想要看的文件编号>>>:').strip()
if choice.isdigit():
choice = int(choice)
if choice in range(len(data_movin_list)+1):
# 获取编号对应的文件名称
file_name = data_movin_list[choice - 1]
# 拼接文件的完整路径(*******)
file_path = os.path.join(movie_dir,file_name) # 专门用于路径拼接 并且能够自动识别当前操作系统的路径分隔符
# 利用文件操作读写文件
with open(file_path,'r',encoding='utf8') as f:
print(f.read())
# 7.删除一个文件
os.remove('a.txt')
# 8.修改文件
os.rename('老文件名','新文件名')
# 9.获取当前工作路径
print(os.getcwd())
# 10.切换路径
os.chdir('D:/')
with open(r'a.txt','wb') as f:
pass
# 11.判断路径是否存在
print(os.path.exists('a.txt')) # False
print(os.path.exists('老师教学视频')) # True
print(os.path.exists('re模块.py')) # True
# 12.判断当前路径是否是文件
print(os.path.isdir('re模块.py')) # True
print(os.path.isdir('老师教学视频')) # False
# 13.判断当前路径是否是文件夹
print(os.path.isdir('re模块.py')) # False
print(os.path.isdir('老师教学视频')) # True
# 14.获取文件大小(字节数)
print(os.path.ge
sys模块
# 主要和python解释器打交道
import sys
print(sys.path) # 文件路径
print(sys.version) # python解释器版本
print(sys.platform) # 操作系统
print(sys.argv) # 获取当前执行文件的绝对路径
try:
username = sys.argv[1]
password = sys.argv[2]
if username == 'jason' and password == '123':
print('正常执行文件内容')
else:
print('用户名和密码错误')
except Exception:
print('用户名和密码')
print('只能让你体验一下(游客模式)')
json模块
# josn格式数据:跨语言传输
import json
d = {'username':'jason','pwd':123}
# 1.将python其他数据转换成json格式字符串(序列化)
res = json.dumps(d)
print(res,type(res)) # {"username": "jason", "pwd": 123} <class 'str'>
# 2.将json格式字符串转换成当前语言对应的某个数据类型(反序列化)
res1 = json.loads(res)
print(res1,type(res1)) # {'username': 'jason', 'pwd': 123} <class 'dict'>
bytes_data = b'{"username":"jason","pwd":123}'
bytes_str = bytes_data.decode('utf8')
bytes_dict = json.loads(bytes_str)
print(bytes_dict,type(bytes_dict))
"""
暂且可以简单的理解为
序列化就是将其他数据类型转换成字符串过程
json.dumps()
反序列化就是将字符串转换成其他数据类型
json.loads()
"""
# 将字典d写入文件
with open(r'a.txt','w',encoding='utf8') as f:
f.write(str(d))
# 将字典取出来
with open(r'a.txt','r',encoding='utf8') as f:
data = f.read()
# print(dict(data)) # 报错
# 将字典d写入文件
with open(r'a.txt','w',encoding='utf8') as f:
res = json.dumps(d) # 序列化成json格式字符串
f.write(res)
# 将字典d取出来
with open(r'a.txt','r',encoding='utf8') as f:
data = f.read()
res1 = json.loads(data)
print(res1,type(res1))
d1 = {'username':'tony好帅哦 我好喜欢','pwd':123,'hobby':[11,22,33]}
print(json.dumps(d1,ensure_ascii=False))
"""
# 并不是所有的数据类型都支持序列化
json.JSONEncoder 查看支持的数据类型
"""
subprocess模块
import subprocess
"""
1.可以基于网络连接上一台计算机(soket模块)
2.让连接上的计算机执行我们需要执行的命令
3.将命令的结果返回
"""
res = subprocess.Popen('tasklist',
shell=True,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE
)
print('stdout',res.stdout.read().decode('gbk')) # 获取正确命令执行之后的结果
print('stderr',res.stderr.read().decode('gbk')) # 获取错误命令执行之后的结果
"""
windows电脑内部编码默认GBK
"""
hashlib模块
"""
加密:将明文数据通过一系列算法变成密文数据(目的是为了数据的安全)
加密算法:md系列 sha系列 base系列 hmac系列
"""
# 基本使用
import hashlib
# 1.先确定算法类型(md5普遍使用)
md5 = hashlib.md5()
# 2.将明文数据传递给md5算法(update只能接受bytes类型数据)
# md5.update('123'.encode('utf8'))
md5.update(b'123')
# 3.获取加密之后的密文数据(没有规则的一串随机字符串)
res = md5.hexdigest()
print(res)
"""
1.加密之后的密文数据是没有办法反解密成明文数据的
市面上所谓的破解 其实就是提前算出一系列明文对应的密文
之后比对密文再获取明文
"""
详细操作
# # 1.明文数据只要是相同的 那么无论如何传递加密结果肯定是一样的
import hashlib
# 1.1先确定算法类型(md5普遍使用)
md5 = hashlib.md5()
# 1.2将明文数据传递给md5算法(update只能接受bytes类型数据)
md5.update('123'.encode('utf8')) # 202cb962ac59075b964b07152d234b70
md5.update(b'hellojason123') # 4bd388611b31c548eedef14b16c9868c
md5.update(b'hello')
md5.update(b'jason')
md5.update(b'123') # 4bd388611b31c548eedef14b16c9868c
# 1.3获取加密之后的密文数据(没有规则的一串随机字符串)
res = md5.hexdigest()
print(res)
# 2.密文数据越长便是内部对应的算法越复杂 越难被正向破解
import hashlib
# 2.1先确定算法类型(md5普遍使用)
md5 = hashlib.sha256()
# 2.2将明文数据传递给md5算法(update只能接受bytes类型数据)
md5.update('123'.encode('utf8')) # 4f045b68547558ac48e5f870a407f87c
md5.update(b'hellojason123')
md5.update(b'hello')
md5.update(b'jason')
md5.update(b'123') # 6ec0f865ec4e9bcf9abc7f6c0e7268acfb2ba36f3e460fd592a214f7eacc92b0
# 2.3获取加密之后的密文数据(没有规则的一串随机字符串)
res = md5.hexdigest()
print(res)
"""
密文越长表示算法越复杂 对应的破解算法的难度越高
但是越复杂的算法所需要消耗的资源也就越多 密文越长基于网络发送需要占据的数据也就越大
具体使用什么算法取决于项目的要求 一般情况下md5足够了
"""
# 3.涉及到用户密码存储 其实都是密文 只有用户自己知道明文是什么
"""
1.内部程序员无法得知明文数据
2.数据泄露也无法得知明文数据
ATM购物车练习 用户密码采取的就是密文存取
"""
# 4.加盐处理
"""
在对明文数据做加密处理过程前添加一些干扰项
"""
import hashlib
# 4.1先确定算法类型(md5普遍使用)
md5 = hashlib.md5()
# 4.2将明文数据传递给md5算法(update只能接受bytes类型数据)
md5.update('123'.encode('utf8'))
# 加盐(干扰项)
md5.update('公司内部自己定义的盐'.encode('utf8'))
# 真实数据
md5.update(b'hellojason123')
# 3.获取加密之后的密文数据(没有规则的一串随机字符串)
res = md5.hexdigest()
print(res) # 4326fe1839d7c4186900eb2b356d4f8a
# 5.动态加盐
"""
在对明文数据做数据加密处理过程前添加一些变化的干扰项]
"""
import hashlib
# 5.1先确定算法类型(md5普遍使用)
md5 = hashlib.md5()
# 5.2将明文数据传递给md5算法(update只能接受bytes类型数据)
md5.update('123'.encode('utf8'))
# 加盐(干扰项)
md5.update('公司内部自己定义的盐'.encode('utf8'))
# 动态加盐(干扰项) 当前时间 用户名的部分 uuid(随机的字符串(永远不会重复))
import time
res1 = str(time.time())
md5.update(res1.encode('utf8'))
# 真实数据
md5.update(b'hellojason123')
# 3.获取加密之后的密文数据(没有规则的一串随机字符串)
res = md5.hexdigest()
print(res) # 703a43558d582335f904ba65007e0452
"""
在IT互联网领域 没有绝对的安全可言 只有更安全
原因在于互联网本质 就是通过网线(网卡)连接计算机
"""
# 6.校验文件一致性
"""
文件不是很大的情况下 可以将所有文件内部全部加密处理
但是如果文件特别大 全部加密处理相当的耗时耗资源 如何解决???
针对大文件可以使用切片读取的方式
"""
import hashlib
md5 = hashlib.md5()
with open(r'a.txt','rb') as f:
for line in f:
md5.update(line)
real_data = md5.hexdigest()
print(real_data)
import os
md5 = hashlib.md5()
# 读取文件总大小
res = os.path.getsize(r'a.txt')
with open(r'a.txt','rb') as f:
# 指定分片读取策略(读几段,每段几个字节)
f.seek(res//4)
line = f.read(8)
f.seek(res//2)
line1 = f.read(8)
md5.update(line)
md5.update(line1)
print(md5.hexdigest())
logging日志模块
是需要掌握的很少(会用即可)
import logging
# 日志有五个等级(从上往下重要程度递增)
logging.debug('debug级别') # 10
logging.info('info级别') # 20
logging.warning('warning级别') # 30
logging.error('error级别') # 40
logging.critical('critical级别') # 50
'''默认记录的级别在30及以上'''
# 简单使用
import logging
file_handler = logging.FileHandler(filename='x1.log', mode='a', encoding='utf-8',)
logging.basicConfig(
format='%(asctime)s - %(name)s - %(levelname)s -%(module)s: %(message)s',
datefmt='%Y-%m-%d %H:%M:%S %p',
handlers=[file_handler,],
level=logging.ERROR
)
logging.error('日志模块很好学 不要自己吓自己')
"""
1.如何控制日志输入的位置
想在文件和终端中同时打印
2.不同位置如何做到不同的日志格式
文件详细一些 终端简单一些
"""
# 详细介绍
import logging
# 1.logger对象:负责产生日志
logger = logging.getLogger('转账记录')
# 2.filter对象:负责过滤日志(直接忽略)
# 3.handler对象:负责日志产生的位置
hd1 = logging.FileHandler('a1.log',encoding='utf8') # 产生到文件的
hd2 = logging.FileHandler('a2.log',encoding='utf8') # 产生到文件的
hd3 = logging.StreamHandler() # 产生在终端的
# 4.formatter对象:负责日志的格式
fm1 = logging.Formatter(fmt='%(asctime)s - %(name)s - %(levelname)s -%(module)s: %(message)s',
datefmt='%Y-%m-%d %H:%M:%S %p',
)
fm2 = logging.Formatter(
fmt='%(asctime)s - %(name)s %(message)s',
datefmt='%Y-%m-%d',
)
# 5.绑定handler对象
logger.addHandler(hd1)
logger.addHandler(hd2)
logger.addHandler(hd3)
# 6.绑定formatter对象
hd1.setFormatter(fm1)
hd2.setFormatter(fm2)
hd3.setFormatter(fm1)
# 7.设置日志等级
logger.setLevel(30)
# 8.记录日志
logger.debug('好烦啊,日志功能')
# 配置字典
# 核心就在于CV
import logging
import logging.config
standard_format = '[%(asctime)s][%(threadName)s:%(thread)d][task_id:%(name)s][%(filename)s:%(lineno)d]' \
'[%(levelname)s][%(message)s]' #其中name为getlogger指定的名字
simple_format = '[%(levelname)s][%(asctime)s][%(filename)s:%(lineno)d]%(message)s'
logfile_path = 'a3.log'
# log配置字典
LOGGING_DIC = {
'version': 1,
'disable_existing_loggers': False,
'formatters': {
'standard': {
'format': standard_format
},
'simple': {
'format': simple_format
},
},
'filters': {}, # 过滤日志
'handlers': {
#打印到终端的日志
'console': {
'level': 'DEBUG',
'class': 'logging.StreamHandler', # 打印到屏幕
'formatter': 'simple'
},
#打印到文件的日志,收集info及以上的日志
'default': {
'level': 'DEBUG',
'class': 'logging.handlers.RotatingFileHandler', # 保存到文件
'formatter': 'standard',
'filename': logfile_path, # 日志文件
'maxBytes': 1024*1024*5, # 日志大小 5M
'backupCount': 5,
'encoding': 'utf-8', # 日志文件的编码,再也不用担心中文log乱码了
},
},
'loggers': {
#logging.getLogger(__name__)拿到的logger配置 空字符串作为键 能够兼容所有的日志
'': {
'handlers': ['default', 'console'], # 这里把上面定义的两个handler都加上,即log数据既写入文件又打印到屏幕
'level': 'DEBUG',
'propagate': True, # 向上(更高level的logger)传递
}, # 当键不存在的情况下 (key设为空字符串)默认都会使用该k:v配置
},
}
# 使用配置字典
logging.config.dictConfig(LOGGING_DIC) # 自动加载字典中的配置
logger1 = logging.getLogger('xxx')
logger1.debug('好好的 不要浮躁 努力就有收获')
第三方模块
# 并不是赔ython自带的 需要基于网络下载!!!
'''pip所在的路径添加环境变量'''
"""
下载第三方模块的方式
方式1:命令行借助于pip工具
pip3 install 模块名 # 不知道版本默认是最新版
pip3 install 模块名==版本号 # 指定版本下载
pip3 install 模块名 -! 仓库地址 # 临时切换
'''命令行形式永久修改需要修改python解释器源文件'''
方式2:pycharm快捷方式
settings
project
project interprter
双击或者加好
点击右下方mannge管理添加源地址即可
"""
# 下载完第三方模块之后 还是使用import或者from import句式导入使用
"""
pip命令默认下载的渠道是国外的python官网(有时候非常慢)
我们可以切换下载的源(仓库)
(1)阿里云 http://mirrors.aliyun.com/pypi/simple/
(2)豆瓣 http://pypi.douban.com/simple/
(3)清华大学 https://pypi.tuna.tsinghua.edu.cn/simple/
(4)中国科学技术大学 http://pypi.mirrors.ustc.edu.cn/simple/
(5)华中科技大学http://pypi.hustunique.com/
pip3 install openpyxl -i http://mirrors.aliyun.com/pypi/simple/
"""
"""
下载第三方模块可能报错的情况及解决措施
1.报错的提示信息中含有关键字timeout
原因:网络不稳定
措施:再次尝试 或者切换更加稳定的网络
2.找不到pip命令
环境变量问题
3.没有任何的关键字 不同的模块报不同的错
原因:模块需要特定的计算机环境
措施:拷贝报错信息 打开浏览器 百度搜索即可
pip下载某个模块报错错误信息
"""
未完待续![]()