C++类模版与友元

全局函数类内实现-直接在类内声明友元即可

全局函数类外实现-需要提前让编译器知道全局函数的存在

cpp 复制代码
#include <iostream>
using namespace std;

//通过全局函数来打印Person的信息


template<class T1,class T2>
class Person{

    //全局函数,类内实现
    friend void printPerson(Person<T1,T2> p)
    {
        cout << "姓名:" << p.m_Name << " 年龄:" << p.m_Age << endl;
    }

    public:
        Person(T1 name,T2 age)
        {
            this->m_Name=name;
            this->m_Age=age;
        }


    private:
        T1 m_Name;
        T2 m_Age;

};
//1.全局函数在类内实现
void test01()
{
    Person<string,int> p("Tom",20);
    printPerson(p);
}
int main()
{
    test01();
    return 0;
}

你看,其实一开始m_Name和m_Age是一个私有的属性,因为其前面有private关键字,因此前面加上friend,也就是友元的全局函数才能访问这个私有属性。

全局函数在类外实现的时候

cpp 复制代码
#include <iostream>
using namespace std;

//通过全局函数来打印Person的信息
template<class T1,class T2>
class Person{

    //全局函数,类内实现
    friend void printPerson2(Person<T1,T2> p);
    // {
    //     cout << "姓名:" << p.m_Name << " 年龄:" << p.m_Age << endl;
    // }

    public:
        Person(T1 name,T2 age)
        {
            this->m_Name=name;
            this->m_Age=age;
        }
    private:
        T1 m_Name;
        T2 m_Age;

};
//1.全局函数在类内实现
// void test01()
// {
//     Person<string,int> p("Tom",20);
//     printPerson(p);
// }
//2.全局函数在类外实现,因为这是一个全局函数,因此其没有必要加作用域
template<class T1,class T2>
void printPerson2(Person<T1,T2> p)
{
    cout << "类外实现的姓名:" << p.m_Name << " 类外实现的年龄:" << p.m_Age << endl;
}
//全局函数在类外实现的测试
void test02()
{
    Person<string,int> p("Jerry",28);
    printPerson2(p);
}
int main()
{
    test02();
    return 0;
}

其会出现一个无法解析的错误。

cpp 复制代码
    friend void printPerson2(Person<T1,T2> p);

这是一个普通函数的函数声明。

但是下面这是一个函数模版的实现,因此就导致了这两个不是一个东西。因此我们需要告诉编译器这是一个函数模版的声明。

cpp 复制代码
//2.全局函数在类外实现,因为这是一个全局函数,因此其没有必要加作用域
template<class T1,class T2>
void printPerson2(Person<T1,T2> p)
{
    cout << "类外实现的姓名:" << p.m_Name << " 类外实现的年龄:" << p.m_Age << endl;
}
cpp 复制代码
#include <iostream>
using namespace std;

template<class T1,class T2>
class Person;

template<class T1,class T2>
void printPerson2(Person<T1,T2> p)
{
    cout << "类外实现的姓名:" << p.m_Name << " 类外实现的年龄:" << p.m_Age << endl;
}

//通过全局函数来打印Person的信息
template<class T1,class T2>
class Person{

    //全局函数,类内实现
    //加空模版的参数列表
    friend void printPerson2(Person<T1,T2>p)
    {
        cout << "姓名:" << p.m_Name << " 年龄:" << p.m_Age << endl;
    }

    friend void printPerson2<>(Person<T1,T2> p);

    public:
        Person(T1 name,T2 age)
        {
            this->m_Name=name;
            this->m_Age=age;
        }
    private:
        T1 m_Name;
        T2 m_Age;

};
//1.全局函数在类内实现
// void test01()
// {
//     Person<string,int> p("Tom",20);
//     printPerson(p);
// }
//2.全局函数在类外实现,因为这是一个全局函数,因此其没有必要加作用域
template<class T1,class T2>
void printPerson2(Person<T1,T2>& p)
{
    cout << "类外实现的姓名:" << p.m_Name << " 类外实现的年龄:" << p.m_Age << endl;
}
//全局函数在类外实现的测试
void test02()
{
    Person<string,int> p("Jerry",28);
    printPerson2(p);
}
int main()
{
    test02();
    return 0;
}

一般没有特殊需求的话,直接就写全局函数配合类内实现就完事儿了,用法非常简单,而且编译器可以直接识别。

相关推荐
Mr_WangAndy6 分钟前
C++设计模式_结构型模式_适配器模式Adapter
c++·设计模式·适配器模式·c++设计模式
bkspiderx12 分钟前
C++设计模式之结构型模式:代理模式(Proxy)
c++·设计模式·代理模式
1710orange12 分钟前
java设计模式:适配器模式
java·设计模式·适配器模式
2401_8414956440 分钟前
【自然语言处理】Universal Transformer(UT)模型
人工智能·python·深度学习·算法·自然语言处理·transformer·ut
RainbowSea1 小时前
9. Spring AI 当中对应 MCP 的操作
java·spring·ai编程
浅川.251 小时前
xtuoj 整数分类
算法
RainbowSea1 小时前
10. Spring AI + RAG
java·spring·ai编程
寻星探路1 小时前
Java EE初阶启程记05---线程安全
java·开发语言·java-ee
Moshow郑锴2 小时前
Java 中配置 Selenium UI 自动化测试 并生成 Cucumber 报告
java·selenium·测试工具
nlog3n2 小时前
分布式短链接系统设计方案
java·分布式