WPF:警惕TextBox会占用过多内存

问题源自这篇文章:WPF的TextBox产生内存泄露的情况。

  整个问题是这样的,文章作者演示使用类似下方的代码来不停地像WPF的TextBox控件赋值:

for (int i = 0; i < 10000; i++)

{

//tbx是界面上的TextBox变量

tbx.Text += string.Format("{0}\n", i);

}

  然后会出现程序占用过多内存的问题。

  很快在那篇文章的评论中有人指出这个和WPF没有关系,因为频繁得拼接字符串会产生过多重复字符串对象,即使不显示在TextBox控件上,也会会占用过多内存的。

  但是原文作者又在回复中讲到他做了相关测试,但是却不会出现占用非常多的内存的情况。

  最后问题不了了之。

  我做了下测试,一种是不断拼接字符串并显示在TextBox中:

//我们就拿个小数字1万来做示例

for (int i = 0; i < 10000; i++)

{

//tbx是界面上的TextBox变量

tbx.Text += string.Format("{0}\n", i);

}

  程序运行后任务管理器显示55.5 MB的内存。(环境:.NET 4.5/Debug编译/64位)

  接着测试另一种情况:先拼接字符串,最后才显示在TextBox中:

var str = String.Empty;

//我们就拿个小数字1万来做示例

for (int i = 0; i < 10000; i++)

{

//tbx是界面上的TextBox变量

str += String.Format("{0}\n", i);

}

tbx.Text = str;

  运行后显示24.1 MB。

  如果把最后一句删掉,也就是根本不在TextBox中显示。运行后是23.4 MB。

  很明显,整个问题确实是和WPF有关系的,但也不是因为原文作者认为的字符串拼接所造成的。真正的问题是TextBox(更准确地说是其父类:TextBoxBase)的UndoLimit属性。也就是说TextBox会因为频繁修改值而堆积过多的撤销项目项目。

  这部分项目会占用过多的内存空间。

  在.NET 3.5和4.0:TextBoxBase.UndoLimit的值默认是-1。代表如果内存够用的话,撤销列表会无限大。(这个有点恐怖)

  在.NET 4.5中:TextBoxBase.UndoLimit的值默认是100。

  我们可以吧TextBoxBase.UndoLimit设置成0(或者把IsUndoEnabled设置成False),也就是命令TextBox不支持撤销功能。再次运行第一次的代码:

//禁止撤销

tbx.UndoLimit = 0;

for (int i = 0; i < 10000; i++)

{

//tbx是界面上的TextBox变量

tbx.Text += string.Format("{0}\n", i);

}

  运行后,任务管理器显示29.4 MB(几秒后又变成了24 MB)。而没有设置UndoLimit,也就是.NET 4.5中默认值是100的情况下,占用内存则能飙升到55.5 MB。在非.NET 3.5或者4下可能会更大,因为默认是没有限制的。

  那么问题的解决方案是:

  适当设置WPF的TextBox.UndoLimit(尤其是.NET 3.5/4.0环境下,默认值-1太可怕了)。当然这仅仅应用在频繁设置TextBox值的情况下,如果没有此类情况,无需担心。另外也要注意如果要进行频繁字符串拼接操作,请使用TextBoxBase.AppendText或者StringBuilder。

  本文来自刘圆圆博客,原文地址:http://www.cnblogs.com/mgen/archive/2013/02/24/2924558.html

时间: 2024-08-02 05:03:51

WPF:警惕TextBox会占用过多内存的相关文章

WPF的TextBox产生内存泄露的情况

前段时间参与了一个WPF编写的项目,在该项目中有这样一个场景:在程序运行过程中需要动态地产生大量文本信息,并追加WPF界面上的一个TextBox的Text中进行显示.编写完之后,运行该项目的程序,发现在产生大量信息之后,发现系统变慢了,打开任务管理器才发现,该项目的程序占用了将近1.5G的内存(天啊!!!这不是一般的耗内存啊!!!).后来通过查资料和探索才发现了WPF的TextBox在追加Text显示文本时会造成内存泄露.下面通过一个小Demo程序来展示一下这个内存泄露. 我的Demo程序很简单

mmdrv.exe进程占用大量内存及其他系统资源

收到阿里云的报警说是服务器进程数过多,然后登陆云监控一看,吓了一跳发现服务器的内存,cpu各种上升,登陆查看任务管理器进程出现好多mmdrv.exe进程,仔细寻觅一番才发现是测试工程师在使用Loadrunner做测试(尼玛也不说一声),原来虚惊一场,以下是自己了解的一些Loadrunner: Load Generator 对系统的要求 如果使用进程方式并发,并发10个虚拟用户,则会在任务管理器中出现10个mmdrv.exe进程: 如果使用线程方式并发,并发10个虚拟用户,只会在任务管理器中出现1

查看进程占用的内存情况

可以直接使用top命令后,查看%MEM的内容.可以选择按进程查看或者按用户查看,如想查看java用户的进程内存使用情况的话可以使用如下的命令: (1)top top命令是Linux下常用的性能分析工具,能够实时显示系统中各个进程的资源占用状况,类似于Windows的任务管理器 可以直接使用top命令后,查看%MEM的内容.可以选择按进程查看或者按用户查看,如想查看tomcat用户的进程内存使用情况的话可以使用如下的命令: $ top -u tomcat 内容解释: PID:进程的ID    US

Android高效内存1:一张图片占用多少内存

在做内存优化的时候,我们发现除了解决内存泄露问题,剩下的就只有想办法减少真实的内存占用.而在App中,大部分内存可能被我们图片占用了,所以减少图片的内存占用可以带来直接的效果.本文就简单介绍一张图片到底占用多少内存,我们先假设我们有一张图片时 600 * 800 的,图片占用空间大小假设是 100KB. 图片内存大小跟占用空间大小有什么关系? 占用空间的大小不是图片占用内存的大小,一些初学者可能会误解一下.占用空间是在磁盘上占用的控件,内存大小是加载到内存中占用的内存大小.两个只是单位是一样的,

由C/C++编译的程序占用的内存分类

一个由C/C++编译的程序占用的内存分为以下几个部分 1.栈区(stack):由编译器自动分配释放 ,存放函数的参数值,局部变量的值等.其操作方式类似于数据结构中的栈. 2.堆区(heap) : 一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 .注意它与数据结构中的堆是两回事,分配方式倒是类似于链表,呵呵. 3.全局区(静态区)(static):,全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另一块区

如何获知PHP程序占用多少内存

想要知道编写的 PHP 脚本需要占用多少内存么?很简单,直接使用 PHP 查看当前分配给 PHP 脚本的内存的函数 memory_get_usage() 就可以了. 下面是使用示例:赌王娱乐城 <?php echo memory_get_usage(), '<br />'; // 313864 $tmp = str_repeat('http://www.nowamagic.net/', 4000); echo memory_get_usage(), '<br />'; //

java对象的内存布局(一):计算java对象占用的内存空间以及java object layout工具的使用

最近在学习java对象内存布局方面的一些知识,主要是想知道一个java对象到底占用多少内存空间,以及java对象在内存中到底是什么样子的.c/c++中的sizeof运算符能够方便地告诉我们一个变量占用的内存空间,但是在java中却没有直接提供这种机制.如果想获取java对象占用的内存大小,可以利用java的Instrumentation机制.java.lang.instrument.Instrumentation这个接口提供了getObjectSize(Object objectToSize),

WPF循环加载图片导致内存溢出的解决办法

程序场景:一系列的图片,从第一张到最后一张依次加载图片,形成"动画". 生成BitmapImage的方法有多种: 1. var source=new BitmapImage(new Uri("图片路径",UriKind.xxx)); 一般的场景使用这种方法还是比较方便快捷,但是对于本场景,内存恐怕得爆. 2. var data =File.ReadAllBytes("图片路径"); var ms = new System.IO.MemoryStr

C++得到当前进程所占用的内存

原文地址:C++得到当前进程所占用的内存作者:雪碧狗 使用SDK的PSAPI (Process Status Helper)中的BOOL GetProcessMemoryInfo(  HANDLE Process,  PPROCESS_MEMORY_COUNTERS ppsmemCounters,  DWORD cb); typedef struct _PROCESS_MEMORY_COUNTERS {  DWORD cb;  DWORD PageFaultCount;  SIZE_T Peak