オブジェクト指向プログラミングにおける重要な概念の一つに、カプセル化があります。 カプセル化 ( encapsulation ) とは、外部からデータの構造が見えないようにし、限られた方法でしかデータにアクセスできないようにすることです。 適切にカプセル化を行いますと、プログラムの作成・変更が容易になります。
カプセル化の本質は情報隠蔽です。 ここで、 情報隠蔽 ( information hiding ) とは、外部に対して必要最小限の情報だけを公開するという考え方です。 公開部分は、外部から利用されますので、簡単には変更できません。 一方、隠蔽部分は、外部から利用されませんので、いつでも変更できます。
関連する情報をひとまとめにしてからカプセル化を行ったプログラム単位は、 モジュール ( module ) と呼ばれます。 プログラムを作成するときに、モジュールを一つ一つ作成するようにしますと、プログラミングがやりやすくなります。
Java言語では、クラスがモジュールの役割を果たします。 つまり、関連する情報は一つのクラスにまとめ、クラスごとにプログラムを作成するということです。 また、情報隠蔽については、クラスのインスタンス変数やインスタンス・メソッドなどが対象となります。 公開するインスタンス変数と隠蔽するインスタンス変数、公開するインスタンス・メソッドと隠蔽するインスタンス・メソッドに分別します。 隠蔽されたインスタンス変数やインスタンス・メソッドを変更しても、他のプログラムは変更しなくてよいのです。
カプセル化の例として、何時何分という時刻のクラスを考えます。 クラス名を Time とし、何時を変数 hour , 何分を変数 minute で表すことにします。 メイン・プログラムでは、13時15分というデータを格納してから取り出します。
/* 1*/ class Time { /* 2*/ int hour; /* 3*/ int minute; /* 4*/ }
/* 1*/ class TimeMain { /* 2*/ public static void main (String[] args) { /* 3*/ Time t = new Time(); /* 4*/ t.hour = 13; /* 5*/ t.minute = 15; /* 6*/ System.out.println(t.hour + ":" + t.minute); /* 7*/ } /* 8*/ }
b04a001@AisaA1:~/comp3b% java TimeMain 13:15 b04a001@AisaA1:~/comp3b%
もし、変数名 hour と minute を、それぞれ ji と fun に変更しますと、メイン・プログラムも変えなければなりません。
/* 1*/ class Time { /* 2*/ int ji; /* 3*/ int fun; /* 4*/ }
/* 1*/ class TimeMain { /* 2*/ public static void main (String[] args) { /* 3*/ Time t = new Time(); /* 4*/ t.ji = 13; /* 5*/ t.fun = 15; /* 6*/ System.out.println(t.ji + ":" + t.fun); /* 7*/ } /* 8*/ }
一か所を変更するとすべてを変更しなければならないようでは、プログラムの変更は困難になってしまいます。
ここでカプセル化を行います。
まず、インスタンス変数は隠蔽します。
次に、インスタンス変数へアクセスするメソッドを用意し、これを公開します。
Java言語では、
private
と書きますと、その情報が隠蔽されます。
/* 1*/ class Time { /* 2*/ private int hour; /* 3*/ private int minute; /* 4*/ void setHour (int hour) { /* 5*/ this.hour = hour; /* 6*/ } /* 7*/ void setMinute (int minute) { /* 8*/ this.minute = minute; /* 9*/ } /* 10*/ int getHour () { /* 11*/ return this.hour; /* 12*/ } /* 13*/ int getMinute () { /* 14*/ return this.minute; /* 15*/ } /* 16*/ }
/* 1*/ class TimeMain { /* 2*/ public static void main (String[] args) { /* 3*/ Time t = new Time(); /* 4*/ t.setHour(13); /* 5*/ t.setMinute(15); /* 6*/ System.out.println(t.getHour() + ":" + t.getMinute()); /* 7*/ } /* 8*/ }
カプセル化を行ったことによって、隠蔽部分であるインスタンス変数を変更しても、メイン・プログラムはそのままでよくなります。
/* 1*/ class Time { /* 2*/ private int ji; /* 3*/ private int fun; /* 4*/ void setHour (int ji) { /* 5*/ this.ji = ji; /* 6*/ } /* 7*/ void setMinute (int fun) { /* 8*/ this.fun = fun; /* 9*/ } /* 10*/ int getHour () { /* 11*/ return this.ji; /* 12*/ } /* 13*/ int getMinute () { /* 14*/ return this.fun; /* 15*/ } /* 16*/ }
/* 1*/ class TimeMain { /* 2*/ public static void main (String[] args) { /* 3*/ Time t = new Time(); /* 4*/ t.setHour(13); /* 5*/ t.setMinute(15); /* 6*/ System.out.println(t.getHour() + ":" + t.getMinute()); /* 7*/ } /* 8*/ }
カプセル化を行うときは、どの情報を隠蔽し、どの情報を公開するかを適切に決定することが大事です。
特に、インスタンス変数はすべて
private
にするのが普通です。
今後の予定は次の通りです。
この授業の成績は、レポートの提出と試験の得点で決まります。
成績に関して次のような事情のある人はメールで連絡してください。 できる限り対処します。
問1. 以下のプログラムは、1から6までの数の中から2つを選ぶ組み合わせを列挙する(つもりの)ものです。 このプログラムの不具合を修正してください。
/* 1*/ class Combination6 { /* 2*/ public static void main (String[] args) { /* 3*/ int i, j; /* 4*/ for (i = 1; i <= 6; i++) { /* 5*/ for (j = 1; j <= i; j++) { /* 6*/ System.out.print(" " + i + "-" + j); /* 7*/ } /* 8*/ System.out.println(); /* 9*/ } /* 10*/ } /* 11*/ }
b04a001@AsiaA1:~/comp3b% java Combination6 1-1 // 1-1 は不要 2-1 2-2 // 2-2 は不要 3-1 3-2 3-3 // 3-3 は不要 4-1 4-2 4-3 4-4 // 4-4 は不要 5-1 5-2 5-3 5-4 5-5 // 5-5 は不要 6-1 6-2 6-3 6-4 6-5 6-6 // 6-6 は不要 b04a001@AsiaA1:~/comp3b%
問2. 以下のプログラムは、空列を括弧で6回囲む(つもりの)ものです。 このプログラムの不具合を修正してください。
/* 1*/ class Parenthesis6 { /* 2*/ public static void main (String[] args) { /* 3*/ printParenthesis(6); /* 4*/ System.out.println(); /* 5*/ } /* 6*/ static void printParenthesis (int n) { /* 7*/ if (n > 0) { /* 8*/ System.out.print("("); /* 9*/ System.out.print(")"); /* 10*/ printParenthesis(n - 1); /* 11*/ } /* 12*/ } /* 13*/ }
b04a001@AsiaA1:~/comp3b% java Parenthesis6 ()()()()()() // 正しくは (((((()))))) b04a001@AsiaA1:~/comp3b%
問3. 以下のプログラムは、1から6までの数を順次リストの先頭に挿入して、[6, 5, 4, 3, 2, 1] というリストを構成する(つもりの)ものです。 このプログラムの不具合を修正してください。
/* 1*/ class ListNode { /* 2*/ int data; /* 3*/ ListNode next; /* 4*/ ListNode (int data, ListNode next) { /* 5*/ this.data = data; /* 6*/ this.next = next; /* 7*/ } /* 8*/ }
/* 1*/ class List6 { /* 2*/ public static void main (String[] args) { /* 3*/ int i, DUMMY = 0; /* 4*/ ListNode x, list = new ListNode(DUMMY, null); /* 5*/ x = list; /* 6*/ for (i = 1; i <= 6; i++) { /* 7*/ x.next = new ListNode(i, x.next); /* 8*/ x = x.next; /* 9*/ } /* 10*/ for (x = list.next; x != null; x = x.next) { /* 11*/ System.out.print(" " + x.data); /* 12*/ } /* 13*/ System.out.println(); /* 14*/ } /* 15*/ }
b04a001@AsiaA1:~/comp3b% java List6 1 2 3 4 5 6 // 正しくは 6 5 4 3 2 1 b04a001@AsiaA1:~/comp3b%
問4. 次のプログラムは、以下の木の節点の個数を数える(つもりの)ものです。 このプログラムの不具合を修正してください。
1 | +-------+-------+ | | 2 3 | | +---+---+ +---+ | | | 4 5 6
/* 1*/ class TreeNode { /* 2*/ int data; /* 3*/ TreeNode left; /* 4*/ TreeNode right; /* 5*/ TreeNode (int data, TreeNode left, TreeNode right) { /* 6*/ this.data = data; /* 7*/ this.left = left; /* 8*/ this.right = right; /* 9*/ } /* 10*/ }
/* 1*/ class CountNodes { /* 2*/ public static void main (String[] args) { /* 3*/ int DUMMY = 0; /* 4*/ TreeNode tree = new TreeNode(DUMMY, null, null); /* 5*/ tree.right = new TreeNode(1, null, null); /* 6*/ tree.right.left = new TreeNode(2, null, null); /* 7*/ tree.right.right = new TreeNode(3, null, null); /* 8*/ tree.right.left.left = new TreeNode(4, null, null); /* 9*/ tree.right.left.right = new TreeNode(5, null, null); /* 10*/ tree.right.right.left = new TreeNode(6, null, null); /* 11*/ System.out.println(countNodes(tree.right)); /* 12*/ } /* 13*/ static int countNodes (TreeNode tree) { /* 14*/ if (tree == null) { /* 15*/ return 0; /* 16*/ } else { /* 17*/ return countNodes(tree.left) + countNodes(tree.right); /* 18*/ } /* 19*/ } /* 20*/ }
b04a001@AsiaA1:~/comp3b% java CountNodes 0 // 正しくは 6 b04a001@AsiaA1:~/comp3b%
問5. 次のプログラムは、長方形 x の幅と高さを長方形 y へコピーする(つもりの)ものです。 このプログラムの不具合を修正してください。
/* 1*/ class Rectangle { /* 2*/ int width; /* 3*/ int height; /* 4*/ void setWidth (int width) { /* 5*/ this.width = width; /* 6*/ } /* 7*/ void setHeight (int height) { /* 8*/ this.height = height; /* 9*/ } /* 10*/ int getWidth () { /* 11*/ return this.width; /* 12*/ } /* 13*/ int getHeight () { /* 14*/ return this.height; /* 15*/ } /* 16*/ void copyFrom (Rectangle rect) { /* 17*/ rect.setWidth(this.getWidth()); /* 18*/ rect.setHeight(this.getHeight()); /* 19*/ } /* 20*/ }
/* 1*/ class RectangleMain { /* 2*/ public static void main (String[] args) { /* 3*/ Rectangle x = new Rectangle(); /* 4*/ Rectangle y = new Rectangle(); /* 5*/ x.setWidth(16); /* 6*/ x.setHeight(9); /* 7*/ y.copyFrom(x); /* 8*/ System.out.println(y.getWidth()); /* 9*/ System.out.println(y.getHeight()); /* 10*/ } /* 11*/ }
b04a001@AsiaA1:~/comp3b% java RectangleMain 0 // 正しくは 16 0 // 正しくは 9 b04a001@AsiaA1:~/comp3b%
今日の演習12の答案(Javaプログラム)をメールで提出してください。 メールの差出人は学内のアドレス(b04a001@twcu.ac.jpなど)とし、メールの宛先はkonishi@twcu.ac.jpとします。 メールの本文には、学生番号、氏名、科目名、授業日(1月13日)を明記してください。