Python 类使用详解
wangzf / 2024-04-02
目录
定义一个类
class Example:
def __init__(self, data1: int, data2: int, data3: int):
self.data = data1 # 类的公有属性
self._data = data2 # 类的受保护属性
self.__data = data3 # 类的私有属性
def func(self):
"""
类的公有方法
"""
print("Example 类的公有方法可以被类和实例调用。")
def _func(self):
"""
类的受保护的方法
"""
print("Example 类的受保护的方法可以被类和实例调用。")
def __func(self):
"""
类的私有方法
"""
print("Example 类的私有方法可以被类和实例调用。")
def show_data(self):
"""
1.调用类的公有方法
2.调用类的受保护的方法
3.调用类的私有方法
4.调用类的公有属性
5.调用类的受保护属性
6.调用类的私有属性
"""
self.func()
self._func()
self.__func()
print(f"Example 类的公有属性: {self.data}")
print(f"Example 类的受保护属性: {self._data}")
print(f"Example 类的私有属性: {self.__data}")
# 测试代码 main 函数
def main():
exp = Example(data1 = 50, data2 = 100, data3 = 200)
exp.show_data()
print()
exp.func()
exp._func()
# exp.__func()
print(exp.data)
print(exp._data)
# print(exp.__data)
if __name__ == "__main__":
main()
Example 类的公有方法可以被类和实例调用。
Example 类的受保护的方法可以被类和实例调用。
Example 类的私有方法可以被类和实例调用。
Example 类的公有属性: 50
Example 类的受保护属性: 100
Example 类的私有属性: 200
Example 类的公有方法可以被类和实例调用。
Example 类的受保护的方法可以被类和实例调用。
50
100
类的实例化
示例 1:
class Example2:
def __init__(self):
print("Example 类的私有方法可以被类和实例调用.")
# 测试代码 main 函数
def main():
# 类 Example2 的实例
exp = Example2()
if __name__ == "__main__":
main()
Example 类的私有方法可以被类和实例调用.
示例 2:
class Example22:
def __init__(self, data1: int, data2: int):
self.data1 = data1
self.data2 = data2
# 测试代码 main 函数
def main():
# 类 Example22 的实例
exp22 = Example22(10, 20)
print(exp22.data1)
if __name__ == "__main__":
main()
10
类的封装
class Example3:
"""
类对数据的封装
"""
def __init__(self, data1: int, data2: int):
self.data1 = data1
self.data2 = data2
# 测试代码 main 函数
def main():
exp = Example3(10, 20)
print(exp.data1)
if __name__ == "__main__":
main()
10
class Example33:
"""
类对逻辑的封装
"""
def __init__(self, data1: int, data2: int):
self.data1 = data1
self.data2 = data2
def add(self):
print(self.data1 + self.data2)
# 测试代码 main 函数
def main():
exp = Example33(10, 20)
exp.add()
if __name__ == "__main__":
main()
30
类的继承
class Father:
def __init__(self, age: int):
self.age = age
print(f"age: {self.age}")
def getAge(self):
return self.age
class Son(Father):
def __init__(self, age: int):
"""
重写父类 Father 的属性
"""
self.age = age
# 测试代码 main 函数
def main():
son = Son(18)
# 子类的实例继承了父类的方法
print(son.getAge())
if __name__ == "__main__":
main()
18
类的私有属性、方法
父类
- 没有下划线的方法或属性,在类的定义中可以调用和访问,类的实例也可以直接访问,
子类也可以访问,能通过
from module import *
的方式导入; - 单下划线的方法或属性,在类的定义中可以调用和访问,类的实例也可以直接访问,
子类也可以访问,不能通过
from module import *
的方式导入; - 双下划线的方法或属性,在类的定义中可以调用和访问,类的实例不可以直接访问,
子类不可以访问,不能通过
from module import *
的方式导入。
class Father0:
"""
父类
"""
# 类属性
__class_private_param = None # 类的私有属性
# TODO _class_protected_param = None # 类的受保护的属性
# TODO class_public_param = None # 类的公有属性
def __init__(self, private_param, protected_param, public_param):
self.__private_param = private_param # 实例的私有属性
# TODO self._protected_param = protected_param # 类、实例的受保护属性
self.public_param = public_param # 实例的公有属性
def __private(self):
print("父类的私有方法 private.")
def _protected(self):
print("父类的受保护的方法 protected.")
def public(self):
print("父类的公有方法 public.")
子类
- 子类双划线方法
__private
方法并没有覆盖父类双下划线__private
方法的权限 - 子类的单下划线方法
_protected
方法是可以覆盖父类的单下划线_protected
方法
class Son0(Father0):
"""
继承了 Father0 的子类
"""
def __private(self):
"""
子类的私有方法, 没有重写父类的 __private
"""
print("子类的重载私有方法 private.")
def _protected(self):
"""
子类的受保护放方法, 重写父类的 _protected
"""
print("子类的重载受保护方法 protected.")
def public(self):
"""
子类的重载公有方法, 重写父类的 public
"""
print("子类的重载公有方法 public.")
类的实例方法、类方法、静态方法
实例方法
- 除了静态方法与类方法外,类的其他方法都属于实例方法;
- 实例方法隐含的参数为类实例
self
; - 实例方法需要将类实例化后才可以调用,如果使用类直接调用实例方法,需要显式地将实例对象作为参数传入;
- 类方法被类和实例调用,实例方法被实例调用,静态方法类和实例都能调用,主要区别在于参数传递上的区别:
- 4.1 实例方法隐藏传递的是类实例
self
引用作为参数 - 4.1 类方法隐藏传递的是
cls
引用作为参数 - 4.3 静态方法无隐含参数
- 4.1 实例方法隐藏传递的是类实例
类方法
- 类方法可以通过类直接调用,或通过实例直接调用. 但无论哪种调用方式,方法隐含的参数为类本身
cls
- 类方法可以实例化对象和类访问,不随实例属性的变化而变化
总结:
- 都是类属性的值,不随实例属性的变化而变化
- 类方法可以被实例对象和类访问
- 类方法则更接近类似 Java 面向对象概念中的静态方法
静态方法
- 静态方法是类中的函数,不需要实例
- 静态方法主要是用来存放逻辑性的代码,主要是一些逻辑属于类, 但是和类本身没有交互,即在静态方法中,不会涉及到类中的方法和属性的操作.
- 可以理解为将静态方法存在此类的名称空间中.
- 事实上,在 Python 引入静态方法之前,通常是在全局名称空间中创建函数
- 关于静态方法只需要记住两方面: 可以被实例对象和类访问; 静态方法直接输出传入方法的值
Python 的静态方法和类方法都可以被类或实例访问,两者概念不容易理清,但还是有区别的:
- 静态方法无需传入
self
参数,类方法需传入代表本类的cls
参数 - 从第 1 条,静态方法是无法访问实例变量的,而类方法也同样无法访问实例变量,但可以访问类变量
- 静态方法有点像函数工具库的作用,而类方法则更接近类似 Java 面向对象概念中的静态方法
示例
class A:
"""
类
"""
sentence = "this is a learning testing"
def __init__(self, instance_param):
"""
类的构造函数(初始化方法)
"""
self.instance_param = instance_param
def normalMethod(self):
"""
普通方法、实例方法
总结:
输出结果随着实例属性改变而改变
"""
print(self.sentence)
@classmethod
def classMethod(cls, sentence):
"""
类方法
"""
print(cls.sentence)
@staticmethod
def staticMethod(sentence):
"""
静态方法
"""
print(sentence)
def test61():
"""
使用实例调用实例方法
1.不改变原始类的属性
2.改变原始类属性
"""
a = A("instance_param")
# 1.不改变原始类的属性
a.normalMethod() # this is a learning testing
# 2.改变原始类属性
sentence = "this is a normalMethod sentence" # 只是修改变量 sentence, 并没有修改类属性
a.normalMethod() # this is a learning testing
a.sentence = "this is a normalMethod sentence" # 修改类属性
a.normalMethod() # this is a normalMethod sentence
def test62():
"""
通过实例、类访问类的静态方法
"""
# 实例对象访问静态方法
a = A("instance_param")
a.staticMethod("static sentence") # static sentence
a.staticMethod("changing sentence") # changing sentence
# 用类直接访问静态方法
A.staticMethod("staticMethod sentence") # staticMethod sentence
A.staticMethod("changing staticMethod sentence") # changing staticMethod sentence
def test63():
"""
通过类、实例访问类方法
"""
# 实例访问类方法
a = A("instance_param")
a.classMethod("classMethod sentence") # this is a learning testing
a.classMethod("changing classMethod sentence") # this is a learning testing
# 类访问类方法
A.classMethod("classMethod sentence") # this is a learning testing
A.classMethod("changing classMethod sentence") # this is a learning testing
子类实例直接调用父类属性、方法
class Father1:
"""
父类
"""
def __init__(self):
self.a = "aaa"
def action(self):
print("调用父类的方法.")
class Son1(Father1):
"""
子类继承父类的属性和方法
"""
pass
def test7():
son1 = Son1()
son1.action()
print(son1.a)
重写父类属性、方法
class Father2():
def __init__(self):
self.a = "aaa"
def action(self):
print("调用父类的方法.")
class Son2(Father2):
def __init__(self):
"""
子类重写父类的属性
"""
self.a = "bbb"
def action(self):
"""
子类重写父类的方法
"""
print("子类重写父类的方法.")
class Son22(Father2):
def __init__(self):
"""
子类重写父类的属性
"""
self.a = "bbb"
def action(self):
"""
子类重写了父类的方法, 子类调用父类的方法
"""
super().action()
def test8():
son2 = Son2()
son2.action()
print(son2.a)
son22 = Son22()
son22.action()
print(son22.a)
强制调用父类私有属性、方法
class Father3():
def __action(self):
print("调用父类的方法.")
class Son3(Father3):
"""
强制调用父类私有属性、方法
"""
def action(self):
super()._Father3__action()
def test9():
son3 = Son3()
son3.action()
调用父类 __init__
方法
class Father4:
def __init__(self, age: int):
self.age = age
print(f"age: {self.age}")
def getAge(self):
print("父类的返回结果:")
return self.age
class Son4(Father4):
"""
显式地调用父类的构造方法
"""
def __init__(self, age: int):
# method 1: 调用父类的 __init__ 方法
super().__init__(age)
# method 2: 调用父类的 __init__ 方法
# Father4.__init__(self, age)
def getAge(self):
print("子类的返回结果:")
return self.age
class Son44(Father4):
"""
不重写父类的构造方法, 实例化子类时, 会自动调用父类定义的构造方法
"""
def getAge(self):
print("子类的返回结果:")
return self.age
def test10():
son4 = Son4(18) # 18
print(son4.getAge()) # 子类返回的结果: 18
son44 = Son44(19)
print(son44.getAge()) # 子列返回的结果: 19
继承父类初始化过程中的参数
class Father5:
def __init__(self):
self.a = 1
self.b = 2
class Son5(Father5):
def __init__(self):
super().__init__()
def add(self):
"""
继承父类初始化过程中的参数
"""
return self.a + self.b
def test11():
son5 = Son5()
print(son5.add())
class Father6:
def __init__(self, a: int, b: int):
self.a = a
self.b = b
def dev(self):
return self.a - self.b
class Son6(Father6):
def __init__(self, a: int, b: int, c: int = 10):
super().__init__(a, b)
self.c = c
def add(self):
return self.a + self.b
def compare(self):
if self.c > (self.a + self.b):
return True
else:
return False
def test111():
son6 = Son6(1, 2)
print(son6.dev())
print(son6.add())
print(son6.compare())
son7 = Son6(1, 2, 1)
print(son7.dev())
print(son7.add())
print(son7.compare())
多继承与 super() 用法
class A:
"""
经典类, 支持多继承
"""
def __init__(self):
print("A")
class B(A):
"""
B 继承 A
"""
def __init__(self):
"""
B 类调用 A 的构造函数方法
"""
A.__init__(self)
print("B")
class C(B, A):
"""
C 继承 B, A
"""
def __init__(self):
"""
C 调用 A, B 的 构造函数方法
"""
A.__init__(self)
B.__init__(self) # 会调用 A 的 __init__ 两次
print("C")
class NewA:
def __init__(self):
print("NewA")
class NewB(NewA):
def __init__(self):
super(NewB, self).__init__()
print("NewB")
class NewC(NewB, NewA):
def __init__(self):
super(NewC, self).__init__()
print("NewC")
本地测试
# 测试代码 main 方法
def main():
test1()
test2()
test22()
test3()
test33()
test4()
test61()
test62()
test63()
test7()
test8()
test9()
test10()
test11()
test111()
if __name__ == "__main__":
main()
类参数
class Person_v1:
def __init__(self, name, gender, **kw):
self.name = name
self.gender = gender
for key, value in kw.items():
setattr(self, key, value)
class Person_v2:
def __init__(self, name, gender, **kw):
self.name = name
self.gender = gender
self.__dict__.update(kw)
p1 = Person_v1(name = "wangzf", gender = "male", age = 18, course = "Python")
p2 = Person_v2(name = "wangzf", gender = "male", age = 18, course = "Python")
print(p1.age)
print(p1.course)
print(p2.age)
print(p2.course)
attr & cattr
# -*- coding: utf-8 -*-
import attr
from attr import s, attributes, attrs
from attr import ib, attr, attrib
from attr import fields, validators, Factory
import cattr
from cattr import structure, unstructure
import typing
# ==========================================
# 基本用法
# ==========================================
class Color_1(object):
"""
Color Object of RGB
R: 0-255
G: 0-255
B: 0-255
RGB(r, g, b)
"""
def __init__(self, r, g, b):
self.r = r
self.g = g
self.b = b
def __repr__(self):
"""
在 Python 里面想要定义某个对象本身的打印输出结果的时候, 需要实现它 __repr__ 方法
"""
return f"{self.__class__.__name__}(r={self.r}, g={self.g}, b={self.b})"
def __eq__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return (self.r, self.g, self.b) == (other.r, other.g, other.b)
def __ne__(self, other):
result = self.__eq__(other)
if result is NotImplemented:
return NotImplemented
else:
return not result
def __lt__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return (self.r, self.g, self.b) < (other.r, other.g, other.b)
def __gt__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return (self.r, self.g, self.b) > (other.r, other.g, other.b)
def __le__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return (self.r, self.g, self.b) <= (other.r, other.g, other.b)
def __ge__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return (self.r, self.g, self.b) >= (other.r, other.g, other.b)
def __hash__(self):
return hash((self.__class__, self.r, self.g, self.b))
@attrs
class Color_2(object):
"""
Color Object of RGB
R: 0-255
G: 0-255
B: 0-255
RGB(r, g, b)
"""
r = attrib(type = int, default = 0)
g = attrib(type = int, default = 0)
b = attrib(type = int, default = 0)
def test_color():
color_1 = Color_1(255, 255, 255)
print(color_1)
color = Color_2(255, 255, 255)
print(color)
# ==========================================
# 声明和比较
# ==========================================
@attrs
class Point_1(object):
"""
Point 数据结构, 包含 x, y 的坐标
"""
x = attrib()
y = attrib()
@attrs
class Point_2(object):
"""
默认值
"""
x = attrib()
y = attrib(default = 100)
@attrs
class Point_3(object):
"""
设置一个初始值, 一直固定不变
"""
x = attrib(init = False, default = 10)
y = attrib(default = 100)
@attrs
class Point_4(object):
"""
强制关键字
"""
x = attrib(default = 0)
y = attrib(kw_only = True)
def to_int(value):
"""
整数类型转换器
Args:
value ([type]): 属性值
Returns:
[type]: [description]
"""
try:
return int(value)
except:
return None
@attrs
class Point_5(object):
"""
转换器
"""
x = attrib(converter = to_int)
y = attrib()
@attrs
class Point_6(object):
"""
类型-原生类型
"""
x = attrib(type = int)
y = attrib()
@attrs
class Point_7(object):
"""
类型-typing 类型、attr 类型
"""
x = attrib(type = int)
y = attrib(type = typing.List[int])
z = attrib(type = attr.Factory(list))
@attrs
class Line(object):
"""
类型-类型嵌套
"""
name = attrib()
points = attrib(type = typing.List[Point_7])
def is_valid_gender(instance, attribute, value):
"""
性别验证器方法
Args:
instance ([type]): 类对象
attribute ([type]): 类属性
value ([type]): 属性名
Raises:
ValueError: [description]
"""
if value not in ["male", "female"]:
raise ValueError(f"gender {value} is not vaild.")
def is_less_than_100(instance, attribute, value):
"""
年龄验证器
Args:
instance ([type]): 类对象
attribute ([type]): 类属性
value ([type]): 属性名
Raises:
ValueError: [description]
"""
if value > 100:
raise ValueError(f"age {value} must less than 100")
@attrs
class Person(object):
"""
验证器
"""
name = attrib()
gender = attrib(validator = is_valid_gender)
age = attrib(validator = [validators.instance_of(int), is_less_than_100])
def test_point():
print("# 声明和比较:")
p1 = Point_1(1, 2)
print(p1)
p2 = Point_1(x = 1, y = 2)
print(p2)
print('Equal:', Point_1(1, 2) == Point_1(1, 2))
print('Not Equal(ne):', Point_1(1, 2) != Point_1(3, 4))
print('Less Than(lt):', Point_1(1, 2) < Point_1(3, 4))
print('Less or Equal(le):', Point_1(1, 2) <= Point_1(1, 4), Point_1(1, 2) <= Point_1(1, 2))
print('Greater Than(gt):', Point_1(4, 2) > Point_1(3, 2), Point_1(4, 2) > Point_1(3, 1))
print('Greater or Equal(ge):', Point_1(4, 2) >= Point_1(4, 1))
print("# 属性定义:")
print(fields(Point_1()))
print(Point_2(x = 1)) # 默认值
print(Point_3(x = 10, y = 3)) # 初始化
print(Point_4(1, y = 3)) # 强制关键字
def test_person():
print(Person(name = "Mike", gender = "male", age = 10))
print(Person(name = "Mike", gender = "mlae", age = 500))
# ==========================================
# 序列转换
# ==========================================
@attrs
class Point(object):
x = attrib(type = int, default = 0)
y = attrib(type = int, default = 0)
def test_convert():
point = Point(x = 1, y = 2)
json_data = cattr.unstructure(point)
print(f"json: {json_data}")
obj_data = cattr.structure(json_data, Point)
print(f"obj: {obj_data}")
def main():
pass
if __name__ == "__main__":
main()