A scope is a condition for reusing injected objects, instead of creating a new one each time. For example, if a network connection was being injected then the scope would be while the network connection was open, reuse the object. When it is closed, open and inject a new one.
The implementation for this is to basically provide a factory object, implementing the Scope interface. This has one method:
Provider scope(Key<T> key, Provider<T> unscoped)
The Key object defines the injected type, the unscoped Provider can be used to create a new instance. Typically the returned Provider implementation will create the required object using the unscoped Provider and keep a reference to it while in scope. When the object is requested again the stored object is returned if still in scope. This example usage above is how I needed to use custom scopes but it is flexible enough to do all kinds of weird things.
Here is the Scope implementation. It is a bit more complex than necessary because for this example I wanted to allow objects to be created out of scope.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public class EnterableScope implements Scope | |
{ | |
private HashMap<Key<?>, Object> values; | |
public void enter() | |
{ | |
values = new HashMap<Key<?>, Object>(); | |
} | |
public void exit() | |
{ | |
values = null; | |
} | |
@Override | |
public <T> Provider<T> scope(final Key<T> key, final Provider<T> unscoped) | |
{ | |
return new Provider<T>() { | |
@Override | |
@SuppressWarnings("unchecked") | |
public T get() | |
{ | |
// if values is null return null otherwise return the stored value of one exists | |
Object object = values != null ? values.get(key) : null; | |
if(object == null) | |
{ | |
object = unscoped.get(); | |
// if we in scope, store it, otherwise always create a new one | |
if(values != null) | |
{ | |
values.put(key, object); | |
} | |
} | |
return (T) object; | |
} | |
}; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
@Target({ TYPE, METHOD }) @Retention(RUNTIME) @ScopeAnnotation | |
public @interface BookScoped | |
{ | |
} | |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public interface Hero | |
{ | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
@BookScoped | |
public class Hobbit implements Hero | |
{ | |
public Hobbit() | |
{ | |
System.out.println("created " + toString()); | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public class MiddleEarthModule extends AbstractModule | |
{ | |
private final EnterableScope scope; | |
public MiddleEarthModule() | |
{ | |
scope = new EnterableScope(); | |
} | |
@Override | |
protected void configure() | |
{ | |
bindScope(BookScoped.class, scope); | |
bind(Hero.class).to(Hobbit.class); | |
} | |
public void enter() | |
{ | |
scope.enter(); | |
} | |
public void exit() | |
{ | |
scope.exit(); | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public class Main | |
{ | |
public static void main(String[] args) | |
{ | |
MiddleEarthModule module; | |
module = new MiddleEarthModule(); | |
Injector injector; | |
injector = Guice.createInjector(module); | |
System.out.println("entering scope"); | |
module.enter(); | |
injector.getInstance(Hero.class); | |
injector.getInstance(Hero.class); | |
injector.getInstance(Hero.class); | |
injector.getInstance(Hero.class); | |
System.out.println("exiting scope"); | |
module.exit(); | |
injector.getInstance(Hero.class); | |
injector.getInstance(Hero.class); | |
injector.getInstance(Hero.class); | |
injector.getInstance(Hero.class); | |
} | |
} |
entering scope
created customscopes.Hobbit@12a54f9
exiting scope
created customscopes.Hobbit@30e280
created customscopes.Hobbit@16672d6
created customscopes.Hobbit@fd54d6
created customscopes.Hobbit@1ccb029
created customscopes.Hobbit@12a54f9
exiting scope
created customscopes.Hobbit@30e280
created customscopes.Hobbit@16672d6
created customscopes.Hobbit@fd54d6
created customscopes.Hobbit@1ccb029
When within scope the four calls to #getInstance only create one instance, when not in scope a new instance is created each time. That's a whole lot of code although it is all fairly simple. From a maintainability viewpoint I do see some issues with this, scopes are so arcane to anyone not experienced with Guice that it would be quite easy to misuse a scope which could cause all kinds of issues with retained state and possibly memory leaks. All I can suggest is document everything and get this over to the team.
No comments:
Post a Comment