2

在我的应用程序中,用户使用像书一样的枢轴,他翻转枢轴项目直到他击中最后一个,然后我删除现有的枢轴项目并加载新的。这一切都是通过简单地将Pivot.ItemsSource属性指向一个新的项目集合来完成的。

我注意到超时内存消耗会增加并且永远不会降低。Pivot 的 VisualTree 似乎没有被垃圾收集。

我创建了一个示例应用程序来演示该问题(这是一个 WP8 应用程序):

重现步骤:

  • 启动应用程序
  • 导航到第 1 页
  • 单击加载更多按钮几次。

    (一次它加载 100 个项目,这当然不是我在实际应用程序中所做的,但它清楚地展示了每次内存消耗如何越来越高,即使导航回来也不会被清除)

我将不胜感激任何降低内存的提示或建议,因为如果以这种方式使用足够长的时间,应用程序不可避免地会崩溃。

MainPage.xaml:

...
<HyperlinkButton NavigateUri="/Page1.xaml">
    Page1
</HyperlinkButton>
...

MainPage.xaml.cs:

    //...
    public MainPage()
    {
        InitializeComponent();
        var timer = new DispatcherTimer();
        timer.Interval = TimeSpan.FromSeconds(2);
        timer.Tick += delegate
                          {
                              Debug.WriteLine("{0:f} MB, {1:f} MB",
                                              DeviceStatus.ApplicationCurrentMemoryUsage/(1024.0*1024),
                                              DeviceStatus.ApplicationPeakMemoryUsage/(1024.0*1024));
                          };
        timer.Start();
    }
    //...

Page1.xaml:

<Grid x:Name="LayoutRoot" Background="Transparent">
        <Grid.RowDefinitions>
            <RowDefinition Height="Auto" />
            <RowDefinition Height="*" />
        </Grid.RowDefinitions>

        <!--TitlePanel contains the name of the application and page title-->
        <StackPanel Grid.Row="0" Margin="12,17,0,28">
            <TextBlock Text="MY APPLICATION" Style="{StaticResource PhoneTextNormalStyle}" />
            <TextBlock Text="page name" Margin="9,-7,0,0" Style="{StaticResource PhoneTextTitle1Style}" />
        </StackPanel>

        <!--ContentPanel - place additional content here-->
        <Grid x:Name="ContentPanel" Grid.Row="1" Margin="12,0,12,0">

            <StackPanel>
                  <phone:Pivot x:Name="pivot1" ItemsSource="{Binding Items}">
                    <!--Pivot item one-->
                    <phone:Pivot.ItemTemplate>
                        <DataTemplate>
                            <TextBlock Text="{Binding}" />
                        </DataTemplate>
                    </phone:Pivot.ItemTemplate>
                </phone:Pivot>
                <Button Click="Load_More">Load More</Button>
            </StackPanel>
        </Grid>
    </Grid>

Page1.xaml.cs:

public partial class Page1 : PhoneApplicationPage, INotifyPropertyChanged
    {
        private List<int> _items;
        public List<int> Items
        {
            get
            {
                if (_items == null)
                {
                    _items = new List<int>();
                }
                return _items;
            }

            set
            {
                _items = value;
                RaisePropertyChanged("Items");
            }
        }
        public Page1()
        {
            InitializeComponent();
            DataContext = this;
        }

        private void Load_More(object sender, RoutedEventArgs e)
        {
            var lst = new List<int>();
            //100 new items just for demonstration, in reality i won't have more than 6 new items
            for (int i = 0; i < 100; i++)
            {
                lst.Add(i);
            }
            pivot1.SelectedIndex = 0;
            Items=lst;
        }

        public event PropertyChangedEventHandler PropertyChanged;

        void RaisePropertyChanged(string property)
        {
            var handler = PropertyChanged;
            if (handler != null)
            {
                handler(this, new PropertyChangedEventArgs(property));
            }
        }

      ~Page1()
      {
          Debug.WriteLine("Page1 GC");
      }
    }
4

3 回答 3

2

看起来 Pivot 控件泄漏,如使用 Panorama 控件所证明的那样。用这个替换你的堆栈面板(我没有更改控件的名称)不会泄漏。(内存在 50 兆左右被回收)

        <StackPanel>
            <Button Click="Load_More">Load More</Button>
            <phone:Panorama x:Name="pivot1" ItemsSource="{Binding Items}">
                <phone:PanoramaItem></phone:PanoramaItem>
            </phone:Panorama>
        </StackPanel>
于 2013-03-01T05:50:06.110 回答
1

一种解决方法是继续重复使用相同的项目集合,而不是创建新集合。当您需要更改它时,请清除该集合,然后重新填充它。该集合可以驻留在 App 全局空间中,例如:

public partial class App : Application
    {
        public static List<int> GlobalList;
...
    }

在 MainPage 构造函数中对其进行初始化,然后通过App.GlobalList.

于 2013-03-28T17:53:34.427 回答
1

解决方案 1

也许你可以试试这个解决方案:Windows Phone 8: LongListSelector Memory Leak

在关闭页面之前尝试清理这些依赖项属性。

解决方案 2

尝试使用OservableCollection<T>而不是List<T>for binding Itemsource。由于 WPF 设计,将 a绑定List<T>到 a可能会导致内存泄漏。Itemsource

在 Windows Presentation Foundation 中使用数据绑定时可能会发生内存泄漏

于 2014-03-18T09:43:39.933 回答