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

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

新規登録して質問してみよう
ただいま回答率
85.48%
SQL

SQL(Structured Query Language)は、リレーショナルデータベース管理システム (RDBMS)のデータベース言語です。大きく分けて、データ定義言語(DDL)、データ操作言語(DML)、データ制御言語(DCL)の3つで構成されており、プログラム上でSQL文を生成して、RDBMSに命令を出し、RDBに必要なデータを格納できます。また、格納したデータを引き出すことも可能です。

PHP

PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

デザインパターン

デザインパターンは、ソフトウェアのデザインでよく起きる問題に対して、解決策をノウハウとして蓄積し再利用出来るようにした設計パターンを指します。

Q&A

解決済

1回答

3707閲覧

リポジトリパターンのクラス設計について

kgtkr

総合スコア49

SQL

SQL(Structured Query Language)は、リレーショナルデータベース管理システム (RDBMS)のデータベース言語です。大きく分けて、データ定義言語(DDL)、データ操作言語(DML)、データ制御言語(DCL)の3つで構成されており、プログラム上でSQL文を生成して、RDBMSに命令を出し、RDBに必要なデータを格納できます。また、格納したデータを引き出すことも可能です。

PHP

PHPは、Webサイト構築に特化して開発されたプログラミング言語です。大きな特徴のひとつは、HTMLに直接プログラムを埋め込むことができるという点です。PHPを用いることで、HTMLを動的コンテンツとして出力できます。HTMLがそのままブラウザに表示されるのに対し、PHPプログラムはサーバ側で実行された結果がブラウザに表示されるため、PHPスクリプトは「サーバサイドスクリプト」と呼ばれています。

デザインパターン

デザインパターンは、ソフトウェアのデザインでよく起きる問題に対して、解決策をノウハウとして蓄積し再利用出来るようにした設計パターンを指します。

0グッド

0クリップ

投稿2016/07/08 14:19

編集2016/07/08 23:55

###前提・実現したいこと
MySQLであるWEBサービスを作っています。
今までは処理も永続化も一つのクラスに書いていたのですが、リポジトリパターンで分けるといいみたいなので分割しました。
しかし、クラス設計で悩んでいます。
以下のような場合、どのようにクラス設計を行うのが正しいのでしょうか?

ここでは、重複禁止のメールアドレスカラム(mail)があるとします。

###分割前のソースコード

PHP

1/** 2 * メールアドレスを変更します 3 * @param mysqli $sql メールアドレス 4 * @param string $mail メールアドレス 5 * 6 * @throws Exception 既に登録されている 7 */ 8public function setMail($sql,$mail){ 9 //引数チェック省略 10 11 //DBのhogeテーブルのmailカラムの更新 12 $stmt = $sql->prepare("UPDATE hoge SET mail=?"); 13 $stmt->bind_param("s", $mail); 14 $stmt->execute(); 15 16 //もし1062エラー(重複データの格納)なら例外を投げる 17 if ($sql->errno === 1062) { 18 throw new Exception("このメールアドレスは登録済みです。"); 19 } 20 21 //重複でなければセット 22 $this->mail=$mail; 23}

###困っていること
このクラスをリポジトリパターンで分割する場合、重複チェック(1062エラーチェック)は、どう行うのが正しいのか?
今考えているのは、setMall()関数では重複チェックを行わずに、引数チェックと、フィールドへの代入のみを行い、Repositoryクラスの永続化関数で重複チェックを行う事です。

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

Chironian

2016/07/09 07:59

回答依頼を頂きましたが、PHP, SQL, リボジトリパターンどれも知らないので回答できません。ごめんなさい。
kgtkr

2016/07/09 08:04

こちらこそいきなり依頼してすみません
guest

回答1

0

ベストアンサー

あくまでも私の考えになりますが、重複チェックはサービス側でやるべきです。(実際に取ってくるのはデータアクセス層ですが)
また、重複チェックとUPDATEを同時にすると2つの処理をやることになってしまうので、基本的には分けるべきでしょう。(もしやる場合はDataAccess層からExceptionを投げる必要があるでしょう)

HogeService

php

1class HogeService 2{ 3 protected $hogeRepository; 4 5 public function __construct(HogeRepositoryInterface $hogeRepository) 6 { 7 $this->hogeRepository = $hogeRepository; 8 } 9 10 public function existsEmail($email) 11 { 12 $this->hogeRepository->exitsEmail($email); 13 } 14 15 public function updateEmail($email) 16 省略 17}

HogeRepository

class HogeRepository implements HogeRepositoryInterface { protected $hoge; public function __construct(Hoge $hoge) { $this->hoge = $hoge; } public function existsEmail($email) { // もしbool値を返さないようなメソッドを使う場合はここで処理して返す $this->hoge->existsEmail($email); } // 省略 }

HogeRepositoryInterface
省略します。

Hoge(データアクセス)

class Hoge { // ここではあえてEmailに限定させましたが、カラムも別途していできるようにしても良いと思います public function existsEmail($email) { $stmt = $sql->prepare("SELECT count(*) FROM hoge WHERE mail = :mail"); $stmt->bind_param("mail", $mail); $stmt->execute(); return !!$stmt->fetchColumn(); } // 省略 }

メインコード

// クラスで管理している場合はメンバ変数に入れておきましょう。 // より良いのはDIコンテナなどを利用してRepositoryはDIするのが良いでしょう。 $hogeService = new HogeService(new HogeRepository); if ($hogeService->exitsEmail($email)) { // メールアドレス存在してるのでエラーを返す } // ここはtry-catchで囲っておいても良いかもしれません $hogeService->updateEmail($email);

って今気づきましたが1年越しの回答でした

投稿2017/07/21 11:20

編集2017/07/21 11:48
fagai

総合スコア2158

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問