Lvalue не соответствует (&&) без шаблона, но соответствует (T &&) с шаблоном? - PullRequest
0 голосов
/ 27 декабря 2018
#include <bits/stdc++.h>
using namespace std;

template<class T = string>
void f(T &&s) {
    cout << s << endl;
}

int main() {
    string s("1234");
    f(s);
    f("1234");

    return 0;
}

Может быть скомпилировано.

#include <bits/stdc++.h>
using namespace std;

void f(string &&s) {
    cout << s << endl;
}

int main() {
    string s("1234");
    f(s);
    f("1234");

    return 0;
}

Я заменяю T на string, код не может быть скомпилирован.

ошибка:

❯ g++-8 -std=c++11 a.cpp && ./a.out
a.cpp: In function 'int main()':
a.cpp:10:11: error: cannot bind rvalue reference of type 'std::__cxx11::string&&' {aka 'std::__cxx11::basic_string<char>&&'} to lvalue of type 'std::__cxx11::string' {aka 'std::__cxx11::basic_string<char>'}
         f(s);
           ^
a.cpp:4:10: note:   initializing argument 1 of 'void f(std::__cxx11::string&&)'
     void f(string &&s) {
          ^

Я так растерялся.

1 Ответ

0 голосов
/ 27 декабря 2018

Существуют некоторые исключения для вывода типа шаблона.Если шаблон функции получает ссылку rvalue, и мы передали ссылку lvalue, компилятор выведет ее как ссылку lvalue. По этой причине std :: move работает правильно.

template <typename T>
typename remove_reference<T>::type&& move(T&& t)
{
    return static_cast<typename remove_reference<T>::type&&>(t);
}
...