После Что такое идиома копирования и подкачки и Как предоставить функцию подкачки для моего класса, я попытался реализовать подкачку функция, как в последнем, приняла вариант ответа номер 2 (имея бесплатную функцию, которая вызывает функцию-член) вместо прямой дружественной бесплатной функции в прежней ссылке.
Однако следующее не компилируется
#include <iostream>
// Uncommenting the following two lines won't change the state of affairs
// class Bar;
// void swap(Bar &, Bar &);
class Bar {
public:
Bar(unsigned int bottles=0) : bottles(bottles) { enforce(); } // (1)
Bar(Bar const & b) : bottles(b.bottles) { enforce(); } // (1)
Bar & operator=(Bar const & b) {
// bottles = b.bottles;
// enforce();
// Copy and swap idiom (maybe overkill in this example)
Bar tmp(b); // but apart from resource management it allows (1)
// to enforce a constraint on the internal state
swap(*this, tmp); // Can't see the swap non-member function (2)
return *this;
}
void swap(Bar & that) {
using std::swap;
swap(bottles, that.bottles);
}
friend std::ostream & operator<<(std::ostream & out, Bar const & b) {
out << b.bottles << " bottles";
return out;
}
private:
unsigned int bottles;
void enforce() { bottles /=2; bottles *= 2; } // (1) -- Ensure the number of bottles is even
};
void swap(Bar & man, Bar & woman) { // (2)
man.swap(woman);
}
int main () {
Bar man (5);
Bar woman;
std::cout << "Before -> m: " << man << " / w: " << woman << std::endl;
swap(man, woman);
std::cout << "After -> m: " << man << " / w: " << woman << std::endl;
return 0;
}
Я знаю, что идиома копирования и подкачки здесь излишня, но она также позволяет наложить некоторые ограничения на внутреннее состояние с помощью конструктора копирования (1) (более конкретным примером может быть сохранение дроби в сокращенной форме). К сожалению, это не компилируется, потому что единственным кандидатом на (2), который видит компилятор, является функция-член Bar::swap. Я застрял с подходом функции друга, не являющегося членом?
РЕДАКТИРОВАТЬ: перейдите к моему ответу ниже, чтобы увидеть, что у меня получилось, благодаря всем ответам и комментариям по этому вопросу.
this->swap(other)
? - person Yakk - Adam Nevraumont   schedule 15.03.2016