In addition to Weibo, there is also WeChat
Please pay attention
WeChat public account
Shulou
2025-03-26 Update From: SLTechnology News&Howtos shulou NAV: SLTechnology News&Howtos > Network Security >
Share
Shulou(Shulou.com)06/01 Report--
1. A simple car
First, let's consider a simple example, where we use the engine class and the car class. In order to describe the problem more clearly, we leave both classes and interfaces empty. Each car will have an engine, and we want to equip the car with the famous MooseEngine.
The Engine class is as follows:
1 public interface Engine {2 3} 4 5 public class SlowEngine implements Engine {6 7} 8 9 public class FastEngine implements Engine {10 11} 12 13 public class MooseEngine implements Engine {14 15}
Then we can get a car class:
1 public class Car {2 3 private MooseEngine engine;4 5}
It's a great car, but even if there are other kinds of engines on the market, we can't have them. We say that the car class and MooseEngine class here are tightly coupled (tightly coupled). Although the MooseEngine is great, what if we want to replace it with another engine?
Back to the top.
two。 Interface programming
You may have noticed that MooseEngine implements the Engine interface. Other engines implement the same interface. We can think about it. When we designed our Car class, we wanted a "car" equipped with a "engine". So we re-implement a Car class, this time using the Engine interface:
1 public class Car {2 3 private Engine engine;4 5}
Interface programming is a very important concept in dependency injection. I heard you scream, "wait a minute, you use the interface here, what about concrete class? where do you set up (set) the engine? I want to equip my car with MooseEngine." We can set it in the following ways:
1 public class Car {2 3 private Engine engine = new MooseEngine (); 4 5}
But is this useful? It doesn't look much different from the first example. Our car is still tightly coupled to MooseEngine. So how do we set up (set or inject) our car engines?
Back to the top.
3. Introduction to dependency injection
Just like the name dependency injection, dependency injection is injecting dependencies, or simply, setting the relationship between different instances. Some people associate it with a Hollywood rule, "Don't call me off, I'll call you." I prefer to call it the "bugger" rule: "I don't care who you are, do as I say." In our first example, Car relies on Engine's realization class MooseEngine. When one class A depends on another class B, the implementation of class B is set directly in class A, and we say that An is tightly coupled to B. In the second example, we decided to use the interface instead of the present class MooseEngine, which makes the Car class more flexible. And we decided not to define the realization class implementation of engine. In other words, we made the Car class loosely coupled (loosely coupled). Car no longer relies on any engine's realization class. So where do we specify which engine we need to use? It's time for dependency injection. Instead of setting a materialized Engine class in the Car class, we inject it from the outside. How can this be achieved?
3.1 use constructors to inject dependencies
One way to set dependencies is to pass the concrete implementation of the dependency class to the constructor. The Car class will look like this:
1 public class Car {2 3 private Engine engine; 4 5 public Car (Engine engine) {6 7 this.engine = engine; 8 9} 10 11}
Then we can use any kind of engine to create Car. For example, one car uses MooseEngine and the other uses crappy SlowEngine:
1 public class Test {2 3 public static void main (String [] args) {4 5 Car myGreatCar = new Car (new MooseEngine ()); 6 7 Car hisCrappyCar = new Car (new SlowEngine ()); 8 9} 10 11}
3.2 use setter to inject dependencies
Another common way to set dependencies is to use the setter method. When you need to inject a lot of dependencies, it is recommended to use the setter method instead of the constructor. Our car class will be implemented as follows:
1 public class Car {2 3 private Engine engine; 4 5 public void setEngine (Engine engine) {6 7 this.engine = engine; 8 9} 10 11}
It is very similar to constructor-based dependency injection, so we can implement the same cars above in the following ways:
1 public class Test {2 3 public static void main (String [] args) {4 5 Car myGreatCar = new Car (); 6 7 myGreatCar.setEngine (new MooseEngine ()); 8 9 Car hisCrappyCar = new Car (); 10 11 hisCrappyCar.setEngine (new SlowEngine ()); 12 13} 14 15}
Back to the top.
4. Using dependency injection in unit testing
If you compare the first example of the Car class with the example of using setter dependency injection, you might think that the latter uses additional steps to implement dependency injection of the Car class. That's right, you have to implement a setter method. But when you are doing unit testing, you will feel that the extra work is worth it. If you are not familiar with unit testing, it is recommended that you read this post that unit testing is toxic. Our Car example is too simple to demonstrate the importance of dependency injection to unit testing. So instead of using this example, we use the previous example of the bonfire story, especially the part of mock used in unit tests. We have a servlet class that "registers" animals on the farm by using remote EJB:
1 public class FarmServlet extends ActionServlet {2 3 public void doAction (ServletData servletData) throws Exception {4 5 String species = servletData.getParameter ("species"); 6 7 String buildingID = servletData.getParameter ("buildingID"); 8 9 if (Str.usable (species) & & Str.usable (buildingID)) {10 11 FarmEJBRemote remote = FarmEJBUtil.getHome (). Create () 12 13 remote.addAnimal (species, buildingID) 14 15} 16 17} 18 19}
You have noticed that FarmServlet is tightly coupled to the FarmEJBRemote instance, retrieving the instance value by calling "FarmEJBUtil.getHome (). Create ()". Doing so can be very difficult to do unit testing. As a unit test, we don't want to use any database. We also don't want to access the EJB server. Because this not only makes unit testing difficult but also slows it down. So in order to unit test the FarmServlet class smoothly, it's best to make it loosely coupled. To clear the tight dependency between FarmServlet and FarmEJBRemote, we can use setter-based dependency injection:
1 public class FarmServlet extends ActionServlet {2 3 private FarmEJBRemote remote; 4 5 public void setRemote (FarmEJBRemote remote) {6 7 this.remote = remote; 8 9} 10 11 public void doAction (ServletData servletData) throws Exception {12 13 String species = servletData.getParameter ("species"); 14 15 String buildingID = servletData.getParameter ("buildingID") 16 17 if (Str.usable (species) & & Str.usable (buildingID)) {18 19 remote.addAnimal (species, buildingID); 20 21} 22 23} 24 25}
In a real deployment package, we ensure that an instance of a FarmServlet remote member created by calling "FarmEJBUtil.getHome (). Create ()" is injected. In our unit test, we used a virtual mock class to simulate FarmEJBRemote. In other words, we implement FarmEJBRemote by using the mock class:
1 class MockFarmEJBRemote implements FarmEJBRemote {2 3 private String species = null; 4 5 private String buildingID = null; 6 7 private int nbCalls = 0; 8 9 public void addAnimal (String species, String buildingID) 10 11 {12 13 this.species = species; 14 15 this.buildingID = buildingID; 16 17 this.nbCalls++ 18 19} 20 21 public String getSpecies () {22 23 return species;24 25} 26 27 public String getBuildingID () {28 29 return buildingID;30 31} 32 33 public int getNbCalls () {34 35 return nbCalls 36 37} 38 39} 40 41 42 43 public class TestFarmServlet extends TestCase {44 45 public void testAddAnimal () throws Exception {46 47 / / Our mock acting like a FarmEJBRemote48 49 MockFarmEJBRemote mockRemote = new MockFarmEJBRemote (); 50 51 / / Our servlet. We set our mock to its remote dependency52 53 FarmServlet servlet = new FarmServlet (); 54 55 servlet.setRemote (mockRemote); 56 57 58 59 / / just another mock acting like a ServletData60 61 MockServletData mockServletData = new MockServletData (); 62 63 mockServletData.getParameter_returns.put ("species", "dog") 64 65 mockServletData.getParameter_returns.put ("buildingID", "27"); 66 67 68 69 servlet.doAction (mockServletData); 70 71 assertEquals (1, mockRemote.getNbCalls ()); 72 73 assertEquals ("dog", mockRemote.getSpecies ()); 74 75 assertEquals (27, mockRemote.getBuildingID ()); 76 77} 78 79}
This makes it easy to test FarmServlet.
Welcome to subscribe "Shulou Technology Information " to get latest news, interesting things and hot topics in the IT industry, and controls the hottest and latest Internet news, technology news and IT industry trends.
Views: 0
*The comments in the above article only represent the author's personal views and do not represent the views and positions of this website. If you have more insights, please feel free to contribute and share.
Continue with the installation of the previous hadoop.First, install zookooper1. Decompress zookoope
"Every 5-10 years, there's a rare product, a really special, very unusual product that's the most un
© 2024 shulou.com SLNews company. All rights reserved.