什么是C#相当于朋友?

时间:2008-10-15 13:41:08

标签: c#

  

可能重复:
  Why does C# not provide the C++ style ‘friend’ keyword?

我希望类的私有成员变量可以被Tester类访问,而不会将它们暴露给其他类。

在C ++中,我只是将Tester类声明为朋友,如何在C#中执行此操作?有人可以举个例子吗?

5 个答案:

答案 0 :(得分:245)

没有直接相当于“朋友” - 最接近的可用(并且不是非常接近)是InternalsVisibleTo。我只使用过这个属性进行测试 - 它非常方便!

示例:放入AssemblyInfo.cs

[assembly: InternalsVisibleTo("OtherAssembly")]

答案 1 :(得分:70)

壁橱等效于创建一个嵌套类,它将能够访问外部类的私有成员。像这样:

class Outer
{
    class Inner
    {
       // This class can access Outer's private members
    }
}

或者如果您更喜欢将Inner类放在另一个文件中:

Outer.cs
partial class Outer
{
}


Inner.cs
partial class Outer
{
    class Inner
    {
       // This class can access Outer's private members
    }
}

答案 2 :(得分:41)

采取一种非常常见的模式。 Class Factory制作小部件。工厂类需要与内部进行混淆,因为它是工厂。两者都在同一个文件中实现,并且根据设计和愿望与性质,紧密耦合的类 - 事实上,Widget实际上只是工厂的输出类型。

在C ++中,让Factory成为Widget类的朋友。

在C#中,我们能做些什么?我遇到的唯一合适的解决方案是创建一个接口IWidget,它只公开公共方法,并让Factory返回IWidget接口。

这涉及相当多的单调乏味 - 在界面中再次暴露所有自然公共财产。

答案 3 :(得分:14)

C#中没有'friend'关键字,但测试私有方法的一个选项是使用System.Reflection来获取方法的句柄。这将允许您调用私有方法。

给定一个具有此定义的类:

public class Class1
{
    private int CallMe()
    {
        return 1;
    }
}

您可以使用以下代码调用它:

Class1 c = new Class1();
Type class1Type = c.GetType();
MethodInfo callMeMethod = class1Type.GetMethod("CallMe", BindingFlags.Instance | BindingFlags.NonPublic);

int result = (int)callMeMethod.Invoke(c, null);

Console.WriteLine(result);

如果您使用的是Visual Studio Team System,则可以通过右键单击该方法并选择“创建单元测试...”来让VS自动生成包含私有访问器的代理类

答案 4 :(得分:6)

如果有权访问的类位于另一个包中,并且您公开的方法被标记为内部或内部受保护,则可以模拟朋友访问。您必须修改要共享的程序集,并将以下设置添加到AssemblyInfo.cs:

// Expose the internal members to the types in the My.Tester assembly
[assembly: InternalsVisibleTo("My.Tester, PublicKey=" +
"012700000480000094000000060200000024000052534131000400000100010091ab9" +
"ba23e07d4fb7404041ec4d81193cfa9d661e0e24bd2c03182e0e7fc75b265a092a3f8" +
"52c672895e55b95611684ea090e787497b0d11b902b1eccd9bc9ea3c9a56740ecda8e" +
"961c93c3960136eefcdf106955a4eb8fff2a97f66049cd0228854b24709c0c945b499" +
"413d29a2801a39d4c4c30bab653ebc8bf604f5840c88")]

根据您的需要,公钥是可选的。