Python - 面向对象编程 - 类变量、实例变量/类属性、实例属性
什么是对象和类
https://www.cnblogs.com/poloyy/p/15178423.html
什么是 Python 类、类对象、实例对象
https://www.cnblogs.com/poloyy/p/15178456.html
类变量、实例变量/类属性、实例属性
前言
只是叫法不一样
实例属性 = 实例变量
类属性 = 类变量
个人认为叫属性更恰当
类属性和实例属性区别
- 类属性,所有实例对象共享该属性
- 实例属性,属于某一个实例对象的属性,用于描述具体的对象
从实际栗子了解类属性、实例属性
有一个表格,四个常见的明星
| 姓名 | 年龄 |
|---|---|
| 周润发 | 58 |
| 成龙 | 55 |
| 刘德华 | 53 |
| 周星驰 | 54 |
总结一下
- 四个人归类为明星
- 每个明星都有两个属性:姓名、年龄
- 明星这个群体具有一个属性:明星数量,在这张表是 4
- 姓名和年龄等属性是用来描述具体的一个对象
- 明星的数量是用于描述明星这个类别的
使用面向对象编程思想来总结的话
- 周润发、成龙、刘德华、周星驰都是实例对象
- 他们都属于明星,明星是类
- 属于实例对象的属性有:姓名、年龄,所以也叫实例属性
- 属于明星类的属性有:数量,所以也叫类属性
类里面的三种类型变量
- 在所有方法之外定义的变量,称为类属性/类变量
- 在方法内部,通过 self.变量名 方式定义的变量,称为实例属性/实例变量
- 在方法内部,通过 变量名=变量值 方式定义的变量,称为局部变量
类属性
类属性在类中的定义
class 类名:
类属性1 = 值
类属性2 = 值 def func(self):
...
类属性、类方法注意点
- 无论是类属性还是类方法,都无法像普通变量或者函数那样,在类的外部直接使用它们(类方法后面详解)
- 可以将类看做一个独立的空间,类属性其实也是在类体中定义的变量,类方法是在类体中定义的函数
- 需要通过类对象/实例对象来调用类属性 ClassName.classProperty (类方法后面详解)
类属性的栗子
# 类属性
class PoloBlog:
# 这就是在所有方法之外 下面定义了 2 个类变量
name = "小菠萝测试笔记"
blog = "https://www.cnblogs.com/poloyy/" # 通过类名调用类属性
print(PoloBlog.name)
print(PoloBlog.blog) # 输出结果
小菠萝测试笔记
https://www.cnblogs.com/poloyy/
通过 Pycharm 的代码联想,可以看到 blog、name、__doc__ 三个类属性

类属性的调用方式
有两种
- 直接通过类名调用
- 也可以通过类的实例对象调用
调用类属性的栗子
# 调用类属性的两种方式
class PoloBlog:
# 这就是在所有方法之外 下面定义了 2 个类变量
name = "小菠萝测试笔记"
blog = "https://www.cnblogs.com/poloyy/" # 通过类名直接调用
print(PoloBlog.name)
print(PoloBlog.blog) # 修改类属性
PoloBlog.name = "blogyuan"
PoloBlog.blog = "https://www.cnblogs.com/" # 通过实例对象调用修改后的类属性
poloBlog = PoloBlog()
print(poloBlog.name)
print(poloBlog.blog) # 输出结果
小菠萝测试笔记
https://www.cnblogs.com/poloyy/
blogyuan
https://www.cnblogs.com/
通过类名修改类属性的值,会影响所有的实例化对象
实例对象修改类属性
# 修改类属性
poloBlog.name = "小菠萝回来了" # 再看看类对象调用修改后的类属性
print(PoloBlog.name)
print(poloBlog.name) # 输出结果
blogyuan
小菠萝回来了
- 会发现,类名.name 仍然返回之前的值,而 实例对象.name 会返回修改的值
- 原因: 实例对象.name 本质上并不是修改类属性的值,而是在定义一个新的实例属性(下面详解)
动态添加类属性
PoloBlog.age = 24
print(PoloBlog.age)
print(poloBlog.age) # 输出结果
24
24
- age 没有在类体中定义
- 可以直接通过 类名.new_property_name 的方式定义一个新的类属性
实例属性
- 属于具体对象的属性,用于描述具体的对象
- 只能通过实例对象访问,无法通过类名访问
实例属性的栗子
class PoloBlog:
def __init__(self):
# 在方法内部,通过 self.name 的方式定义的变量就是实例变量
self.name = "小菠萝测试笔记"
self.add = "https://www.cnblogs.com/poloyy/" # 下面定义了一个 say 实例方法
def say(self):
self.age = 13 # 实例化对象
blog = PoloBlog()
blog.say() print(blog.name, blog.add, blog.age) # 输出结果
小菠萝测试笔记 https://www.cnblogs.com/poloyy/ 13
- 重点:__init__ 会在实例化对象的时候自动调用,因此 blog1 创建成功就有 name、add 两个实例属性
- 调用 say() 方法之后才有第三个实例属性 age
修改实例属性的栗子
blog.name = "小菠萝"
blog.add = "xiaopolo.com"
blog.age = 24 print(blog.name, blog.add, blog.age) # 输出结果
小菠萝 xiaopolo.com 24
动态添加实例属性
blog.phone = 13501489999
print(blog.phone) # 输出结果
13501489999
上面也有说到,通过 实例对象.属性名 的方式并不会给类变量赋值,而是定义一个新的实例变量
综合栗子
# 综合栗子
class PoloBlogObjectTest:
# 类变量
sum = 0 # 初始化方法
def __init__(self, name, age):
# 实例变量
self.name = name
self.age = age
# 类变量
PoloBlogObjectTest.sum += 1 # 实例方法
def printNameAge(self):
print(self.name, self.age) poloTest1 = PoloBlogObjectTest("小菠萝一号", 24)
poloTest2 = PoloBlogObjectTest("小菠萝二号", 14) print(PoloBlogObjectTest.sum)
# 调用实例方法
poloTest1.printNameAge()
poloTest2.printNameAge() # 输出结果
2
小菠萝一号 24
小菠萝二号 14
不推荐实例属性和类属性同名
- 类中,实例属性和类属性可以同名
- 但这种情况下使用实例对象将无法调用类变量,它会首选实例变量,无论这个变量是否已定义
- 实例独享绑定新的实例属性时,会直接覆盖掉重名的类属性
实例属性、类属性同名栗子
class Person:
# 只有一个类变量
name = "cool guy" # 实例化一个对象
p = Person()
# 打印实例属性 name,因为实例对象并没有name属性,所以会继续查找class的name属性
print(p.name)
# 打印类属性 name
print(Person.name) # 给实例绑定 name、age 属性
p.name = "bad guy"
p.age = 12 # 打印 name、age 属性
print(p.age)
# 由于实例属性优先级比类属性高,因此,它会屏蔽掉类的 name 属性
print(p.name)
# 仍然打印类的 name 属性
print(Person.name) # 输出结果
cool guy
cool guy
12
bad guy
cool guy
实例对象属性引用的查找过程

Python - 面向对象编程 - 类变量、实例变量/类属性、实例属性的更多相关文章
- python 面向对象编程学习
1. 问题:将所有代码放入一个py文件:无法维护 方案:如果将代码才分放到多个py文件,好处: 1. 同一个名字的变量互相不影响 2.易于维护 3.引用模块: import module 2.包:解决 ...
- Python - 面向对象编程 - 什么是 Python 类、类对象、实例对象
什么是对象和类 https://www.cnblogs.com/poloyy/p/15178423.html Python 类 类定义语法 最简单的类定义看起来像这样 class ClassName: ...
- python面向对象编程进阶
python面向对象编程进阶 一.isinstance(obj,cls)和issubclass(sub,super) isinstance(obj,cls)检查是否obj是否是类 cls 的对象 1 ...
- Python面向对象编程(上)
Python不仅支持面向过程编程,同时也支持面向对象编程.面向工程就是分析解决问题所需的步骤,然后用函数把这些步骤逐一实现,使用的时候再一个个调用函数就可以.面向对象则是把解决的问题按照一定规则划分为 ...
- python - 面向对象编程(初级篇)
写了这么多python 代码,也常用的类和对象,这里准备系统的对python的面向对象编程做以下介绍. 面向对象编程(Object Oriented Programming,OOP,面向对象程序设计) ...
- Python 面向对象编程详解
Python 的创始人为吉多·范罗苏姆(Guido van Rossum).1989年的圣诞节期间,吉多·范罗苏姆为了在阿姆斯特丹打发时间,决心开发一个新的脚本解释程序,作为ABC语言的一种继承.Py ...
- 图解python | 面向对象编程
作者:韩信子@ShowMeAI 教程地址:http://www.showmeai.tech/tutorials/56 本文地址:http://www.showmeai.tech/article-det ...
- python面向对象编程(上)
面向对象编程(OOP,Object Oriented Programming)是每一个高级编程语言都支持的编程方法,比如JAVA/C++/C#等等.学习面向对象编程是每一个程序员都绕不开的重点内容. ...
- python 面向对象编程(一)
一.如何定义一个类 在进行python面向对象编程之前,先来了解几个术语:类,类对象,实例对象,属性,函数和方法. 类是对现实世界中一些事物的封装,定义一个类可以采用下面的方式来定义: class c ...
随机推荐
- 深入理解JavaScript中的继承
1前言 继承是JavaScript中的重要概念,可以说要学好JavaScript,必须搞清楚JavaScript中的继承.我最开始是通过看视频听培训班的老师讲解的JavaScript中的继承,当时看的 ...
- 更改Nginx网站根目录以及导致的403 forbidden问题
Nginx采用默认配置,只修改了root的网站根目录位置,再访问网站的时候提示403Forbidden的错误. 仔细检查了新文件夹的权限,也对比了心就网站根目录的权限,都是一样的. 最后尝试关闭了SE ...
- 前端性能之LightHouse
"灯塔"(LightHouse)前端性能优化测试工具 (谷歌亲儿子) 一 灯塔v6/v7版是通过几种性能指标及不同权重来进行计分的 前端性能指标主要是根据PerformanceTi ...
- 右键发送 (sendto),创建快捷方式到自定义的位置,不仅仅是复制,就像 发送到 桌面快捷方式 一样
TL;DR 在 SendTo 文件夹里加上一文件夹的快捷方式后,在右键发送到这个文件夹的是这些文件的一个副本,实际上是一个复制的过程,有时候我们只希望是快捷方式,那就得另想办法了. 方案如下: 创建一 ...
- Django GIS SQL注入漏洞(CVE-2020-9402)
影响版本 Django 1.11.29之前的1.11.x版本.2.2.11之前的2.2.x版本和3.0.4之前的3.0.x版本中存在SQL注入漏洞 提示有admin.vuln.vuln2,3个页面,存 ...
- SSH远程端口转发实战详解
问题 前段时间在外地没有在实验室,随身携带了一个笔记本电脑.但是笔记本性能不够,想用SSH远程连接实验室的电脑.问如何连接?现有以下设备 设备 IP 备注 系统 实验室电脑C1 192.168.0.2 ...
- 【FATE】设置虚拟机固定IP以及免密登录
一.前期准备 1.VMWare上新建三个Centos7的虚拟机 2.VMWare虚拟机的三种联网方式 1.桥接模式 -- 桥接: 默认使用VMnet0 这一种联网方式最简单,在局域网内,你的主机是怎么 ...
- BSTestRunner增加历史执行记录展示和重试功能
之前对于用例的失败重试,和用例的历史测试记录存储展示做了很多的描述呢,但是都是基于各个项目呢,不方便使用,为了更好的使用,我们对这里进行抽离,抽离出来一个单独的模块,集成到BSTestRunner中, ...
- jadx的使用
使用jadx之前必须安装配置java环境才能正常打开运行 https://www.cnblogs.com/yhoil/p/14808648.html 一.前言 今天介绍一个非常好用的反编译的工具 ja ...
- 剑指 Offer 39. 数组中出现次数超过一半的数字
剑指 Offer 39. 数组中出现次数超过一半的数字 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字. 你可以假设数组是非空的,并且给定的数组总是存在多数元素. 示例 1: 输入: [ ...