Difference between revisions of "MasterDetail/ja"
Line 67: | Line 67: | ||
---- | ---- | ||
− | == ''' | + | == '''国と都市の例''' == |
− | + | このチュートリアルは SQLite を使用して作成されたが、原則は他のデータベースでも同じである。 | |
− | + | もしSQLiteの管理プログラムを持っていない場合、SQLite Studio http://sqlitestudio.pl を推奨する。 | |
− | * | + | *初めに新しく SQLite データベースを作り、database.db3として保存する。 |
− | ( | + | (別のデータベース システムを使用する場合は、慣れているようにテーブルを作成し、例に適切なコンポーネントを使用するだけだ)。 |
− | ''' | + | '''ここで、2つのテーブルを作る''' |
− | + | 「国」テーブル: | |
<syntaxhighlight lang="SQL"> | <syntaxhighlight lang="SQL"> | ||
Line 87: | Line 87: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
− | ''' | + | '''テーブルに2つの国を追加する:''' |
*Sweden | *Sweden | ||
*Norway | *Norway | ||
− | Because COUNTRY_ID | + | Because COUNTRY_ID が自動的に加算されるので、それぞれの国は自動的にIDがふられる。 |
− | |||
− | ''' | + | '''「都市」テーブルを作る:''' |
<syntaxhighlight lang="SQL"> | <syntaxhighlight lang="SQL"> | ||
CITY_ID (Primary key, integer, autoincrement) | CITY_ID (Primary key, integer, autoincrement) | ||
Line 101: | Line 100: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
− | ''' | + | '''3つの都市をこのテーブルに追加する:''' |
− | *Stockholm | + | * Stockholm COUNTRY_ID 1 |
− | *Gothenburg | + | * Gothenburg COUNTRY_ID 1 |
− | *Oslo | + | * Oslo COUNTRY_ID 2 |
− | + | 空のテーブルに最初にスウェーデンが追加されたため、スウェーデンは COUNTRY_ID 1 を持ち、ノルウェーは COUNTRY_ID 2 を持つ。追加する都市ごとに、COUNTRY_ID を指定する必要がある。COUNTRY_ID によってテーブルがリンクされ、マスター/詳細の概念が可能になるためである。 | |
+ | * Lazarus で新しいプロジェクトを作成し、MasterDetail として保存する。 | ||
+ | * TSQLite3Connection をフォームに配置する(SQLdb tab)。 | ||
+ | * TSQLite3Connection の DatabaseName プロパティに、作成したばかりの SQLite データベース ファイルのフル パスとファイル名を入力する。 | ||
+ | * TSQLTransaction をフォームに配置する(SQLdb tab)。 | ||
+ | * そのデータベース プロパティを: SQLite3Connection1へ。 | ||
+ | * TSQLite3Connectionに戻り、TransactionプロパティをSQLTransaction1に設定する。 | ||
+ | * TSQLite3Connection.Connected を true にする。 | ||
− | + | '''これでデータベースに接続できるようになり、次のステップに進むことができます。''' | |
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | + | * TSQLQueryをフォームに配置(SQLdb tab)。 | |
− | + | * そのデータベースプロパティを: SQLite3Connection1にする | |
− | * | + | * Transaction プロパティを: SQLTransaction1にする。 |
− | * | + | * SQLステートメントに、以下を入力する: |
− | * | ||
− | * | ||
<syntaxhighlight lang="SQL"> | <syntaxhighlight lang="SQL"> | ||
select * from countries | select * from countries | ||
</syntaxhighlight> | </syntaxhighlight> | ||
− | * | + | * TDataSourceをフォームに配置(Data Access tab)。 |
− | * | + | * このデータセットを: SQLQuery1にする。 |
− | * | + | * TDBGridをフォームに配置(Data Controls tab)。 |
− | * | + | * そのDataSourceプロパティを: DataSource1へ設定。 |
− | * | + | * SQLQuery1 に戻り、Active を trueにする。 |
− | ''' | + | '''国テーブル (マスター) の内容が DBGrid1 に表示されるようになる。''' |
− | * | + | *次に都市テーブルに進みます(ディテール) |
− | * | + | *2つ目のTSQLQueryをフォームに配置(SQLdb tab)。 |
− | * | + | *そのデータベースプロパティを: SQLite3Connection1へ設定する。 |
− | * | + | *そのTransaction プロパティを: SQLTransaction1へ設定する。 |
− | * | + | *SQL ステートメントに、以下を入力する: |
<syntaxhighlight lang=pascal> | <syntaxhighlight lang=pascal> | ||
Line 147: | Line 145: | ||
where cities.COUNTRY_ID = :COUNTRY_ID | where cities.COUNTRY_ID = :COUNTRY_ID | ||
</syntaxhighlight> | </syntaxhighlight> | ||
− | + | :COUNTRY_ID のコロン記号は、これが変数パラメーターであり、値が他の場所から取得されることを意味する。それはマスター データソースから値を取得する。したがって、この例では、:COUNTRY_ID 値はマスター データソースである DataSource1 によって提供される(パラメータ名は任意ではない。マスターテーブル内の関連するフィールド名である)。 | |
+ | *そのDatasourceプロパティをDatasource1 | ||
− | * | + | これではまだ都市を追加できない、これをする: |
+ | * 上記の #正しい外部キーを使用した詳細レコードの追加 にある情報を使用のこと。 | ||
− | |||
− | |||
+ | '''ディテールビューを続行する:''' | ||
− | + | *2つ目のTDataSourceをフォームに配置。 | |
+ | *そのDataSetプロパティをQuery2 (ディテール クエリ)にする。 | ||
+ | *2つ目のDBGridをフォームに配置(ディテール グリッド)にする。 | ||
+ | *そのDataSourceを DataSource2 (ディテール ソース)にする。 | ||
− | |||
− | |||
− | |||
− | |||
− | + | '''次に、グリッドにデータを流し込む。 これらをすべてアクティブ/接続済みに設定する:''' | |
− | ''' | ||
*DBConnection | *DBConnection | ||
Line 170: | Line 167: | ||
*Query2 | *Query2 | ||
+ | これで、DBGrid2にディテールデータがあるはずだ。 | ||
− | + | 覚えておくべきことは、(上記のように) 詳細クエリのデータソースをマスター データソースに設定する必要があるということである (この例では DataSource1)。 | |
− | |||
− | |||
− | |||
− | |||
− | |||
− | ''' | + | '''それでも理解できなかった場合は、次の連結スキームを見ること:''' |
Master Query.DataSource := None | Master Query.DataSource := None |
Revision as of 11:06, 31 March 2024
概要
リレーショナル データベースでは、マスター/ディテールの概念は 1 対多または 1:N とも呼ばれる。これは、テーブル/データセット内の 1 つのレコードが、別のテーブル/データセット内の多数のレコードに関連付けられる可能性があることを意味する。これは、あるデータセットのデータを使用して別のデータセットのデータをフィルター処理するモデルである。データのフィルタリングに使用されるデータセットはマスターと呼ばれ、フィルタリングされたデータセットはディテールと呼ばれる。たとえば、国には多くの都市があります。 国を選択し、そのすべての都市を表示したい場合は、マスター/ディテールに適した仕事である。国テーブルをマスター、都市テーブルをディテールにすることができる。フィルタリングが機能するのは、ディテール表にマスター表の識別子 (通常はマスター表の主キー) が含まれているためである。 したがって、このキーはマスターテーブルとディテールテーブルの両方に存在する。 詳細テーブルでは外部キーと呼ばれる。
マスター/ディテール関係の典型的な例は次のとおりである:
- 顧客と注文
- 注文した商品を含む注文
以下の例では、customer テーブルと order テーブルを操作する。
SQLDB 実装
詳細データセットの datasource プロパティを使用して、マスター データセットにリンクできる。
以下のアプローチは、組み込みの SQLdb Package データベース層を使用する Lazarus と FPC コードの両方に適用される。 他のデータベース層 (Zeo など) にも適用される場合がある。 ドキュメントを確認されたい。
EMPLOYEE データベースの例 (SQLdb_Tutorial0/ja および他のチュートリアルで使用されているもの):
- 整数の主キー CUST_NO とその他のフィールドを持つ CUSTOMER テーブル
- CUSTOMER テーブルの CUST_NO フィールドにリンクする外部キーである CUST_NO 整数フィールドを持つ SALES テーブル
フォーム上で:
- 1 つの接続、1 つのトランザクションを使用するが、2 つのクエリ、2 つのデータ ソースを使用する
- qryCustomers というマスター クエリは CUSTOMER テーブルから選択する
- SALES テーブルから選択する qrySales というディテールクエリ
ディテールクエリqrySalesで:
- 通常通り、データベースプロパティを設定する
- データベースプロパティをマスターデータソースを示すように設定する
- クエリ SQL では、WHERE クエリを使用して選択を制限する。 詳細フィールド名を使用し、マスター テーブル内のフィールドの名前を持つパラメーターに制限する。
SELECT * from SALES WHERE SALES.CUST_NO=:CUST_NO
この場合、たまたまマスター フィールド CUST_NO と同じフィールド名を持つ SALES.CUST_NO フィールドを使用するが、そうである必要はない。マスター データソースはマスター クエリ qryCustomers の現在のレコードを追跡するため、FPC はマスター/qryCustomers クエリの CUST_NO フィールドの現在の値への参照として CUST_NO パラメーターを確認できる。
マスター クエリにリンク すべきではない 追加のパラメーターをクエリで使用する場合は、マスター データセットを開く前に、その Bound
プロパティが true であることを確認すること。doc:fcl/db/tparam.bound.htmlを参照のこと。
フィールドを検索できるように、ディテールクエリの前にマスター クエリが開いていることを確認すること。
適切な外部キーを使用したディテールレコードの追加
ディテールレコードはマスター レコードとともにスクロールしますが、追加のコードが必要である。
新しいディテールレコードを追加するとき、SALES.CUST_NO フィールドは、値を入力しない限り NULL のままである。したがって、qrySales の AfterInsert イベント ハンドラーを設定する必要がある:
procedure TForm1.qrySalesAfterInsert(DataSet: TDataSet);
begin
DataSet.FieldByName('CUST_NO').AsInteger := qryCustomers.FieldByName('CUST_NO').AsInteger;
end;
マスターデータを使用してフィルタリングしないディテールパラメータの使用
デフォルトでは、詳細データセット内のすべてのパラメーターの値はマスター データセットによって提供される。 詳細データセットに個別にフィルター処理する別のパラメーターがあるとする:
Master Query: Same as above
Detail query:
SELECT * FROM SALES WHERE SALES.CUST_NO=:CUST_NO AND OVERDUE=:CREDITOVERDUE
解決策は、次のいずれかの方法を使用して、SQLDB に CREDITOVERDUE パラメータを「バインド」するように指示することである:
- オブジェクトインスペクタでパラメータの値を設定する。
- マスター データセットを開く前に、パラメーターを (値に) バインド済みとしてマークする:
qryDetail.Params.ParamByName('CREDITOVERDUE').Bound:=True;
Bound property documentation参照。 - マスター データセットを開く前にパラメーター値を設定する:
qryDetail.Params.ParamByName('CREDITOVERDUE').AsBoolean:=true;
ディテール ブックマーク
注意: TBufDataset および bufdataset の派生 (SQLQuery など)、マスター データセットのアクティブ レコードが移動するたびに詳細データセットが再ロードされる; したがって、既存のディテール ブックマークは、たとえ有効であっても、移動後は同じディテール レコードを指さなくなる。
国と都市の例
このチュートリアルは SQLite を使用して作成されたが、原則は他のデータベースでも同じである。
もしSQLiteの管理プログラムを持っていない場合、SQLite Studio http://sqlitestudio.pl を推奨する。
- 初めに新しく SQLite データベースを作り、database.db3として保存する。
(別のデータベース システムを使用する場合は、慣れているようにテーブルを作成し、例に適切なコンポーネントを使用するだけだ)。
ここで、2つのテーブルを作る
「国」テーブル:
COUNTRY_ID (Primary key, integer, autoincrement)
COUNTRY_NAME (VARCHAR(50), not null)
テーブルに2つの国を追加する:
- Sweden
- Norway
Because COUNTRY_ID が自動的に加算されるので、それぞれの国は自動的にIDがふられる。
「都市」テーブルを作る:
CITY_ID (Primary key, integer, autoincrement)
CITY_NAME (VARCHAR(50), not null)
COUNTRY_ID (integer, not null) (foreign key)
3つの都市をこのテーブルに追加する:
- Stockholm COUNTRY_ID 1
- Gothenburg COUNTRY_ID 1
- Oslo COUNTRY_ID 2
空のテーブルに最初にスウェーデンが追加されたため、スウェーデンは COUNTRY_ID 1 を持ち、ノルウェーは COUNTRY_ID 2 を持つ。追加する都市ごとに、COUNTRY_ID を指定する必要がある。COUNTRY_ID によってテーブルがリンクされ、マスター/詳細の概念が可能になるためである。
- Lazarus で新しいプロジェクトを作成し、MasterDetail として保存する。
- TSQLite3Connection をフォームに配置する(SQLdb tab)。
- TSQLite3Connection の DatabaseName プロパティに、作成したばかりの SQLite データベース ファイルのフル パスとファイル名を入力する。
- TSQLTransaction をフォームに配置する(SQLdb tab)。
- そのデータベース プロパティを: SQLite3Connection1へ。
- TSQLite3Connectionに戻り、TransactionプロパティをSQLTransaction1に設定する。
- TSQLite3Connection.Connected を true にする。
これでデータベースに接続できるようになり、次のステップに進むことができます。
- TSQLQueryをフォームに配置(SQLdb tab)。
- そのデータベースプロパティを: SQLite3Connection1にする
- Transaction プロパティを: SQLTransaction1にする。
- SQLステートメントに、以下を入力する:
select * from countries
- TDataSourceをフォームに配置(Data Access tab)。
- このデータセットを: SQLQuery1にする。
- TDBGridをフォームに配置(Data Controls tab)。
- そのDataSourceプロパティを: DataSource1へ設定。
- SQLQuery1 に戻り、Active を trueにする。
国テーブル (マスター) の内容が DBGrid1 に表示されるようになる。
- 次に都市テーブルに進みます(ディテール)
- 2つ目のTSQLQueryをフォームに配置(SQLdb tab)。
- そのデータベースプロパティを: SQLite3Connection1へ設定する。
- そのTransaction プロパティを: SQLTransaction1へ設定する。
- SQL ステートメントに、以下を入力する:
select *
from cities
where cities.COUNTRY_ID = :COUNTRY_ID
- COUNTRY_ID のコロン記号は、これが変数パラメーターであり、値が他の場所から取得されることを意味する。それはマスター データソースから値を取得する。したがって、この例では、:COUNTRY_ID 値はマスター データソースである DataSource1 によって提供される(パラメータ名は任意ではない。マスターテーブル内の関連するフィールド名である)。
- そのDatasourceプロパティをDatasource1
これではまだ都市を追加できない、これをする:
- 上記の #正しい外部キーを使用した詳細レコードの追加 にある情報を使用のこと。
ディテールビューを続行する:
- 2つ目のTDataSourceをフォームに配置。
- そのDataSetプロパティをQuery2 (ディテール クエリ)にする。
- 2つ目のDBGridをフォームに配置(ディテール グリッド)にする。
- そのDataSourceを DataSource2 (ディテール ソース)にする。
次に、グリッドにデータを流し込む。 これらをすべてアクティブ/接続済みに設定する:
- DBConnection
- Transaction
- Query1
- Query2
これで、DBGrid2にディテールデータがあるはずだ。
覚えておくべきことは、(上記のように) 詳細クエリのデータソースをマスター データソースに設定する必要があるということである (この例では DataSource1)。
それでも理解できなかった場合は、次の連結スキームを見ること:
Master Query.DataSource := None
Master DataSource.Dataset := Master Query
Master DBGrid.DataSource := Master DataSource
Detail Query.DataSource := Master DataSource
Detail DataSource.Dataset := Detail Query
Detail DBGrid.DataSource := Detail DataSource
Delete all Detail records if Master record is deleted
This is easy to do with Cascading Deletes. Once again, this is for SQLite, but other systems work similarly.
- First turn on foreign keys for SQLite because it's off by default. Do this by adding FOREIGN_KEYS=ON to the Params property of TSQLite3Connection.
- Then make sure the detail table is created so the foreign key (country_id) references the master table. In this case it is the cities table:
try
{ Master table }
Conn.ExecuteDirect('CREATE TABLE IF NOT EXISTS countries (' +
' country_id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, ' +
' country_name VARCHAR(30) ' +
')');
{ Detail table }
Conn.ExecuteDirect('CREATE TABLE IF NOT EXISTS cities (' +
' city_id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, ' +
' city_name VARCHAR(30), ' +
' country_id INTEGER REFERENCES countries ON DELETE CASCADE' +
')');
{ Sample data }
Conn.ExecuteDirect('INSERT INTO countries(country_name) VALUES(''Sweden'')');
Conn.ExecuteDirect('INSERT INTO cities(city_name, country_id) VALUES(''Stockholm'', 1)');
Conn.ExecuteDirect('INSERT INTO cities(city_name, country_id) VALUES(''Malmo'', 1)');
Conn.ExecuteDirect('INSERT INTO countries(country_name) VALUES(''Norway'')');
Conn.ExecuteDirect('INSERT INTO cities(city_name, country_id) VALUES(''Oslo'', 2)');
Conn.ExecuteDirect('INSERT INTO cities(city_name, country_id) VALUES(''Bergen'', 2)');
Conn.Transaction.Commit;
Conn.Close;
except
on E:Exception do
begin
TX.Rollback;
ShowMessage( E.Message );
end;
end;
Thats it. From now on, all detail records are deleted automatically whenever the master record is deleted.
Note: DON'T try to turn on foreign keys this way:
Conn.ExecuteDirect('PRAGMA foreign_keys = ON');
TSQLite3Connection starts a transaction before the first query is sent to the database, even in case of a ExecuteDirect. It is not possible to enable or disable foreign key constraints in the middle of a multi-statement transaction (when SQLite is not in autocommit mode). Attempting to do so does not return an error; it simply has no effect. Source: Ludob/SQLite documentation.
One more example
There is a complete running example with source code on this forum topic:
How to Append a new line and save it in DBGrid?
http://forum.lazarus.freepascal.org/index.php/topic,42088.msg293305.html#msg293305
See also
- http://www.freepascal.org/docs-html/fcl/sqldb/usingparams.html Documentation on using parameters, including implementation for master/detail
- http://docwiki.embarcadero.com/RADStudio/XE3/en/Establishing_Master-detail_Relationships_Using_Parameters The documentation for the equivalent Delphi feature
- http://forum.codecall.net/topic/62683-master-detail-datasets/
- Lazarus Tdbf Tutorial#Master table relations Example of using master/detail with the TDBF components
- SQLdb Tutorial0 - Set up instructions for sample data for the subsequent database tutorials. Has master/detail tables