decltype не работает должным образом в gcc 4.3.2? - PullRequest
2 голосов
/ 10 марта 2011
#include <iostream>
#include <map>
using namespace std;

int main()
{
    int x = 5;
    decltype(x) y = 10;
    map<int, int> m;
    decltype(m) n;
    decltype(m)::iterator it;
}

g++ -std=c++0x main.cpp

main.cpp: In function `int main()':
main.cpp:11: error: expected initializer before `it'

Первые 2 работы decltypes. Третье приводит к ошибке компилятора. Это проблема этой версии GCC?

g++ -v
Using built-in specs.
Target: x86_64-suse-linux
Configured with: ../configure --prefix=/usr --infodir=/usr/share/info --mandir=/usr/share/man --libdir=/usr/lib64 --libexecdir=/usr/lib64 --enable-languages=c,c++,objc,fortran,obj-c++,java,ada --enable-checking=release --with-gxx-include-dir=/usr/include/c++/4.3 --enable-ssp --disable-libssp --with-bugurl=http://bugs.opensuse.org/ --with-pkgversion='SUSE Linux' --disable-libgcj --disable-libmudflap --with-slibdir=/lib64 --with-system-zlib --enable-__cxa_atexit --enable-libstdcxx-allocator=new --disable-libstdcxx-pch --enable-version-specific-runtime-libs --program-suffix=-4.3 --enable-linux-futex --without-system-libunwind --with-cpu=generic --build=x86_64-suse-linux
Thread model: posix
gcc version 4.3.2 [gcc-4_3-branch revision 141291] (SUSE Linux) 

Ответы [ 5 ]

6 голосов
/ 10 марта 2011

Возможность использовать выражение decltype для формирования квалифицированного идентификатора ( type :: type ) была добавлена ​​только недавно (ну, год назад) в рабочий документ C ++ 0x, но это будет частью готового стандарта. Таким образом, код, который вы написали, на самом деле является правильно сформированным C ++ 0x. В конце концов gcc догонит.

Тем временем вы можете использовать что-то вроде

#include <map> 
template<typename T> 
struct decltype_t 
{ 
typedef T type; 
}; 

#define DECLTYPE(expr) decltype_t<decltype(expr)>::type 

int main() 
{ 
 std::map<int, int> m;
 decltype(m) n; 
 DECLTYPE(m)::iterator it; // works as expected
} 

Хотя, если вы похожи на меня, глупо будет прибегать к таким уловкам:)

1 голос
/ 10 марта 2011

Согласно последнему стандарту, decltype-specier уже разрешено делать это, я думаю, вы увидите поддержку GCC в GCC 4.6.

qualified-id:
  ::opt nested-name-specifier unqualified-id
  :: identifier
  :: operator-function-id
  :: literal-operator-id
  :: template-id

nested-name-specifier:
  type-name ::
  namespace-name ::
  decltype-specifier ::
  nested-name-specifier identifier ::
  nested-name-specifier templateopt simple-template-id ::
1 голос
/ 10 марта 2011

Согласно википедии это известная проблема в спецификации, которая просто еще не была исправлена. Из-за этого decltype не может быть частью квалифицированного идентификатора .

0 голосов
/ 12 февраля 2013

У меня нет gcc 4.3.2 на моей машине, но я тестировал с gcc 4.4 и использовал std::remove_reference после того, как мой коллега предложил его для аналогичной ошибки компиляции.

$ g ++ - 4.4 - версия g ++ - 4.4 (Ubuntu / Linaro 4.4.7-1ubuntu2) 4.4.7 Copyright (C) 2010 Free Software Foundation, Inc. Это бесплатно программного обеспечения; см. источник для условий копирования. Здесь нет гарантия; даже не для ИЗДЕЛИИ или ФИТНЕСА ДЛЯ ОСОБЕННОСТИ ЦЕЛЬ.

#include <iostream>
#include <map>
using namespace std;

int main()
{ 
  int x = 5;
  decltype(x) y = 10;
  map<int, int> m;
  decltype(m) n;
  std::remove_reference<decltype(m)>::type::iterator it;
  std::cout << "Done"  << std::endl;
}

Вывод командной строки:

$ g++-4.4 -std=c++0x decltype.cpp 
$ ./a.out 
Done
0 голосов
/ 10 марта 2011

Также не работает с GCC 4.5. См. Википедию.

...