So I am trying to implement the xorshift PRNGs as a parameterised STL-style class from random
, like e.g. std::mersenne_twister_engine
, so I can use it with those quite convenient distributions from the random
library etc.
Anyway, I have a problem overloading the operator<<
and, frankly, I am completely stumped.
The class is parameterised like this:
template <size_t __n,
int_least8_t __a, int_least8_t __b, int_least8_t __c,
uint64_t __m>
class xorshift_engine
{
...
The overload is declared as a friend
inside the class like this:
template <size_t __n_,
int_least8_t __a_, int_least8_t __b_, int_least8_t __c_,
uint64_t __m_,
typename _CharT, typename _Traits>
friend std::basic_istream<_CharT, _Traits>&
operator<< (std::basic_ostream<_CharT, _Traits>& _os,
const xorshift_engine<__n_, __a_, __b_, __c_, __m_>& _x);
and its implementation outside the class looks like this:
template <size_t __n,
int_least8_t __a, int_least8_t __b, int_least8_t __c,
uint64_t __m,
typename _CharT, typename _Traits>
std::basic_ostream<_CharT, _Traits>&
operator<< (std::basic_ostream<_CharT, _Traits>& _os,
const xorshift_engine<__n, __a, __b, __c, __m>& _x)
{
...
}
When I try to compile the following:
#include <iostream>
#include <random>
#include "xorshift.hpp"
using namespace std;
int main()
{
xorshift1024star bip(2345);
mt19937_64 bip2(2345);
cout << bip << endl;
cout << endl << bip2 << endl;
return 0;
}
(xorshift1024star
is just an instantiation of the xorshift_engine
class:
typedef xorshift_engine<16, -31, 11, 30, 1181783497276652981ULL> xorshift1024star;
) I get this error (I replaced my username in the file paths with ---
):
C:\Users\---\Documents\randgen.cpp: In function 'int main()':
C:\Users\---\Documents\randgen.cpp:11:7: error: ambiguous overload for 'operator<<' (operand types are 'std::ostream {aka std::basic_ostream<char>}' and 'xorshift1024star {aka xorshift_engine<16ull, -31, 11, 30, 1181783497276652981ull>}')
cout << bip << endl;
^
C:\Users\---\Documents\randgen.cpp:11:7: note: candidates are:
In file included from C:\Users\---\Documents\randgen.cpp:3:0:
C:\Users\---\Documents\xorshift.hpp:898:1: note: std::basic_ostream<_CharT, _Traits>& operator<<(std::basic_ostream<_CharT, _Traits>&, const xorshift_engine<__n, __a, __b, __c, __m>&) [with long long unsigned int __n = 16ull; signed char __a = -31; signed char __b = 11; signed char __c = 30; long long unsigned int __m = 1181783497276652981ull; _CharT = char; _Traits = std::char_traits<char>]
operator<< (std::basic_ostream<_CharT, _Traits>& _os,
^
C:\Users\---\Documents\xorshift.hpp:858:2: note: std::basic_istream<_CharT, _Traits>& operator<<(std::basic_ostream<_CharT, _Traits>&, const xorshift_engine<__n_, __a_, __b_, __c_, __m_>&) [with long long unsigned int __n_ = 16ull; signed char __a_ = -31; signed char __b_ = 11; signed char __c_ = 30; long long unsigned int __m_ = 1181783497276652981ull; _CharT = char; _Traits = std::char_traits<char>; long long unsigned int __n = 16ull; signed char __a = -31; signed char __b = 11; signed char __c = 30; long long unsigned int __m = 1181783497276652981ull]
operator<< (std::basic_ostream<_CharT, _Traits>& os,
^
In file included from C:/mingw-w64/x86_64-4.9.0-posix-seh-rt_v3-rev1/mingw64/x86_64-w64-mingw32/include/c++/iostream:39:0,
from C:\Users\---\Documents\randgen.cpp:1:
C:/mingw-w64/x86_64-4.9.0-posix-seh-rt_v3-rev1/mingw64/x86_64-w64-mingw32/include/c++/ostream:602:5: note: std::basic_ostream<_CharT, _Traits>& std::operator<<(std::basic_ostream<_CharT, _Traits>&&, const _Tp&) [with _CharT = char; _Traits = std::char_traits<char>; _Tp = xorshift_engine<16ull, -31, 11, 30, 1181783497276652981ull>] <near match>
operator<<(basic_ostream<_CharT, _Traits>&& __os, const _Tp& __x)
^
C:/mingw-w64/x86_64-4.9.0-posix-seh-rt_v3-rev1/mingw64/x86_64-w64-mingw32/include/c++/ostream:602:5: note: no known conversion for argument 1 from 'std::ostream {aka std::basic_ostream<char>}' to 'std::basic_ostream<char>&&'
For reference, this is how the overload is declared inside mersenne_twister_engine
in bits/random.h
:
template<typename _UIntType1,
size_t __w1, size_t __n1,
size_t __m1, size_t __r1,
_UIntType1 __a1, size_t __u1,
_UIntType1 __d1, size_t __s1,
_UIntType1 __b1, size_t __t1,
_UIntType1 __c1, size_t __l1, _UIntType1 __f1,
typename _CharT, typename _Traits>
friend std::basic_ostream<_CharT, _Traits>&
operator<<(std::basic_ostream<_CharT, _Traits>& __os,
const std::mersenne_twister_engine<_UIntType1, __w1, __n1,
__m1, __r1, __a1, __u1, __d1, __s1, __b1, __t1, __c1,
__l1, __f1>& __x);
and its implementation in bits/random.tcc
:
template<typename _UIntType, size_t __w,
size_t __n, size_t __m, size_t __r,
_UIntType __a, size_t __u, _UIntType __d, size_t __s,
_UIntType __b, size_t __t, _UIntType __c, size_t __l,
_UIntType __f, typename _CharT, typename _Traits>
std::basic_ostream<_CharT, _Traits>&
operator<<(std::basic_ostream<_CharT, _Traits>& __os,
const mersenne_twister_engine<_UIntType, __w, __n, __m,
__r, __a, __u, __d, __s, __b, __t, __c, __l, __f>& __x)
{
...
}
If I comment out the cout << bip << endl;
line in my test code, it compiles and runs without error, so this overload is fine, but mine is not.
I am completely out of ideas. What am I missing? I would be more than grateful if someone could help me.
edit: For those suggesting it's the "template friends" problem which I should solve by forward-declaring the function, I have just added
template <size_t __n,
int_least8_t __a, int_least8_t __b, int_least8_t __c,
uint64_t __m>
class xorshift_engine;
template <size_t __n,
int_least8_t __a, int_least8_t __b, int_least8_t __c,
uint64_t __m,
typename _CharT, typename _Traits>
std::basic_ostream<_CharT, _Traits>&
operator<< (std::basic_ostream<_CharT, _Traits>& _os,
const xorshift_engine<__n, __a, __b, __c, __m>& _x);
before the class body/implementation, and I get the same error upon attempting to compile.
edit2: Simplified example giving the same compile error:
//file "failclass.hpp"
#ifndef _FAILCLASS_HPP
#define _FAILCLASS_HPP
#include <iosfwd>
#include <type_traits>
template <int n>
class failclass
{
private:
static constexpr int k = n;
public:
template<int n1, typename _CharT, typename _Traits>
friend std::basic_istream<_CharT, _Traits>&
operator<< (std::basic_ostream<_CharT, _Traits>& _os,
const failclass<n1>& _x);
};
template<int n1, typename _CharT, typename _Traits>
std::basic_ostream<_CharT, _Traits>&
operator<< (std::basic_ostream<_CharT, _Traits>& _os,
const failclass<n1>& _x)
{
_os << _x.k;
return _os;
}
#endif // _FAILCLASS_HPP
Try to compile this:
//file "failtest.cpp"
#include <iostream>
#include "failclass.hpp"
using namespace std;
int main()
{
failclass<5> a;
cout << a;
return 0;
}