Python的创建型设计模式之建造者模式

书上说此模式和前面的抽象工厂模式相似,它不仅提供了创建复杂对象所需的方法,而且还保存了对象里各部分的细节。

个人觉得,抽象工厂里面因为包含了类中嵌套类的模式,就没有那么像父子类的模式,在看了建造者模式之后,觉得这个才真的很类似于父子类的模式。

再借用书上的话:尤其适用于需要把复杂对象各部分的细节与其创建流程相分离的场合。

现在是要用HTML和TKinter完成一个表单的创建,来看看部分事例代码:

1.首先是一个只能用作抽象基类的类,里面包括了添加表单,标题,内容等等方法

class AbstractFormBuilder(metaclass=abc.ABCMeta):

    @abc.abstractmethod
    def add_title(self, title):
        pass

    @abc.abstractmethod
    def form(self):
        pass

    @abc.abstractmethod
    def add_label(self, text, row, column, **kwargs):
        pass
	
    ...
2.html下完成表单的类,在继承AbstractFormBuilder的基础上完成具体方法

class HtmlFormBuilder(AbstractFormBuilder):

    def __init__(self):
        self.title = "HtmlFormBuilder"
        self.items = {}

    def add_title(self, title):
        pass

    def add_label(self, text, row, column, **kwargs):
        self.items[(row, column)] = ('<td><label for="{}":</label></td>')\
            .format(kwargs["target"], xml.sax.saxutils.escape(text))

    def form(self):
        html = ["<!doctype html>\n<html><head><title>{}</title></head>"
                "body".format(self.title), '<form><table border="0">']
        thisRow = None
        for key, value in sorted(self.items.items()):
            row, column = key
            if thisRow is None:
                html.append("  <tr>")
            elif thisRow != row:
                html.append("  </tr>\n <tr>")
                thisRow = row
                html.append("  " + value)
            html.append("</tr>")
            return "\n".join(html)
    ...

3.tkinter下完成表单的类

class TkFormBuilder(AbstractFormBuilder):

    def __init__(self):
        self.title = "TkFormHtml"
        self.statements = []

    def add_title(self, title):
        pass

    def add_label(self, text, row, column, **kwargs):
        name = self._canonicalize(text)
        create = """self.{}Label = ttk.Label(self, text="{}:")""".format(name, text)
        layout = """self.{}Label.grid(row={}, column={}, sticky=tk.W)""".format(name, row, column)
        self.statements.append((create, layout))

    TEMPLATE = """创建表单的代码,使用到了tkinter"""

    def form(self):
        return TkFormBuilder.TEMPLATE.format(title=self.title,
                                             name=self._canonicalize(self.title, False),
                                             statements="\n     ".join(self.statements))

    def _canonicalize(self, text, startLower=True):
        text = re.sub(r"\W", "", text)
        if text[0].isdigit():
            return "_" + text
        return text if not startLower else text[0].lower() + text[1:]

盖楼房,也就是这个道理吧。有根基和大致布局,才能去具体完成。




發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章