skillup

技術ブログ

Database

論理設計のアンチパターン その2

投稿日:

今回は論理設計のアンチパターンの続きです。

今までに比べると何も意図がないというものではなく、パフォーマンスを考えて設計されているようなものが多いです。

ただし、中には絶対に許されないタイプのものもあるため、注意が必要なようです。

テーブル分割

水平分割

年度 会社コード 売上
2001 c001 2,000
2001 c002 3,000

これは例えば上記のようなテーブルがあったとして、パフォーマンス向上を目的として、レコード数を絞りたいがために年度ごとにテーブルを分割してしまうようなケースです。

意図はありますが、現実として下記のような欠点があります。

分割する意味的な理由がない

テーブルを分割する場合、正規化すべき理由がなければ基本的にテーブルを分割すべきではありません。

拡張性に乏しい

このテーブルの場合、前年度のデータを総なめで検索することがないという前提が常に成り立っています。そしてその前提が常になりたつかどうかは保証できませんし、カラムを追加する場合、複数のテーブルへの追加する必要がでてきます。

他の代替手段がある

こうした水平分割に代わる代替手段としてパーティションという機能があります。

パーティション機能を用いることで、テーブルを分割することなく、特定のキーを軸として物理的に書く領域を分離することが可能になり、SQLがアクセスするレコード数を1/nに減らすことができます。

インデックスとパーティションのどちらを使うかで迷うことがあるようですが、パーティションは一般的にはカーディナリティが小さく、値の変更があまり起きない列をキーにするときに利用します。

典型的な例としては「年」や「都道府県」などカーディナリティが十~数十のものが対象になります。

垂直分割

これは水平分割よりもさらによくありそう。

会社コード 社員ID 名前 年齢 部署コード
c001 s001 山田 23 m001
c001 s002 鈴木 33 m002
c002 k001 渡辺 44 p002

こういったテーブルを 会社コード、社員ID、名前、年齢のテーブルと 会社コード、社員ID、名前、部署コードのテーブル分けることを垂直分割といいます。

これも正規化といった点から分割すべきでないテーブルですね。なんとなく意味的に分けておきたいからといった理由でたまに見かけることがあります。

しかし、これらも下記のような代替手段があることからそれらを使うべきです。

集約

列の絞り込み

本来のテーブルから特定の参照したい列だけを絞ったテーブルを新しいテーブルを作成し、定期的に同期するようにします。このようにして作られるテーブルのことをデータマート、あるいは単にマートと呼びます。

サマリーテーブル

サマリーテーブルも集約の一手段です。列の絞り込みと違い、集計関数を使い、レコードを集約した状態で保持することです。例えば社員の平均年齢を見るときに会社コードごとの平均年齢を作成したテーブルをあらかじめ作っておくなどがこのようなテーブルになります。

ともにもしテーブルを分割したい場合には上記のような技術を使うことが望ましいです。ただし、ともに同期しておかないと整合性がとれなくなることやこれらを作ることでストレージの容量を消費することにもつながります。これらがデメリットです。

不適切なキー

キーに関してはなるべく固定長文字列を使うべき、というのが本来のあり方のようです。

というのもキーはその性質上不変であることが望ましいため、長さが変わる可能性があるものはふさわしくないということですね。

また全く同じ文字列でも固定長文字列と可変長文字列では違った値として判定されます。私が今までやってきた現場ではそこまでの意識がなかったのか。ほぼ可変長文字列でしたね・・・・

ダブルマスタ

これは全く同じ性質のテーブルが2つ(あるいはそれ以上)作られてしまうことです。

0からシステムを構築した場合このようなことはめったに起きないのですが、もともと別のシステムで利用されていたマスタ同士が、システム統廃合によって同じドメインに存在する、となったときにこのようなことが起こってしまうようです。この場合、やはりデータをしっかり精査して、1つのテーブルに収めるのが正しい手法です。

-Database
-

執筆者:


comment

メールアドレスが公開されることはありません。 * が付いている欄は必須項目です

関連記事

no image

SQL問題

今までやったSQL問題などのまとめ。定期的にやる予定です・・ 自分用なのでテーブルデータとかあったりなかったりいい加減です(汗) SQLドリル 問題1 nameとageで構成されたテーブルがあるとして …

no image

slow-query-logについて

データベースを伴う部分でののチューニングですが、大きく分けると SQLを書き直す インデックスを張りなおす プログラム内部でキャッシュを有効化する 設定ファイルの修正 上記のようなかんじになるのではな …

no image

SQL基礎 case&groupbyの活用について

主に集計タイプの計算で大活躍するgroupbyについて。 主な用途は集計とカット。特にcase式と連動した集計はかなり使える 例 nameとageで構成されたテーブルがあるとして、年代ごとの人数を出し …

no image

JPQLでの算術関数

複雑なJPQLを書いていると、通常のレコードの取り出しだけではなく、合計(SUM)や算出(COUNT)などのいわゆる算術関数を使うことが一般的です。 JPQLでもこれらを通常通り扱うことができます。 …

no image

cakeでのトランザクション、コミット、ロールバック

cakePHP(2.X系)でのトランザクション、コミット、ロールバックについて。 cakePHPでトランザクションを書ける場合、Model内に [crayon-5d304e1ac144f6323268 …