このワークシートは[url=https://www.geogebra.org/m/twxxx3yq]Math by Code[/url]の一部です。
これまで扱ったデザインパタンは[br]構造を前に進む探索について扱ったきました。[br]進みたいときもあれば、戻りたいときもあるでしょう。[br][br][b]処理を取り消したい。Undo。[br]可能な状態を行き来したい。StateTraveler。[br]過去の状態に一瞬で世界を上書きしたい。Warp。[br][/b][br]いろんな戻り方があります。[br]今回のテーマは「数学では、進めるなら戻れるはず」[br][size=150][size=100]扱うパタンは[br][/size][color=#0000ff][br]Commandクラスで記憶の追加と消去,[br]Stateクラスで主客逆転してすごろく化,[br]Mementoクラスは没入型の思い出写真。[br]の3つです。[br][/color][/size][br]数学でも日常生活でも重要な[b][size=150]「可逆性の視点と方法」[/size][/b]に取り組もう。
[b][size=150]<コマンド履歴パタン>[/size][/b][br]PCを動かしていると、Ctrl+Zで画面操作が1回取り消されます。[br]別に神様が監視して元に戻しているわけではありません。[br]アプリへのユーザの選んだコマンド選択が履歴として時系列で残します。[br]そのコマンドを取り消すことで、1つ前の画面状態に戻れるというしくみです。[br][color=#0000ff][b][size=150]コマンドパタン:[br]メソッド呼び出しのカプセル化をすることで、[br]メソッドログ保存やUndoの実装ができるようにする。[br][/size][/b][/color]ということです。ただ、本によって実装方法もカプセル化もちがってますから、[br]ひとつの例として読んでください。[br][br]大昔のPCゲームでは2次元なので、コントローラのかわりに矢印キーで[br]自分の分身や攻撃の発射台など動かしてましたね。[br][b]敵のいないプレーンなゲームのフレームを作りたいとしましょう。[br]画面に現れた自分Meを矢印キー(←、→、↑、↓)のどれかを押すことができます[/b]。[br][b]キャレットを1つ前の位置に戻すにはZキーを押します[/b]。[br][br]ユーザのキー入力は[b]呼び出しInvokeクラスのpressKey[/b]コマンドで受け取ります。[br]それを情報として記録するための情報記録系のコマンドクラスを雇います。それがInvokeクラスのSetCommandです。ただし、コマンドクラスはインターフェースでしかないので、実際はその子クラスの情報部員MacroCommandになります。[br]キー入力に応じてInvokeのpresskeyのコマンドを通して、情報部員MacroCommandに情報が渡ります。[br][br]では、情報を受け取った[b]情報部員MacroCommandのインスタンス[/b]は何をするのでしょうか。一番大切なのは指令の実行、executeであることは当然ですが、それだけではありません。[br]使えるメソッドはイニシャライズのときに受け取りMeInterfaceインターフェースの登録です。[br][b]Invokeのpresskeyに応じたコマンド[/b]があります。[br]・appendは情報部員のログcommandslogにキー操作の追加。[br]・executeはappendしたあとにMeInterfaceに矢印キーに対応したleft,right, up,downを伝えます。[br]・undoはpresskeyがzなら、commandslogからのキー入力履歴の最新をPopで抜いておき、MeInterfaceにUndoを伝えます。[br][br]コマンドの[b]最後の受け取りと実行をするのは受け取り系のインターフェースReceiver[/b]です。[br]ゲームで動く自分の分身にちなんで、[b]MeInterface[/b]とします。[br]メソッドはleft,right,up,down,undo_actionの5つです。[br]この子クラスは、自分のposition、過去の位置の記録リストhistoryをもちます。[br]自分の位置の記録のための私的メソッド_save_current_positionを持ちます。[br]矢印キーの情報を時系列に追記し、連動してhistoryも時系列に追記します。[br]zキーを押すと、commandslogもhistoryもpopさせます。[br][color=#0000ff][b][size=150]最新の情報が消えるので、戻った状態になる。記憶を1つ消したのですね[/size][/b][/color]。[br][br][b]サブクラス化だけでなく、移譲(has)によって、情報をリレーして投げるようす[/b]は[br]次のようになります。[br][b]矢印キー→呼び出しのpresskey→MacroCommandのlogへのappendとexecute→Meクラスで実行。[br]zキー→→呼び出しのpresskey→MacroCommandのlogのpop→Meクラスのhistoryもpop。[br][size=150][br]<コマンド履歴パタンの実装例>[br][/size][/b]# Command.py[br][color=#0000ff]# ==========================================[br]# 末端の実行者である受取系(Receiver)のMeInterface[br]# ==========================================[br][/color]class MeInterface:[br] """(入力待ちの自分の分身)が持つべき具体的なアクションの規格"""[br] def left(self): raise NotImplementedError()[br] def right(self): raise NotImplementedError()[br] def up(self): raise NotImplementedError()[br] def down(self): raise NotImplementedError()[br] def undo_action(self): raise NotImplementedError() # 過去の位置に戻るアクション[br][br][color=#0000ff]# ==========================================[br]# サブクラス(現在位置と位置の履歴を持つアバター)[br]# 末端とはいえ、自分の情報と記憶を持っている。[br]# ==========================================[br][/color]class Avator(MeInterface):[br] """位置(x, y)と、自身の中に位置の履歴を持つ"""[br] def __init__(self):[br] self.position = [0, 0] # [x座標, y座標][br] self.history = [] # 過去の位置座標を記録するスタック[br] def _save_current_position(self):[br] """現在の位置を履歴スタックに保存する(可逆性のための記憶)"""[br] self.history.append(list(self.position))[br] def left(self):[br] self._save_current_position()[br] self.position[0] -= 1 # 左矢印[br] print(f"[Action] Left: 自分の分身が左に動きました。現在地: {self.position}")[br] def right(self):[br] self._save_current_position()[br] self.position[0] += 1 # 右矢印[br] print(f"[Action] Right: 自分の分身が右に動きました。現在地: {self.position}")[br] def up(self):[br] self._save_current_position()[br] self.position[1] += 1 # 上矢印[br] print(f"[Action] Up: 自分の分身が上に動きました。現在地: {self.position}")[br] def down(self):[br] self._save_current_position()[br] self.position[1] -= 1 # 下矢印[br] print(f"[Action] Down: 自分の分身が下に動きました。現在地: {self.position}")[br] def undo_action(self):[br] """位置の巻き戻し(逆演算)"""[br] if self.history:[br] # 1つ前の位置を取り出して上書き(タイムトラベル)[br] self.position = self.history.pop()[br] print(f"[Undo] Z-Key: 1つ前の位置に戻しました。現在地: {self.position}")[br] else:[br] print("[Warning] これ以上戻る歴史(履歴)がありません。")[br][br][color=#0000ff]# ==========================================[br]# 【情報管理系】雇ったアバターにexcuteで実行させる[br]# ==========================================[br][/color]class Command:[br] """コマンド情報管理系の根底にある共通規格"""[br] def execute(self):[br] raise NotImplementedError()[br][color=#0000ff]# ==========================================[br]# 情報部員は情報を受け取って実行するアバターを雇う[br]# 情報部員はInvoke通して受け付けたキーのlogを持つ。[br]# ==========================================[br][/color]class MacroCommand(Command):[br] """[コマンド情報管理系] 受け取り実行アバターを保持し、操作の歴史を統括する"""[br] def __init__(self, receiver: MeInterface):[br] self.receiver = receiver [br] self.commands_log = [] # ユーザが行ったキー操作の文字列ログ[br] def append(self, key_name):[br] """ユーザの操作を歴史ログに追加する"""[br] self.commands_log.append(key_name)[br] def execute(self):[br] """最新のキー名に応じて、具体的なアクションをキックする"""[br] if not self.commands_log:[br] return[br] last_key = self.commands_log[-1] # 直近に追加されたキーを取得[br] if last_key == "left":[br] self.receiver.left()[br] elif last_key == "right":[br] self.receiver.right()[br] elif last_key == "up":[br] self.receiver.up()[br] elif last_key == "down":[br] self.receiver.down()[br] def undo(self):[br] """取り消し:ログから最後の1件を削除し、逆演算を命令する"""[br] if self.commands_log:[br] removed_key = self.commands_log.pop()[br] print(f"[Macro] {removed_key} の操作を取り消します。")[br] self.receiver.undo_action()[br] else:[br] print("[Macro] 取り消すログが存在しません。")[br][color=#0000ff]# ==========================================[br]# ユーザーのキー入力を仲介する(Invoke)[br]# ==========================================[br][/color]class KeyInvoker:[br] """ユーザからのキー入力を受け取り、セットした情報部員にコマンドの実行を依頼する"""[br] def __init__(self):[br] self.macro_command = None[br] def setCommand(self, macro_command: MacroCommand):[br] """操作対象となる情報部員をセットする"""[br] self.macro_command = macro_command[br] def press_key(self, key):[br] """ユーザがキーボードから指令を送る(left, right, up, down, Z)"""[br] if not self.macro_command:[br] print("コマンドがセットされていません。")[br] return[br] if key == "Z":[br] # Zキーなら、マクロに対して取り消し(undo)を要求する[br] self.macro_command.undo()[br] else:[br] # それ以外の移動キーなら、履歴に追加(append)してから実行(execute)を依頼する[br] self.macro_command.append(key)[br] self.macro_command.execute()
# ==========================================[br]# 【作動テスト】ユーザのキーボード入力をシミュレート[br]# ==========================================[br]print("=== Command History Algorithm: Start ===\n")[br][br][b]# 1. 各系統のインスタンス化と連携の構築[br]Me = Avator() # 受取系アバターのインスタンスをMeとする。[br]my_macro = MacroCommand(Me) # Meを雇う情報部員my_macroインスタンス。[br]keyboard = KeyInvoker() # 呼出系のインスタンスをKeybordとする。[br]keyboard.setCommand(my_macro) # 呼出系のKeybordは情報部員を持つ。[br][/b][br]# 2. ユーザのキーボード操作(指令の送信)[br]print("--- 1. 自分の分身を矢印キーで動かす ---")[br]keyboard.press_key("right")[br]keyboard.press_key("up")[br]keyboard.press_key("right")[br]keyboard.press_key("right")[br]keyboard.press_key("right")[br]keyboard.press_key("up")[br]keyboard.press_key("up")[br]keyboard.press_key("left")[br][br]print("\n--- 2. Zキーで操作を取り消してみる(Undo) ---")[br]keyboard.press_key("Z") # [br]keyboard.press_key("Z") # [br]keyboard.press_key("Z") # [br]keyboard.press_key("Z") # [br][br][OUT][br]=== Command History Algorithm: Start ===[br][br]--- 1. 自分の分身を矢印キーで動かす ---[br][Action] Right: 自分の分身が右に動きました。現在地: [1, 0][br][Action] Up: 自分の分身が上に動きました。現在地: [1, 1][br][Action] Right: 自分の分身が右に動きました。現在地: [2, 1][br][Action] Right: 自分の分身が右に動きました。現在地: [3, 1][br][Action] Right: 自分の分身が右に動きました。現在地: [4, 1][br][Action] Up: 自分の分身が上に動きました。現在地: [4, 2][br][Action] Up: 自分の分身が上に動きました。現在地: [4, 3][br][Action] Left: 自分の分身が左に動きました。現在地: [3, 3][br][br]--- 2. Zキーで操作を取り消してみる(Undo) ---[br][Macro] left の操作を取り消します。[br][Undo] Z-Key: 1つ前の位置に戻しました。現在地: [4, 3][br][Macro] up の操作を取り消します。[br][Undo] Z-Key: 1つ前の位置に戻しました。現在地: [4, 2][br][Macro] up の操作を取り消します。[br][Undo] Z-Key: 1つ前の位置に戻しました。現在地: [4, 1][br][Macro] right の操作を取り消します。[br][Undo] Z-Key: 1つ前の位置に戻しました。現在地: [3, 1]
[b][size=150]<状態クラスは主客逆転のすごろく化>[br][/size][/b][br][b]イベント駆動の開発[/b]が必要なときに、[br]あまりにも[b]イベント中心思考[/b]となり、[br]状態を従となる制約条件としてみてしまうと、[br]大変なことが起きることがよくあります。[br][br]例えばイベントがX,Y,Zの4つあり、[br]それに連動して変わるstateがa,b,c,dの4つがあり、[br]メッセージm1,m2,m3,m4,m5,m6がある。[br]ゲームマシンContextの動作をイベント駆動でかくと、[br]イベントX{[br]if state==a{m1}[br]elif state==b{[br] m2[br] state =c }[br]elif state==c{m3}[br]elif state==d{[br] m4[br] state=a}[br]elif state==e{m5}[br]}[br]イベントY{[br]if state==a{[br] m2[br] state=b}[br]}[br]elif state==b{m3}[br]elif state==c{[br] m4[br] state=d}[br]elif state==d{m5}[br]elif state==e{m6}[br]}[br]イベントZ{[br]if state==a{m3}[br]elif state==b{m4}b-[br]elif state==c{[br] m5[br] state=a}[br]elif state==d{m6}[br]elif state==e{m1}[br]}[br]となったする。すごい大量のコードだ。[br][b][size=200][color=#0000ff]ここで逆転の発想をしましょう。[br]これを状態を中心に書き直してみよう。[br][/color][/size][/b][br]状態a,b,c,dをクラスにしてStateA,StateB.StateC,StateDとする。[br]ゲームマシンContextは状態クラスのインターフェースIStateを持ち、[br]その子クラスが上にある4つのクラスだ。[br]Contextは3つのイベントX,Y,Zに対応するリクエストreqX,reqY,reqZができる。[br]これを受け取るとIStateはevntX,evntY,evntZ持つ。[br]全部はかかないけれど、StateAクラスのコードは簡単だ。[br]最初のイベント内のコードのif文の1行目から、[br][size=150][b]StateA{evntX{m1}; evntY{m2;changeState(stateB)};evntZ{m3}}[/b][br][/size][color=#0000ff][b][size=200]if文が消えたね。[/size][/b][/color][br][br]そして、[b]Contextクラスは[color=#0000ff][size=200]現在の状態を格納する変数self.state[/size][/color]がある。[br]そうすることで、状態クラスをA→B→C→A, D→Aと推移することで、[br]Contextが移譲する状態子クラスが変化し、それに対応してイベントに対するメッセージが変わる[/b]。[br][br]このように、状態を中心に逆転した書き方をするだけでカンタンなるけれど、[br]状態をクラスにしたことで、移譲する子クラスを変えるという表現にすることで、[br][br][b]ゲームマシンコンテキストの状態推移が手にとるようにわかる。[br]記述も簡単になる。[br][/b][br]状態たちを〇がこみして、そのときの推移を線でつなぐ。[br]外からイベントがやってきて、ただのメッセージで終わることもあれば、次の〇にすすめることもある。[br]まるで「すごろく」だね。[br][b][size=150][color=#0000ff][br]状態クラスを作り、状態つまりマスは次に進むだけです。[br][/color][/size][/b][br]状態の子クラスは問題空間Contextで決められるし、あとで変更・追加も簡単にできるでしょう。[br][br]ストラテジーパタンのようには、持っている(移譲した)コメンテータを適当に選ぶことはできません。[br]でも、複数の状態を持てる(移譲する)ならば、それを渡る歩くStateTravelができます。[br][b]移譲でうまく切り替えるという意味では共通[/b]してますね。[br]
[b][size=150]<状態クラスの実装例>[br][/size][/b]# State.py[br][color=#0000ff]# ==========================================[br]# 状態の共通規格(インターフェース)[br]# ==========================================[br][/color]class IState:[br] """各状態(すごろくのマス目)が処理すべきイベント(X, Y, Z)の規格"""[br] def evntX(self, context): raise NotImplementedError()[br] def evntY(self, context): raise NotImplementedError()[br] def evntZ(self, context): raise NotImplementedError()[br][br][color=#0000ff]# ==========================================[br]# ゲームマシン(コンテキスト)[br]# ==========================================[br][/color]class GameMachineContext:[br] """状態を保持し、外からのリクエストを現在の状態へとリレーする"""[br] def __init__(self):[br] self.state = StateA() # 初期状態はマスA (state_a)[br] def change_state(self, new_state: IState):[br] """状態を次のマス(子クラス)へと差し替えて推移(相転移)する"""[br] self.state = new_state[br] def reqX(self): self.state.evntX(self) # イベントXの受付[br] def reqY(self): self.state.evntY(self) # イベントYの受付[br] def reqZ(self): self.state.evntZ(self) # イベントZの受付[br][br][color=#0000ff]# ==========================================[br]# 状態の子クラス群(すごろくの各マス目)[br]# ==========================================[br][/color]class StateA(IState):[br] """状態aのマス:イベントYが来たら、m2を出してマスBへ進むルールに進化"""[br] def evntX(self, context): print("[Message] m1 (StateAのまま)")[br] def evntY(self, context):[br] print("[Message] m2 ➔ [Transit] マスA から マスB へ進みます。")[br] context.change_state(StateB()) # A ➔ B への推移[br] def evntZ(self, context): print("[Message] m3 (StateAのまま)")[br][br]class StateB(IState):[br] """状態bのマス:イベントXが来たら、m2を出してマスCへ進む"""[br] def evntX(self, context):[br] print("[Message] m2 ➔ [Transit] マスB から マスC へ進みます。")[br] context.change_state(StateC()) # B ➔ C への推移[br] def evntY(self, context): print("[Message] m3 (StateBのまま)")[br] def evntZ(self, context): print("[Message] m4 (StateBのまま)")[br][br]class StateC(IState):[br] """状態cのマス:イベントYならdへ進み、イベントZならaへ戻る"""[br] def evntX(self, context): print("[Message] m3 (StateCのまま)")[br] def evntY(self, context):[br] print("[Message] m4 ➔ [Transit] マスC から マスD へ進みます。")[br] context.change_state(StateD()) # C ➔ D への推移[br] def evntZ(self, context):[br] print("[Message] m5 ➔ [Undo/Transit] マスC から マスA へ戻ります!")[br] context.change_state(StateA()) # C ➔ A への引き戻し(StateTravel)[br][br]class StateD(IState):[br] """状態dのマス:イベントXが来たら、最初のマスAへと引き戻される"""[br] def evntX(self, context):[br] print("[Message] m4 ➔ [Undo/Transit] マスD から 最初のマスA へ戻ります!")[br] context.change_state(StateA()) # D ➔ A への引き戻し(StateTravel)[br] def evntY(self, context): print("[Message] m5 (StateDのまま)")[br] def evntZ(self, context): print("[Message] m6 (StateDのまま)")[br][br]class StateE(IState):[br] """状態eのマス:別の世界線"""[br] def evntX(self, context): print("[Message] m5 (StateEのまま)")[br] def evntY(self, context): print("[Message] m6 (StateEのまま)")[br] def evntZ(self, context): print("[Message] m1 (StateEのまま)")[br][color=#0000ff][br]# ==========================================[br]# 【作動テスト】ゲームマシンの状態すごろくをトラベルする[br]# ==========================================[br][/color]print("=== State Traveler Algorithm: Start ===\n")[br][br]# 1. コンテキスト(ゲームマシン)の起動[br]machine = GameMachineContext()[br][br]print("--- 1. マスA:イベントYをトリガーして、正当にマスBへ進む ---")[br]machine.reqX() # マスAのまま(m1)[br]machine.reqY() # マスA ➔ マスB へ![br][br]print("\n--- 2. マスB:イベントXをトリガーして、自動的にマスCへ進む ---")[br]machine.reqY() # マスBのまま(m3)[br]machine.reqX() # マスB ➔ マスC へ![br][br]print("\n--- 3. マスC:イベントYをトリガーして、自動的にマスDへ進む ---")[br]machine.reqX() # マスCのまま(m3)[br]machine.reqY() # マスC ➔ マスD へ![br][br]print("\n--- 4. マスD:イベントXをトリガーして、最初のマスAへと自動で「戻る」 ---")[br]machine.reqY() # マスDのまま(m5)[br]machine.reqX() # マスD ➔ 最初のマスA へ引き戻し![br][b][OUT][br]=== State Traveler Algorithm: Start ===[br][br]--- 1. マスA:イベントYをトリガーして、正当にマスBへ進む ---[br][Message] m1 (StateAのまま)[br][Message] m2 ➔ [Transit] マスA から マスB へ進みます。[br][br]--- 2. マスB:イベントXをトリガーして、自動的にマスCへ進む ---[br][Message] m3 (StateBのまま)[br][Message] m2 ➔ [Transit] マスB から マスC へ進みます。[br][br]--- 3. マスC:イベントYをトリガーして、自動的にマスDへ進む ---[br][Message] m3 (StateCのまま)[br][Message] m4 ➔ [Transit] マスC から マスD へ進みます。[br][br]--- 4. マスD:イベントXをトリガーして、最初のマスAへと自動で「戻る」 ---[br][Message] m5 (StateDのまま)[br][Message] m4 ➔ [Undo/Transit] マスD から 最初のマスA へ戻ります![br][/b]
[b][size=150]<Mementoクラスは思い出写真>[/size][/b][br][br]Commandはユーザの入力記録を追加してから、画面に表示していました。[br]呼び出し係が情報部員を持ち、情報部員が行動係を持ち、という移譲の連鎖で[br]2種の情報スタックの最上部が1つPopされて、記憶も画面も1つ戻るという仕組みでした。[br][br]これに対して[b][color=#0000ff][size=150]移譲とは言っても、[br]状態を連続的に丸ごと記録したり、すべてを公開するのはしたくない場合[/size][/color][/b]があるでしょう。[br][br]2つの素数の積を乱数で表示するアプリRandN_Appがあるとします。[br][br]このアプリはplayボタンをおすたびに、2素数のかけ算の式と答えで表示します。[br]たとえば、[b]2×17=34、11×13=143、23×31=713、31×127=3937[/b]、…のように。[br]丸ごと記録ではなく、パシャとスナップ写真のように、[br]状況・画面を作る[b]変数だけを状態のインスタンスとして記録するのが思い出Memento[/b]です。[br]だから、713が素敵だと思ったら。createMementoコマンドでパシャと(23,31)を保存したとします。[br]素敵な713を思い出したいと思ったらrestoreMementoコマンドを選ぶと、23×31=713が表示される。[br][br]Mementoクラスは没入型の思い出写真。[br]時が過ぎても記録しておけば、その時の状態が丸ごと再現できるわけです。[br]ゲームとかならば、その時の状態からリプレーできたりするので、うれしい機能ですね。[br][br]細かくいうと、Mementoクラスの他に、[br][b]情報の中身は知らず[/b]に保存と呼び出しをするお世話係[b]Caretaker[/b]クラスを用意します。[br]Mementoは1枚だけのアルバムの写真1枚としてのインスタンスで使われます。主体性はありません。[br]アプリRandN_Appが写真として作り、再現します。ユーザはCaretakerが持っているアルバムに写真を入れたり、出したりするという構図によって、情報が守られているのです。[br][br][br][b][size=150]<Mementoクラスの実装例>[/size][/b][br][br]# Memento.py[br]import random[br][color=#0000ff]# ==========================================[br]# 【スナップ写真】思い出アルバムの1枚(Memento)[br]# ==========================================[br][/color]class Memento:[br] """内部の秘密の変数(2つの素数)を外に漏らさず凍結保存するカプセル"""[br] def __init__(self, p1, p2):[br] self._p1 = p1 # 外部から直接書き換えられないように私的変数にする[br] self._p2 = p2[br] def get_saved_primes(self):[br] """写真の現像:保存された素数のペアを返す"""[br] return self._p1, self._p2[br][br][color=#0000ff]# ==========================================[br]# 【アプリ本体】写真機能のある、2つの素数の積を表示するマシン(Originator)[br]# ==========================================[br][/color]class RandN_App:[br] """乱数で素数算をつくる。自身の写真を撮ったり、写真から復元する"""[br] def __init__(self):[br] self.p1 = 2[br] self.p2 = 3[br] self.prime_list = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 127] # 候補の素数たち[br] def play(self):[br] """時間を進める:ランダムに2つの素数を選んで式と答えを表示"""[br] self.p1 = random.choice(self.prime_list)[br] self.p2 = random.choice(self.prime_list)[br] print(f"{self.p1} × {self.p2} = {self.p1 * self.p2}")[br] def createMemento(self) -> Memento:[br] """カシャッ!とスナップ写真を撮影してMementoインスタンスを作る"""[br] print(f"[Save] 現在の {self.p1}×{self.p2} の瞬間をアルバムに記録しました。")[br] return Memento(self.p1, self.p2)[br] def restoreMemento(self, memento: Memento):[br] """アルバムから写真を1枚選び、その時の世界線へ没入ワープする"""[br] self.p1, self.p2 = memento.get_saved_primes()[br] print(f"[Restore] アルバムから復元しました!")[br] print(f" {self.p1} × {self.p2} = {self.p1 * self.p2}")[br][br][color=#0000ff]# ==========================================[br]# 【アルバム帳とお世話係】(Caretaker)[br]# ==========================================[br][/color]class AlbumKeeper:[br] """[情報管理系] Mementoの中身には一切触れず、ただ写真を預かるだけの係"""[br] def __init__(self):[br] self.photo_album = {} # お気に入りの名前をつけて保存する辞書[br] def add_photo(self, label, memento: Memento):[br] self.photo_album[label] = memento[br] def get_photo(self, label) -> Memento:[br] return self.photo_album.get(label)[br][br]# ==========================================[br]# 【作動テスト】思い出アルバムによる時空ワープ[br]# ==========================================[br]print("=== Memento Album Algorithm: Start ===\n")[br][br]# 1. アプリとアルバム帳の準備[br]app = RandN_App()[br]album = AlbumKeeper()[br][br]print("--- 1. アプリを何回か遊ぶ ---")[br]app.play()[br]app.play() # 偶然素敵な数式が出たとしましょう[br][br]print("\n--- 2. おっ、お気に入りの数式が出たのでカシャッと保存! ---")[br]# ここで 23 × 31 = 713 が出たと仮定して、アルバムに「素敵」という名前で保存[br]saved_photo = app.createMemento()[br]album.add_photo("素敵", saved_photo)[br][br]print("\n--- 3. さらに時間は過ぎ、全く別の数式へ移り変わる ---")[br]app.play()[br]app.play()[br]app.play()[br][br]print("\n--- 4. あの時の「素敵」な思い出の数式にワープする(Restore) ---")[br]target_photo = album.get_photo("素敵")[br]if target_photo:[br] app.restoreMemento(target_photo)[br][b][OUT]やるたびに中身は変わります。[br]=== Memento Album Algorithm: Start ===[br][br]--- 1. アプリを何回か遊ぶ ---[br]23 × 23 = 529[br]7 × 43 = 301[br][br]--- 2. おっ、お気に入りの数式が出たのでカシャッと保存! ---[br][Save] 現在の 7×43 の瞬間をアルバムに記録しました。[br][br]--- 3. さらに時間は過ぎ、全く別の数式へ移り変わる ---[br]47 × 17 = 799[br]127 × 13 = 1651[br]47 × 127 = 5969[br][br]--- 4. あの時の「素敵」な思い出の数式にワープする(Restore) ---[br][Restore] アルバムから復元しました![br] 7 × 43 = 301[/b]
振り返りをかねて、つぎの課題にとりくもう。[br][br][color=#9900ff][u][b][size=150]課題:フィボナッチ数列を使って3つのパタンを擬人化することで、探求のヒントをつかもう。[br][/size][/b][/u][/color]フィボナッチ数列が素数Pの倍数になる番号をみつけたいとする。[br]フィボナッチ数のM番目までのリストがあるとする。fibM={1,1,2,3,5,8,13,21,34,55,89,....}[br][br]コマンド君は、戻るとはいってもただの消去だから、基本的にリストに追加して記録するのが得意技だ。[br]ひたすらMを増やして周期を見つけることはできるだろう。[br]P=2なfibMの剰余はrem={1,1,0,1,1,0,....}となり、周期3が0 mod2となる。[br][br]ステート君は、計算結果を状態として、状態を主に考える逆算思考ができる。[br]だから、剰余が一度0になると、次は剰余が0の前の剰余xと同じになり、その次は0+x=xとなるから、[br]2周目目の剰余は1周目のx倍になるだけだから、調べなくても周期的になることに気づくだろう。[br]p=3での剰余になおすと、[br]最初の4つが1,1,2,0だから、次の4つは2倍して2,2,4,0,つまり、2,2,1,0になることを予想して[br]確かめられた。rem={1,1,2,0,2,2,1,0,....}となり、周期4の倍数番目が0 mod3となる。[br][br]メメント君は、剰余が0になるときの番号を取りためることで、kを色々変化させてデータをアルバムに追加する。[br]そのために、剰余が0になる番号のリストを作りアルバム化することに気づく。[br]素数P={2,3,5,7,11,13,17,19,23,...}に対してfibMがPの倍数になる周期cycleを作ることにした。[br] cycle2=3[br]cycle5=5[br]cycle7=8[br]p=5のときの周期は5だけど、[br]それ以外ではp+1またはp-1になることに気がつくかもしれないと気づくかもしれない。[br][br]そこで、ウォールの定理の存在につながっていく。[br]kの倍数は最初に0(modk)となる番号p(k)を周期にして現れます。[br]kが5以外の素数なら、kの倍数の周期p(k)は、p-1,p+1またはその約数になる。[br]これを参考に探求のきっかにしてみよう。[br][br]M=slider(1,100,1)[br]fibM=IterationList(a+b,a,b,{1,1},M)={1,1,2,3,5,8,13,21,34,55,89,....}[br]p={2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53}[br]n=slider(1,Length(p),1)[br]P=p(n)とすると、n番目の素数がPとなる。[br]rem=Sequence(Mod(fibM(s),P),s,1,M)[br]plus=DivisorsList(P-1)[br]minus=DivisorsList(P+1)[br]a=cycle ∈ plus[br]b=cycle ∈ minus[br]c=a ∨ b[br]text1="rem="+rem+[br]"//素数Pの倍数の周期はcycle="+cycle+[br]"//cycleはP+1の約数"+plus+"か[br]//P-1の約数"+minus+"の中にあるか。\\"+c[br]ウォールの定理からcはいつもtrueを表示するでしょう。[br]text2="素数"+P+"の倍数分布"[br]text3=fibM[br]text3があると目や暗算でも確かめやすくなりますね。