対策する資格
Java Silver ・ 近日ORACLE Silver ・ 近日
Gold 保有者 監修1Z0-829 対応
モジュール(exports vs opens)難易度 高無料

モジュール com.lib が次のように宣言されている(exports は無く opens のみ)。

1   module com.lib {
2       opens com.lib.model;             // public class Entity を含む
3   }

別モジュール com.clientrequires com.lib;)が、ソースコードで次のように com.lib.model.Entity を直接 import して使おうとする。結果は?

4   package com.client;
5   import com.lib.model.Entity;
6   public class Main {
7       public static void main(String[] a) {
8           System.out.println(new Entity().name());
9       }
10  }
  1. Aコンパイル成功。opens により com.clientcom.lib.model の public 型をコンパイル時に参照できる。
  2. Bコンパイルエラー。opens実行時のリフレクティブアクセスのみを許可し、コンパイル時/通常コードの型参照は exports でなければ許可されない。
  3. Cコンパイルは成功するが、実行時に IllegalAccessException がスローされる。
  4. Dopensexports は同義であり、どちらを書いても結果は同じ。
正解・解説・誤答理由・ひっかけを見る▼ open
✓ 正解:BGold監修

解説

exportsopens は「公開」の意味がまったく違う。1段ずつ整理する。

exports com.x = 他モジュールが com.xpublic 型の public メンバを、コンパイル時も実行時も通常コードから使える。
opens com.x = 他モジュールが com.x実行時の深いリフレクションsetAccessible(true) を含む。private にも届く)でアクセスできる。ただし通常のコンパイル時の型参照は許可しない

opens はフレームワーク(Jackson の JSON マッピング、JPA、DI 等)が privateフィールドにリフレクションで触るために使うもの。 本問の com.clientimport して new Entity() という通常のコンパイル時参照をしているので、opens しかない com.lib.model は「コンパイルからは見えない」。よってコンパイルエラー

各誤答が違う理由
  • Aopens はコンパイル時可視性を与えない。通常の import/new は通らない。
  • Cそもそもコンパイルが通らないので、実行時例外に至らない。
  • D同義ではない。exports=通常アクセス(public のみ)、opens=実行時リフレクション(private 含む)。役割が直交する。
ひっかけ:opens の方が private まで開くんだから、当然 public も使えるはず(exports の上位互換)」という直感。実際は軸が別で、コンパイル時参照は exports 専用。両方欲しければ両方書く(exportsopens)。
実機確認の答え合わせ
opens のみの com.lib.model を client が import → javac がコンパイルエラー:
Main.java:5: error: package com.lib.model does not exist
import com.lib.model.Entity;
                    ^
Main.java:8: error: cannot find symbol  (class Entity)
Gold 保有者による書き下ろし解説・実機で検証済
この分野をもっと解いて、得点源に
モジュール を含む全問を分野別に演習できます(無料)。
演習する →