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

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

ただいまの
回答率

90.99%

  • Java

    12206questions

    Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

  • Spring Boot

    381questions

    Spring Bootは、Javaのフレームワークの一つ。Springプロジェクトが提供する様々なフレームワークを統合した、アプリケーションを高速で開発するために設計されたフレームワークです。

  • Oracle Database

    52questions

    Oracle Databaseは、米オラクルが開発・販売を行うリレーショナルデータベース管理システムです。

Spring Boot で IndexOutOfBoundsException

解決済

回答 2

投稿

  • 評価
  • クリップ 0
  • VIEW 112

mjxg

score 6

Spring Boot でチャットアプリを作成しています。
/roomsでformのボタンをクリックするとPostメソッドからgetUserByIdメソッドでUsersテーブルからselectするのですが、
レコードが0件だったようでIndexOutOfBoundsExceptionになりました。
まだユーザー認証を作ってないのでダミーのユーザーID1を決め打ちしてます。
ログを貼ろうとしたら字数制限にかかってしまいましたので、必要な場合はお申しつけください。
どのあたりから調べればよいかご指導いただけないでしょうか。

よろしくお願いいたします。

DDL

create table app.users (
  id number generated always as identity primary key,
  name nvarchar2(32) not null,
  password nvarchar2(32) not null);

create table app.rooms (
  id number generated always as identity primary key,
  owner_id number references app.users(id));

create table app.memberships (
  membership_id number generated always as identity primary key,
  room_id number references app.users(id),
  user_id number references app.rooms(id));

create table app.messages (
  id number generated always as identity primary key,
  sender_id number references app.users(id),
  room_id number references app.rooms(id),
  text nvarchar2(256) not null,
  created_at timestamp default systimestamp not null);


usersテーブルの状態
select * from app.users

        ID NAME                             PASSWORD
---------- -------------------------------- --------------------------------
         1 test01                           password


application.yml

spring:
  datasource:
  url: jdbc:oracle:thin:@localhost:1521:chatdb
  username: app
  password: password
  driverClassName: oracle.jdbc.driver.OracleDriver
server:
  port: 80


RoomsController.java

package com.springTalk.web;

import java.util.List;

import javax.servlet.http.HttpServletRequest;
import javax.validation.Valid;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.validation.BindingResult;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;

import com.springTalk.domain.Message;
import com.springTalk.domain.Room;
import com.springTalk.domain.User;
import com.springTalk.service.MessageService;
import com.springTalk.service.RoomService;
import com.springTalk.service.UserService;

@Controller
public class RoomsController {
  @Autowired
  private UserService userService;
  @Autowired
  private RoomService roomService;
  @Autowired
  private MessageService messageService;

  @GetMapping("/rooms")
  public String roomsGet(Model model) {
    List<Room> rooms = roomService.getRoomsOrderByIdDesc();
    model.addAttribute("room", rooms);
    model.addAttribute("roomsForm", new RoomsForm());
    return "rooms";
  }

  @PostMapping("/rooms")
  public String roomsPost(Model model, @Valid RoomsForm roomsForm, BindingResult bindingResult,
      HttpServletRequest request) {
    if (bindingResult.hasErrors()) {
      List<Room> rooms = roomService.getRoomsOrderByIdDesc();
      model.addAttribute("room", rooms);
      model.addAttribute("roomsForm", new RoomsForm());
      return "rooms";
    }

    // get dummy user
    // ToDo: get user id from request
    final Long OWNER_ID = 1L;
    User user = userService.getUserById(OWNER_ID).get(0);
    Room room = roomService.saveAndFlush(new Room(user));
    messageService.save(new Message(user, room, roomsForm.getText()));
    return "redirect:/rooms";
  }
}

UserService.java

package com.springTalk.service;

import java.util.List;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import com.springTalk.domain.User;
import com.springTalk.domain.UserRepository;

@Service
public class UserService {

  @Autowired
  private UserRepository repository;

  public List<User> getUserById(Long id) {
    return repository.findById(id);
  }

  public List<User> getUserByNameOrderById(String name) {
    return repository.findByNameOrderById(name);
  }

  @Transactional
  public void save(User user) {
    repository.save(user);
  }
}


UserRepository.java

package com.springTalk.domain;

import java.util.List;

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

@Repository
public interface UserRepository extends JpaRepository<User, Long> {
  List<User> findById(Long id);

  List<User> findByNameOrderById(String name);
}


User.java

package com.springTalk.domain;

import java.util.List;

import javax.persistence.CascadeType;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.OneToMany;
import javax.persistence.Table;

@Entity
@Table(name = "app.users")
public class User {
  @Id
  @GeneratedValue(strategy = GenerationType.AUTO)
  private Long id;

  @Column(nullable = false)
  private String name;

  @Column(nullable = false)
  private String password;

  @OneToMany(mappedBy = "ownerId", fetch = FetchType.LAZY, cascade = CascadeType.ALL)
  private List<Room> rooms;

  @OneToMany(mappedBy = "userId", fetch = FetchType.LAZY, cascade = CascadeType.ALL)
  private List<Membership> memberships;

  @OneToMany(mappedBy = "senderId", fetch = FetchType.LAZY, cascade = CascadeType.ALL)
  private List<Message> messages;

  // JPA requirement
  protected User() {
  }

  public User(String name, String password) {
    this.name = name;
    this.password = password;
  }

  public Long getId() {
    return id;
  }

  public String getName() {
    return name;
  }

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

  public String getPassword() {
    return password;
  }

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


環境
Spring Boot 1.5.9 RELEASE
Oracle VM VirtualBox 5.1.22 r115126 (Qt5.6.2)
CentOS 7.3.1611
openjdk version "1.8.0_141"
Oracle Database 12c Release 2

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 2

+1

RoomsController の roomsPost メソッドにて、

User user = userService.getUserById(OWNER_ID).get(0);


と書かれていますが、ここで1件も取得できないのであれば get(0)で指定しても取得できず IndexOutOfBoundsException がスローされてしまうでしょうか。

userService.getUserById した後に要素があるのかどうかをチェックして値を格納すべきでしょう。
 

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/01/22 00:36

    ありがとうございます。
    ただ、この場合はID=1のレコードが1件返される想定なのですが、これに関してはなぜなのでしょうか?

    キャンセル

  • 2018/01/22 00:45

    実際テーブルにIDが1のレコードが含まれていないか、JPAの設定に何らかの抜けがあるのだと思います。
    ほかにも Userクラスのidに対し、ミューテータ(setメソッド)がありません。

    キャンセル

  • 2018/01/22 23:46

    ありがとうございます。
    解決しました。
    ちなみにidは自動生成なのでsetterはつけていません。

    キャンセル

check解決した方法

0

OracleではなくH2に接続していたようです。
application.ymlのインデントが誤っていました。
また別事象でトラブルに遭っていますが、別途質問したいと思います。
ご覧いただきありがとうございました。

application.yml

spring:
  datasource:
    url: jdbc:oracle:thin:@localhost:1521:chatdb
    username: app
    password: password
    driverClassName: oracle.jdbc.driver.OracleDriver
server:
  port: 80

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

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

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

関連した質問

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

  • Java

    12206questions

    Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

  • Spring Boot

    381questions

    Spring Bootは、Javaのフレームワークの一つ。Springプロジェクトが提供する様々なフレームワークを統合した、アプリケーションを高速で開発するために設計されたフレームワークです。

  • Oracle Database

    52questions

    Oracle Databaseは、米オラクルが開発・販売を行うリレーショナルデータベース管理システムです。