質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.64%

SpringBootsの実行時エラー: No property findEmployees found for type EmployeeInfo!

受付中

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 2,251

solli

score 4

言語:java 
FW : Spring Boot

spring bootを使ってログイン画面、一覧画面、詳細画面を表示する簡単なwebアプリケーションを作成しています。

ログイン画面の実装は終了し、
一覧画面の実装に入りましたがアプリ実行時エラーが出て詰まってしまいます。

やりたいこと
EmployeeInfoRepositoryImplクラスで employee_infoテーブルの(とりあえず)全件取得をしようとしていますが、 No property findEmployees found for type EmployeeInfo!というエラーが出てしまいます。

初心者&独学で大変困っております。どなたかご教授お願いします。

エラー内容

2019-11-06 14:09:35.987 ERROR 1635 --- [           main] o.s.boot.SpringApplication               : Application run failed

org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'listController': Unsatisfied dependency expressed through field 'listservice'; nested exception is org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'listServiceImpl': Unsatisfied dependency expressed through field 'listRepository'; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'employeeInfoRepository': Invocation of init method failed; nested exception is java.lang.IllegalArgumentException: Failed to create query for method public abstract java.util.List com.employee.repository.EmployeeInfoRepository.findEmployees()! No property findEmployees found for type EmployeeInfo!

関連するクラスを↓に貼ります。

EmployeeInfoRepository

package com.employee.repository;

import java.util.List;

import org.springframework.data.jpa.repository.JpaRepository;
import com.employee.entity.EmployeeInfo;

public interface EmployeeInfoRepository extends JpaRepository<EmployeeInfo, String> {

    public List<EmployeeInfo> findEmployees();


}

EmployeeInfoRepositoryImpl

package com.employee.repository;

import java.util.List;

import javax.persistence.EntityManager;
import javax.persistence.Query;

import org.springframework.stereotype.Repository;

import com.employee.entity.EmployeeInfo;

@Repository
public abstract class EmployeeInfoRepositoryImpl implements EmployeeInfoRepository {


    private EntityManager entityManager;

    @SuppressWarnings("unchecked")
    @Override
    public List<EmployeeInfo> findEmployees() {

    List<EmployeeInfo> empInfoList = null;

    //String sql = "FROM employee_info AS I INNER JOIN employee_state AS S ON S.employee_info_id = I.employee_id INNER JOIN company_info AS C ON C.company_id = I.company_info_id";

    String sql = "FROM employee_info";

    Query query = entityManager.createQuery(sql);

    empInfoList = query.getResultList();

    return empInfoList;
    }
}


以上になります。どなたかお助けいただけるとありがたいです!!

    

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

+1

EmployeeInfoRepositoryImplクラスの実装に問題があるようです。下記のように実装していますが、

@Repository
public abstract class EmployeeInfoRepositoryImpl implements EmployeeInfoRepository {

}

abstractだとインスタンスが生成できないので外してください。(もしくはこのEmployeeInfoRepositoryImplをextendsした具象クラスを作成してください)
・またEmployeeInfoRepositoryをimplementsするとこのインタフェースに定義されているメソッドをすべてオーバライドしなければならなくなるので、implementsしないようにしてください。(もしくはすべてのメソッドをオーバーライドしてください)

修正すると下記のようになります。インタフェースをimplementsしないようにすることでクラス名がおかしくなるので、ここでは暫定的にEmployeeInfoRepositoryServiceというクラス名に変えて説明を続けます。
同様にアノテーションも@Serviceに変えています。
この時点で質問文に記載のエラーは解消されると思います。

@Service
public class EmployeeInfoRepositoryService {

}

ただ、ほかにも実行時にエラーになる問題がありますので修正します。このままではEntityManagerは使えないので、Springにインジェクトしてもらう必要があります。下記のようにコンストラクタを宣言してインジェクトしてください。

@Service
public class EmployeeInfoRepositoryService {
    private final EntityManager entityManager;

    public EmployeeInfoRepositoryService(EntityManager entityManager) {
        this.entityManager = entityManager;
    }

}

クエリの書き方にも問題があるので修正します。sqlを"FROM employee_info";と書いていますがFROMには実テーブル名ではなくクラス名を記述します。
※なおTypedQueryを使っていますが、これはあくまでも一例なのでこのように書かなければいけないということではありません。

String sql = "FROM EmployeeInfo e";
TypedQuery<EmployeeInfo> query = entityManager.createQuery(sql, EmployeeInfo.class);
List<EmployeeInfo> empInfoList = query.getResultList();

以上の修正でやりたいことの

EmployeeInfoRepositoryImplクラスで employee_infoテーブルの(とりあえず)全件取得をしようとしています

employee_infoテーブルのデータを取得することができると思います。

上記の実装はEntityManagerを使ってデータを取得する方法ですが、もっと簡単な方法としてEmployeeInfoRepositoryを使用する方法があります。

public interface EmployeeInfoRepository extends JpaRepository<EmployeeInfo, String> {
    public List<EmployeeInfo> findEmployees();
}

findEmployeesメソッドを宣言していますが不要なので削除します。
修正後は下記のようになります。

public interface EmployeeInfoRepository extends JpaRepository<EmployeeInfo, String> {
}

あとはemployee_infoテーブルを取得したいクラスでこのリポジトリをインジェクトして使用します。
たとえばコントローラクラスで使用する場合、下記のようにインジェクトし、リポジトリのfindAllメソッドでデータを取得します。findAllはメソッド名の通りデータを全件取得します。

@RestController
public class EmployeeInfoController {
    private final EmployeeInfoRepository repository;

    public EmployeeInfoController(EmployeeInfoRepository repository) {
        this.repository= repository;
    }

    @GetMapping(path = "list", produces = MediaType.APPLICATION_JSON_VALUE)
    public List<EmployeeInfo> findAll() {
        return repository.findAll();
    }

}

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/11/06 22:44

    大変わかりやすいです。ありがとうございます!

    1点、質問なのですが
    外部結合したテーブルの値を取得する場合はfindAllメソッドや、
    リポジトリのメソッドは使用できないよ思います。
    外部結合されたテーブルの値を取得する場合はどんなメソッドを使用すればいいでしょうか?
    (findEmployeesメソッドを書いたのは、のちに独自のメソッドを作成しようとしたからです)

    すみませんが、よろしくお願いします。

    キャンセル

  • 2019/11/07 00:14

    テーブルの関連はエンティティクラスで@OneToManyや@ManyToOneなどのアノテーションで表現するのですが、複雑な問い合わせの場合はJPQLで記述する必要があるかもしれません。

    正確にお答えするには、具体的にどのようなクエリを発行したいかを知る必要がありますので、
    エンティティクラスのソースコード(ないしはcreate table文)と、実行したいSQL文を質問内容に追記ください。

    それと質問に質問で返すようで申し訳ありませんが、”やりたいこと”に記載されている

    > employee_infoテーブルの(とりあえず)全件取得をしようとしています

    という部分は解決できたのでしょうか?

    キャンセル

  • 2019/11/24 13:15

    オフラインの質問で解決できました!
    ありがとうございました。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.64%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る