Map:: emplace() с пользовательским типом значения

У меня возникли проблемы с использованием map::emplace(). Может ли кто-нибудь помочь мне разобраться в правильном синтаксисе? Я эффективно пытаюсь сделать то же самое, что и в в этом примере. Вот моя версия:

#include <map>
using namespace std;

class Foo
{
  // private members

  public:
    Foo(int, char, char) /* :init(), members() */ {  }

    // no default ctor, copy ctor, move ctor, or assignment
    Foo() = delete;
    Foo(const Foo&) = delete;
    Foo(Foo &&) = delete;
    Foo & operator=(const Foo &) = delete;
    Foo & operator=(Foo &&) = delete;
};


int main()
{
  map<int, Foo> mymap;
  mymap.emplace(5, 5, 'a', 'b');

  return 0;
}

В GCC 4.7 (с флагом -std=c++11) я ошибаюсь в строке emplace. Пример который я связал выше, не говорит ничего о том, как обращаться с настраиваемыми типами вместо примитивов.

Ответ 1

Член контейнера emplace создает элемент, используя предоставленные аргументы.

value_type вашей карты std::pair<const int, Foo> и этот тип не имеет конструктора, принимающего аргументы { 5, 5, 'a', 'b' } i.e, это не сработает:

std::pair<const int, Foo> value{ 5, 5, 'a', 'b' };
map.emplace(value);

Вам нужно вызвать emplace с аргументами, которые соответствуют одному из конструкторов pair.

С соответствующей реализацией С++ 11 вы можете использовать:

mymap.emplace(std::piecewise_construct, std::make_tuple(5), std::make_tuple(5, 'a', 'b'));

но GCC 4.7 также не поддерживает этот синтаксис (GCC 4.8 будет выпущен.)

Ответ 2

GCC 4.7 не имеет полной поддержки функций emplace.

Вы можете увидеть поддержку С++ 11 в GCC 4.7.2 здесь.