一、封装

1.1封装的定义

在程序设计中,封装是对具体对象的一种抽象,即将某些部分隐藏起来,在程序外部看不到,其含义是其他程序无法调用。

要了解封装,离不开“私有化”,就是将类或者是函数中的某些属性限制在某个区域之内,外部无法调用。

1.2封装的用处

封装数据的主要原因是:保护隐私(把不想别人知道的东西封装起来)

封装方法的主要原因是:隔离复杂度

注意:在编程语言里,对外提供的接口(接口可理解为了一个入口),就是函数,称为接口函数,这与接口的概念还不一样,接口代表一组接口函数的集合体。

 

1.3封装的两个层面

(1)类就是麻袋,这本身就是一种封装

(2)类中定义私有的,只有类的内部使用,外部无法访问

 

class People:
    _star='earth'     #单下划线表示被隐藏起来
    def __init__(self,name,id,age,salary):
        self.name=name
        self.id=id
        self.age=age
        self.salary=salary

    def get_id(self):
        print('[%s的身份证号是%s]'%(self.name,self.id))

p1=People('czd','62012393208',18,100000)
print(p1._star)




class People:
    __star='earth'          #双下划綫
    def __init__(self,name,id,age,salary):
        self.name=name
        self.id=id
        self.age=age
        self.salary=salary

    def get_id(self):
        print('[%s的身份证号是%s]'%(self.name,self.id))

p1=People('czd','62012393208',18,100000)
# print(p1.__star)      #报错

print(p1._People__star)

 

  这种自动变形的特点:

    1、类中定义的__star只能在内部使用,如品p1.__star,引用的就是变形的结果。

    2、这种变形其实正是针对外部的变形,在外部是无法通过__star这个名字访问到的。

    3、在子类定义的__star不会覆盖在父类定义的__star,因为子类中变形成了:_子类名__star,而父类中变形成了:_父

类名__star,即双下滑线开头的属性在继承给子类时,子类是无法覆盖的。

  注意:对于这一层面的封装(隐藏),我们需要在类中定义一个函数(接口函数)在它内部访问被隐藏的属性,然后

外部就可以使用了

二、多态

2.1多态的概念

多态的特性是调用不同的子类将会产生不同的行为,而无需明确知道这个子类实际上是什么。

多态实际上是依附于继承的两种含义:"改变"和"扩展"本身就意味着必须有机制去选用你改变/扩展过的版本,多态实质上就是继承的实现细节

子类实例化后调用基类的方法,w1.turn_ice()叫做多态;

Python 函数怎么封装参数 python封装函数接口_子类

Python 函数怎么封装参数 python封装函数接口_封装_02

class H2O:
    def __init__(self,name,temperature):
        self.name=name
        self.temperature=temperature
    def turn_ice(self):
        if self.temperature<0:
            print('[%s]温度太低结冰了'%self.name)
        elif self.temperature>0 and self.temperature<100:
            print('[%s]液化成水'%self.name)
        elif self.temperature>100:
            print('[%s]温度太高变为水蒸气'%self.name)

class Water(H2O):
    pass
class Ice(H2O):
    pass
class Steam(H2O):
    pass

w1=Water('水',25)

i1=Ice('冰',-20)

s1=Steam('蒸汽',200)

# w1.turn_ice()     #执行过程中不同的对象调用相同的方法
# i1.turn_ice()
# s1.turn_ice()

#或者定义一个接口来执行上面的调用
def func(obj):
    obj.turn_ice()

func(w1)   #---->w1.turn_ice()
func(i1)
func(s1)

View Code

三、反射

3.1反射的介绍

反射说简单点 --> 就是利用字符串的形式去对象(模块)中操作(寻找/检查/设置/删除)成员。

  • hasattr(object,"name")
  • getattr(object,"name",default=None)
  • setattr(x,y,v)
  • delattr(x,y)

例如:

Python 函数怎么封装参数 python封装函数接口_子类

Python 函数怎么封装参数 python封装函数接口_封装_02

class BlackMedium:
    feture = 'Ugly'
    def __init__(self,name,addr):
        self.name=name
        self.addr=addr

    def sell_house(self):
        print('[%s]正在卖房子,傻逼才买呢'%self.name)

    def rent_house(self):
        print('[%s]正在租房子,傻逼才租呢'%self.name)

b1=BlackMedium('万成置地','天露园')

print(hasattr(b1,'name'))
print(hasattr(b1,'sell_house'))   #显示True
print(hasattr(b1,"sell_houseaaa"))#显示False


print(getattr(b1,'name'))
print(getattr(b1,'rent_house'))
func=getattr(b1,'rent_house')
func()
# print(getattr(b1,'rent_houseaaa'))  #没有则报错
print(getattr(b1,'rent_houseaaa','没有这个属性'))

#数据属性
setattr(b1,'sb','True')    #相当于b1.sb=True
setattr(b1,'123','czd')
setattr(b1,'name','fuzhou')
#函数属性
setattr(b1,'func',lambda x:x+1)
setattr(b1,'func1',lambda self:self.name+'sb')
print(b1.__dict__)
print(b1.func)
print(b1.func(1))
print(b1.func1(b1))

delattr(b1,'sb')         #相当于del b1.sb
print(b1.__dict__)

代码示范

3.2 反射的好处

(1)好处一:实现可插拔机制

有俩程序员,一个czd,一个是egon,lili在写程序的时候需要用到egon所写的类,但是egon去跟女朋友度蜜月去了,还没有完成他写的类,lili想到了反射,使用了反射机制czd可以继续完成自己的代码,等egon度蜜月回来后再继续完成类的定义并且去实现czd想要的功能。

总之反射的好处就是,可以事先定义好接口,接口只有在被完成后才会真正执行,这实现了即插即用,这其实是一种‘后期绑定’,什么意思?即你可以事先把主要的逻辑写好(只定义接口),然后后期再去实现接口的功能

Python 函数怎么封装参数 python封装函数接口_子类

Python 函数怎么封装参数 python封装函数接口_封装_02

class FtpClient:
    'ftp客户端,但是还么有实现具体的功能'
    def __init__(self,addr):
        print('正在连接服务器[%s]' %addr)
        self.addr=addr

lei

Python 函数怎么封装参数 python封装函数接口_子类

Python 函数怎么封装参数 python封装函数接口_封装_02

#from module import FtpClient
f1=FtpClient('192.168.1.1')
if hasattr(f1,'get'):
    func_get=getattr(f1,'get')
    func_get()
else:
    print('---->不存在此方法')
    print('处理其他的逻辑')

导入模块

(2)好处二:动态导入模块(基于反射当前模块成员)