Welcome to Experts Exchange

Add your voice to the tech community where 5M+ people, just like you, are talking about what matters.

  • Help others & share knowledge
  • Earn cash & points
  • Learn & ask questions
Solved

default value initializer in templatized class

Posted on 2002-04-15
5
248 Views
Last Modified: 2010-04-02
I want to create a templatizable class which includes a default value for the templatized data type.  The following oversimplified example will work for atomic types like int and BOOL but not for data types that are structures or classes:

template <class DataType, DataType DefaultValue>
class CMyClass
{
protected:
  DataType m_Data;
public
  CMyClass() {m_Data = DefaultValue;}
}; //end CMyClass

I get several errors if I try to instantiate the following:
CMyClass<CString, "default string value"> MyObject;

I found that CString is not a valid type for a template argument because structures are not allowed.  MSDN (my C++ documentation) says to use a pointer instead if that is the case, but when I have tried that, I get an error like "expected constant value" because I have to instantiate it something like this:

const CString DefaultValue("default string value")
CMyClass<CString*, &DefaultValue> MyObject;

and &DefaultValue is where it expects a constant value and I can't give it one.  I use CString as an example, but I want to be able to use this on custom classes as well (all of which would support operator= so that the templatized constructor code m_Data = DefaultValue would work) otherwise, there are a variety of alternative solutions I could use just for strings.

--------

I want to be able to make a similar class for arrays like this:

template <class DataType, int nNumElements, const DataType* pDefaultValue>
class CMyArrayClass
{
protected:
  DataType m_arData[nNumElements];
public:
  CMyClass()
  {
    for (int i=0; i<nNumElements; i++)
       m_arData[i] = pDefaultValue[i];
  }
}; //end CMyArrayClass

I have tried the above code using a char array as follows:
CMyArrayClass<char, 4, "test"> MyObject;
but I get an error that "test" is an invalid template argument (even though it is a const char*).  If I use NULL instead, it works, but that doesn't help me because I want to pass in an initial value other than a non-value like NULL.

So what can I do?
0
Comment
Question by:ris
  • 4
5 Comments
 
LVL 30

Expert Comment

by:Axter
ID: 6942980
Default values are not allowed as a template value.

Continue.....
0
 
LVL 30

Expert Comment

by:Axter
ID: 6942986
>>So what can I do?

There is a way around this, but it's not a clean method.  In other words, it's ugly.
I'll post an example code in a follow-up comment.

The basic idea would be to create either a singleton class, or a class with a single static function, that would return the default value.
0
 
LVL 30

Accepted Solution

by:
Axter earned 100 total points
ID: 6943012
Here's an example method.

#include <stdlib.h>

#pragma warning (disable:4786)
#include <string>
using namespace std;


template<typename T>
class DefaultValueTemplateClass
{
public:
     static T DefaultValue();
};

template<class T, class T_DF = DefaultValueTemplateClass<T> >
class foo
{
public:
     foo(const T& Src = T_DF::DefaultValue()) : m_data(Src)
     {
     }
     T m_data;
};


#define MAKE_DEFAULT_TEMPL_VAUE(T,V) T DefaultValueTemplateClass<T>::DefaultValue(){return V;}

MAKE_DEFAULT_TEMPL_VAUE(int,54)
MAKE_DEFAULT_TEMPL_VAUE(string,"Hello World")
MAKE_DEFAULT_TEMPL_VAUE(long,99)

int main(int argc, char* argv[])
{
     foo<int, DefaultValueTemplateClass<int> > foo_int;
     printf("foo_int = %i\n", foo_int.m_data);

     foo<int> foo_int_with_default_default;
     printf("foo_int_with_default_default = %i\n", foo_int_with_default_default.m_data);

     foo<long> foo_long;
     printf("foo_long = %i\n", foo_long.m_data);

     foo<string> foo_string;
     printf("foo_string = %s\n", foo_string.m_data.c_str());


     system("pause");
     return 0;
}

0
 
LVL 30

Expert Comment

by:Axter
ID: 6943054
Correction:
Even though the previous macro I posted, compiles and runs on VC++ 6.0, it shouldn't have.
The correct syntax, is the following:

#define MAKE_DEFAULT_TEMPL_VAUE(T,V) template<> T DefaultValueTemplateClass<T>::DefaultValue(){return V;}
0
 
LVL 1

Author Comment

by:ris
ID: 6943070
I see what you are getting at there.  Thanks.
0

Featured Post

Free Tool: IP Lookup

Get more info about an IP address or domain name, such as organization, abuse contacts and geolocation.

One of a set of tools we are providing to everyone as a way of saying thank you for being a part of the community.

Question has a verified solution.

If you are experiencing a similar issue, please ask a related question

Suggested Solutions

When writing generic code, using template meta-programming techniques, it is sometimes useful to know if a type is convertible to another type. A good example of when this might be is if you are writing diagnostic instrumentation for code to generat…
Templates For Beginners Or How To Encourage The Compiler To Work For You Introduction This tutorial is targeted at the reader who is, perhaps, familiar with the basics of C++ but would prefer a little slower introduction to the more ad…
The viewer will learn how to user default arguments when defining functions. This method of defining functions will be contrasted with the non-default-argument of defining functions.
The viewer will learn how to clear a vector as well as how to detect empty vectors in C++.

837 members asked questions and received personalized solutions in the past 7 days.

Join the community of 500,000 technology professionals and ask your questions.

Join & Ask a Question