Почему неявное преобразование типов не работает при выводе шаблона? - PullRequest
0 голосов
/ 09 ноября 2018

В следующем коде я хочу вызвать функцию шаблона путем неявного преобразования int в Scalar<int> объект.

#include<iostream>
using namespace std;

template<typename Dtype>
class Scalar{
public:
  Scalar(Dtype v) : value_(v){}
private:
  Dtype value_;
};

template<typename Dtype>
void func(int a, Scalar<Dtype> b){ 
  cout << "ok" <<endl;
}

int main(){
  int a = 1;
  func(a, 2); 
  //int b = 2;
  //func(a, b);
  return 0;
}

Почему вычет / замена аргумента шаблона не удалась? И закомментированные коды тоже неверны.

test.cpp: In function ‘int main()’:
test.cpp:19:12: error: no matching function for call to ‘func(int&, int)’
   func(a, 2);
            ^
test.cpp:19:12: note: candidate is:
test.cpp:13:6: note: template<class Dtype> void func(int, Scalar<Dtype>)
 void func(int a, Scalar<Dtype> b){
      ^
test.cpp:13:6: note:   template argument deduction/substitution failed:
test.cpp:19:12: note:   mismatched types ‘Scalar<Dtype>’ and ‘int’
   func(a, 2);

Ответы [ 2 ]

0 голосов
/ 09 ноября 2018
template<typename Dtype>
void func(int a, Scalar<Dtype> b){ 
  cout << "ok" <<endl;
}
template<typename Dtype>
void func(int a, Dtype b){ 
  func(a, Scalar<Dtype>(std::move(b)));
}

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

Преобразование выполняется позже, при разрешении перегрузки и времени вызова функции.

Здесь мы добавляем еще одну перегрузку, которая явно перенаправляет на ту, которую вы хотите.

0 голосов
/ 09 ноября 2018

Потому что вывод аргумента шаблона не такой умный: он (по замыслу) не учитывает определяемые пользователем преобразования. И int -> Scalar<int> является пользовательским преобразованием.

Если вы хотите использовать TAD, вам нужно преобразовать свой аргумент на сайте вызывающей стороны:

func(a, Scalar<int>{2}); 

или определите руководство по удержанию 1 для Scalar и позвоните f:

func(a, Scalar{2}); // C++17 only

Кроме того, вы можете явно создать экземпляр f:

func<int>(a, 2); 

1) Достаточно руководства по выводу по умолчанию: demo .

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...