Python-面向对象(下)

目录

一、私有权限

二、私有属性和私有方法

1、私有方法

2、私有属性

三、类部调用私有属性和私有方法

四、子类不能继承父类私有属性和方法

五、修改私有属性的值

六、类属性和实例属性

七、实例属性(对象属性)

八、通过实例(对象)去修改类属性 

九、静态方法和类方法

1、类方法

2、静态方法

3、实例方法

4、实例方法,静态方法,类方法的总结

十、__new__ 方法

十一、单例模式


一、私有权限

封装的意义

将属性和方法放到一起做为一个整体,然后通过实例化对象来处理;

隐藏内部实现细节,只需要和对象及其属性和方法交互就可以了;

对类的属性和方法增加 访问权限控制。

私有权限:在属性名和方法名 前面 加上两个下划线 __

类的私有属性 和 私有方法,都不能通过对象直接访问,但是可以在本类内部访问;

类的私有属性 和 私有方法,都不会被子类继承,子类也无法访问;

私有属性 和 私有方法 往往用来处理类的内部事情,不通过对象处理,起到安全作用。

二、私有属性和私有方法

1、私有方法

代码:

# 私有方法
class Person(object):
     def __init__(self,name,score):
         self.name = name
         self.__score = score
     def __primethod(self):
         print('私有方法')
p = Person('Bob',59)
print(p.name)
print(p.__primethod) #调用私有方法

结果:

2、私有属性

 

代码:

class Person(object):
    def __init__(self,name,score):
        self.name = name
        self.__score = score

p = Person('bay',66)
print(p.name)
print(p.__score)#调用私有属性

结果:

 

三、类部调用私有属性和私有方法

代码:

# 子类通过调用通过实例化方法调用私有属性和方法,不能直接调用
class Person(object):
    def __init__(self, name, score):
        self.name = name
        self.__score = score

    def __primethod(self):
        print("私有方法")

    def method(self):
        return self.__score

    def method1(self):
        return self.__primethod()


p = Person('Bob', 59)
print(p.method())
p.method1()

结果:

 

四、子类不能继承父类私有属性和方法

 

代码1:

#子类不能继承父类私有属性和方法
class Father(object):
    def __init__(self,name):
        self.name = name
        self.__age = 19
    def eat(self):
        print('吃东西')
    def __PlayPingPang(self):
        print('喜欢玩乒乓球')
class Son(Father):
    pass
son = Son('张三')
print(son.name)
print(son.__age)

结果1:

代码2:

class Father(object):
    def __init__(self,name):
        self.name = name
        self.__age = 19
    def eat(self):
        print('吃东西')
    def __PlayPingPang(self):
        print('喜欢玩乒乓球')
class Son(Father):
    pass
son = Son('张三')
son.eat()
son.__PlayPingPang()

结果2:

 

总结一下私有属性和私有方法:

1). 私有属性,可以在类内部通过self调用,但不能通过对象访问

2). 私有方法,可以在类内部通过self调用,但不能通过对象访问

3). 对象不能访问私有权限的属性和方法

4). 子类不能继承父类私有权限的属性和方法

5). Python中没有像C++中 public 和 private, protected 这些关键字来区别公有属性和私有属性。

6). Python是以属性命名方式来区分,如果在属性和方法名前面加了2个下划线'__',则表明该属性和方法是私有权限,否则为公有权限。

 

五、修改私有属性的值

 

如果需要修改一个对象的属性值,通常有2种方法

1、对象名.属性名 = 数据 ----> 直接修改

2、对象名.方法名() ----> 间接修改

私有属性不能直接访问,所以无法通过第一种方式修改,一般的通过第二种方式修改私有属性的值:定义一个可以调用的公有方法,在这个公有方法内访问修改。

代码:

class People(object):
    def __init__(self):
        self.__money = 1000

    def set_money(self, money):
        self.__money = money

    def get_money(self):
        return self.__money

p = People()
p.set_money(2000)
print(p.get_money())

结果:

总结:

现代软件开发中,通常会定义get_xxx()方法和set_xxx()方法来获取和修改私有属性值

get_xxx()方法-->返回私有属性的值

set_xxx()方法-->接收参数,修改私有属性的值

对象不能访问私有权限的属性和方法,可以通过访问公有方法set_money()来修改私有属性的值,可以通过访问公有方法get_money()来获取私有属性的值

 

六、类属性和实例属性

在前面的例子中我们接触到的就是实例属性(实例对象属性),顾名思义,类属性就是类对象所拥有的属性,它被所有类对象的实例对象所共有,在内存中只存在一个副本,这个和C++中类的静态成员变量有点类似。对于公有的类属性,可以通过类或者实例对象访问

实例属性只能通过对象来调用,类不能调用

代码:

class People(object):
    name = 'Tom'  # 公有的类属性
    __age = 12  # 私有的类属性

p = People()
print(p.name)  # 正确
print(People.name)  # 正确
print(p.__age)  # 错误,不能在类外通过实例对象访问私有的类属性
print(People.__age)  # 错误,不能在类外通过类对象访问私有的类属性

结果:

可以通过类或者实例对象调用

 

七、实例属性(对象属性)

 

代码:

class People(object):
    address = '山东'  # 类属性
    def __init__(self):
        self.name = 'xiaowang'  # 实例属性
        self.age = 20  # 实例属性
p = People()
p.age = 12  # 实例属性
print(p.address)  # 正确
print(p.name)  # 正确
print(p.age)  # 正确
print(People.address)  # 正确
print(People.name)  # 错误
print(People.age)  # 错误

结果:

可以通过实例化对象调用,类不能调用 

 

八、通过实例(对象)去修改类属性 

 

代码:

class People(object):
    country = 'china' #类属性
print(People.country)
p = People()
print(p.country)
p.country = 'japan' 
print(p.country)  # 实例属性会屏蔽掉同名的类属性
print(People.country)
del p.country  # 删除实例属性
print(p.country)

结果:

总结:

对象修改类属性,只对本对象有效果,对别的 对象没有影响

 

九、静态方法和类方法

1、类方法

是类对象所拥有的方法,需要用修饰器@classmethod来标识其为类方法,对于类方法,第一个参数必须是类对象,一般以cls作为第一个参数(当然可以用其他名称的变量作为其第一个参数,但是大部分人都习惯以'cls'作为第一个参数的名字,就最好用'cls'了),能够通过实例对象和类对象去访问。

代码:

class People(object):
    # 类属性
    age =18
    #类方法,用classmethod来进行修饰
    @classmethod
    def get_country(cls):
        return cls.age
p = People()
print(p.get_country())    #可以用过实例对象引用
print(People.get_country())    #可以通过类对象引用

结果:

类方法还有一个用途就是可以对类属性进行修改:

代码:

class People(object):
    # 类属性
    age= 18
    #类方法,用classmethod来进行修饰
    @classmethod
    def get_country(cls):
        return cls.age
    @classmethod
    def set_country(cls,age):
        cls.age = age
p = People()
print(p.get_country())   #可以用过实例对象访问
print(People.get_country())    #可以通过类访问
p.set_country(23)
print(p.get_country())
print(People.get_country())
p1 = People()
print(p1.get_country())

结果:

 结果显示在用类方法对类属性修改之后,通过类对象和实例对象访问都发生了改变(全部改变)

 

2、静态方法

需要通过修饰器@staticmethod来进行修饰,静态方法不需要多定义参数,可以通过对象和类来访问。

代码:

class People(object):
    country = 'china'
    @staticmethod
    #静态方法
    def get_country():
        return People.country
p = People()
# 通过对象访问静态方法
print(p.get_country())
# 通过类访问静态方法
print(People.get_country())

结果:

静态方法中不需要额外定义参数,因此在静态方法中引用类属性的话,必须通过类实例对象来引用,调用静态方法可以通过对象或者类调用

 

3、实例方法

实例方法中的第一个参数是self,只能通过对象来访问。

代码:

class People(object):
    def selfmethod(self):
        print("我是实例方法")
p = People()
p.selfmethod()
#People.selfmethod()    #报错

结果:

实例方法中需要self参数,因此调用实例方法只能通过实例对象调用 也可以通过类调用但是一般不这样用

 

4、实例方法,静态方法,类方法的总结

类方法使用@classmethod装饰,第一个参数为类(cls),调用时可以通过类的实例或者类本身来调用。 实例方法定义时第一个参数为类的一个实例(self),调用时必须通过实例调用。 静态方法使用@staticmethod装饰,调用时可以使用类的实例或者类本身来调用。

 

十、__new__ 方法

代码:

class A(object):
    def __init__(self):
        print('这是init方法')

    def __new__(cls):
        print('这是new方法')
        return object.__new__(cls)

a = A()

结果:

总结:

1). __new__至少要有一个参数cls,代表要实例化的类,此参数在实例化时由Python解释器自动提供

2). __new__必须要有返回值,返回实例化出来的实例,这点在自己实现__new__时要特别注意,可以return父类__new__出来的实例,或者直接是object的__new__出来的实例

3). __init__有一个参数self,就是这个__new__返回的实例,__init__在__new__的基础上可以完成一些其它初始化的动作,__init__不需要返回值

4). 我们可以将类比作制造商,__new__方法就是前期的原材料购买环节,__init__方法就是在有原材料的基础上,加工,初始化商品环节

注意点:

代码:

#__new__ 和 __init__ 的作用
class A(object):
    def __init__(self):
        print('*'*40)
        print(self)
        print('这是init方法')

    def __new__(cls):
        print(id(cls))
        print('这是new方法')
        ret = object.__new__(cls)
        print(ret)
        return ret

a = A()

结果:

 

十一、单例模式

 

单例模式:

永远用一个对象得实例,避免新建太多实例浪费资源    

实质:

使用__new__方法新建类对象时先判断是否已经建立过,如果建过就使用已有的对象

代码:

class Foo(object):
   instance = None
   def __init__(self):
      self.name = 'alex'
   def __new__(cls):
      if Foo.instance:
         return Foo.instance
      else:
         Foo.instance = object.__new__(cls)
         return Foo.instance
obj1 = Foo()
obj2 = Foo()
print(obj1,obj2

结果:

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值