org.sf.beans.factory.NoSuchBeanDefinitionException: нет подходящего bean-компонента типа X

Я пытаюсь настроить простой проект с пользовательским репозиторием в Spring Boot с Hibernate как JPA, я попробовал этот и это учебники, и в обоих (и везде) @SpringBootApplication кажется чтобы сделать репозиторий доступным для всей области действия приложения, поскольку он включает @ComponentScan, за исключением того, что в моем случае это не так. Даже этот ответ выглядит адаптированным к моей ситуации, но нет.

Структура проекта

project
│   └── src
│       └── main
│           └── java
│               └── medisam
|                    └── Application.java
│   └── src
│       └── main
│           └── java
│               └── medisam
│                   └── entity
│                       └── User.java
|                       └── UserRepository.java
│    └── src
│        └── main
│            └── java
│               └── medisam
│                   └── controller
│                       └── MainController.java

С такой структурой и этим кодом:

@SpringBootApplication
public class Application {
    ...
    private static final Logger LOG = LoggerFactory.getLogger(Application.class);
    @Autowired
    private UserRepository repo;

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

Я получил:

... Ошибка создания bean-компонента с именем 'application' : неудовлетворенная зависимость, выраженная через поле 'repo'; вложенное исключение org.springframework.beans.factory.NoSuchBeanDefinitionException : нет подходящего bean-компонента типа ' medisam.entity.UserRepository ' available: ожидается как минимум 1 bean-компонент, который квалифицируется как autowire кандидат . Аннотации зависимостей: {@ org.springframework.beans.factory.annotation.Autowired (required = true)} ...

Только когда я это сделаю:

@ComponentScan(basePackages = {"medisam.controller", "medisam.entity"})
@EntityScan({"medisam.entity"})
@EnableJpaRepositories(basePackages = {"medisam.entity"})
public class Application {
    ...
    private static final Logger LOG = LoggerFactory.getLogger(Application.class);
    @Autowired
    private UserRepository repo;

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

что я получаю рабочий код, но если действительно есть способ добиться этого с помощью только @SpringBootApplication, я бы хотел его использовать.

Может быть, @SpringBootApplication сканирует неправильно?

Помогите, пожалуйста

ИЗМЕНИТЬ

UserRepository.java

package medisam.entity;

import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.stereotype.Repository;

@Repository
public interface UserRepository extends JpaRepository<User, Integer> {

}

User.java

package medisam.entity;

import java.util.Set;
import javax.persistence.Access;
import javax.persistence.AccessType;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.JoinTable;
import javax.persistence.ManyToMany;

@Entity
public class User {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Integer id;

    private String name;
    private String email;
    private String username;
    private String password;
    private boolean enabled;

    @ManyToMany(targetEntity = Role.class)
    @Access(AccessType.FIELD)
    @JoinTable(name = "user_role", joinColumns = @JoinColumn(name = "user_id", referencedColumnName = "id"),
            inverseJoinColumns = @JoinColumn(name = "role_id", referencedColumnName = "id"))
    private Set<Role> roles;

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    public boolean isEnabled() {
        return enabled;
    }

    public void setEnabled(boolean enabled) {
        this.enabled = enabled;
    }

    public Set<Role> getRoles() {
        return roles;
    }

    public void setRoles(Set<Role> roles) {
        this.roles = roles;
    }
}

person Scaramouche    schedule 11.10.2018    source источник
comment
@SpringBootApplication будет автоматически сканировать ваши объекты / репо, если они находятся в корневом каталоге или в его корневых пакетах, если они находятся за пределами вашего основного класса, тогда вам необходимо использовать автоматическое сканирование пакетов для репозиториев Jpa   -  person kj007    schedule 11.10.2018
comment
@ kj007, учитывая структуру вопроса, правильно ли расположены мои объекты, чтобы @SpringBootApplication мог их правильно сканировать?   -  person Scaramouche    schedule 11.10.2018
comment
Нет, ваш пакет сущностей должен находиться в рамках проекта- ›project-main-› src- ›main-java-› medisam, и автоматическая настройка будет работать.   -  person kj007    schedule 11.10.2018
comment
@ kj007, у меня плохо, но, возможно, структура, которую я опубликовал, немного вводит в заблуждение: project-main, project-entities и project-controllers не существуют, они просто существуют для логического разделения классов, они не являются папками в моем проекте, попробуйте увидеть дерево без них , src находится прямо внутри project. Извините, я обновил вопрос, чтобы отразить реальную структуру   -  person Scaramouche    schedule 11.10.2018
comment
ваша структура кажется прекрасной и должна автоматически настраивать объекты / репозиторий и контроллеры, не могли бы вы просто аннотировать UserRepository с помощью @Repository, а также очистить свой проект, запустив mvn clean install. если все еще не работает, не могли бы вы поделиться своим кодом на github, я могу посмотреть ... пожалуйста, также покажите класс UserRepository.   -  person kj007    schedule 11.10.2018
comment
@ kj007, я только что добавил User.java и UserRepository.java. Я использую не Maven, а Gradle. Я уже очистил и построил свой проект в NetBeans. Стоит ли мне делать что-нибудь еще в Gradle?   -  person Scaramouche    schedule 11.10.2018
comment
Позвольте нам продолжить это обсуждение в чате.   -  person kj007    schedule 11.10.2018


Ответы (1)


Я просмотрел ваш код на gihub и обнаружил, что вы импортировали неправильное имя пакета в класс Application.java и ServiceResponse.java.

Пожалуйста, импортируйте основной пакет как package medisam; вместо package hello;.

как показано ниже:

Класс вашего приложения:

package medisam;
import medisam.entity.UserRepository;
import nz.net.ultraq.thymeleaf.LayoutDialect;
import nz.net.ultraq.thymeleaf.decorators.strategies.GroupingStrategy;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Description;
import org.springframework.web.servlet.view.InternalResourceViewResolver;
import org.springframework.web.servlet.view.JstlView;
import org.thymeleaf.extras.java8time.dialect.Java8TimeDialect;
import org.thymeleaf.extras.springsecurity4.dialect.SpringSecurityDialect;
import org.thymeleaf.spring5.SpringTemplateEngine;
import org.thymeleaf.spring5.templateresolver.SpringResourceTemplateResolver;
import org.thymeleaf.spring5.view.ThymeleafViewResolver;
import org.thymeleaf.templatemode.TemplateMode;
import org.thymeleaf.templateresolver.ITemplateResolver;
import org.thymeleaf.templateresolver.UrlTemplateResolver;

@SpringBootApplication
public class Application {

а вот ваш ServiceResponse

package medisam;

import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import org.apache.catalina.util.ResourceSet;

import java.util.ArrayList;
import java.util.List;

@JsonIgnoreProperties(ignoreUnknown = true)
public class ServiceResponse {
person kj007    schedule 11.10.2018