WPF概念解析一: FrameworkElement的Loaded事件和Initialized事件

WPF FrameworkElement的Loaded事件和Initialized事件

在WPF中FrameworkElement类定义了两个事件:Loaded和Initialized事件。当控件被加载到页面上的 时候,这两个事件会相继发生, 那么这两个事件的差别是什么哪?在进行自定义控件开发的时候,应该如何使用这两个事件那?
小结:

  • Initialized事件:该事件表明Frameworkelement已经被创建, 而且它的所有属性已经被设置。通常情况下子Element的事件会先于父Element触发,当Element接到该事件的时候,只 是说明其sub-tree已经被初始化,而父Element还没有被初始化。当Element的XMAL sub-tree被加载的 时候,该事件会被触发,这个时候FrameworkElement的IsInitialized属性为True。
  • Loaded事件:该事件表明Frameworkelement 不但被创建,初始化,而且该Element已经被加载到VirtualTree上,接下来该Element将被Render。从根节 点开始,该事件会以广播的方式在子Element触发,这个时候,FrameworkElement的IsLoaded属性为 True。

Initialized事件

Initialized事件标示当前Element的所有属性都已经被设置。WPF中 FrameworkElement实现了ISupportInitialize接口,当EndInit方法被调用的时候,置 IsInitialized属性为True,然后触发Initialized 事件。

 
ISupportInitialize接口允许开发人员在控件的 BeginInit后推迟某些属性的设置,直至EndInit时机才统一进行 多个属性的设置。利用ISupportInitialize接口可以解决由于属性设置顺序以及属性之间相互约束产生的问题。

 
WPF的XAML Loader已经支持了ISupportInitialize接口,创建完成对象后(解析完对象的开始节点),会调用 BeginInit方法,当加载完成该对象的XAML tree之后(解析完对象的结束节点)回调用EndInit方法。
如下面的例子:

<Button Width="100">
     Hello World!
</Button>

Button被创建的时候,BeginInit被调用,然后Width属性被设置,Content 属性被设置,最后EndInit方法被调用。


另外一件事情是在Initialized之前Element的隐式Style不会被应用,如下面的例 子:在Button类型的Style中定义了Background为Blue,但是Button1对象Initialized之前它的Background为Null。

<Page xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" >
  <Page.Resources>
    <Style TargetType="Button">
      <Setter Property="Background" Value="Blue" />
    </Style>
  </Page.Resources>
  <Button Name="Button1">Clack</Button>
</Page>

如果使用代码来创建FrameworkElement对象,同时也没有调用该对象的BeginInit和EndInit方法,那么在该对象被添加到VisualTree上的时候,Initialized事件会被触发。

Initialized事件是自底向上触发,由父向子。通常情况下只有子对象被初始化之后,父对象才应该被初始化。

Loaded事件

FrameworkElement的Loaded事件被触发的时候,该对象已经被初始化,而且即 将被绘制。开发人员可以在该事件中进行一些应用程序的初始化工作。


Initialized事件也可以做一些初始化的事情,但是在该事件中Element的有一些属性是不准确的,例如ActualWidth和 ActualHeight,因为该对象仅仅是被初始化,还没有开始计算ActualWidth和ActualHeight,一些数据绑定 的属性也还没有开始计算。


Loaded事件通常会在第一次Render之前触发,Element已经被Layout,数据 绑定也已经被计算。需要注意的是如果你在Loaded中Invalidate了Layout,在Render之前该事件有可能会被重新触 发。


Loaded事件是自顶向下以广播方式触发,从根节点,由父向子触发。而Initialized 事件是自底向上触发,先子后父。

 

posted @ 2011-11-08 11:33  ted  阅读(10007)  评论(3编辑  收藏  举报