python中代理模式分为几种_浅谈Python设计模式 - 代理模式

声明:本系列文章主要参考《精通Python设计模式》一书,并且参考一些资料,结合自己的一些看法来总结而来。

一、在某些应用中,我们想要在访问某个对象之前执行一个或者多个重要的操作,例如,访问敏感信息 -- 在允许用户访问敏感信息之前,我们希望确保用户具备足够的去权限。同时在网络访问时,限制某些网络的访问等操作。

二、把一个计算成本较高的对象的创建过程延迟到用户首次真正使用它的时候才进行。

以上的情况就可以使用 代理设计模式。

代理模式:因使用代理对象再访问实际对象之前执行重要操作而得其名。

示例:

之前想用《精通Python设计模式》中的示例来说明,但是发现很负责不太好理解,于是有了接下来的示例:阿里云:Python与设计模式 -代理模式。

一、首先构件一个网络服务器:

ContractedBlock.gif

ExpandedBlockStart.gif

#该服务器接受如下格式数据,addr代表地址,content代表接收的信息内容

info_struct=dict()

info_struct["addr"]=10000info_struct["content"]=""

classServer:

content=""

defrecv(self,info):pass

defsend(self,info):pass

defshow(self):pass

classinfoServer(Server):defrecv(self,info):

self.content=inforeturn "recv OK!"

defsend(self,info):pass

defshow(self):print "SHOW:%s"%self.content

普通的网络服务器

infoServer有接收和发送的功能,发送功能由于暂时用不到,保留。另外新加一个接口show,用来展示服务器接收的内容。接收的数据格式必须如info_struct所示,服务器仅接受info_struct的content字段。

二、若此时有需求,该网络服务器只允许部分网络IP进行访问,那么需要设置白名单,该怎么做呢?显然可以有如下两种方式:

①、修改Server结构是个方法,即在进入server时,做一系列逻辑判断。但这显然不符合软件设计原则中的单一职责原则。

②、使用代理,即利用代理来进行逻辑判定,若在白名单中,则允许访问,若不在则拒绝。

classserverProxy:pass

classinfoServerProxy(serverProxy):

server=""

def __init__(self,server):

self.server=serverdefrecv(self,info):returnself.server.recv(info)defshow(self):

self.server.show()classwhiteInfoServerProxy(infoServerProxy):

white_list=[]defrecv(self,info):try:assert type(info)==dictexcept:return "info structure is not correct"addr=info.get("addr",0)if not addr inself.white_list:return "Your address is not in the white list."

else:

content=info.get("content","")returnself.server.recv(content)defaddWhite(self,addr):

self.white_list.append(addr)defrmvWhite(self,addr):

self.white_list.remove(addr)defclearWhite(self):

self.white_list=[]

代理中有一个server字段,控制代理的服务器对象,infoServerProxy充当Server的直接接口代理,而whiteInfoServerProxy直接继承了infoServerProxy对象,同时加入了white_list和对白名单的操作。这样,在场景中使用一个白名单服务器代理类来实现,在接收请求时,做验证:内容是否符合规则、访问者的IP地址是否在白名单中,若通过则接收内容。

那么有了白名单服务器代理,该怎么使用呢?

if __name__=="__main__":

info_struct=dict()

info_struct["addr"] = 10010info_struct["content"] = "Hello World!"info_server=infoServer()

info_server_proxy=whiteInfoServerProxy(info_server)print(info_server_proxy.recv(info_struct))

info_server_proxy.show()

info_server_proxy.addWhite(10010)print(info_server_proxy.recv(info_struct))

info_server_proxy.show()

打印如下:

Your address is not inthe white list.

SHOW:

recv OK!

SHOW:Hello World!

这边我也把书中的示例放置在此。

ContractedBlock.gif

ExpandedBlockStart.gif

classLazyProperty(object):'''利用装饰器的特性作为代理,给_resource初始化值'''

def __init__(self, method):

self.method=method

self.method_name= method.__name__

print('func name is:{}'.format(self.method_name))def __get__(self, obj, cls):'''使用值来替代方法'''

if notobj:returnNone

value=self.method(obj)print('value {}'.format(value))

setattr(obj, self.method_name, value)returnvalueclassTest(object):def __init__(self):

self.x= 'foo'self.y= 'bar'self._resource=None

@LazyProperty#resource = LazyProperty(resource)

defresource(self):print('init self._resource which is:{}'.format(self._resource))

self._resource= tuple(range(5))returnself._resourcedefmain():

t=Test()print(t.x)print(t.y)print(t._resource)print(t.resource)print(t.__dict__)print(t.resource)#print(t._resource)

if __name__ == '__main__':

main()

精通Python设计模式--示例

该示例:使用的是装饰器来实现对 resource方法的惰性加载,而该装饰器是使用数据描述符来实现的,故需要对数据描述符有一定的了解。

over~~,参考:https://yq.aliyun.com/articles/70738?utm_content=m_15329,感谢。。。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值