Перегрузка функции для строковых литералов lvalue и rvalue - PullRequest
12 голосов
/ 27 сентября 2019

Функция test ниже перегружена для lvalue пустых строк, lvalue непустых строк и rvalue строк.Я попытался скомпилировать с Clang и GCC, но в обоих случаях у меня нет ожидаемого результата.

#include <iostream>

void test(const char (&)[1]){ std::cout << __PRETTY_FUNCTION__ << std::endl; }

template <unsigned long int N>
void test(const char (&)[N]){ std::cout << __PRETTY_FUNCTION__ << std::endl; }

void test(char*&&){ std::cout << __PRETTY_FUNCTION__ << std::endl; }

int main(){
    char str1[] = "";
    char str2[] = "test";
    test("");
    test("test");
    test(str1);
    test(str2);
}

Вывод clang версия 6.0.0-1ubuntu2 :

clang++ test.cpp -o test.out && ./test.out
void test(const char (&)[1])
void test(const char (&)[N]) [N = 5]
void test(char *&&)
void test(char *&&)

Вывод с g ++ (MinGW.org GCC-8.2.0-3) :

g++ test.cpp -o test.exe && test.exe
test.cpp: In function 'int main()':
test.cpp:15:11: error: call of overloaded 'test(char [1])' is ambiguous
  test(str1);
           ^
test.cpp:3:6: note: candidate: 'void test(const char (&)[1])'
 void test(const char (&)[1]){ std::cout << __PRETTY_FUNCTION__ << std::endl; }
      ^~~~
test.cpp:6:6: note: candidate: 'void test(const char (&)[N]) [with long unsigned int N = 1]'
 void test(const char (&)[N]){ std::cout << __PRETTY_FUNCTION__ << std::endl; }
      ^~~~
test.cpp:8:6: note: candidate: 'void test(char*&&)'
 void test(char*&&){ std::cout << __PRETTY_FUNCTION__ << std::endl; }
      ^~~~

Мои вопросы:

  1. Какиекомпилятор верен?
  2. С Clang, почему test(str1) и test(str2) выбирают перегрузку rvalue, пока они являются lvalues?
  3. С GCC, почему вызов test(str1) неоднозначен?
  4. Есть ли стандартное правило для этой ситуации?
  5. Как исправить два последних звонка?

Спасибо.

Ответы [ 3 ]

10 голосов
/ 27 сентября 2019
  1. Какой компилятор верен?

GCC верен.

С помощью clang, почему str1 и str2 выбирают перегрузку rvalue, в то время как они являются lvalues?

Clang неверен на test(str1);, это должно быть неоднозначно.Для test(str2);, str2 может преобразовываться в указатель неявно, то есть затухание массива в указатель.Преобразованный char* является значением.По той же причине, что и # 3, последовательности неявного преобразования имеют одинаковое ранжирование, тогда предпочтительна не шаблонная функция;test(char*&&) выбран.

С gcc, почему вызов с str1 неоднозначен?

Для вызова test(const char (&)[1]) требуется преобразование квалификации с char[1] в const char[1];для вызова test(char*&&) требуется преобразование массива в указатель.Оба квалифицируются как точное совпадение и имеют одинаковый рейтинг.

Существует ли стандартное правило для этой ситуации?

См. ранжирование последовательностей неявного преобразования в разрешении перегрузки и неявных преобразований .

Как исправить два последних звонка?

Это зависит от ваших намерений.

2 голосов
/ 27 сентября 2019

Строковые литералы не являются значениями.()

Как исправить два последних звонка?

Вы можете устранить неоднозначность с шаблонами специализаций:

#include <iostream>

template<typename C, std::size_t N>
void test(const C (&)[N]) { std::cout << __PRETTY_FUNCTION__ << std::endl; }
template<typename C>
void test(const C (&)[1]) { std::cout << __PRETTY_FUNCTION__ << std::endl; }
template<typename C, std::size_t N>
void test(const C (&&)[N]) { std::cout << __PRETTY_FUNCTION__ << std::endl; }
template<typename C>
void test(const C (&&)[1]) { std::cout << __PRETTY_FUNCTION__ << std::endl; }
template<typename C, std::size_t N>
void test(C (&)[N]) { std::cout << __PRETTY_FUNCTION__ << std::endl; }
template<typename C>
void test(C (&)[1]) { std::cout << __PRETTY_FUNCTION__ << std::endl; }
template<typename C, std::size_t N>
void test(C (&&)[N]) { std::cout << __PRETTY_FUNCTION__ << std::endl; }
template<typename C>
void test(C (&&)[1]) { std::cout << __PRETTY_FUNCTION__ << std::endl; }

int main(){
    char str1[] = "";
    char str2[] = "test";
    test("");
    test("test");
    test(str1);
    test(str2);
    test(std::move(str1));
    test(std::move(str2));
    const char str3[] = "";
    const char str4[] = "test";
    test(std::move(str3));
    test(std::move(str4));
}

дает

void test (const C (&) [1]) [с C = char]
void test (const C (&) [N]) [с C = char;long unsigned int N = 5]
void test (C (&) [1]) [с C = char]
void test (C (&) [N]) [с C = char;long unsigned int N = 5]
void test (C (&&) [1]) [с C = char]
void test (C (&&) [N]) [с C = char;long unsigned int N = 5]
void test (const C (&&) [1]) [with C = char]
void test (const C (&&) [N]) [with C = char;long unsigned int N = 5]

1 голос
/ 30 сентября 2019

Спасибо @songyuanyao за ваш ответ, теперь я понимаю, почему test(char*&&) выбрано в двух последних случаях.Мне удалось устранить неоднозначность с помощью специализации шаблона при первой перегрузке благодаря ответу @Darklighter.

Так что я решил свою проблему, например ниже:

#include <iostream>

template <unsigned long int N>
void test(const char (&)[N]){
    std::cout << __PRETTY_FUNCTION__ << " //non-empty literal" << std::endl;
}

template <>
void test(const char (&)[1]){
    std::cout << __PRETTY_FUNCTION__ << " //empty literal" << std::endl;
}

void test(char*&&){
    std::cout << __PRETTY_FUNCTION__ << " //string variable" << std::endl;
}

int main(){
    char str1[] = "";
    char str2[] = "test";
    test("");
    test("test");
    test(str1);
    test(str2);
}

Вывод:

clang++ test.cpp -o test.out && ./test.out
void test(const char (&)[1]) //empty literal
void test(const char (&)[N]) [N = 5] //non-empty literal
void test(char *&&) //string variable
void test(char *&&) //string variable

g++ test.cpp -o test.exe && test.exe
void test(const char (&)[N]) [with long unsigned int N = 1] //empty literal
void test(const char (&)[N]) [with long unsigned int N = 5] //non-empty literal
void test(char*&&) //string variable
void test(char*&&) //string variable
...