Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.68/34: Рейтинг темы: голосов - 34, средняя оценка - 4.68
337 / 185 / 80
Регистрация: 22.08.2013
Сообщений: 724

Шаблонный оператор присваивания в шаблонном классе

29.12.2014, 23:16. Показов 7168. Ответов 6
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Почему, в таком коде, срабатывает оператор присваивания по умолчанию?
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
template <class T>
class A
{
    T x;
 
public:
    A(T a) : x(a)
    {}
    A()
    {}
    ~A()
    {}
    
    template <class T>
    A<T> & operator=(const A<T>& right)
    {
        cout << "= template" << endl;
        x = right.x;
        return *this;
    }
};
 
int main()
{
   A <int> a(4);
   A <int> b(5);
  
   a = b;
   
   return 0;
}
0
Лучшие ответы (1)
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
29.12.2014, 23:16
Ответы с готовыми решениями:

Шаблонный друг в шаблонном классе
Добрый день! Создаю шаблонный класс, в котором перегружаю сложение: сложение какого-то нешаблонного типа и шаблонного объекта моего класса,...

Шаблонный конструктор в шаблонном классе
Не получается определить этот конструктор. Класс шаблонный и еще конструктор тоже шаблонный Подскажите как правильно сделать ...

Параметры в шаблонном классе
Почему в шаблонном классе при описании функции надо два раза указывать тип параметра? В 1-ой строчке есть T и во 2-ой тоже написано T ...

6
4 / 4 / 1
Регистрация: 30.11.2010
Сообщений: 69
29.12.2014, 23:45
Во первых:
Цитата Сообщение от nord_v Посмотреть сообщение
template <class T>
* * A<T> & operator=(const A<T>& right)
* * {
* * * * cout << "= template" << endl;
* * * * x = right.x;
* * * * return *this;
* * }
надо:
C++
1
2
3
4
5
6
7
8
9
10
template <class T>
    A<T> & operator=(const A<T>& right)
    {  
        cout << "= template" << endl;
 
        if( &right != this)
         x = right.x;
 
       return *this;
    }
Во вторых:
Если оператора перегрузки нет( Вы его не написали ), то компилятор создаёт побитовую копию объекта самостоятельно. Если же он у Вас есть, то он вызывается. В Вашем примере всё должно работать.
0
19500 / 10105 / 2461
Регистрация: 30.01.2014
Сообщений: 17,818
29.12.2014, 23:46
Лучший ответ Сообщение было отмечено nord_v как решение

Решение

Цитата Сообщение от nord_v Посмотреть сообщение
Почему, в таком коде, срабатывает оператор присваивания по умолчанию?
Такое вообще не должно компилироваться. У тебя параметр шаблона в классе имеет то же имя, что и параметр оператора=.
Но логику я объясню. Так как T в операторе= - это другое T. То этот оператор считается шаблонным, а шаблонная функция всегда имеет более низкий приоритет по сравнению с нешаблонной (та, что доступна по-умолчанию). Если ты хотел определить шаблонный оператор= правильно, то это надо было делать так:
C++
1
2
3
4
5
6
    template <class T1> // параметр шаблона имеет другое имя
    A<T> & operator=(const A<T1> & right)
    {
        x = right.x;
        return *this;
    }
А вот нешаблонный оператор присваивания определяется так:
C++
1
2
3
4
5
    A<T> & operator=(const A<T> & right)
    {
        x = right.x;
        return *this;
    }
Шаблонный оператор будет вызываться например в таких ситуациях:
C++
1
2
3
4
   A <int> a(4);
   A <double> b(5.3);
 
   a = b;
А нешаблоный, определенный самостоятельно, либо сгенерированный по-умолчанию, в таких:
C++
1
2
3
4
   A <int> a(4);
   A <int> b(5);
 
   a = b;
1
Эксперт С++
 Аватар для hoggy
8973 / 4319 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
29.12.2014, 23:53
Цитата Сообщение от nord_v Посмотреть сообщение
Почему, в таком коде, срабатывает оператор присваивания по умолчанию?
Ваш код не по стандарту.


C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
#include <iostream>
using namespace std;
 
template <class T> class A
{
    T x;
public:
    A(T a):x(a){}
    A(){} 
    ~A(){}
    
    
    A& operator=(const A& right)  //<--- переопределяем дефолтный
    {
        cout << "user" << endl;
        x = right.x;
        return *this;
    }
    
    template<class U>  //<--- U, а не T, иначе нарветесь на 
                       //error:  shadows template parm ‘class T’
        
    //возвращаем A&, а не A<U>&    
    A& operator=(const A<U>& right)
    {
        cout << "template" << endl;
        x = right.x;
        return *this;
    }
};
 
int main()
{
   A <int> a(4);
   A <int> b(5);
  
   a = b;
   
   return 0;
}
Ответ на ваш вопрос:
Компилятор генерирует дефолтный оператор присваивания в случае, если его не определил пользователь.
Он имеет вид: T& operator=(const T& rhs)

Шаблонная версия оператора не соответствует этому виду, и поэтому, компилятор сгенерировал дефолтную версию.
0
337 / 185 / 80
Регистрация: 22.08.2013
Сообщений: 724
30.12.2014, 00:26  [ТС]
Цитата Сообщение от DrOffset Посмотреть сообщение
Такое вообще не должно компилироваться.
Тогда бы я не выкладывал код, и не спрашивал. Студия компилирует.
Цитата Сообщение от DrOffset Посмотреть сообщение
Так как T в операторе= - это другое T. То этот оператор считается шаблонным, а шаблонная функция всегда имеет более низкий приоритет по сравнению с нешаблонной (та, что доступна по-умолчанию).
То есть, если так:
C++
1
2
3
4
5
6
template <class T1>
A<T1> & operator=(const A<T1>& right)
{
    x = right.x;
    return *this;
}
то всегда будет срабатывать оператор по умолчанию, потому что у него (как не шаблонного) приоритет больше?
0
19500 / 10105 / 2461
Регистрация: 30.01.2014
Сообщений: 17,818
30.12.2014, 00:33
Цитата Сообщение от nord_v Посмотреть сообщение
Студия компилирует.
Я в курсе. Она компилирует, но по стандарту не должна. Это из-за отсутствия two-phase lookup. Если интересно, эта особенность легко гуглится.
Цитата Сообщение от nord_v Посмотреть сообщение
то всегда будет срабатывать оператор по умолчанию, потому что у него (как не шаблонного) приоритет больше?
Этот код по смыслу эквивалентен тому, что написан у тебя. Поведение верное, вызывается оператор по-умолчанию, потому что он разумеется нешаблонный.

PS. обрати внимание, на возвращаемое значение. У меня там T, а ты переписал с T1. Если T = int, а T1 = double, то у типа не найдется подходящего способа преобразовать T<int> (это *this), к ссылке на T<double>. Внимательней к деталям
1
337 / 185 / 80
Регистрация: 22.08.2013
Сообщений: 724
30.12.2014, 00:44  [ТС]
Цитата Сообщение от DrOffset Посмотреть сообщение
PS. обрати внимание, на возвращаемое значение. У меня там T, а ты переписал с T1.
Это не твой код, а мой, из первого поста, переписал, чтобы по стандарту было, и ещё раз уточнил ответ (именно по моему коду). Насчёт твоего кода, с разными типами, я всё понял. Благодарю за разъяснения!
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
30.12.2014, 00:44
Помогаю со студенческими работами здесь

Ошибки в шаблонном классе
Добрый день. Реализовал класс динамического массива. При проверке его работы столкнулся с ошибками, которые мне пока не удаётся исправить....

Ошибка в шаблонном классе
Помогите исправить ошибку в 12 строчке: error C2248: CMyClass&lt;T&gt;::m_value: невозможно обратиться к private член, объявленному в классе...

String в шаблонном классе
Как сделать так, чтобы в Map&lt;int, std::string&gt; A; работало со string'ом С интами чарами и т.д. все хорошо работает, а на string прога...

Работа с матрицей в шаблонном классе
Часть работы разобрал, не получается сделать такое задание как: создать метод который дает возможность изменять кол-во строк или столбцов...

Перегрузка оператора в шаблонном классе
Здравствуйте! Есть шаблонный класс Array, описывающий массив. Такая проблема: нужно перегрузить оператор &quot;&lt;&quot;, который...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
7
Ответ Создать тему
Новые блоги и статьи
My Business CRM
MaGz GoLd 16.04.2026
Всем привет, недавно возникла потребность создать CRM, для личных нужд. Собственно программа предоставляет из себя базу данных клиентов, в которой можно фиксировать звонки, стадии сделки, а также. . .
Знаешь почему 90% людей редко бывают счастливыми?
kumehtar 14.04.2026
Потому что они ждут. Ждут выходных, ждут отпуска, ждут удачного момента. . . а удачный момент так и не приходит.
Фиксация колонок в отчете СКД
Maks 14.04.2026
Фиксация колонок в СКД отчета типа Таблица. Задача: зафиксировать три левых колонки в отчете. Процедура ПриКомпоновкеРезультата(ДокументРезультат, ДанныеРасшифровки, СтандартнаяОбработка) / / . . .
Настройки VS Code
Loafer 13.04.2026
{ "cmake. configureOnOpen": false, "diffEditor. ignoreTrimWhitespace": true, "editor. guides. bracketPairs": "active", "extensions. ignoreRecommendations": true, . . .
Оптимизация кода на разграничение прав доступа к элементам формы
Maks 13.04.2026
Алгоритм из решения ниже реализован на нетиповом документе, разработанного в конфигурации КА2. Задачи, как таковой, поставлено не было, проделанное ниже исключительно моя инициатива. Было так:. . .
Контроль заполнения и очистка дат в зависимости от значения перечислений
Maks 12.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "ПланированиеПерсонала", разработанного в конфигурации КА2. Задача: реализовать контроль корректности заполнения дат назначения. . .
Архитектура слоя интернета для сервера-слоя.
Hrethgir 11.04.2026
В продолжение https:/ / www. cyberforum. ru/ blogs/ 223907/ 10860. html Знаешь что я подумал? Раз мы все источники пишем в голове ветки, то ничего не мешает добавить в голову такой источник, который сам. . .
Подстановка значения реквизита справочника в табличную часть документа
Maks 10.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "ПланированиеПерсонала", разработанного в конфигурации КА2. Задача №1: при указании работ (справочник РаботыПоРемонтуСпецтехники),. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru