[小西ホームページ]   [目次・索引]   [前の授業]   [次の授業]

コンピュータIIIB(Javaアルゴリズム)第12回

目次
12.1 カプセル化
12.1.1 カプセル化とは
12.1.2 カプセル化の例
12.2 今後の予定
12.3 成績評価について
12.4 演習12
12.5 レポート課題
12.6 参考文献
索引
privateキーワード   カプセル化   情報隠蔽   モジュール  

12.1 カプセル化

12.1.1 カプセル化とは

オブジェクト指向プログラミングにおける重要な概念の一つに、カプセル化があります。 カプセル化encapsulation ) とは、外部からデータの構造が見えないようにし、限られた方法でしかデータにアクセスできないようにすることです。 適切にカプセル化を行いますと、プログラムの作成・変更が容易になります。

カプセル化の本質は情報隠蔽です。 ここで、 情報隠蔽information hiding ) とは、外部に対して必要最小限の情報だけを公開するという考え方です。 公開部分は、外部から利用されますので、簡単には変更できません。 一方、隠蔽部分は、外部から利用されませんので、いつでも変更できます。

関連する情報をひとまとめにしてからカプセル化を行ったプログラム単位は、 モジュールmodule ) と呼ばれます。 プログラムを作成するときに、モジュールを一つ一つ作成するようにしますと、プログラミングがやりやすくなります。

Java言語では、クラスがモジュールの役割を果たします。 つまり、関連する情報は一つのクラスにまとめ、クラスごとにプログラムを作成するということです。 また、情報隠蔽については、クラスのインスタンス変数やインスタンス・メソッドなどが対象となります。 公開するインスタンス変数と隠蔽するインスタンス変数、公開するインスタンス・メソッドと隠蔽するインスタンス・メソッドに分別します。 隠蔽されたインスタンス変数やインスタンス・メソッドを変更しても、他のプログラムは変更しなくてよいのです。

12.1.2 カプセル化の例

カプセル化の例として、何時何分という時刻のクラスを考えます。 クラス名を Time とし、何時を変数 hour , 何分を変数 minute で表すことにします。 メイン・プログラムでは、13時15分というデータを格納してから取り出します。

Time.java(第1版)
/*  1*/ class Time {
/*  2*/     int hour;
/*  3*/     int minute;
/*  4*/ }
TimeMain.java(第1版)
/*  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%

もし、変数名 hourminute を、それぞれ jifun に変更しますと、メイン・プログラムも変えなければなりません。

Time.java(第2版)
/*  1*/ class Time {
/*  2*/     int ji;
/*  3*/     int fun;
/*  4*/ }
TimeMain.java(第2版)
/*  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 と書きますと、その情報が隠蔽されます。

Time.java(第3版)
/*  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*/ }
TimeMain.java(第3版)
/*  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*/ }

カプセル化を行ったことによって、隠蔽部分であるインスタンス変数を変更しても、メイン・プログラムはそのままでよくなります。

Time.java(第4版)
/*  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*/ }
TimeMain.java(第3版のまま)
/*  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 にするのが普通です。


12.2 今後の予定

今後の予定は次の通りです。

1月20日(金)
授業中に試験を行います。 今日の演習問題をよく見直しておいてください。
1月27日(金)
全員にレポートの提出状況をメールで知らせます。
1月31日(火)
レポートの最終締め切り日です。 これ以降提出されたレポートは採点しません。

12.3 成績評価について

この授業の成績は、レポートの提出と試験の得点で決まります。

S
レポートはすべて提出。 レポートの完成度が高い。 試験の得点は「優」相当。
A
レポートはすべて提出。 試験の得点は「優」相当。
B
レポートはおおむね提出。 試験の得点は「良」相当。
C
レポートは未提出が目立つ。 試験の得点は「可」相当。
F
レポートは未提出が多い。 試験の得点は「不可」相当。

成績に関して次のような事情のある人はメールで連絡してください。 できる限り対処します。


12.4 演習12

問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.5 レポート課題

今日の演習12の答案(Javaプログラム)をメールで提出してください。 メールの差出人は学内のアドレス(b04a001@twcu.ac.jpなど)とし、メールの宛先はkonishi@twcu.ac.jpとします。 メールの本文には、学生番号、氏名、科目名、授業日(1月13日)を明記してください。


12.6 参考文献


[小西ホームページ]   [目次・索引]   [前の授業]   [次の授業]

2006年1月13日更新
小西 善二郎 <konishi@twcu.ac.jp>
Copyright (C) 2006 Zenjiro Konishi. All rights reserved.