内置函数属性
1.声明是内置类属性
声明类的时候系统自动添加的属性(可能是字段也可能是对象属性)
class Person:
"""
说明文档:人类
num:人的数量
"""
num = 61
def __init__(self,name,gender,age):
self.name = name
self.gender = gender
self.age = age
def eat(self,food):
print('{}在吃{}'.format(self.name,food))
定制当前类的对象的打印
1)重写str方法,这个方法的返回值就是对应的打印结果(类型必须是字符串),只能单独打印
def __str__(self):
return str(self.__dict__)
2)重写repr方法,这个方法的返回值就是对应的打印结果(类型必须是字符串),可以打印列表等
def __repr__(self):
return str(self.__dict__)
p1=Person('小明','男',18)
1.name
类的字段;类名.name - 获取类的名字(字符串)
print(type(Person),Person) #
print(type(int)) #
print(Person.__name__) # Person
2.doc
获取类的说明文档
print(Person.__doc__)
3.class
对象属性;对象.class - 获取对象对应的类,返回的是类
(和type(对象)功能一样)
print(p1.__class__) #
print(type(p1)) #
type1 = p1.__class__ # 拿到的是p1这个对象的类
print(type1.__name__) # Person
4.dict(将对象转换成字典)
对象属性:对象.dict - 将对象中所有的属性和对应的值转换成一个字典中的键值对(一个对象对应一个字典)
类的字段:类.dict - 将类转换成一个字典,字典中的元素是类中所有的字段和对应的值
注意:slots和dict不可同时使用,会报错,一般我们选择放弃slots
print(p1.__dict__) # {'name': '小明', 'gender': '男', 'age': 18}
print(Person.__dict__) # {'__module__': '__main__', '__doc__': '\n 说明文档:人类\n num:人的数量\n ', 'num': 61,
# '__init__': , 'eat': ,
# '__dict__': , '__weakref__': }
print([p1,p1])
5.module
类的字段;类.module - 获取当前类是在哪个模块中声明的(返回的是模块的名字)
6.bases
类的字段:类.bases - 获取当前类的父类(返回的是一个元祖)
print(Person.__bases__) # (,)
练习:将字典转换成对象
class Data:
# 将字典作为属性添加,然后将键值对添加为属性
def __init__(self,dict1:dict):
for key in dict1:
setattr(self,key,dict1[key])
def __repr__(self):
return str(self.__dict__)
私有化
1.访问权限:公开、保护、私有
公开 - 公开的属性和方法在类的内部、外部狗能使用也能被继承
保护 - 保护的属性和方法在类的内部能够使用,外部不能使用,但是可以被继承
私有 - 私有的属性和方法只能在类的内部使用,外部不能使用,也不能被继承
2.python中属性和方法的访问权限
python类中所有的属性和方法本质都是公开的;
私有化是假私有化,只是提示程序员这个属性或者方法在外部不能使用,也不要去继承
怎么私有化:在需要私有化的属性名或者方法名前加''(注意:不能以结尾)
python私有化的原理:在私有的属性和方法前加了'_类名'
class Person:
num = 61
__num1 = 54 # 私有化
def __init__(self, name, age):
self.name = name
self.__age = age
def eat(self, food='米饭'):
print('{}在吃{}'.format(self.name, food))
print('内部:', Person.num) # 内部: 61
print('内部:', Person.__num1) # 内部: 54
p1 = Person('小明', 21)
print(Person.num) # 61
p1.eat() # 小明在吃米饭
# print(Person.__num) # AttributeError: type object 'Person' has no attribute '__num'
# 私有化属性的获取(实质上的私有就是换了个名字,所以无法直接取到)
print(p1.__dict__) # {'name': '小明', '_Person__age': 21}
print(p1._Person__age) # 21
属性的getter和setter
1.什么是getter和setter
当我们需要在获取属性值之前做点别的事情就需要给这个属性添加getter;
当需要给属性赋值之前做别的事情就需要给这个属性添加setter
2.给属性添加getter
1)属性命名的时候前面加一个下划线(对类本身无影响,主要是标识这个属性要添加getter或者setter)
2)在装饰器@property的后面声明一个对象方法
a.将属性去掉下划线作为方法名
b.方法除了self以外不需要其他参数
c.函数的返回值就是获取这个属性的时候得到的值
3)在外部使用属性的时候,通过'对象,不带下划线的'
注意:获取属性值的时候,就会自动去调用getter对应的函数
可以指定返回值、做提醒、做警告等
3.给属性添加setter
属性添加setter之前必须先添加getter
1)保证属性名前有一个下划线
2)在装饰器@属性名.setter的后面声明一个对象方法
a.将属性去掉下划线作为方法名
b.需要一个self以为的参数
c.不需要返回值
3)在外部使用属性的时候,通过'对象,不带下划线的属性’去使用
注意:当给属性赋值的时候,实质是调用setter对应的方法
可以修改属性,结合if raise 可以做判断,或者变成只读(一旦检测到修改操作就报错raise)
class Person:
def __init__(self, name, age, gender):
self.name = name
self._age = age
self.gender = gender
self._week = 0
# 添加getter
@property
def week(self):
if self._week == 0:
return '星期天'
else:
return '不是星期天'
@property
def age(self):
print('年龄值被访问!!!')
return self._age
# 添加setter
@age.setter
def age(self, value):
print('年龄被修改')
self._age = value
# p1 = Person('小明', 18, '男')
# print(p1.week) # 星期天 - 这儿实质实质爱调用week方法获得返回值
# print(p1.age) # 年龄值被访问!!! 18
# p1.age = 20 # 年龄被修改
# print(p1.age) # 年龄值被访问!!! 20
练习:写一个矩形类
有属性:长、宽、面积、周长
要求:从生活的角度看这个矩形
class Rec:
def __init__(self,length,wide):
self._length = length
self._wide = wide
self._area = length*wide
self.per = 2*(length+wide)
@property
def length(self):
return self._length
@length.setter
def length(self,value):
if not isinstance(value,int) or isinstance(value,float):
raise ValueError
if value <0:
raise ValueError
self._length = value
@property
def area(self):
self._area = self._length * self._wide
return self._area
@area.setter
def area(self,value):
print('休想修改面积')
raise ValueError
r1 = Rec(3, 4)
# r1.area = 12
r1.length = 14
print(r1.area)
类方法和静态方法
1.类中的函数
类中的方法分为:对象方法、类方法和静态方法
1)对象方法
a.怎么声明:直接声明
b.怎么调用:用对象来调用
c.特点:有指向当前对象的self
d.什么时候用:如果实现函数的功能需要用到对象属性,就使用对象方法
2)类方法
a.怎么声明:声明在@classmethod后面
b.怎么调用:通过类来调用(类.类方法)
c.特点:有自带的参数cls,表示当前类,这个参数在调用的时候不用传参,系统会自动将当前类传给它
cls:谁调用就指向谁(如果是对象指向的是对象对应的类)
d.什么时候用:如果实现函数的功能不需要对象属性,但需要类的字段(严格来说是需要类),就是使用类方法
3)静态方法
a.怎么声明:声明在@staticmethod后面
b.怎么调用:通过类来调用(类.静态方法)
c.特点:没有默认参数
d.什么时候用:如果实现函数的功能既不需要对象属性,也不需要类,就使用静态方法
class Person:
num = 61
def __init__(self,name, age, gender):
self.name = name
self.age = age
self.gender = gender
def eat(self,food):
# 对象能做的事情,self都能做
print(self.name+'在吃'+food)
@classmethod
def func1(cls):
# 类能做的事情,cls都能做
print('这是一个类方法',cls.num)
@staticmethod
def func2():
print('这是一个静态方法')
Person.func1() # 这是一个类方法 61
Person.func2() # 这是一个静态方法
继承
1.什么是继承
继承就是让子类直接拥有父类的属性和方法
子类 - 继承者
父类/超类 - 被继承者
2.怎么继承
1)语法
class 类名(父类1,父类2,父类...):
说明文档
类的内容
2)说明:
() - 固定写法,如果省略,相当于(object)
声明类的时候如果没有写父类,默认继承object(object又叫基类)
父类 - 一个类的父类可以有一个或多个,但是一般情况下只有一个
class Person:
num = 61
def __init__(self,name, age, gender):
self.name = name
self.age = age
self.gender = gender
def eat(self,food):
print('{}在吃{}'.format(self.name,food))
def func1(self):
print('这是父类')
def func2(self):
print('父类的功能2')
@classmethod
def show(cls):
print('人类的数量{}'.format(cls.num))
class Student(Person):
num = 10
def __init__(self,name, age, gender):
# 在子类中添加对象属性,需要先通过super()去调用父类的__init__来继承父类
super().__init__(name, age, gender)
self.id = '001'
self.class1 = 'py1904'
# 添加功能
def study(self):
print(self.name + '在认真敲代码')
# 功能重写
def func1(self):
print('重写覆盖父类')
def func2(self):
print('子类的功能2')
# 在子类中可以通过super()可以去调用父类的方法
# 注意:super()只能在对象方法和类方法中使用,因为静态方法找不到父类
super().func2()
s1 = Student('小米',32,'女')
s1.eat('菠萝')
s1.func2()