Rumah >pembangunan bahagian belakang >C++ >Ralat kompilasi C++: Pemuatan templat tidak sah, bagaimana untuk menyelesaikannya?

Ralat kompilasi C++: Pemuatan templat tidak sah, bagaimana untuk menyelesaikannya?

PHPz
PHPzasal
2023-08-22 12:49:051321semak imbas

Ralat kompilasi C++: Pemuatan templat tidak sah, bagaimana untuk menyelesaikannya?

C++ ialah bahasa pengaturcaraan berkuasa yang sering digunakan untuk membangunkan pelbagai aplikasi. Walau bagaimanapun, dalam proses menulis kod C++, anda pasti akan menghadapi pelbagai masalah, salah satunya adalah masalah overloading template yang tidak sah. Masalah ini, jika tidak ditangani dengan betul, akan membawa kepada ralat penyusunan. Jadi, bagaimana kita menyelesaikan masalah ini?

Pertama sekali, kita perlu faham apa itu overloading templat. Dalam C++, lebihan templat merujuk kepada mengisytiharkan berbilang templat dengan nama yang sama tetapi nombor atau jenis parameter yang berbeza. Apabila menggunakan templat kelas atau fungsi dengan parameter yang berbeza, pengkompil secara automatik memilih templat yang sesuai berdasarkan jenis parameter yang ditakrifkan oleh templat. Walau bagaimanapun, apabila kami mentakrifkan templat, jika dua atau lebih templat ditakrifkan dengan jenis dan nombor parameter yang sama, dan jenis pemulangan juga sama, masalah lebihan templat yang tidak sah akan berlaku.

Seterusnya, mari kita lihat beberapa masalah biasa dengan lebihan templat tidak sah dan penyelesaiannya:

  1. Contoh ralat 1:
template <typename T>
void print(T x) {
    cout << "x = " << x << endl;
}

template <typename T>
void print(T* x) {
    cout << "x* = " << *x << endl;
}

int main() {
    int x = 1;
    int* ptr = &x;
    print(x);   // 1
    print(ptr); // Cannot resolve overloaded function 'print' 
    return 0;
}

Dalam contoh ini, kami mentakrifkan dua templat dengan nama yang sama tetapi parameter yang berbeza Fungsi templat print digunakan untuk mencetak pembolehubah dan penunjuk masing-masing. Walau bagaimanapun, apabila kami menggunakan fungsi print dengan parameter penunjuk, kami mendapat ralat penyusunan. print,分别用于打印变量和指针。然而,在我们使用带有指针参数的print函数时,我们得到了一个编译错误。

这是因为C++编译器需要通过参数类型来确定调用哪个函数模板。在这个例子中,指针也是一种T类型的参数,但与int类型不同。因此,编译器无法确定调用哪个函数,导致模板重载无效。解决这个问题的方法是为指针参数提供一个不同的类型,如下所示:

template <typename T>
void print(T x) {
    cout << "x = " << x << endl;
}

template <typename T>
void print(T* x) {
    cout << "x* = " << *x << endl;
}

template <typename T>
void print(T*& x) { // 指针引用增加参数类型
    cout << "x* = " << *x << endl;
}

int main() {
    int x = 1;
    int* ptr = &x;
    print(x);   // 1
    print(ptr); // x* = 1
    return 0;
}

在这个例子中,我们添加了一个新的模板函数print(T*& x),该函数有一个指针引用参数类型,结果可以成功打印一个指针。

  1. 错误示例2:
template <typename T1,typename T2>
void swap(T1& a, T2& b) {
    T1 temp = a;
    a = b;
    b = temp;
}

template <typename T>
void swap(T& a, T& b) {
    T temp = a;
    a = b;
    b = temp;
}

int main() {
    int x = 1,y = 2;
    double d1 = 1.1,d2 = 2.2;
    swap(x,y);
    swap(d1,d2);
    swap(x,d2); // Cannot resolve overloaded function 'swap'
    return 0;
}

在这个例子中,我们定义了两个名字相同但参数不同的模板函数swap,一个用于交换两个不同类型的变量,另一个则用于交换同一类型的变量。然而,在我们交换一个int类型的变量和一个double类型的变量时,我们又得到了一个编译错误。

这是因为在这种情况下,编译器无法根据参数类型区分哪个swap函数应该被调用,导致模板重载无效。为了解决这个问题,我们需要强制指定调用哪个swap函数,如下所示:

template <typename T1,typename T2>
void swap(T1& a, T2& b) {
    T1 temp = a;
    a = b;
    b = temp;
}

template <typename T>
void swap(T& a, T& b) {
    T temp = a;
    a = b;
    b = temp;
}

int main() {
    int x = 1,y = 2;
    double d1 = 1.1,d2 = 2.2;
    swap(x,y);
    swap(d1,d2);
    swap<int,double>(x,d2); // 使用模板实参指定调用哪个模板函数
    return 0;
}

在这个例子中,我们在调用swap<int>(x,d2)</int>时,使用了模板实参<int></int>来指定调用哪个swap

Ini kerana pengkompil C++ perlu menentukan templat fungsi mana yang hendak dipanggil melalui jenis parameter. Dalam contoh ini, penunjuk juga merupakan parameter jenis T, tetapi bukan jenis int. Oleh itu, pengkompil tidak dapat menentukan fungsi yang hendak dipanggil, menyebabkan beban templat menjadi tidak sah. Cara untuk menyelesaikan masalah ini adalah dengan menyediakan jenis yang berbeza untuk parameter penunjuk, seperti ini:

rrreee

Dalam contoh ini, kami telah menambah fungsi templat baharu print(T*& x), The fungsi mempunyai jenis parameter rujukan penunjuk, dan hasilnya boleh berjaya mencetak penunjuk.

    Contoh ralat 2: 🎜🎜rrreee🎜Dalam contoh ini, kami mentakrifkan dua fungsi templat tukar dengan nama yang sama tetapi parameter berbeza, satu untuk menukar dua pembolehubah berbeza jenis, dan satu lagi digunakan untuk menukar pembolehubah daripada jenis yang sama. Walau bagaimanapun, apabila kita menukar pembolehubah jenis int dengan pembolehubah jenis berganda, kita mendapat satu lagi ralat penyusunan. 🎜🎜Ini kerana dalam kes ini, pengkompil tidak dapat membezakan fungsi swap yang harus dipanggil berdasarkan jenis parameter, mengakibatkan lebihan templat tidak sah. Untuk menyelesaikan masalah ini, kita perlu memaksa untuk menentukan fungsi swap yang dipanggil, seperti yang ditunjukkan di bawah: 🎜rrreee🎜Dalam contoh ini, kami memanggil swap<int>(x,d2 ) </int>, parameter sebenar templat <int></int> digunakan untuk menentukan fungsi swap yang hendak dipanggil, yang menyelesaikan masalah lebihan templat yang tidak sah. 🎜🎜Ringkasan: 🎜🎜Pembebanan templat tidak sah ialah ralat biasa dalam penulisan C++, biasanya disebabkan oleh fungsi terlampau beban dengan definisi yang sama tetapi parameter atau jenis pengembalian yang sedikit berbeza. Untuk mengelakkan masalah ini, kami perlu menyediakan jenis parameter dan jenis pulangan yang berbeza untuk setiap fungsi templat dan menggunakan hujah templat untuk menentukan fungsi yang perlu dipanggil. Melalui kaedah ini, kami boleh menyelesaikan masalah lebihan templat yang tidak sah dan menjadikan kod C++ kami lebih mantap dan lengkap. 🎜

Atas ialah kandungan terperinci Ralat kompilasi C++: Pemuatan templat tidak sah, bagaimana untuk menyelesaikannya?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Kenyataan:
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn