6

使用 g++ 4.7.0 ( ) 编译此代码-Wall -Wextra -Werror -Wconversion -std=c++11

#include <iostream>  // std::cout, std::endl
#include <string>    // std::string
#include <utility>   // std::move

void out(std::string const &message)
{
   static int count{0};
   std::cout << count++ << " = " << message << std::endl;
}

struct Foo
{
   Foo()                         {out("constructor");}
  ~Foo()                         {out("destructor");}
   Foo(Foo const &)              {out("copy constructor");}
   Foo & operator=(Foo const &)  {out("copy via assignment"); return *this;}
   Foo(Foo &&)                   {out("move constructor");}
   Foo & operator=(Foo &&)       {out("move via assignment"); return *this;}
};

int main()
{
   auto bar{std::move(Foo())};
   out("exiting main");
}

...导致以下错误:

error: unused variable 'bar' [-Werror=unused-variable]

我可以通过将bar初始化更改为以下任何一项来消除错误:

/* 0 */ auto bar(std::move(Foo()));
/* 1 */ Foo bar{std::move(Foo())};
/* 2 */ Foo bar(std::move(Foo()));
/* 3 */ auto bar = std::move(Foo());
/* 4 */ Foo bar = std::move(Foo());
/* 5 */ auto bar __attribute__((unused)) {std::move(Foo())};

更改bar初始化后,输出始终为:

0 = constructor
1 = move constructor
2 = destructor
3 = exiting main
4 = destructor

为什么原始bar初始化报告一个未使用的变量?

4

2 回答 2

8
auto bar{std::move(Foo())};

在此声明之后,bar是 type std::initializer_list<Foo>,它具有简单的复制/移动操作和析构函数。您的其他声明

auto bar(std::move(Foo()));
Foo bar{std::move(Foo())};
Foo bar(std::move(Foo()));
auto bar = std::move(Foo());
Foo bar = std::move(Foo());

声明barFooor Foo&&,它会抑制警告,因为它具有非平凡的特殊成员函数。

auto除非您特别打算创建一个std::inializer_list对象,否则您通常不希望使用括号初始化。

于 2012-03-18T17:55:29.810 回答
2

嗯,bar 用过。您可能想为编译器提交一个缺陷,因为这似乎在其他情况下错误地未被检测到。

于 2012-03-18T16:54:54.177 回答