Разрешение перегрузки для внешних версий "C" и "C++" qsort()/bsearch()

В C++ есть две версии qsort() предоставляется стандартной библиотекой:

extern "C" void qsort(void* base, size_t nmemb, size_t size, int (*compar)(const void*, const void*));
extern "C++" void qsort(void* base, size_t nmemb, size_t size, int (*compar)(const void*, const void*));

bsearch() примерно так же.

Мой вопрос, как работает разрешение перегрузки при вызове qsort()? Будет ли он автоматически ссылаться на соответствующую функцию на основе типа связи ("C" или "C++") указателя функции, переданного в качестве последнего аргумента? Или вызывающий должен указать явно с каким-то дополнительным синтаксисом?

(Давайте просто уберем искушение позвонить std::sort на секунду...)

1 ответ

Решение

int (*compar)(const void*, const void*) Параметр имеет разные типы для двух разных перегрузок. Для первой перегрузки это extern "C" параметр указателя функции. Для второй перегрузки это extern "C++" параметр указателя функции. Любой указатель на функцию, которую вы передаете qsort уже будет иметь какую-то связь, и это то, что используется, чтобы определить, какую перегрузку вызывать.

Чтобы процитировать стандарт:

7.5 Спецификации сцепления [dcl.link]

Все типы функций, имена функций с внешней связью и имена переменных с внешней связью имеют языковую связь. [...] Языковая связь по умолчанию для всех типов функций, имен функций и имен переменных - это языковая связь C++. Два типа функций с разными языковыми связями являются разными типами, даже если они идентичны.

На самом деле, я не думаю, что стандарт на самом деле означает, что два qsort перегрузки действительно имеют различную связь. В отличие от C, пользовательские объявления стандартных функций библиотеки не допускаются; соответствующая разница между ними является тип compar, Они могли быть объявлены

extern "C" typedef int (*__compar_fnp_c)(const void *, const void *);
extern "C++" typedef int (*__compar_fnp_cxx)(const void *, const void *);
void qsort(void* base, size_t nmemb, size_t size, __compar_fnp_c compar);
void qsort(void* base, size_t nmemb, size_t size, __compar_fnp_cxx compar);

где должно быть более очевидно, что __compar_fnp_c а также __compar_fnp_cxx разные типы. Тем не менее, правило "как если" не допускает такую ​​реализацию, поскольку оно нарушит код, который принимает указатель или ссылку на qsort,

Обратите внимание, что GCC, как и некоторые другие компиляторы, не реализуют это правильно и не рассматривают связывание как часть типа указателя функции. На таких реализациях только одна версия qsort будет сделано доступным, чтобы предотвратить конфликт при разрешении перегрузки.

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