[comp.lang.c++] C++ typing not so strong

cok@islsun.Kodak.COM (David Cok) (02/13/91)

I and others recently completed a package of C++ classes in which we tried to
provide both strong typing and inheritance and we ran into the following
situation, in which the two requirements conflicted.

We had a base class Image from which some derived classes DoubleImage,
FloatImage, IntImage etc. were derived.  We defined lots of relevant 
functions including (for example) operator- .

Now intuitively operator- applied to a DoubleImage produces a new
DoubleImage value and similarly for the other derived classes.  With this and
strong typing in mind, we defined member functions of the form

		const DoubleImage& DoubleImage::operator-() const;
		const FloatImage&  FloatImage::operator-() const;

But we also want some "virtualness" here.  If we apply operator- to an Image
we should get a new Image, having applied the correct - operator depending on
the runtime derived type.  That is we want:

		const Image& Image::operator-() const;

The resulting partial program is

Example A:

class Image {
    public:
	virtual const Image& operator-() const = 0;
};

class DoubleImage: public Image {
    public:
	const DoubleImage& operator-() const;
};

BUT THIS IS ILLEGAL because the two operator- declarations do not return the
same type.

The best way we could find to give our desired interface was a kludge like this:

Example B:

class Image {
    public:
	virtual const Image& operator_minus() const = 0;
	inline const Image& operator-() const
		{ return this->operator_minus(); }
};

class DoubleImage {
    public:
	const Image& operator_minus() const; // the real work is here
	inline const DoubleImage& operator-() const
		{ return (DoubleImage&)(((Image*)this)->operator_minus());}
};

which has type-unsafe casts and twice the number of functions.

Question 1: Does anyone have a better suggestion?
Question 2: What is the rationale for requiring that the implementations of
	virtual functions in derived classes return exactly the same type as is
	declared in the base class?  It would seem sufficient to require
	that the declaration in a derived class return a type which can be
	converted (using at least some set of implicit conversions) to that 
	used in the base class.  For example,
	if the base class returns Image& the derived class may return
	DoubleImage& ; if the base class returns Image*, the derived class
	may return DoubleImage* .

This simple relaxation of the rule would not break any existing C++
code.  The compiler can type-check everything just as well, and it would
help avoid the problem of needing casts ("casts considered harmful...")
to avoid types drifting to the top of the inheritance tree.
It would have simplified our code above greatly -- almost
every function needed a virtual internal function so that we could
get the correct interface and virtual behavior as well.

It does require the compiler to do a little work.  Consider the rule relaxed to
allow at least the pointer and reference conversions indicated above, and add to
example A above the corresponding declaration of FloatImage::operator-().
Applying operator- to a FloatImage& would simply return a FloatImage&.  Applying
operator- to an Image& needs to return an Image&.  The compiler needs to create
a version of FloatImage::operator- which converts its FloatImage& result to an 
Image& and a version of DoubleImage::operator- which converts its DoubleImage&
result to an Image& (change all references to pointers if you like).  For these
conversions that is trivial; more complicated conversions (which may not be
allowed) or the presence of MI would mean some real work had to be done. In
effect, the compiler converts example A into example B, but does so in a 
type-safe manner.  When the programmer does it with casts, type-checking is 
lost.  

David R. Cok
Eastman Kodak Company
cok@Kodak.COM
716-477-7086

schweitz@lexvmc.iinus1.ibm.com ("Eric Schweitz") (02/19/91)

| David R. Cok
| Eastman Kodak Company
| cok@Kodak.COM
| 716-477-7086

writes:

| We had a base class Image from which some derived classes DoubleImage,
| FloatImage, IntImage etc. were derived.  We defined lots of relevant
| functions including (for example) operator- .
|
| ... we defined member functions of the form
|
|                 const DoubleImage& DoubleImage::operator-() const;
|                 const FloatImage&  FloatImage::operator-() const;
|
| But we also want some "virtualness" here.

Perhaps you could tell us why you `needed' these member functions to be
virtual.

| Example A:
|
| class Image {
|     public:
|         virtual const Image& operator-() const = 0;
| };
|
| class DoubleImage: public Image {
|     public:
|         const DoubleImage& operator-() const;
| };
|
| BUT THIS IS ILLEGAL because the two operator- declarations do not return the
| same type.

You're right here. However, if ``operator-'' had been declared as a
non-`pure virtual' member function, you would have simple overloading of
the unary ``-'' operator.

| Question 1: Does anyone have a better suggestion?

Yeah, don't make the operator- a pure virtual function if you don't have to.

| Question 2: What is the rationale for requiring that the implementations of
|         virtual functions in derived classes return exactly the same type as is
|         declared in the base class?

The rationale behind this is that it would make the following possible :

   //....
   Image i, *ip;
   DoubleImage *dp;

   ip = &i;
   dp = func ((DoubleImage*) ip);
   //....

where, func () is:

   //....
   DoubleImage* func (DoubleImage* p)
   {
      return p->operator-();
   }
   //....

hence, func() would return a Image* to dp which is expecting a DoubleImage*.

Schweitz.   schweitz@lexvmc.iinus1.ibm.com

cok@islsun.Kodak.COM (David Cok) (02/19/91)

In article <9102182022.AA05904@ucbvax.Berkeley.EDU> schweitz@lexvmc.iinus1.ibm.com ("Eric Schweitz") writes:
>| David R. Cok
>| Eastman Kodak Company
>| cok@Kodak.COM
>| 716-477-7086
>
>writes:
>
>| We had a base class Image from which some derived classes DoubleImage,
>| FloatImage, IntImage etc. were derived.  We defined lots of relevant
>| functions including (for example) operator- .
>|
>| ... we defined member functions of the form
>|
>|                 const DoubleImage& DoubleImage::operator-() const;
>|                 const FloatImage&  FloatImage::operator-() const;
>|
>| But we also want some "virtualness" here.
>
>Perhaps you could tell us why you `needed' these member functions to be
>virtual.
>
>
>
> ...
>

Craig Hubley already responded to the theory here: compile-time overloading
vs runtime virtual function resolution, so I'll just provide the pragmatic
justification for wanting an unusual thing like virtual functions.

We were putting together a package to support image processing and hence had
classes DoubleImage, FloatImage, IntImage etc derived from Image.  Some
operations apply only to specific classes: bit twiddling on IntImages for
example.  Others apply to all the derived classes (+ - negation * / etc.).
If one knows at compile time that an Image has a certain pixel data type, one
wants the compiler to do type checking -- hence strong typing.  However,
often one does not know the pixel data type at compile time, such as if the 
image were read in from a disk file.  Then it must be passed around as an
Image* and virtual functions are needed to act upon it.  Obviously, if one is
going to do any type-specific operations (e.g. bit anding) one has to check
the dynamic type somehow at run-time.  But, one can write useful routines
which use only virtual functions (e.g. convolution: pad around the edge of
the image with 0's (or whatever); loop over the image taking sums and products
of neighboring pixel elements).  If one were restricted to compile-time
overloading, this routine would have to be duplicated -- until templates come
around -- for every derived type.

Is that enough to `need' virtual member functions?

David R. Cok
Eastman Kodak Company  cok@Kodak.COM

craig@gpu.utcs.utoronto.ca (Craig Hubley) (02/20/91)

In article <1991Feb19.124328.23934@kodak.kodak.com> cok@islsun.Kodak.COM (David Cok) writes:
>In article <9102182022.AA05904@ucbvax.Berkeley.EDU> schweitz@lexvmc.iinus1.ibm.com ("Eric Schweitz") writes:
>>| We had a base class Image from which some derived classes DoubleImage,
>>| FloatImage, IntImage etc. were derived.  We defined lots of relevant
>>| functions including (for example) operator- .
>>|
>>| ... we defined member functions of the form
>>|
>>|                 const DoubleImage& DoubleImage::operator-() const;
>>|                 const FloatImage&  FloatImage::operator-() const;
>>|
>>| But we also want some "virtualness" here.
>>
>>Perhaps you could tell us why you `needed' these member functions to be
>>virtual.
>> ...
>
>Craig Hubley already responded to the theory here: compile-time overloading
>vs runtime virtual function resolution, so I'll just provide the pragmatic
>justification for wanting an unusual thing like virtual functions.

"unusual" ?

>We were putting together a package to support image processing and hence had
>classes DoubleImage, FloatImage, IntImage etc derived from Image.  Some
>operations apply only to specific classes: bit twiddling on IntImages for
>example.  Others apply to all the derived classes (+ - negation * / etc.).
>If one knows at compile time that an Image has a certain pixel data type, one
>wants the compiler to do type checking -- hence strong typing.  However,
>often one does not know the pixel data type at compile time, such as if the 
>image were read in from a disk file.  Then it must be passed around as an
>Image* and virtual functions are needed to act upon it.  Obviously, if one is
>going to do any type-specific operations (e.g. bit anding) one has to check
>the dynamic type somehow at run-time.  But, one can write useful routines
>which use only virtual functions (e.g. convolution: pad around the edge of
>the image with 0's (or whatever); loop over the image taking sums and products
>of neighboring pixel elements).  If one were restricted to compile-time
>overloading, this routine would have to be duplicated -- until templates come
>around -- for every derived type.

You provide a good example of a capability that can be provided by templates
(parametrized polymorphism) or inheritance (hierarchical polymorphism).  It
seems to me that in the case that you describe, either could solve it.  But
which method leaves you more flexibility for extension ?  If the functions
you write would apply to any type, without needing to be adjusted for different
class hierarchies, then it is a good candidate for a template.  However, if
you are writing functions specific to a given class hierarchy, as you seem to
be here, you would like to take advantage of the relationship that these
types already have.  In other words, you would rather these functions be
members than be free, because they are really part of the type specification
rather than part of the application specification.  If you do this with
templates you will be hauling them around with the Image class hierarchy
and so they are effectively part of it.

I often find it necessary to use *all* the polymorphism power of C++
(overloading, conversion, public derivation) to provide a uniform and
consistent set of behaviors to the user.  Templates add some power to
that, but they are not a substitute for a unified resolution or assertion
mechanism.

>Is that enough to `need' virtual member functions?

In theory, sure.  In C++, apparently not.  :)  At least once templates are
widespread you will have a way to do this, even if (to my mind) it's not the
best way.  

>David R. Cok
>Eastman Kodak Company  cok@Kodak.COM


-- 
  Craig Hubley   "...get rid of a man as soon as he thinks himself an expert."
  Craig Hubley & Associates------------------------------------Henry Ford Sr.
  craig@gpu.utcs.Utoronto.CA   UUNET!utai!utgpu!craig   craig@utorgpu.BITNET
  craig@gpu.utcs.toronto.EDU   {allegra,bnr-vpa,decvax}!utcsri!utgpu!craig