Re: Template functions and avoiding implicit conversions
On Feb 27, 2:56 pm, "tarjei.knaps...@gmail.com"
<tarjei.knaps...@gmail.com> wrote:
Consider the following:
class CommandLineOptions
{
public:
template <typename ValueType>
void
Add(std::string& arg, bool required);
template <typename ValueType>
void
Add(std::string& arg, ValueType default);
};
The problem I have is that if I write something like:
CommandLineOptions clo;
clo.Add<std::string>("method", "invert");
the second argument is implicitly converted to bool and the first
template function is instantiated.
It has nothing to do with templates. If you have two functions:
void f( std::string ) ; // or std::string const&...
void f( bool ) ;
f( "string literal" ) ;
will call the second. The conversion char const[] to bool is a
better match than the conversion char const[] to std::string,
since it is a standard conversion, and the conversion to
std::string is a user defined conversion.
The possible resolutions I've come up with are:
1. Rename the first function to for instance AddRequired(). I'd like
to avoid this to keep the interface as simple as possible.
Do the functions do different things? Then they should have
different names.
2. Require the user to explicitly instantiate a std::string in the
call,
clo.Add<std::string>("method", std::string("invert"));. This is far
too error prone and not really a solution at all.
Is there any other neat trick I could use to avoid the implicit
conversion from happening?
I'm not sure that you could adopt it directly, but I had a
similar problem when I extended Fallible to accept an extended
error status. You might want to take a look at the code:
http://kanze.james.neuf.fr/code/Util/Basic/Fallible/gb/Fallible.hh.
(Generally, I prefer people going through the documentation
first, starting at http://kanze.james.neuf.fr/code-en.html. But
in this case, you're really just interested in the particular
implementation trick, documented in the .hh file above.)
The trick involves using an intermediate class without any
implicit conversions. In my case, it works because anytime you
have the string, you want false, and if you want the version
without the string, you'll want true, and will use that variant
of the function, instead of the one with the intermediate type
as parameter. In your case, you'll probably want to provide two
pre-defined instances of the intermediate type, say isRequired
and isNotRequire (which is, by the way, far clearer than true or
false anyway).
But does it make any sense in your case to ever call Add(
"toto", false )? If not, wouldn't just two functions:
Add( std::string const& arg ) ;
Add( std::string const& arg, ValueType defaultValue ) ;
do the trick? (BTW: I've corrected two other "obvious" errors:
the strings should be passed by const reference, so that string
literals can be used as arguments as well, and I think you'll
find that the compiler doesn't like default as the name of a
parameter.)
--
James Kanze (Gabi Software) email: james.kanze@gmail.com
Conseils en informatique orient?e objet/
Beratung in objektorientierter Datenverarbeitung
9 place S?mard, 78210 St.-Cyr-l'?cole, France, +33 (0)1 30 23 00 34
--
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]