A <t>的朋友也可以是A <a <t>&gt;的朋友吗?

时间:2015-11-01 16:07:00

标签: c++ c++14 friend friend-function

请考虑以下代码:

#include <vector>

template<typename T> class Container;
template<typename T> Container<Container<T>> make_double_container(const std::vector<std::vector<T>>&);

template<typename T>
class Container {
    std::vector<T> v;
    friend Container<Container<T>> make_double_container<T>(const std::vector<std::vector<T>>&);

public:
    Container() {}
    explicit Container(std::vector<T> v) : v(v) {}
};

template<typename T>
Container<Container<T>> make_double_container(const std::vector<std::vector<T>>& v) {
    Container<Container<T>> c;
    for(const auto& x : v) {
        c.v.push_back(Container<T>(x));
    }
    return c;
}

int main() {
    std::vector<std::vector<int>> v{{1,2,3},{4,5,6}};
    auto c = make_double_container(v);
    return 0;
}

编译器告诉我:

main.cpp: In instantiation of 'Container<Container<T> > make_double_container(const std::vector<std::vector<T> >&) [with T = int]':
main.cpp:27:37:   required from here
main.cpp:8:20: error: 'std::vector<Container<int>, std::allocator<Container<int> > > Container<Container<int> >::v' is private
     std::vector<T> v;
                    ^
main.cpp:20:9: error: within this context
         c.v.push_back(Container<T>(x));

我认为这是正确的,因为make_double_containerContainer<T>的朋友,而不是Container<Container<T>>的朋友。如何在这种情况下让make_double_container工作?

2 个答案:

答案 0 :(得分:10)

显然,您可以将make_double_container的所有专业化作为朋友:

template <typename U>
friend Container<Container<U>> make_double_container(const std::vector<std::vector<U>>&);

如果您希望将友谊保持在最低限度而不需要部分专业化等,请尝试

template <typename> struct extract {using type=void;};
template <typename U> struct extract<Container<U>> {using type=U;};
friend Container<Container<typename extract<T>::type>>
     make_double_container(const std::vector<std::vector<typename extract<T>::type>>&);

Demo

答案 1 :(得分:3)

make_double_container定义为S的模板函数似乎可以使其编译和工作。

template<typename T>
class Container {
    std::vector<T> v;

    template<class S>
    friend Container<Container<S>> make_double_container(const std::vector<std::vector<S>>&);

public:
    Container() {}
    explicit Container(std::vector<T> v) : v(v) {}
};

http://coliru.stacked-crooked.com/a/bdc23a0451a2125b

当编译器在结构中看到某些内容时:

template<class T>
class X{};

当您指定T时,它会对类进行实例化,并将类型名为T的所有内容替换为指定的类型。

当你写

 Container<Container<T>> c;

T实际上是Container<T>,而make_double_container是进入

Container<Container<Container<T>>> make_double_container(const std::vector<std::vector<Container<T>>>&); 

然后(在主要内部)进入

Container<Container<Container<int>>> make_double_container(const std::vector<std::vector<Container<int>>>&); 

将友谊改为:

 template<class S>
    friend Container<Container<S>> make_double_container(const std::vector<std::vector<S>>&);

你强制编译器从S的模板中找出Container<Container<T>>的内容,然后找出S的正确类型,即int < / p>