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

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

ただいまの
回答率

88.58%

Spring bootにおけるビルド失敗

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 19K+

sanezane

score 85

Spring boot 2.1.0にてWebアプリケーションを勉強中です。その中で、ビルドエラーが発生し対処が不明のためご意見を頂戴したいと思っております。

アプリケーション仕様:部屋予約サイト

//エラーログ

Error starting ApplicationContext. To display the conditions report re-run your application with 'debug' enabled.
2018-11-23 03:43:45.939 ERROR 23104 --- [           main] o.s.b.d.LoggingFailureAnalysisReporter   : 

***************************
APPLICATION FAILED TO START
***************************

Description:

Field roomService in com.example.demo.RoomsController required a bean of type 'domain.room.RoomService' that could not be found.

The injection point has the following annotations:
    - @org.springframework.beans.factory.annotation.Autowired(required=true)

RoomsControllerRoomServiceのBean定義がないと言われていると解釈していますがRoomsControllerには@AutowiredアノテーションがついているのでBeanアノテーションは必要ないかと思っているですが、そもそもエラーの解釈が違い、違うアプローチが必要なのでしょうか?

package com.example.demo;

import java.time.LocalDate;
import java.util.List;
import domain.model.ReservableRoom;
import domain.room.RoomService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.format.annotation.DateTimeFormat;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
@RequestMapping("rooms")
public class RoomsController {
    @Autowired
    RoomService roomService;

    @RequestMapping(value = "{date}", method = RequestMethod.GET)
    String listRooms(@DateTimeFormat(iso = DateTimeFormat.ISO.DATE) @PathVariable("date") LocalDate date, Model model) {
        List<ReservableRoom> rooms = roomService.findReservableRooms(date);
        model.addAttribute("rooms", rooms);
        return "room/listRooms";
    }

    @RequestMapping(method = RequestMethod.GET)
    String listRooms(Model model) {
        LocalDate today = LocalDate.now();
        model.addAttribute("date", today);
        return listRooms(today, model);
    }
}
package domain.room;

import java.time.LocalDate;
import java.util.List;
import domain.model.ReservableRoom;
import domain.repository.room.ReservableRoomRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

@Service
@Transactional
public class RoomService {
    @Autowired
    ReservableRoomRepository reservableRoomRepository;
    @Autowired
    MeetingRoomRepository meetingRoomRepository;

    public List<ReservableRoom> findReservableRooms(LocalDate date) {
        return reservableRoomRepository.findByReservableRoomId_reservedDateOrderByReservableRoomId_roomIdAsc(date);
    }

//    public MeetingRoom findMeetingRoom(Integer roomId) {
//        return meetingRoomRepository.findOne(roomId);
//    }
}

教材サンプルソースコードはダウンロード可能なのですが公共の場にリンクを貼っていいのか判断がつきませんのでエラーに該当するクラスのみソースコードを貼っておきます。

追加情報:パッケージエクスプローラのキャプチャ追加
イメージ説明

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

質問への追記・修正、ベストアンサー選択の依頼

  • rubytomato

    2018/11/23 16:58

    パッケージの定義(構造)に問題があるように思います。パッケージの構造がわかるような情報(たとえばパッケージとソースコードの一覧が確認できるプロジェクトエクスプローラーのスクリーンショットなど)があるといいと思います。

    キャンセル

  • sanezane

    2018/11/23 17:03

    すみません!貼ろうと思って忘れていました。。見にくいと思ったのでフォルダ構成は該当ソースのフォルダしか開いていませんが確認できますでしょか。

    キャンセル

  • rubytomato

    2018/11/23 17:54

    情報の追記ありがとうございます。回答致しましたのでご確認ください。

    キャンセル

回答 2

checkベストアンサー

+1

現在のパッケージの構造をフォルダ風に表現すると以下のようになるとおもいます。
ご覧の通り、com.example.demodomainが別々の階層になっています。

com/
  |
  +--- example/
         |
         +--- demo/
               |
               +--- RoomsController.java
               +--- SampleApplication.java
               +--- TyhmeleafConfig.java

domain/
  |
  +--- model/
  |     |
  |     +--- ReservableRoom.java
  |
  +--- repository/
  |     |
  |     +--- room/
  |            |
  |            +--- ReservableRoomRepository.java
  |
  +--- room/
        |
        +--- MeetingRoomRepository.java
        +--- RoomService.java

これを、以下のように(model、repository、roomパッケージをcom.example.demoパッケージ下に移動)統合し、

com/
  |
  +--- example/
         |
         +--- demo/
               |
               +--- RoomsController.java
               +--- SampleApplication.java
               +--- TyhmeleafConfig.java
               |
               +--- model/
               |     |
               |     +--- ReservableRoom.java
               |
               +--- repository/
               |     |
               |     +--- room/
               |            |
               |            +--- ReservableRoomRepository.java
               |
               +--- room/
                     |
                     +--- MeetingRoomRepository.java
                     +--- RoomService.java

移動したjavaソースのpackage宣言も修正すると解決すると考えています。
たとえば、RoomService.javaであれば

package domain.room;

@Service
@Transactional
public class RoomService {
}

から

package com.example.demo.room;

@Service
@Transactional
public class RoomService {
}

のようにです。

このようなパッケージ構造にするとなぜ解決するか(解決するかはまだわからいないですが)ですが、Spring Bootでは@SpringBootApplicationアノテーションが付いたクラスがある位置からパッケージスキャンが行われます。
パッケージスキャンを簡単に説明しますと、アプリケーションの起動時に@Controller@Service@Repository@Componentアノテーションが付いているクラスを探し出してインスタンス化し、Springの管理下に納めます。
そして、Springの管理下にあるインスタンスは@Autowiredアノテーションが付いたフィールドへ、必要なタイミングで代入(インジェクト)されるという風になっています。

今回のご質問されたプロジェクトでは、@SpringBootApplicationアノテーションが付いたクラスは、おそらくSampleApplicationクラスだと思いますが、パッケージスキャンはこのクラスの位置、つまりcom.example.demoから始まります。このため別の階層にあるdomainはスキャンの対象外になり、このパッケージ下の@Serviceアノテーションなどが付いたクラスは、Springの管理下にはなりません。

これがRoomServiceがAutowiredされない原因だと考えています。
この考え方に間違いがなければ、上記の内容で問題は解決できるとおもっているのですが、この方法で解決した場合、参考にされている教材とプロジェクトの内容に差が出てしまい、このさき困ることが出てくるかもしれません。

なので、いったん教材の方の@SpringBootApplicationアノテーションが付いているクラスやコントローラクラスがどのパッケージの位置にあるのかを確認して頂き、そちらの内容にあわせてcom.example.demoのパッケージ下のクラスを移動させることがいいかもしれません。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/11/23 21:48

    ありがとうございます。rubytomatoさんのおかげで勉強になっております。今回の質問でアノテーションの基本的な使い方と意味を勉強できました。今回のエラーは解決いたしました。

    キャンセル

+1

SpringBootの @SpringBootApplication は、@Configuration@EnableAutoConfiguration@ComponentScan を内包します。

特に @ComponentScan はSpringFrameworkの管理下におくクラスをパッケージ名で指定しますが、これを指定しないときは、このアノテーションが存在するパッケージとサブパッケージを対象にします。これは @SpringBootApplicationでも同様です。

もしデフォルトの動作を変えたい場合は、スキャン対象のパッケージを全て記載します。

以上の詳細は、https://docs.spring.io/spring-boot/docs/2.1.0.RELEASE/reference/htmlsingle/#using-boot-structuring-your-code とその前後の章や節を参考にされると良いでしょう。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

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

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

関連した質問

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