Modo standard:
@RestController
public class Main {
UserService userService;
public Main(){
userService = new UserServiceImpl();
}
@GetMapping("/")
public String index(){
return userService.print("Example test");
}
}
Interfaccia del servizio utente:
public interface UserService {
String print(String text);
}
Classe UserServiceImpl:
public class UserServiceImpl implements UserService {
@Override
public String print(String text) {
return text + " UserServiceImpl";
}
}
Produzione: Example test UserServiceImpl
Questo è un ottimo esempio di classi ad accoppiamento stretto, cattivo esempio di progettazione e ci saranno problemi con i test (anche PowerMockito è male).
Ora diamo un'occhiata all'iniezione di dipendenza SpringBoot, un bell'esempio di accoppiamento lento:
L'interfaccia rimane la stessa,
Classe principale:
@RestController
public class Main {
UserService userService;
@Autowired
public Main(UserService userService){
this.userService = userService;
}
@GetMapping("/")
public String index(){
return userService.print("Example test");
}
}
Classe ServiceUserImpl:
@Component
public class UserServiceImpl implements UserService {
@Override
public String print(String text) {
return text + " UserServiceImpl";
}
}
Produzione: Example test UserServiceImpl
e ora è facile scrivere test:
@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");
}
}
Ho mostrato @Autowired
un'annotazione sul costruttore ma può anche essere utilizzata su setter o campo.