thinking

Spring has taken over half of our Java development framework, and we’ve been using it since the beginning. But why use Spring at all, which many people may not have thought about? Many people may also be too overwhelmed by demand to think about such seemingly natural questions. Today, let’s take a closer look at why we should use Spring IOC.

Reverse thinking

Suppose we developed a common piece of user logic using the traditional MVC approach when there was no Spring IOC framework.

User DAO

public class UserDAO {

    private String database;

    public UserDAO(String dataBase) {
        this.database = dataBase;
    }
    public void doSomething() {
        System.out.println("Save users!"); }}Copy the code

The user Service

public class UserService {

    private UserDAO dao;

    public UserService(UserDAO dao) {
        this.dao = dao;
    }
    public void doSomething() { dao.doSomething(); }}Copy the code

The user Controller

public class Controller {

    public UserService service;

    public Controller(UserService userService) {
        this.service = userService;
    }

    public void doSomething() { service.doSomething(); }}Copy the code

Then we have to manually create the objects one by one and assemble the DAO, service, and Controller in order before we can call them.

    public static void main(String[] args) {

        UserDAO dao = new UserDAO("mysql");
        UserService service = new UserService(dao);
        Controller controller = new Controller(service);

        controller.doSomething();

    }

Copy the code

What are the disadvantages of this approach?

  1. Where controllers are generated, we have to create a DAO, then a service, and then a Controller, which is a tedious process.
  2. In each of these three layers, the upper layer needs to know how the lower layer is created, and the upper layer must create the lower layer itself, resulting in tight coupling. Why do business programmers need to know the database password and create their own DAOs when writing a business? Not only that, but if the DAO’s database password changes, it needs to be changed everywhere the Controller is generated.
  3. Concrete objects are generated through the new keyword, which is hardcoded in a way that violates the principles of interface oriented programming. When we change from Hibernate to Mybatis one day, we will need to change in every new DAO.
  4. We waste resources by creating objects too often.

Now let’s take a look at SpringIOC and see what the code looks like.

public static void main(String[] args) {

        ApplicationContext context = new ClassPathXmlApplicationContext("classpath:application.xml");
        Controller controller = (Controller) context.getBean("controller");
        controller.doSomething();
        
    }
Copy the code

Obviously, with IOC, we just ask the container for the beans we need. The IOC has addressed the following pain points:

  1. Decoupling between beans, which is reflected in the fact that we do not hardcode the dependencies between beans in our code. (Building the objects sequentially without a new operation was done internally by springIOC to help implement dependency injection.) On the one hand, the IOC container switches from setting dependencies via new objects to dynamically setting dependencies at runtime.
  2. The IOC container naturally provides us with singletons.
  3. When the DAO needs to be replaced, we only need to change the DAO implementation class in the configuration file without breaking the previous code.
  4. The upper layer now does not need to know how the lower layer was created.

In case you’re wondering, let’s take a look at the definition of IOC.

define

Inversion of Control, or IoC, is a design principle used in object-oriented programming to reduce coupling between computer code. One of the most common is called Dependency Injection, or DI, and another is called Dependency Lookup. With inversion of control, when an object is created, it is passed a reference to the object on which it depends by an external entity that regulates all objects in the system. In other words, dependencies are injected into objects.

Interpretation of the

Inversion of control, where is the control? The control is in our first example, the upper layer needs to control the new lower layer object. And what does reversal mean? Inverting means that we hand over control of object creation to whom? This is handed over to the IOC container, which is responsible for creating and populating specific objects wherever they are declared to be needed. IOC and DI may be confusing to students, but IOC is the interface. It dictates what is to be achieved. Dependency injection (DI) is the concrete implementation that achieves what IOC wants. The IOC’s philosophy embodies one of the laws of object-oriented design, the Hollywood law: “Don’t come to us, we’ll come to you”; That is, the IoC container helps the object to find the corresponding dependent object and inject, rather than the object to find.

How to reduce coupling? For example, when we add a plug-in in the IDEA development environment, we need to know what dependencies the plug-in needs. How can we successfully start the plug-in? Isn’t it pretty user unfriendly? Since we’re going to use the plug-in, that means we’re coupled to the plug-in and can’t avoid it. But if we also need to know the dependencies required by the plug-in and the steps to enable the plug-in, the coupling between us and the plug-in is stronger. So SpringIOC’s definition says “de-coupling” rather than “de-coupling,” but as long as the coupling degree is reduced, it is easier to maintain and extend the code. Think about it? Is it similar to Maven’s mechanics? When a dependency is declared in Maven, Maven automatically recurses to find other dependencies needed for that dependency. You can think of yourself as the designer of an object. You design a drawing of the internal structure of the object, give it to SpringIOC, and it automatically generates the object from that drawing.

The IOC container implements a componentization of object granularity in development, placing each object in the container as a component. Each component is pluggable, which is the best way to decouple objects. One is by deferring the coupling between objects from compile time to run time. The object can be used as soon as it is needed, and the client programmer does not need to know the ins and outs of the object. And the unified management of beans by IOC container makes the implementation of AOP more convenient. In this way, our client programmers can focus more on writing business code.

I hope this article will be helpful to readers. If there are mistakes, please point them out.

My personal blog, vc2x.com