default parameter value

E

earl

class temp
{
public:
temp();
foo(char, char, char*);
private:
char matrix[150];
};

temp::foo(char p, char o, char m[150] = matrix )
{
//code
}

Is it possible to set char matrix[150] as default in the parameter list for
the mehod foo ? If so, how ?

Thanks for any help :)
 
R

Rob Williscroft

earl wrote in
class temp
{
public:
temp();
foo(char, char, char*);
private:
char matrix[150];
};

temp::foo(char p, char o, char m[150] = matrix )
{
//code
}

Is it possible to set char matrix[150] as default in the parameter
list for the mehod foo ? If so, how ?

Well yes and no, this works:

void foo(char char m[150] = "moose" )
{
std::cout << m << std::endl;
}

But what is really going on is that you are passing a pointer not an
array, i.e. you are really writing:

void foo(char char *m = "moose" )
{
std::cout << m << std::endl;
}

You should really write the 2nd form as its less missleading.

Note you can also do:

void foo(char char m[3] = "moose" )
{
std::cout << m << std::endl;
}

Note the m[3] with a 6 char initializer, the compiler doesn't care
it just treated the array like it was declared a pointer.

Here's a way to do maybe what you want:

#include <iostream>

int a[3] = { 1, 2, 3 };

void foo(int (&m)[3] = a )
{
std::cerr << m[1] << "\n";
}

int main()
{
foo();
}

The above is clearer, as your asking for an array to be passed
by reference and that is what you get, not a pointer pretending
to be an array.

If you don't want to change the array (a bit like it was passed by
value) use:

void foo(int const (&m)[3] = a )
{
}

HTH

Rob.
 
E

earl

Rob Williscroft said:
earl wrote in
class temp
{
public:
temp();
foo(char, char, char*);
private:
char matrix[150];
};

temp::foo(char p, char o, char m[150] = matrix )
{
//code
}

Is it possible to set char matrix[150] as default in the parameter
list for the mehod foo ? If so, how ?

Well yes and no, this works:

void foo(char char m[150] = "moose" )
{
std::cout << m << std::endl;
}

But what is really going on is that you are passing a pointer not an
array, i.e. you are really writing:

void foo(char char *m = "moose" )
{
std::cout << m << std::endl;
}

You should really write the 2nd form as its less missleading.

Note you can also do:

void foo(char char m[3] = "moose" )
{
std::cout << m << std::endl;
}

Note the m[3] with a 6 char initializer, the compiler doesn't care
it just treated the array like it was declared a pointer.

Here's a way to do maybe what you want:

#include <iostream>

int a[3] = { 1, 2, 3 };

void foo(int (&m)[3] = a )
{
std::cerr << m[1] << "\n";
}

int main()
{
foo();
}

The above is clearer, as your asking for an array to be passed
by reference and that is what you get, not a pointer pretending
to be an array.

If you don't want to change the array (a bit like it was passed by
value) use:

void foo(int const (&m)[3] = a )
{
}

HTH

Rob.

I tryed this code :
int main()
{
bar temp;
temp.foo();
system("PAUSE");
}

#include <iostream>
class bar
{
public:
bar();
void foo(int (&m)[5] );
private:
int matrix[5];
};


#include "bar.h"
bar::bar()
{
for(int x=0; x < 5;x++)
matrix[x] = x;
}


void bar::foo(int (&m)[5] = matrix)
{
cout << matrix[1] << endl;
}

I get an unvalid use of member bar::matrix.

What am I doing wrong ?
 
R

Rob Williscroft

earl wrote in
Rob Williscroft said:
earl wrote in
[snip]

I tryed this code :
int main()
{
bar temp;
temp.foo();
system("PAUSE");
}

#include <iostream>
class bar
{
public:
bar();

You should put the default argument in here:

void foo( int (&m)[5] = matrix );

Otherwise it will only be visible in bar.cpp or werever you define
bar::foo().
void foo(int (&m)[5] );
private:

Make this static:

static_int matrix[ 5 ];
int matrix[5];
};


#include "bar.h"

int bar::matrix[5] = { 0, 1, 2, 3, 4 };
bar::bar()
{

If you go with the static solution above you don't need this any more,
but see below if matrix must be a non-static member variable.
for(int x=0; x < 5;x++)
matrix[x] = x;
}


void bar::foo(int (&m)[5] = matrix)
{
cout << matrix[1] << endl;

cout << m[1] << endl;
}

I get an unvalid use of member bar::matrix.

The default argument can't be to a non-static member variable.

If you actually need matrix to be non-static then define 2 overloads
of bar::foo() like so:

class bar
{
public:
bar(); // init matrix
void foo( int (&m)[5] ); // no default
void foo() { foo( matrix ); }
private:
int matrix[5];
};


Rob.
 
L

lilburne

Rob said:
earl wrote in
Is it possible to set char matrix[150] as default in the parameter
list for the mehod foo ? If so, how ?



You should really write the 2nd form as its less missleading.

Quite right! I'm sure there is a valid reason for default
arguments, but I've never encountered one.
 
D

Dan Cernat

lilburne said:
Rob said:
earl wrote in
[snip]


Quite right! I'm sure there is a valid reason for default
arguments, but I've never encountered one.

Avoid code duplication.

void f(int a)
{
return a*2;
}

void f(int a, int b)
{
return a*b;
}

or, using default values:
void f(int a, int b = 2)
{
return a*b;
}

of course, the example is trivial.

/dan
 
K

Karl Heinz Buchegger

Dan said:
lilburne said:
Rob said:
earl wrote in
[snip]


Quite right! I'm sure there is a valid reason for default
arguments, but I've never encountered one.

Avoid code duplication.

void f(int a)
{
return a*2;
}

void f(int a, int b)
{
return a*b;
}

or, using default values:
void f(int a, int b = 2)
{
return a*b;
}

of course, the example is trivial.

/dan

.... and could be solved without default arguments:

void f( int a, int b )
{
return a*b;
}

void f( int a )
{
return f( a, 2 );
}

This comes in handy, if the default arument is not constant, eg.
if the above is a member function of a class and the 'default argument'
is some class member.
 
D

Dan Cernat

Karl Heinz Buchegger said:
Dan said:
lilburne said:
Rob Williscroft wrote:

earl wrote in
[snip]


Quite right! I'm sure there is a valid reason for default
arguments, but I've never encountered one.

Avoid code duplication.

void f(int a)
{
return a*2;
}

void f(int a, int b)
{
return a*b;
}

or, using default values:
void f(int a, int b = 2)
{
return a*b;
}

of course, the example is trivial.

/dan

... and could be solved without default arguments:

void f( int a, int b )
{
return a*b;
}

void f( int a )
{
return f( a, 2 );
}
you still have 2 functions f, even the second one is trivial. using
default arguments, reduces their number to 1 as in my example.
This comes in handy, if the default arument is not constant,
Let us not confuse the readers. The default argument could be a
function call that returns the apropriate type, except your example
below.
eg.
if the above is a member function of a class and the 'default argument'
is some class member.
yes, here you are right. However, a *static* member of a class or a
static function call could be used as default arguments.

/dan
 
L

lilburne

Dan said:
Karl Heinz Buchegger said:
Dan said:
Rob Williscroft wrote:


earl wrote in

[snip]


Quite right! I'm sure there is a valid reason for default
arguments, but I've never encountered one.


Avoid code duplication.

void f(int a)
{
return a*2;
}

void f(int a, int b)
{
return a*b;
}

or, using default values:
void f(int a, int b = 2)
{
return a*b;
}

of course, the example is trivial.

/dan

... and could be solved without default arguments:

void f( int a, int b )
{
return a*b;
}

void f( int a )
{
return f( a, 2 );
}

you still have 2 functions f, even the second one is trivial. using
default arguments, reduces their number to 1 as in my example.

Not really. You still have in effect two functions f but
they share one body. You haven't reduced complexity just
disguised it.

In the calling code you have:

f(3);
f(2,6);

this looks like two different functions and anyone reading
the code will wonder what the difference is. Also anyone
calling f() needs to know whether they 'want' the default or
not, so you haven't simplified anything for them, all you
have done is saved them from typing a couple of characters.

In my experience programmers tend to work from an example.
They'll cut&paste snippets of working code, and simply
tweaking it for their current needs. They'll use the f(3)
form without realising that there is a default argument, or
whether it is appropriate for the given circumstances.

By not using default arguments you make users confront the
issues each time they use the f, not hide behind defaults.
Also you can get strange behaviour when default arguments
are used with virtual functions as the defaults aren't
inherited.

The method:

void f(int a, int b);

is still just one function, with one body, and all calls to
it are standardized.
 
D

Dan Cernat

lilburne said:
Dan said:
Dan Cernat wrote:


Rob Williscroft wrote:


earl wrote in

[snip]


Quite right! I'm sure there is a valid reason for default
arguments, but I've never encountered one.


Avoid code duplication.

void f(int a)
{
return a*2;
}

void f(int a, int b)
{
return a*b;
}

or, using default values:
void f(int a, int b = 2)
{
return a*b;
}

of course, the example is trivial.

/dan

... and could be solved without default arguments:

void f( int a, int b )
{
return a*b;
}

void f( int a )
{
return f( a, 2 );
}

you still have 2 functions f, even the second one is trivial. using
default arguments, reduces their number to 1 as in my example.

Not really. You still have in effect two functions f but
they share one body. You haven't reduced complexity just
disguised it.

In the calling code you have:

f(3);
f(2,6);

this looks like two different functions and anyone reading
the code will wonder what the difference is. Also anyone
calling f() needs to know whether they 'want' the default or
not, so you haven't simplified anything for them, all you
have done is saved them from typing a couple of characters.

In my experience programmers tend to work from an example.
They'll cut&paste snippets of working code, and simply
tweaking it for their current needs. They'll use the f(3)
form without realising that there is a default argument, or
whether it is appropriate for the given circumstances.
Still, in your eaxmple, they can copy and paste an example that uses f(3)
not knowing that there is another function with the same name that has two
parameters they can use. So, IMO having default values means that the
programmers should know about the default parameters, and not having default
values meant the programmers have to knou about the multiple signatures of
the functions. I guess it is sort of the same thing.

default values version:
f(int a, int b = 2); // must know about b

without default values
f(int a);
f(int a, int b); // must know about this second function too

By not using default arguments you make users confront the
issues each time they use the f, not hide behind defaults.
Also you can get strange behaviour when default arguments
are used with virtual functions as the defaults aren't
inherited.

you have a point here

The method:

void f(int a, int b);

is still just one function, with one body, and all calls to
it are standardized.
programmers will end up declaring constants for the calls that require the
default parameter.

I am wondering why the default parameters are in the standard if they are
soo bad (I'll start a new thread about this)
Use them or don't use them.

/dan
 
L

lilburne

Dan said:
programmers will end up declaring constants for the calls that require the
default parameter.

If it helps to document what is going on then sobeit.
I am wondering why the default parameters are in the standard if they are
soo bad (I'll start a new thread about this)
Use them or don't use them.

I think its a question of scale. For small programs using a
100 or so classes then default arguments might have a place.
For large systems encompassing 1000s of classes the focus is
on having things pretty upfront in the source code. The
little bits of syntactic sugar, and candy interfaces that
default arguments, and operators give, don't carry as much
weight as trying to get programmers to write code that they
really meant to write.

When default arguments creep into our programs it is usually
because someone is retrofitting a flag to the API of some
method, and they can't be bothered to fix up all the places
that the API change requires. In other words they are
hacking (perjorative sense).
 
K

Karl Heinz Buchegger

Dan said:
I am wondering why the default parameters are in the standard

IMHO: to save a few keystrokes. As has been shown, it is easy to life
without them by creating a second function with a different signature.

When I dsicovered default arguments I used them a lot. After confusing
myself in the calling code I lowered their use. Today I use default
arguments in constructors only.
if they are
soo bad (I'll start a new thread about this)
Use them or don't use them.

Right. It's a personal preference.
 
K

Karl Heinz Buchegger

lilburne said:
When default arguments creep into our programs it is usually
because someone is retrofitting a flag to the API of some
method, and they can't be bothered to fix up all the places
that the API change requires. In other words they are
hacking (perjorative sense).

Even in this cases I prefer to add the missing arguments
on the callers side. The compiler will flag all code spots
where a call is done. Since it is possible to give a default
value, it is also possible to insert that very same defaut
value on the callers side. But it has the nice benefit of
looking at each call and doing a quick search in the callers
code, if the default is really apropriete. It happend more then
once that I figured out in less then 30 seconds that it was not :)
 

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

Forum statistics

Threads
474,145
Messages
2,570,827
Members
47,373
Latest member
Desiree036

Latest Threads

Top