Constructor problem

C

Chiller

I am in the process of writing a class that will represent metric distances
by accepting a value (ie, 3) and a unit of measure (ie, m).

I've written my constructor in the .h file as

Distance (int, char);

I've written the constructor in the .cpp implementation file as:

Distance :: Distance ( int n, char m) : nu(n), me(m) {}

When I try to enter a value in main using:
Distance a = Distance (5, m);

I get an error saying the value "m" is undeclared, yet the 5 is accepted.
Considering I declared these variables in the initial constructor I can't
understand why I'm getting this error.

I'd appreciate it if someone could give me an explanation of what I'm doing
wrong and how I can correct my mistake.

Thanks
 
T

Thomas Tutone

I am in the process of writing a class that will represent metric distances
by accepting a value (ie, 3) and a unit of measure (ie, m).

I've written my constructor in the .h file as

Distance (int, char);

I've written the constructor in the .cpp implementation file as:

Distance :: Distance ( int n, char m) : nu(n), me(m) {}

When I try to enter a value in main using:
Distance a = Distance (5, m);

I get an error saying the value "m" is undeclared, yet the 5 is accepted.
Considering I declared these variables in the initial constructor I can't
understand why I'm getting this error.

I'd appreciate it if someone could give me an explanation of what I'm doing
wrong and how I can correct my mistake.

If you mean the literal character 'm', then you can do:

Distance a = Distance(5, 'm');

or, more concisely:

Distance a(5, 'm');

If you don't mean the literal character 'm', then you need to be more
specific about what it is you are trying to do. In particular, provide an
actual compilable piece of code - what you provided is not compilable as is.

Best regards,

Tom
 
C

Chiller

Is there a way that I can write the constructor so that it will simply
accept two variables, ie. when I create an object I can do so by simply
typing

Distance a = Distance (5, m); or
Distance a = Distance (5, km); etc

without having to use ' ' or declaring m, km?

Thanks
 
C

Chiller

Below I have included the .h file .cpp implementation file. The .cpp file
has a TEST_DISTANCE driver that needs to be declared to the preprocessor to
allow it to be compiled.

I'd greatly appreciate some advice on what I'm doing wrong and how I can
correct the problem.

Whenever I compile I get the following errors:

Distance.cpp(32) : error C2556: 'char Distance::measure(void) const' :
overloaded function differs only by return type from 'int
Distance::measure(void) const'

Distance.h(25) : see declaration of 'Distance::measure'

Distance.cpp(32) : error C2371: 'Distance::measure' : redefinition;
different basic types

Distance.h(25) : see declaration of 'Distance::measure'

Distance.cpp(39) : error C2264: 'Distance::measure' : error in function
definition or declaration; function not called



..h file as follows

**************************************************

#ifndef DISTANCE_H

#define DISTANCE_H

#include <iostream>

using namespace std;

class Distance

{

public :

Distance (int, char) ; // constructor - takes int and char values

Distance (int) ; // constructor - takes int value

Distance (void) ; // default - zero

//access member functions

int number (void) const;

int measure (void) const;


private :

int nu ; // the value

char me ; // the unit of measure (m)

} ;

// provide an overload of "<<" for easy display

ostream& operator<< (ostream&, const Distance&);

#endif



..cpp file as follows

*****************************************

#include "Distance.h"

#include <iostream>

#include <string>

using namespace std;

/*-------------------------------------------------------*\

| implementation of member functions |

\*-------------------------------------------------------*/

// constructor

Distance :: Distance (int n, char m) : nu(n), me(m) {}

Distance :: Distance (int n) : nu(n) {}

Distance :: Distance (void) : nu(0) {}

// access functions

int Distance :: number (void) const

{

return nu;

}

char Distance :: measure (void) const

{

return me;

}

// provide an overload of "<<" for easy display

ostream& operator<< (ostream& out, const Distance& d)

{

out << "(" << d.number() << "," << d.measure() << ")" ;

return out;

}

/*-------------------------------------------------------*\

| test driver for the Distance class |

\*-------------------------------------------------------*/

#ifdef TEST_DISTANCE // .... Distance class .... test driver

int main (void)

{

// create test input

Distance a = Distance (6);

Distance b (4);

Distance c (2);

Distance d;

Distance e (5, 'm');


cout << a << endl << b << endl << c << endl << d << endl << e << endl;



cin.ignore();

return 0; // normal termination

}

#endif
 
M

Mark A. Gibbs

Chiller said:
Is there a way that I can write the constructor so that it will simply
accept two variables, ie. when I create an object I can do so by simply
typing

Distance a = Distance (5, m); or
Distance a = Distance (5, km); etc

without having to use ' ' or declaring m, km?

yes, but really, i'm still not sure what exactly you're looking for.

it *sounds* like you want to do something more like this:

class DistanceUnit
{
private:
DistanceUnit(int value) :
v_(value),
n_(name)
{}

DistanceUnit(DistanceUnit const&);
void operator=(DistanceUnit const&);

int v_;

friend class Distance;
};

class Distance
{
public:
static const DistanceUnit m;
static const DistanceUnit km;

Distance(int n, const DistanceUnit& unit) :
value_(n),
unit_(unit)
{}

private:
int value_;
DistanceUnit unit_;
};

Distance::DistanceUnit m(1);
Distance::DistanceUnit km(1000);

to use:

using Distance::m;
Distance a(5, m);

or:

Distance a(5, Distance::m);

this is just a *very* rough starting point - i wrote this off the cuff
without even checking anything (maybe you should put DistanceUnit in
Distance - that's up to you). unless you're willing to supply more
information, i can't offer you any more than this. but starting from
here you can create a system that can handle arbitrary distance units in
meters, kilometers, millimeters, inches, feet, nautical miles or cubits
(though you'd have to give up on the ints and go to floats at least),
can convert automatically between any two of them and (if you're clever
enough to add a string to DistanceUnit) print out the length in the
given unit (or any other chosen unit).

mark
 
C

Chiller

Thanks Mark,

I'm trying to write a class (Distance.h file and a Distance.cpp) file that
will accept two arguments, one being a value and the other being a unit (ie
km, m, or cm). I'm trying to write the class to include overloads for ==,
!=, <, <=, > and >= which will return a boolean value. As well as overloads
of + and -. Obviously the overloads will only work on multimple instances,
ie when I enter more than a single value.

What I've written so far is included within this thread.

Thanks
 
J

John Harrison

Chiller said:
Is there a way that I can write the constructor so that it will simply
accept two variables, ie. when I create an object I can do so by simply
typing

Distance a = Distance (5, m); or
Distance a = Distance (5, km); etc

Its more common to write

Distance a(5, m);
Distance a(5, km);
without having to use ' ' or declaring m, km?

Thanks

Probably what you want is an enum

enum
{
m,
km,
miles,
};

etc.

Put that at the top of you program and then change char to int. I.e. you
constructor becomes

Distance (int, int);

But I'd seriously consider using longer names than m, km etc. How about

enum
{
units_meters,
units_kilometers,
units_miles,
};

that's how it is normally done.

john
 
J

John Harrison

Chiller said:
Below I have included the .h file .cpp implementation file. The .cpp file
has a TEST_DISTANCE driver that needs to be declared to the preprocessor to
allow it to be compiled.

I'd greatly appreciate some advice on what I'm doing wrong and how I can
correct the problem.

Whenever I compile I get the following errors:

Distance.cpp(32) : error C2556: 'char Distance::measure(void) const' :
overloaded function differs only by return type from 'int
Distance::measure(void) const'

[snip]

int measure (void) const;

You declare measure as returning an int here

[snip]
char Distance :: measure (void) const

But you define measure and returning a char here. The definition and
dclaration must be the same.

john
 
C

Chiller

John,

I've taken your suggestion and included enum; however, when I output the
stored values to screen, all the values are int values. What I'm trying to
do is to be able to enter a value and unit and be able to output the same
value and unit. ie, if I input Distance a (5, m) I'd like to be able to
output the same. Once I've got this working my next step will be to
implement some overloaded operators to perform addition, subtraction etc on
any objects created by the class.

I've included what I've done so far below. If you compile and execute the
program you'll see what I mean by the output being in the wrong format.I'd
appreciate any advice on this.

Thanks


Distance.h

************************************
#ifndef DISTANCE_H

#define DISTANCE_H

#include <iostream>

using namespace std;

class Distance

{

public :

Distance (int, int) ; // constructor - takes int values

Distance (int) ; // constructor - takes int value

Distance (void) ; // default - zero

//access member functions

int number (void) const;

int measure (void) const;


private :

int nu ; // the value

int me ; // the unit of measure (m)

} ;

// provide an overload of "<<" for easy display

ostream& operator<< (ostream&, const Distance&);

#endif



Distance.cpp as follows:

*****************************************

#include "Distance.h"

#include <iostream>

using namespace std;

/*-------------------------------------------------------*\

| implementation of member functions |

\*-------------------------------------------------------*/

// constructor

Distance :: Distance (int n, int m) : nu(n), me(m) {}

Distance :: Distance (int n) : nu(n) {}

Distance :: Distance (void) : nu(0) {}



enum

{

m,

km,

};



// access functions

int Distance :: number (void) const

{

return nu;

}

int Distance :: measure (void) const

{

return me;

}

// provide an overload of "<<" for easy display

ostream& operator<< (ostream& out, const Distance& d)

{

out << "(" << d.number() << "," << d.measure() << ")" ;

return out;

}

/*-------------------------------------------------------*\

| test driver for the Distance class |

\*-------------------------------------------------------*/

#ifdef TEST_DISTANCE // .... Distance class .... test driver

int main (void)

{

// create test input

Distance a = Distance (6);

Distance b (4);

Distance c (2);

Distance d;

Distance e (5, m);


cout << a << endl << b << endl << c << endl << d << endl << e << endl;



cin.ignore();

return 0; // normal termination

}

#endif
 
S

SaltPeter

Chiller said:
Is there a way that I can write the constructor so that it will simply
accept two variables, ie. when I create an object I can do so by simply
typing

Distance a = Distance (5, m); or
Distance a = Distance (5, km); etc

without having to use ' ' or declaring m, km?

Thanks
You need to refine your knowledge of constants.
The arguements provided are not variables. 5 is a constant ( a literal
value). The second arguement can be a std::string or enum but its also a
constant here. You don't have to use the second arguement if you plan on
providing a consistant measurement base (like meters). Personally, i'ld
prefer a constructor with an init list and passing constants by reference:

#include <iostream>
#include <string>

class Distance
{
public:
Distance(const int& r_distance) : meters(r_distance) { }
virtual ~Distance() { }
void display() const { std::cout << "distance = " << meters <<
std::endl; }
private:
int meters;
};

int main()
{
Distance dist(5000);
dist.display();
return 0;
}
 
J

John Harrison

Chiller said:
John,

I've taken your suggestion and included enum; however, when I output the
stored values to screen, all the values are int values.

That's because enum's are integer constants.
What I'm trying to
do is to be able to enter a value and unit and be able to output the same
value and unit. ie, if I input Distance a (5, m) I'd like to be able to
output the same. Once I've got this working my next step will be to
implement some overloaded operators to perform addition, subtraction etc on
any objects created by the class.

There's no way to do that other than the hard way

enum { m, km, };

switch (d.measure())
{
case m:
out << "m";
break;
case km:
out << "km";
break;
}

There's various ways of tarting this up, for instance you could have an
array of unit strings.

enum { m, km, };

const char* const unit_str[] = { "m", "km" };

out << unit_str[d.measure()];

but essentially you have to test the int value of your units and print out
the appropriate string. There is no way of automatically converting the name
of something in your program to a string.

john
 
D

DaKoadMunky

enum
{
m,
km,
miles,
};

etc.

Put that at the top of you program and then change char to int. I.e. you
constructor becomes

Distance (int, int);

I am curious about your suggestion to change the second parameter from a char
to an int.

Wouldn't it be better to have the enumeration typed and then use that type as
the second parameter? Then the author of the class need not worry that he has
been passed an invalid indicator of unit as the compiler will allow only values
specified in the enumeration to be passed. Of course the caller could cast any
value to the enumerated type, but then they get what they deserve.

Regards.


Brian F. Seaberg
Naperville, Illinois
Delray Beach, Florida
 
J

John Harrison

I am curious about your suggestion to change the second parameter from a char
to an int.

Wouldn't it be better to have the enumeration typed and then use that type as
the second parameter? Then the author of the class need not worry that he has
been passed an invalid indicator of unit as the compiler will allow only values
specified in the enumeration to be passed. Of course the caller could cast any
value to the enumerated type, but then they get what they deserve.

Maybe, but I was just trying to introduce one new concept at a time.

Incidentally the OP took my advice but if you read his latest post he's
changed back from an int to a char for some spurious reason.

john
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
474,164
Messages
2,570,898
Members
47,439
Latest member
shasuze

Latest Threads

Top