Por que o Silverlight ContentControls não é coletado como lixo?

Estive investigando por que alguns dos meus controles não estão sendo coletados como lixo e notei que é fácil impedir que controles simples que herdam do ContentControl sejam destruídos. Aqui está um exemplo:

Aqui está o meu ContentControl personalizado:

 public class MyCustomControl : ContentControl
{

    public MyCustomControl()
    {
        Debug.WriteLine("Constructed");
    }

    ~MyCustomControl()
    {
        Debug.WriteLine("Destroyed");
    }
}

Agora, se eu colocá-lo em uma página como esta:

<navigation:Page x:Class="SimpleTestBed.Views.CustomControl" 
       xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
       xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" 
       xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
       xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
       mc:Ignorable="d"
       xmlns:navigation="clr-namespace:System.Windows.Controls;assembly=System.Windows.Controls.Navigation"
             xmlns:local="clr-namespace:SimpleTestBed"
       d:DesignWidth="640" d:DesignHeight="480"
       Title="CustomControl Page">
<Grid x:Name="LayoutRoot">

    <StackPanel>
        <local:MyCustomControl>
            <TextBox Text="{Binding SomeProperty,Mode=TwoWay}"></TextBox>
        </local:MyCustomControl>
    </StackPanel>

</Grid>

Com o seguinte código por trás:

 public partial class CustomControl : Page
{
    public CustomControl()
    {
        InitializeComponent();

        this.DataContext = new CustomControlViewModel();

        this.Unloaded += new RoutedEventHandler(OnUnloaded);
    }

    void OnUnloaded(object sender, RoutedEventArgs e)
    {
        this.DataContext = null;
    }

    // Executes when the user navigates to this page.
    protected override void OnNavigatedTo(NavigationEventArgs e)
    {
    }



}

Então o modelo de visualização é:

  public class CustomControlViewModel : INotifyPropertyChanged
{

    #region INotifyPropertyChanged
    public event PropertyChangedEventHandler PropertyChanged;
    protected void OnPropertyChanged(string propertyName)
    {
        RaisePropertyChanged(propertyName);
    }
    private void RaisePropertyChanged(string propertyName)
    {
        if (PropertyChanged != null)
            PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
    }
    #endregion


    private string _someProperty = "Initial Value";
    public string SomeProperty
    {
        get { return _someProperty; }
        set
        {
            if (_someProperty != value)
            {
                string oldValue = _someProperty;
                _someProperty = value;
                OnPropertyChanged("SomeProperty");
                OnSomePropertyChanged(oldValue, value);
            }
        }
    }



    protected virtual void OnSomePropertyChanged(string oldValue, string newValue)
    {

    }


}

Agora, quando eu navego para longe desta página e tento coletar lixo com GC.Collect (), desde que eu não tenha feito alterações no texto na caixa de texto, o ContentControl e Page são destruídos conforme o esperado pelo GC. Mas se eu editei algum texto e naveguei para fora da página e tentei GC.Collect (), o ContentControl não obtém o lixo coletado.

Alguém pode explicar esse comportamento?

Na verdade, você pode forçar o GC a coletar o controle 'movendo' o Modelo do controle ao descarregar:

  void MyCustomControl_Unloaded(object sender, RoutedEventArgs e)
    {
        Debug.WriteLine("MyCustomControl Unloaded");
        ControlTemplate oldTemplate = this.Template;
        this.Template = null;
        this.Template = oldTemplate;
    }

Presumo que isso destrua a árvore visual atual, perdendo referências do primeiro componente da árvore para seu pai (o controle personalizado). Certamente força o controle a chamar OnApplyTemplate quando o controle é recarregado.

Esse é o padrão correto para o desenvolvimento de controles do Silverlight sem vazamento? Nesse caso, me parece um pouco estranho que o modelo não seja descartado automaticamente quando o controle é descarregado.

Uma boa explicação desse comportamento seria muito apreciada, pois vai direto ao cerne do ciclo de vida dos controles do Silverlight.

questionAnswers(1)

yourAnswerToTheQuestion