第八节(1) 面对对象初步(2)

方法没有重载&方法的动态性

方法没有重载

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

方法的动态性

Python 是动态语言,我们可以动态的为类添加新的方法,或者动态的修改类的已有的方法。

#测试方法的动态性
class Person:
    def work(self):
        print("努力上班!")
def play_game(self):
    print("{0}玩游戏".format(self))
def work2(s):
    print("好好工作,努力上班!")

Person.play = play_game
Person.work = work2
p = Person()
p.play()

result:
<__main__.Person object at 0x7fe60a48f940>玩游戏

我们可以看到,Person 动态的新增了 play_game 方法,以及用 work2 替换了 work 方法。


私有属性和私有方法(实现封装)

Python 对于类的成员没有严格的访问控制限制,这与其他面向对象语言有区别。关于私有 属性和私有方法,有如下要点:

  1. 通常我们约定,两个下划线开头的属性是私有的(private)。其他为公共的(public)。
  2. 类内部可以访问私有属性(方法)
  3. 类外部不能直接访问私有属性(方法)
  4. 类外部可以通过“_类名__私有属性(方法)名”访问私有属性(方法)

【注】方法本质上也是属性!只不过是可以通过()执行而已。所以,此处讲的私有属性和公 有属性,也同时讲解了私有方法和公有方法的用法。如下测试中,同时也包含了私有方法和 公有方法的例子。

class Employee:
    __company = "xxx" #私有类属性. 通过 dir 可以查到

    def __init__(self, name, age):
        self.name = name
        self.__age = age  # 私有实例属性

    def say_company(self):
        print("我的公司是:", Employee.__company)  # 类内部可以直接访问私有属性
        print(self.name, "的年龄是:", self.__age)
        self.__work()

    def __work(self):  ##私有实例方法 通过 dir 可以查到 _Employee__work
        print("工作!好好工作!")

p1 = Employee("Joshua", 30)
print(p1.name)
print(dir(p1))
p1.say_company()
print(p1._Employee__age) #通过这种方式可以直接访问到私有属性 。通过 dir 可以查到属性:_Employee__age
#print(p1.__age) #直接访问私有属性,报错
#p1.__work() #直接访问私有方法,报错

result:
Joshua
['_Employee__age', '_Employee__company', '_Employee__work', '__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__', 'name', 'say_company']
我的公司是: xxx
Joshua 的年龄是: 30
工作!好好工作!
30

@property 装饰器

@property 可以将一个方法的调用方式变成“属性调用”。下面是一个简单的示例,让大 家体会一下这种转变:

class Employee:
    @property
    def salary(self):
        return 30000;

emp1 = Employee()
print(emp1.salary)  # 打印 30000
print(type(emp1.salary))  # 打印<class 'int'>

#emp1.salary() # 报错:TypeError: 'int' object is not callable

# emp1.salary =1000 # @property 修饰的属性,如果没有加setter方法,则为只读属性。此处修改报错: AttributeError: can't set attribute

@property 主要用于帮助我们处理属性的读操作、写操作。对于某一个属性,我们可以直 接通过:

  emp1.salary = 30000

如上的操作读操作、写操作。但是,这种做法不安全。比如,我需要限制薪水必须为 1-10000 的数字。这时候,我们就需要通过 getter、setter 方法来处理。

class Employee:
    def __init__(self,name,salary):
        self.name = name
        self.__salary = salary

    @property #相当于 salary 属性的 getter 方法
    def salary(self):
        print("月薪为{0},年薪为 {1}".format(self.__salary,(12*self.__salary)))
        return self.__salary;

    @salary.setter
    def salary(self, salary):  # 相当于 salary 属性的 setter 方法
        if (0 < salary < 1000000):
            self.__salary = salary
        else:
            print("薪水录入错误!只能在 0-1000000 之间")

emp1 = Employee("高淇", 100)
print(emp1.salary)
emp1.salary = -200

result:
月薪为100,年薪为 1200
100
薪水录入错误!只能在 0-1000000 之间

属性和方法命名总结

1 _xxx:保护成员,不能用“from module import * ”导入,只有类对象和子类对象能访 问这些成员。

2 xxx:系统定义的特殊成员

3 __xxx: 类中的私有成员,只有类对象自己能访问,子类对象也不能访问。(但,在类外 部可以通过“对象名. _类名__xxx”这种特殊方式访问。Python 不存在严格意义的私有成员)

注:再次强调,方法和属性都遵循上面的规则。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值