隐藏

System.ComponentModel.Component入门

发布:2021/6/18 10:47:29作者:管理员 来源:本站 浏览次数:1039

我们经常看见许多.NET的Class是从System.ComponentModel.Component继承过来的,他实现了System.ComponentModel.IComponent接口,而且MS也在Visual Studio .NET中,推荐你经常使用System.ComponentModel.Component作为基础类。但是很遗憾,很多人不知道为什么这样做。

MS的解释包括:
1、控制外部资源
      IComponent 接口继承自 System.IDisposable 接口,这样可以控制对象的释放。

2、设计时支持
      只要是支持IComponent接口,都可以看见一个设计器,并且拖入到这个组件中的子组件都会自动产生以下代码:
this.errorProvider1 = new System.Windows.Forms.ErrorProvider(this.components);

以便加入站点。

3、承载组件
     所有的子组件都是通过 IContainer 管理的,所以你的子组件都是被管理的。

事实上,很多人看见这些特性还是云里雾里,对于ISite、IContainer和IComponment的关系还是难以理解,那么我们看看这个图形。


通过这个图,我想理解应该好一些吧。

讨论:
1、IComponent使用了注入依赖的思想;
      我发现IComponent的Site属性是Get 和Set的,也就是说,IComponent实现需要有人初始化Site才能正常工作,不信的话你直接实例化一个Component对象,访问Site你会发现是Null的。
      Visual Studio .NET的设计器在实例化一个IComponent对象时,会自动产生如下的代码:

this.errorProvider1 = new System.Windows.Forms.ErrorProvider(this.components);
这段代码实际上就是帮助你的errorProvider1对象初始化Site属性。
       另外,ISite继承了System.IServiceProvider,所以也对应的注入了服务的提供者。
       关于注入依赖,请参考查找AOP和Ioc

2、.NET 2.0 增加了INestedContainer
     容器一般来说,也是被另外一个IComponent持有,你也许注意到我们的设计器总是会生成这样的代码:
private System.ComponentModel.IContainer components = null;
     但是这个容器对象,我们发现他并不包含 Owner 的引用,也就是说,你的子对象可以互相通讯,但是你的子对象不能访问到 他的父。
     基于以上问题,.NET 2.0中增加了INestedContainer接口(继承自IContainer),他新增了Owner属性,可以访问到容器所在的所有者,即父。

3、IComponent适合的范围
      我注意到,自动生成的代码中,这个组件公用一个容器,也就是说,只有整个组件释放,容器才会释放。所以当你使用一个组件时,请及时的释放这个对象。
      如果你打算设计无状态的服务类,请不要使用设计器,而是需要某个子组件时,才创建,并使用容器统一管理,诸如以下的代码:
        public void Save() {
            
//使用容器统一管理资源。
            using (Container container = new Container()) {
                Component1 c1 
= new Component1(container);
                c1.Do();

                Component2 c2 
= new Component2(container);
                c2.Do();
            }

        }