类和成员函数模板专业化出错了

时间:2017-08-15 13:48:55

标签: c++ class templates visual-studio-2013 template-specialization

我正在尝试专门化一个Color类及其成员函数to(),以便从一个颜色空间转换为另一个颜色空间并返回。这就是我到目前为止所拥有的:

enum ColorSpace {
    BGR, RGB, LAB, HSV, XYZ, YUV, HSL, CMY, YIQ, LUV, HLAB, LCH, OHTA
};

template<ColorSpace _CS, typename _Dtp> class Color;

template<typename _Dtp>
class Color<RGB, _Dtp> {

public:

    Color(_Dtp R, _Dtp G, _Dtp B) : R(R), G(G), B(B) { ;; };
    Color() : Color(0, 0, 0) { ;; };
    ~Color() { ;; };

    _Dtp R, G, B;

    template<ColorSpace _CS, typename _Dtp2 = _Dtp>
    Color<_CS, _Dtp2> to<>();

    template<typename _Dtp2 = _Dtp>
    Color<HSV, _Dtp2> to<HSV, _Dtp2>() {
        Color<HSV, _Dtp2> res;

        rgb2hsv(R, G, B, res.H, res.S, res.V);

        return res;
    }

};

template<typename _Dtp = double>
class Color<HSV, _Dtp> {

public:

    Color(_Dtp H, _Dtp S, _Dtp V) : H(H), S(S), V(V) { ;; };
    Color() : Color(0, 0, 0) { ;; };
    ~Color() { ;; };

    _Dtp H, S, V;

    template<ColorSpace _CS, typename _Dtp2 = _Dtp>
    Color<_CS, _Dtp2> to<>();

    template<typename _Dtp2 = _Dtp>
    Color<RGB, _Dtp2> to<RGB, _Dtp2>() {
        Color<RGB, _Dtp2> res;

        hsv2rgb(H, S, V, res.R, res.G, res.B);

        return res;
    }

};

每个颜色空间都有一个类。每个类都有针对每个可能的转换的专用函数to()。但是,由于显而易见的原因,我只发布其中的2个。所以,当我尝试使用转换函数时:

Color<RGB, double> c(255, 255, 125);

Color<HSV, double> c2 = c.to<HSV, double>();

我收到以下错误:

  

错误:课程&#34;颜色&#34;没有会员&#34;到&#34;

当我尝试使用类Color的默认类型时,我的最后一个问题出现了:

template<typename _Dtp = double>
class Color<RGB, _Dtp> {
...
...

在这种情况下,我甚至无法写下这个:

Color<RGB> c;

它给了我这个错误:

  

错误:预期声明

2 个答案:

答案 0 :(得分:1)

您的代码应如下所示:

enum ColorSpace {
    BGR, RGB, LAB, HSV, XYZ, YUV, HSL, CMY, YIQ, LUV, HLAB, LCH, OHTA
};

template<ColorSpace _CS, typename _Dtp> class Color;

template<typename _Dtp>
class Color<RGB, _Dtp> {

public:

    Color( _Dtp R, _Dtp G, _Dtp B ) : R( R ), G( G ), B( B ) { ;; };
    Color() : Color( 0, 0, 0 ) { ;; };
    ~Color() { ;; };

    _Dtp R, G, B;


    template<ColorSpace _CSO, typename _Dtp2> struct STo;
    template<typename _Dtp2> struct STo<HSV, _Dtp2>
    {
        Color<HSV, _Dtp2> to()
        {
            Color<HSV, _Dtp2> res;

            rgb2hsv( R, G, B, res.H, res.S, res.V );

            return res;
        }
    };

    template<ColorSpace _CSO, typename _Dtp2>
    Color<_CSO, _Dtp2> to()
    {
        STo<_CSO, _Dtp2> object;
        return object.to();
    }

};

template<typename _Dtp>
class Color<HSV, _Dtp> {

public:

    Color( _DtpH, _DtpS, _DtpV ) : H( H ), S( S ), V( V ) { ;; };
    Color() : Color( 0, 0, 0 ) { ;; };
    ~Color() { ;; };

    _DtpH, S, V;

    template<ColorSpace _CSO, typename _Dtp2> struct STo;
    template<typename _Dtp2> struct STo<RGB, _Dtp2>
    {
        Color<RGB, _Dtp2> to()
        {
            Color<RGB, _Dtp2> res;

            hsv2rgb( H, S, V, res.R, res.G, res.B );

            return res;
        }
    };

    template<ColorSpace _CSO, typename _Dtp2>
    Color<_CSO, _Dtp2> to()
    {
        STo<_CSO, _Dtp2> object;
        return object.to();
    }
};

你无法在没有定义的情况下编写函数专业化。这就是为什么你应该删除括号&lt;&gt;从&#34;到&#34;功能。但是在这种情况下我们遇到了另一个问题 - 我们不能对函数使用部分特化,但对于类可以。在这种情况下,我使用&#34;来&#34;功能类似于可能的结构专业化的条目&#34; STo&#34;

答案 1 :(得分:0)

添加构造函数进行转换可能更简单:

enum ColorSpace {
    BGR, RGB, LAB, HSV, XYZ, YUV, HSL, CMY, YIQ, LUV, HLAB, LCH, OHTA
};

template <ColorSpace, typename = double> class Color;

template <typename T>
class Color<RGB, T> {
public:

    Color(T R, T G, T B) : R(R), G(G), B(B) {}
    Color() : Color(0, 0, 0) {}
    Color(const Color&rhs) = default;
    ~Color() = default;

    T R, G, B;

    template <typename T2>
    /*explicit*/ Color(const Color<HSV, T2>& rhs) 
    {
        hsv2rgb(rhs.H, rhs.S, rhs.V, R, G, B);
    }
};
template <typename T>
class Color<HSV, T> {
public:
    Color(T H, T S, T V) : H(H), S(S), V(V) {}
    Color() : Color(0, 0, 0) {}
    Color(const Color&rhs) = default;
    ~Color() = default;

    T H, S, V;

    template <typename T2>
    /*explicit*/ Color(const Color<RGB, T2>& rhs) 
    {
        rgb2hsv(rhs.R, rhs.G, rhs.B, H, S, V);
    }

};

可能的用法:

Color<RGB, double> c(255, 255, 125);

Color<HSV, double> c2{c};