What is Spring Framework?
More
than Dependency Injection
Last updated on May 14, 2020 - 29 comments
You can use this guide to understand what Spring
Quick Links
framework is and how its core features like
Introduction
dependency injection or aspected oriented
Dependency programming work. Also, a comprehensive FAQ.
Injection
Basics (Editor’s note: At ~7800 words, you probably don’t
Spring’s want to try reading this on a mobile device.
Dependency Bookmark it and come back later. And even on a
Injection desktop, eat read this elephant one bite at a time.)
Spring’s
Aspect-
Oriented Introduction
Programming
(AOP)
Spring’s The complexity of the Spring ecosystem
Resources
A lot of companies are using Spring, but then you go
Additional to spring.io and see that the Spring universe actually
Modules
consists of 21 di erent, active projects. Ouch!
FAQ
Furthermore, if you started programming with Spring
Fin
in the last couple of years, there is a very high
Acknowledgment
chance that you went directly into Spring Boot or
Spring Data.
However, this guide is solely about one, the most
important one, of these projects: Spring Framework.
Why?
Because it is essential to learn that Spring
Framework is the basis for all other projects. Spring
Boot, Spring Data, Spring Batch all build on top of
Spring.
This has two implications:
Without proper Spring framework knowledge,
you will sooner or later get lost. You won’t fully
grok e.g. Spring Boot, no matter how
unimportant you think that core knowledge is.
Spending ~15 minutes reading this guide, which
covers the most important 80% of Spring
framework, will pay-o a million times in your
professional career.
What is Spring Framework?
The short answer:
At its core, Spring framework is really just a
dependency injection container, with a couple of
convenience layers (think: database access, proxies,
aspect-oriented programming, RPC, a web mvc
framework) added on top. It helps you build Java
application faster and more conveniently.
Now, that doesn’t really help, does it?
Luckily, there’s also a long answer:
The remainder of this document.
Dependency Injection Basics
In case you already know what dependency injection
is, feel free to skip straight to Spring’s Dependency
Injection. Otherwise, read on.
What is a dependency?
Imagine you are writing a Java class that lets you
access a users table in your database. You would call
these classes DAOs (data access object) or
Repositories. So, you are going to write a UserDAO
class.
public class UserDao {
public User findById(Integer id) {
// execute a sql query to find the user
}
}
Your UserDAO has only one method which lets you
nd users in your database table by their respective
IDs.
To execute the appropriate SQL query, your UserDAO
needs a database connection. And in the Java world,
you (usually) get that database connection from
another class, called a DataSource. So, your code
now would look something like this:
import javax.sql.DataSource;
public class UserDao {
public User findById(Integer id) throws
SQLException {
try (Connection connection =
dataSource.getConnection()) { // (1)
PreparedStatement selectStatement =
connection.prepareStatement("select * from users
where id = ?");
// use the connection etc.
}
}
1. The question is now, where does your UserDao
get its dataSource dependency from? The DAO
obviously depends on a valid DataSource to re
those SQL queries.
Instantiating dependencies with new()
The naive solution would be to simply create a new
DataSource through a constructor, every time you
need one. So, to connect to a MySQL database your
UserDAO could look like this:
import com.mysql.cj.jdbc.MysqlDataSource;
public class UserDao {
public User findById(Integer id) {
MysqlDataSource dataSource = new
MysqlDataSource(); // (1)
dataSource.setURL("jdbc:mysql://localhost:3306/myData
dataSource.setUser("root");
dataSource.setPassword("s3cr3t");
try (Connection connection =
dataSource.getConnection()) { // (2)
PreparedStatement selectStatement =
connection.prepareStatement("select * from users
where id = ?");
// execute the statement..convert the
raw jdbc resultset to a user
return user;
}
}
}
1. We want to connect to a MySQL database;
hence we are using a MysqlDataSource and
hardcoding url/username/password here for
easier reading.
2. We use our newly created DataSource for the
query.
This works, but let’s see what happens when we
extend our UserDao class with another method,
ndByFirstName.
Unfortunately, that method also needs a DataSource
to work with. We can add that new method to our
UserDAO and apply some refactorings, by introducing
a newDataSource method.
import com.mysql.cj.jdbc.MysqlDataSource;
public class UserDao {
public User findById(Integer id) {
try (Connection connection =
newDataSource().getConnection()) { // (1)
PreparedStatement selectStatement =
connection.prepareStatement("select * from users
where id = ?");
// TODO execute the select , handle
exceptions, return the user
}
}
public User findByFirstName(String firstName)
{
try (Connection connection =
newDataSource().getConnection()) { // (2)
PreparedStatement selectStatement =
connection.prepareStatement("select * from users
where first_name = ?");
// TODO execute the select ,
handle exceptions, return the user
}
}
public DataSource newDataSource() {
MysqlDataSource dataSource = new
MysqlDataSource(); // (3)
dataSource.setUser("root");
dataSource.setPassword("s3cr3t");
dataSource.setURL("jdbc:mysql://localhost:3306/myData
return dataSource;
}
}
1. ndById has been rewritten to use the new
newDataSource() method.
2. ndByFirstName has been added and also uses
the new newDataSource() method.
3. This is our newly extracted method, able to
create new DataSources.
This approach works, but has two drawbacks:
1. What happens if we want to create a new
ProductDAO class, which also executes SQL
statements? Your ProductDAO would then also
have a DataSource dependency, which now is
only available in your UserDAO class. You would
then have another similar method or extract a
helper class that contains your DataSource.
2. We are creating a completely new DataSource
for every single SQL query. Consider that a
DataSource opens up a real, socket connection
from your Java program to your database. This
takes time and is rather expensive. It would be
much nicer if we opened just one DataSource
and re-used it, instead of opening and closing
tons of them. One way of doing this could be by
saving the DataSource in a private eld in our
UserDao, so it can be reused between methods
- but that does not help with the duplication
between multiple DAOs.
Dependencies in a global Application
class
To accommodate these issues, you could think about
writing a global Application class, that looks
something like this:
import com.mysql.cj.jdbc.MysqlDataSource;
public enum Application {
INSTANCE;
private DataSource dataSource;
public DataSource dataSource() {
if (dataSource == null) {
MysqlDataSource dataSource = new
MysqlDataSource();
dataSource.setUser("root");
dataSource.setPassword("s3cr3t");
dataSource.setURL("jdbc:mysql://localhost:3306/myData
this.dataSource = dataSource;
}
return dataSource;
}
}
Your UserDAO could now look like this:
import com.yourpackage.Application;
public class UserDao {
public User findById(Integer id) {
try (Connection connection =
Application.INSTANCE.dataSource().getConnection())
{ // (1)
PreparedStatement selectStatement =
connection.prepareStatement("select * from users
where id = ?");
// TODO execute the select etc.
}
}
public User findByFirstName(String firstName)
{
try (Connection connection =
Application.INSTANCE.dataSource().getConnection())
{ // (2)
PreparedStatement selectStatement =
connection.prepareStatement("select * from users
where first_name = ?");
// TODO execute the select etc.
}
}
}
It is an improvement in two ways:
1. Your UserDAO does not have to construct its
own DataSource dependency anymore, instead it
can ask the Application class to give it a fully-
functioning one. Same for all your other DAOs.
2. Your application class is a singleton (meaning
there will only be one INSTANCE created), and
that application singleton holds a reference to a
DataSource singleton.
There are however still several drawbacks to this
solution:
1. The UserDAO actively has to know where to get
its dependencies from, it has to call the
application class →
Application.INSTANCE.dataSource().
2. If your program gets bigger, and you get more
and more dependencies, you will have one
monster Application.java class, which handles
all your dependencies. At which point you’ll
want to try and split things up into more
classes/factories etc.
Inversion of Control (IoC)
Let’s go one step further.
Wouldn’t it be nice if you and the UserDAO didn’t
have to worry about nding dependencies at all?
Instead of actively calling
Application.INSTANCE.dataSource(), your UserDAO
could shout (somehow) that it needs one, but has no
control anymore when/how/where it gets it from?
This is what is called inversion of control.
Let’s have a look at what our UserDAO could look
like, with a brand-new constructor.
import javax.sql.DataSource;
public class UserDao {
private DataSource dataSource;
private UserDao(DataSource dataSource) { //
(1)
this.dataSource = dataSource;
}
public User findById(Integer id) {
try (Connection connection =
dataSource.getConnection()) { // (2)
PreparedStatement selectStatement =
connection.prepareStatement("select * from users
where id = ?");
// TODO execute the select etc.
}
}
public User findByFirstName(String firstName)
{
try (Connection connection =
dataSource.getConnection()) { // (2)
PreparedStatement selectStatement =
connection.prepareStatement("select * from users
where first_name = ?");
// TODO execute the select etc.
}
}
}
1. Whenever a caller creates a new UserDao
through its constructor, the caller also has to
pass in a valid DataSource.
2. The ndByX methods will then simply use that
DataSource.
From the UserDao perspective this reads much nicer.
It doesn’t know about the application class anymore,
or how to construct DataSources itself. It only
announces to the world "if you want to construct
(i.e. use) me, you need to give me a datasource".
But imagine you now want to run your application.
Whereas you could call "new UserService()"
previously, you’ll now have to make sure to call new
UserDao(dataSource).
public class MyApplication {
public static void main(String[] args) {
UserDao userDao = new
UserDao(Application.INSTANCE.dataSource());
User user1 = userDao.findById(1);
User user2 = userDao.findById(2);
// etc ...
}
}
Dependency Injection Containers
Hence, the issue is that you, as a programmer are
still actively constructing UserDAOs through their
constructor and thus setting the DataSource
dependency manually.
Wouldn’t it be nice if someone knew that your
UserDAO has a DataSource constructor dependency
and knew how to construct one? And then magically
construct both objects for you: A working
DataSource and a working UserDao?
That someone is a dependency injection container
and is exactly what Spring framework is all about.
Learning Spring
If you want to practice writing web
applications with self-made
dependency injection, without Learning
using Spring at rst, check out my Spring
Learning Spring course. It helps
tremendously in understanding
what Spring does.
Spring’s Dependency
Injection
As already mentioned at the very beginning, Spring
Framework, at its core, is a dependency injection
container that manages the classes you wrote and
their dependencies for you (see the previous
section). Let’s nd out how it does that.
What is an ApplicationContext? What do
you need it for?
That someone, who has control over all your classes
and can manage them appropriately (read: create
them with the necessary dependencies), is called
ApplicationContext in the Spring universe.
What we want to achieve is the following code (I
described the UserDao and DataSource in the
previous section, go skim it if you came right here
and skipped it):
import
org.springframework.context.ApplicationContext;
import
org.springframework.context.annotation.AnnotationConf
import javax.sql.DataSource;
public class MyApplication {
public static void main(String[] args) {
ApplicationContext ctx = new
AnnotationConfigApplicationContext(someConfigClass);
// (1)
UserDao userDao =
ctx.getBean(UserDao.class); // (2)
User user1 = userDao.findById(1);
User user2 = userDao.findById(2);
DataSource dataSource =
ctx.getBean(DataSource.class); // (3)
// etc ...
}
}
1. Here we are constructing our Spring
ApplicationContext. We’ll go into much more
detail on how this works in the next paragraphs.
2. The ApplicationContext can give us a fully
con gured UserDao, i.e. one with its DataSource
dependency set.
3. The ApplicationContext can also give us the
DataSource directly, which is the same
DataSource that it sets inside the UserDao.
This is pretty cool, isn’t it? You as the caller don’t
have to worry about constructing classes anymore,
you can simply ask the ApplicationContext to give
you working ones!
But how does that work?
What is a
ApplicationContextCon guration? How
to construct ApplicationContexts from
Con gurations.
In the code above, we put a variable called
"someCon gClass" in the
AnnotationCon gApplicationContext constructor.
Here’s a quick reminder:
import
org.springframework.context.annotation.AnnotationConf
public class MyApplication {
public static void main(String[] args) {
ApplicationContext ctx = new
AnnotationConfigApplicationContext(someConfigClass);
// (1)
// ...
}
}
What you really want to pass into the
ApplicationContext constructor, is a reference to a
con guration class, which should look like this:
import
org.springframework.context.annotation.Bean;
import
org.springframework.context.annotation.Configuration;
@Configuration
public class MyApplicationContextConfiguration {
// (1)
@Bean
public DataSource dataSource() { // (2)
MysqlDataSource dataSource = new
MysqlDataSource();
dataSource.setUser("root");
dataSource.setPassword("s3cr3t");
dataSource.setURL("jdbc:mysql://localhost:3306/myData
return dataSource;
}
@Bean
public UserDao userDao() { // (3)
return new UserDao(dataSource());
}
1. You have a dedicated ApplicationContext
con guration class, annotated with the
@Con guration annotation, that looks a bit like
the Application.java class from Dependencies in
a global Application class.
2. You have a method that returns a DataSource
and is annotated with the Spring-speci c
@Bean annotation.
3. You have another method, which returns a
UserDao and constructs said UserDao by calling
the dataSource bean method.
This con guration class is already enough to run
your very rst Spring application.
import
org.springframework.context.ApplicationContext;
import
org.springframework.context.annotation.AnnotationConf
public class MyApplication {
public static void main(String[] args) {
ApplicationContext ctx = new
AnnotationConfigApplicationContext(MyApplicationConte
UserDao userDao =
ctx.getBean(UserDao.class);
// User user1 = userDao.findById(1);
// User user2 = userDao.findById(1);
DataSource dataSource =
ctx.getBean(DataSource.class);
}
}
Now, let’s nd out what exactly Spring and the
AnnotationCon gApplicationContext do with that
Con guration class you wrote.
Why did we construct an
AnnotationCon gApplicationContext?
Are there other ApplicationContext
classes?
There are many ways to construct a Spring
ApplicationContext, for example through XML les,
annotated Java con guration classes or even
programmatically. To the outside world, this is
represented through the single ApplicationContext
interface.
Look at the MyApplicationContextCon guration class
from above. It is a Java class that contains Spring-
speci c annotations. That is why you would need to
create an Annotation Con gApplicationContext.
If, instead, you wanted to create your
ApplicationContext from XML les, you would create
a ClassPathXmlApplicationContext.
There are also many others, but in a modern Spring
application, you will usually start out with an
annotation-based application context.
What does the @Bean annotation do?
What is a Spring Bean?
You’ll have to think of the methods inside your
ApplicationContext con guration class as factory
methods. For now, there is one method that knows
how to construct UserDao instances and one
method that constructs DataSource instances.
These instances that those factory methods create
are called beans. It is a fancy word for saying: I (the
Spring container) created them and they are under
my control.
But this leads to the question: How many instances
of a speci c bean should Spring create?
What are Spring bean scopes?
How many instances of our DAOs should Spring
create? To answer that question, you need to learn
about bean scopes.
Should Spring create a singleton: All your DAOs
share the same DataSource?
Should Spring create a prototype: All your DAOs
get their own DataSource?
Or should your beans have even more complex
scopes, like saying: A new DataSource per
HttpRequest? Or per HttpSession? Or per
WebSocket?
You can read up on a full list of available bean
scopes here, but for now it is su ce to know that
you can in uence the scope with yet another
annotation.
import
org.springframework.context.annotation.Bean;
import
org.springframework.context.annotation.Scope;
import
org.springframework.context.annotation.Configuration;
@Configuration
public class MyApplicationContextConfiguration {
@Bean
@Scope("singleton")
// @Scope("prototype") etc.
public DataSource dataSource() {
MysqlDataSource dataSource = new
MysqlDataSource();
dataSource.setUser("root");
dataSource.setPassword("s3cr3t");
dataSource.setURL("jdbc:mysql://localhost:3306/myData
return dataSource;
}
}
The scope annotation controls how many instances
Spring will create. And as mentioned above, that’s
rather simple:
Scope("singleton") → Your bean will be a
singleton, there will only be one instance.
Scope("prototype") → Every time someone
needs a reference to your bean, Spring will
create a new one. (There’s a couple of caveats
here, like injecting prototypes in singletons,
though).
Scope("session") → There will be one bean
created for every user HTTP session.
etc.
The gist: Most Spring applications almost entirely
consist of singleton beans, with the occasional other
bean scope (prototype, request, session, websocket
etc.) sprinkled in.
Now that you know about ApplicationContexts,
Beans & Scopes, let’s have another look at
dependencies, or the many ways our UserDAO could
obtain a DataSource.
What is Spring’s Java Con g?
So far, you explicitly con gured your beans in your
ApplicationContext con guration, with the help of
@Bean annotated Java methods.
This is what you would call Spring’s Java Con g, as
opposed to specifying everything in XML, which was
historically the way to go with Spring. Just a quick
recap of what this looks like:
import
org.springframework.context.annotation.Bean;
import
org.springframework.context.annotation.Configuration;
@Configuration
public class MyApplicationContextConfiguration {
@Bean
public DataSource dataSource() {
MysqlDataSource dataSource = new
MysqlDataSource();
dataSource.setUser("root");
dataSource.setPassword("s3cr3t");
dataSource.setURL("jdbc:mysql://localhost:3306/myData
return dataSource;
}
@Bean
public UserDao userDao() { // (1)
return new UserDao(dataSource());
}
1. One question: Why do you have to explicitly call
new UserDao() with a manual call to
dataSource()? Cannot Spring gure all of this
out itself?
This is where another annotation called
@ComponentScan comes in.
What does @ComponentScan do?
The rst change you’ll need to apply to your context
con guration is to annotate it with the additional
@ComponentScan annotation.
import
org.springframework.context.annotation.Bean;
import
org.springframework.context.annotation.ComponentScan;
import
org.springframework.context.annotation.Configuration;
@Configuration
@ComponentScan // (1)
public class MyApplicationContextConfiguration {
@Bean
public DataSource dataSource() {
MysqlDataSource dataSource = new
MysqlDataSource();
dataSource.setUser("root");
dataSource.setPassword("s3cr3t");
dataSource.setURL("jdbc:mysql://localhost:3306/myData
return dataSource;
}
// (2)
// no more UserDao @Bean method!
}
1. We added the @ComponentScan annotation.
2. Note, that the UserDAO de nition is now missing
from the context con guration!
What this @ComponentScan annotation does, is tell
Spring: Have a look at all Java classes in the same
package as the context con guration if they look like
a Spring Bean!
This means if your
MyApplicationContextCon guration lives in package
com.marcobehler, Spring will scan every package,
including subpackages, that starts with
com.marcobehler for potential Spring beans.
How does Spring know if something is a Spring
bean? Easy: Your classes need to be annotated with
a marker annotation, called @Component.
What do @Component & @Autowired
do?
Let’s add the @Component annotation to your
UserDAO.
import javax.sql.DataSource;
import org.springframework.stereotype.Component;
@Component
public class UserDao {
private DataSource dataSource;
private UserDao(DataSource dataSource) { //
(1)
this.dataSource = dataSource;
}
}
1. This tells Spring, similarly to that @Bean
method you wrote before: Hey, if you nd me
annotated with @Component through your
@ComponentScan, then I want to be a Spring
bean, managed by you, the dependency injection
container!
(When you look at the source code of annotations
like @Controller, @Service or @Repository later on,
you’ll nd that they all consist of multiple, further
annotations, always including @Component!).
There’s only one little piece of information missing.
How does Spring know that it should take the
DataSource that you speci ed as a @Bean method
and then create new UserDAOs with that speci c
DataSource?
Easy, with another marker annotation: @Autowired.
Hence, your nal code will look like this.
import javax.sql.DataSource;
import org.springframework.stereotype.Component;
import
org.springframework.beans.factory.annotation.Autowire
@Component
public class UserDao {
private DataSource dataSource;
private UserDao(@Autowired DataSource
dataSource) {
this.dataSource = dataSource;
}
}
Now, Spring has all the information it needs to
create UserDAO beans:
UserDAO is annotated with @Component →
Spring will create it
UserDAO has an @Autowired constructor
argument → Spring will automatically inject the
DataSource that is con gured via your @Bean
method
Should there be no DataSources con gured in
any of your Spring con gurations, you will
receive a NoSuchBeanDe nition exception at
runtime.
Constructor Injection & Autowired
Revisited
I have been lying to you a tiny bit in the previous
section. In earlier Spring versions (pre 4.2), you
needed to specify @Autowired in order for
constructor injection to work.
With newer Spring versions, Spring is actually smart
enough to inject these dependencies without an
explicit @Autowired annotation in the constructor.
So this would also work.
@Component
public class UserDao {
private DataSource dataSource;
private UserDao(DataSource dataSource) {
this.dataSource = dataSource;
}
}
Why did I mention @Autowired then? Because it
does not hurt, i.e. makes things more explicit and
because you can use @Autowired in many other
di erent places, apart from constructors.
Let’s have a look at di erent ways of dependency
injection - constructor injection just being one many.
What is Field Injection? What is Se er
Injection?
Simply put, Spring does not have to go through a
constructor to inject dependencies.
It can also directly inject elds.
import javax.sql.DataSource;
import org.springframework.stereotype.Component;
import
org.springframework.beans.factory.annotation.Autowire
@Component
public class UserDao {
@Autowired
private DataSource dataSource;
Alternatively, Spring can also inject setters.
import javax.sql.DataSource;
import org.springframework.stereotype.Component;
import
org.springframework.beans.factory.annotation.Autowire
@Component
public class UserDao {
private DataSource dataSource;
@Autowired
public void setDataSource(DataSource
dataSource) {
this.dataSource = dataSource;
}
These two injection styles ( elds, setters) have the
same outcome as constructor injection: You’ll get a
working Spring Bean. (In fact, there’s also another
one, called method injection which we won’t cover
here.)
But obviously, they di er from one another, which
means there has been a great many debates about
which injection style is best and which one you
should use in your project.
Constructor Injection vs. Field Injection
There have been a great many debates online,
whether constructor injection or eld injection is
better, with a number of strong voices even claiming
that eld injection is harmful.
To not add further noise to these arguments, the gist
of this article is:
1. I have worked with both styles, constructor
injection and eld injection in various projects
over the recent years. Based solely on personal
experience, I do not truly favor one style over
the other.
2. Consistency is king: Do not use constructor
injection for 80% of your beans, eld injection
for 10% and method injection for the remaining
10%.
3. Spring’s approach from the o cial
documentation seems sensible: Use constructor
injection for mandatory dependencies and
setter/ eld injection for optional dependencies.
Be warned: Be really consistent with that.
Learning Spring
If you want to practice the various
Learning
ways of dependency injection in a
Spring
real-world application, check out
my Learning Spring course.
Summary: Spring’s IoC container
By now, you should know pretty much everything you
need to know about Spring’s dependency container.
There is of course more to it, but if you have a good
grasp of ApplicationContexts, Beans, dependencies
and di erent methods of dependency injection, then
you are already on a good path.
Let’s see what else Spring has to o er, apart from
pure dependency injection.
Spring’s Aspect-Oriented
Programming (AOP)
Dependency injection might lead to better structured
programs, but injecting a dependency here and there
is not what Spring’s ecosystem is all about. Let’s
have a look at a simple
ApplicationContextCon guration again:
import
org.springframework.context.annotation.Bean;
import
org.springframework.context.annotation.Configuration;
@Configuration
public class MyApplicationContextConfiguration {
@Bean
public UserService userService() { // (1)
return new UserService();
}
}
1. Let’s assume that UserService is a class that
lets you nd users from a database table - or
save users to that database table.
Here’s where Spring’s hidden killer feature comes in:
Spring reads in that context con guration, containing
the @Bean method you wrote and therefore Spring
knows how to create and inject UserService beans.
But Spring can cheat and create something else than
your UserService class. How? Why?
Spring can create proxies
Because under the hood, any Spring @Bean method
can return you something that looks and feels like
(in your case) a UserService, but actually isn’t.
It can return you a proxy.
The proxy will at some point delegate to the
UserService you wrote, but rst, will execute its own
functionality.
More speci cally, Spring will, by default, create
dynamic Cglib proxies, that do not need an interface
for proxying to work (like JDK’s internal proxy
mechanism): Instead, Cglib can proxy classes
through subclassing them on the y. ( If you are
unsure about the individual proxy patterns, read
more about the proxies on Wikipedia. )
Why would Spring want to create
proxies?
Because it allows Spring to give your beans
additional features, without modifying your code. In
a gist, that is what aspect-oriented (or: AOP)
programming is all about.
Let’s have a look at the most popular AOP example,
Spring’s @Transactional annotation.
Spring’s @Transactional
Your UserService implementation above could look a
bit like this:
import org.springframework.stereotype.Component;
import
org.springframework.transaction.annotation.Transactio
@Component
public class UserService {
@Transactional // (2)
public User activateUser(Integer id) { // (1)
// execute some sql
// send an event
// send an email
}
}
1. We wrote an activateUser method, which, when
called, needs to execute some SQL to update
the User’s state in the database, maybe send an
email or a messaging event.
2. @Transactional on that method signals Spring
that you need an open database
connection/transaction for that method to work
and that said transaction should also be
committed at the end. And that Spring needs to
do this.
The problem: While Spring can create your
UserService bean through the applicationContext
con guration, it cannot rewrite your UserService. It
cannot simply inject code in there that opens a
database connection and commits a database
transaction.
But what it can do, is to create a proxy around your
UserService that is transactional. So, only the proxy
needs to know about how to open up and close a
database connection and can then simply delegate
to your UserService in between.
Let’s have a look at that innocent
ContextCon guration again.
@Configuration
@EnableTransactionManagement // (1)
public class MyApplicationContextConfiguration {
@Bean
public UserService userService() { // (2)
return new UserService();
}
}
1. We added an annotation signaling Spring: Yes,
we want @Transactional support, which
automatically enables Cglib proxies under the
hood.
2. With the above annotation set, Spring does not
just create and return your UserService here. It
creates a Cglib proxy of your bean, that looks,
smells and delegates to your UserService, but
actually wraps around your UserService and
gives its transaction management features.
This might seem a bit unintuitive rst, but most
Spring developers encounter proxies very soon in
debugging sessions. Because of the proxies, Spring
stacktraces can get rather long and unfamiliar: When
you step inside a method, you could very well step
inside the proxy rst - which scares people o . It is,
however, completely normal and expected behavior.
Does Spring have to use Cglib proxies?
What about real bytecode weaving?
Proxies are the default choice when programming
AOP with Spring. You are however not restricted to
using proxies, you could also go the full AspectJ
route, that modi es your actual bytecode, if wanted.
Covering AspectJ is however outside the scope of
this guide.
Also see: What is the di erence between Spring AOP
& AspectJ?.
Spring’s AOP Support: A Summary
There is of course much more to be said about
aspect-oriented programming, but this guide gives
you an idea of how the most popular Spring AOP
use-cases like @Transactional or Spring Security’s
@Secured work. You could even go as far as write
your own AOP annotations, if wanted.
As a consolation for the abrupt end, if you want to
get more information on how Spring’s
@Transactional management works in detail, have a
look at my @Transactional guide.
Spring’s Resources
We’ve been talking about dependency injection &
proxies for a while. Let’s now have a rst look at
what I would call important convenience utilities in
Spring framework. One of these utilities is Spring’s
resources support.
Think about how you would try to access a le in
Java via HTTP or FTP. You could use Java’s URL class
and write some plumbing code.
Similarly, how would you read in les from your
application’s classpath? Or from a servlet context,
that means from a web applications root directory
(admittedly, this gets rarer and rarer in modern,
packaged.jar application).
Again, you’d need to write a fair amount of
boilerplate code to get that working and
unfortunately the code would di er for each use
case (URLs, classpaths, servlet contexts).
But there’s a solution: Spring’s resource abstraction.
It is easily explained in code.
import org.springframework.core.io.Resource;
public class MyApplication {
public static void main(String[] args) {
ApplicationContext ctx = new
AnnotationConfigApplicationContext(someConfigClass);
// (1)
Resource aClasspathTemplate =
ctx.getResource("classpath:somePackage/application.pr
// (2)
Resource aFileTemplate =
ctx.getResource("file:///someDirectory/application.pr
// (3)
Resource anHttpTemplate =
ctx.getResource("https://marcobehler.com/application.
// (4)
Resource depends =
ctx.getResource("myhost.com/resource/path/myTemplate.
// (5)
Resource s3Resources =
ctx.getResource("s3://myBucket/myFile.txt"); //
(6)
}
}
1. As always, you need an ApplicationContext to
start o .
2. When you call getResource() on an
applicationContext with a string that starts with
classpath:, Spring will look for a resource on
your..well..application classpath.
3. When you call getResource() with a string that
starts with le:, Spring will look for a le on
your harddrive.
4. When you call getResource() with a string that
starts with https: (or http), Spring will look for a
le on the web.
5. If you don’t specify a pre x, it actually depends
on what kind of applicationContext you
con gured. More on that here.
6. This does not work out of the box with Spring
Framework, but with additional libraries like
Spring Cloud, you can even directly access s3://
paths.
In short, Spring gives you the ability to access
resources via a nice little syntax. The resource
interface has a couple of interesting methods:
public interface Resource extends
InputStreamSource {
boolean exists();
String getFilename();
File getFile() throws IOException;
InputStream getInputStream() throws
IOException;
// ... other methods commented out
}
As you can see, it allows you to execute the most
common operations on a resource:
Does it exist?
What is the lename?
Get a reference to the actual File object.
Get a direct reference to the raw data
(InputStream).
This lets you do everything you want with a resource,
independent of it living on the web or on your
classpath or your hard drive.
The resources abstraction looks like such a tiny
feature, but it really shines when combined with the
next convenience feature o ered by Spring:
Properties.
What is Spring’s Environment?
A big part of any application is reading in properties,
like database username & passwords, email server
con guration, Stripe payment detail con guration,
etc.
At its simplest form, these properties live in
.properties les and there could be many of them:
Some of them on your classpath, so you have
access to some development related passwords.
Others in the lesystem or a network drive, so a
production server can have its own, secure
properties.
Some could even come in the form of operating
system environment variables.
Spring tries to make it easy for you to register and
automatically nd properties across all these
di erent sources, through its environment
abstraction.
import org.springframework.core.env.Environment;
public class MyApplication {
public static void main(String[] args) {
ApplicationContext ctx = new
AnnotationConfigApplicationContext(someConfigClass);
Environment env = ctx.getEnvironment();
// (1)
String databaseUrl =
env.getProperty("database.url"); // (2)
boolean containsPassword =
env.containsProperty("database.password");
// etc
}
}
1. Through an applicationContext, you can always
access the current program’s environment.
2. The environment on the other hand, let’s you,
among other things, access properties.
Now, what is an environment exactly?
What are Spring’s @PropertySources?
In a nutshell, an environment consists of one to
many property sources. For example:
/mydir/application.properties
classpath:/application-default.properties
(Note: An environment also consists of pro les, i.e.
"dev" or "production" pro les, but we won’t go into
detail on pro les in this revision of this guide).
By default, a Spring MVC web application
environment consists of ServletCon g/Context
parameter, JNDI and JVM system property sources.
They are also hierarchical, that means they have an
order of importance and override each other.
However, it is rather easy to de ne new
@PropertySources yourself:
import
org.springframework.context.annotation.PropertySource
import
org.springframework.context.annotation.PropertySource
@Configuration
@PropertySources(
{@PropertySource("classpath:/com/${my.placeholder:def
@PropertySource("file://myFolder/app-
production.properties")})
public class MyApplicationContextConfiguration {
// your beans
}
Now it makes much more sense, why we talked
about Spring’s Resources before. Because both
features go hand in hand.
The @PropertySource annotation works with any
valid Spring con guration class and lets you de ne
new, additional sources, with the help of Spring’s
resources abstraction: Remember, it’s all about the
pre xes: http://, le://, classpath: , etc.
De ning properties through @PropertySources is
nice, but isn’t there a better way than having to go
through the environment to access them? Yes, there
is.
Spring’s @Value annotation & Property
injection
You can inject properties into your beans, similarly
like you would inject a dependency with the
@Autowired annotation. But for properties, you need
to use the @Value annotation.
import org.springframework.stereotype.Component;
import
org.springframework.beans.factory.annotation.Value;
@Component
public class PaymentService {
@Value("${paypal.password}") // (1)
private String paypalPassword;
public PaymentService(@Value("${paypal.url}")
String paypalUrl) { // (2)
this.paypalUrl = paypalUrl;
}
}
1. The @Value annotation works directly on elds…
2. Or on constructor arguments.
There really isn’t much more to it. Whenever you use
the @Value annotation, Spring will go through your
(hierarchical) environment and look for the
appropriate property - or throw an error message if
such a property does not exist.
Learning Spring
If you want to practice how
Spring's property and environment
Learning
support works and especially how
Spring
Spring Boot nds all your
.properties les, check out my
Learning Spring course.
Additional Modules
There’s even more modules that Spring Framework
consists of. Let’s have a look at them now.
Spring Web MVC
Spring Web MVC, also known as Spring MVC, is
Spring’s Web framework. It lets you build anything
web related, from HTML-based websites to RESTful
web services that return JSON or XML. It also
powers frameworks like Spring Boot.
Note - April 2020: You can nd an extensive
description of Spring MVC in my new guide: Spring
MVC: In-Depth Guide.
Data Access, Testing, Integration &
Languages
Spring framework consists of even more
convenience utilities than you have seen so far. Let’s
call them modules and do not confuse these
modules with the 20 other Spring projects on
spring.io. To the contrary, they are all part of the
Spring framework project.
So, what kind of convenience are we talking about?
You’ll have to understand that basically everything
Spring o ers in these modules, is also available in
pure Java. Either o ered by the JDK or a third-party
library. Spring framework always builds on top of
these existing features.
Here’s an example: Sending email attachments with
Java’s Mail API is certainly doable, but a bit
cumbersome to use. See here for a code example.
Spring provides a nice little wrapper API on top of
Java’s Mail API, with the added bene t that
everything it o ers blends in nicely with Spring’s
dependency injection container. It is part of Spring’s
integration module.
import
org.springframework.core.io.FileSystemResource;
import
org.springframework.mail.javamail.JavaMailSender;
import
org.springframework.mail.javamail.MimeMessageHelper;
public class SpringMailSender {
@Autowired
private JavaMailSender mailSender; // (1)
public void sendInvoice(User user, File pdf)
throws Exception {
MimeMessage mimeMessage =
mailSender.createMimeMessage();
MimeMessageHelper helper = new
MimeMessageHelper(mimeMessage, true); // (2)
helper.setTo("john@rambo.com");
helper.setText("Check out your new
invoice!");
FileSystemResource file = new
FileSystemResource(pdf);
helper.addAttachment("invoice.pdf", file);
mailSender.send(mimeMessage);
}
}
1. Everything related to con guring an email server
(url, username, password) is abstracted away
into the Spring speci c MailSender class, that
you can inject in any bean that wants to send
emails.
2. Spring o ers convenience builders, like the
MimeMessageHelper, to create multipart emails
from, say, les as fast as possible.
So, to sum it up, Spring framework’s goal is to
'springify' available Java functionality, preparing it for
dependency injection and therefore making the APIs
easier to use in a Spring context.
Module Overview
I’d like to give you a quick overview of the most
common utilities, features and modules you might
encounter in a Spring framework project. Note,
however, that detailed coverage of all these tools is
impossible in the scope of this guide. Instead, have a
look at the o cial documentation for a full list.
Spring’s Data Access: Not to be confused with
Spring Data (JPA/JDBC) libraries. It is the basis
for Springs @Transactional support, as well as
pure JDBC and ORM (like Hibernate) integration.
Spring’s Integration Modules: Makes it easier for
you to send emails, integrate with JMS or AMQP,
schedule tasks, etc.
Spring Expression Language (SpEL): Even though
this is not really correct, think about it as a DSL
or Regex for Spring Bean
creation/con guration/injection. It will be
covered in more detail in future versions of this
guide.
Spring’s Web Servlet Modules: Allows you
writing web applications. Includes Spring MVC,
but also support for WebSockets, SockJS and
STOMP messaging.
Spring’s Web Reactive Modules: Allows you
writing reactive web applications.
Spring’s Testing Framework: Allows you to
(integration) test Spring contexts and therefore
Spring applications, including helper utilities for
testing REST services.
Learning Spring
Interested in taking your rst
steps with @Transactional, Learning
JdbcTemplate and Spring Data Spring
JDBC? Check out my Learning
Spring course.
FAQ
What is the latest Spring Framework
version?
The current latest version is Spring 5.2.6.RELEASE
and you can always nd new version announcements
on the Spring Blog.
What Spring version should I use?
Choosing a Spring version is relatively simple:
If you are building new Spring Boot projects, the
version of Spring you are using is already pre-
de ned for you. If you are using Spring Boot
2.2.x, for example, you will be using Spring 5.2.x
(though, in theory you could override this).
If you are using plain Spring in a green eld
project, you can obviously choose whatever
version you want, which will usually be the
latest one.
If you are using Spring in a legacy project, you
can always think about upgrading to a newer
Spring version if that makes sense from a
business perspective (or if you want to respect
the EOL announcements)- Spring versions have
a high amount of compatibility (see next
paragraph).
Realistically, you’ll nd Spring versions 4.x-5.x used
by companies, though the rare, legacy 3.x (initial
release: 2009) Spring project also pops up.
How o en do new versions of Spring get
released? How long are they supported?
Here’s a nice, little graph showing you Spring’s
version history:
You can see that the initial Spring release was ~17
years ago, with major framework versions being
released every 3-4 year. This does not account for
maintenance branches, however.
Spring 4.3 for example, has been released in
June 2016, and will be supported until the end
of 2020.
Even support for Spring 5.0 and 5.1 will be cut
o at the end of 2020, in favor of Spring 5.2,
which has been released in September 2019.
Note: The current EOL() (no more updates & support)
for all Spring versions, except 5.2 is currently set to
Dec 31st, 2020.
What libraries do you need to get
started with Spring?
There is, in fact only one dependency you need
whenever you want to setup a Spring project. It is
called spring-context. It is the absolute minimum to
get Spring’s dependency injection container working.
If you are working on a Maven or Gradle project, you
can simply add the following dependency to it (also
see: What is the latest Spring Framework version?) -
instead of downloading said .jar le and adding it to
your project manually.
<!-- Maven -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.2.6.RELEASE</version>
</dependency>
// Gradle
compile group: 'org.springframework', name:
'spring-context', version: '5.2.6.RELEASE'
For additional Spring features (like Spring’s JDBC or
JMS support), you’ll need other, additional libraries.
You can nd a list of all available modules in the
o cial Spring documentation, though in terms of
Maven dependencies, the artifactIds really do follow
the module’s name. Here’s an example:
<!-- Maven -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.2.6.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.2.6.RELEASE</version>
</dependency>
What are the differences between
Spring versions?
Similar to the JVM, Spring versions are insanely
backwards compatible, which means you can
(essentially) still run your Spring 1.0 xml les with
the latest Spring 5.0 (though I admittedly haven’t
tried that out, yet). In addition, upgrading from, say 3
to 5 is also possible with a bit of e ort (see this
migration guide).
So, in general, newer Spring versions build on top of
the older Spring versions and have minimal breaking
changes (compared with, say Python 2 vs 3). So, all
core concepts you learned for Spring version 3 or 4
are still true for Spring version 5.
You can get a great overview of what changed over
the last 7 years in the individual Spring versions,
here:
What’s new in Spring 4
What’s new in Spring 5
To give you an executive summary:
The core (dependency injection, transaction
management etc.) always stays the same or gets
extended. Spring, however, goes with time and o ers
support for newer Java language versions, test
framework enhancements, Websockets, reactive
programming, etc.
What do the 20 other Spring.io projects
do now?
In the scope of this guide, I cannot go into detail of
all the di erent projects, but let’s have a look at the
ones you are most likely going to encounter.
Spring Boot: Probably the most popular Spring
project. Spring Boot is an opinionated version of
Spring Framework. Look at What is the
di erence between Spring & Spring Boot? to
nd out what that rather meaningless phrase
actually means.
Spring Batch: A library that helps you write good
old batch jobs.
Spring Cloud: A set of libraries that help your
Spring project integrate easier with "the cloud"
(think: AWS) or write microservices.
Spring Security: A library that helps you secure,
e.g. your web-application with OAuth2 or Basic
Auth.
and many more…
Takeaway: All these libraries extend Spring
Framework and build on top of its dependency
injection core principles.
What is the difference between Spring &
Spring Boot?
If you have read this guide, you should understand
by now that Spring Boot builds on top of Spring.
While a comprehensive Spring Boot guide is coming
up soon, here’s an example what "opinionated
defaults" in Spring Boot mean.
Spring o ers you the ability to read in .properties
les from a variety of places, e.g. with the help of
@PropertySource annotations. It also o ers you the
ability to write JSON REST controllers with the help
of its Web MVC framework.
The issue is, you have to write and con gure all
these individual pieces yourself. Spring Boot, on the
other hand, takes these single pieces and bundles
them up together. Example:
Always and automatically look for
application.properties les in various places and
read them in.
Always booting up an embedded Tomcat so you
can immediately see the results of writing your
@RestControllers.
Automatically con guring everything for you to
send/receive JSON, without needing to worry a
ton about speci c Maven/Gradle dependencies.
All, by running a main method in a Java class, which
is annotated with the @SpringBootApplication
annotation. Even better, Spring Boot o ers
Maven/Gradle plugins that let you package up your
application into a .jar le, which you can run like
this:
java -jar mySpringBootApp.jar
So, Spring Boot is all about taking the existing Spring
framework parts, pre-con guring and packaging
them up - with as little development work needed
as possible.
What is the difference between Spring
AOP & AspectJ?
As mentioned in the Does Spring have to use Cglib
proxies? What about real bytecode weaving? section,
Spring is using proxy-based AOP by default. It wraps
your beans in proxies to achieve things like
transaction management. This has a couple of
limitations and caveats but is a pretty simple and
straightforward way of implementing the most
common AOP problems that Spring developers face.
AspectJ on the other hand allows you to change
actual bytecode through load-time-weaving or
compile-time-weaving. This gives you a lot more
possibilities, in exchange for a lot more complexity.
You can however con gure Spring to use AspectJ’s
AOP, instead of its default, proxy-based AOP.
Here are a couple of links if you want to get more
information on this topic:
AspectJ Homepage
Spring AOP vs AspectJ
Spring AOP o cial documentation
What is the difference between Spring &
Spring Batch?
Spring Batch is a framework that makes it simpler to
write batch jobs, i.e. "read in these 95 CSV les
every night at 3am and call an external validation
service for every single entry".
Again, it builds on top of Spring Framework, but is
very much its own project.
Do note, though, that it is essentially impossible to
build a rock-solid batch job without having a good
understanding of Spring Framework’s general
transaction management and how this relates to
Spring Batch.
What is the difference between Spring &
Spring Web MVC?
If you have read this guide, you should understand
by now that Spring Web MVC is part of Spring
framework.
On a very high-level it allows you to turn your Spring
application into a web application with the help of a
DispatcherServlet that routes to @Controller classes.
These can be RestControllers (where you send XML
or JSON to the client) or good old HTML Controllers
where you generate HTML with frameworks like
Thymeleaf, Velocity or Freemarker.
What is the difference between Spring &
Struts?
The question should really be: What is the di erence
between Spring Web MVC & Struts?
The short, historical answer is: Spring Web MVC
started out as a competitor of Struts, which was,
allegedly, perceived as poorly designed by Spring
developers (see wikipedia).
The modern answer is, that while Struts 2 is
certainly still being used in the odd legacy project,
Spring Web MVC is the basis for everything web-
related in the Spring universe. From Spring Web ow
to Spring Boot’s RestControllers.
Which is be er? Spring XML or
annotation or Java con guration?
Spring started out with XML con guration only. Then,
slowly, more and more annotation / Java
con guration features came out.
Today, you’ll nd XML con guration mainly used in
older, legacy projects - with newer projects all going
for the Java / annotation-based con guration.
Do note two things though:
1. There is essentially nothing stopping you from
combining XML / Annotations / Java Con g in
the same project, which usually leads to a mess.
2. You want to strive for homogeneity in your
Spring con guration, i.e. not randomly generate
some con gurations with XML, some with Java
con g and some with component-scanning.
Which is be er? Constructor or Field
Injection?
As mentioned in the dependency injection section,
this is a question which prompts many di erent
opinions. Most importantly, your choice should be
consistent across your project: Don’t use constructor
injection for 83% of your beans and eld injection for
the other 17%.
A sensible approach would be to use the
recommended way in the Spring documentation:
Using constructor injection for mandatory
dependencies, setter/ eld injection for optional
dependencies and then null-checking those optional
dependencies throughout your class.
Most importantly, keep in mind: The overall success
of your software project will not depend on the
choice of your favorite dependency injection method
(pun intended).
Are there alternatives to Spring’s
dependency injection container?
Yes, two popular ones in the Java ecosystem are:
Google’s Guice.
Google’s Dagger, formerly Square’s.
Note that Dagger only o ers dependency injection,
with no additional convenience features. Guice o ers
dependency injection and other features like
transaction management (with the help of Guice
Persist).
Fin
If you have read this far, you should now have a
pretty thorough understanding of what Spring
framework is about.
You’ll nd out how this connects to other Spring
ecosystem libraries (like Spring Boot, or Spring Data)
in the follow-up guides, but for now I want you to
keep this metaphor in mind when trying to answer
the question What is Spring Framework?
Imagine you want to renovate a house (~= build a
software project).
Spring Framework is your DIY store (~=dependency
injection container), which o ers a ton di erent
tools, from Bunsen burners (~= resources /
properties) to sledgehammers (~= Web MVC) for your
renovation. These tools simply help you renovate
your house (=~ build your Java application) faster
and more conveniently.
(note: Don’t ask me how I came up with these
comparisons ;) )
That’s it for today. If you have any questions or
suggestions, drop me a mail at
marco@marcobehler.com or leave a comment below.
For hands-on practice, check out the Learning Spring
exercise course.
Thanks for reading. Auf Wiedersehen.
Acknowledgments
Big thanks go out to:
Patricio "Pato" Moschcovich, for doing an
amazing job proof-reading this article and
pointing out a ton of tiny mistakes.
Maciej Walkowiak, for rightfully pointing out that
@RestController has always been part of Spring
MVC, not Spring Boot.
There's more where that came
from
I'll send you an update when I publish new
guides. Absolutely no spam, ever.
Unsubscribe anytime.
e.g. martin@fowler.com
I want more!
Share:
Comments
Login
Add a comment
M ↓ MARKDOWN COMMENT ANONYMOUSLY ADD COMMENT
Upvotes Newest Oldest
Anonymous
? 1 point · 4 months ago
What a great way to explain Spring Framework, this is indeed a hidden
jewel. Many thanks!
Anonymous
? 0 points · 56 days ago
Excellent article! Thank you very much for putting the time and effort into
it
Anbu Sampath
A 0 points · 4 months ago
Excellent article, clear explanation to understand various spring modules.
Anonymous
? 0 points · 53 days ago
Hey Marco,
Great article! My only comment regards the Constructor/Field/Setter
injection. You link to an external article that is titled 'Field Injection
Considered Harmful' where the article goes into detail on how field
injection leads to DI container coupling, mutability, etc. However, in your
article, you label the external resource as 'setter injection is harmful.' In
the rest of your article, you put more emphasis on the combination of
field and constructor injection.
Is this a mishap on your part, or am I missing an argument you're trying
to make in favor of field injection?
Marco Behler
0 points · 53 days ago
Mishap on my part! Thanks for bringing it up, will be fixed soon.
Anonymous
? 0 points · 4 months ago
Thanks a lot for this master-piece article!
Anonymous
? 0 points · 4 months ago
Awesome Article! Thanks!
Anonymous
? 0 points · 4 months ago
This is an amazing article, thanks! That's what I missed before is here on
one html page :)
Anonymous
? 0 points · 4 months ago
Really awesome and clear article. Thank you!
Anonymous
? 0 points · 4 months ago
Explaining core concepts to its brim.Thanks !
Anonymous
? 0 points · 3 months ago
Thank you very much for this very insightful article. I am a Spring
developer since one year, but always had some issues understanding how
the internals of Spring work and what is actually doing. Thanks to your
article I now have a better understanding of all things that are happening
intrinsically. Kudos!
Anonymous
? 0 points · 4 months ago
Best thing for Spring beginners.
Anonymous
? 0 points · 4 months ago
Is it ok to use 78% Constructor injection and 22% field?
Marco Behler
0 points · 4 months ago
Best would be 66% Constructor, 33% field and 1% method injection ;)
Anonymous
? 0 points · 3 months ago
Best doc I have seen for Spring boot, easy understand
Anonymous
? 0 points · 3 months ago
Thanks for this article! I loved the one on Microservices and this one as
well. Keep em coming :)
goyalshub
G 0 points · 3 months ago
"Spending ~15 minutes reading this guide" are you kidding me?
Marco Behler
0 points · 3 months ago
Time flies by when you enjoy yourself ;)
Anonymous
? 0 points · 3 months ago
Awesome overview of what is spring and what it does ! Many thanks.
Anonymous
? 0 points · 3 months ago
Great article and summary. I would just add that the @Scope("singleton")
is the default setting for a new creation of a bean. Your text suggest it but
doesn't mention it explicitly. Another idea: The @Scope("prototype")
could be used if one connection should have 'UTF-8' encoding while the
other not or different timezones. Than one could easily get the same
basic settings but different flavors of the same settings. Beside this -
wonderful to read this article especially a er working some time already
with Spring. Now getting a little bit more detailed idea. THX
Anonymous
? 0 points · 3 months ago
Well written; with a good deal of links to follow up for more details.
Sterling job!
Anonymous
? 0 points · 3 months ago
thanks so much
Anonymous
? 0 points · 3 months ago
I read other tutorials to explore Spring, and they were winding and had
unnecessary details. Yours was crisp, clear and much useful. Explained
clearly why the framework is so popular. Thanks!!
Anonymous
? 0 points · 42 days ago
Hi Marco, great article , than you for you efforts.
ricksonmenezes
R 0 points · 3 months ago
Marco, what made you write this? The fact that you found this need to fill
the gap with an in-depth write-up requires many shout-outs. Thanks you.
Anonymous
? 0 points · 7 days ago
Thank you for this excellent post! 15 minutes well spent :)
ricksonmenezes
R 0 points · 3 months ago
Hi Marco,
import org.springframework.context.ApplicationContext; import
org.springframework.context.annotation.AnnotationConfigApplicationCon
public class MyApplication {
public static void main(String[] args) {
ApplicationContext ctx = new AnnotationConfigApplicationCont
UserDao userDao = ctx.getBean(UserDao.class);
// User user1 = userDao.findById(1);
// User user2 = userDao.findById(1);
DataSource dataSource = ctx.getBean(DataSource.class);
}
In the code here there is a main method. In my spring mvc project, there
is no main. We have web.xml, our main servlet mapped there, other bean
objects declared in dispatcher-servlet.xml but no main().
Why is it that your demo code has it? Does it not use a servlet? If the
answer is too long, do you think you can point me to a resource?
Thank you for reading.
Marco Behler
1 point · 3 months ago
Hi Rickson, so, there's a couple of things that need to be untangled
here.
Spring, as in the core dependency injection container, has nothing to
do with the web or servlets. You can instantiate an ApplicationContext
anywhere you like, so for example in the main() method or really
anywhere else.
Now, you seem to be using Spring "WebMVC" (!), with the old-school
web.xml & dispatcher-servlet.xml configuration. That means you will
have a Tomcat (or similar servlet container), package up your
application into a .war file and put it into your Tomcat. Your Tomcat will
be able to read he web.xml and - in the end - instantiate a
DispatcherServlet which internally needs (to instantiate) an
ApplicationContext. So what we do here in the main method is
basically hidden for you that way.
There are other, more modern ways, though. First, you could use an
embedded Tomcat and ,second, then use the servlet 3+ servlet
initialization mechanisms without having a web.xml, and at the end
with a Spring Java Config instead of the dispatcher-servlet.xml.
As for further reading, I actually explain and make you practice pretty
much all of these concepts in my new Spring Framework course
https://www.marcobehler.com/academy/learning-spring , which will
be published this week. You will be able to try out the first part for free,
which explains most of these concepts.
Cheers
Anonymous
? 0 points · 2 months ago
Nice overview! The only thing I'd complain about is that it uses annotated
controllers with field injection, which was the standard pattern about 10
years ago. These days, most people agree that @Configuration classes
with constructor injection are the way to go, mostly because that
combination leads to better testability. Don't want to create a login here.
Disagree with me at https://twitter.com/oldJavaGuy
Powered by Commento
Privacy & Terms | Imprint | Contact