カスタムスコープ
カスタムスコープの仕組みについてマニュアルに一応の説明があるが、これは非常にわかりづらい。 しかし、誤解を恐れずに一言で言えば、「その時の条件に応じたプロバイダを返すオブジェクト」と言えるかと思う。 この動きをつぶさに追ってみる。
単純な例
package sample; import com.google.inject.*; public class Scope1 { public static final Scope SCOPE_OBJECT = new Scope() { public <T> Provider<T> scope(Key<T> key, Provider<T> provider) { return provider; } }; public static class Person {} public static void main(String[] args) { Injector injector = Guice.createInjector(new AbstractModule() { protected void configure() { bind(Person.class).in(SCOPE_OBJECT); } }); System.out.println(injector.getInstance(Person.class)); System.out.println(injector.getInstance(Person.class)); } }
これは非常に単純な例である。bind(Person.class).in(SCOPE_OBJECT)としているが、このSCOPE_OBJECTがこの場合はProvider<Person>を返すオブジェクト(スコープの実態)である。しかし、この例ではあまりスコープらしくない。 ともあれ、上記を実行すると、
sample.Scope1$Person@1ccb029 sample.Scope1$Person@1415de6
となる。この例ではスコープを使った意味が全くない。
アノテーションを使う例
次に専用のアノテーション(SampleScoped)を登場させてみる。この例も前と同じで全く意味がない。 injector.getInstance(Person.class)するたびに異なるPersonが返される。
package sample; import java.lang.annotation.*; import com.google.inject.*; public class Scope2 { /** スコープ用アノテーション */ @Target({ ElementType.TYPE }) @Retention(RetentionPolicy.RUNTIME) @ScopeAnnotation public @interface SampleScoped {} public static final Scope SCOPE_OBJECT = new Scope() { public <T> Provider<T> scope(Key<T> key, Provider<T> provider) { return provider; } }; public static class Person {} public static class Sample { @Inject Person person; } public static void main(String[] args) { Injector injector = Guice.createInjector(new AbstractModule() { protected void configure() { // アノテーションをスコープにバインド bindScope(SampleScoped.class, SCOPE_OBJECT); } }); System.out.println(injector.getInstance(Sample.class).person); System.out.println(injector.getInstance(Sample.class).person); } }