Errore del compilatore C2666
'identifier': gli overload dei numeri hanno conversioni simili
Una funzione o un operatore di overload è ambiguo. Gli elenchi di parametri formali possono essere troppo simili per il compilatore per risolvere l'ambiguità. Per risolvere questo errore, eseguire il cast esplicito di uno o più parametri effettivi.
Esempi
L'esempio seguente genera l'errore C2666:
// C2666.cpp
struct complex {
complex(double);
};
void h(int,complex);
void h(double, double);
int main() {
h(3,4); // C2666
}
Questo errore può essere generato in seguito al lavoro di conformità del compilatore eseguito per Visual Studio 2019 versione 16.1:
- Una conversione che promuove un'enumerazione il cui tipo sottostante è fisso al tipo sottostante è migliore di uno che promuove al tipo sottostante alzato di livello, se i due sono diversi.
L'esempio seguente illustra come cambia il comportamento del compilatore in Visual Studio 2019 versione 16.1 e versioni successive:
#include <type_traits>
enum E : unsigned char { e };
int f(unsigned int)
{
return 1;
}
int f(unsigned char)
{
return 2;
}
struct A {};
struct B : public A {};
int f(unsigned int, const B&)
{
return 3;
}
int f(unsigned char, const A&)
{
return 4;
}
int main()
{
// Calls f(unsigned char) in 16.1 and later. Called f(unsigned int) in earlier versions.
// The conversion from 'E' to the fixed underlying type 'unsigned char' is better than the
// conversion from 'E' to the promoted type 'unsigned int'.
f(e);
// Error C2666. This call is ambiguous, but previously called f(unsigned int, const B&).
f(e, B{});
}
Questo errore può essere generato anche in seguito al lavoro di conformità del compilatore eseguito per Visual Studio .NET 2003:
operatori binari e conversioni definite dall'utente in tipi di puntatore
la conversione della qualifica non equivale alla conversione delle identità
Per gli operatori <binari , , <>=, e >=, un parametro passato viene ora convertito in modo implicito nel tipo dell'operando se il tipo del parametro definisce un operatore di conversione definito dall'utente da convertire nel tipo dell'operando. È ora possibile che si verifichino ambiguità.
Per il codice valido nelle versioni di Visual Studio .NET 2003 e Visual Studio .NET di Visual C++, chiamare l'operatore di classe in modo esplicito usando la sintassi della funzione.
// C2666b.cpp
#include <string.h>
#include <stdio.h>
struct T
{
T( const T& copy )
{
m_str = copy.m_str;
}
T( const char* str )
{
int iSize = (strlen( str )+ 1);
m_str = new char[ iSize ];
if (m_str)
strcpy_s( m_str, iSize, str );
}
bool operator<( const T& RHS )
{
return m_str < RHS.m_str;
}
operator char*() const
{
return m_str;
}
char* m_str;
};
int main()
{
T str1( "ABCD" );
const char* str2 = "DEFG";
// Error - Ambiguous call to operator<()
// Trying to convert str1 to char* and then call
// operator<( const char*, const char* )?
// OR
// trying to convert str2 to T and then call
// T::operator<( const T& )?
if( str1 < str2 ) // C2666
if ( str1.operator < ( str2 ) ) // Treat str2 as type T
printf_s("str1.operator < ( str2 )\n");
if ( str1.operator char*() < str2 ) // Treat str1 as type char*
printf_s("str1.operator char*() < str2\n");
}
L'esempio seguente genera l'errore C2666
// C2666c.cpp
// compile with: /c
enum E
{
E_A, E_B
};
class A
{
int h(const E e) const {return 0; }
int h(const int i) { return 1; }
// Uncomment the following line to resolve.
// int h(const E e) { return 0; }
void Test()
{
h(E_A); // C2666
h((const int) E_A);
h((int) E_A);
}
};