命名空间不能直接包含字段或方法之类的成员是什么意思_Python 学习笔记之类与实例...

Python 学习笔记之类与实例

一、定义

1.1、定义

类 (class) 封装一组相关数据,使之成为一个整体,并使用一种方法持续展示和维护。

这有点像把零件组装成整车提供给用户,无须了解汽车的内部结构和工作原理,只要知道方向盘,刹车和油门这些外部接口就可以正常行驶。

类存在两种关系

  1. 继承(inheritance,is-a)自某个族类
    继承可以用来表达本车属于某厂的哪个车族系列,除了继承原车系的技术和优势,还可基于某些环境进行改进。
  2. 组合(composition,has-a)了哪些部件
    组合可用来表述该车使用了哪些零部件,比如最新的发动机。

类与模块的不同之处

  1. 类可生成多个实例。
  2. 类可被继承和扩展。
  3. 类实例的生命周期可控。
  4. 类支持运算符,可按需重载。

这些特性模块没有或者不需要,同时,模块粒度大,模块可用来提供游戏场景级别的解决方案,而类则是该场景下的特定家族和演员。

1.2、创建

定义类,以此为个体为例。关键字 class 同样是运行期指令,用于完成类型对象的创建。

  class User:pass
可在函数内定义,以限制其作用范围。

类型与实例

如果类在模块中定义,那么其生命周期与模块等同,如果被放在函数内,那么每次都是新建。即便名字和内容相同,也属于不同类型。

  def test():class X:passreturn X()​>>> a,b = test(),test()>>> a.__class__ is b.__class__Out[1]: False
函数内定义的类型对象,在所有实例死亡后,会被垃圾回收。

类型对象除了用来创建实例,也为所有实例定义了基本操作接口,其负责管理整个家族的可共享数据和行为目标。

实例只保存私有特征,其以内部引用从所属类型或其它所属祖先类查找所需的方法,用来驱动展现个体面貌。

dddc8d36edc51d0f9491cf6a154abff0.png

名字空间

类型有自己的名字空间,存储当前类型定义的字段和方法。这其中并不包括所继承的祖先成员,其同样以引用关联祖先类型,无须复制到本地。

  class A:a = 100                 #类字段def __init__(self, x):  #实例初始化方法self.x = x          #实例字段def_get_x(self):        #实例方法return self.xclass B(A):                 #继承自Ab = "hello"def __init__(self, x, y):super().__init__(self,x)    #调用父类的初始化方法self.y = ydef get_y(self):return self.y

c7a7160dbc9a565093a42345cee112d0.png

实例 instance o 会保存所有继承层次的实例字段,因为这些都属于其私有数据。

  >>> o = B(1,2)>>> print(A.__dict__){'a': 100,'__init__': <function A.__init__ at 0x109d04f28>, '_get_x': <function A._get_x at 0x109d046a8>, ...}>>> print(B.__dict__){'b': 'hello', '__init__': <function B.__init__ at 0x109d272f0>, 'get_y': <function B.get_y at 0x109d27378>, ...}

当通过实例或类访问某个成员时,会从当前对象开始(instance o 开始查找),依次由近到远向祖先类查找

(即 o --> class B --> class A 进行成员查找)。

如此做的好处就是祖先类的新增功能可以直接 【广播】给所有后代。

在继承层次的不同名字空间中允许有同名成员,并按顺序优先命中。

二、字段

依照所处空间不同,我们将字段分为类型字段实例字段

官方将成员统称为 Attribute,我们可按例将数据当做字段。

2.1、类型字段

【类型字段】在 class 语句块内直接定义,而实例字段必须通过实例引用(self)赋值定义。

仅从执行方式来看,无论实例方法存在于哪级类型,其隐式参数 self 总指向当前调用实例。

  class A:def test(self):             #self 总是指向引用的当前实例,这与继承层次无关print(self)self.x = 100class B(A):pass​if __name__=="__main__":o = B()print(hex(id(o)))print(o.test())0x109d2f358         <__main__.B object at 0x109d2f358>  
实例参数 self 只是约定成俗的名字,这类似于其它语言中的 this,换成 this 同样有效。

2.2、字段赋值

可使用赋值语句为类型实例添加的新字段

  class A:            pass​if __name__=="__main__":A.x = 100                   #新增类型字段print(A.x)print(vars(A))​'''100{x': 100, ...}'''

可一旦实例重新赋值,就将会在其名字空间建立同名字段,并会遮蔽原字段。

  a = A()a.b = 200print(a.b)print(vars(a))'''200{'b': 200}'''

2.3、私有字段

将私有字段暴露给用户是很危险的。因为无论是修改还是删除都无法截获,由此可能引发意外错误。因为语言没有严格意义上的访问权限设置,所以只好将它们隐藏起来。

如果成员名字以双下划线开头,但没有以双下划线结尾,那么编译器会自动对其重命名。

  class X:__table = "user"            #类型变量def __init__(self,name):self.name = name        #实例变量def get_name(self):return self.name
同时双下划线开头课结尾的,通常是系统方法,比如 __ init __ ,__ hash __ ,__ main __等。

所谓重命名,就是编译器附加了类型名称前缀。虽然这种做法不能真正阻止用户访问,但基于名字的约定也算一种提示。这种方式让继承类也无法访问。

重命名机制总是针对当前类型,继承类型无法访问重命名后的基类成员。

可将双下划线前缀改为单下划线,这样虽然不能自动重命名,不过提示作用依旧。

  class A:__name = "user"    #双下划线成员class B(A):def test(self):print(self.__name)​>>> B().test()'''AttributeError: 'B' object has no attribute '_B__name''''​class A:_name = "user"    #单下划线成员class B(A):def test(self):print(self._name)>>> B().test()'''user'''

三、属性

对私有字段会进行重命名保护,那公开字段如何处理呢?

问题是核心在于访问拦截,必须由内部逻辑决定如何返回结果。而属性(property)机制就是将读、写和删除操作映射到指定的方法调用上,从而实现操作控制。

  class C:def __init__(self, name):self.__name = name@propertydef name(self):                 #读return self.__name​@name.setterdef name(self, value):          #写self.__name = value         @name.deleterdef name(self):                 #删除raise AttributeError("can't delete attribute")​c = C("user")print(c.name)# user    c.name = "abc"print(c.name)# abcdel c.nameprint(c.name)#can't delete attribute
这种 @ 语法被称作装饰器(decorator)。
多个方法名必须相同,默认从读方法尅是定义属性,随后以属性名定义写和删除。
如果实现只读,或禁止删除,则只需去掉对应的方法即可。

四、方法

方法是一种特殊函数,其与特定对象绑定,用来获取或修改对象状态。

实际上,无论是对象构造,初始化,析构还是运算符,都以方法实现。根据绑定目标和调用方法的不同,方法可分为实例方法,类型方法,以及静态方法

名字以上下划线开始和结束的方法,通常有特殊用途,其由解释器和内部机制调用。

实例方法

实例方法与实例对象绑定,在其参数列表中,将绑定对象作为第一参数,以便在方法中读取或修改数据状态。在以实例引用调用方法时,无须显式传入第一实参,而由解释器自动完成。

官方建议参数名用 self,同样以 cls 作为类型方法的第一参数名。
  class W:def __init__(self, name):self.__name = namedef get(self):              #以实例引用调用,自动传入 self 参数return self.__namedef set(self, value):self.__name = value>>> w = W("python")>>> w.get()                 #忽略第一参数'''python'''

类型方法

类型方法用来维护类型状态,面向族群提供服务接口。除绑定的第一参数名称不同外,还需添加专门的装饰器,以便解释器将其实例方法区分开来。

  class D:__data = "D.data"@classmethod                #定义为类型方法def get(cls):               #解释器自动将类型对象 D 作为 cls 参数传入return cls.__data@classmethoddef set(cls, value):cls.__data = value>>> D.get()                     #同样忽略 cls 参数'''D.data'''

静态方法

静态方法,则更像是普通函数。其既不接收实例引用,也不参与类型处理,所以就没有自动传入第一参数。使用静态方法,更多原因是将类型作为一个作用域,或者当前类型添加便捷接口。

  class DES:def __init__(self, key):self.__key = keydef encrypt_bytes(self, value):return str(self.__key) + str(value)@staticmethoddef encrypt(key, s):return DES(key).encrypt_bytes(s.encode("utf-8"))>>> DES.encrypt("key", "value")'''keyb'value''''

特殊方法

下面又解释器自动调用,与对象生命周期相关的方法。

  1. __ new __:构造方法,创建对象实例
  2. __ init __:初始化方法,设置实例的相关属性
  3. __ del __:析构方法,实例被回收时调用

创建实例时,会先调用析构方法初始化方法

  class E:def __new__(cls, *args):            #与__init__接收相同的调用函数print("__new__", args)return super().__new__(cls)def __init__(self, *args):          ##self 由 __new__创建并返回print("__init__", args)>>> E(1,2)'''__new__ (1, 2)__init__ (1, 2)'''

如果 __ new __ 返回实例与 cls 类型不符,将导致 __ init __ 无法执行。

五、总结

学习到此,我总算把类的创建,属性和方法等弄清楚了,我最想强调一点,希望读者把 实例 self 参数弄明白,后续编码过程中使用较多。

还要清楚实例方法和静态方法的区别。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

跨平台开发框架 Lynx 初探

跨平台开发是目前开发较热门的方向&#xff0c;React Native 在这方面取得了很大的成功&#xff0c;同时 Flutter 也获得了非常多的关注。React Native 采用 Web 框架开发并使用 Native UI 进行渲染&#xff0c;很大程度上降低了 Native 开发的门槛并且提高迭代的效率&#xff…

ajax包含mysql吗_php 实例ajax与mysql怎么只查询出一条数据?

http://www.runoob.com/php/php...使用这个实例操作之后为什么只显示一条数据&#xff0c;如何让符合条件的数据全部显示出来如&#xff0c;我使用的查询字段是yesterday_str&#xff0c;查询2017-04-18这个数据怎么样才能把2017-04-18包含这个的全部数据提取出来&#xff1f;p…

aspx写入mysql_Asp.net用户登陆数据库验证与注册写入数据库

1.思路与效果图Index.aspx注册注册成功登陆登陆验证通过进入内容页1登陆没通过验证思路&#xff1a;首先建一个Sqlserver数据库Student,再建一个student表(name,pwd)存放用户名和密码。然后注册功能的实现&#xff1a;通过数据库插入信息到表的Sql语句来实现&#xff0c;成功提…

页面体验提升小技巧—渐进式图片

前端性能方面有许多可优化的点&#xff0c;而这些优化带来的就是用户体验的提升。今天我们要聊的东西并不能给性能带来提升&#xff0c;但却能在一定程度上提升用户的体验。 参考博客 场景&#xff1a;在访问页面的时候如果图片较大或者网速慢的情况我们会看到图片加载起来是有…

微信小程序搭配小白接口,自己没有服务器也能开发哦

这里将重点介绍&#xff0c;在自己没有服务器的情况下&#xff0c;如何在微信小程序里直接调用小白接口。 前提 假设你已经开通微信小程序&#xff0c;如果还没有&#xff0c;可前往微信公众平台开通&#xff1a;https://mp.weixin.qq.com 假设你已经开通小白接口&#xff0c…

怎么将自己做好的网站发布到互联网上呢?

如何将自己的网站上传到网站空间。 1.需要有一个上传网站的软件&#xff0c;在这里推荐大家使用 FTP全称是flashfxp这个软件&#xff0c;这个功能功能齐全而且操作简单。大家可以先去下载一下这个软 件 2. 打开FTP&#xff0c;界面如下 3.我们要点击链接按钮&#xff0c;然后FT…

[贝聊科技]网页端「应用跳转」技术实现演变

本文作者&#xff1a;Mr.Luo &#xff0c;贝聊前端经理。本文同时发布于作者 个人博客 。 由于网页传播的便捷性&#xff0c;从网页向APP导流几乎是所有APP厂商都会采用的推广手段&#xff0c;具体来说就是在网页上提供一些触发点&#xff08;例如按钮、链接&#xff09;&#…

Linux服务器配置---安装vsftpd

安装vsftpd 大多数Linux系统都使用vsftpd&#xff0c;因此这里我们也安装vsftpd 1、安装vsftpd [rootlocalhost phpMyAdmin]# yum install -y vsftpd Loaded plugins: fastestmirror, refresh-packagekit, security Installed: vsftpd.i686 0:2.2.2-11.el6_4.1 …

mysql怎么制作柱状图_从数据库中取出最近三十天的数据并生成柱状图

在终端用cd 命令进入文件目录说明&#xff1a;此处例子我是拿项目中的一个例子讲解的。1、新建一个项目 &#xff1a;用终端输入&#xff1a;zf create project Airline 格式&#xff1a;zf create action project project-name 备注&#xff1a;这些格式可以在终端输入zf 查看…

关于ES6的Promise

JavaScript的异步处理 提到JavaScript的异步处理&#xff0c;也许很多人和我一样想到利用回调函数。 例如&#xff1a; firstAsync(function(data){//处理得到的 data 数据//....secondAsync(function(data2){//处理得到的 data2 数据//....thirdAsync(function(data3){//处…

[转]我是如何走进黑客世界的?

*本文原创作者&#xff1a;MyselfExplorer&#xff1b;翻译编辑&#xff1a;楼兰&#xff0c;本文属FreeBuf原创奖励计划&#xff0c;未经许可禁止转载 我想给你一把打开这扇门的钥匙&#xff0c;而你要做的便是静静的聆听接下来的故事。挖掘 0day 一般需要掌握fuzzing&#xf…

Socket.io 深入理解

最近在做项目优化工作时&#xff0c;用到了Socket.io , Socket.io 文档比较少&#xff0c; 结合官网介绍以及自己在项目开发中的摸索&#xff0c;总结如下内容&#xff1b; Socket.io将Websocket和轮询 &#xff08;Polling&#xff09;机制以及其它的实时通信方式封装成了通用…

python填表_小Python填表得到d

我正在尝试使用Scrapy从网站自动下载数据。在我要做的是&#xff1a;使用我的凭据登录网站通过在“RIC”行中写入代码并选择感兴趣的时段来选择我想要的数据单击“获取数据”后&#xff0c;将生成.csv文件&#xff0c;我可以从“下载/”url下载该文件&#xff0c;其中我的所有文…

我理解中的“大前端”/“大无线”

本文内容较长&#xff0c;大概需要15分钟时间阅读。 内容包含五部分&#xff1a;前言&#xff0c;NodeJS职能变化&#xff0c;ReactNative的大规模应用&#xff0c;专门的架构组职能&#xff0c;总结。主要是介绍我所在团队最近的一些变化和思考。 更多信息可以加入我的小密圈关…

Dirichlet分布

1.预备知识 Beta分布函数是一种定义在实数区间[0,1]的特殊函数&#xff0c;它是二项式分布的共轭分布&#xff1b;与Beta分布相同&#xff0c;Dirichlet分布也是定义在实数区间[0,1]的概率度量函数&#xff0c;Dirichlet分布是多项式分布的共轭分布&#xff0c;Dirichlet分布的…

布局定位

布局与定位 摆放元素 1&#xff0c;使用流 流实际上就是浏览器在页面上摆放HTML元素所用的方法。浏览器从HTML文件最上面开始&#xff0c;从上到下沿着元素流逐个显示所遇到的各个元素。 每个块元素会按它在HTML标记中出现的顺序放置在页面上。每个新的块元素会带来一个换行。并…

python语言精通_Python语言基础从入门到精通

1、python关键字False await else import passNone break except in raiseTrue class finally is returnand continue for lambda tryas def from nonlocal whileassert del global not withasync elif if or yield2、命令行参数williamdeMBP-2:~ william$ python -c "imp…

FreeNAS:创建 CIFS 共享(权限)

第一部分&#xff1a;新建账户与指定数据集权限 简单起见&#xff0c;本教程主要介绍带基本身份验证的 CIFS 共享&#xff0c;即只有输入正确的用户名和密码才可以访问共享目录。关于创建匿名共享、多用户权限管理以及域控制器相关内容&#xff0c;我们会另外发布教程专门介绍。…

oracle日记账单据编号未生成_商管财务数据平台Oracle与共享未付池差异如何核对、解决?...

‍‍近期&#xff0c;总部新上线财务数据平台啦&#xff01;各个系统间的差异异常数据清晰可见&#xff0c;随时可查&#xff0c;今天就和小伙伴们一起分享一下Oracle与共享未付池差异如何核对、解决。首先&#xff0c;将Oracle与共享未付池差异数据导出。由于导出的数据包括本…

android paint 圆角 绘制_[BOT] 一种android中实现“圆角矩形”的方法

内容简介文章介绍ImageView(方法也可以应用到其它View)圆角矩形(包括圆形)的一种实现方式&#xff0c;四个角可以分别指定为圆角。思路是利用“Xfermode Path”来进行Bitmap的裁剪。背景圆角矩形实现的方法应该很多&#xff0c;网上一大堆。很怀疑为啥安卓的控件不内置这样的属…