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

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

ただいまの
回答率

88.64%

springbootでプルダウンに列挙型(DB登録)

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 1,849

KurisOswad

score 15

 前提・実現したいこと

スプリングブートで在庫一覧表に新規登録画面と更新画面を作成し
formを使ってプルダウンメニューを作成。
そのまま入力値をDBに登録するところまで出来ているのですが
プルダウンメニューに列挙型で定数を設定したいのですが
ネットで色々調べるも理解できる記事が見つからず困っています。
初心者で知識不足なので申し訳ないですが
どなたかご教授いただけないでしょうか。
DBの型がchar型になっていて、enunではStringで作成してます。
こちらは変換なども必要なのでしょうか。

追加の情報です。
A-pzさんのアドバイスに明確に答えられないかもしれません。
以下、私が使っているものです。

・エクリプスのSTSでスタータースプリングブート
・MYSQL
・Marven
・トムキャット
・mysql.jdbc.Driver
・ライブラリがよくわかっていないのですが
JREシステムライブラリというフォルダがプロジェクトにあります。

<?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.funayama.springboot</groupId>
    <artifactId>pda_k_funayama</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>pda_k_funayama</name>
    <description>sample project for Spring Boot</description>

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

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

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

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

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-thymeleaf</artifactId>
        </dependency>
    </dependencies>

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


</project>

コンバーターと列挙型を作成してみましたがうまくいきませんでした。

package com.funayama.springboot;

import javax.persistence.AttributeConverter;

//フォーム用クラス
public class Converter implements AttributeConverter<JenreEnum, String> {
    //データーベースに登録する値(列挙型使用)
    @Override
    public String convertToDatabaseColumn(JenreEnum jenreCode) {
        if (jenreCode == null) {
            return null;
        }
        switch (jenreCode) {
        case Clock:
            return "1";

        case Electron:
            return "2";

        case Phone:
            return "3";

        default:
            return null;

        }
    }

    //Entityに登録する値(列挙型使用)
    @Override
    public JenreEnum convertToEntityAttribute(String dbData) {
        if (dbData == null) {
            return null;
        }
        switch (dbData) {
        case "1":
            return JenreEnum.Clock;
        case "2":
            return JenreEnum.Electron;
        case "3":
            return JenreEnum.Phone;
        default:
            return null;
        }
    }

}
ackage com.funayama.springboot;

import javax.persistence.Convert;

@Convert(converter = Converter.class)
public enum JenreEnum {

    Unspecified("0", "指定なし"),

    Clock("1", "時計"),

    Electron("2", "電子"),

    Phone("3", "携帯");

    //ジャンルID
    private String genreId;

    //ジャンル名
    private String genreName;

    //コンストラクタ
    private JenreEnum(String genreId, String genreName) {
        this.genreId = genreId;
        this.genreId = genreName;
    }

    //商品IDゲッター
    public String getGenreId() {
        return genreId;
    }

    //ジャンル名ゲッター
    public String getGenreName() {
        return genreName;
    }

}
@Convert(converter = Converter.class)
    @Column(name = "product_genre",nullable = false,length = 2)
    private String genre;

EntityをEnumに設定できないとエラーが発生してしまいました。
データーベースの型がcharなのにStringでしてしまっているからなのか
HTMLからの取り出し方に問題があるのか
なにがおかしいのか見当もつきません。
どなたかお力添えをしていただけないでしょうか。。
ちなみにHTMLも載せておきます

<form method="post" action="/ProductList" th:action="@{/ProductList}" th:object="${formList}">
            <div class='form-id'>
                <!--商品ID-->
                <label>※必須※ 商品ID</label> <br> <input type="text"
                    name="id">
            </div>
            <!--ジャンル-->
            <div class='form-jenre'>
                <label>ジャンル</label><br> <select style="width: 160px;"
                    name="genre">
                    <option value='0'></option>
                    <option value='1'>時計</option>
                    <option value='2'>電子</option>
                    <option value='3'>携帯</option>
                </select>
            </div>
  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

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

  • A-pZ

    2018/10/17 16:44

    データベースのアクセスに使っているライブラリを書くと明確な回答が得られるかもしれません。例えばSpring-JDBC、MyBatis、Doma2などがあります。

    キャンセル

  • A-pZ

    2018/10/18 14:15

    なるほど、spring-boot-starter-data-jpa を指定しているので、JPAをお使いで、enumの変換で困っているのですね、ありがとうございます。

    キャンセル

  • KurisOswad

    2018/10/18 20:43

    使ってるもので処理が違うということでしょうか…調べ方を変えてみます!ありがとうございます!

    キャンセル

回答 2

checkベストアンサー

+2

JPAの @Convert 注釈は、enumにつけるのではなく、@Entityで変換したいフィールドに付与します。

例:Genre.java

import javax.persistence.Column;
import javax.persistence.Convert;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.Table;

import lombok.Getter;
import lombok.Setter;
import lombok.ToString;

@Entity
@Table(name="GENRE")
@Getter @Setter @ToString
public class Genre {

    @Id
    @Column(name="ID")
    private Integer id;

    @Column(name="NAME")
    @Convert(converter = GenreAttributeConverter.class)
    private GenreEnum name;
}

ただし、わざわざAttributeConverterを作成せずとも、@Enumerated を付与することでenum型の名前とデータベースの値が合致する要素にもできますし、データベースの値が数値で、その数値の順番どおりにenumの要素が定義されている場合は、自動的に変換が可能です。

変換対象のEnumは以下の通りです。

GenreEnum.java

public enum GenreEnum {
    Other, Clock, Electron, Phone;
}

これに対するGenreクラスは、以下のどちらかを適用します。(両方も可能ですが、それはあまり意味がないでしょう)

enumの要素名と、検索結果を照合させる場合のGenre.java

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.EnumType;
import javax.persistence.Enumerated;
import javax.persistence.Id;
import javax.persistence.Table;

import lombok.Getter;
import lombok.Setter;
import lombok.ToString;

@Entity
@Table(name="GENRE")
@Getter @Setter @ToString
public class Genre {

    @Id
    @Column(name="ID")
    private Integer id;

    @Column(name="NAME")
    @Enumerated(EnumType.STRING)
    private GenreEnum name;
}

enumの要素の順番と、検索結果の数値を照合させる場合のGenre.java

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.Table;

import lombok.Getter;
import lombok.Setter;
import lombok.ToString;

@Entity
@Table(name="GENRE")
@Getter @Setter @ToString
public class Genre {

    @Id
    @Column(name="ID")
    private GenreEnum id;

    @Column(name="NAME")
    private String name;
}

この場合は、@Idによって、ordinary(順序指定=数値)が暗黙的に指定され、enumの場合は自動的に ordinal() が実行された結果が入ります。

例えばデータベースから取得した値が 2 だった場合は、ordinal値が2のもの(ordinailは0から開始される数値)なので、GenreEnum.Electron が返されます。


参考までに、Repositoryは以下のインタフェースで作成し、

import org.springframework.data.repository.CrudRepository;

import com.github.apz.springsample.entity.Genre;

public interface GenreRepository extends CrudRepository<Genre, Integer> {

}

@Controller@Serviceの責務がついたクラスなどからは以下で呼び出せるでしょう。

@RestController
@RequiredArgsConstructor
public class GenreController {

    private final GenreRepository repository;

    @GetMapping("/genre")
    public Iterable<Genre> getAllGenre() {
        return repository.findAll();
    }
}

テーブル定義

CREATE TABLE GENRE (
    ID INT NOT NULL,
    NAME VARCHAR(256) NOT NULL
);

INSERT INTO GENRE (ID, NAME) VALUES (1, 'Clock');
INSERT INTO GENRE (ID, NAME) VALUES (2, 'Electron');
INSERT INTO GENRE (ID, NAME) VALUES (3, 'Phone');

余談:提示していただいたコードに、Jenre と Genre が入り混じっていますので、Genreに統一した方が良いかと思います。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/10/21 23:13

    返事が遅くなり申し訳ありません
    詳しく教えていただいてありがとうございます!
    コンバーターを作らずとも出来るのですね。。
    色々ねっと情報を見ていたのですが今までで一番わかりやすかったです!
    早速参考にさしていただきます。
    大変助かりました。
    また機会があればよろしくお願いします!
    ありがとうございました!

    キャンセル

  • 2018/10/22 13:51

    d(・ω・

    Converterを作成する理由は、回答のような「デフォルトの振る舞い」ではなく、それ以外の動きをしてほしい場合に指定します。たいていは既存システムの都合やDBの設計都合などによりデフォルトの振る舞いでは条件を満たせない場合があるのと、より複雑な構造を要する場合もあるため、Converterを使うことはよくあります。

    キャンセル

  • 2018/10/22 22:48

    なるほどですね。Converterについても気になっていたところでした。使う機会も多いということですね。どちらのパターンでも使えるように試してみたいと思います。知識が少なすぎて質問の仕方ですらよくわかっていなかったのですが、丁寧にお答え下さってありがとうございます。まだまだ理解しきれないところが多いのですが、教えてもらえたことしっかりプラスにしていきたいと思います!!

    キャンセル

0

ormによって違うので一概には言えません。

設定(ファイル/アノテーション)で対応可能なもの。(無設定で可能な場合はこちら)

コンバーターを作成して設定で対応可能なもの

が一般的になります。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/10/17 22:36

    asahina1979さん
    アドバイスをありがとうございます。
    今日これからですが、コンバーターで調べ再チャレンジしてみようかと思います。
    昨日まで途方にくれていましたが、今日は少しでも進められるように頑張ってみます!!
    助かります!

    キャンセル

  • 2018/10/17 22:51

    public interface AttributeConverter<X,Y> {
    public Y convertToDatabaseColumn (X attribute);
    public X convertToEntityAttribute (Y dbData);
    }

    上記のものでうまくいくか作成してみようかと思います。

    キャンセル

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

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

関連した質問

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