聊聊C# CLR中那些大量的友元函数,友元类的底层玩法

一:理解友元

如果你看过 CLR 代码就会发现这里面有很多的 friend 修饰符, 比如: MethodTable.cpp 文件下。

class MethodTable
{
    /************************************
     *  FRIEND FUNCTIONS
     ************************************/
    // DO NOT ADD FRIENDS UNLESS ABSOLUTELY NECESSARY
    // USE ACCESSORS TO READ/WRITE private field members

    // Special access for setting up String object method table correctly
    friend class ClassLoader;
    friend class JIT_TrialAlloc;
    friend class Module;
    friend class EEClass;
    friend class MethodTableBuilder;
    friend class CheckAsmOffsets;
 ...
}

这对于 C# 来说是个新鲜玩意,这一篇我们就来探索下 friend 的底层玩法。

我个人觉得类中引入了 friend 修饰符会让类之间更具有人情味,在现实生活中,人都是一个独立的个体,什么该拿出来,什么不该拿出来,自己心里很有数, 这就是很好的封装性,但这个封装性最终还是取决于对面是什么人,如果是 亲戚 在遇到困难的时候你绝对可以慷概援助, 陌生人 你可能就需要考虑再三了。

那这个 friend 就是用来标记 谁和谁 是亲戚,谁和谁是朋友,大家相互利用对方的资源,一起做大当地的市场。

在 C++ 中存在两种友元,一个叫 友元函数,一个叫 友元类,我们逐一了解下。

一:C++ 中的友元函数

1. 一个小例子

为了方便说明,先上一段代码:

#include<iostream>

using namespace std;

class Person {

public:
 Person(int money) :money(money) {}
private:
 string name;
 int money;
};

int getMoney(const Person& person)
{
 int money = person.money;

 return money;
}

int main() {

 Person p(100);

 int age = getMoney(p);

 cout << "money: " << age << endl;
}

假定上面的 getMoney 函数想直接提取 Person 的私有字段 money ,这么粗暴的做法,我想是个人都会拒绝,何况是函数了,在现实社会实践中,总有办法能偷出来的 😄,代码中也是一样,比如用 指针 直接暴力偷,如下图所示:

8003072d5b554efb0d0c3574c94ba038.png

这么玩的话其实没什么意思,合规的做法是将 getMoney() 设置为 Person 的友元函数。

2. 使用友元函数

使用友元函数之前,我们先看下 友元函数 的定义格式。

class Person {

 //友元列表
 friend <返回类型> <函数名> (<参数列表>);

private:
 xxxx

public:
 xxx
};

看起来非常简单,就是在 的开始处定义一个 友元列表 即可,接下来我们修改代码如下:

#include<iostream>

using namespace std;

class Person {

 friend int getMoney(const Person& person);

public:
 Person(int money) :money(money) {}
private:
 string name;
 int money;
};

//友元函数
int getMoney(const Person& person)
{
 int money = person.money;

 return money;
}

int main() {

 Person p(100);

 int age = getMoney(p);

 cout << "money: " << age << endl;
}
8da54856ae53118e6b9c69cd3515d16c.png

本质上来说,能不能提取得到,就是 编译器 的一个限定而已,在汇编上没有任何不一样的地方,都是从栈单元中获取数值。

二:C++ 中的友元类

1. 一个小例子

除了将 函数 设置为友元,当然还可以将 设置为友元了, 格式也基本差不多。

class Person {

 //友元列表
 friend class 类名;

private:
 xxxx

public:
 xxx
};

有了这个模板,接下来就可以上代码了。

#include <iostream>

using namespace std;

class Person {

 friend class Test;

public:
 Person(int money) :money(money) {}
private:
 string name = "hello";
 int money;
};

class Test {
public:
 int getMoney(const Person& person) {

  int sum = person.money;

  return sum;
 }
};

int main() {

 Person p(100);

 Test t;

 int money = t.getMoney(p);

 cout << "money: " << money << endl;
}
799f0fe7877ef7a4fbc13d7de438fce8.png

三:总结

友元的思想还是挺符合现实社会的,用起来也很简单,最后就是友元只是一个编译器限定,在汇编层没有什么不同,都是从 栈单元 上提取出目标值。

好了,本篇就聊这么多,希望对你有帮助。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值