SQL上級者こそ知って欲しい、なぜO/Rマッパーが重要か?

K
PgDay2012 LightningTalk
SQL上級者にこそ知って欲しい


なぜ
O/Rマッパー
             が重要か?
      makoto kuwata <kwa@kuwata-lab.com>
                 http://www.kuwata-lab.com/
問:次のSQLは
何が問題でしょう?



 copyright© 2012 kuwata-lab.com all rights reserved
SQL
--	 販売成績上位10コを抽出
select	 *	 from	 sales
where	 deleted	 =	 false
order	 by	 amount	 desc
limit	 10

                                 文法的には正しいけど…




        copyright© 2012 kuwata-lab.com all rights reserved
話は変わって、
他のプログラミング言語



  copyright© 2012 kuwata-lab.com all rights reserved
Ruby
##	 販売実績上位Nコを抽出
def	 sales_top(n,	 rows)
	 	 xs	 =	 rows.map	 {|row|	 Sale.new(*row)	 }
	 	 xs	 =	 xs.select	 {|x|	 !	 x.deleted_at	 }
	 	 xs	 =	 xs.sort_by	 {|x|	 -	 x.amount	 }
	 	 xs	 =	 xs[0,	 10]
	 	 return	 xs
end




            copyright© 2012 kuwata-lab.com all rights reserved
Ruby
##	 販売実績上位Nコを抽出
def	 sales_top(n,	 rows)
	 	 xs	 =	 rows.map	 {|row|	 Sale.new(*row)	 }
	 	 xs	 =	 xs.select	 {|x|	 !	 x.deleted_at	 }
	 	 xs	 =	 xs.sort_by	 {|x|	 -	 x.amount	 }
	 	 xs	 =	 xs[0,	 10]          select from相当
	 	 return	 xs
end




            copyright© 2012 kuwata-lab.com all rights reserved
Ruby
##	 販売実績上位Nコを抽出
def	 sales_top(n,	 rows)
	 	 xs	 =	 rows.map	 {|row|	 Sale.new(*row)	 }
	 	 xs	 =	 xs.select	 {|x|	 !	 x.deleted_at	 }
	 	 xs	 =	 xs.sort_by	 {|x|	 -	 x.amount	 }
	 	 xs	 =	 xs[0,	 10]
	 	 return	 xs            where相当
end




            copyright© 2012 kuwata-lab.com all rights reserved
Ruby
##	 販売実績上位Nコを抽出
def	 sales_top(n,	 rows)
	 	 xs	 =	 rows.map	 {|row|	 Sale.new(*row)	 }
	 	 xs	 =	 xs.select	 {|x|	 !	 x.deleted_at	 }
	 	 xs	 =	 xs.sort_by	 {|x|	 -	 x.amount	 }
	 	 xs	 =	 xs[0,	 10]
	 	 return	 xs         order by相当
end




            copyright© 2012 kuwata-lab.com all rights reserved
Ruby
##	 販売実績上位Nコを抽出
def	 sales_top(n,	 rows)
	 	 xs	 =	 rows.map	 {|row|	 Sale.new(*row)	 }
	 	 xs	 =	 xs.select	 {|x|	 !	 x.deleted_at	 }
	 	 xs	 =	 xs.sort_by	 {|x|	 -	 x.amount	 }
	 	 xs	 =	 xs[0,	 10]
	 	 return	 xs
                        limit相当
end




            copyright© 2012 kuwata-lab.com all rights reserved
Ruby
##	 販売実績上位Nコを抽出
def	 sales_top(n,	 rows)
	 	 xs	 =	 rows.map	 {|row|	 Sale.new(*row)	 }
	 	 xs	 =	 xs.select	 {|x|	 !	 x.deleted_at	 }
	 	 xs	 =	 xs.sort_by	 {|x|	 -	 x.amount	 }
	 	 xs	 =	 xs[0,	 10]
	 	 return	 xs
end                       1つの関数でいろんな
                                      ことをやりすぎている




            copyright© 2012 kuwata-lab.com all rights reserved
複数の関数に分解
(リファクタリング)




 copyright© 2012 kuwata-lab.com all rights reserved
Ruby
def	 to_sales(rows)
	 	 rows.map	 {|row|	 Sale.new(*row)	 }
end

def	 active(sales)
	 	 sales.select	 {|x|	 !	 x.deleted_at	 }
end

def	 top(n,	 sales)
	 	 sales	 =	 sales.sort_by	 {|x|	 -	 x.amount	 }
	 	 return	 sales[0,	 10]
end


            copyright© 2012 kuwata-lab.com all rights reserved
Ruby
##	 使い方
def	 sales_top(n,	 rows)
	 	 return	 top(n,	 active(to_sales(rows)))
end
                                           分解した関数を
                                            組み合わせる




           copyright© 2012 kuwata-lab.com all rights reserved
専用のクラスを定義
 (オブジェクト指向)




 copyright© 2012 kuwata-lab.com all rights reserved
Ruby
clas	 Sales
	 	 def	 initialize(rows)
	 	 	 	 @all	 =	 rows.map	 {|row|	 Sales.new(*row)	 }
	 	 end
	 	 attr_reader	 :all
	 	 def	 active
	 	 	 	 @all	 =	 @all.select	 {|x|	 !	 x.deleted	 }
	 	 	 	 return	 self
	 	 end
	 	 def	 top(n)
	 	 	 	 @all	 =	 @all.sort_by	 {|x|	 -	 x.amount	 }
	 	 	 	 @all	 =	 @all[0,	 n]
	 	 	 	 return	 self
	 	 end
end

              copyright© 2012 kuwata-lab.com all rights reserved
Ruby
##	 使い方
Sales.new(rows).active().top(10).all()


                                      分解したメソッドを
                                        組み合わせる




          copyright© 2012 kuwata-lab.com all rights reserved
プログラミング言語が
持っている基本機能



  copyright© 2012 kuwata-lab.com all rights reserved
• 「全体」 「部分」
      を    に分解する機能

• 「部分」 「全体」
      から   を構築する機能

• 「部分」に名前をつけて抽象化する機能




     copyright© 2012 kuwata-lab.com all rights reserved
改めて、次のSQLは
何が問題でしょう?



 copyright© 2012 kuwata-lab.com all rights reserved
SQL
--	 販売成績上位10コを抽出
select	 *	 from	 sales
where	 deleted	 =	 false
order	 by	 amount	 desc
limit	 10




        copyright© 2012 kuwata-lab.com all rights reserved
SQLは、分解・構築・
抽象化の機能が弱っちい!



  copyright© 2012 kuwata-lab.com all rights reserved
そこで
O/R Mapper!!

  copyright© 2012 kuwata-lab.com all rights reserved
ORMを使うことで、

•SQLを小さい部品に分解
• 部品からSQL全体を構築
• 部品に名前をつけて抽象化
できるようになる!

   copyright© 2012 kuwata-lab.com all rights reserved
サンプルコード:

ActiveRecord (Rails3)

     copyright© 2012 kuwata-lab.com all rights reserved
Ruby
                             where deleted = false
class	 Sales	 <	 ActiveReocrd::Base
                                                  に相当する「部分」
	 	 named_scope	 :active,	 
	 	 	 	 {:conditions=>"deleted	 =	 false"}

	 	 named_scope	 :top,	 lambda	 do	 |n|
	 	 	 	 {:order=>"amount	 desc",	 :limit=>n}
	 	 end
                            order by amount desc limit n
end                         に相当する「部分」




            copyright© 2012 kuwata-lab.com all rights reserved
「部分」を組み合わせて                                              Ruby
          「全体」を構築

Sales.active().top(10).all()

##	 これは
##	 	 select	 *	 from	 sales
##	 	 where	 deleted	 =	 false
##	 	 order	 by	 amount	 desc	 limit	 10
##	 を生成して実行する




        ※(2012-12-19) 「と同じ」を「を生成して実行する」に修正

            copyright© 2012 kuwata-lab.com all rights reserved
サンプルコード:

DataMapper

 copyright© 2012 kuwata-lab.com all rights reserved
Ruby
class	 Sales
                          where deleted = false
	 	 include	 DataMapper::Resource
                                          に相当する「部分」
	 	 def	 self.active
	 	 	 	 where(:deleted	 =>	 false)
	 	 end

	 	 def	 self.top(n)
	 	 	 	 order(:amount.desc).limit(n)
	 	 end
end                 order by amount desc limit n
                          に相当する「部分」



             copyright© 2012 kuwata-lab.com all rights reserved
「部分」を組み合わせて                                              Ruby
          「全体」を構築

Sales.active().top(10).all()

##	 これも
##	 	 select	 *	 from	 sales
##	 	 where	 deleted	 =	 false
##	 	 order	 by	 amount	 desc	 limit	 10
##	 を生成して実行する




        ※(2012-12-19) 「と同じ」を「を生成して実行する」に修正

            copyright© 2012 kuwata-lab.com all rights reserved
別のサンプル:

副問い合わせ


copyright© 2012 kuwata-lab.com all rights reserved
SQL
--	 年齢が20歳の社員が所属する部署一覧
select	 *	 from	 departments
where	 id	 in
	 	 (select	 dept_id	 from	 employees
	 	 	 where	 age	 =	 20)
order	 by	 name
                              SQLに「部品化」の機能が
                              ないことを示す典型例




                ※(2012-12-13) "select id" を "select dept_id" に修正

         copyright© 2012 kuwata-lab.com all rights reserved
サンプルコード:

      Sequel

copyright© 2012 kuwata-lab.com all rights reserved
副問い合わせを切り出して
     名前を付ける
                                                        Ruby
dept_ids	 =	 Employee.select(:dept_id)	 
	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 .filter(:age=>20)
Department.filter(:id=>dept_ids).all()
            SQL構築が簡潔になる
                                                                   SQL
select	 *	 from	 departments
where	 id	 in
	 	 (select	 dept_id	 from	 employees
	 	 	 where	 age	 =	 20)


              copyright© 2012 kuwata-lab.com all rights reserved
Ruby
dept_ids	 =	 Employee.select(:dept_id)	 
	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 .filter(:age=>20)
Department.filter(:id=>dept_ids).all()
Budget.filter(:dept_id=>dept_ids).all()


         部品化した副問い合わせを複数の
         SQLで共用できる(with句より便利)




              copyright© 2012 kuwata-lab.com all rights reserved
別のサンプル:

重複した式


copyright© 2012 kuwata-lab.com all rights reserved
SQL
--	 誕生日をもとに年齢を計算
select	 date_part('year',	 age(birth))	 	 	 
	 	 	 	 	 	 	 as	 age,	 count(*)
from	 users
where	 date_part('year',	 age(birth))	 <	 20
group	 by	 date_part('year',	 age(birth))
order	 by	 age
                               同じ式が重複して出現




           copyright© 2012 kuwata-lab.com all rights reserved
サンプルコード:

SQLAlchemy

 copyright© 2012 kuwata-lab.com all rights reserved
年齢の計算式を表す構文木を作り、                                       Python
from	 sqlalchemy.sql	 import	 func	 as	 fn
age	 =	 fn.date_part('year',	 
	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 fn.age(User.birth))
rows	 =	 DBSession	 
	 	 	 	 	 	 	 .query(age,	 fn.count('*'))	 
	 	 	 	 	 	 	 .select_from(User)	 
	 	 	 	 	 	 	 .filter(age	 <	 20)	 
	 	 	 	 	 	 	 .group_by(age)	 
	 	 	 	 	 	 	 .order_by(age)	 
	 	 	 	 	 	 	 .all()
                             それを複数箇所で使用できる



              copyright© 2012 kuwata-lab.com all rights reserved
つまり



copyright© 2012 kuwata-lab.com all rights reserved
• SQLには部品化の機能がない
 → SQLが長くなる・わかりにくい
 → 複数のSQLで要素の重複が多い

• ORMはSQL要素の部品化ができる
 → SQL構築が簡潔・わかりやすい
 → 複数のSQLで部品を共用できる


    copyright© 2012 kuwata-lab.com all rights reserved
• SQLには部品化の機能がない
 → SQLが長くなる・わかりにくい
 → 複数のSQLで要素の重複が多い

• ORMはSQL要素の部品化ができる
 → SQL構築が簡潔・わかりやすい
 → 複数のSQLで部品を共用できる


    copyright© 2012 kuwata-lab.com all rights reserved
たとえるなら



copyright© 2012 kuwata-lab.com all rights reserved
まだ進化の途中な
               高水準へと進化                                  ので暖かい目を!

Lisp, Ruby                                     Modern ORM



C, Pascal                                         JDBC, DBI



Assembler                                               SQL
                   低水準から


        copyright© 2012 kuwata-lab.com all rights reserved
まとめ



copyright© 2012 kuwata-lab.com all rights reserved
• 一般のプログラミング言語には
 「分解」「構築」「抽象化」
 の機能がある

• SQLはそれらが弱っちい
• モダンなORMなら、それらの機能
 をSQLに提供できる

    ORMはSQLの高水準言語 or DSL!



    copyright© 2012 kuwata-lab.com all rights reserved
おしまい
copyright© 2012 kuwata-lab.com all rights reserved
1 of 46

More Related Content

What's hot(20)

イミュータブルデータモデル(入門編)イミュータブルデータモデル(入門編)
イミュータブルデータモデル(入門編)
Yoshitaka Kawashima185.7K views

Similar to SQL上級者こそ知って欲しい、なぜO/Rマッパーが重要か?(20)

ScalaMatsuri 2016ScalaMatsuri 2016
ScalaMatsuri 2016
Yoshitaka Fujii7.1K views
Ruby on Rails 入門Ruby on Rails 入門
Ruby on Rails 入門
Yasuko Ohba11.3K views
20110820 metaprogramming20110820 metaprogramming
20110820 metaprogramming
Masanori Kado1.2K views
Functional JavaScript with Lo-Dash.jsFunctional JavaScript with Lo-Dash.js
Functional JavaScript with Lo-Dash.js
Shogo Sensui2.1K views
Clojure programming-chapter-2Clojure programming-chapter-2
Clojure programming-chapter-2
Masao Kato1.1K views
Scala on HadoopScala on Hadoop
Scala on Hadoop
Shinji Tanaka2.5K views
Oracle Database Connect 2017 / JPOUG#1Oracle Database Connect 2017 / JPOUG#1
Oracle Database Connect 2017 / JPOUG#1
Noriyoshi Shinoda2.7K views
Web技術勉強会 第25回Web技術勉強会 第25回
Web技術勉強会 第25回
龍一 田中649 views
Ruby on Rails on MySQL チューニング入門Ruby on Rails on MySQL チューニング入門
Ruby on Rails on MySQL チューニング入門
だいすけ さとう21.5K views

More from kwatch(20)

なんでもIDなんでもID
なんでもID
kwatch1.4K views
Fantastic DSL in PythonFantastic DSL in Python
Fantastic DSL in Python
kwatch13K views

SQL上級者こそ知って欲しい、なぜO/Rマッパーが重要か?