I've recently had the opportunity to investigate building web services using Java. I think it's fair to say that Spring is the most commonly used dependency injection framework for Java, and it comes with a lot of other useful tools too, but I have no experience using it so I decided to investigate Guice, a dependency injection framework written by Google and used in their AdWords back-end. It doesn't provide any of the extras that Spring gives you, but does strive to be a lightweight and simple to use framework.
If you're using Guice to build servlets, you probably want to make use of the guice-servlet extension. This builds on the dependency injection core of Guice, adding @RequestScoped
, an annotation which allows injected objects to last for the duration of a request (as opposed to having ephemeral or singleton scope), and the GuiceServletContextListener, which is what ties Guice in to the request pipeline. Piping requests through Guice lets you avoid having to define routing in web.xml, and is required for injection to work correctly as the injector can't inject into an object which it did not construct. It's perfectly possible to continue to serve non-Guice servlets at the same time as Guice-provided servlets.
I'm going to build a simple example web application; if you want to follow along then you can find full sources on Github in my guice-webapp-example repository.
So, in your web.xml
, you need to define two objects. Firstly, a <filter>
, provided by Guice, which captures requests and delivers them according to configuration (set up in a custom module, as we'll see later). A mapping defines which requests are captured, usually all of them. Secondly, a <listener>
, which provides a service to the servlet container -- the container will call the functions contextInitialised()
and contextDestroyed()
at the appropriate times, and these provide the hooks required to initialise the injector and provide the configuration to the filter. A sample web.xml follows, the only customisation required in a new project is the class name of the listener.
As the filter is a Guice-provided class, we don't need any customisation there. We will, though, need to provide the ServletConfig
class.
Our ServletConfig
class inherits from GuiceServletContextListener
. As the super-class handles registering the injector into the servlet context, all we need to do is to override the (abstract) getInjector()
method, calling Guice.createInjector()
as you would in a regular application:
You'll see we use a ServletModule
to drive the injection, binding a servlet to a route. We're not making much use of Guice yet -- there's nothing here that couldn't be done very easily with web.xml
-- but we'll get on to bindings when we define our servlet.
I don't want to do much in the servlet, as this is just an example, so we'll just output a greeting:
Here, we have a bit more Guice in evidence: we're going to inject our greeting, and the class is marked as a singleton. The singleton annotation means that Guice will only create a single instance of the servlet, no matter how often the servlet is injected. The greeting is only injected when the class is created -- usually, we'd inject a class with actual behaviour (or even a provider of a class) rather than a string, or we'd tag the injection site to make sure we get the right string, but for the purposes of the example you can add bind(String.class).toInstance("Hello, World!");
to configureServlets()
in ServletConfig
to always inject a cheerful greeting.
We've now got enough code to serve our test string: if you've cloned the git repository, you should have an Eclipse project and a Maven pom to play with: the latter will build you a war
, the former should be able to deploy to a Tomcat server if you set one up -- you may need to install the Java EE tools first.
In later instalments we'll look at injection scopes and testing using dependency injection.