Today Dynamic dispatch aka late binding aka virtual method calls – Call to self.m2() in method m1 defined in class C can resolve to a method m2 defined in a subclass of C CSE341: Programming Languages – Important characteristic of OOP Need to define the semantics of objects and method lookup as Late Binding in Ruby carefully as we defined variable lookup for functional programming Multiple Inheritance, Interfaces, Mixins Then consider advantages, disadvantages of dynamic dispatch Recall earlier encoding OOP / dynamic dispatch with functions in Alan Borning Racket (bank account example) Spring 2018 Resolving identifiers Ruby instance variables and methods • self maps to some "current" object The rules for "looking up" various symbols in a programming language is a key part of the language's definition • Look up local variables in environment of method – So discuss in general before considering dynamic dispatch • Look up instance variables using object bound to self • Look up class variables using object bound to self.class • Haskell: Look up variables in the appropriate environment – Key point of closures' lexical scope is defining "appropriate” A syntactic distinction between local/instance/class means there is no ambiguity or shadowing rules • Racket: Like Haskell plus hygienic macros – Contrast: Java locals shadow fields unless use this.f • Ruby: But there is ambiguity/shadowing with local variables and zero- – Local variables and blocks mostly like Haskell and Racket argument no-parenthesis calls – But also have instance variables, class variables, and – What does m+2 mean? methods (all more like record fields) • Local shadows method if exists unless use m()+2 • Contrast: Java forces parentheses for syntactic distinctions 1 Method names are different Ruby message lookup The semantics for method calls aka message sends • self, locals, instance variables, class variables all map to objects e0.m(e1,…,en) 1. Evaluate e0, e1, …, en to objects obj0, obj1, …, objn • Have said "everything is an object" but that's not quite true: – As usual, may involve looking up self, variables, fields, etc. – Method names 2. Let C = the class of obj0 (every object has a class) – Blocks 3. If m is defined in C, pick that method, else recur with the superclass – Argument lists of C unless C is already Object – If no m is found, call method_missing instead • First-class values are things you can store, pass, return, etc. • Definition of method_missing in Object raises an error – In Ruby, only objects (almost everything) are first-class 4. Evaluate body of method picked: – Example: cannot do e.(if b then m1 else m2 end) – With formal arguments bound to obj1, …, objn • Have to do if b then e.m1 else e.m2 end – With self bound to obj0 -- this implements dynamic dispatch! – Example: can do (if b then x else y).m1 Note: Step (3) complicated by mixins: will revise definition later Java method lookup (very similar) The punch-line again The semantics for method calls aka message sends e0.m(e1,…,en) e0.m(e1,…,en) 1. Evaluate e0, e1, …, en to objects obj0, obj1, …, objn – As usual, may involve looking up this, variables, fields, etc. To implement dynamic dispatch, evaluate the method body with 2. Let C = the class of obj0 (every object has a class) self mapping to the receiver 3. [Complicated rules to pick "the best m" using the static types of e0, e1, …, en] • That way, any self calls in the body use the receiver's class, – Static checking ensures an m, and in fact a best m, will always be – Not necessarily the class that defined the method found – Rules similar to Ruby except for this static overloading • This much is the same in Ruby, Java, C#, Smalltalk, etc. – No mixins to worry about (interfaces irrelevant here) 4. Evaluate body of method picked: – With formal arguments bound to obj1, …, objn – With this bound to obj0 -- this implements dynamic dispatch! 2 Comments on dynamic dispatch A simple example In Ruby (and other object-oriented languages), subclasses can • This is why last lecture's distFromOrigin2 worked in change the behavior of methods they don't override PolarPoint class A – distFromOrigin2 implemented with self.x, self.y def even x – If receiver's class is PolarPoint, then will use PolarPoint's if x==0 then true else odd (x-1) end x and y methods because self is bound to the receiver end def odd x if x==0 then false else even (x-1) end • More complicated than the rules for closures end – Have to treat self specially end – May seem simpler only because you learned it first class B < A # improves odd in B objects – Complicated doesn't imply superior or inferior def even x ; x % 2 == 0 end end • Depends on how you use it… class C < A # breaks odd in C objects • Overriding does tend to be overused def even x ; false end end The OOP trade-off What next? Any method that makes calls to overridable methods can have its Have used classes for OOP's essence: inheritance, overriding, behavior changed in subclasses even if it is not overridden dynamic dispatch – Maybe on purpose, maybe by mistake Now, what if we want to have more than just 1 superclass • Makes it harder to reason about "the code you're looking at" • Multiple inheritance: allow > 1 superclasses – Can avoid by disallowing overriding (Java final) of helper – Useful But has some proBlems (see C++) methods you call • Java-style interfaces: allow > 1 types – Mostly irrelevant in a dynamically typed language, But fewer • Makes it easier for subclasses to specialize behavior without problems copying code – Provided method in superclass isn't modified later • RuBy-style mixins: 1 superclass; > 1 method providers – Often a fine suBstitute for multiple inheritance and has fewer problems 3 Multiple Inheritance Trees, dags, and diamonds • If inheritance and overriding are so useful, why limit ourselves to one • Note: The phrases subclass, superclass can be ambiguous superclass? – There are immediate subclasses, superclasses – Because the semantics is often awkward (next couple of slides) – And there are transitive subclasses, superclasses – Because it makes static type-checking harder (not discussed) A – Because it makes efficient implementation harder (not discussed) • Single inheritance: the class hierarchy is a tree – Nodes are classes B C D • Is it useful? Sure! – Parent is immediate superclass E – Example: Make a ColorPt3D by inheriting from Pt3D and – Any number of children allowed (or maybe just from ) ColorPt Color X – Example: Make a StudentAthlete by inheriting from Student • Multiple inheritance: the class hierarchy no longer a tree and Athlete W – Cycles still disallowed (a directed-acyclic graph) V – With single inheritance, end up copying code or using non-OOP- Z style helper methods – If multiple paths show that X is a (transitive) superclass of Y, then we have diamonds Y What could go wrong? X 3DColorPoints W V If Ruby had multiple inheritance, we would want ColorPt3D to • If V and Z both define a method m, Z inherit methods that share one @x and one @y what does Y inherit? What does super mean? Y class Pt – Directed resends useful (e.g., Z::super) attr_accessor :x, :y … • What if X defines a method m that Z but not V overrides? end class ColorPt < Pt – Can handle like previous case, but sometimes undesirable attr_accessor :color (e.g., wants 's overrides to "win") ColorPt3D Pt3D … end • If X defines fields, should Y have one copy of them (f) or two class Pt3D < Pt (V::f and Z::f)? attr_accessor :z – Turns out each behavior is sometimes desirable (next slides) … # override methods like distance? end – So C++ has (at least) two forms of inheritance class ColorPt3D < Pt3D, ColorPt # not Ruby! end 4 ArtistCowboys Java interfaces This code has Person define a pocket for subclasses to use, but Recall (?), Java lets us define interfaces that classes explicitly an ArtistCowboy wants two pockets, one for each draw method implement class Person interface Example { attr_accessor :pocket void m1(int x, int y); … Object m2(Example x, String y); end } class Artist < Person # pocket for brush objects def draw # access pocket class A implements Example { … public void m1(int x, int y) {…} end public Object m2(Example e, String s) {…} class Cowboy < Person # pocket for gun objects } def draw # access pocket class B implements Example { … public void m1(int pizza, int beer) {…} end public Object m2(Example e, String s) {…} class ArtistCowboy < Artist, Cowboy # not Ruby! } end What is an interface? Multiple interfaces interface Example { void m1(int x, int y); • Java classes can implement any number of interfaces Object m2(Example x, String y); } • Because interfaces provide no methods or fields, no questions of • An interface is a type! method/field duplication arise – Any implementer (including subclasses) is a subtype of it – No problem if two interfaces both require of implementers and promise to clients the same method – Can use an interface name wherever a type appears – (In Java, classes are also types in addition to being classes) • Such interfaces aren't much use in a dynamically typed language • An implementer type-checks if it defines the methods as required – We don't type-check implementers – Parameter names irrelevant to type-checking; it's a bit strange that Java requires them in interface definitions – We already allow clients to send any message • A user of type Example can objects with that type have the – Presumably these types would change the meaning of is_a?, methods promised but we can just use instance_methods to find out what methods an object has – I.e., sending messages with appropriate arguments type-checks 5 Why no interfaces in C++? Mixins If you have multiple inheritance and abstract methods (called pure • A mixin is (just) a collection of methods virtual methods in C++), there is no need for interfaces – Less than a class: no fields, constructors, instances, etc.
Details
-
File Typepdf
-
Upload Time-
-
Content LanguagesEnglish
-
Upload UserAnonymous/Not logged-in
-
File Pages6 Page
-
File Size-