在C ++中理解显式构造函数的问题

时间:2010-09-15 12:16:03

标签: c++ constructor explicit

阅读此帖后 What does the explicit keyword mean in C++?

我编写了这个程序

class MyClass
{
  public:
  explicit MyClass(int a)
  {
    cout<<"Int was called"<<endl;
    val = a;
  }

  MyClass(char *a)
  {
    cout<<"Char was called"<<endl;
    val = atoi(a);
  }

  MyClass(const MyClass& copy)
  {
    cout<<"Copy Const was called"<<endl;
    this->val = copy.val;
  }

  inline const int getval() const
  { return val; }

  private:
  int val ;
};

主要代码

int main()
{
  int x=4;
  char y='4';
  char *z = &y;

  MyClass a(x);
  MyClass b(z);
  MyClass c(a);
  MyClass d('4');

  cout<<a.getval()<<endl;
  cout<<b.getval()<<endl;
  cout<<c.getval()<<endl;
  cout<<d.getval()<<endl;

  return 0;
}

输出结果为:

Int was called
Char was called
Copy Const was called
Int was called
4
4
4
52

现在,按照上面的线程,它应该在构造函数调用对象d之后抛出错误,但它没有。

g ++版本信息

Using built-in specs.
Target: i486-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 4.4.3-4ubuntu5' --with-bugurl=file:///usr/share/doc/gcc-4.4/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --enable-shared --enable-multiarch --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.4 --program-suffix=-4.4 --enable-nls --enable-clocale=gnu --enable-libstdcxx-debug --enable-plugin --enable-objc-gc --enable-targets=all --disable-werror --with-arch-32=i486 --with-tune=generic --enable-checking=release --build=i486-linux-gnu --host=i486-linux-gnu --target=i486-linux-gnu
Thread model: posix
gcc version 4.4.3 (Ubuntu 4.4.3-4ubuntu5) 

我不确定我在上面的代码中是否做错了。 请帮忙

7 个答案:

答案 0 :(得分:4)

显式构造函数不允许您执行类似“隐式转换”的操作,例如:初始化对象时:

MyClass d = 4;

或使用参数调用函数时:

void foo( const MyClass& param);

... 

foo( 4);

在你的情况下,在调用显式构造函数之前,有一个从char到int的转换,这是设计的。

答案 1 :(得分:1)

实际上,使用d构造函数构建int。您的char'4')被隐式投放到int。这在C / C ++中是合法的,因为它们都是整数,int至少与char一样宽(反过来需要显式转换,因为可能导致精度损失)。 / p>

对象d的输出是字符“4”的ASCII代码。

答案 2 :(得分:1)

explicit关键字定义构造函数不可用于隐式转换,但在代码中,您明确要求从值构造对象。

如果您将代码更改为:

MyClass d = 'a';
MyClass x = 5;

然后转换是隐式的,你会收到错误。

请注意,重要的转化不是从'a'int,而是从intMyClass。允许编译器执行从'a'int的转换,因为原始代码中明确请求MyClass(int)构造函数。

答案 3 :(得分:0)

'4'是一个char,可以自由转换为int。 52是'4'的字符代码

答案 4 :(得分:0)

“char”(如'd')在C ++中有一个int值。实际上,52是'd'的整数ASCII值。

答案 5 :(得分:0)

没有错误 - 它只是用整数值'4'调用你的int构造函数,它实际上是U + 0034(十六进制)或数字52.C / C ++中的所有char常量都存储为int内部

答案 6 :(得分:0)

任何char字面值都是int,您可以说int i = 'a;`