Slide 1

Slide 1 text

"DUJWF4VQQPSU$PODFSO
 ͰֶͿ3VCZ &CJTVSC ,B[VNB4V[VLJ

Slide 2

Slide 2 text

ࣗݾ঺հ w ླ໦࿨ਅʢ!LB[VNBYʣ w ܦྺ w ɹΧΧΫίϜೖࣾ w ʙ৯΂ϩά w ʙΩφϦϊ w झຯ w ཱྀ !"#$%&'()*+,-./0123456789:;<=>?@

Slide 3

Slide 3 text

wʮ฻Β͠Λૉఢʹஸೡʹɻʯ ΛςʔϚʹͨ͠ϥΠϑελΠϧ ϝσΟΞɺ&$Ϟʔϧ w݄ؒ ສ66 w୅ޙ൒ʙ୅ঁੑ͕ϝΠϯ λʔήοτ

Slide 4

Slide 4 text

ࠓ೔ͷΰʔϧ w "DUJWF4VQQPSU$PODFSOͷίʔυΛಡΉ͜ͱΛ௨ͯ͠ 3VCZ΁ͷཧղΛਂΊΔ w ର৅ɿ3VCZॳதڃऀ

Slide 5

Slide 5 text

"DUJWF4VQQPSU$PODFSOͱ͸ w 3BJMTͷ"DUJWF4VQQPSUϥΠϒϥϦͷϞδϡʔϧͷͭ w ϞδϡʔϧΛJODMVEF͢Δࡍʹগͳ͍هड़ͰΠϯελϯε ϝιουͱΫϥεϝιουΛ௥Ճͯ͘͠ΕΔ w JODMVEF͢ΔϞδϡʔϧͷґଘؔ܎Λ؅ཧͯ͘͠ΕΔ

Slide 6

Slide 6 text

෮श

Slide 7

Slide 7 text

0CKFDUFYUFOE w ࢦఆͨ͠ϞδϡʔϧͷΠϯελϯεϝιουΛTFMGͷಛ ҟϝιουͱͯ͠௥Ճ͢Δ module M def my_method 'Ebisu.rb' end end class C1 include M end class C2 extend M end C1.new.my_method #=> "Ebisu.rb" C2.my_method #=> "Ebisu.rb"

Slide 8

Slide 8 text

"DUJWF4VQQPSU$PODFSOͷྫᶃ module M def self.included(base) base.extend ClassMethods base.class_eval do scope :disabled, ->{ where(disabled: true) } end end module ClassMethods ... end end module M extend ActiveSupport::Concern included do scope :disabled, ->{ where(disabled: true) } end class_methods do ... end end $PODFSOͳ͠ $PODFSO͋Γ w Ϋϥεϝιουͷ௥Ճ΍JODMVEFͨ͠ଆͷΫϥεϝιουͷݺͼग़͠Λ؆ ୯ʹॻ͚Δ

Slide 9

Slide 9 text

"DUJWF4VQQPSU$PODFSOͷྫᶄ module M1 def self.included(base) base.extend(ClassMethods) end module ClassMethods def m1_class_method 'M1!' end end end module M2 include M1 def self.included(base) base.extend(ClassMethods) end module ClassMethods def m2_class_method m1_class_method + ' M2!' end end end class C include M2 end C.m2_class_method #=> NameError: undefined local variable or method `m1_class_method' for C:Class $PODFSOͳ͠ w JODMVEF͢ΔϞδϡʔϧͷґଘؔ܎Λ؅ཧͯ͘͠ΕΔ module M1 extend ActiveSupport::Concern class_methods do def m1_class_method 'M1!' end end end module M2 extend ActiveSupport::Concern include M1 class_methods do def m2_class_method m1_class_method + ' M2!' end end end class C include M2 end C.m2_class_method #=> "M1! M2!" $PODFSO͋Γ

Slide 10

Slide 10 text

Ͱ͸ɺૣ଎ "DUJWF4VQQPSU$PODFSOͷ ίʔυΛಡΜͰΈ·͠ΐ͏

Slide 11

Slide 11 text

1 module ActiveSupport 2 module Concern 3 class MultipleIncludedBlocks < StandardError #:nodoc: 4 def initialize 5 super "Cannot define multiple 'included' blocks for a Concern" 6 end 7 end 8 9 def self.extended(base) #:nodoc: 10 base.instance_variable_set(:@_dependencies, []) 11 end 12 13 def append_features(base) 14 if base.instance_variable_defined?(:@_dependencies) 15 base.instance_variable_get(:@_dependencies) << self 16 return false 17 else 18 return false if base < self 19 @_dependencies.each { |dep| base.include(dep) } 20 super 21 base.extend const_get(:ClassMethods) if const_defined?(:ClassMethods) 22 base.class_eval(&@_included_block) if instance_variable_defined?(:@_included_block) 23 end 24 end 25 26 def included(base = nil, &block) 27 if base.nil? 28 raise MultipleIncludedBlocks if instance_variable_defined?(:@_included_block) 29 30 @_included_block = block 31 else 32 super 33 end 34 end 35 36 def class_methods(&class_methods_module_definition) 37 mod = const_defined?(:ClassMethods, false) ? 38 const_get(:ClassMethods) : 39 const_set(:ClassMethods, Module.new) 40 41 mod.module_eval(&class_methods_module_definition) 42 end 43 end 44 end

Slide 12

Slide 12 text

ͨͬͨߦʂ ۭߦൈ͘ͱߦʂ

Slide 13

Slide 13 text

1 module ActiveSupport 2 module Concern 3 class MultipleIncludedBlocks < StandardError #:nodoc: 4 def initialize 5 super "Cannot define multiple 'included' blocks for a Concern" 6 end 7 end 8 9 def self.extended(base) #:nodoc: 10 base.instance_variable_set(:@_dependencies, []) 11 end 12 13 def append_features(base) 14 if base.instance_variable_defined?(:@_dependencies) 15 base.instance_variable_get(:@_dependencies) << self 16 return false 17 else 18 return false if base < self 19 @_dependencies.each { |dep| base.include(dep) } 20 super 21 base.extend const_get(:ClassMethods) if const_defined?(:ClassMethods) 22 base.class_eval(&@_included_block) if instance_variable_defined?(:@_included_block) 23 end 24 end 25 26 def included(base = nil, &block) 27 if base.nil? 28 raise MultipleIncludedBlocks if instance_variable_defined?(:@_included_block) 29 30 @_included_block = block 31 else 32 super 33 end 34 end 35 36 def class_methods(&class_methods_module_definition) 37 mod = const_defined?(:ClassMethods, false) ? 38 const_get(:ClassMethods) : 39 const_set(:ClassMethods, Module.new) 40 41 mod.module_eval(&class_methods_module_definition) 42 end 43 end 44 end

Slide 14

Slide 14 text

$PODFSOͷίʔυΛಡΉᶃ Ϟδϡʔϧ͕"DUJWF4VQQPSU$PODFSOΛFYUFOE͢Δͱ TFMGFYUFOEFE͕ݺͼग़͞ΕΔ FYUFOE͞ΕͨϞδϡʔϧʹ!@EFQFOEFODJFTΛఆٛ͠ɺۭ഑ྻ Λ୅ೖ͢Δ ˞͜ΕҎ߱"DUJWF4VQQPSU$PODFSO͕FYUFOE͞ΕͨϞδϡʔϧͷ͜ͱΛDPODFSOͱදه͠·͢ def self.extended(base) #:nodoc: base.instance_variable_set(:@_dependencies, []) end

Slide 15

Slide 15 text

1 module ActiveSupport 2 module Concern 3 class MultipleIncludedBlocks < StandardError #:nodoc: 4 def initialize 5 super "Cannot define multiple 'included' blocks for a Concern" 6 end 7 end 8 9 def self.extended(base) #:nodoc: 10 base.instance_variable_set(:@_dependencies, []) 11 end 12 13 def append_features(base) 14 if base.instance_variable_defined?(:@_dependencies) 15 base.instance_variable_get(:@_dependencies) << self 16 return false 17 else 18 return false if base < self 19 @_dependencies.each { |dep| base.include(dep) } 20 super 21 base.extend const_get(:ClassMethods) if const_defined?(:ClassMethods) 22 base.class_eval(&@_included_block) if instance_variable_defined?(:@_included_block) 23 end 24 end 25 26 def included(base = nil, &block) 27 if base.nil? 28 raise MultipleIncludedBlocks if instance_variable_defined?(:@_included_block) 29 30 @_included_block = block 31 else 32 super 33 end 34 end 35 36 def class_methods(&class_methods_module_definition) 37 mod = const_defined?(:ClassMethods, false) ? 38 const_get(:ClassMethods) : 39 const_set(:ClassMethods, Module.new) 40 41 mod.module_eval(&class_methods_module_definition) 42 end 43 end 44 end

Slide 16

Slide 16 text

.PEVMFBQQFOE@GFBUVSFT w .PEVMFJODMVEFͷ࣮ମ w JODMVEF͞ΕͨϞδϡʔϧ͕ܧঝνΣʔϯʹؚ·Ε͍ͯͳ͚Ε͹ܧ ঝνΣʔϯʹϞδϡʔϧΛ௥Ճ͢Δ w ΦʔόʔϥΠυ͢ΔͱϞδϡʔϧ͕JODMVEF͞Εͳ͘ͳΔ module M def self.append_features end end class C include M end C.ancestors #=> [C, Object, Kernel, BasicObject]

Slide 17

Slide 17 text

$PODFSOͷίʔυΛಡΉᶄ !@EFQFOEFODJFT͕ఆٛ͞Ε͍ͯΕ͹ʢDPODFSOͰ͋Ε͹ʣɺ !@EFQFOEFODJFTʹࣗ෼ࣗ਎Λ௥Ճͯ͠GBMTFΛฦ͢JODMVEF͠ͳ͍ DPODFSOͰͳ͍৔߹ɺܧঝνΣʔϯʹࣗ෼ࣗ਎͕௥Ճ͞Ε͍ͯͳ͚Ε͹ɺґଘ ͨ͠DPODFSOΛ࠶ؼతʹJODMVEF͍͖ͯ͠ɺ࠷ޙʹTVQFSͰܧঝνΣʔϯʹࣗ ෼ࣗ਎Λ௥Ճ͢Δ $MBTT.FUIPETϞδϡʔϧΛFYUFOE͢Δ JODMVEFEʹ౉ͨ͠ϒϩοΫΛJODMVEFͨ͠ଆͷΫϥεͷίϯςΩετͰ࣮ߦ͢Δ def append_features(base) if base.instance_variable_defined?(:@_dependencies) base.instance_variable_get(:@_dependencies) << self return false else return false if base < self @_dependencies.each { |dep| base.include(dep) } super base.extend const_get(:ClassMethods) if const_defined?(:ClassMethods) base.class_eval(&@_included_block) if instance_variable_defined?(:@_included_block) end end

Slide 18

Slide 18 text

1 module ActiveSupport 2 module Concern 3 class MultipleIncludedBlocks < StandardError #:nodoc: 4 def initialize 5 super "Cannot define multiple 'included' blocks for a Concern" 6 end 7 end 8 9 def self.extended(base) #:nodoc: 10 base.instance_variable_set(:@_dependencies, []) 11 end 12 13 def append_features(base) 14 if base.instance_variable_defined?(:@_dependencies) 15 base.instance_variable_get(:@_dependencies) << self 16 return false 17 else 18 return false if base < self 19 @_dependencies.each { |dep| base.include(dep) } 20 super 21 base.extend const_get(:ClassMethods) if const_defined?(:ClassMethods) 22 base.class_eval(&@_included_block) if instance_variable_defined?(:@_included_block) 23 end 24 end 25 26 def included(base = nil, &block) 27 if base.nil? 28 raise MultipleIncludedBlocks if instance_variable_defined?(:@_included_block) 29 30 @_included_block = block 31 else 32 super 33 end 34 end 35 36 def class_methods(&class_methods_module_definition) 37 mod = const_defined?(:ClassMethods, false) ? 38 const_get(:ClassMethods) : 39 const_set(:ClassMethods, Module.new) 40 41 mod.module_eval(&class_methods_module_definition) 42 end 43 end 44 end

Slide 19

Slide 19 text

म০ w ϝιουͷ࠷ޙʹ෇͖ͷҾ਺Λ௥Ճ͢Δ͜ͱͰɺϝιουʹϒϩο ΫΛ౉͢͜ͱ͕Ͱ͖Δ w Λ෇͚ͨ··ͩͱϒϩοΫͱͯ͠ɺΛ෇͚ͳ͍ͱ1SPDͱͯ͠࢖ ͑Δ def my_method(&hoge) hoge end a = my_method {|name| "Hello, #{name}!" } a.class #=> Proc a.call("Ebisu.rb") #=> "Hello, Ebisu.rb!"

Slide 20

Slide 20 text

$PODFSOͷίʔυΛಡΉᶅ !@JODMVEFE@CMPDLʹJODMVEFEʹ౉͞ΕͨϒϩοΫͷ1SPDΦϒ δΣΫτΛ୅ೖ͢Δ def included(base = nil, &block) if base.nil? raise MultipleIncludedBlocks if instance_variable_defined?(:@_included_block) @_included_block = block else super end end def class_methods(&class_methods_module_definition) mod = const_defined?(:ClassMethods, false) ? const_get(:ClassMethods) : const_set(:ClassMethods, Module.new) mod.module_eval(&class_methods_module_definition) end $MBTT.FUIPETϞδϡʔϧʹDMBTT@NFUIPETʹ౉͞ΕͨϒϩοΫͷ ಺༰Λ௥Ճ͢Δ

Slide 21

Slide 21 text

Ҏ্ʂ

Slide 22

Slide 22 text

·ͱΊ w "DUJWF4VQQPSU$PODFSO͕Ͳ͏࣮ݱ͞Ε͍ͯΔ͔Λ࣮ࡍ ʹίʔυΛಡΜͰֶͼɺ3VCZ΁ͷཧղΛਂΊͨ w 3BJMTΛ࢖͍ͬͯͯ͜ΕͬͯͲ͏΍࣮ͬͯݱͯ͠ΔΜͩΖ ͏ͱࢥͬͨΒɺ3BJMTͷίʔυΛಡΜͰΈΔͱ͍ΖΜͳൃ ݟ͕͋Δ