VC++ 2013: использование-объявление + переопределение функции-члена приводит к ошибке компиляции

Я хочу разрешить изменять поведение моего класса, указав политику. Эта политика должна использоваться в качестве посетителя для boost:: option. Существует политика по умолчанию, которая подходит для большинства случаев, но пользователю может потребоваться добавить или заменить некоторые перегрузки.

Я обнаружил, что vC++ 2013 не компилирует этот код с ошибкой C3066: there are multiple ways that an object of this type can be called with these arguments, Тот же код компилируется и работает как положено в gcc и clang.

Это ошибка vC++ 2013?

#include <iostream>

struct DefaultPolicy
{
    void operator()( bool ) { std::cout << "Base: bool" << std::endl; }
    void operator()( int ) { std::cout << "Base: int" << std::endl; }
};

struct UserModifiedPolicy : public DefaultPolicy
{
    using DefaultPolicy::operator();
    void operator()( int ) { std::cout << "Derived: int" << std::endl; }
    void operator()( float ) { std::cout << "Derived: float" << std::endl; }
};

int main()
{
    UserModifiedPolicy()(true);
    UserModifiedPolicy()(1); // <-- ERROR HERE
    UserModifiedPolicy()(1.f);
    return 0;
}

UPD Этот пример работает в vC++ 2010. Похоже, это ошибка в версии 2013 года.


UPD Обходной путь

#include <iostream>

struct DefaultPolicy
{
    void operator()( bool ) { std::cout << "Base: bool" << std::endl; }
    void operator()( int ) { std::cout << "Base: int" << std::endl; }
};

struct UserModifiedPolicy : public DefaultPolicy
{
    // Using template to forward a call to the base class:
    template< class T >
    void operator()( T && t ) { DefaultPolicy::operator()( std::forward<T>(t) ); }

    void operator()( int ) { std::cout << "Derived: int" << std::endl; }
    void operator()( float ) { std::cout << "Derived: float" << std::endl; }
};

int main()
{
    UserModifiedPolicy()(true);
    UserModifiedPolicy()(1);
    UserModifiedPolicy()(1.f);
    return 0;
}

1 ответ

Решение

Код правильно сформирован. 7.3.3/15:

Когда объявление использования переносит имена из базового класса в область производного класса, функции-члены и шаблоны функций-членов в производном классе переопределяют и / или скрывают функции-члены и шаблоны функций-членов с одинаковыми именами, параметр-тип-список (8.3.5) cv-квалификация и ref-квалификатор (если есть) в базовом классе (а не конфликтующий).

Так UserModifiedPolicy::operator()(int) должен еще спрятаться DefaultPolicy::operator()(int), И название поиска для operator() должен найти трех членов DefaultPolicy::operator()(bool), UserModifiedPolicy::operator()(int), а также UserModifiedPolicy::operator()(float),

Другие вопросы по тегам