Python打卡十四——类

Python打卡(十四)

1、简单的类结构和构造函数

'''
    类中的函数必须使用self参数,并且次参数必须位于第一位
    注意:
    1、构造函数固定格式为__init__(self)
    2、创建对象,我们需要定义构造函数__init__()方法。构造方法用于执行“实例对象的初始化工
       作”,即对象创建后,初始化当前对象的相关属性,无返回值。
    3、如果我们不定义__init__方法,系统会提供一个默认的__init__方法。如果我们定义了带参
       的__init__方法,系统不创建默认的__init__方法。
    4、在使用类似 s1 = Student("任嘉伦",31) 的语句后系统会自动调用__new__()函数和__init__()
       两个函数,但是__new__()函数无需自己定义
'''
class Student:
    def __init__(self,name,age):
        self.name = name
        self.age = age
    def see_age(self):
        print("该生姓名为{0},年龄为{1}".format(self.name,self.age))

s1 = Student("任嘉伦",31)
s1.see_age()

运行结果:

该生姓名为任嘉伦,年龄为31

2、实例属性和实例方法

'''
    实例属性是从属于实例对象的属性,也称为“实例变量”。他的使用有如下几个要点:
    1. 实例属性一般在__init__()方法中通过如下代码定义:
        self.实例属性名 = 初始值
    2. 在本类的其他实例方法中,也是通过 self 进行访问:
        self.实例属性名
    3. 创建实例对象后,通过实例对象访问:
        obj01 = 类名() #创建对象,调用__init__()初始化属性
        obj01.实例属性名 = 值 #可以给已有属性赋值,也可以新加属性

    实例方法:(从属于实例属性)
    a = Student()
    a.say_score() ——————> 解释器翻译为:Student.say_score(a)
'''
class Student:
    def __init__(self,name,age):
        self.name = name       #name和age均为示例属性
        self.age = age
    def see_age(self):         #see_age为实例方法
        print("该生姓名为{0},年龄为{1}".format(self.name,self.age))

s1 = Student("任嘉伦",31)
s1.see_age()

s1.salary = 3000  #创建完对象之后可以为该对象增加实例属性
print(s1.salary)

Student.see_age(s1)   #与s1.see_age()结果一致

s2 = Student("肉骨茶",21)
print(s2.salary)   #s2对象没有salary属性,因为类这个模具中并未定义salary属性

运行结果:

该生姓名为任嘉伦,年龄为31
3000
该生姓名为任嘉伦,年龄为31
Traceback (most recent call last):
  File "D:/Python/project/studyPython/Object_Oriented/shilishuxing.py", line 31, in <module>
    print(s2.salary)   #s2对象没有salary属性,因为类这个模具中并未定义salary属性
AttributeError: 'Student' object has no attribute 'salary'

其他操作:

'''
其他操作:
    1. dir(obj)可以获得对象的所有属性、方法
    2. obj.__dict__ 对象的属性字典
    3. pass 空语句
    4. isinstance(对象,类型) 判断“对象”是不是“指定类型”
'''

print(dir(s1))
print(s2.__dict__)

class Man:
    pass

print(isinstance(s1,Student))
print(isinstance(s1,Man))

运行结果:

['__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'age', 'name', 'salary', 'see_age']
{'name': '肉骨茶', 'age': 21}
True
False

3、类对象、类属性、类方法

(1)类对象

'''
    类对象是针对模具本身来说的,当解释器执行class语句时,就会创建一个类对象
    一个类只有一个类对象
'''
class Student:
    def __init__(self,name,age):
        self.name = name
        self.age = age

    def say_age(self):
        print("{0}的年龄是{1}".format(self.name,self.age))

stu2 = Student
s1 = Student("任嘉伦",31)
s1.say_age()
s2 = stu2("肉骨茶",21)
s2.say_age()

运行结果:

任嘉伦的年龄是31
肉骨茶的年龄是21

(2)类属性

'''
    类属性是从属于“类对象”的属性,也称为“类变量”。由于,类属性从属于类对象,可以被所有实例对象共享。
    类属性的定义方式:
        class 类名:
            类变量名= 初始值
    在类中或者类的外面,我们可以通过:“类名.类变量名”来读写
'''

class Student:
    company = "BaiDu"  # 类属性
    count = 0  # 类属性

    def __init__(self, name, score):
        self.name = name  # 实例属性
        self.score = score
        Student.count = Student.count + 1

    def say_score(self):  # 实例方法
        print("我的公司是:", Student.company)
        print(self.name, '的分数是:', self.score)

s1 = Student('张三', 80)  # s1 是实例对象,自动调用__init__()方法
s1.say_score()
s2 = Student('李四', 90)
s3 = Student('王五', 20)
s4 = Student('赵柳', 150)
print('一共创建{0}个 Student 对象'.format(Student.count))

运行结果:

我的公司是: BaiDu
张三 的分数是: 80
一共创建4个 Student 对象

内存分析:
在这里插入图片描述
(3)类方法

'''
    类方法是从属于“类对象”的方法。类方法通过装饰器@classmethod 来定义,格式如下:
    @classmethod
    def 类方法名(cls [,形参列表]) :
        函数体
    要点如下:
    1. @classmethod 必须位于方法上面一行
    2. 第一个 cls 必须有;cls 指的就是“类对象”本身;
    3. 调用类方法格式:“类名.类方法名(参数列表)”。 参数列表中,不需要也不能给 cls 传值。
    4. 类方法中访问实例属性和实例方法会导致错误
    5. 子类继承父类方法时,传入 cls 是子类对象,而非父类对象
'''
class Student:

    company = "BaiDu"

    def __init__(self,name,age):
        self.name = name
        self.age = age

    def say_score(self):  # 实例方法
        print("我的公司是:", Student.company)
        print(self.name, '的年龄是:', self.age)

    @classmethod
    def printcompany(cls):
        print(cls.company)
        #print(self.name) #类方法中访问实例属性和实例方法会导致错误
        #print(self.say_score())   #类方法中访问实例属性和实例方法会导致错误

Student.printcompany()
s1 = Student("ll",23)
Student.say_score(s1)

运行结果:

BaiDu
我的公司是: BaiDu
ll 的年龄是: 23

(4)静态方法(与本类无关的函数)

'''
    Python 中允许定义与“类对象”无关的方法,称为“静态方法”。
    “静态方法”和在模块中定义普通函数没有区别,只不过“静态方法”放到了“类的名字空间里面”,需要通过“类调用”。

    静态方法通过装饰器@staticmethod 来定义,格式如下:
    @staticmethod
    def 静态方法名([形参列表]) :
        函数体

    要点如下:
    1. @staticmethod 必须位于方法上面一行
    2. 调用静态方法格式:“类名.静态方法名(参数列表)”。
    3. 静态方法中访问实例属性和实例方法会导致错误

'''
class Student:
    company = "SXT" # 类属性

    @staticmethod
    def add(a, b): # 静态方法
        print("{0}+{1}={2}".format(a,b,(a+b)))
        return a+b

Student.add(20,30)

运行结果;

20+30=50

4、析构方法

'''
        __del__方法称为“析构方法”,用于实现对象被销毁时所需的操作。比如:释放对象
    占用的资源,例如:打开的文件资源、网络连接等。

    Python 实现自动的垃圾回收,当对象没有被引用时(引用计数为 0),由垃圾回收器调用__del__方法。

    我们也可以通过 del 语句删除对象,从而保证调用__del__方法。

    系统会自动提供__del__方法,一般不需要自定义析构方法。
'''

class Test:

    def __del__(self):
        print("释放对象{0}".format(self))

t1 = Test()
t2 = Test()
del t2
print("程序结束")

运行结果:

释放对象<__main__.Test object at 0x000002174F7D5A48>
程序结束
释放对象<__main__.Test object at 0x000002174F7D28C8>

5、可调用对象

'''

    定义了__call__方法的对象,称为“可调用对象”,即该对象可以像函数一样被调用。

'''
class Salary:

    def __call__(self, salary):
        print("算工资啦!!!")
        yearsalary = salary * 12
        daysalary = salary // 22.75
        hoursalary = daysalary // 8

        return dict(yearsalary=yearsalary,monthsalary=salary,daysalary=daysalary,hoursalary=hoursalary)

s = Salary()
print(s(30000))

运行结果:

算工资啦!!!
{'yearsalary': 360000, 'monthsalary': 30000, 'daysalary': 1318.0, 'hoursalary': 164.0}

6、Python中的方法没有重载

'''

        在其他语言中,可以定义多个重名的方法,只要保证方法签名唯一即可。方法签名包含 3个
    部分:方法名、参数数量、参数类型。

        Python 中,方法的的参数没有声明类型(调用时确定参数的类型),参数的数量也可以由
    可变参数控制。因此,Python 中是没有方法的重载的。定义一个方法即可有多种调用方式,相
    当于实现了其他语言中的方法的重载。

        如果我们在类体中定义了多个重名的方法,只有最后一个方法有效。

        建议:不要使用重名的方法!Python 中方法没有重载。

'''
#在Python中定义多个同名方法,只有最后一个方法有效

class Person:

    def say_hi(self):
        print("Hello!")

    def say_hi(self,name):
        print("Hello {0}!".format(name))

p1 = Person()
#p1.say_hi()     #不带参数报错:  TypeError: say_hi() missing 1 required positional argument: 'name'
p1.say_hi("任嘉伦")

运行结果:

Hello 任嘉伦!

7、方法的动态性

'''
    Python 是动态语言,我们可以动态的为类添加新的方法,或者动态的修改类的已有的方法。
'''
class Person:
    def work(self):
        print("好好上班,挣钱养家!!!")

def play_game(self):
    print("事都干完啦!还玩?还玩?还玩?")

def work2(s):
    print("工作做完啦!!")

p1 = Person()
p1.work()
#p1.play_game()  #此时p1没有play_game()这个方法属性   AttributeError: 'Person' object has no attribute 'play_game'
Person.play = play_game
Person.work = work2
p = Person()
p.play()
p.work()

运行结果:

好好上班,挣钱养家!!!
事都干完啦!还玩?还玩?还玩?
工作做完啦!!
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值