加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 编程开发 > Python > 正文

python设计模式之工厂模式

发布时间:2020-12-17 00:17:38 所属栏目:Python 来源:网络整理
导读:div class="markdown-here-wrapper" data-md-url="https://i.cnblogs.com/EditPosts.aspx?postid=10029960"gt; h1 id="-" style="margin: 20px 0px 10px; padding: 0px; font-weight: bold; color: black; font-size: 24px; border-bottom: 2px solid #aaaaa

<div class="markdown-here-wrapper" data-md-url="https://i.cnblogs.com/EditPosts.aspx?postid=10029960"&gt;
<h1 id="-" style="margin: 20px 0px 10px; padding: 0px; font-weight: bold; color: black; font-size: 24px; border-bottom: 2px solid #aaaaaa;">一.理解工厂模式
<blockquote style="margin: 1.2em 0px; border-left: 4px solid #dddddd; padding: 0px 1em; color: #777777; quotes: none;">
<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">在面向对象编程中,术语“工厂”表示一个负责创建替他类型对象的类。通常情况下,作为一个工厂的类有一个对象以及与它关联的多个方法。客户端使用某些参数调用此方法,之后,工厂会据此创建所需类型的对象,然后将它们返回给客户端。

 abc  ABCMeta,abstractmethod

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Coke<span class="hljs-params" style="color: #ffc58f;">(metaclass=ABCMeta):
<span class="hljs-decorator">@abstractmethod
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">pass

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Coca<span class="hljs-params" style="color: #ffc58f;">(Coke):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
print(<span class="hljs-string" style="color: #d1f1a9;">'drink Coca-Cola')

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Pepsi<span class="hljs-params" style="color: #ffc58f;">(Coke):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
print(<span class="hljs-string" style="color: #d1f1a9;">'drink Pepsi-Cola')

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Fast_food_restaurant<span class="hljs-params" style="color: #ffc58f;">():
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_coke<span class="hljs-params" style="color: #ffc58f;">(self,name):
<span class="hljs-keyword" style="color: #ebbbff;">return eval(name)()

KCD=Fast_food_restaurant()
coke=KCD.make_coke(<span class="hljs-string" style="color: #d1f1a9;">'Coca')
coke.drink()<span class="hljs-comment" style="color: #7285b7;">#drink Coca-Cola


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">ABCMeta是python的一个元类,用于在Python程序中创建抽象基类,抽象基类中声明的抽象方法,使用abstractmethod装饰器装饰。


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">eval(类名)返回的是一个class类型的对象


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">我们建立一个可乐的抽象类,百事可乐和可口可乐继承这个抽象类,我们又建立了快餐店类,也就是所说的工厂类,让它生产可乐。当用户需要可乐时,只需要告诉快餐店做一份什么品牌的可乐,告诉快餐店可乐的名字,然后快餐店使用make_coke方法做可乐,返回了你所需要的对象——一杯可口可乐,然后就可以快乐的喝可乐了。。


<h3 id="2-" style="margin: 20px 0px 10px; padding: 0px; font-weight: bold; color: black; font-size: 20px; border-bottom: 1px solid #aaaaaa;">2.工厂方法模式


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">简单工厂模式已经帮我们做到我们需要某种对象时,可以不关心对象是怎么创建的,只需要向工厂类要对象即可,但是如果我们又多了一种对象,例如又出现了一个可乐品牌,嗯,我们叫它sfencs可乐吧,那么我们快餐店也得新添加这种可乐,也就是工厂类也得能够创建sfencs可乐对象了,但是这样就得在工厂类中加入新的逻辑判断来根据用户需求制造新添加的这个对象,显然是不恰当的,因为这样每当有一个新的类型的可乐增加的时候,我们都得修改工厂类的逻辑代码,使之能够判断出新的类型。这个问题使用工厂方法模式可以得到解决。


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">这里有一个小问题,如上面的简单工厂模式的代码,如果要新加sfencs可乐,似乎并不需要修改快餐店工厂类,只需要在sfencs可乐类定义之后,依然向make_coke函数传递类名即可,那这样岂不是不需要工厂方法模式了吗?


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">对于这个现象,其实是得益于eval()函数的功能,它能依据传入的字符串转换成相应的类,也就是eval函数就是工厂类的逻辑判断,如果不使用eval,那么逻辑判断可能就是多个if语句了,判断条件就是输入的字符串参数是否等于这个,是否等于那个。。。等。但是eval使用也有很大的局限性,它只能根据字符串来判断,我们不一定创建对象时都知道它的类名。除此之外,个人感觉eval这个函数有点厉害,越厉害就有可能造成不必要的麻烦,eval使用时还是得多注意。


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">接下来介绍工厂方法模式:

 <span class="hljs-title" style="color: #7285b7;">Coke<span class="hljs-params" style="color: #ffc58f;">(metaclass=ABCMeta):
<span class="hljs-decorator">@abstractmethod
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">pass

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Coca<span class="hljs-params" style="color: #ffc58f;">(Coke):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
print(<span class="hljs-string" style="color: #d1f1a9;">'drink Coca-Cola')

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Pepsi<span class="hljs-params" style="color: #ffc58f;">(Coke):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
print(<span class="hljs-string" style="color: #d1f1a9;">'drink Pepsi-Cola')

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Sfencs<span class="hljs-params" style="color: #ffc58f;">(Coke):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
print(<span class="hljs-string" style="color: #d1f1a9;">'drink Sfencs-Cola')

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Fast_food_restaurant<span class="hljs-params" style="color: #ffc58f;">(metaclass=ABCMeta):
<span class="hljs-decorator">@abstractmethod
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_coke<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">pass

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Coca_produce<span class="hljs-params" style="color: #ffc58f;">(Fast_food_restaurant):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_coke<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">return Coca()

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Pepsi_produce<span class="hljs-params" style="color: #ffc58f;">(Fast_food_restaurant):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_coke<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">return Pepsi()

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Sfencs_produce<span class="hljs-params" style="color: #ffc58f;">(Fast_food_restaurant):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_coke<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">return Sfencs()

KCD=Sfencs_produce()
coke=KCD.make_coke()
coke.drink()<span class="hljs-comment" style="color: #7285b7;">#drink Sfencs-Cola


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">工厂方法模式将原来的工厂类变为了抽象类,不同类型的可乐通过不同的子类生产,也就是工厂方法模式定义了一个创建对象的接口,但具体创建哪个类的对象由子类来决定,这种方式的逻辑判断相当于交给了客户端,也就是KCD=Sfencs_produce()来选择使用哪个子类,这样如果有新可乐产品出现的话,只需要再写一个子类继承工厂抽象类。


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">这里的类中,快餐店抽象类也叫做抽象工厂类,它的子类称为具体工厂类。可乐也一样,Coke为抽象产品类,它的子类为具体产品类。


<h3 id="3-" style="margin: 20px 0px 10px; padding: 0px; font-weight: bold; color: black; font-size: 20px; border-bottom: 1px solid #aaaaaa;">3.抽象工厂模式
<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">抽象工厂模式的主要目的是提供一个接口来创建一系列相关对象而无需指定具体的类。这个模式与与工厂方法模式的区别在于,它的一个方法子类,可以创建一系列的对象。


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">依然用可乐来举例,只喝普通的可乐还不足以让我们非常快乐,那么如果有一杯冰可乐,想必就能满足我们了。


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">于是我们的抽象产品类变为了两个,一个是冰可乐,一个是普通可乐,具体产品类有百事冰可乐、可口可乐冰可乐,普通百事,普通可口可乐。抽象工厂类有生产冰可乐和生产普通可乐的抽象方法,具体工厂类有百事工厂,可口可乐工厂。

 <span class="hljs-title" style="color: #7285b7;">Ice_coke<span class="hljs-params" style="color: #ffc58f;">(metaclass=ABCMeta):
<span class="hljs-decorator">@abstractmethod
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">pass

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Ordinary_coke<span class="hljs-params" style="color: #ffc58f;">(metaclass=ABCMeta):
<span class="hljs-decorator">@abstractmethod
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">pass

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Coca_ice<span class="hljs-params" style="color: #ffc58f;">(Ice_coke):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
print(<span class="hljs-string" style="color: #d1f1a9;">'drink Coca-ice-Cola')

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Pepsi_ice<span class="hljs-params" style="color: #ffc58f;">(Ice_coke):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
print(<span class="hljs-string" style="color: #d1f1a9;">'drink Pepsi-ice-Cola')

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Coca_ordinary<span class="hljs-params" style="color: #ffc58f;">(Ordinary_coke):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
print(<span class="hljs-string" style="color: #d1f1a9;">'drink Coca-ordinary-Cola')

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Pepsi_ordinary<span class="hljs-params" style="color: #ffc58f;">(Ordinary_coke):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">drink<span class="hljs-params" style="color: #ffc58f;">(self):
print(<span class="hljs-string" style="color: #d1f1a9;">'drink Pepsi-ordinary-Cola')

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Fast_food_restaurant<span class="hljs-params" style="color: #ffc58f;">(metaclass=ABCMeta):
<span class="hljs-decorator">@abstractmethod
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_ice_coke<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">pass

<span class="hljs-decorator"&gt;@abstractmethod</span>
<span class="hljs-function" style="color: #bbdaff;"&gt;<span class="hljs-keyword" style="color: #ebbbff;"&gt;def</span> <span class="hljs-title" style="color: #7285b7;"&gt;make_ordinary_coke</span><span class="hljs-params" style="color: #ffc58f;"&gt;(self)</span>:</span>
    <span class="hljs-keyword" style="color: #ebbbff;"&gt;pass</span>

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Coca_produce<span class="hljs-params" style="color: #ffc58f;">(Fast_food_restaurant):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_ice_coke<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">return Coca_ice()
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_ordinary_coke<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">return Coca_ordinary()

<span class="hljs-class"><span class="hljs-keyword" style="color: #ebbbff;">class <span class="hljs-title" style="color: #7285b7;">Pepsi_produce<span class="hljs-params" style="color: #ffc58f;">(Fast_food_restaurant):
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_ice_coke<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">return Pepsi_ice()
<span class="hljs-function" style="color: #bbdaff;"><span class="hljs-keyword" style="color: #ebbbff;">def <span class="hljs-title" style="color: #7285b7;">make_ordinary_coke<span class="hljs-params" style="color: #ffc58f;">(self):
<span class="hljs-keyword" style="color: #ebbbff;">return Pepsi_ordinary()

KCD=Coca_produce()
coke=KCD.make_ice_coke()
coke.drink()<span class="hljs-comment" style="color: #7285b7;">#drink Coca-ice-Cola


<p style="margin: 0px 0px 1.2em !important; font-size: 16px; line-height: 1.75em; padding-right: 0.5em; padding-left: 0.5em;">这样再有其他类型的可乐或者其他品牌的可乐,只需要添加方法或者类就可以了。


(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读