today we've found a critical issue regarding the ListView from
Windows.Forms. it was confirmed on several machines with Win2K and XP.
here's the problem: create a ListView with about 50000 rows. now use task
manager to see the GDI usage of the process. everything seems normal.
now catch the ListView's scroller and start to move it downwards. you have
to hold the constant speed so that the ListView is constantly repainted.
look at the GDI usage of the application. it starts to increase, starting at
21 and going through 1000-5000. if you get the bottom of the ListView,
immediately start to scroll it upwards so that the ListView is constantly
repainted. you'll see that the GDI objects count reaches the 10000 and then
jumps to over 4297000000! remember to keep the scrolling speed!
running from the IDE can even make the Windows GDI subsystem completely
broken and hang the whole system (this is what our application does when it
is misused!).
here's the simple code. compile it, press the button "button 1" and watch
the GDI usage.
in the attachment I've included the snapshot of the Task Manager showing the
GDI usage. I am really desperate to hear from anyone about that. could
anyone confirm the problem or point my mistake?
Regards,
Wiktor
-----------------------------------------------------
using System;
using System.Drawing;
using System.Collections;
using System.ComponentModel;
using System.Windows.Forms;
using System.Data;
namespace vicTestF
{
public class Form1 : System.Windows.Forms.Form
{
private System.Windows.Forms.ListView listView1;
private System.Windows.Forms.ColumnHeader columnHeader1;
private System.Windows.Forms.ColumnHeader columnHeader2;
private System.Windows.Forms.ColumnHeader columnHeader3;
private System.Windows.Forms.Button button1;
private System.ComponentModel.Container components = null;
public Form1()
{
InitializeComponent();
}
protected override void Dispose( bool disposing )
{
if( disposing )
{
if (components != null)
{
components.Dispose();
}
}
base.Dispose( disposing );
}
#region Windows Form Designer generated code
private void InitializeComponent()
{
this.listView1 = new System.Windows.Forms.ListView();
this.columnHeader1 = new System.Windows.Forms.ColumnHeader();
this.columnHeader2 = new System.Windows.Forms.ColumnHeader();
this.columnHeader3 = new System.Windows.Forms.ColumnHeader();
this.button1 = new System.Windows.Forms.Button();
this.SuspendLayout();
//
// listView1
//
this.listView1.Columns.AddRange(new System.Windows.Forms.ColumnHeader[] {
this.columnHeader1,
this.columnHeader2,
this.columnHeader3});
this.listView1.Dock = System.Windows.Forms.DockStyle.Fill;
this.listView1.GridLines = true;
this.listView1.Location = new System.Drawing.Point(0, 32);
this.listView1.Name = "listView1";
this.listView1.Size = new System.Drawing.Size(496, 289);
this.listView1.TabIndex = 0;
this.listView1.View = System.Windows.Forms.View.Details;
//
// columnHeader1
//
this.columnHeader1.Width = 143;
//
// columnHeader2
//
this.columnHeader2.Width = 115;
//
// columnHeader3
//
this.columnHeader3.Width = 229;
//
// button1
//
this.button1.Dock = System.Windows.Forms.DockStyle.Top;
this.button1.Name = "button1";
this.button1.Size = new System.Drawing.Size(496, 32);
this.button1.TabIndex = 1;
this.button1.Text = "button1";
this.button1.Click += new System.EventHandler(this.button1_Click);
//
// Form1
//
this.AutoScaleBaseSize = new System.Drawing.Size(5, 13);
this.ClientSize = new System.Drawing.Size(496, 321);
this.Controls.AddRange(new System.Windows.Forms.Control[] {
this.listView1,
this.button1});
this.Name = "Form1";
this.Text = "Form1";
this.ResumeLayout(false);
}
#endregion
[STAThread]
static void Main()
{
Application.Run(new Form1());
}
private void button1_Click(object sender, System.EventArgs e)
{
listView1.BeginUpdate();
ListViewItem li;
for ( int i=0; i<50000; i++ )
{
// ListViewItem li = listView1.Items.Add( "item" + i.ToString() );
li = listView1.Items.Add( "item" + i.ToString() );
li.SubItems.Add( "qwer" );
li.SubItems.Add( "qwer" );
}
MessageBox.Show( "before endupdate" );
listView1.EndUpdate();
}
}
} 15 3805
Well I dont know exactly whats going on here although you should probably
download a profiler and see exactly what is being used by your program.
Either way, 50,000 rows if far more than you should ever consider putting
into a ListView! No user is ever going to be able to take in all of these!
HTH
Kieran
"Wiktor Zychla" <ie****@microsoft.com.no.spam> wrote in message
news:%2*****************@TK2MSFTNGP09.phx.gbl... today we've found a critical issue regarding the ListView from Windows.Forms. it was confirmed on several machines with Win2K and XP.
here's the problem: create a ListView with about 50000 rows. now use task manager to see the GDI usage of the process. everything seems normal.
now catch the ListView's scroller and start to move it downwards. you have to hold the constant speed so that the ListView is constantly repainted. look at the GDI usage of the application. it starts to increase, starting
at 21 and going through 1000-5000. if you get the bottom of the ListView, immediately start to scroll it upwards so that the ListView is constantly repainted. you'll see that the GDI objects count reaches the 10000 and
then jumps to over 4297000000! remember to keep the scrolling speed!
running from the IDE can even make the Windows GDI subsystem completely broken and hang the whole system (this is what our application does when
it is misused!).
here's the simple code. compile it, press the button "button 1" and watch the GDI usage.
in the attachment I've included the snapshot of the Task Manager showing
the GDI usage. I am really desperate to hear from anyone about that. could anyone confirm the problem or point my mistake?
Regards, Wiktor ----------------------------------------------------- using System; using System.Drawing; using System.Collections; using System.ComponentModel; using System.Windows.Forms; using System.Data;
namespace vicTestF { public class Form1 : System.Windows.Forms.Form { private System.Windows.Forms.ListView listView1; private System.Windows.Forms.ColumnHeader columnHeader1; private System.Windows.Forms.ColumnHeader columnHeader2; private System.Windows.Forms.ColumnHeader columnHeader3; private System.Windows.Forms.Button button1; private System.ComponentModel.Container components = null;
public Form1() { InitializeComponent(); }
protected override void Dispose( bool disposing ) { if( disposing ) { if (components != null) { components.Dispose(); } } base.Dispose( disposing ); }
#region Windows Form Designer generated code private void InitializeComponent() { this.listView1 = new System.Windows.Forms.ListView(); this.columnHeader1 = new System.Windows.Forms.ColumnHeader(); this.columnHeader2 = new System.Windows.Forms.ColumnHeader(); this.columnHeader3 = new System.Windows.Forms.ColumnHeader(); this.button1 = new System.Windows.Forms.Button(); this.SuspendLayout(); // // listView1 // this.listView1.Columns.AddRange(new System.Windows.Forms.ColumnHeader[]
{ this.columnHeader1, this.columnHeader2, this.columnHeader3}); this.listView1.Dock = System.Windows.Forms.DockStyle.Fill; this.listView1.GridLines = true; this.listView1.Location = new System.Drawing.Point(0, 32); this.listView1.Name = "listView1"; this.listView1.Size = new System.Drawing.Size(496, 289); this.listView1.TabIndex = 0; this.listView1.View = System.Windows.Forms.View.Details; // // columnHeader1 // this.columnHeader1.Width = 143; // // columnHeader2 // this.columnHeader2.Width = 115; // // columnHeader3 // this.columnHeader3.Width = 229; // // button1 // this.button1.Dock = System.Windows.Forms.DockStyle.Top; this.button1.Name = "button1"; this.button1.Size = new System.Drawing.Size(496, 32); this.button1.TabIndex = 1; this.button1.Text = "button1"; this.button1.Click += new System.EventHandler(this.button1_Click); // // Form1 // this.AutoScaleBaseSize = new System.Drawing.Size(5, 13); this.ClientSize = new System.Drawing.Size(496, 321); this.Controls.AddRange(new System.Windows.Forms.Control[] { this.listView1, this.button1}); this.Name = "Form1"; this.Text = "Form1"; this.ResumeLayout(false);
} #endregion
[STAThread] static void Main() { Application.Run(new Form1()); }
private void button1_Click(object sender, System.EventArgs e) { listView1.BeginUpdate();
ListViewItem li; for ( int i=0; i<50000; i++ ) { // ListViewItem li = listView1.Items.Add( "item" + i.ToString() ); li = listView1.Items.Add( "item" + i.ToString() ); li.SubItems.Add( "qwer" ); li.SubItems.Add( "qwer" ); }
MessageBox.Show( "before endupdate" ); listView1.EndUpdate(); } } }
> Either way, 50,000 rows if far more than you should ever consider putting into a ListView! No user is ever going to be able to take in all of these!
it also happens when you have 2000 items or even a 1000. this IS a
reasonable count. 50000 was only an example!
just try it and help me explain it if you can. the problem DOES NOT OCCUR if
you do the same in old VB for example. so it is a Windows.Forms issue.
You probably run the debug version, can you check this in release mode.
Willy.
"Wiktor Zychla" <ie****@microsoft.com.no.spam> wrote in message news:%2*****************@TK2MSFTNGP09.phx.gbl... today we've found a critical issue regarding the ListView from Windows.Forms. it was confirmed on several machines with Win2K and XP.
here's the problem: create a ListView with about 50000 rows. now use task manager to see the GDI usage of the process. everything seems normal.
now catch the ListView's scroller and start to move it downwards. you have to hold the constant speed so that the ListView is constantly repainted. look at the GDI usage of the application. it starts to increase, starting at 21 and going through 1000-5000. if you get the bottom of the ListView, immediately start to scroll it upwards so that the ListView is constantly repainted. you'll see that the GDI objects count reaches the 10000 and then jumps to over 4297000000! remember to keep the scrolling speed!
running from the IDE can even make the Windows GDI subsystem completely broken and hang the whole system (this is what our application does when it is misused!).
here's the simple code. compile it, press the button "button 1" and watch the GDI usage.
in the attachment I've included the snapshot of the Task Manager showing the GDI usage. I am really desperate to hear from anyone about that. could anyone confirm the problem or point my mistake?
Regards, Wiktor ----------------------------------------------------- using System; using System.Drawing; using System.Collections; using System.ComponentModel; using System.Windows.Forms; using System.Data;
namespace vicTestF { public class Form1 : System.Windows.Forms.Form { private System.Windows.Forms.ListView listView1; private System.Windows.Forms.ColumnHeader columnHeader1; private System.Windows.Forms.ColumnHeader columnHeader2; private System.Windows.Forms.ColumnHeader columnHeader3; private System.Windows.Forms.Button button1; private System.ComponentModel.Container components = null;
public Form1() { InitializeComponent(); }
protected override void Dispose( bool disposing ) { if( disposing ) { if (components != null) { components.Dispose(); } } base.Dispose( disposing ); }
#region Windows Form Designer generated code private void InitializeComponent() { this.listView1 = new System.Windows.Forms.ListView(); this.columnHeader1 = new System.Windows.Forms.ColumnHeader(); this.columnHeader2 = new System.Windows.Forms.ColumnHeader(); this.columnHeader3 = new System.Windows.Forms.ColumnHeader(); this.button1 = new System.Windows.Forms.Button(); this.SuspendLayout(); // // listView1 // this.listView1.Columns.AddRange(new System.Windows.Forms.ColumnHeader[] { this.columnHeader1, this.columnHeader2, this.columnHeader3}); this.listView1.Dock = System.Windows.Forms.DockStyle.Fill; this.listView1.GridLines = true; this.listView1.Location = new System.Drawing.Point(0, 32); this.listView1.Name = "listView1"; this.listView1.Size = new System.Drawing.Size(496, 289); this.listView1.TabIndex = 0; this.listView1.View = System.Windows.Forms.View.Details; // // columnHeader1 // this.columnHeader1.Width = 143; // // columnHeader2 // this.columnHeader2.Width = 115; // // columnHeader3 // this.columnHeader3.Width = 229; // // button1 // this.button1.Dock = System.Windows.Forms.DockStyle.Top; this.button1.Name = "button1"; this.button1.Size = new System.Drawing.Size(496, 32); this.button1.TabIndex = 1; this.button1.Text = "button1"; this.button1.Click += new System.EventHandler(this.button1_Click); // // Form1 // this.AutoScaleBaseSize = new System.Drawing.Size(5, 13); this.ClientSize = new System.Drawing.Size(496, 321); this.Controls.AddRange(new System.Windows.Forms.Control[] { this.listView1, this.button1}); this.Name = "Form1"; this.Text = "Form1"; this.ResumeLayout(false);
} #endregion
[STAThread] static void Main() { Application.Run(new Form1()); }
private void button1_Click(object sender, System.EventArgs e) { listView1.BeginUpdate();
ListViewItem li; for ( int i=0; i<50000; i++ ) { // ListViewItem li = listView1.Items.Add( "item" + i.ToString() ); li = listView1.Items.Add( "item" + i.ToString() ); li.SubItems.Add( "qwer" ); li.SubItems.Add( "qwer" ); }
MessageBox.Show( "before endupdate" ); listView1.EndUpdate(); } } }
Sorry Wiktor but I can't reproduce this on my .NET 1.1 machine, what version
are you using? Task manager reports just 4 GDI objects no matter how
fast/slow I scroll. Memory usage (At least as reported by TM) goes up but
this is to be expected and is absolutely normal. Jsut tried it on a 128Mb
machine as well and performance was still adequate.
Kieran
"Wiktor Zychla" <ie****@microsoft.com.no.spam> wrote in message
news:u%****************@TK2MSFTNGP11.phx.gbl... Either way, 50,000 rows if far more than you should ever consider
putting into a ListView! No user is ever going to be able to take in all of
these! it also happens when you have 2000 items or even a 1000. this IS a reasonable count. 50000 was only an example!
just try it and help me explain it if you can. the problem DOES NOT OCCUR
if you do the same in old VB for example. so it is a Windows.Forms issue.
> You probably run the debug version, can you check this in release mode.
Willy, I've checked some compiler swiches (/debug+, /debug-) and it does not
help.
Check without /debug (debug+ or /debug- both generate a debug version).
I checked with your sample and the phenomenon you describe only occurs when a debug build is run. Note that the value 429700000 is
completely bogus, the max. number of GDI objects per process is 16384.
Willy.
"Wiktor Zychla" <ie****@microsoft.com.no.spam> wrote in message news:%2****************@tk2msftngp13.phx.gbl... You probably run the debug version, can you check this in release mode.
Willy, I've checked some compiler swiches (/debug+, /debug-) and it does not help.
> I checked with your sample and the phenomenon you describe only occurs
when a debug build is run. Note that the value 429700000 is completely bogus, the max. number of GDI objects per process is 16384.
alas, it happens to me on both debug and release build of the application.
anyway, isn't it a bug then? it does not happen with the ListView itself -
I've checked the same code with VB6.0 and there's no problem there.
I think that the number 4297000000 could be a 16-bit negative value (given
by the system) converted to 32-bit unsigned value (seen by TM). it would
explain why I get the number just after 10000.
Regards, Wiktor
Wiktor,
Weird, I'm running your sample on XP using version 1.1 of the .NET runtime, running the release version I noticed the GDI handle
count going up to ~2000 starting from ~70 and dropping regularly to ~70 after a CG run. The debug build shows a different pattern,
just because non managed resources are not reclaimed so aggressively, this is considered normal behavior not a bug.
However, the TM GDI object counter seems to bug out when exceeding 10000.
What version of the framework of the runtime are you running on?
Willy.
"Wiktor Zychla" <ie****@microsoft.com.no.spam> wrote in message news:e7**************@TK2MSFTNGP11.phx.gbl... I checked with your sample and the phenomenon you describe only occurs when a debug build is run. Note that the value 429700000 is completely bogus, the max. number of GDI objects per process is 16384.
alas, it happens to me on both debug and release build of the application. anyway, isn't it a bug then? it does not happen with the ListView itself - I've checked the same code with VB6.0 and there's no problem there.
I think that the number 4297000000 could be a 16-bit negative value (given by the system) converted to 32-bit unsigned value (seen by TM). it would explain why I get the number just after 10000.
Regards, Wiktor
> What version of the framework of the runtime are you running on?
tried on 1.0 and 1.1. the same result. the GDI handles count is never
decreased. it starts at 21 and then when I scroll the listview, it increases
up to 10000 and then jumps to this big number.
Ive just tried on a 1.0 machine and I get the same behaviour as Willy. Very
strange. The debug build doesn't kill my PC but does less aggressively GC.
This is inexplicable, anyone got any ideas?
Kieran
"Wiktor Zychla" <ie****@microsoft.com.no.spam> wrote in message
news:el*************@TK2MSFTNGP11.phx.gbl... What version of the framework of the runtime are you running on? tried on 1.0 and 1.1. the same result. the GDI handles count is never decreased. it starts at 21 and then when I scroll the listview, it
increases up to 10000 and then jumps to this big number.
nonsense
in one of my application, whole accounting book can be loaded into listview
(if user selects dates 1.1 to 31.12).When I was testing I used 100000 rows
and tested on celeron 466,same as customer has. There were no slowdowns
while scrolling, only a lag of about 3 seconds to load everything (using
addrange).
Hmm .... Who's talking about slowdows?
Willy.
"martin" <fa********@yahoo.com> wrote in message news:bk**********@ls219.htnet.hr... nonsense
in one of my application, whole accounting book can be loaded into listview (if user selects dates 1.1 to 31.12).When I was testing I used 100000 rows and tested on celeron 466,same as customer has. There were no slowdowns while scrolling, only a lag of about 3 seconds to load everything (using addrange).
> nonsense in one of my application, whole accounting book can be loaded into
listview (if user selects dates 1.1 to 31.12).When I was testing I used 100000 rows and tested on celeron 466,same as customer has. There were no slowdowns while scrolling, only a lag of about 3 seconds to load everything (using addrange).
I do not have any slowdowns, either. I even did not mention any slowdowns.
So, what is your point? Did you check the GDI objects leakage in your
application? This thread has been closed and replies have been disabled. Please start a new discussion. Similar topics
by: Greg Busby |
last post by:
I have a client who wants to use Windows Integrated Security for authentication and authorization to use this application. They also want this application to run as soon as Windows comes up. So, I...
|
by: Job Lot |
last post by:
Can some please provide some tips, links whatever possible on creating Graphs in WINDOWS FORMS application (not web forms), using DataSet, DataTable etc etc… anything.
I am totally lost and cud...
|
by: Steph |
last post by:
I am working on a Windows Forms App. I have attached a standard
UnhandledExceptionEventHandler to the current domain to catch
unexpected errors. In my application, when I trigger an unhandled...
|
by: Visually Seen # |
last post by:
Hey everyone,
Previously, about a month ago I posted a question asking how to remove
the command prompt that always appeared when I ran my Windows Forms
Application EXE.
Well, that was visually...
|
by: Sanjeeva Reddy |
last post by:
1)how to get the preview of movie file to add in imagelist(i mean the first
frame of movie file. )(in .net frame work,windows Forms application) in a
listview control.
here i am giving code for...
|
by: Sandy |
last post by:
My background is web based. I am attempting to write a Windows
application and I am finding some simple things difficult.
Currently I am trying to find out how to store information session
wide....
|
by: Ludovic DE FREITAS |
last post by:
Hello,
I am making a prototype system with a Pocket PC that must connect to a
Server Application.
Unfortunately, the server cannot be a Web Server, but a simple Windows Forms
Application.
...
|
by: Jordan S. |
last post by:
SQL Server will be used as the back-end database to a non trivial client
application.
In question is the choice of client application:
I need to be able to speak intelligently about when one...
|
by: samueltilden |
last post by:
I have inherited a complex application with dozens of controls.
The new functionality I am to surgically insert is to prevent the user
from proceeding further if any of the data is incorrect in...
|
by: ryjfgjl |
last post by:
ExcelToDatabase: batch import excel into database automatically...
|
by: isladogs |
last post by:
The next Access Europe meeting will be on Wednesday 6 Mar 2024 starting at 18:00 UK time (6PM UTC) and finishing at about 19:15 (7.15PM).
In this month's session, we are pleased to welcome back...
|
by: jfyes |
last post by:
As a hardware engineer, after seeing that CEIWEI recently released a new tool for Modbus RTU Over TCP/UDP filtering and monitoring, I actively went to its official website to take a look. It turned...
|
by: ArrayDB |
last post by:
The error message I've encountered is; ERROR:root:Error generating model response: exception: access violation writing 0x0000000000005140, which seems to be indicative of an access violation...
|
by: PapaRatzi |
last post by:
Hello,
I am teaching myself MS Access forms design and Visual Basic. I've created a table to capture a list of Top 30 singles and forms to capture new entries. The final step is a form (unbound)...
|
by: CloudSolutions |
last post by:
Introduction:
For many beginners and individual users, requiring a credit card and email registration may pose a barrier when starting to use cloud servers. However, some cloud server providers now...
|
by: Defcon1945 |
last post by:
I'm trying to learn Python using Pycharm but import shutil doesn't work
|
by: Shællîpôpï 09 |
last post by:
If u are using a keypad phone, how do u turn on JavaScript, to access features like WhatsApp, Facebook, Instagram....
|
by: af34tf |
last post by:
Hi Guys, I have a domain whose name is BytesLimited.com, and I want to sell it. Does anyone know about platforms that allow me to list my domain in auction for free. Thank you
| |