[comp.lang.c++] Data hiding with derived classes

haydens@natasha.juliet.ll.mit.edu (Hayden Schultz) (05/22/91)

I want to create a class Sample which will hide the real data type
(byte, int, float, complex, vector, etc.) of my samples, but will
allow certain arithmetic operatons to be performed without my
application software knowing about the hidden data types.

From what I've read, I think that templates should do a very good job
of this. I'm somewhat reluctant to build a major amount of software on
an "experimental feature." An even more compelling reason for avoiding
templates is that none of my c++ compilers (Saber and Sun CC) support
it yet.

Here's a simple code fragment that illustrates the kind of ideas I've
been fooling around with. I realize that the Sample:: functions are
pretty bad ways to do things, they're just there to get the code to
hobble along until the basic scheme is fleshed out.

class Sample {
  public:
    Sample() {}
    ~Sample() {}
    virtual 		operator float() const {
	return(this->operator float());
    }
    virtual Sample	&operator=(const float f) {
	return(this->operator=(f));
    }
    virtual void	operator+=(const Sample &s) {
	this->operator+=(s);
    }
};


class FloatSample : public Sample {
    float	fval;
  public:
    FloatSample() { fval = 0.; }
    FloatSample(float f) { fval = f; }
    ~FloatSample() {}

    		operator float() const { return(fval); }
    Sample	&operator=(const float f);
    void	operator+=(const Sample &s);
};


Sample	operator+(const Sample &s1, const Sample &s2)
{
    Sample	sum = s1;

    sum += s2;
    return(sum);
}

void	FloatSample::operator+=(const Sample &s) {
    fval += float(s);  // this is where the error occurs on the 2nd addition
}

Sample	&FloatSample::operator=(const float f)
{
    fval = f;
    return(*this);
}

main()
{
    Sample	*i = new FloatSample(0.);
    Sample	*j = new FloatSample(1.);
    Sample	*k = new FloatSample(2.);
    Sample	l;
    
    l = (*i) + (*j) + (*k);
}

The problem which bothers me, is that the first addition (*i) + (*j)
works, and the result is placed in a temp Sample. The second addition
of the temp Sample and (*k) fails because, although the virtual table
pointer of the temp points to the FloatSample functions, it's really
a Sample.

It occured to me to put a pointer (say a void *) to the data so that
sizeof(Sample) == sizeof(FloatSample). But I don't really think that's
a good solution. If a user defined class comes along:

class UserSample : public Sample {
    int		size;
  public:
.....
};

Now sizeof(Sample) != sizeof(UserSample), and the same problem will
arise.

So what's a good way of doing this without using templates? Please
email a copy of any responses to haydens@juliet.ll.mit.edu

	Thanks,

	Hayden Schultz (haydens@juliet.ll.mit.edu)
	MIT Lincoln Lab