{}I言語自習書_02:リレーショナルデータベースとSQL
●リレーショナルデータベース(RDB)は他のDBと異なりデータを一番簡単な構造の2次元の表で持っています。構造が簡単なのでそれを扱う言語のSQLも文法が簡単で命令数も少なく習得が容易ですが、結構難しいニーズに容易に対応出来る優れ物ですので、I言語で使用しています。
●I言語ではデータの定義と操作はSQLで行うのが基本なので、SQLを理解しておく必要が有り、ここで自習します。
- リレーショナルデータベースとは。
1. リレーショナルデータベースは専用の言語のSQLを使ってデータの定義と操作をします。
2. リレーショナルデータベースはデータをテーブルと呼ばれる2次元の表で持ちます。縦が列(カラム、フィールド)で列名(データの名前)とデータ型を持ち、横が行(ロー、レコード)で、一件づつのデータと成ります。
テーブルの例名前 NCHAR(12) | 性別 NCHAR(2) | 生年月日 DATE |
山田太郎 | 男 | 1948-11-01 |
山田花子 | 女 | 1952-04-01 |
3. テーブルは一般的には正規化で構造設計をします。全部で8種類もの正規形が有りますが、実際のデータは2次元の表で表せるような単純な構造ではないので、正しい構造にするのではなく、あくまでも矛盾が起きにくい構造にする為ですので、矛盾が起きないでプログラムが複雑に成らなければそれで良しと思って下さい。
4. 正規化は第3正規化までで殆どが解決します、なので、下記3点に注意して作れば殆ど問題は起きません。。
4.1 列には繰り返しデータは持たないので、例えば、家族データをテーブルにしたい場合、名前は構成人数分必要ですので、1家族1行では無く、構成人数分の行で持ち、家族を区別出来る列を設けます。(第1正規化)
家族テーブル世帯主 | 世帯主枝番 | 名前 | 性別 | 生年月日 |
山田太郎 | 1 | 山田太郎 | 男 | 1948-11-01 |
山田太郎 | 1 | 山田花子 | 女 | 1952-04-01 |
(世帯主枝番は世帯主に同姓同名の人がいた場合の区別に使いますが、家族IDで家族を区別出来る列名を新規に設け、世帯主枝番を持たない方法も有ります)
4.2 主キーと呼ばれる他の行と区別出来るユニーク値を持つ列(複数の列でユニークでも可)を探し、主キーが決まれば値が決まる列を含めて1個のテーブルとします。(第2及び第3正規化)
◎リレーショナルデータベースのテーブルはExcelの表とは異なり、行の記憶場所は決まっていません、なので、テーブル中の1行を特定するには列の値がユニークな物を必要とします、これを主キーと呼んできます。
〇世帯主、世帯主枝番、名前、性別、生年月日では、世帯主、世帯主枝番、名前の3個で主キーとなり、主キーが決まれば性別と生年月日が決まるので、これで良しとなります。
4.3 他の列から導き出せる物は、他の列が変更された時点で矛盾が起きてしまうので、元のテーブルには持たない事が基本ですが,月次処理等で計算した結果のテーブルに無いと逆に毎回計算が必要になったり、その時の計算結果に誤りが無かったか確認出来ないので、こちらは持った方が良いです。
〇世帯主、世帯主枝番、名前、性別、生年月日では、他の列から導き出せる物は無いので、これが完成形と成ります。
5. リレーショナルデータベースはRDBと表現する事があります。
6. RDBを実現するソフトをRDBMS(リレーショナルデータベースマネージメントシステム)と言います。
- SQLとは。
1.SQLはJIS規格にもなっており、基本部分はどのRDBMSでも共通ですが、拡張部分は方言のような異なる文法の部分も多いので注意が必要です。
◎I言語では、拡張部分をI言語の文法でカバーし、極力同じ書き方でどのRDBMSでも動くようにしています。
2. 定義は物に対する定義とそれ以外が有ります。
2.1 定義する物はテーブルが主ですが主な物が下記です、createで作り、alterで変更し、dropで消すSQL文が有ります。
2.1.1 table---データを記憶する物です。
2.1.2 index---データの検索を早くしたり、ユニークな値以外を持てない制約を定義出来ます。
2.1.3 view---仮想的なテーブルを定義します。(プログラムの解析が難しくなるので、極力使わない事を推奨します、I言語側ではRDBMS毎にシステム系テーブル構造が異なるのでViewで同じ構造にしています)
★黄色の物は、RDBMS毎に文法が異なるので、必要な物は可能な限りI言語側で対応する事で、直接使わなくても良くしたので、極力使わない事を推奨している物です。このような物が存在する事は記憶に留めておいて下さい。
2.1.4 function---ユーザー関数を定義します。(プログラムの解析が難しくなるので、極力使わない事を推奨します)
2.1.5 procedure---RDBMS固有のプログラムを定義します。(RDBMS毎に文法が大きく異なるので、極力使わない事を推奨します、希にcousorとfetchを使って一行づつ処理したい場合、処理スピードを保証する為使う場合が有りいます)
2.1.6 trigger---テーブルが更新された事を引き金に実行する処理を定義します。(システムの解析が難しくなるので、極力使わない事を推奨します)
2.1.7 sequence---連続した数値を発生する物を定義します。(プログラムの解析が難しくなるので、極力使わない事を推奨します)
2.1.8 synonym---物に別名を定義します。(プログラムの解析が難しくなるので、極力使わない事を推奨します)
2.1.9 user---ユーザーを定義します。(I言語では運用支援プログラムが定義します、RDBMSへの接続はどれかのユーザーを指定して接続します)
2.1.10 schema---スキーマを定義します(SQL Serverの場合dboスキーマが設定されており、schemaを増やすと管理が大変ですので、極力使わない事を推奨します、テーブル等は必ずどれかのスキーマに所属します)
2.1.11 database---データベースを定義します。(I言語では運用支援プログラムが定義します)
2.2 物以外の定義です。
2.2.1 grant---ユーザーに権限を付与します。(I言語では開発支援プログラムが使います、下記記入の操作のSELECT,INSERT,UPDATE,DELETEの権限を個別に設定出来ます)
2.2.2 revoke---ユーザーの権限を削除します。(I言語では開発支援プログラムが使います)
2.2.3 begin---トランザクションを開始します。(トランザクションはcommitかrollbackが実行れるまでテーブル更新を保留します)
〇トランザクションは適切な排他処理をする事で、複数個所から同時に更新をしても、処理中に何らかのトラブルが有っても、矛盾が起きないようにする為の物です。
2.2.4 commit---トランザクション開始後の、更新を実行します。
2.2.5 rollback---トランザクション開始後の、更新を取り消します。
3. 操作するSQLは主に下記の4個です。
3.1 select---データを検索します。
□ select [distinct] 式... [from テーブル名 [別名] [結合 テーブル名 [別名][on 条件]].....]
[where 条件] [group by 式...] [having 条件] [order by 式 [desc]...]
★[]内は省略可能の意味です。
★...はカンマ区切で複数個書ける意味です。
★.....は空白区切で複数個書ける意味です。
〇selectが一番多くの場面で使用し、色々な事が出来るので、しっかり覚えて下さい。
〇式は列名ばかりでなく、関数、計算式、文字リテラル、等が書けます。
〇selectで始まり検索したい式をカンマ区切りで書きます。(distinctは同じ内容の行は1行に成ります)
〇from以下にテーブルを書き結合で複数のテーブルを結合しonで結合条件を書きます、whereで検索条件を書きます。(結合は、inner join,left join,right join,full joinの4種類が有ります)
〇groupは合計等の集計関数を使う場合のグループを書き、havingには集計関数を使った検索条件を書きます。
〇orderは検索する行の順番を書きます、RDBでは行の格納順番は何の保証も無いので、orderで順番を指定します。(descが有ると降順に成ります、無いと昇順に成ります)
3.2 insert---データを作成します。
□ insert into テーブル名 [(列名...)] values(式...)
□ insert into テーブル名 [(列名...)] select文
3.3 update---データを修正します。
3.3.1 テーブルが1個の場合。
□ update テーブル名 set 列名=式... [where 条件]
〇where以下を書かないと全行を対象に処理されます、なので、通常は1行処理の為主キーを使って条件を設定します。
3.3.2 テーブルが2個の場合。(RDBMS毎に文法が異なります,SQL Serverの場合)
□ update O set 列名=式... from テーブル名1 O,テーブル名2 I where 条件
3.3.2.1 PostgreSQLの場合 (参考)
□ update テーブル名1 O set 列名=式... from テーブル名2 I where 条件
3.3.2.2 Oracle,DB2の場合 (参考)
□update テーブル名1 O set (列名...)=(select 式... from テーブル名2 I where 条件) where exists(select 0 from テーブル名2 I where 条件)
3.3.2.3 MariaDBの場合 (参考)
□update テーブル名1 O set 列名=(select 式 from テーブル名2 I where 条件)... where exists(select 0 from テーブル名2 I where 条件)
3.3.2.4 I言語の場合(参考)
□=UPDATE{テーブル名1,列名...}{式...}{テーブル名2 I WHERE 条件};
3.4 delete---データを削除します。
□ delete from テーブル名 [where 条件]
◎I言語ではdeleteは使わないで削除履歴を残す為、updateを使って論理削除で対応しています、最終的に運用支援プログラムがdeleteしています。
- SQLを使うのでSSMS(SQL Server Management Studio)を実行します。
1. 左下の「START」をクリックします。
2.「Microsoft SQL Server Tools」をクリックします、
3.「Microsoft SQL Server Management Studio」をクリックします、
4.「サーバーへの接続」でサーバー名が「コンピュータ名¥SQLEXPRESS」で有る事を確認し「接続」をクリックします。
(認証はWindows認証です、これはWindowsにログインしているログインIDでSQL Serverに接続する意味で、SQL ServerをインストールしたユーザーIDはSQL Serverの管理者権限が与えられており、SQL Serverに接続出来るように成っています)
5.「新しいクエリ」をクリックします。
6.新しく表示された画面上でSQLを実行します。
- テーブルを消すdrop table テーブル名
を実行します。
drop table t_a
★青い部分がSQLです、青い部分をマウスでドラッグし右ボタンでコピーしSSMSに貼り付けで「実行」をクリックすれば実行します。
1.実行するとメッセージに「テーブル 't_a' を 削除 できません。存在しないか、権限がありません。」とテーブルが無いのでエラーが出ます。
- テーブルを作るcreate table テーブル名 (列名 データ型 [not null] [default 初期値]...[,primary key(主キー用列名...)])を実行します。
drop table t_a
go
create table t_a(c_a char(6) not null default ' ',c_b nchar(6) not null default ' ',c_c decimal(5,2) not null default 0,primary key(c_a,c_b))
1.SSMSでは複数のSQLを書けますが、SQLの区切りには「go」を書くルールになっています。
2.charは半角文字コード用のデータ型で、6は文字数です。(ただし、全角文字も登録出来ます)
3.ncharは全角文字コード用のデータ型です。(ただし、半角文字も登録出来ます)
4.deimalは数値用のデータ型です、5は有効桁数で、2は小数部桁数です。
5.char,nchar,decimal以外にもデータ型が有りますが、基本は上記3個とdate型とdatetime型です、その他は、charの固定長に対しvarcharの可変長、ncharの固定長に対しnvarcharの可変長で、小数点無しで有効桁数8桁以内であればint、小さな値から大きな値まで持てるが計算結果は近似値で良い物にfloatが有ります。
6.not nullはnull(値が無い)が作成出来ないように成ります。
◎nullの時は「列名!='A'」(列が'A'では無い)としてもnullは'A'では無いのに真と成りません、このような凡ミスが起きやすいので、I言語ではnullをテーブル上に持たない事を基本としています。
7.default ' 'はinsert文(行の作成)で列名を指定しない時に初期値として設定されます。
◎not nullを定義しているので、該当列名の無いinsert文は作成出来ないエラーと成るので、I言語ではdefaultも定義します。
8.primary keyは主キーの列名を指定します。
9.実行でメッセージにdropのエラーが出ますが、再度実行で、テーブルが作られており、テーブルを消して作っているのでメッセージに「コマンドは正常に完了しました。」が出ます。
- データを作成するinsert into テーブル名[(列名...)] values(内容...)を実行します。
◎列名は省略できますが、プログラム内では、列名の数が変わるとエラーと成ってしますので、列名を必ず書くようにします。
insert into t_a (c_a,c_b,c_c) values('abcd','efgh',1)
go
insert into t_a (c_a,c_b,c_c) values('abcd','ijkl',-2.6)
go
insert into t_a (c_a,c_b,c_c) values('mnop','qrst',3.4)
go
insert into t_a (c_a,c_b,c_c) values('abcd','efgh',4)
1.charとncharの内容は引用符で囲みます。
2.実行でメッセージに「PRIMARY KEY 違反」が出ており、最後のデータのPRIMARY KEYが同じ物は作成出来ない事が分かります。
- データを検索しますselect 式... from テーブル名[order by 式...]を実行します。
★式は列名や値やそれ以外の値を行単位に1個又は0個返す物が書けます。
◎「select *」で全列の指定が出来ますが、プログラム内では、列の数が変わると検索結果が変わってしまうので使いません。
select c_a,c_b,c_c from t_a order by c_a,c_b
1.実行で結果に3件の行が表示されます。
2.orderを書かないと順不同で表示されます。
3.SQL Serverの場合テーブルが必要無い時は「from テーブル名」を省略する事も出来ます。
◎SQLは今までのプログラムのように1件づつのデータを順次処理する方法ではなく、全てのデータを対象に処理されます,
尚、今までのように1件づつ処理する方法もcursorとfetchを使えば可能ですが、出番は殆ど無いです。
- 条件を付けてデータを検索しますselect 式... from テーブル名 [where 条件]を実行します。
select c_a,c_b,c_c from t_a where c_a='abcd'
1.実行で結果に条件に合致した2件の行が表示されます。
2.条件の演算子は更に下記の物が有ります。
2.1=(等しい),>(大きい),<(小さい),>=(以上)
,<=(以下),!=(等しくない),<>(等しくない)
,式 [not] between a and b( a以上でb以下で真)
,式 [not] in(a,b...) (一覧の1つに一致で真,in内にサブクエリと呼ばれるselect文を書く事も出来ます)
,式 [not] like 'ab%'(パターンに一致,'%'が0個以上の全ての文字に一致、'_'が1文字の全ての文字に一致,'ab%'は先頭がabの物全ての意味に成ります)
,[not] exists(サブクエリ)(サブクエリでデータが有れば真)
2.2複数の条件が必要な場合はandで両方の条件が真,orでどちらか一方が真で真となり、andとorではandが先に判定されますが、カッコを付ければカッコ内を先に判定します、又notで否定判定も出来ます。
2.3null(値無しのデータ)の判定は「is null」の判定以外は真に成らないので注意して下さい。
◎I言語のZZZZ010319に(SQL条件(WHERE) サンプル)が作ってあるので参考にして下さい。
- 式にcastを使えばデータ型の変換が出来ますcast(式 as データ型)[as 別名]。
select c_c,cast(c_c as decimal(3,0))as c_c2 from t_a
1.castで小数点以下を無くしています。
- castで変換できないデータ型の変換はエラーと成ります。
select c_a,cast(c_a as decimal(6,0))as c_a2 from t_a
1.文字は数値に変換出来ません。
- groupを付けて集計関数でデータを検索するselect [式...,]集計関数(式)[as 別名]... from テーブル名 [group by 式...][having 条件]を実行します。
select c_a,sum(c_c) as sum,max(c_c) as max,min(c_c) as min,avg(c_c) as avg,count(*) as count
,stdev(c_c)as stdev,stdevp(c_c)as stdevp,var(c_c)as var,varp(c_c)as varp
,string_agg(c_a,',')as string_agg from t_a group by c_a having count(*)>1
1.sum(合計),max(最大値),min(最小値),avg(平均値),count(件数)
,●stdev(標本標準偏差),●stdevp(標準偏差)、●var(標本分散),●varp(分散)
,▲string_agg(区切り文字で結合)。
★RDBMSによっては使えない物も有ります、DB2、Oracle、PostgreSQL、MariaDBを含む5個のRDBMSで、
文法が異なるが使える物は●を、複数のRDBMSで使える物は■を、SQL Serverのみ使える物は▲を付けています。
2.集計関数を使った条件判定はwhereでは無くhavingを使います。
- caseで条件により内容を変更する事が出来ますcase 式 when 値1 then 新値1 ... else 値 end [as別名]です。
select c_a,c_b,c_c,case c_c when 1 then 'abc' when -2.6 then 'xyz' else 'zzz' end as case1 from t_a
1.別の書き方でcase when c_c=1 then 'abc' when c_c=2.6 then 'xyz' else 'zzz' endが有り、等しい以外の演算子も使えます。
- caseに似た物にnullifが有りますnullif(式1,式2) end [as別名]。
select nullif(c_a,c_a),nullif(c_a,c_b) from t_a
1.式1と式2が等しい時nullをそれ以外は式1を返します。
2.nullが発生するので、ほとんど出番は有りません。
- selectから始まるサブクエリは何処でも使えます、式とテーブルに書いてみます。
select c_a,c_b,c_c,(select c_c from t_a t2 where t2.c_a=t1.c_a and t2.c_b=t1.c_b)as c_c2 from (select * from t_a)t1
1.サブクエリは便利ですので、複雑なニーズに柔軟に対応出来ます。
- テーブルを単純に結合しますselect 式... from テーブル名1,テーブル名2... 。
create table t_b(c_a char(4) default ' ' not null,c_d decimal(5,2) default 0 not null,primary key(c_a))
go
insert into t_b(c_a,c_d) values('abcd',2)
go
insert into t_b(c_a,c_d) values('wxyz',2)
go
select * from t_a,t_b
1.行同士が結合し、両テーブルの列が集合して表示されます。
2.「実行」で実行します。
3.結果に6件の行が表示されます。t_aの3件の1行づつにt_bの2件を単純に付加して、6件が表示されます。
4.昔のSQLは、これにwhereで条件を付けて処理をしていましたが、最近は別の方法が使われています。尚、複雑な処理には必要な場合も有ります。
- inner joinで条件に有った物を結合し検索しますselect 式... from テーブル名1 inner join テーブル名2 on 条件 [.....]。
★[.....]は複数個書ける意味です。
select * from t_a inner join t_b on t_a.c_a=t_b.c_a
1.実行で結果にonの条件が真の2件の行が表示されます。
- left outer joinで結合し検索しますselect 式... from テーブル名1 left outer join テーブル名2 on 条件 [.....]。
select * from t_a left outer join t_b on t_a.c_a=t_b.c_a
1.実行で結果にonの条件が真の2件の行とt_aで結合相手が無い1行の計3行が表示されます。
◎一般的なシステムでは処理対象のデータテーブルと基本情報を持っているマスタテーブルと結合し必要な情報を得て、処理を行います、left outer joinはマスタテーブルの主キーと結合すれば、データテーブルの件数は常に同じと成るので、一番多く使われる結合方法です。
3.right outer joinでleftと逆のテーブルのonでの結合が無い行が付加されます。
- full outer joinで結合し検索しますselect 式... from テーブル名1 left outer join テーブル名2 on 条件 [.....]。
select coalesce(t_a.c_a,t_b.c_a),c_b,c_c,c_d from t_a full outer join t_b on t_a.c_a=t_b.c_a
1.両方のテーブルのonでの結合が無い行が付加され4行表示されます。
2.coaresce(t_a.c_a,t_b.c_a)は最初のnull以外の値を採用するので、full outer joinの結合キーの表示には最適です。
◎I言語のZZZZ010318に(SQLテーブル結合 サンプル)が作ってあるので参考にして下さい。
- union allでselect 式... from テーブル名1 union all select 式... from テーブル名2 [.....]両方の行がそれぞれ表示されます。
select c_a,c_b,c_c from t_a union all select c_a,null,c_d from t_b
1.実行で結果にt_aの3行とt_bの2行の計5行が表示されます。
2.unionのみにすると、全く同じ内容の行は1行にまとめられます、尚、普通のselectはselect distinct ...とすれば同じ内容の行が1行と成ります。
- 式で計算も出来ます。
提供し
select c_c,c_c+2 as p,c_c-2 as mi,c_c*2 as mu,c_c/2 as d,c_c%2 as r from t_a
1.*は乗算で、/は除算で、■%は除算の余りです。
尚、I言語では可能な限りシステム変数で提供し、RDBMSが異なってもそのまま動くように対応しています。
2.算術演算子を複数使った複雑な計算も可能ですし、括弧を使って計算の順序を変える事もできます。
- 式に数学関数が使えます。
select c_c,abs(c_c)as abs,acos(c_c/10)as acos,asin(c_c/10)as asin
,atan(c_c)as atan,atn2(c_c,c_c*2)as atn2,ceiling(c_c)as ceiling,cos(c_c)as cos
,cot(c_c)as cot,degrees(c_c)as degrees,exp(c_c)as exp,floor(c_c)as floor
,log(abs(c_c)*10)as log,log10(abs(c_c)*10)as log10,pi()as pi,power(c_c,2)as power
,radians(c_c)as radians,rand(c_c) as rand,round(c_c,0)as round0
,round(c_c,0,1)as round1,sign(c_c)as sign,sin(c_c)as sin,sqrt(abs(c_c))as sqrt
,square(c_c)as square,tan(c_c)as tan from t_a
1.abs(絶対値),acos(逆余弦),asin(逆正弦,atan(逆正接),●atn2(点 (y, x) までの線の間の角度をラジアンで返す)
,●ceiling(最小整数),cos(余弦),■cot(コタンジェント),■degrees(ラジアンを度に変換),exp(指数値),floor(最大整数)
,log(自然対数),log10(底が10の対数),pi(円周率),power(べき乗),■radians(度をラジアンに変換),●rand(疑似乱数)
,round(式,まるめ位置)(四捨五入),■round(式,まるめ位置,1)(切捨て),sign(正は+1,負は-1、0は0)
,sin(サイン),sqrt(平方根),▲square(2乗),tan(タンジェント)。
◎I言語のZZZZ010320に(SQL算術 サンプル)が作ってあるので参考にして下さい。
- 式に文字関数も使えます。
select c_a,c_b,c_c,ascii(c_b)as ascii,char(ascii(c_b))as char,charindex('fg',c_b)as charindex,c_a+c_b as concat0,concat(c_a,c_b)as concat
,concat_ws('#',c_a,c_b,c_a)as concat_ws,difference(c_a,c_b)as difference,format(c_c+123456,'##/##/##')as format
,left(c_b,3)as left0,len(c_b)as len,lower(c_b)as lower,ltrim(' '+c_b)as ltrim,nchar(ascii(c_a))as nchar
,patindex('%fg%',c_b)as patindex,quotename(rtrim(c_a)+']'+rtrim(c_b))as quotename,replace(c_b,'fg','xx')as peplace
,replicate('x',3)as replicate,reverse(c_b)as reverse,right(c_b,4)as right0,rtrim(c_b)+'x' as rtrim,soundex(c_b) as soundex
,'|'+space(c_c)+'|' as space,str(c_c,4,1)as str,string_escape('/"','json')as string_escape,stuff(c_a,2,2,'xy')as stuff
,substring(c_a,2,2)as substring,translate(c_a,'bcno','1234')as translate,trim('ap' from rtrim(c_a))as trim
,unicode(c_a)as unicode,upper(c_a)as upper from t_a
ascii(先頭文字をアスキーコードに変換),char(アスキーコードを文字に変換),●charindex(検索),●+(結合),concat(結合)
,■concat_ws(区切り文字付きで結合),▲difference(soundex()値の差)
,▲format(書式設定),■left(左から抽出),●len(文字数),lower(小文字化),ltrim(左の空白を削除)
,■nchar(ユニコードを文字に変換),▲patindex(パターン検索),■quotename(引用符変更),peplace(置き換え)
,■replicate(繰り返す),■reverse(反転),■right(右から抽出),rtrim(右の空白を削除),▲soundex(2 つの文字列の類似性を評価)
,■space(空白繰り返シ),▲str(数値を文字に変換),▲string_escape(エスケープを付加します),▲stuff(部分入れ替え)
,●substring(部分抜き出し),trim(両端の指定文字と空白を削除します),■translate(複数の文字単位の置き換え)
,▲unicode(先頭文字をユニコードに変換),upper(大文字化)。
- 式に分析関数も使えます。
select c_a,c_b,c_c,cume_dist() over(partition by c_a order by c_c)as cume_dist
,first_value(c_b) over(partition by c_a order by c_c)as first_value
,lag(c_b,1) over(partition by c_a order by c_c)as lag
,last_value(c_b) over(partition by c_a order by c_c)as last_value
,lead(c_b,1) over(partition by c_a order by c_c)as lead
,percentile_cont(0.5) within group(order by c_c) over(partition by c_a)as percentile_cont
,percentile_disc(0.5) within group(order by c_c) over(partition by c_a)as percentile_disc
,percent_rank() over(partition by c_a order by c_c)as percent_rank
,dense_rank() over(partition by c_a order by c_c)as dense_rank
,ntile(2) over(partition by c_a order by c_c)as ntile
,rank() over(partition by c_a order by c_c)as rank
,row_number() over(partition by c_a order by c_c)as row_number
from (select * from t_a union all select 'mnop',c_a,c_d from t_b)t order by c_a,c_c,c_b
■cume_dist(累積分布),■first_value(最初の行の値),■lag(前の行の値),■last_value(最後の行の値)■lead(後の行の値)
■percentile_cont(百分位数),■percentile_disc(特定の百分位数),■percent_rank(相対的な順位)
■dense_rank(順位が飛ばない),■ntile(指定値で分割),■rank(順位が飛ぶ)■row_number(順番)
- 日付け関数も使えます、尚、日付け関数はRDBMS毎に文法が大幅に異なるので、下記はSQL Serverのみの説明です。
create table t_c(c_d date default getdate() not null,c_t datetime default getdate() not null)
go
insert into t_c(c_d,c_t) values('2018-09-01','2018-04-01 03:30:30.333')
go
insert into t_c(c_d,c_t) values('2018-08-31','2018-05-31 23:59:59.333')
go
insert into t_c(c_d,c_t) values(getdate(),getdate())
go
select c_d,dateadd(yy,3,c_d)as yy3,dateadd(q,-3,c_d)as q_3,dateadd(m,3,c_d)as m3,dateadd(y,3,c_d)as y3
,dateadd(d,3,c_d)as d3,dateadd(ww,3,c_d)as ww3,dateadd(w,3,c_d)as w3 from t_c
go
select c_t,dateadd(hh,3,c_t)as hh3,dateadd(mi,3,c_t)as mi3,dateadd(s,3,c_t)as s3,dateadd(ms,3,c_t)as ms3 from t_c
go
select c_d,datediff(yy,c_d,getdate())as yy,datediff(q,c_d,getdate())as q,datediff(m,c_d,getdate())as m,datediff(y,c_d,getdate())as y
,datediff(d,c_d,getdate())as d,datediff(ww,c_d,getdate())as ww,datediff(w,c_d,getdate())as w
,c_t,datediff(hh,c_t,getdate())as hh,datediff(mi,c_t,getdate())as mi,datediff(s,c_t,getdate())as s,datediff(ms,c_t,c_t)as ms from t_c
go
select c_d,datepart(yy,c_d)as p_yy,datepart(q,c_d)as p_q,datepart(m,c_d)as p_m,datepart(y,c_d)as p_y
,datepart(d,c_d)as p_d,datepart(ww,c_d)as p_ww,datepart(w,c_d)as p_w
,c_t,datepart(hh,c_t)as p_hh,datepart(mi,c_t)as p_mi,datepart(s,c_t)as p_s,datepart(ms,c_t)as p_ms from t_c
go
1.データ型は日付のみのdateと日付と時間のdatetimeが有ります。
2.getdate()で現時点の日付と時間が設定出来ます。
3.dateaddで日付と時間の計算(負の値で戻る計算も出来ます)が,datediffで差が、datepartで部分が出せます。
yy(年),q(四半期),y(年始からの日数),d(日数),ww(週),w(曜日)、hh(時),mi(分),s(秒,ms(ミリ秒)
◎I言語のZZZZ010317に(SQL関数関連(集計含む)サンプル)が作ってあるので参考にして下さい。
- updateで修正が出来ます。
update t_a set c_b=case c_b when 'efgh' then 'zzzz' else 'zzzzzz' end,c_c=999.99 where c_a='abcd'
go
select * from t_a
1.whereを書かないと全行が対象となってしまうので注意が必要です。
2.2行分が修正されています。
- deleteで削除が出来ます。
delete t_a where c_a='abcd'
go
select * from t_a
1.whereを書かないと全行が対象となってしまうので注意が必要です。
2.2行分が削除されて、1行に成っています。
- create indexでインデックスを作ります。create [unique] index インデックス名 on テーブル名 (列名...)
create index t_a1 on t_a(c_b,c_c)
1.uniqueを書くと重複を認めません。
- drop indexでインデックスを消します。drop index インデックス名 on テーブル名
drop index t_a1 on t_a
1.「on テーブル名」を書かないRDBMSも有ります。
- with句の再帰クエリと呼ばれる機能を使って、部品表を表示する事が出来ます。
create table t_d(c_oya char(6) default ' ' not null,c_ko char(6) default ' ' not null,primary key(c_oya,c_ko))
go
insert into t_d (c_oya,c_ko) values('a','aa')
go
insert into t_d (c_oya,c_ko) values('a','ab')
go
insert into t_d (c_oya,c_ko) values('a','ac')
go
insert into t_d (c_oya,c_ko) values('ab','aba')
go
insert into t_d (c_oya,c_ko) values('ac','aca')
go
insert into t_d (c_oya,c_ko) values('ac','acb')
go
insert into t_d (c_oya,c_ko) values('acb','acba')
go
insert into t_d (c_oya,c_ko) values('acb','acbb')
go
with t_with as (select * from t_d where c_oya='a'
union all select t_d.* from t_d,t_with where t_d.c_oya=t_with.c_ko
)
select * from t_with order by 1
1.withで事前にSQLを宣言出来ますが、宣言したテーブル自身をwith内に書く事で再帰的にSQLを実行出来ます。
- 発注システムで在庫を引き当てる等一行づつ処理する必要が有る場合、curssorとfetchで一行づつ在庫を引き当てます。
◎初心者にはかなり難しい内容ですので、ここでの理解はしなくても良いですが、必要な場合も有るので、このような機能が有る事だけは覚えておいて下さい。
出番は少ないですが、必要な場合が無い訳では無いので、必要に成った時点でしっかりと理解して下さい。
create table t_in(c_item char(1),c_date char(8),c_order decimal(5,0),c_order_start decimal(5,0),primary key(c_item,c_date ))
go
create table t_out(c_item char(1),c_date char(8) ,c_order decimal(5,0),c_order_start decimal(5,0),primary key(c_item,c_date))
go
create table t_stock(c_item char(1),stock decimal(5,0),stock_start decimal(5,0),primary key(c_item))
go
insert into t_in(c_item,c_date,c_order,c_order_start)values('a','20180110',100,100)
go
insert into t_in(c_item,c_date,c_order,c_order_start)values('a','20180111',200,200)
go
insert into t_in(c_item,c_date,c_order,c_order_start)values('b','20180112',300,300)
go
insert into t_in(c_item,c_date,c_order,c_order_start)values('c','20180112',600,600)
go
insert into t_stock(c_item,stock,stock_start)values('a',250,250)
go
insert into t_stock(c_item,stock,stock_start)values('b',500,500)
go
insert into t_stock(c_item,stock,stock_start)values('d',900,900)
go
begin
declare @o_end_sw decimal(5,0)=0; /*1で終了判定*/
declare @f_item char(5); /*@f_...はfetchで設定*/
declare @f_date char(8);
declare @f_order decimal(5,0);
declare @f_order_start decimal(5,0);
declare @w_item char(5)=' '; /*@w_...は作業用*/
declare @w_date char(8)=' ';
declare @w_order decimal(5,0)=0;
declare @w_order_start decimal(5,0)=0;
declare @w_stock decimal(5,0)=0;
declare @w_stock_start decimal(5,0)=0;
declare @w_update decimal(5,0)=0;
declare @w_count decimal(5,0)=0;
declare o_cursor cursor for select c_item,c_date,c_order,c_order_start from t_in order by c_item,c_date; /*cursorのselect文*/
begin open o_cursor;
while 0 = 0 begin /*無限ループ*/
fetch o_cursor into @f_item,@f_date,@f_order,@f_order_start; /*fetchで@f_...に値を設定*/
if @@fetch_status<>0 set @o_end_sw=1;/*終了時@o_end_swを1に設定*/
if @w_item<>@f_item OR @o_end_sw<>0 begin /*処理対象のitem値が変化で後処理実行*/
if @w_update=1 begin update t_stock set stock=@w_stock where c_item=@w_item;end; /*引き当て有った場合はt_stockをupdate*/
if @o_end_sw=0 begin /*終了でない場合前処理実施*/
set @w_item=@f_item;set @w_update=0;/*前処理*/
select @w_count=count(*) from t_stock where c_item=@w_item and stock>0 ; /*t_stockにデータ有るか件数select*/
if @w_count=1 begin /*t_stockにデータ有リは引き当て計算開始*/
select @w_stock=stock,@w_stock_start=stock_start /*t_stock情報@w_...に設定*/
from t_stock where c_item=@w_item and stock>0;end;
else begin set @w_stock=0;
end;end;end;
if @o_end_sw=0 begin /*引き当て計算*/
if @f_order>0 and @w_count=1 begin
if @w_stock>@f_order begin /*stockが大き*/
set @w_stock=@w_stock - @f_order; /*stockから引く*/
set @f_order=0; /*orderは0*/
end; else begin
set @f_order=@f_order - @w_stock; /*orderから引く*/
set @w_stock=0; /*stockは0*/
end;
set @w_update=1;/*引き当てした*/
end;
insert into t_out(c_item,c_date,c_order,c_order_start) values(@f_item,@f_date,@f_order,@f_order_start); /*t_outにinsert*/
end;
if @o_end_sw<>0 break; /*終了していたら、無限ループ解消*/
end;
close o_cursor; deallocate o_cursor;end;end /*cursorを閉じる*/
go
select * from t_out
go
select * from t_stock
1.aが100と200で合計300ですが、在庫が250しかないので、最後のオーダーが50と成ります。
2.bは在庫が500有り、発注側の300が引かれ発注は0で在庫は200mと成ります。
3.cは在庫が無いのでそのまま600を発注します。
4.dは発注が無いので、在庫が900のままとなります。
5.cursorとfetch関連はRDBMSで大きく文法が異なるので、上記はSQL Serverのみ動きます、尚、I言語ではRDBMS間で共通に動く命令を用意してあります。
◎I言語のZZZZ010327に(CURSOR{}サンプル(発注数計算))が作ってあるので参考にして下さい。
◎このようにSQLでかなりの処理を実現できますが、データの定義と操作以外は出来ないので、プログラムを作るにはプログラミング言語が必要と成ります。
All Rights Reserved, Copyright (C) 2018-2023 Nobumichi Harasawa.