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;
}

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

相关推荐
百***78754 分钟前
Grok-4.1与GPT-5.2深度对比:技术差异、适用场景及Python集成指南
java·python·gpt
月光下的麦克18 分钟前
如何查案动态库版本
linux·运维·c++
Mr -老鬼21 分钟前
Java VS Rust
java·开发语言·rust
北凉军28 分钟前
java连接达梦数据库,用户名是其他库的名称无法指定库,所有mapper查询的都是以用户名相同的库内的表
java·开发语言·数据库
小六子成长记34 分钟前
【C++】:搜索二叉树的模拟实现
数据结构·c++·算法
汉克老师38 分钟前
GESP2025年9月认证C++二级真题与解析(编程题1(优美的数字))
c++·算法·整除·枚举算法·求余·拆数
程序员张338 分钟前
Mybatis条件判断某属性是否等于指定字符串
java·spring boot·mybatis
wuk99838 分钟前
基于C#与三菱PLC通过TCPIP实现MC协议通信示例
java·网络·c#
沛沛老爹1 小时前
Web转AI架构篇 Agent Skills vs MCP:工具箱与标准接口的本质区别
java·开发语言·前端·人工智能·架构·企业开发
码农小卡拉1 小时前
Maven与Gradle选型指南:如何匹配项目的依赖管理需求
java·gradle·maven