link もっと前
   2013年 3月 28日 -
      2013年 3月 28日  
link もっと後

link 未来から過去へ表示(*)
link 過去から未来へ表示

日々

link permalink

Java のリフレクションとコンストラクタ

Java のリフレクションを使ったコンストラクタの取得で躓いています。正解が分からない…。

あるクラス Test があり、パラメータにクラス A しか取らないコンストラクタがあるとします。

クラス Test のコンストラクタに対し、クラス A と全く関係ないクラス C のインスタンスを渡せばコンパイルエラーになります。クラス A の派生クラス B のインスタンスであれば渡せます。これはオブジェクト指向言語なら当たり前です。

しかしリフレクションを使ってコンストラクタを得ようとすると、派生クラスを渡せるコンストラクタをどうやって取得すれば良いかわからないのです。


public class Test {
    public Test(A a) {
    }
}

public class A {
    public A() {
    }
}

public class B extends A {
    public B() {
    }
}

public class Main {
    public static void main() {
        Test obj1 = new Test(new A()); //OK
        Test obj2 = new Test(new B()); //OK

        Constructor<Test> cons1 = Test.class.getConstructor(A.class); //OK
        Constructor<Test> cons2 = Test.class.getConstructor(B.class); //NG

        Test obj3 = cons1.newInstance(new A()); //OK
        Test obj4 = cons1.newInstance(new B()); //OK

        //Test obj5 = cons2.newInstance(new B());
    }
}

上記のようなコードがあったとします(try 〜 catch は省いています)。

クラス Test には、クラス A をパラメータにとるコンストラクタしかありません。従って、派生クラス B をパラメータに取るコンストラクタをくれ(= Test.class.getConstructor(B.class))と頼むと「そんなコンストラクタは無い」と例外がスローされます。

どうして Test.class.getConstructor(B.class) としたとき、「ありません」なんだろうか?基底クラスを取るコンストラクタ(Test(A a))があるのだから、そちらを教えてくれれば良いのに…。

鶏と卵

クラス Test には、クラス B を取るコンストラクタはないから、この動きは正しい。
とか、
基底クラス A を取るコンストラクタ(cons1)を得て、派生クラス B のインスタンスを渡せば良いだけだ。(つまり obj4 を生成しているように書く)
という反論はわかるのですが、それだとせっかくのリフレクションなのに本末転倒になってしまいませんか?

クラス Test の利用者側の立場からすれば、クラス Test に、クラス A を取るコンストラクタがあるか?クラス B を取るコンストラクタがあるか?なんて知らないわけです。自分が呼び出したいコンストラクタが既にあるとわかっているなら、リフレクションなど使わずに直接呼べば良いのです…。

クラス Test のコンストラクタを全部取得して、クラス B の基底クラスを取るコンストラクタがないかどうか、全て探すしかないのでしょうか…。うーん、それはさすがに格好悪いような…。

[編集者: すずき]
[更新: 2013年 3月 30日 03:04]
link 編集する

コメント一覧

  • コメントはありません。
open/close この記事にコメントする



link もっと前
   2013年 3月 28日 -
      2013年 3月 28日  
link もっと後

管理用メニュー

link 記事を新規作成

合計:  counter total
本日:  counter today

link About www.katsuster.net
RDF ファイル RSS 1.0
QR コード QR コード

最終更新: 4/20 00:42

カレンダー

<2013>
<<<03>>>
-----12
3456789
10111213141516
17181920212223
24252627282930
31------

最近のコメント 5件

  • link 19年04月01日
    すずき 「どの CPU というかシステムでも同じ傾...」
    (更新:04/05 11:03)
  • link 19年04月01日
    hdk 「去年Ryzen 7 1700で測りました...」
    (更新:04/02 22:48)
  • link 19年03月05日
    すずき 「> オシロの波形見てて気がつかなか...」
    (更新:03/21 17:45)
  • link 19年03月05日
    kml 「> 自分が持っている RockPr...」
    (更新:03/20 21:30)
  • link 19年03月10日
    すずき 「ありがとー!!」
    (更新:03/13 09:46)

最近の記事 3件

link もっとみる
  • link 19年04月19日
    すずき 「[RISC-V の SoC を見ていた] Linux が動くくらい...」
    (更新:04/20 00:42)
  • link 19年04月18日
    すずき 「[Linux の DMA] 昨今のキャッシュを持った CPU では...」
    (更新:04/20 00:29)
  • link 19年04月13日
    すずき 「[レジスタダンプ、書き換えツール memaccess - ちょ] ...」
    (更新:04/19 23:35)

こんてんつ

open/close wiki
open/close Java API

過去の日記

open/close 2002年
open/close 2003年
open/close 2004年
open/close 2005年
open/close 2006年
open/close 2007年
open/close 2008年
open/close 2009年
open/close 2010年
open/close 2011年
open/close 2012年
open/close 2013年
open/close 2014年
open/close 2015年
open/close 2016年
open/close 2017年
open/close 2018年
open/close 2019年
open/close 過去日記について

その他の情報

open/close アクセス統計
open/close サーバ一覧
open/close サイトの情報