3 python进阶篇

文章目录

  • 面向对象
    • 类属性和类方法
      • 类属性
      • 类方法
      • 静态方法
  • 单例模式
    • __new__ 方法
    • 类实现单例模式
  • 异常 、模块和包
    • 异常
      • 自定义异常
    • 模块和包
      • 模块的搜索顺序
      • 包的init文件
      • 发布模块(了解)
  • 文件
    • seek
    • 文件/目录的常用管理操作
    • eval函数
  • 补充性知识
    • 位运算小技巧

参考我的别的帖子,此篇博客只进行补充:
链接

面向对象

链接

is和==的差别:

  1. is指的是两个变量引用的对象是否是同一个
  2. ==指的是引用变量的值是否是同一个

而关于实现多态实际上就是通过实现重写功能来实现的多态功能。

在这里插入图片描述
在这里插入图片描述
补充:

类属性和类方法

  1. 创建出来的对象叫做类的实例
  2. 创建对象的动作叫做实例化
  3. 对象的属性叫做实例属性
  4. 对象调用的方法叫做实例方法

在程序执行时:

  1. 对象各自拥有自己的 实例属性
  2. 调用对象方法,可以通过 self.
    – 访问自己的属性
    – 调用自己的方法

结论

  1. 每一个对象 都有自己 独立的内存空间,保存各自不同的属性
  2. 多个对象的方法,在内存中只有一份,在调用方法时,需要把对象的引用传递到方法内部(知道为啥要搞 self 了吧)

在这里插入图片描述

类属性

定义:
类属性就是给类对象中定义的属性,通常用来记录与这个类相关的特征,类属性不会用于记录具体对象的特征。

例子:

class Tool(object):
	# 使用赋值语句,定义类属性,记录创建工具对象的总数
	count = 0
	def __init__(self, name):
		self.name = name
		# 针对类属性做一个计数+1
		Tool.count += 1

# 创建工具对象
tool1 = Tool("斧头")
tool2 = Tool("榔头")
tool3 = Tool("铁锹")
# 知道使用 Tool 类到底创建了多少个对象?
print("现在创建了 %d 个工具" % Tool.count)

对于count来说就是对应的类属性,而对于count属性来说对应的就是对应的对象属性或者叫实例属性。

而在访问类属性的时候有两种方法:

  1. 类名.类属性
  2. 对象.类属性 (不推荐) 别用

值得关注的是:
如果使用对象.类属性 = 值 赋值语句,只会给对象添加一个属性,而不会影响到 类属性的值

# 拼接上面的代码
tool3.count = 99
print("工具对象总数 %d" % tool3.count)
print("===> %d" % Tool.count)
# 运行完会发现实际上tool3.count变成tool3的一个新定义的属性了,不是原本的tool的属性。

类方法

类方法就是针对类对象定义的方法
语法如下:

@classmethod
def 类方法名(cls):
pass

  1. 类方法需要用修饰器 @classmethod 来标识,告诉解释器这是一个类方法
  2. 类方法的第一个参数应该是cls
    – 由哪一个类调用的方法,方法内的cls就是哪一个类的引用
    – 这个参数和实例方法的第一个参数是和self类似
  3. 通过类名. 调用类方法,调用方法时,不需要传递 cls 参数
  4. 在方法内部
    – 可以通过 cls. 访问类的属性
    – 也可以通过 cls. 调用其他的类方法

例子:

class Tool(object):
	# 使用赋值语句定义类属性,记录所有工具对象的数量
	count = 0
	# 定义类方法
	@classmethod
	def show_tool_count(cls):
		print("工具对象的数量 %d" % cls.count)
		
	def __init__(self, name):
		self.name = name
		# 让类属性的值+1
		Tool.count += 1

# 创建工具对象
tool1 = Tool("斧头")
tool2 = Tool("榔头")
# 调用类方法
Tool.show_tool_count()

静态方法

在开发时,如果需要在类中封装一个方法,这个方法:既不需要访问实例属性 或者调用实例方法,也不需要访问类属性或者调用类方法,这个时候,可以把这个方法封装成一个静态方法,例如打印一些帮助。
语法:

@staticmethod
def 静态方法名():
	pass

例子:

class Dog(object):
	@staticmethod
	def run():
	# 不访问实例属性/类属性
	print("小狗要跑...")

# 通过类名.调用静态方法 - 不需要创建对象
Dog.run()

总结:

  1. 实例方法 —— 方法内部需要访问实例属性
    – 实例方法 内部可以使用类名. 访问类属性
  2. 类方法 —— 方法内部只需要访问类属性
  3. 静态方法 —— 方法内部,不需要访问实例属性和类属性

在这里插入图片描述

单例模式

链接

new 方法

  1. 使用类名()创建对象时,Python的解释器首先会调用__new__ 方法为对象
    分配空间
  2. new 是一个由object基类提供的内置的静态方法,主要作用有两个:
    – 1) 在内存中为对象分配空间
    – 2) 返回对象的引用
  3. Python的解释器获得对象的引用后,将引用作为第一个参数,传递给
    init 方法
    重写 new 方法 的代码非常固定!
  4. 重写 new 方法 一定要 return super().new(cls)
  5. 否则Python的解释器得不到分配了空间的对象引用,就不会调用对象的初
    始化方法
  6. 注意:new 是一个静态方法,在调用时需要主动传递cls参数

类实现单例模式

class Single(object):
    _instance = None
    def __new__(cls, *args, **kw):
        if cls._instance is None:
            cls._instance = object.__new__(cls, *args, **kw)
        return cls._instance
    def __init__(self):
        pass

single1 = Single()
single2 = Single()
print(id(single1) == id(single2))

补充,但是个人建议使用导包法,或者装饰器方法进行实现,不然实际上这个单例模式会进行覆盖,实际上就是每次进行一次类型的创建,就相当于对于这个地址的相对创建,其中的所有的内容都会进行一次初始化。

在这里插入图片描述

异常 、模块和包

链接

异常

使用异常可以提高程序鲁棒性(稳定性)

比较好用的模板补充:
Python 如何捕获异常发生的文件和具体行数

try:
	print(a)
except Exception as e:
	print(e)
	print(e.__traceback__.tb_frame.f_globals["__file__"]) # 发生异常所在的文件
	print(e.__traceback__.tb_lineno) # 发生异常所在的行数

自定义异常

示例:
这个一般用于只应用一俩次的异常

def input_password():
	# 1. 提示用户输入密码
	pwd = input("请输入密码:")
	# 2. 判断密码长度 >= 8,返回用户输入的密码
	if len(pwd) >= 8:
		return pwd
	# 3. 如果 < 8 主动抛出异常
	print("主动抛出异常")
	# 1> 创建异常对象 - 可以使用错误信息字符串作为参数
	ex = Exception("密码长度不够")
	# 2> 主动抛出异常
	raise ex

# 提示用户输入密码
try:
	print(input_password())
except Exception as result:
	print(result)

方法二:自定义异常类,不去使用exception:

创建新的异常很简单——定义新的类,让它继承自 Exception (或者是任何一个已存在的异常类型)。 例如,如果你编写网络相关的程序,你可能会定义一些类似如下的异常:

class NetworkError(Exception):
    pass

class HostnameError(NetworkError):
    pass

class TimeoutError(NetworkError):
    pass

class ProtocolError(NetworkError):
    pass

然后用户就可以像通常那样使用这些异常了,例如:

try:
    msg = s.recv()
except TimeoutError as e:
    ...
except ProtocolError as e:
    ...

在这里插入图片描述

模块和包

补充:

模块的搜索顺序

Python 的解释器在 导入模块 时,会:

  1. 搜索当前目录指定模块名的文件,如果有就直接导入
  2. 如果没有,再搜索系统目录
    在开发时,给文件起名,不要和系统的模块文件重名,还有导入的函数也会满足后导入的覆盖前导入的,所以这边对于from ··· import *是很不推荐的,原因就是后面迟早会发生名字覆盖的问题
  3. Python 中每一个模块都有一个内置属性 file 可以 查看模块 的 完整路径
    示例
import random
print(random.__file__)
rand = random.randint(0, 10)
print(rand)

注意:如果当前目录下,存在一个 random.py 的文件,程序就无法正常执行了!这个时候,Python 的解释器会 加载当前目录 下的 random.py 而不会加载 系统的 random 模块。

包的init文件

init.py
• 要在外界使用包中的模块,需要在 init.py 中指定对外界提供的模块列

# 从当前目录导入模块列表
# .就是当前文件夹,这里要写的就是文件的相对路径
from . import send_message
from . import receive_messag

发布模块(了解)

如果希望自己开发的模块,分享给其他人,可以按照以下步骤操作:

  1. 制作发布压缩包步骤
    新建发布模块的文件夹,然后将 setup.py 放入(setup.py 和要发布的模块在同级目录)
    (1) 创建 setup.py
from distutils.core import setup
setup(name="", # 包名
	version="", # 版本
	description="", # 描述信息
	long_description="", # 完整描述信息
	author="", # 作者
	author_email="", # 作者邮箱
	url="", # 主页
	py_modules=["wd_message.send_message",
	"wd_message.receive_message"])

(2) 构建模块(ubuntu 下,windows 都可以)

$ python3 setup.py build

(3) 生成发布压缩包

$ python3 setup.py sdist

注意:要制作哪个版本的模块,就使用哪个版本的解释器执行!
2. 安装模块

$ tar xf wd_message-1.0.tar.gz
$ sudo python3 setup.py install

卸载模块
直接从安装目录下,把安装模块的 目录 删除就可以

$ cd /usr/local/lib/python3.6/dist-packages/
$ sudo rm -r wd_message*

在这里插入图片描述

文件

链接

访问模式扩展:

访问方式说明
r以只读方式打开文件。文件的指针将会放在文件的开头,这是默认模式。如果文件不存在,抛出异常
w以只写方式打开文件。如果文件存在会被覆盖。如果文件不存在,创建新文件
a以追加方式打开文件。如果该文件已存在,文件指针将会放在文件的结尾。如果文件不存在,创建新文件进行写入r+ 以读写方式打开文件。文件的指针将会放在文件的开头。如果文件不存在,抛出异常.
w+以读写方式打开文件。如果文件存在会被覆盖。如果文件不存在,创建新文件
a+以读写方式打开文件。如果该文件已存在,文件指针将会放在文件的结尾。如果文件不存在,创建新文件进行写入

值得关注的是:
文本模式下:\n 写入磁盘时,会保存为\r\n,当遇到\r\n 时,读到的是\n,所以对于一个文件流来说,我们打开都尽量以二进制打开,而不要使用文本模式打开。

seek

seek(offset, from)
offset :文件指针偏移量(offset 定义为指针偏移量)
from : 0-文件开头 1-当前位置 2-文件末尾(这里的 0,1,2 只是代表了文件位
置,而不是说 0,1,2 可以参与指针偏移的计算。from 是可选项,默认为 0.)
二进制模式下才可以往前偏

文件/目录的常用管理操作

  1. 在终端 / 文件浏览器中可以执行常规的 文件 / 目录 管理操作,例如:
    – 创建、重命名、删除、改变路径、查看目录内容、……
  2. 在 Python 中,如果希望通过程序实现上述功能,需要导入 os 模块
方法名说明示例
rename重命名文件os.rename(源文件名, 目标文件名)
remove删除文件,不能删除文件夹os.remove(文件名)
listdir目录列表os.listdir(目录名)
mkdir创建目录os.mkdir(目录名)
rmdir删除目录os.rmdir(目录名)
getcwd获取当前目录os.getcwd()
chdir修改工作目录os.chdir(目标目录)
path.isdir判断是否是文件夹os.path.isdir(文件路径)

在这里插入图片描述

eval函数

eval() 函数十分强大 —— 将字符串当成有效的表达式来求值并返回计算结果,实际上就是去引号化的最简式

例子:

# 基本的数学计算
In [1]: eval("1 + 1")
Out[1]: 2
# 字符串重复
In [2]: eval("'*' * 10")
Out[2]: '**********'
# 将字符串转换成列表
In [3]: type(eval("[1, 2, 3, 4, 5]"))
Out[3]: list
# 将字符串转换成字典
In [4]: type(eval("{'name': 'xiaoming', 'age': 18}"))
Out[4]: dict

对于eval的警惕:
切记不能将eval和前端传回的消息相沟通,只能用之与自己的本地的文件相连,不然很可能出现问题。

补充性知识

位运算小技巧

# 找101个数中的出现1次的1个数
def find_list101_one():
    list1 = [8, 3, 2, 6, 3, 8, 2]  # 所有的数都异或起来,就得到了出现1次的那个数
    result = 0
    for i in list1:
        result ^= i
    print(result)
    

实际上就是通过异或具有交换律进行了解的,相同的异或成0,不同的异或成非零数,得到,我们只需要将这每一个数都异或起来即可得到唯一一个只出现过一次的一个数。

# 找102个数中的出现1次的两个数,如何把出现1次的两个数分到两堆
def find_list102_one():
    list1 = [8, 3, 2, 6, 3, 8, 2, 11]
    result = 0
    for i in list1:
        result ^= i
    split_flag = result & -result
    result1 = 0
    result2 = 0
    for i in list1:
        if split_flag & i:
            result1 ^= i
        else:
            result2 ^= i
    print('出现1次的两个数分别是 %d %d' % (result1,result2))

此处实现全部异或得到一个数,这个数实际上就是那两个数的异或值,比如此题就是6和11的异或值,然后我么将这个数据和他的负数相异或可以得到最低的不同位不同的最小数(通过补码自己计算一遍),然后把这个数去和这个数组在异或一遍,就会将原本的数分成两堆,一堆有6,一堆有11,实际上就是分开的过程,然后再用第一题的方法就可以实现求出两个数。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/31275.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Python入门教程:掌握for循环、while循环、字符串操作、文件读写与异常处理等基础知识

文章目录 for循环while循环字符串操作访问字符串中的字符切片总结字符串拼接 文件读写try...except 异常处理函数模块和包类和面向对象编程完结 for循环 在 Python 中&#xff0c;for 循环用于遍历序列&#xff08;list、tuple、range 对象等&#xff09;或其他可迭代对象。for…

Java中反射机制,枚举,Lambda的使用

目录 一、反射机制 1、含义 2、作用 3、※反射相关的几个类 3.1、Class类&#xff08;Class对象是反射的基石&#xff09; 3.2、Class类中相关的方法 3.2.1 (※重要)常用获得类相关的方法 3.2.2 (※重要)常用获得类中属性、变量Field相关的方法 3.2.3 获得类中注解相…

N-Gram语言模型工具kenlm的详细安装教程

【本配置过程基于Linux系统】 下载源代码&#xff1a; wget -O - https://kheafield.com/code/kenlm.tar.gz |tar xz 编译&#xff1a; makdir kenlm/build cd kenlm/build cmake .. && make -j4 发现报错&#xff1a; 系统中没有cmake&#xff0c;按照错误提示&am…

ChatGPT 指南:角色扮演让回答问题更专业

让 ChatGPT 进行角色扮演 Act as ...&#xff0c;比如&#xff0c;律师、内科医生、心理医生、运动教练、哲学家、翻译、平面设计师、IT 工程师等等&#xff0c;从而才能让 ChatGPT 从这个角色角度来分析我们的问题&#xff0c;不然&#xff0c;它的回答可能会过于广泛。 下面以…

实在智能RPA亮相2023全球人工智能技术博览会,“能对话的数字员工”引领智能自动化新篇章

随着ChatGPT火爆全网&#xff0c;人工智能再次成为学术界和科技领域“新宠”&#xff0c;一场“智能革命”的序幕悄然掀开。 6月13日&#xff0c;“智能驱动 砥砺前行”为主题的2023全球人工智能技术博览会在杭州未来科技城学术交流中心圆满落下帷幕。此次博览会以展示智能科技…

51单片机 - 期末复习重要图

AT89S51片内硬件结构 1.内部硬件结构图 2.内部部件简单介绍 3. 26个特殊功能寄存器分类 按照定时器、串口、通用I/O口和CPU 中断相关寄存器&#xff1a;3IE - 中断使能寄存器IP - 中断优先级寄存器 定时器相关寄存器6TCON - 定时器/计数器控制寄存器TMOD - 定时器/计数器模…

【Leetcode60天带刷】day07哈希表——454.四数相加II , 383. 赎金信 ,15. 三数之和 , 18. 四数之和

题目&#xff1a;454.四数相加II 454. 四数相加 II 给你四个整数数组 nums1、nums2、nums3 和 nums4 &#xff0c;数组长度都是 n &#xff0c;请你计算有多少个元组 (i, j, k, l) 能满足&#xff1a; 0 < i, j, k, l < nnums1[i] nums2[j] nums3[k] nums4[l] 0 …

神经网络:参数更新

在计算机视觉中&#xff0c;参数更新是指通过使用梯度信息来调整神经网络模型中的参数&#xff0c;从而逐步优化模型的性能。参数更新的作用、原理和意义如下&#xff1a; 1. 作用&#xff1a; 改进模型性能&#xff1a;参数更新可以使模型更好地适应训练数据&#xff0c;提高…

I/O多路复用+高性能网络模式

前言&#xff1a; 本篇文章将介绍客户端-服务端之间从最简单的Socket模型到I/O多路复用的模式演变过程&#xff0c;并介绍Reactor和Proactor两种高性能网络模式 文章内容摘自&#xff1a;小林Coding I/O多路复用高性能网络模式 . 传统Socket模型传统Socket模型的性能瓶颈多进程…

SpringCloud Alibaba入门5之使用OpenFegin调用服务

我们继续在上一章的基础上进行开发 SpringCloud Alibaba入门4之nacos注册中心管理_qinxun2008081的博客-CSDN博客 Feign是一种声明式、模板化的HTTP客户端。使用Feign&#xff0c;可以做到声明式调用。Feign是在RestTemplate和Ribbon的基础上进一步封装&#xff0c;使用RestT…

链路追踪SkyWalking整合项目以及数据持久化

1. 微服务整合SkyWalking 1.1 通过jar包方式整合 首先我们将一个简单的springboot服务打成jar包。 将其上传到Linux服务器中。 准备一个启动脚本&#xff0c;脚本内容如下&#xff1a; #!/bin/sh # SkyWalking Agent配置 export SW_AGENT_NAMEskywalking‐test #Agent名字,一…

【MQTT 5.0】协议 ——发布订阅模式、Qos、keepalive、连接认证、消息结构

一、前言1.1 MQTT 协议概述1.2 MQTT规范 二、MQTT 协议基本概念2.1 发布/订阅模式2.11 MQTT 发布/订阅模式2.12 MQTT 发布/订阅中的消息路由2.13 MQTT 与 HTTP 对比2.14 MQTT 与消息队列 2.2 服务质量&#xff1a;QoS2.21 QoS 0 最多分发一次2.22 QoS1 至少分发一次2.23 QoS 2 …

Windows远程桌面(mstsc)不能复制粘贴的解决办法

最近突然发现Windows远程桌面(mstsc)不能在远程端和本地端之间自由的复制和粘贴了&#xff0c;这还是非常影响使用体验的&#xff1b;因此记录一下解决方法&#xff0c;以便后续再遇到此类问题时查看如何解决&#xff1b; 文章目录 一、背景二、解决办法2.1 方法1 重启rdpclip.…

Java并发编程学习16-线程池的使用(中)

线程池的使用&#xff08;中&#xff09; 引言1. 配置 ThreadPoolExecutor1.1 线程的创建与销毁1.2 管理队列任务1.3 饱和策略1.4 线程工厂1.5 定制 ThreadPoolExecutor 2. 扩展 ThreadPoolExecutor总结 引言 上篇分析了在使用任务执行框架时需要注意的各种情况&#xff0c;并…

OpenCV 笔记_1

笔记_1 文章目录 笔记_1Mat类数据类型读取Mat类支持的运算图像读取&#xff0c;显示&#xff0c;保存imread 图像读取namedWindow 创建要显示的窗口imshow 窗口显示imwrite 图像保存 视频加载与摄像头的使用VideoCapture 加载视频或摄像头get 获取属性VideoWriter 保存视频 图像…

【五子棋实战】第6章 调用接口进行联调

【五子棋实战】第6章 调用接口进行联调 Ajax调用接口 调用五子棋接口 点击优化 尾声 更多待开发的功能 Ajax调用接口 引入Jquery&#xff0c;使用JQ封装的ajax&#xff0c;demo如下&#xff1a; <script src"jquery-3.5.0.min.js"></script> <…

Python 操作 Excel 全攻略 | 包括读取、写入、表格操作、图像输出和字体设置

文章目录 前言Python 操作 Excel 教程1. Excel 文件的读取与写入2. Excel 表格的操作2.1 插入和删除行和列2.2 遍历表格中的单元格并修改值 3. 图像的输出3.1 输出柱状图 4. 字体的设置4.1 设置单元格的字体大小和颜色4.2 设置单元格的加粗和斜体4.3 设置单元格的边框和填充颜色…

CSS弹性布局常用设置

目录 一、单位元素 二、弹性容器 三、常用属性 三、项目实战效果 一、单位元素 vm 1vm 为视口的1% vh 视口高的1% vmin 参照长边 vmax 参照长边 rem 等比缩放 需要设置最外层盒子html设置vw 根字号html的--- font-- 1vm 去适配 初始化 //初始化*{padding: 0;margin: 0}//…

【Python GUI编程系列 01】安装python pycharm 和 pyside6

Python GUI编程系列 01 安装python pycharm 和 pyside61、安装python2、安装pycharm3、安装 pyside6 安装python pycharm 和 pyside6 本系列使用python3 pycharmpyside6 来进行python gui设计&#xff0c;首先我们来配置编程环境 PS&#xff1a;为了减少复杂程度&#xff0c;本…

MySQL:事务

事务 在介绍事务之前&#xff0c;我们先来了解一个案例&#xff1a; 在一个买票的软件中&#xff0c;当客户端A检查还有一张票时&#xff0c;将票卖点&#xff0c;但是还没有更新数据库&#xff0c;客户端B检查了票数&#xff0c;发现大于0&#xff0c;于是又卖掉了一张票。然…