C#类的嵌套

本文详细介绍了C#中的嵌套类概念,包括静态嵌套类和非静态嵌套类的特点,以及它们如何与外部类交互。探讨了访问权限、继承、懒加载等关键特性,并提供了示例代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

C# 类的嵌套
(2009-07-17 20:56:22)
 
嵌套类( Nested Class )是在类中定义的类。以下把拥有内嵌类的类称为外部类。嵌套类分为静态嵌套类和非静态嵌套类,其中非静态嵌套类也被称为内部类。嵌套类在 UML 中是 composite 的另外一种代码表示形式,表示耦合度更高,并且与外部类更加紧密。
一般类的访问修饰符可以定义为默认的 internal 或者 public ,而内嵌类就有比较多的选择,可以是为 protected internal public 以及默认的 private
内嵌类与外部类的访问限制
嵌套类可以访问外部类的方法、属性、字段而不管访问修饰符的限制。如:
  public class A
  {
      private static int _AInt;
      private int _instanceInt;
      private static void AMethod()
      {
          Console.WriteLine(_AInt);
      }
      public void SayIt()
      {
          NestedA.Method(this);
      }      
      private class NestedA
      {
         public static void Method(A a)
         {
            //
静态成员
            _AInt = 100;
            AMethod();
            //
实例成员
            a._instanceInt = 10;
            a.SayIt();
         }
      }
  }
   但是外部类只能够访问修饰符为 public internal 嵌套类的字段、方法、属性。示例如下:
  public class A
  {
     public static void AMethod()
     {
        //
成功
        NestedA.StaticMethod();
        //
编译报错
        NestedA._Int = 100;       
        NestedA ins=new NestedA();
        //
成功
        ins.Method();
        //
编译报错
        ins._instanceInt = 100;
     }
     private class NestedA
     {
        private static int _Int;
        private int _instanceInt;
        public static void StaticMethod() { }
        public void Method(){}
     }
  }
  
嵌套类访问外部类实例的方法、字段、属性时候。一般在采取构造函数输入外部类。如下:
  public class A
  {
     private int _a;
     private class NestedA
     {
        public NestedA(A a)
        {
           a._a = 9;
        }             
     }
  }
继承
继承类,也就是继承类外部类的类,只能使用父类中嵌套类的 public 或者 internal( 同一个程序集合 ) 方法。但是继承类可以再定义一个内嵌类并从继承父类中嵌套类。如:
  public class A
  {
     protected class Nested
     {
        protected virtual void BaseNested_Method(){}
     }
  }
  public class C : A
  {   
    protected class C_Nested:Nested
    {
       protected override void BaseNested_Method()
       {
           //
重写部分
       }
    }
  }
因为 C A 中继承,因此 C_Nested 可以继承 Nested 类,从而获取重写父嵌套类的机会。但是 Nested 必须是可继承类及可访问的(非 private  sealed static )。
嵌套类可以随意访问外部类的任何数据属性,而外部类访问嵌套类就只能遵守访问修饰符。从这个角度看,嵌套类是外部类的补充,通过嵌套类可以获取更好的封装性,增加外部类的可维护性和可读性。       
从程序结构看,嵌套类在逻辑上更加接近使用类。可以更有效地表示类与类之间的紧密程度。为类管理提供除命名空间外的另一种方法。
懒加载
嵌套类的静态构造函数不会随着外部类的触发而初始化。因此可以有效地避免创建时候初始化时间,当需要使用内嵌类的时候,嵌套类才开始初始化。
 public class Outside
 {
    static Outside()
    {
       Console.WriteLine("Outside Inilizlized");
    }
    public void SayIt()
    {
        Nested.Run();
    }
    private class Nested
    {
        static Nested()
        {
            Console.WriteLine("Nested initilized");
        }
        public static void Run()
        {
            Console.WriteLine("Nested Run");
        }
    }
}
执行结果
Outside o = new Outside(); //
打印 "Outside Inilizlized"
Console.ReadLine();
o.SayIt();//
首先打印 "Nested initilized"  再打印  "Nested Run"
Console.ReadLine();
一般应用这个特性会在一些 C# 单例模式中找到,而这种模式可以被称为 Fully lazy singleton 模式。下面是简单的演示代码( Singleton 模式可以在这里有更加详细的解释):
  public class Singleton
  {
      public static Singleton Instance
      {
         get { return Nested.instance; }
      }
      private class Nested
      {
          public readonly static Singleton instance=new Singleton();
      }
  }
  反射
  反射内嵌类需要使用 "+" 而不是我们常使用的 "."  ,如 A 类在名称为 InsideClass Assembly 中。
 namespace InsideClass
 {
    public class A
    {
        public class Nested
        {
            protected void BaseNested_Method()
            {
            }
        }
     }
 }
执行
//
成功
object o1 = System.Activator.CreateInstance("InsideClass", "InsideClass.A+Nested");
//
失败   抛出 System.TypeLoadException 异常
object o2 = System.Activator.CreateInstance("InsideClass", "InsideClass.A.Nested");
 

本文转自gauyanm 51CTO博客,原文链接:http://blog.51cto.com/gauyanm/305110,如需转载请自行联系原作者
### C# 中实现自嵌套数据C# 中,可以通过定义 **嵌套** 来创建一种特殊的结构形式——即所谓的“自嵌套数据”。这种设计模式允许外部包含其自身的实例作为成员变量。以下是关于如何构建此结构的详细说明。 #### 自嵌套数据的概念 当提到“自嵌套数据”,通常是指一个可以持有另一个相同型的对象作为自己的属性或字段。这似于树形结构的设计方式,在许多场景下非常有用,比如表示层次化的数据模型或者递归关系[^1]。 #### 创建自嵌套数据的具体方法 为了展示这一特性,以下是一个简单的例子: ```csharp public class TreeNode { // 属性:节点名称 public string Name { get; set; } // 嵌套:子节点列表 public List<TreeNode> Children { get; set; } = new List<TreeNode>(); // 构造函数初始化 public TreeNode(string name) { this.Name = name; } // 添加子节点的方法 public void AddChild(TreeNode childNode) { Children.Add(childNode); } } ``` 上述代码片段展示了 `TreeNode` 是如何通过维护一个名为 `Children` 的集合来存储其他同型 (`TreeNode`) 对象的引用。这样就实现了所谓“自嵌套”的效果[^3]。 #### 使用示例 下面是基于上面定义好的 `TreeNode` 结构的一个简单应用案例: ```csharp class Program { static void Main() { // 创建根节点 var root = new TreeNode("Root"); // 创建第一个子节点并附加到根上 var child1 = new TreeNode("Child 1"); root.AddChild(child1); // 继续扩展第二个级别的孩子节点 var grandchild1 = new TreeNode("Grand Child 1"); child1.AddChild(grandchild1); // 打印整个树状结构 (简化版打印逻辑) PrintTree(root, ""); } private static void PrintTree(TreeNode node, string indent) { Console.WriteLine(indent + "- " + node.Name); foreach (var child in node.Children) PrintTree(child, indent + " "); } } ``` 这段程序会输出如下结果: ``` - Root - Child 1 - Grand Child 1 ``` 这里可以看到,即使是在同一个中,“父”对象也可以轻松管理多个“子”对象,并且这些子对象还可以继续拥有更多的后代,形成复杂的层级体系[^2]。 #### 总结 通过这种方式,可以在 C# 编程语言里方便地建立具有自我参照性质的数据结构。这技术特别适合用来描述那些天然具备分层特征的信息单元,例如文件夹目录、组织架构图等等。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值