Qore Programming Language

  • Increase font size
  • Default font size
  • Decrease font size

Multiple Inheritance Example

E-mail Print PDF
Example script demonstrating how multiple inheritance works in Qore.
#!/usr/bin/env qore
#
# this file gives examples of how class inheritance in qore works
#
# QORE INHERITANCE
# when a class inherits another class, all of the methods of the base
# class are available in the subclass (inherting class).  If the
# subclass defines methods with the same names as methods in base
# classes, then the subclasses methods will override the base class
# methods.
#
# MULTIPLE INHERITANCE
# a class can inherit from more than one class (called multiple
# inheritance).  In this case, if there are base class methods with the
# same names, the class listed first in the inheritance list will take
# precedence over classes listed afterwards (breadth-first, left-to-
# right resolution order)
#
# "SPECIAL" METHODS, ::constructor(), ::destructor(), and ::copy() base
# class methods will be automatically called when the equivalent
# subclass methods are executed.
# In this case, a depth-first, left-to-right order is used for
# constructor and copy methods, which means that the lowest-level
# base class constructors are called first, using a left-to-right
# order (the declaration order) for classes at the same level.
# destructors will be called in the reverse order.
#
# NOTE: base class special methods will only be called once, even if
# the class has been inherited multiple times in the hierarchy (for
# example, if a subclass "Final" inherits classes "Base1" and "Mid",
# but "Base1" is already a subclass of class "Mid", the constructor
# for "Base1" will only be executed once when the Final::constructor()
# method is executed
#
# NOTE: base class constructor arguments can be overridden by subclasses.
# if a subclass specifies arguments for base class, and another inherited
# class also specifies arguments for the base class, the lowest level
# class' base constructor argument specification will be used (and the
# other class' base class arguments will not be evaluated in this case)
#
# NOTE: the other "special" methods ::methodGate() and ::memberGate()
# are not called for base classes, subclasses have to define their own
# versions of these methods.  However, these special methods can be
# explicitly called from subclasses

# we require qore >= 0.8 because we use the cast<> operator below
%requires qore >= 0.8

# require global variables to be declared with "our" before use
%require-our
%enable-all-warnings

# Base1 will be a base class
class Base1 {
    # declare some private members
    private $.base1, $.x;

    constructor($a) {
	printf("Base1::constructor(%n)\n", $a);
	$.a = $a;
    }

    destructor() {
	printf("Base1::destructor() (%n)\n", $.a);
    }

    copy() {
	printf("Base1::copy() (%n)\n", $.a);
	$.a = $.a + "-copy";
    }

    hello() {
	# note that here to call method "subclass()" in a derived class, we have
	# to use $self.subclass() instead of $.subclass(), because calls like
	# "$.subclass()" are resolved at parse time (which would resolve to run
	# Base::subclass(), which is not what we want), but "$self.subclass()"
	# is resolved at run-time, and so will run the "subclass" method in the
	# derived class

	# to avoid a warning about the unknown method, we use the cast<> operator
	printf("Base1 hello (%n, derived class %n)\n", $.a, cast<Mid>($self).subclass());
    }

    # note that "Base1" here is just a regular method, the constructor
    # is always named "constructor"
    Base1($a) {
	printf("Base1::Base1() (%n) %n\n", $.a, $a);
    }

    # "static" functions are not associated with an object; they are
    # like regular functions and can be called outside the class (if
    # they are not private)
    static test() {
	printf("Base1::test() static function\n");
    }
}

# Base2 is a base class
class Base2 {
    # declare some private members
    private $.base2, $.y;

    constructor($a) {
	printf("Base2::constructor(%n)\n", $a);
	$.b = $a;
    }
    copy() {
	printf("Base2::copy() (%n)\n", $.b);
	$.b = $.b + "-copy";
    }
    destructor() {
	printf("Base2::destructor() (%n)\n", $.b);
    }
    hello() {
	printf("Base2 hello (%n)\n", $.b);
    }

    # note that "Base2" here is just a regular method, the constructor
    # is always named "constructor"
    Base2($a) {
	printf("Base2::Base2() (%n) %n\n", $.b, $a);
    }
}

# Mid is a subclass of Base1 and Base2 (and will be a base class for
# class Final)
namespace Mid;
class Mid::Mid inherits Base1, Base2
{
    # declare some private members
    private $.mid, $.z;

    constructor($m) : Base1("Mid->Base1-" + $m), Base2("Mid->Base2-" + $m) {
	printf("Mid::constructor(%n)\n", $m);
	$.m = $m;
    }
    copy() {
	printf("Mid::copy() (%n)\n", $.m);
	$.m = $.m + "-copy";
    }
    destructor() {
	printf("Mid::destructor() (%n)\n", $.m);
    }
    hello() {
	Base1::$.hello();
	Base2::$.hello();
	printf("Mid hello (%n)\n", $.m);
    }
    subclass() {
        return "Mid";
    }
    # note that "Mid" here is just a regular method, the constructor
    # is always named "constructor"
    Mid($a) {
	$.Base1("Mid");
	$.Base2("Mid");
	printf("Mid::Mid() (%n) %n\n", $.m, $a);
    }
}

# class Final is a subclass of Mid, Base1, and Base2 (even though
# Base1 and Base2 are already subclasses of Mid)
# private members of this class: $.base1, $.base2, $.x, $.y, $.z
class Final inherits Mid::Mid, Base1, Base2;

Final::constructor($a) : Mid("Final->Mid-" + $a), Base1("Final->Base1-" + $a) {
    printf("Final::constructor(%n)\n", $a);
    $.f = $a;
}

Final::copy() {
    printf("Final::copy() (%n)\n", $.f);
    $.f = $.f + "-copy";
}

Final::destructor() {
    printf("Final::destructor() (%n)\n", $.f);
}

Final::subclass() {
    return "Final";
}

Final::hello() {
    # here we make a copy of ourselves and print out the result
    my $x = $.copy();
    printf("x=%N\n", $x);

    Mid::$.hello();
    Base1::$.hello();
    Base2::$.hello();
    $.Base1("Final");
    $.Base2("Final");
    $.Mid("Final");
    printf("Final hello\n");
}

# test constructors
#$m = new Mid("Mid");
#thread_exit;
our $a = new Final("Final");
# execute a subclass method
$a.hello();
# test that base class methods are available in the subclass
$a.Base1("top");
$a.Base2("top");
$a.Mid("top");

# objects are always references (unless explicitly copied), so $b and $a will represent the same object after the following assignment
our $b = $a;

# however, we can make an explicit copy
our $c = $a.copy();

our $x = new Final("hi");
# execute destructors as objects go out of scope
Last Updated on Saturday, 25 December 2010 20:19