Skip to content

Latest commit

 

History

History
146 lines (114 loc) · 5.42 KB

README.md

File metadata and controls

146 lines (114 loc) · 5.42 KB

Inheritance and "pure virtual" functions in Go

This article tries to answer the question: whether it is possible to simulate "the classic OOP" in Go.

And the short answer is: yes, it's possible. But in result we get lots of clumsy boilerplate code, so the author warns that it’s more of a theoretical possibility than a practically usable solution. Actually this code is just a first idea born during discussion of this article’s main question with one of my colleagues. Use it at your own risk :).

Assume we need to have this class hierarchy:

We know that there are no classes and no inheritance in Go, but internally the OOP concept is nothing more than a structure with fields and dispatch tables containing function pointers. From the other side in Go we have structs and first class functions (we can assign functions to variables). So in theory nothing prevents us from storing both the fields and a dispatch table in the same struct. Also we will need to link structs to simulate inheritance, this adds one extra field into each “child” structure.

To keep the method presented more or less simple we are allow to have only a single level of Parent->Child inheritance. Multiple levels of inheritance will require more complex logic to keep the correct instance pointer in the middle of hierarchy. Even with this simplification we still getting structures with lots of boilerplate:

In Shape we want to have a “pure virtual” function GetArea() float64 which is later implemented in both Circle and Square. Also on any instance of the Shape we should be able to call IsLargerThan(area float64) bool which then calls the correct version of GetArea().

Our Shape is very simple:

  type Shape struct {
    instance interface{}

    IsLargerThan func(this interface{}, area float64) bool
    GetArea func(this interface{}) float64
  }

It just holds a reference to actual Circle or Square in its instance field (remember, we support only single level of "inheritance"). IsLargerThan function will call the GetArea. Let``s construct the Shape:

  func NewShape(instance interface{}) *Shape {
    this := &Shape{}
    this.instance = instance
    this.IsLargerThan = ShapeIsLargerThan

    return this
  }

As our GetArea is "pure virtual" its value is nil after constructing Shape. But we call the GetArea from IsLargerThan:

  func ShapeIsLargerThan(this interface{}, area float64) bool {
    if shape, ok := this.(*Shape); ok {
      return shape.GetArea(shape.instance) > area
    } else {
      panic(fmt.Errorf("wrong type passed %T", this))
    }
  }

Here we also meet with boilerplate this parameter (yes, I know that it is a bad practice in Go to name it this or self, but we are simulating the "classic OOP"). Before calling the GetArea we are checking that the *Shape is actually passed to IsLargerThan. Then we getting a pointer to actual instance and make the call. It is obvious that our "descendants" should implement GetArea to allow this to work. Let's check how the Circle does that:

  func CircleGetArea(this interface{}) float64 {
    if circle, ok := this.(*Circle); ok {
      return math.Pi * circle.radius * circle.radius
    } else {
      panic(fmt.Errorf("wrong type passed %T", this))
    } 
  }

This function just converts the type of the parameter passed and then uses radius field to compute the area value. Complete definistion of Circle also includes the parent field and the dispatch table for both IsLargerThan and GetArea.

  type Circle struct {
    parent *Shape

    radius float64

    IsLargerThan func(this interface{}, area float64) bool
    GetArea func(this interface{}) float64
  }

On creation of the new Circle we need not only to store the radius, but also do all the work of our OOP manually:

  • create the Shape;
  • add the implementation of the "pure virtual" function to Skape``s dispatch table;
  • fill the dispatch table of Circle.
  func NewCircle(radius float64) *Circle {
    this := &Circle{}
    this.parent = NewShape(this)
    this.parent.GetArea = CircleGetArea
    this.radius = radius
    this.IsLargerThan = CircleIsLargerThan
    this.GetArea = CircleGetArea

    return this
  }

Our clumsy implementation of OOP forces us to "override" IsLargerThan in Circle to be able to convert the type and call the parent's one.

  func CircleIsLargerThan(this interface{}, area float64) bool {
    if circle, ok := this.(*Circle); ok {
      return circle.parent.IsLargerThan(circle.parent, area)
    } else {
      panic(fmt.Errorf("wrong type passed %T", this))
    }
  }

Additional note

From my point of view lots of boilerplate code is not as bad as the necessity of specifying an instance pointer everywhere (as a first parameter of each function call):

  area := 16.0

  square := NewSquare(5)
  fmt.Printf("%v is larger than %f: %t\n", square, area, square.IsLargerThan(square, area))

Uniform Function Call Syntax (UFCS) may help here, but unfortunately we don’t have it in Go (sometimes UFCS helps to write elegant code for example in Nim).