Nie można autowire pole: RestTemplate w aplikacji Spring boot

109

Otrzymuję poniższy wyjątek podczas uruchamiania aplikacji rozruchu sprężynowego podczas uruchamiania:

org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'testController': Injection of autowired dependencies failed; nested exception is org.springframework.beans.factory.BeanCreationException: Could not autowire field: private org.springframework.web.client.RestTemplate com.micro.test.controller.TestController.restTemplate; nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type [org.springframework.web.client.RestTemplate] found for dependency: expected at least 1 bean which qualifies as autowire candidate for this dependency. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}

Jestem autowiring RestTemplate w moim TestController. Używam Mavena do zarządzania zależnościami.

TestMicroServiceApplication.java

package com.micro.test;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class TestMicroServiceApplication {

    public static void main(String[] args) {
        SpringApplication.run(TestMicroServiceApplication.class, args);
    }
}

TestController.java

    package com.micro.test.controller;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.MediaType;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;

@RestController
public class TestController {

    @Autowired
    private RestTemplate restTemplate;

    @RequestMapping(value="/micro/order/{id}",
        method=RequestMethod.GET,
        produces=MediaType.ALL_VALUE)
    public String placeOrder(@PathVariable("id") int customerId){

        System.out.println("Hit ===> PlaceOrder");

        Object[] customerJson = restTemplate.getForObject("http://localhost:8080/micro/customers", Object[].class);

        System.out.println(customerJson.toString());

        return "false";
    }

}

POM.xml

    <?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.micro.test</groupId>
    <artifactId>Test-MicroService</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>Test-MicroService</name>
    <description>Demo project for Spring Boot</description>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.3.3.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>


</project>
Khuzi
źródło
1
Głosowanie za twoim pytaniem, ponieważ nie jest oczywiste, że kiedy wszystko jest magicznie połączone, RestTemplatenie jest automatycznie tworzone dla ciebie.
daniel.eichten
Upvoted - samouczek na własnej stronie Spring Boot nie mówi nic o tworzeniu fasoli RestTemplate !!
Matt,
Podobny: stackoverflow.com/q/28024942/86967
Brent Bradburn

Odpowiedzi:

174

Dokładnie to mówi błąd. Nie stworzyłeś żadnej RestTemplatefasoli, więc nie może ona automatycznie podłączać. Jeśli potrzebujesz RestTemplate, musisz go zapewnić. Na przykład dodaj następujący kod do TestMicroServiceApplication.java :

@Bean
public RestTemplate restTemplate() {
    return new RestTemplate();
}

Uwaga, we wcześniejszych wersjach Spring Cloud Starter dla Eureki stworzono RestTemplatefasolę, ale to już nie prawda.

g00glen00b
źródło
Bardzo dziękuję za twoją odpowiedź. To pomogło!
Khuzi
19
Głosowałem za pytaniem i odpowiedzią, ponieważ Nie jest oczywiste, że musisz ręcznie utworzyć, RestTemplatekiedy wszystko inne jest magicznie utworzone i połączone. Zwłaszcza jeśli wcześniej używano wiosennej chmury, która zapewnia autokonfigurację RestTemplate. ;-)
daniel.eichten
2
Szczerze mówiąc, z tego powodu umieściłem ten problem tutaj na forum. Spodziewałem się, że RestTemplate zostanie dla mnie połączony. :-) To działało dobrze, gdy dodałem zależność Eureka do POM.xml. To działało dobrze bez definiowania beana RestTemplate. Jedna z klas Eureki mogła zdefiniować tę fasolę.
Khuzi
4
Tylko aktualizacja. Od wersji Spring Boot 1.4.0 RestTemplateBuildermożna używać do zarządzania RestTemplateinstancjami. Przykład tutaj spring.io/guides/gs/consuming-rest
Mensur
Nie mogę jeszcze zaktualizować do SB 1.4.0. Chcę to zrobić z 1.3.8.RELEASE, ale rozwiązanie @ g00glen00b nie działa dla mnie. Używam również spring-cloud-netflixartefaktydu z wersją 1.1.5.RELEASE. Mój RestTemplate jest wywoływany z @RestControllerklasy Java, która używa @Autowireddla RestTemplate. Czy ktoś może pomóc?
ZeroGraviti
33

W zależności od tego, jakich technologii używasz i jakie wersje będą miały wpływ na to, jak zdefiniujesz RestTemplatew swojej @Configurationklasie.

Wiosna> = 4 bez butów sprężynowych

Po prostu zdefiniuj @Bean:

@Bean
public RestTemplate restTemplate() {
    return new RestTemplate();
}

Spring Boot <= 1.3

Nie musisz go definiować, Spring Boot automatycznie definiuje go za Ciebie.

Spring Boot> = 1.4

Spring Boot nie definiuje już automatycznie a, RestTemplateale zamiast tego definiuje, RestTemplateBuilderco pozwala na większą kontrolę nad tworzonym RestTemplate. Możesz wstrzyknąć RestTemplateBuilderargument jako argument w swojej @Beanmetodzie, aby utworzyć RestTemplate:

@Bean
public RestTemplate restTemplate(RestTemplateBuilder builder) {
   // Do any additional configuration here
   return builder.build();
}

Używanie go w klasie

@Autowired
private RestTemplate restTemplate;

Odniesienie

Sahil Chhabra
źródło
8

Jeśli TestRestTemplate jest prawidłową opcją w teście jednostkowym, ta dokumentacja może być odpowiednia

http://docs.spring.io/spring-boot/docs/1.4.1.RELEASE/reference/htmlsingle/#boot-features-rest-templates-test-utility

Krótka odpowiedź: jeśli używasz

@SpringBootTest(webEnvironment=WebEnvironment.RANDOM_PORT)

wtedy @Autowiredzadziała. Jeśli używasz

@SpringBootTest(webEnvironment=WebEnvironment.MOCK)

następnie utwórz TestRestTemplate w ten sposób

private TestRestTemplate template = new TestRestTemplate();
Mark K.
źródło
1

Błąd wskazuje bezpośrednio, że RestTemplatefasola nie jest zdefiniowana w kontekście i nie może jej załadować.

  1. Zdefiniuj ziarno dla RestTemplate, a następnie użyj go
  2. Użyj nowego wystąpienia RestTemplate

Jeśli masz pewność, że ziarno jest zdefiniowane dla RestTemplate, użyj poniższego, aby wydrukować ziarna dostępne w kontekście załadowanym przez aplikację rozruchową sprężyny

ApplicationContext ctx = SpringApplication.run(Application.class, args);
String[] beanNames = ctx.getBeanDefinitionNames();
Arrays.sort(beanNames);
for (String beanName : beanNames) {
    System.out.println(beanName);
}

Jeśli zawiera ziarno o podanej nazwie / typie, wszystko jest w porządku. Albo zdefiniuj nową fasolę i użyj jej.

VinayVeluri
źródło
1

Ponieważ instancje RestTemplate często wymagają dostosowania przed użyciem, Spring Boot nie udostępnia żadnego automatycznie konfigurowanego komponentu bean RestTemplate.

RestTemplateBuilder oferuje właściwy sposób konfigurowania i tworzenia instancji komponentu bean szablonu reszty, na przykład dla podstawowego uwierzytelniania lub przechwytywaczy.

@Bean
public RestTemplate restTemplate(RestTemplateBuilder builder) {
    return builder
                .basicAuthorization("user", "name") // Optional Basic auth example
                .interceptors(new MyCustomInterceptor()) // Optional Custom interceptors, etc..
                .build();
}
Pierrick HYMBERT
źródło
1
  • Musisz dodać @Bean public RestTemplate restTemplate(RestTemplateBuilder builder){ return builder.build(); }
Ranushka Lakmal Sankalpa
źródło
0

Upewnij się, że dwie rzeczy:

1- Użyj @Beanadnotacji z metodą.

@Bean
public RestTemplate restTemplate(RestTemplateBuilder builder){
    return builder.build();
}

2- Zakres tej metody powinien być publiczny, a nie prywatny .

Kompletny przykład -

@Service
public class MakeHttpsCallImpl implements MakeHttpsCall {

@Autowired
private RestTemplate restTemplate;

@Override
public String makeHttpsCall() {
    return restTemplate.getForObject("https://localhost:8085/onewayssl/v1/test",String.class);
}

@Bean
public RestTemplate restTemplate(RestTemplateBuilder builder){
    return builder.build();
}
}
winorośl
źródło
0

W najprostszy sposób udało mi się osiągnąć podobny wyczyn, używając poniższego kodu ( odniesienie ), ale radziłbym nie wykonywać wywołań API w kontrolerach ( zasady SOLID ). Również autoprzewodowanie w ten sposób jest lepiej zoptymalizowane niż tradycyjny sposób.

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.web.client.RestTemplateBuilder;
import org.springframework.http.MediaType;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;

@RestController
public class TestController {

    private final RestTemplate restTemplate;


    @Autowired
    public TestController(RestTemplateBuilder builder) {
        this.restTemplate = builder.build();
    }

    @RequestMapping(value="/micro/order/{id}", method= RequestMethod.GET, produces= MediaType.ALL_VALUE)
    public String placeOrder(@PathVariable("id") int customerId){

        System.out.println("Hit ===> PlaceOrder");

        Object[] customerJson = restTemplate.getForObject("http://localhost:8080/micro/customers", Object[].class);

        System.out.println(customerJson.toString());

        return "false";
    }
}
Padi kodwo
źródło
0

próbujesz wstrzyknąć restTemplate, ale musisz utworzyć klasę konfiguracji. wtedy musisz utworzyć fasolę, która zwróci ci nowy RestTemplate, patrz poniższy przykład.

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;


@Configuration
public class YourConfigClass {


    @Bean
    public RestTemplate restTesmplate() {
        return new RestTemplate();
    }

}
Fazle Subhan
źródło