3

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;
    }
}
4

3 に答える 3

4

このコードのスローダウンを測定しました。XP は約 47 ミリ秒でリストビューを描画します。Win7 は約 96 ミリ秒必要で、約 2 倍遅くなります。それが「地獄」のモニカに値するかどうかはわかりません。原因はわかりませんが、エアロが関係していることは間違いありません。オフにすることは、望ましいオプションではありません。

幸いなことに、コードに間違いがありました。私の測定では 48 ミリ秒かかりました。XP で持っていた正確なパフォーマンスを取り戻します。同じ ListViewItem を再利用する場合は、新しいものを作成する必要があります。次のコード行を RetrieveVirtualItem イベント ハンドラーに追加します。

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

そして、フィールドを取り除きます。

于 2010-10-03T19:17:52.393 に答える
1

SetStyle 行を次のように変更してみてください。

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

ユーザーペイント -

「true の場合、オペレーティング システムが描画するのではなく、コントロールが自分自身を描画します」( MSDN )

.

于 2010-10-05T13:47:45.423 に答える
0

WS_EX_COMPOSITEDを使用してみてください。これにより、仮想リストのパフォーマンスが大幅に向上します。

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

編集済み:少なくとも、描画アーティファクトが減少します。

于 2010-10-05T14:51:33.637 に答える