Forma estándar:
@RestController
public class Main {
UserService userService;
public Main(){
userService = new UserServiceImpl();
}
@GetMapping("/")
public String index(){
return userService.print("Example test");
}
}
Interfaz de servicio de usuario:
public interface UserService {
String print(String text);
}
Clase UserServiceImpl:
public class UserServiceImpl implements UserService {
@Override
public String print(String text) {
return text + " UserServiceImpl";
}
}
Salida: Example test UserServiceImpl
Ese es un gran ejemplo de clases estrechamente acopladas, mal ejemplo de diseño y habrá problemas con las pruebas (PowerMockito también es malo).
Ahora echemos un vistazo a la inyección de dependencia SpringBoot, un buen ejemplo de acoplamiento flojo:
La interfaz sigue siendo la misma.
Clase principal:
@RestController
public class Main {
UserService userService;
@Autowired
public Main(UserService userService){
this.userService = userService;
}
@GetMapping("/")
public String index(){
return userService.print("Example test");
}
}
Clase ServiceUserImpl:
@Component
public class UserServiceImpl implements UserService {
@Override
public String print(String text) {
return text + " UserServiceImpl";
}
}
Salida: Example test UserServiceImpl
y ahora es fácil escribir prueba:
@RunWith(MockitoJUnitRunner.class)
public class MainTest {
@Mock
UserService userService;
@Test
public void indexTest() {
when(userService.print("Example test")).thenReturn("Example test UserServiceImpl");
String result = new Main(userService).index();
assertEquals(result, "Example test UserServiceImpl");
}
}
Mostré @Autowired
anotaciones en el constructor pero también se puede usar en setter o campo.