与XP相比,WinForms ListView在Windows 7中绘制了HELL SLOW

时间:2010-08-01 18:01:26

标签: winforms listview drawing performance virtualmode

我在VirtualMode和详细信息视图中使用ListView和小图标。

这个ListView里面有100,000件物品。

问题是,与XP相比,在Windows 7中绘制此列表视图要慢得多。

您可以在滚动ListView或多选项目时填充慢速绘图。

此外,我注意到添加每列后绘图会变慢。

RetrieveVirtualItem事件处理程序现在什么都不做,只返回文字值,所以这不是瓶颈。

有什么想法吗?

更新:重现的源代码:

FlickerFreeListView.cs:

using System;
using System.Collections.Generic;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;
using System.ComponentModel;
using System.Drawing;
using System.Reflection;
using System.Diagnostics;

namespace ListViewTest
{
    public class FlickerFreeListView : ListView
    {
        public FlickerFreeListView()
        {
            base.SetStyle(ControlStyles.OptimizedDoubleBuffer | ControlStyles.AllPaintingInWmPaint, true);
        }
    }
}

Form1.cs中:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Diagnostics;

namespace ListViewTest
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();

            _item = new ListViewItem(new string[6]);
        }

        private ListViewItem _item;

        private void listView1_RetrieveVirtualItem(object sender, RetrieveVirtualItemEventArgs e)
        {
            string itemIndexStr = e.ItemIndex.ToString();
            _item.Text = itemIndexStr;
            _item.SubItems[1].Text = "blablablablablablablablablablablablablablablablablablablablablablablablablablablablablablablablabla";
            _item.SubItems[2].Text = itemIndexStr;
            _item.SubItems[3].Text = itemIndexStr;
            _item.SubItems[4].Text = itemIndexStr;
            _item.SubItems[5].Text = itemIndexStr;
            e.Item = _item;
        }
    }
}

Form1.Designer.cs:

namespace ListViewTest
{
    partial class Form1
    {
        /// <summary>
        /// Required designer variable.
        /// </summary>
        private System.ComponentModel.IContainer components = null;

        /// <summary>
        /// Clean up any resources being used.
        /// </summary>
        /// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param>
        protected override void Dispose(bool disposing)
        {
            if (disposing && (components != null))
            {
                components.Dispose();
            }
            base.Dispose(disposing);
        }

        #region Windows Form Designer generated code

        /// <summary>
        /// Required method for Designer support - do not modify
        /// the contents of this method with the code editor.
        /// </summary>
        private void InitializeComponent()
        {
            this.listView1 = new ListViewTest.FlickerFreeListView();
            this.columnHeader1 = new System.Windows.Forms.ColumnHeader();
            this.columnHeader2 = new System.Windows.Forms.ColumnHeader();
            this.columnHeader3 = new System.Windows.Forms.ColumnHeader();
            this.columnHeader4 = new System.Windows.Forms.ColumnHeader();
            this.columnHeader5 = new System.Windows.Forms.ColumnHeader();
            this.columnHeader6 = new System.Windows.Forms.ColumnHeader();
            this.SuspendLayout();
            // 
            // listView1
            // 
            this.listView1.AutoArrange = false;
            this.listView1.Columns.AddRange(new System.Windows.Forms.ColumnHeader[] {
        this.columnHeader1,
        this.columnHeader2,
        this.columnHeader3,
        this.columnHeader4,
        this.columnHeader5,
        this.columnHeader6});
            this.listView1.Dock = System.Windows.Forms.DockStyle.Fill;
            this.listView1.FullRowSelect = true;
            this.listView1.HideSelection = false;
            this.listView1.Location = new System.Drawing.Point(0, 0);
            this.listView1.Name = "listView1";
            this.listView1.Size = new System.Drawing.Size(1032, 388);
            this.listView1.TabIndex = 0;
            this.listView1.UseCompatibleStateImageBehavior = false;
            this.listView1.View = System.Windows.Forms.View.Details;
            this.listView1.VirtualListSize = 100000;
            this.listView1.VirtualMode = true;
            this.listView1.RetrieveVirtualItem += new System.Windows.Forms.RetrieveVirtualItemEventHandler(this.listView1_RetrieveVirtualItem);
            // 
            // columnHeader1
            // 
            this.columnHeader1.Width = 92;
            // 
            // columnHeader2
            // 
            this.columnHeader2.Width = 405;
            // 
            // Form1
            // 
            this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
            this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
            this.ClientSize = new System.Drawing.Size(1032, 388);
            this.Controls.Add(this.listView1);
            this.Name = "Form1";
            this.Text = "Form1";
            this.WindowState = System.Windows.Forms.FormWindowState.Maximized;
            this.ResumeLayout(false);

        }

        #endregion

        private FlickerFreeListView listView1;
        private System.Windows.Forms.ColumnHeader columnHeader1;
        private System.Windows.Forms.ColumnHeader columnHeader2;
        private System.Windows.Forms.ColumnHeader columnHeader3;
        private System.Windows.Forms.ColumnHeader columnHeader4;
        private System.Windows.Forms.ColumnHeader columnHeader5;
        private System.Windows.Forms.ColumnHeader columnHeader6;
    }
}

3 个答案:

答案 0 :(得分:4)

我确实测量了这段代码的减速,XP在大约47毫秒内绘制了列表视图,Win7需要大约96毫秒,大约是慢两倍。不确定这是否值得'地狱'绰号。我不知道是什么原因引起的,但不要怀疑Aero是否与它有关。关闭它并不是一个理想的选择。

幸运的是,你在代码中犯了一个错误,在我的测量中花了48毫秒。给你回到你在XP上的确切性能。你重复使用相同的ListViewItem,你应该创建一个新的。将此行代码添加到RetrieveVirtualItem事件处理程序:

    private void listView1_RetrieveVirtualItem(object sender, RetrieveVirtualItemEventArgs e)
    {
        var _item = new ListViewItem(new string[6]);
        // etc...
    }

摆脱这个领域。

答案 1 :(得分:1)

尝试将SetStyle行更改为:

base.SetStyle(ControlStyles.UserPaint | ControlStyles.OptimizedDoubleBuffer | ControlStyles.AllPaintingInWmPaint, true); 

UserPaint -

  

“如果属实,控件会自行绘制   而不是操作系统在做什么   所以“(MSDN

答案 2 :(得分:0)

尝试使用WS_EX_COMPOSITED,它可以提高虚拟列表的性能:

protected override CreateParams CreateParams
{
    get
    {
        CreateParams cp = base.CreateParams;
        cp.ExStyle |= 0x02000000;
        return cp;
    }
}

编辑:至少它减少了绘图工件。