What is the proper way to enable my swap
in STL algorithms?
1) Member swap
. Does std::swap
use SFINAE trick to use the member swap
.
2) Free standing swap
in the same namespace.
3) Partial specialization of std::swap
.
4) All of the above.
Thank you.
EDIT: Looks like I didn't word my question clearly. Basically, I have a template class and I need STL algos to use the (efficient) swap method I wrote for that class.
swap
. Write it this way when you write "library" code and want to enable ADL (argument-dependent lookup) on swap
. Also, this has nothing to do with SFINAE.// some algorithm in your code template<class T> void foo(T& lhs, T& rhs) { using std::swap; // enable 'std::swap' to be found // if no other 'swap' is found through ADL // some code ... swap(lhs, rhs); // unqualified call, uses ADL and finds a fitting 'swap' // or falls back on 'std::swap' // more code ... }
swap
function for your class:namespace Foo { class Bar{}; // dummy void swap(Bar& lhs, Bar& rhs) { // ... } }
If swap
is now used as shown in 1), your function will be found. Also, you may make that function a friend if you absolutely need to, or provide a member swap
that is called by the free function:
// version 1 class Bar{ public: friend void swap(Bar& lhs, Bar& rhs) { // .... } }; // version 2 class Bar{ public: void swap(Bar& other) { // ... } }; void swap(Bar& lhs, Bar& rhs) { lhs.swap(rhs); } ...
std::swap
for template classes, you have to provide a free function in your namespace. Not a bad thing, if I may say so. Now, an explicit specialization is also possible, but generally you do not want to specialize a function template:namespace std { // only allowed to extend namespace std with specializations template<> // specialization void swap<Bar>(Bar& lhs, Bar& rhs) noexcept { // ... } }
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With