深入浅析python 中的self和cls的区别


Posted in Python onJune 20, 2020

python 中的self和cls

一句话描述:self是类(Class)实例化对象,cls就是类(或子类)本身,取决于调用的是那个类。
@staticmethod 属于静态方法装饰器,@classmethod属于类方法装饰器。我们需要从声明和使用两个方面来理解。

详细介绍

一般来说,要使用某个类的方法,需要先⚠️实例化一个对象再调用方法。而使用@staticmethod或@classmethod,就可以不需要实例化,直接类名.方法名()来调用。这有利于组织代码,把某些应该属于某个类的函数给放到那个类里去,同时有利于命名空间的整洁。?

首先定义一个类A,类A中有三个函数,foo1为静态函数,用@staticmethod装饰器装饰,这种方法与类有某种关系但不需要使用到实例或者类来参与。

class A(object):
 a = 'a'
 @staticmethod
 def foo1(name):
 print('hello', name, A.a)
 def foo2(self, name):
 print('hello', name, self.a)
 @classmethod
 def foo3(cls, name):
 print('hello', name, cls.a)
class B(A):
 a = 'b'
 @staticmethod
 def foo1(name):
 print('hello', name, B.a)
 def foo2(self, name):
 print('subclass B')
 print('hello', name, self.a)
 @classmethod
 def foo3(cls, name):
 print('hello', name, cls.a)

如下两种方法都可以正常输出,也就是说

既可以作为类的方法使用,也可以作为类的实例的方法使用。

a = A()
b = B()
a.foo1("小熊猫") # hello 小熊猫 
A.foo1("小熊猫") # hello 小熊猫 
b.foo1("大熊猫") # subclass B, hello 大熊猫 b
B.foo1("大熊猫") # subclass B, hello 大熊猫 b

foo2为正常的函数,是类的实例的函数,调用方式如下。

实参实例化对象或者类名称传入self对象,取到不同的属性和方法。

a.foo2("小熊猫") # hello 小熊猫 a
A.foo2(a, "小熊猫") # hello 小熊猫 a
A.foo2(b, "小熊猫") # hello 小熊猫 b
A.foo2(A, "小熊猫") # hello 小熊猫 a 
A.foo2(B, "小熊猫") # hello 小熊猫 b
B.foo2(a, "小熊猫") # subclass B, hello 小熊猫 a

foo3为类函数,cls作为第一个参数用来表示类本身. 在类方法中用到,类方法是只与类本身有关而与实例无关的方法。如下两种方法都可以正常输出。

可以看出,传入形参cls的值为前面的调用函数,如果再传入对象或者类名称,会报类型错误,多传了一个参数。

a.foo3("小熊猫")
A.foo3("小熊猫")
# a.foo3(a, "小熊猫") # TypeError: foo3() takes 2 positional arguments but 3 were given
# A.foo3(A, "小熊猫") # TypeError: foo3() takes 2 positional arguments but 3 were given
b.foo3("大熊猫")
B.foo3("大熊猫")

@staticmethod和@classmethod的用法

相同:

@staticmethod和@classmethod都可以直接类名.方法名()来调用

区别:

  • 从它们的使用上来看,@staticmethod不需要表示自身对象的self和自身类的cls参数,就跟使用函数一样。@classmethod也不需要self参数,但第一个参数需要是表示自身类的cls参数。
  • 如果在@staticmethod中要调用到这个类的一些属性方法,只能直接类名.属性名或类名.方法名。
  • 而@classmethod因为持有cls参数,可以来调用类的属性,类的方法,实例化对象等,避免硬编码。
class A(object):
 a = 'a'
 @staticmethod
 def foo1(name):
 print('hello foo1', name, A.a)
 print("hello foo4 ", B.foo2(B, "小熊猫"))
 def foo2(self, name):
 print('hello foo2', name, self.a)
 @classmethod
 def foo3(cls, name):
 print('hello foo3', name, cls.a)
 print("hello foo5", cls().foo2(name))
 print("hello foo6", cls().foo1(name))
class B(A):
 a = 'b'
 @staticmethod
 def foo1(name):
 print('subclass B, hello', name, B.a)
 def foo2(self, name):
 print('subclass B, hello', name, self.a)
 @classmethod
 def foo3(cls, name):
 print('subclass B, hello', name, cls.a)

 重点应关注@staticmethod和@classmethod调用本类或其他类的函数和属性的区别

例子1:

关键看第二句 subclass B, hello 小熊猫 b,在调用 B.foo2(B, “小熊猫”) 时,执行了B类型下的foo2()方法,该方法无返回值,因此 下句输出为 hello foo4 None

a = A()
a.foo1("小熊猫")
# 输出
hello foo1 小熊猫 a
subclass B, hello 小熊猫 b
hello foo4 None

例子2:

a.foo3("小熊猫")
# 输出
hello foo3 小熊猫 a
hello foo2 小熊猫 a
hello foo5 None
hello foo1 小熊猫 a
subclass B, hello 小熊猫 b
hello foo4 None
hello foo6 None

PS:下面看下python中self和cls的区别

1、self表示一个具体的实例本身。如果用了staticmethod,那么就可以无视这个self,将这个方法当成一个普通的函数使用。

2、cls表示这个类本身。

>>> class A(object):
    def foo1(self):
      print "Hello",self
    @staticmethod
    def foo2():
      print "hello"
    @classmethod
    def foo3(cls):
      print "hello",cls

>>> a = A()
>>> a.foo1()     #最常见的调用方式,但与下面的方式相同
Hello <__main__.A object at 0x9f6abec>
>>> A.foo1(a)     #这里传入实例a,相当于普通方法的self
Hello <__main__.A object at 0x9f6abec>
>>> A.foo2()     #这里,由于静态方法没有参数,故可以不传东西
hello
>>> A.foo3()     #这里,由于是类方法,因此,它的第一个参数为类本身。
hello <class '__main__.A'>

>>> A         #可以看到,直接输入A,与上面那种调用返回同样的信息。
<class '__main__.A'>

3、whats more,类先调用__new__方法,返回该类的实例对象,这个实例对象就是__init__方法的第一个参数self,即self是__new__的返回值。

总结

到此这篇关于深入浅析python 中的self和cls的区别的文章就介绍到这了,更多相关python 中的self和cls内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
python封装对象实现时间效果
Apr 23 Python
python选择排序算法的实现代码
Nov 21 Python
Python实现备份MySQL数据库的方法示例
Jan 11 Python
python dataframe 输出结果整行显示的方法
Jun 14 Python
Python pandas用法最全整理
Aug 04 Python
DataFrame.to_excel多次写入不同Sheet的实例
Dec 02 Python
如何在django中添加日志功能
Feb 06 Python
Django-rest-framework中过滤器的定制实例
Apr 01 Python
python实现飞船大战
Apr 24 Python
Python将list元素转存为CSV文件的实现
Nov 16 Python
Python Numpy之linspace用法说明
Apr 17 Python
Python 全局空间和局部空间
Apr 06 Python
浅谈Python 参数与变量
Jun 20 #Python
python字符串的index和find的区别详解
Jun 20 #Python
Django Admin 上传文件到七牛云的示例代码
Jun 20 #Python
什么是python的函数体
Jun 19 #Python
浅谈Python中的生成器和迭代器
Jun 19 #Python
python中有帮助函数吗
Jun 19 #Python
python中导入 train_test_split提示错误的解决
Jun 19 #Python
You might like
PHP中一个控制字符串输出的函数
2006/10/09 PHP
php实现首页链接查询 友情链接检查的代码
2010/01/05 PHP
微信公众平台接口开发入门示例
2014/12/24 PHP
如何通过View::first使用Laravel Blade的动态模板详解
2017/09/21 PHP
php-msf源码详解
2017/12/25 PHP
Laravel访问出错提示:`Warning: require(/vendor/autoload.php): failed to open stream: No such file or di解决方法
2019/04/02 PHP
JavaScript 学习小结(适合新手参考)
2009/07/30 Javascript
jQuery级联操作绑定事件实例
2014/09/02 Javascript
使用JavaScript脚本无法直接改变Asp.net中Checkbox控件的Enable属性的解决方法
2015/09/16 Javascript
基于jQuery实现收缩展开功能
2016/03/18 Javascript
MVC+jQuery.Ajax异步实现增删改查和分页
2020/12/22 Javascript
React创建组件的三种方式及其区别
2017/01/12 Javascript
详解Node.js access_token的获取、存储及更新
2017/06/20 Javascript
React路由管理之React Router总结
2018/05/10 Javascript
浅谈JavaScript_DOM学习篇_图片切换小案例
2019/03/19 Javascript
layui实现把数据表格时间戳转换为时间格式的例子
2019/09/12 Javascript
原理深度解析Vue的响应式更新比React快
2020/04/04 Javascript
[04:19]DOTA2完美大师赛第四天精彩集锦
2017/11/26 DOTA
tensorflow实现简单逻辑回归
2018/09/07 Python
用uWSGI和Nginx部署Flask项目的方法示例
2019/05/05 Python
Python如何把多个PDF文件合并代码实例
2020/02/13 Python
python中前缀运算符 *和 **的用法示例详解
2020/05/28 Python
python3.7调试的实例方法
2020/07/21 Python
Python为何不支持switch语句原理详解
2020/10/21 Python
Vuori官网:运动服装的终级表现
2021/01/27 全球购物
自我评价怎么写正确呢?
2013/12/02 职场文书
开会迟到检讨书
2014/01/08 职场文书
高中生家长寄语大全
2014/04/03 职场文书
经贸专业毕业生求职信范文
2014/05/01 职场文书
祖国在我心中演讲稿400字
2014/05/04 职场文书
党员干部廉洁承诺书
2014/05/28 职场文书
民政局副局长民主生活会个人对照检查材料
2014/09/19 职场文书
入团介绍人意见范文
2015/06/04 职场文书
《绝招》教学反思
2016/02/20 职场文书
基于Python的EasyGUI学习实践
2021/05/07 Python
「地球外少年少女」BD发售宣传CM公开
2022/03/21 日漫