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

情報処理技法(Javaプログラミング)I 第13回

目次
索引

レコードとしてのインスタンス

レコードとは

レコード record )とは、(関連性のある)いくつかのデータをひとまとめにしたものです。 次の図はレコードのイメージを表しています。

レコードのイメージ
レコードのイメージ

レコードの構成要素を フィールド field )とよびます。

以前、データをまとめたものとして配列を取り上げました。 配列とレコードには次のような違いがあります。

ここで、レコード型とは、レコードを表すデータ型です。

レコード型は、データ型としての名前、フィールドの名前、およびそれぞれのフィールドのデータ型で定義されます。 上記のイメージの場合は、データ型としての名前をTime, フィールドの名前を hour minute , フィールドのデータ型を整数型(int型)と定義します。

実は、Javaにはレコードはありません。 しかし、 インスタンス instance )というものがレコードの機能を持っています。 レコードとインスタンスの対応は次の通りです。

レコードとインスタンスの対応
一般 Java
レコード インスタンス
フィールド インスタンス変数
レコード型 クラス

この授業では、インスタンス、フィールド、およびクラスという用語を用います。

ちなみに、プログラミング言語によっては、レコードのことを構造体とよびます。

インスタンスの使い方

レコードとしてのインスタンスを使うには、まず、クラスを定義します。 フィールドのデータ型が整数型(int型)なら、クラスの定義は次のようになります。

class クラス {
    int フィールド1;
    ...;
    int フィールドn;
}

ここで、「 クラス 」はクラス名、「 フィールド 」はフィールド名です。 上記の例なら、次の通りです。

/*  1*/ class Time { // クラス名
/*  2*/     int hour; // フィールド名
/*  3*/     int minute; // フィールド名
/*  4*/ }

これをファイルTime.javaに保存します。 このファイルがないと、以下のプログラムは動きません。

クラスが定義できたら、プログラムを作成します。 次のプログラムは、変数を宣言し、インスタンスを生成してその変数に格納し、インスタンスのフィールドにデータを格納し、その値を表示するものです。

/*  1*/ class TimeTest { // Timeクラスのテスト
/*  2*/     public static void main (String[] args) {
/*  3*/         Time x; // 宣言
/*  4*/         x = new Time(); // 生成
/*  5*/         x.hour = 9; // hourフィールドに格納
/*  6*/         x.minute = 30; // minuteフィールドに格納
/*  7*/         System.out.println(x.hour + ":" + x.minute); // フィールドの値を出力
/*  8*/     }
/*  9*/ }
24102a1:java1 k12x1001$ java TimeTest
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
9:30
24102a1:java1 k12x1001$

変数に整数(int型)を格納するには、変数の宣言が必要でした。 変数にインスタンスを格納するにも、同様に変数の宣言が必要です。 これは次のように書きます。

クラス 変数;

今まで int と書いていた部分をクラス名に変えます。 3行目で、変数 x をTimeクラスと宣言します。

クラスのインスタンスは、明示的に生成してはじめて使えます。 インスタンスを生成するには、キーワード new を用います。 「 クラス 」のインスタンスを生成して、「 変数 」に格納するには、

変数 = new クラス();

と書きます。 4行目で、Timeクラスのインスタンスを生成して、変数 x に格納します。

なお、次のように書くと、変数の宣言とインスタンスの生成、そしてその変数への格納が同時に行えます。

クラス 変数 = new クラス();

インスタンスのフィールドにデータを格納するには、次のような文を用います。

変数.フィールド = ;

これで、「 変数 」に格納されているインスタンスの「 フィールド 」に、「 」の値が格納されます。 5行目で、インスタンス x hour フィールドに9が格納され、6行目で、 minute フィールドに30が格納されます。

インスタンスのフィールドに格納されたデータを取り出すには、式

変数.フィールド

を用います。 この式の値は、「 変数 」に格納されているインスタンスの「 フィールド 」に格納されているデータです。 7行目で、インスタンス x hour フィールドと minute フィールドからデータを取り出して、出力します。

コンストラクタ

クラスのインスタンスを扱う場合、これまでは、インスタンスを生成してからフィールドに値を格納しました。 つまり、次のように書きました。

Time x = new Time();
x.hour = 9;
x.minute = 30;

配列では、初期化を使ってプログラムを短くしました。 インスタンスでも、 コンストラクタ constructor )というものを用いると、初期化に相当することができます。 上記の例なら、次のように書けます。

Time x = new Time(9, 30);

もし、クラスが

class クラス {
    int フィールド1;
    ...;
    int フィールドn;
}

と定義されていたら、次のように定義し直すと、コンストラクタが使えるようになります。

class クラス {
    int フィールド1;
    ...;
    int フィールドn;
    クラス (int フィールド1, ..., int フィールドn) {
        this.フィールド1 = フィールド1;
        ...
        this.フィールドn = フィールドn;
    }
}

これで、式

new クラス(引数1, ..., 引数n)

を呼び出すと、「 フィールド 1 」に「 引数 1 」を格納し、…、「 フィールド n 」に「 引数 n 」を格納したインスタンスになります。

コンストラクタはメソッドに似ています。 どちらも、あらかじめ定義しておき、呼出しによってその機能が働きます。 メソッドとコンストラクタは以下の点が異なります。

次のプログラムでは、コンストラクタも定義したクラスTime2を定義しています。 Time2.javaの4行目から7行目までが、コンストラクタ定義です。 このコンストラクタは、第1引数を hour フィールドに格納し、第2引数を minute フィールドに格納します。 Time2Test.javaの3行目で、このコンストラクタが呼び出され、 hour フィールドに9が格納され、 minute フィールドに30が格納されます。

Time2.java
/*  1*/ class Time2 { // 時刻2
/*  2*/     int hour;
/*  3*/     int minute;
/*  4*/     Time2 (int hour, int minute) { // コンストラクタ定義
/*  5*/         this.hour = hour;
/*  6*/         this.minute = minute;
/*  7*/     }
/*  8*/ }
Time2Test.java
/*  1*/ class Time2Test { // Time2クラスのテスト
/*  2*/     public static void main (String[] args) {
/*  3*/         Time2 x = new Time2(9, 30); // コンストラクタ呼出し
/*  4*/         System.out.println(x.hour + ":" + x.minute);
/*  5*/     }
/*  6*/ }
24102a1:java1 k12x1001$ java Time2Test
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
9:30
24102a1:java1 k12x1001$

注意1. 上記のコンストラクタは、最も単純なものです。 一般的には、コンストラクタはより複雑なことが行えます。

注意2. 上記のようにコンストラクタを定義すると、 new Time2() ではインスタンスが生成できなくなります。 実は、コンストラクタを定義しなければ引数なしのコンストラクタが自動的に用意され、コンストラクタを定義すると定義したものだけが使えるという規則になっています。 引数ありと引数なしの両方のコンストラクタを使うには、次のように両方定義します。

class Time3 {
    int hour;
    int minute;
    Time3 () {
    }
    Time3 (int hour, int minute) {
        this.hour = hour;
        this.minute = minute;
    }
}

インスタンスの使用例

インスタンスの例として、以前考えた買い物の問題を再び取り上げます。

A子は店に行き、150円のペットボトルを1本と120円の缶ジュースを3本買いました。 A子はいくら支払わなければならないでしょうか。

/*  1*/ class SomeDrinks2 { // いくつかの飲み物2
/*  2*/     public static void main (String[] args) {
/*  3*/         int total, pet = 150, can = 120;
/*  4*/         total = pet + 3 * can;
/*  5*/         System.out.println(total);
/*  6*/     }
/*  7*/ }

このプログラムでは、値段だけを変数に格納し、個数は変数に格納しませんでした。 インスタンスを利用すれば、値段と個数をまとめて格納できます。

ファイルBuy.javaでは、買った物を格納するために、Buyというクラスを定義します。 price というフィールドには値段を格納し、 count というフィールドには個数を格納します。

/*  1*/ class Buy { // 買った物
/*  2*/     int price; // 値段
/*  3*/     int count; // 個数
/*  4*/ }

ファイルBuyMain.javaでは、Buyクラスの変数 pet を宣言し、Buyクラスのインスタンスを生成して pet に格納します。 変数 can についても同様に宣言・生成します。 インスタンス pet にはペットボトルの値段と個数を格納し、インスタンス can には缶ジュースの値段と個数を格納します。 最後に、合計金額を計算します。

/*  1*/ class BuyMain { // 買った物のメイン
/*  2*/     public static void main (String[] args) {
/*  3*/         int total;
/*  4*/         Buy pet = new Buy(); // 宣言と生成
/*  5*/         Buy can = new Buy(); // 宣言と生成
/*  6*/         pet.price = 150; // ペットボトルの値段は150円
/*  7*/         pet.count = 1; // ペットボトルの個数は1個
/*  8*/         can.price = 120; // 缶ジュースの値段は120円
/*  9*/         can.count = 3; // 缶ジュースの個数は3個
/* 10*/         total = pet.count * pet.price + can.count * can.price;
/* 11*/         System.out.println(total);
/* 12*/     }
/* 13*/ }
24102a1:java1 k12x1001$ java BuyMain
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
510
24102a1:java1 k12x1001$

続いて、コンストラクタを利用して、このプログラムを短くします。

ファイルBuy2.javaでは、クラスBuy2を定義します。 このクラスのコンストラクタは、引数が値段、個数の順になるように定義します。

/*  1*/ class Buy2 { // 買った物2
/*  2*/     int price;
/*  3*/     int count;
/*  4*/     Buy2 (int price, int count) { // コンストラクタ定義
/*  5*/         this.price = price;
/*  6*/         this.count = count;
/*  7*/     }
/*  8*/ }

ファイルBuy2Main.javaでは、コンストラクタを呼び出して、Buy2クラスの変数 pet にペットボトルの値段と個数を格納します。 同様に、Buy2クラスの変数 can に缶ジュースの値段と個数を格納します。 最後に、合計金額を計算します。

/*  1*/ class Buy2Main { // 買った物2のメイン
/*  2*/     public static void main (String[] args) {
/*  3*/         int total;
/*  4*/         Buy2 pet = new Buy2(150, 1); // コンストラクタ呼出し
/*  5*/         Buy2 can = new Buy2(120, 3); // コンストラクタ呼出し
/*  6*/         total = pet.count * pet.price + can.count * can.price;
/*  7*/         System.out.println(total);
/*  8*/     }
/*  9*/ }
24102a1:java1 k12x1001$ java Buy2Main
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
510
24102a1:java1 k12x1001$

試験について

シラバスに書いた通り、来週の授業中に試験を行います。 来週は欠席しないようにしてください。

来週の試験は、授業の教室で、授業の後半の30分間で実施します。 試験の方法は筆記試験で、資料の持ち込みやパソコンの利用は不可とします。

以下に練習問題(全5問)を載せますので、試験勉強の参考にしてください。 この問題はレポート課題ではないので、回答を提出する必要はありません。

問1. 以下のJavaのプログラムは、入力された半径の円の周囲の長さを求めるものです。 ここで、円の周囲の長さは2×円周率×半径で求めます。 【ア】、【イ】、【ウ】、【エ】に適切な文字列を記入して、プログラムを完成させてください。

/*  1*/ import java.io.*;
/*  2*/
/*  3*/ class CircleLength {
/*  4*/     public static void main (String[] args) throws IOException {
/*  5*/         InputStreamReader isr = new InputStreamReader(System.in);
/*  6*/         BufferedReader br = new BufferedReader(isr);
/*  7*/         【ア】 radius, length;
/*  8*/         System.out.print("半径を入力してください: ");
/*  9*/         radius = Double.parseDouble(br.readLine());
/* 10*/         length 【イ】 2.0 【ウ】 Math.PI 【エ】 radius;
/* 11*/         System.out.println("円の周囲の長さは " + length + " です。");
/* 12*/     }
/* 13*/ }
24102a1:java1 k12x1001$ java CircleLength
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
半径を入力してください: 50.0
円の周囲の長さは 314.1592653589793 です。
24102a1:java1 k12x1001$

問2. 以下のJavaのプログラムは、入力された年がオリンピック・イヤーかどうかを判定するものです。 ここで、オリンピック・イヤーとは、4で割り切れる年とします。 【ア】、【イ】、【ウ】、【エ】に適切な文字列を記入して、プログラムを完成させてください。

/*  1*/ import java.io.*;
/*  2*/
/*  3*/ class OlympicYear {
/*  4*/     public static void main (String[] args) throws IOException {
/*  5*/         InputStreamReader isr = new InputStreamReader(System.in);
/*  6*/         BufferedReader br = new BufferedReader(isr);
/*  7*/         int year;
/*  8*/         System.out.print("年を入力してください: ");
/*  9*/         year = Integer.parseInt(br.readLine());
/* 10*/         【ア】 (year 【イ】 4 【ウ】 0) {
/* 11*/             System.out.println("オリンピック・イヤーです。");
/* 12*/         } 【エ】 {
/* 13*/             System.out.println("オリンピック・イヤーではありません。");
/* 14*/         }
/* 15*/     }
/* 16*/ }
24102a1:java1 k12x1001$ java OlympicYear
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
年を入力してください: 2012
オリンピック・イヤーです。
24102a1:java1 k12x1001$ java OlympicYear
年を入力してください: 2015
オリンピック・イヤーではありません。
24102a1:java1 k12x1001$

問3. 以下のJavaのプログラムは、1から1000までの合計を計算するものです。 【ア】、【イ】、【ウ】、【エ】に適切な文字列を記入して、プログラムを完成させてください。

/*  1*/ class Summation1000 {
/*  2*/     public static void main (String[] args) {
/*  3*/         int i, sum = 【ア】;
/*  4*/         【イ】 (i = 1; i 【ウ】 1000; i++) {
/*  5*/             sum 【エ】 i;
/*  6*/         }
/*  7*/         System.out.println("合計は " + sum + " です。");
/*  8*/     }
/*  9*/ }
24102a1:java1 k12x1001$ java Summation1000
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
合計は 500500 です。
24102a1:java1 k12x1001$

問4. 以下のJavaのプログラムは、{300, 200, 100, 300, 200}で表される配列の最小値を求めるものです。 【ア】、【イ】、【ウ】、【エ】に適切な文字列を記入して、プログラムを完成させてください。

/*  1*/ class ArrayMinimum {
/*  2*/     public static void main (String[] args) {
/*  3*/         int i, min;
/*  4*/         【ア】 a = {300, 200, 100, 300, 200};
/*  5*/         min = a[0];
/*  6*/         for (i = 1; i 【イ】 a.【ウ】; i++) {
/*  7*/             if (a[i] 【エ】 min) {
/*  8*/                 min = a[i];
/*  9*/             }
/* 10*/         }
/* 11*/         System.out.println("最小値は " + min + " です。");
/* 12*/     }
/* 13*/ }
24102a1:java1 k12x1001$ java ArrayMinimum
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
最小値は 100 です。
24102a1:java1 k12x1001$

問5. 以下のJavaのプログラムは、2つの整数の差を求めるメソッド diff を定義し、それを呼び出すものです。 ここで、 x y の差は、 x y より大きいならば x y , そうでなければ y x とします。 【ア】、【イ】、【ウ】、【エ】に適切な文字列を記入して、プログラムを完成させてください。

/*  1*/ class IntDifference {
/*  2*/     public static void main (String[] args) {
/*  3*/         System.out.println("diff(5, 2) = " + diff(5, 2));
/*  4*/         System.out.println("diff(4, 4) = " + diff(4, 4));
/*  5*/         System.out.println("diff(1, 7) = " + diff(1, 7));
/*  6*/     }
/*  7*/     static 【ア】 diff (int x, 【イ】 y) {
/*  8*/         if (x 【ウ】 y) {
/*  9*/             return x - y;
/* 10*/         } else {
/* 11*/             【エ】 y - x;
/* 12*/         }
/* 13*/     }
/* 14*/ }
24102a1:java1 k12x1001$ java IntDifference
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
diff(5, 2) = 3
diff(4, 4) = 0
diff(1, 7) = 6
24102a1:java1 k12x1001$

演習13

ある小学校で国語と算数の試験が行われました。 太郎君は国語が70点、算数が80点でした。 花子さんは国語が90点、算数が100点でした。 インスタンスを利用して、これらの得点データを格納し、全体の平均点(小数点以下切り捨て)を計算してください。

24102a1:java1 k12x1001$ java ScoreMain
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
85
24102a1:java1 k12x1001$

Score.javaでは、試験の得点を格納するするために、クラスScoreを定義します。 フィールド japanese には国語の得点を格納し、フィールド arithmetic には算数の得点を格納します。

ScoreMain.javaでは、Scoreクラスの変数 taro hanako を宣言・生成します。 インスタンス taro には太郎君の得点を格納し、インスタンス hanako には花子さんの得点を格納します。 最後に、すべての得点を合計し、4で割ります。

余力のある人は、Score2.javaで、コンストラクタも定義したクラスScore2を定義してください。

Score2Main.javaでは、コンストラクタを呼び出して、Score2クラスの変数 taro には太郎君の得点を格納し、Score2クラスの変数 hanako には花子さんの得点を格納します。 最後に、すべての得点を合計し、4で割ります。

24102a1:java1 k12x1001$ java Score2Main
Picked up JAVA_TOOL_OPTIONS: -Dfile.encoding=UTF-8
85
24102a1:java1 k12x1001$

レポート課題

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


参考文献


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

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