文書コンバータは数学だった。

このワークシートは[url=https://www.geogebra.org/m/twxxx3yq]Math by Code[/url]の一部です。
[color=#0000ff][b]ビルドパタン:1つの作成で多様な表現形式のオブジェクトを作る。[br][/b][/color][br]たとえば、[br]タイトル、文章、箇条書き、数式がある文書ページを作りたい。[br]しかし、テキストファイルのベタ打ちに近いものの欲しいし、[br]あとで、図形を入れるかもしれないから、ワード文書にもしたい。[br]でも、ネットにのせには、HTML形式のページにもできたらいいなあ。[br]さらに、PDFファイルで保存できたら便利かも。。。。。[br][br]これは数学に限らず、仕事でも、趣味でもよくあることですね。[br][br][b]文書コンバートは、いろんなソフトで可能になっています。[br]これが、文書コンバータ、ビルドパタンです。[br][/b][br]今回は、困った状況の脱出というお話ではなく、[br]「よくある優れた機能の裏側(構造)を覗き見よう」というテーマです。[br][br]
1.文書コンバータパタン
フレームワーク会社Docに登場するのは2人です。[br]1人はディレクターで文書の構成と内容を決めます。[br]もう1人は文書の要素を決める名誉職人です。[br][br]ユーザが自分の欲しい表現形式の職人スタッフを準備することになるでしょう。[br][br]Doc社の名誉職人は手を汚しません。名前はビルドさんです。[br]文書の構成要素を抽象的に提示する、抽象契約者です。AbstractFactoryですね。[br]Doc社のディレクターは、名誉職人の提示したキーワードにそって[br]文章を構成し、文字内容を決めるでしょう。[br][br]ユーザは、AbstractFactoryの下請け工場を作ることになります。[br]テキストファイル、HTMLファイル。[br]とりあえずこの2タイプがページができるように実装してみましょう。
2.文書コンバータを実装する
[b][IN]build.py[/b][br]# ==========================================[br]# 【クラスC:名誉職人】ビルドさん(抽象契約)[br]# ==========================================[br]class AbstractDocBuilder:[br]    """【名誉職人】一切手は汚さない。[br]    ただ、文書を構成するためのキーワード(インターフェース)を提示するだけ。"""    [br]    def add_title(self, text): raise NotImplementedError()[br]    def add_paragraph(self, text): raise NotImplementedError()[br]    def add_bullet_point(self, item): raise NotImplementedError()[br]    def add_formula(self, expr): raise NotImplementedError()[br]    def get_result(self): raise NotImplementedError()[br][br]# ==========================================[br]# 【クラスA:ディレクター】編集長[br]# ==========================================[br]class DocumentDirector:[br]    """【ディレクター】文書の構成(順番)と内容(文字)を決定する。[br]    ただし、最終的にそれがどんな形式(表現)になるかは『知らない』。"""[br]    def __init__(self, builder: AbstractDocBuilder):[br]        self.builder = builder # ユーザーが連れてきた具象職人をセットする[br][br]    def construct_page(self):[br]        # 名誉職人のキーワードに沿って、上から順番に原稿を組み立てる[br]        self.builder.add_title("デザインパターンを学ぼう")[br]        self.builder.add_paragraph("二元論で語ることは簡単だ。")[br]        self.builder.add_bullet_point("下請け工場パタン")[br]        self.builder.add_bullet_point("コピー屋さんパタン")[br]        self.builder.add_formula("E = mc^2")[br][br]# ==========================================[br]# 【カスタマイズ1】テキストファイル担当の職人[br]# ==========================================[br]class TextDocBuilder(AbstractDocBuilder):[br]    def __init__(self):[br]        self.lines = [] # ここにテキストを書き溜めていく(手を汚す場所)[br][br]    def add_title(self, text):[br]        self.lines.append(f"【★ {text} ★】\n")[br][br]    def add_paragraph(self, text):[br]        self.lines.append(f" {text}\n")[br][br]    def add_bullet_point(self, item):[br]        self.lines.append(f" ・ {item}\n")[br][br]    def add_formula(self, expr):[br]        self.lines.append(f" [式: {expr}]\n")[br][br]    def get_result(self):[br]        # 最後に、溜まった文字列をガッチャンコして完成品を渡す[br]        return "".join(self.lines)[br][br][br]# ==========================================[br]# 【カスタマイズ2】HTMLファイル担当の職人[br]# ==========================================[br]class HtmlDocBuilder(AbstractDocBuilder):[br]    def __init__(self):[br]        self.tags = [] # ここにHTMLタグを書き溜めていく(手を汚す場所)[br][br]    def add_title(self, text):[br]        self.tags.append(f"{text}\n")[br][br]    def add_paragraph(self, text):[br]        self.tags.append(f"{text}\n")[br][br]    def add_bullet_point(self, item):[br]        self.tags.append(f"[*]{item}[/*]\n")[br][br]    def add_formula(self, expr):[br]        self.tags.append(f"$${expr}$$\n")[br][br]    def get_result(self):[br]        # タグをすべて結合して、一つのHTMLドキュメントとして完成させる[br]        return "\n\n" + "".join(self.tags) + "\n"[br][br]print("--- 1. 【テキスト職人】を配属して構築する ---")[br]text_worker = TextDocBuilder() # 現場のテキスト職人[br]director_T = DocumentDirector(builder = text_worker)[br]director_T.construct_page() # 編集長が原稿を読み上げる[br][br]# 最後に職人から完成品を受け取る[br]text_result = text_worker.get_result()[br]print(text_result)[br][br][br]print("--- 2. 【HTML職人】へ一発チェンジして構築する ---")[br]html_worker = HtmlDocBuilder() # 現場のHTML職人[br]director_H = DocumentDirector(builder = html_worker) # 編集長はそのまま[br]director_H.construct_page() # まったく同じ原稿をもう一度読み上げる[br][br]# 最後に職人から完成品を受け取る[br]html_result = html_worker.get_result()[br]print(html_result)[br][br][b][OUT][br]--- 1. 【テキスト職人】を配属して構築する ---[br]【★ デザインパターンを学ぼう ★】[br] 二元論で語ることは簡単だ。[br] ・ 下請け工場パタン[br] ・ コピー屋さんパタン[br] [式: E = mc^2][br][br]--- 2. 【HTML職人】へ一発チェンジして構築する ---[br][br][br]デザインパターンを学ぼう[br]二元論で語ることは簡単だ。[br][*]下請け工場パタン[/*][br][*]コピー屋さんパタン[/*][br]$$E = mc^2$$[br][br][/b]
3.振り返り
文書コンバータパタンはどうでしたか。[br][br]ビルドさんは構成要素の抽象契約者でしかなく、構築はユーザの作ったサブクラスがやってました。[br]そして、文書の内容はディレクターが決めてました。[br]だから、手を汚しているのはユーザの方ですね。サブクラスまで作っているし。[br][br]それに対して、抽象契約の数式作成はどうだったでしょうか。[br]フレームワーク社の画面工作員は楽ちんとはいいながらも画面に要素を追加する作業をしてました。[br]そう考えると、文書コンバータパタンは究極の省エネパタンですね。[br][br]このような違いはあるものの、共通点としてはサブクラスの効果です。[br]抽象クラスは、ルールの明確化と情報の安定的な流れを保証する。[br]サブクラスに詳細をゆだねることで、抽象クラス自体は不変性が高まるし、変更はサブクラスで柔軟にできる。[br][b]私一人でなんでも解決します。わが社がすべて相談に乗ります。なんでも解決します。[br][/b]ではなく、やることを明確に限定して、変わらないことを価値とする。その代わり変わる部分と連携して、一体となって問題を解決する。[br][br][b][color=#0000ff][size=150]硬と軟を分離してから結合する。異種に分解しておいて合成・連携して動かす。[br][/size][/color][/b][br]これって、[b]ベクトルを基底と係数にわけた線形結合で表す、多変数関数・波動ベクトルを偏微分や射影で取り出し、全体として合成する[/b]ことも[br]わすれないという数学思考と同じ思考回路だなと感じますね。[br][br]デザインパターン思考は数学思考だ。[br][br][color=#9900ff][b][u][size=150]課題:文書コンバータをgeogabraで感じとるにどうしたらよいですか。[br][/size][/u][/b][/color][br]ディレクターの入れる文書内容を2つ用意します。[br]title = "第3章:構築パタン"[br]p1 = "1つの手順で、多様な表現を作る。"[br]a = Checkbox() # これの設定で、見出しを「ふつうのテキスト」などにしましょう。[br]次に表示コンバータの中身を決めます。Textの第1引数はオブジェクト、第2は表示座標です。[br]TextTitle = If(a, Text(title, (1, 5)), Text("◆ " + title, (1, 5)))[br]TextBody = If(a, Text(p1, (1, 4)), Text(" ・ " + p1, (1, 3)))[br]geogebraのTextは文字列+オブジェクト名のように編集表示ができるし、[br]第2引数で表示位置を決められます。[br]ということは、パワーポイントのように動的なメッセージ画面も作れるということですね。[br]ただし、背景の設定を選んで格子線や座標軸を消すのをお忘れなく。[br]ビューの設定で背景色まで変えられますので、好きな色にカスタマイズするのもいいね。
文書コンバータ

Information: 文書コンバータは数学だった。