独習java第4版でわからない所があります。
abstract class Widget {
String color;
abstract int getMass();
public String toString() {
return getClass().getName() + ": " +
color + ", " + getMass();
}
}
class WidgetA extends Widget {
final static int MASS = 4;
WidgetA(String color) {
this.color = color;
}
int getMass() {
return MASS;
}
}
class WidgetB extends Widget {
final static int MASS = 1;
WidgetB(String color) {
this.color = color;
}
int getMass() {
return MASS;
}
}
class WidgetC extends Widget {
final static int MASS = 5;
WidgetC(String color) {
this.color = color;
}
int getMass() {
return MASS;
}
}
class WidgetD extends Widget {
final static int MASS = 17;
WidgetD(String color) {
this.color = color;
}
int getMass() {
return MASS;
}
}
class WidgetTypes {
static int NUMWIDGETS = 6;
public static void main(String args[]) {
// 部品を格納する領域を宣言して割り当てる
Widget widgets[] = new Widget[NUMWIDGETS];
// 部品を作成する
widgets[0] = new WidgetC("Red");
widgets[1] = new WidgetA("Green");
widgets[2] = new WidgetD("Yellow");
widgets[3] = new WidgetB("Magenta");
widgets[4] = new WidgetA("Black");
widgets[5] = new WidgetC("White");
// 部品を処理する
int totalMass = 0;
for(int i = 0; i < NUMWIDGETS; i++) {
Widget w = widgets[i];
System.out.println(w);
totalMass += w.getMass();
}
// 総重量を表示する
System.out.println("Total mass = " + totalMass);
}
}
これはある問題の解答ですが、僕にはどうしても理解出来ない部分があります。
mainのforループ内で
Widget w = widgets[i];
totalMass += w.getMass();
となっています。
この仕組がわかりません。
まず Widget w = widgets[i] でWidget型の変数wにWidget型のwidget[i]を代入しているのにw.getMass()がエラーにならない理由がわかりません。(Widgetクラスは抽象クラスなのにwがWidgetクラスのインスタンスになっている?)
これは 抽象メソッド( abstract int getMass() ) があるためでしょうか?
ちなみにWidgetクラスとそのサブクラスからこの抽象メソッドを削除したらコンパイルエラーが出ました。
ではなぜ抽象クラスの抽象メソッドから、そのサブクラスのメソッドまで範囲が伸びるのでしょうか?
どういう仕組でしょうか?
この質問を書きながら思ったのですが、どうも配列の仕組みや抽象クラス・メソッドの仕組み、「オブジェクト」と「インスタンス」の違いがよくわかってないようです。
多分問題の本質はそこにあると思うんです。
駄文で申し訳ないです。
よろしくお願いします。
No.3ベストアンサー
- 回答日時:
質問者さんがわかっていないのは「ポリモーフィズム」ですね。
それはさておき、> Widget w = widgets[i] でWidget型の変数wにWidget型のwidget[i]を代入しているのに
widgets[i]に入っているのはWidgetクラスのインスタンスではなく、
WidgetCやWidgetAやWidgetDクラスのインスタンスです。
具体的には
widgets[0] = new WidgetC("Red");
としているのですから、forループの最初(i=0)の
Widget w = widgets[i];
System.out.println(w);
totalMass += w.getMass();
でwに代入されるのは、WidgetCクラスのインスタンスです。
w.getMass()で呼ばれるのもWidgetCクラスのgetMass()です。
No.2
- 回答日時:
WidgetA、WidgetB、WidgetC、WidgetD
ちょっとだけ違う似たようなクラスですね。
配列にしないで、一個ずつ処理すると、
ステップが冗長になると思いませんか?
「なんとなく似ているクラス」でまとめる
ことができれば便利ですよね。
そこで登場するのがインターフェースとか
抽象クラスとか言うものです。
widgets[]の形式はWidgetですが、個別の
要素の実体はWidgetAとかWidgetBです。
どれもgetMassメソッドを持っていますから
w.getMass()はエラーになりません。
というか、抽象クラスの抽象メソッドでの
定義があるからgetMassがないとエラーに
なります。つまり、Widget型、あるいはその
サブクラスには必ずgetMassがあるはず
(なければならない)ということです。
抽象メソッドを削除すると、getMassは
各クラスの勝手なメソッドになるわけで、
個別のWidgetAやWidgetBにあるかないか
分かりませんよね。少なくともWidget型には
無いのですから、そういうメソッドは使えない
ことになります。
No.1
- 回答日時:
> この質問を書きながら思ったのですが、どうも配列の仕組みや抽象クラス・メソッドの仕組み、「オブジェクト」と「インスタンス」の違いがよくわかってないようです。
> 多分問題の本質はそこにあると思うんです。
違う、分かっていないのはそこじゃない。
抽象クラスも普通のクラスも(これから出るであろうインターフェースも)、それを派生させたクラスでメソッドをオーバーロードしたときの動作は変わりません。
継承したクラスでオーバーロードしたメソッドを継承元のメソッドの代わりに実行するだけです。
抽象クラスの抽象メソッドはその取って代わられたメソッドが、外の形だけあって実体が無いという状態になっているだけです。
だから、継承とオーバーロードの部分をもう一度勉強しましょう。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・漫画をレンタルでお得に読める!
- ・人生のプチ美学を教えてください!!
- ・10秒目をつむったら…
- ・あなたの習慣について教えてください!!
- ・牛、豚、鶏、どれか一つ食べられなくなるとしたら?
- ・【大喜利】【投稿~9/18】 おとぎ話『桃太郎』の知られざるエピソード
- ・街中で見かけて「グッときた人」の思い出
- ・「一気に最後まで読んだ」本、教えて下さい!
- ・幼稚園時代「何組」でしたか?
- ・激凹みから立ち直る方法
- ・1つだけ過去を変えられるとしたら?
- ・【あるあるbot連動企画】あるあるbotに投稿したけど採用されなかったあるある募集
- ・【あるあるbot連動企画】フォロワー20万人のアカウントであなたのあるあるを披露してみませんか?
- ・映画のエンドロール観る派?観ない派?
- ・海外旅行から帰ってきたら、まず何を食べる?
- ・誕生日にもらった意外なもの
- ・天使と悪魔選手権
- ・ちょっと先の未来クイズ第2問
- ・【大喜利】【投稿~9/7】 ロボットの住む世界で流行ってる罰ゲームとは?
- ・推しミネラルウォーターはありますか?
- ・都道府県穴埋めゲーム
- ・この人頭いいなと思ったエピソード
- ・準・究極の選択
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
Java初級 引数に適用できません
-
レコード件数の表示
-
CSVから読み込んだデータの保持...
-
Strutsでチェックボックスの値...
-
readLine()ではじめから読み直...
-
Google Apps Script で getRang...
-
java のメソッドの使い方について
-
String.containsの反対機能はあ...
-
クラスを作るとメソッドの数が...
-
public static void main (Stri...
-
StringBufferからStringへキャ...
-
なぜprotected overrideなのか
-
C# でメソッドに送られてきたOb...
-
androidのインターフェースにつ...
-
djUnit コンストラクタがpriva...
-
非静的 public final フィール...
-
voidとかstaticとか
-
メソッドのオーバーライド(java)
-
「タイプ初期化子が例外をスロ...
-
エクセルVBAで、条件に一致する...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
Java初級 引数に適用できません
-
String.containsの反対機能はあ...
-
レコード件数の表示
-
StringBufferからStringへキャ...
-
C# 点の描き方をおしえてくだ...
-
Google Apps Script で getRang...
-
YYYYMMDD書式の日付に対する適...
-
C# でメソッドに送られてきたOb...
-
なぜprotected overrideなのか
-
Java初心者です、エラーの意味...
-
Strutsでチェックボックスの値...
-
メソッド宣言の戻り値の型にク...
-
abstract と static を一緒に付...
-
return new使用時
-
Javaの関数名が長い?
-
メソッドの引数に指定されてい...
-
クラスを作るとメソッドの数が...
-
public static void main (Stri...
-
説明文書でのメソッド・変数の...
-
readLine()ではじめから読み直...
おすすめ情報