ページの作成
親となるページを選択してください。
親ページに紐づくページを子ページといいます。
例: 親=スポーツ, 子1=サッカー, 子2=野球
子ページを親ページとして更に子ページを作成することも可能です。
例: 親=サッカー, 子=サッカーのルール
親ページはいつでも変更することが可能なのでとりあえず作ってみましょう!
| この記事の要点 |
|---|
|
@Entity の基本
import jakarta.persistence.*; // Jakarta EE 9+ (Spring Boot 3+)
// または javax.persistence.* (Jakarta EE 8 以前)
@Entity
@Table(name = "users")
public class User {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@Column(name = "user_name", nullable = false, length = 100)
private String name;
@Column(unique = true)
private String email;
@Column(name = "created_at", updatable = false)
private LocalDateTime createdAt;
// 引数なしコンストラクタ (必須)
public User() {}
// ビジネス用コンストラクタ
public User(String name, String email) {
this.name = name;
this.email = email;
}
// getter / setter (省略可: Lombok @Data)
}
必須要件
- クラスに @Entity を付与
- 主キーが必要: 少なくとも 1 つの
@Id付きフィールド - 引数なしコンストラクタ: Hibernate がリフレクションで生成するため
- final クラスにしない: プロキシ生成不可
- Serializable 実装推奨(必須ではないが、キャッシュ・分散環境で便利)
テーブル名のマッピング
// クラス名から自動推測 (snake_case)
@Entity
public class UserProfile { }
// → DB テーブル "user_profile"
// @Table で明示
@Entity
@Table(name = "user_profiles")
public class UserProfile { }
// → DB テーブル "user_profiles"
// スキーマ指定
@Entity
@Table(name = "users", schema = "auth")
public class User { }
// → "auth.users"
// インデックス・UNIQUE 制約も定義
@Entity
@Table(name = "users",
indexes = {
@Index(name = "idx_users_email", columnList = "email", unique = true),
@Index(name = "idx_users_status", columnList = "status")
},
uniqueConstraints = {
@UniqueConstraint(columnNames = {"tenant_id", "username"})
}
)
public class User { }
主キーの定義
// 単一主キー
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
// UUID 主キー (Hibernate 6+)
@Id
@GeneratedValue(strategy = GenerationType.UUID)
private UUID id;
// 複合主キー (@EmbeddedId)
@Embeddable
public class OrderItemId implements Serializable {
private Long orderId;
private Long productId;
// equals/hashCode 必須
}
@Entity
public class OrderItem {
@EmbeddedId
private OrderItemId id;
}
カラムマッピング
@Entity
public class User {
@Id @GeneratedValue
private Long id;
// デフォルト: フィールド名 = カラム名 (snake_case 変換あり)
private String name; // → name カラム
private String firstName; // → first_name カラム (Spring Boot 標準)
// カラム名明示
@Column(name = "email_address", length = 255, nullable = false, unique = true)
private String email;
// 不変 (INSERT/UPDATE 含めない)
@Column(insertable = false, updatable = false)
private LocalDateTime serverGenerated;
// BigDecimal (金額)
@Column(precision = 10, scale = 2)
private BigDecimal price;
// LOB (Large Object: TEXT, BLOB)
@Lob
@Column(columnDefinition = "TEXT")
private String bio;
@Lob
private byte[] image;
// Enum
@Enumerated(EnumType.STRING)
private UserStatus status; // 文字列で保存
// 日時
@Column(name = "created_at")
@CreationTimestamp
private LocalDateTime createdAt;
@Column(name = "updated_at")
@UpdateTimestamp
private LocalDateTime updatedAt;
// 一時的フィールド (DB に保存しない)
@Transient
private String displayName; // 計算用 / メモ用
// バージョン (楽観ロック)
@Version
private Long version;
}
関連 (リレーションシップ)
@OneToMany / @ManyToOne
// 1:N の親
@Entity
public class User {
@Id @GeneratedValue
private Long id;
@OneToMany(mappedBy = "user", cascade = CascadeType.ALL, orphanRemoval = true)
private List orders = new ArrayList<>();
}
// N:1 の子
@Entity
public class Order {
@Id @GeneratedValue
private Long id;
@ManyToOne(fetch = FetchType.LAZY) // ← LAZY 推奨
@JoinColumn(name = "user_id", nullable = false)
private User user;
}
@OneToOne
@Entity
public class User {
@Id @GeneratedValue
private Long id;
@OneToOne(mappedBy = "user", cascade = CascadeType.ALL)
private UserProfile profile;
}
@Entity
public class UserProfile {
@Id
private Long id;
@OneToOne(fetch = FetchType.LAZY)
@MapsId
@JoinColumn(name = "id")
private User user;
private String bio;
}
@ManyToMany(中間テーブル)
@Entity
public class User {
@Id @GeneratedValue
private Long id;
@ManyToMany
@JoinTable(name = "user_roles",
joinColumns = @JoinColumn(name = "user_id"),
inverseJoinColumns = @JoinColumn(name = "role_id"))
private Set roles = new HashSet<>();
}
@Entity
public class Role {
@Id @GeneratedValue
private Long id;
private String name;
@ManyToMany(mappedBy = "roles")
private Set users = new HashSet<>();
}
// → user_roles 中間テーブルが自動生成される
継承戦略
// 戦略 1: 単一テーブル (デフォルト)
@Entity
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = "type")
public abstract class Animal {
@Id @GeneratedValue
private Long id;
private String name;
}
@Entity
@DiscriminatorValue("DOG")
public class Dog extends Animal { }
@Entity
@DiscriminatorValue("CAT")
public class Cat extends Animal { }
// → 1 つの animal テーブルに type カラムで区別
// 戦略 2: テーブル分割 (JOINED)
@Inheritance(strategy = InheritanceType.JOINED)
// → animal + dog + cat テーブル
// 戦略 3: クラスごとテーブル (TABLE_PER_CLASS)
@Inheritance(strategy = InheritanceType.TABLE_PER_CLASS)
ライフサイクルイベント
@Entity
@EntityListeners(AuditListener.class) // 別クラスでリスナー
public class User {
@PrePersist
public void onCreate() {
this.createdAt = LocalDateTime.now();
}
@PreUpdate
public void onUpdate() {
this.updatedAt = LocalDateTime.now();
}
@PostLoad
public void afterLoad() {
// DB から読み込み後
}
@PreRemove
public void beforeDelete() {
// 削除前
}
}
Lombok との併用
@Entity
@Table(name = "users")
@Getter @Setter
@NoArgsConstructor
@AllArgsConstructor
@Builder
@EqualsAndHashCode(of = "id") // id だけで equals
@ToString(exclude = "orders") // orders 除外 (LAZY のため)
public class User {
@Id @GeneratedValue
private Long id;
@Column(nullable = false)
private String name;
@OneToMany(mappedBy = "user")
private List orders;
}
// 利用
User user = User.builder()
.name("Alice")
.build();
注意点
- 引数なしコンストラクタ必須:
@NoArgsConstructorまたは手動 - FetchType.LAZY 推奨: デフォルトの EAGER は N+1 問題の元
- equals/hashCode: id だけで判定(オブジェクトの内容で比較するとリレーション読込で無限ループ)
- toString で関連を含めない: LAZY 読込時に余分なクエリ発行
- FetchType.EAGER は慎重に: 大量データで爆発
関連記事
ページの作成
親となるページを選択してください。
親ページに紐づくページを子ページといいます。
例: 親=スポーツ, 子1=サッカー, 子2=野球
子ページを親ページとして更に子ページを作成することも可能です。
例: 親=サッカー, 子=サッカーのルール
親ページはいつでも変更することが可能なのでとりあえず作ってみましょう!
子ページ
子ページはありません
人気ページ
- 1 Eclipseで「サーバーに追加または除去できるリソースがありません。」の原因と対処法
- 2 tomcat の起動 / 停止ログと catalina.log・catalina.out の違い
- 3 JavaScript base URL 取得方法|window.location.origin と SSR/Node.js 対応
- 4 YouTube Data API v3 エラー一覧|403/400/404 の主要原因と切り分け
- 5 Spring Frameworkのアノテーション一覧
- 6 Laravel エラー一覧|500/Blade/DB 接続/ルーティングの代表エラー
- 7 3Dグラフィックスとは|モデリング/レンダリング/主要ソフトウェア (Blender / Maya)
- 8 【Spring】@Valueアノテーションとは
- 9 CATALINA_HOME の確認方法 (Linux / Mac)
- 10 【Spring】@Autowiredアノテーションとは
最近更新/作成されたページ
- Laravel キャッシュクリア完全ガイド(cache:clear / config:clear / 2026-05-18 07:42:07
- プロジェクトの作成と削除 2026-05-18 07:42:07
- インストール直後にNetbeansが反応しない 2026-05-18 07:42:07
- 動画やチャンネルの検索 2026-05-18 07:42:07
- APIキー取得方法 2026-05-18 07:42:07
- チャンネル情報の取得 2026-05-18 07:42:07
- API 入門 — Web API(REST / GraphQL / gRPC / 2026-05-18 07:42:07
- インストール(eclipseプラグイン) 2026-05-18 07:42:07
- Laravel「Dotenv values containing spaces must be surrounded 2026-05-18 07:42:07
- エラー一覧 2026-05-18 07:42:07
- curl: (51) SSL: certificate subject name '~' does not match 2026-05-18 07:42:07
- インストール方法(Windows版) 2026-05-18 07:42:07
- JSONから配列に変換 2026-05-18 07:42:07
- 処理を一定時間待つ 2026-05-18 07:42:07
- A non well formed numeric value encountered 2026-05-18 07:42:07
コメントを削除してもよろしいでしょうか?