032.Python魔术方法__new__和单态模式
一 __new__ 魔术方法
1.1 介绍
- 触发时机:实例化类生成对象的时候触发(触发时机在__init__之前)
- 功能:控制对象的创建过程
- 参数:至少一个cls接受当前的类,其他根据情况决定
- 返回值:通常返回对象或None
对象.属性
对象.方法() 类.属性
类.方法()
1.2 基本用法
class MyClass():
abc = 123
def __new__(cls): #把class这个类传递__new__这个方法
print (123)
return None #返回一个None #实例化对象
obj = MyClass()
print (obj)
执行
这样就不能使用使用obj这个对象调用abc的值
class MyClass():
abc = 123
def __new__(cls):
print (123)
return None #实例化对象
obj = MyClass()
print (obj)
print (obj.abc) #在这里就相当于使用None.abc.这种此写法不允许,就会报错
执行
借助父类创建对象
返回本类对象
class MyClass():
abc = 123
def __new__(cls):
print (cls)
print (123)
#要借助父类object的__new__创建对象
obj = object.__new__(cls)
return obj #实例化对象
obj = MyClass()
print (obj)
print (obj.abc)
执行
[root@node10 python]# python3 test.py
<class '__main__.MyClass'> #cls是一个类
123
<__main__.MyClass object at 0x7ff2808ba3c8>
123
[root
返回空对象
class MyClass():
abc = 123
def __new__(cls):
print (cls)
print (123)
#要借助父类object的__new__创建对象
obj = object.__new__(cls)
#return obj
return None #实例化对象
obj = MyClass()
print (obj)
print (obj.abc)
执行
返回一个其他类的对象
class MyClass2():
ccc = 4
obj2 = MyClass2()
class MyClass():
abc = 123
def __new__(cls):
print (cls)
print (123)
#要借助父类object的__new__创建对象
obj = object.__new__(cls)
#return obj
#return None
return obj2 #实例化对象
obj = MyClass()
print (obj)
print (obj.abc)
执行
修改属性
[root@node10 python]# cat test.py
class MyClass2():
ccc = 4
obj2 = MyClass2()
class MyClass():
abc = 123
def __new__(cls):
print (cls)
print (123)
#要借助父类object的__new__创建对象
obj = object.__new__(cls)
#return obj
#return None
return obj2 #实例化对象
obj = MyClass()
print (obj)
print (obj.ccc)
执行
[root@node10 python]# python3 test.py
<class '__main__.MyClass'>
123
<__main__.MyClass2 object at 0x7fb463cc7470>
4
1.3 对__new__和__init__这两个魔术方法的区别
调用的顺序
class Boat():
def __new__(cls):
print (1)
return object.__new__(cls)
def __init__(self):
print (2)
obj = Boat()
print (obj)
执行
[root@node10 python]# python3 test.py
1
2
<__main__.Boat object at 0x7f9bb5fcf400>
先调用__new__在调用__init__
调用和定义的顺序无关
class Boat():
def __init__(self):
print (2)
def __new__(cls):
print (1)
return object.__new__(cls)
obj = Boat()
print (obj)
执行
[root@node10 python]# python3 test.py
1
2
<__main__.Boat object at 0x7f95a73cf400>
结果一样
__new__ 的触发时机要快于 __init__
__new__ 是用来创建对象的
__init__ 是用来初始化对象的 先得有对象
才能初始化对象 new 和 init 他门的参数要保持一致.
1.4 如果返回的不是一个本类对象__init__就不会触发
class Boat():
def __new__(cls):
print (1)
#return object.__new__(cls)
return None
def __init__(self):
print (2)
obj = Boat()
print (obj)
执行
[root@node10 python]# python3 test.py
1
None
初始化参数
class Boat():
def __new__(cls):
print (1)
return object.__new__(cls)
#return None
def __init__(self,name):
self.name = name
obj = Boat("John")
print (obj.name)
执行
[root@node10 python]# python3 test.py
Traceback (most recent call last):
File "test.py", line 8, in <module>
obj = Boat("John")
TypeError: __new__() takes 1 positional argument but 2 were given
这是因为,__new__先于__init__调用,但是,在调用的时候,已经传递了一个类(cls)的参数,name的参数就不能对应报错
class Boat():
def __new__(cls,name):
print (1)
return object.__new__(cls)
#return None
def __init__(self,name):
self.name = name
obj = Boat("John")
print (obj.name)
执行,而可以调用
[root@node10 python]# python3 test.py
1
John
1.5 使用参数的普通参数和关键字实参
lass Boat():
def __new__(cls,*args,**kwargs):
print (1)
return object.__new__(cls)
#return None
def __init__(self,*args,**kwargs):
strvar = ""
for i in args:
strvar += i + " "
print (strvar)
print (kwargs)
obj = Boat("John","Jim","Tom",name="David")
执行
[root@node10 python]# python3 test.py
1
John Jim Tom
{'name': 'David'}
1.6 如果__new__魔术方法返回的时其他类的对象,不会触发__init__ 本类的魔术方法
class MyClass2():
ccc = 4
obj2 = MyClass2()
class Boat():
def __new__(cls,*args,**kwargs):
print (1)
#return object.__new__(cls)
return obj2
def __init__(self,*args,**kwargs):
print (111)
obj = Boat("John","Jim","Tom",name="David")
print (obj.ccc)
执行
[root@node10 python]# python3 test.py
1
4
二 单态模式
无论实例化类几次,都有且只有一个对象.为了节省内存空间
2.1 创建一个类,实例化三个对象
[root@node10 python]# cat test.py
class Singleton():
pass
obj1 = Singleton()
print (obj1)
obj2 = Singleton()
print (obj2)
obj3 = Singleton()
print (obj3)
执行
[root@node10 python]# python3 test.py
<__main__.Singleton object at 0x7efd7ec76400>
<__main__.Singleton object at 0x7efd7ec76438>
<__main__.Singleton object at 0x7efd7ec76470>
可以看到占用不同的内存空间
2.2 使用单态模式
class Singleton():
#创建一个私有的对象,类外无法调用
__obj = None
#定义一个方法
def __new__(cls):
#判断这个对象是否为None对象
if cls.__obj is None:
obj = object.__new__(cls) #如果是,就重新创建一个新的对象
cls.__obj = obj #把这新的对象夫给类的私有方法
return cls.__obj #如果不是None,直接返回
obj1 = Singleton()
print (obj1)
obj2 = Singleton()
print (obj2)
obj3 = Singleton()
print (obj3)
执行
[root@node10 python]# python3 test.py
<__main__.Singleton object at 0x7fbae7edb4a8>
<__main__.Singleton object at 0x7fbae7edb4a8>
<__main__.Singleton object at 0x7fbae7edb4a8>
发现三个对象都是一个内存地址
2.3 实际的含义,对象和init之间的关系
class Singleton():
#创建一个私有的对象,类外无法调用
__obj = None
#定义一个方法
def __new__(cls,name):
#判断这个对象是否为None对象
if cls.__obj is None:
cls.__obj = object.__new__(cls)
return cls.__obj
def __init__(self,name):
self.name = name
obj1 = Singleton("John")
obj2 = Singleton("Frnak")
print (obj1.name)
print (obj2.name)
执行
[root@node10 python]# python3 test.py
Frnak
Frnak
032.Python魔术方法__new__和单态模式的更多相关文章
- day24:多态&魔术方法__new__&单态模式
目录 1.多态 2.__new__魔术方法 2.1 关于魔术方法__new__ 2.2 基本语法 2.3 __new__ 触发时机快于构造方法 2.4 __new__ 和 __init__ 参数一一对 ...
- python魔术方法
在类中有一些特殊的方法具有特殊的意义,比如__init__和__del__方法,它们的重要性我们已经学习过了. 一般说来,特殊的方法都被用来模仿某个行为.例如,如果你想要为你的类使用x[key]这样的 ...
- Python 魔术方法笔记
魔术方法总是被__包围, 如__init__ , __len__都是常见的魔术方法,这里主要写一下我遇到的一些魔术方法 setitem 对某个索引值赋值时 即可以进行赋值操作,如 def __seti ...
- 记一次 Apache HUE 优化之因使用 Python 魔术方法而遇到的坑
最近的工作是基于 Apache HUE 做二次开发.刚接手 HUE 的代码的时候,内心是崩溃的:开源的代码,风格很多种, 代码比较杂乱; 虽是基于 Django 开发的,但是项目的结构改变很大; 很多 ...
- Python的程序结构[1] -> 方法/Method[3] -> 魔术方法 __getattr__ 与代理模式
__getattr__ 方法 __getattr__ 方法当对象调用内部属性(包括方法等)且未找到对应属性的时候会调用的特殊方法.利用这一特性,可是对函数实现一个代理模式. __getattr__方法 ...
- Python魔术方法-Magic Method
介绍 在Python中,所有以"__"双下划线包起来的方法,都统称为"Magic Method",例如类的初始化方法 __init__ ,Python中所有的魔 ...
- Python 魔术方法指南
入门 构造和初始化 构造定制类 用于比较的魔术方法 用于数值处理的魔术方法 表现你的类 控制属性访问 创建定制序列 反射 可以调用的对象 会话管理器 创建描述器对象 持久化对象 总结 附录 介绍 此教 ...
- [python]魔术方法
一.初始化: 1.__new__方法,初始化过程中第一个用到的方法(用处不大). 2.之后,__init__方法,构造方法. 3.最后,在对象回收时,调用__del__方法.如果解释器退出时,对象还存 ...
- Python 魔术方法及调用方式
魔术方法 调用方式 解释 __new__(cls [,...]) instance = MyClass(arg1, arg2) __new__ 在创建实例的时候被调用 __init__(self [, ...
随机推荐
- java面试-synchronized底层实现机制
一.synchronized的三种应用方式 1.修饰实例方法,锁是当前实例对象,进入同步代码前要获得当前实例的锁 /** * synchronized修饰实例方法,当前线程的锁是实例对象account ...
- python基础(十):集合的使用(上)
集合的作用 去重:把一个列表变成集合,就自动去重了. 关系测试:测试两组数据之前的交集.差集.并集等关系. 集合的特征 集合使用 set 表示: 集合也使用{ }表示, 与字典不同的是:字典中存储的是 ...
- python基础(二):数据类型
数据类型 学习Python时,一定要注意Python中数据类型和数据结构的学习,这对于你是否能够学好Python其中很重要的作用. 什么是数据类型? 人类有思想,很容易区分汉字和数字的区别,例如,你知 ...
- c 结构体内存对齐详解
0x00简介 首先要知道结构体的对齐规制 1.第一个成员在结构体变量偏移量为0的地址处 2.其他成员变量对齐到某个数字的整数倍的地址处 对齐数=编辑器默认的一个对齐数与该成员大小的较小值 vs中默认的 ...
- JAP 1.0.1 以及 《JAP产品技术白皮书》正式发布
快讯 JAP 1.0.1 正式发布 <JAP产品技术白皮书>正式发布.立即获取:白皮书 JAP 1.0.1 版本内容 新增功能/支持 添加 com.fujieid.jap.core.uti ...
- Day16_93_IO_FileInputStream_读取文件字节流read()方法(一)
读取文件字节流read()方法 * 文件字节输入流:按照字节方式读取文件 * java.io.* java.io.InputStream; java.io.FileInputStream; read( ...
- Day11_52_将Set集合转换为List集合
Set集合转换为List集合 ``` import java.util.*; public class SetReverseLsit { public static void main(String[ ...
- 基于.Net Core 5.0 Worker Service 的 Quart 服务
前言 看过我之前博客的人应该都知道,我负责了相当久的部门数据同步相关的工作.其中的艰辛不赘述了. 随着需求的越来越复杂,最近windows的计划任务已经越发的不能满足我了,而且计划任务毕竟太弱智,总是 ...
- 谈谈react hooks的优缺点
前言Hook 是 React 16.8 的新增特性.它是完全可选的,并且100%向后兼容.它可以让你使用函数组件的方式,运用类组件以及 react 其他的一些特性,比如管理状态.生命周期钩子等.从概念 ...
- Oralce注入 bypass waf出数据
发存货: 探测banner 版本号: ' and (SELECT banner FROM v$version where rownum=1) like 'O%' and '1'like'1 rownu ...