読み込み中...

Composite パターン

フリー百科事典『ウィキペディア(Wikipedia)』より
Composite パターン(コンポジット・パターン)とは、GoF(Gang of Four; 4人のギャングたち)によって定義された デザインパターンの1つである。

Composite パターンを用いるとディレクトリなどのような、木構造を伴う再帰的データ構造を表すことができる。

Composite パターンにおいて登場するオブジェクトは、「枝」と「葉」であり、

これらは共通のインターフェースを実装している。そのため、枝と葉を同様に扱えるというメリットがある。

クラス図

Composite パターンのクラス図を以下に挙げる。

利用例

Composite パターンを用いてディレクトリ構造を表すJavaプログラムの例を示す。 このプログラムは、
  • 枝を表すクラス : Folderクラス
  • 葉を表すクラス : Fileクラス
  • 共通インタフェース : FileInterfaceインタフェース
  • 実行例を示すためのクラス : DirectoryUserクラス

から構成される。

import java.util.ArrayList; import java.util.List; interface FileInterface { public void defaultMethod(int depth); public List getChildren(); public boolean addComponent(FileInterface c); public boolean removeComponent(FileInterface c); } class File implements FileInterface { private String name; public File(String name) { this.name = name; } public void defaultMethod(int depth) { for (int i = 0; i < depth; i++) System.out.print(" "); System.out.println("file:" + this.name); } public List getChildren() { return null; } public boolean addComponent(FileInterface c) { return false; } public boolean removeComponent(FileInterface c) { return false; } } class Folder implements FileInterface { private String name; private List fileList = new ArrayList(); public Folder(String name) { this.name = name; } public void defaultMethod(int depth) { for (int i = 0; i < depth; i++) System.out.print(" "); System.out.println("folder:" + name); for (FileInterface file : fileList) { file.defaultMethod(depth + 1); } } public List getChildren() { return this.fileList; } public boolean addComponent(FileInterface c) { return this.fileList.add(c); } public boolean removeComponent(FileInterface c) { return this.fileList.remove(c); } } public class DirectoryUser { public static void main(String [] args){ FileInterface root = new Folder("root"); FileInterface usr = new Folder("usr"); FileInterface var = new Folder("var"); FileInterface home = new Folder("home"); FileInterface user1 = new Folder("user1"); FileInterface file1 = new File("file1"); root.addComponent(usr); usr.addComponent(var); root.addComponent(home); home.addComponent(user1); user1.addComponent(file1); root.defaultMethod(0); } } 実行結果 folder:root folder:usr folder:var folder:home folder:user1 file:file1

注意事項

Wikipedia画像へのリンク(自分自身を参照している例)
Wikipedia画像へのリンク(循環している例)
Composite パターンを用いる際には、データ構造がきちんと木構造を保つようにしなければならない。 親子関係が循環してしまった場合、Component#operation()

を実行した際に無限ループに陥るからである。

例えば、利用例のソースコードを書き換えた以下のプログラムは、処理が
dir1.defaultMethod(0);

の行に達した時に、無限に出力し続けてしまう。

public class LoopExample { public static void main(String [] args){ FileInterface dir1 = new Folder("dir1"); FileInterface dir2 = new Folder("dir2"); FileInterface dir3 = new Folder("dir3"); dir1.addComponent(dir2); dir2.addComponent(dir3); dir3.addComponent(dir1); dir1.defaultMethod(0); } }

関係するパターン

Interpreter パターン : 文法表現が Composite パターンとなる場合が多い。

関連項目

 読み込み中...

ブログレシピコミュニティお小遣いふくびき壁紙写真

Copyright(C)2008 GMO Media, Inc. All Rights Reserved.