Rumah >pembangunan bahagian belakang >C++ >Bagaimana untuk Menghantar Dengan Selamat Antara `kosong*` dan Penunjuk kepada Fungsi Ahli dalam C ?

Bagaimana untuk Menghantar Dengan Selamat Antara `kosong*` dan Penunjuk kepada Fungsi Ahli dalam C ?

Linda Hamilton
Linda Hamiltonasal
2024-10-27 20:06:02309semak imbas

How to Safely Cast Between `void*` and a Pointer to a Member Function in C  ?

Memutuskan Antara kekosongan* dan Penunjuk kepada Fungsi Ahli

Dalam dunia pengaturcaraan C, menghantar antara void* dan penunjuk ke fungsi ahli boleh menjadi sakit kepala yang sebenar. Mari kita selami masalah khusus yang menggambarkan cabaran ini.

Pertimbangkan coretan kod berikut:

<code class="cpp">template <class T>
int call_int_function(lua_State *L) 
{
    void (T::*method)(int, int) = reinterpret_cast<void (T::*)(int, int)>(lua_touserdata(L, lua_upvalueindex(1))); // <-- problematic line
    T *obj = reinterpret_cast<T *>(lua_touserdata(L, 1));

    (obj->*method)(lua_tointeger(L, 2), lua_tointeger(L, 3));
    return 0;
}</code>

Di sini, GCC mengadu bahawa pelakon daripada void* kepada void (T::*) (int, int) tidak sah. Jadi, apakah penyelesaiannya?

Memahami Masalah

Memahami masalah memerlukan mendalami sifat penunjuk kepada fungsi ahli. Tidak seperti penunjuk biasa yang menunjuk ke alamat, penunjuk kepada fungsi ahli adalah lebih kompleks. Ia mengandungi maklumat khusus untuk pelaksanaan pengkompil dan susun atur ahli kelas.

Pendekatan Alternatif

Sebenarnya, anda tidak boleh terus membuang kekosongan* ke penunjuk kepada fungsi ahli. Penyelesaian yang dicadangkan melibatkan pembalut fungsi ahli dalam fungsi biasa dan sebaliknya mengembalikannya. Begini caranya:

<code class="cpp">template <class T>
int call_int_function(lua_State *L) 
{
    void (*method)(T*, int, int) = reinterpret_cast<void (*)(T*, int, int)>(lua_touserdata(L, lua_upvalueindex(1)));
    T *obj = reinterpret_cast<T *>(lua_touserdata(L, 1));

    method(obj, lua_tointeger(L, 2), lua_tointeger(L, 3));
    return 0;
}</code>

Pendekatan ini berfungsi kerana kami kini menghantar ke penuding fungsi biasa, yang boleh disimpan dalam kekosongan* dan diambil semula tanpa isu.

Atas ialah kandungan terperinci Bagaimana untuk Menghantar Dengan Selamat Antara `kosong*` dan Penunjuk kepada Fungsi Ahli dalam C ?. 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