c#:使用using关键字自动释放资源未必一定就会有明显好处
记录这篇文章的灵感来源来自今天下班前与同事的小小争论,我现在开发的一个项目中,有这样一段代码:
public string ToXML()
{
string strXml = string.Empty;
try
{
MemoryStream ms = new MemoryStream();
XmlSerializer xml = new XmlSerializer(this.GetType());
xml.Serialize(ms, this);
byte[] arr = ms.ToArray();
strXml = Encoding.UTF8.GetString(arr, 0, arr.Length);
return strXml;
}
catch
{
return "";
}
}
同事说象MemoryStream这类资源,应该用using包起来自动释放资源,否则会有内存泄漏问题。在using的使用上,我也同意应该使用using,但由于这类风格的代码在原项目中非常多(有一部分历史原因),如果一一修改,工作量太大,时间不允许。于是我就在内心评估:如果不改,现在这种代码的风险到底有多大?
我想很多人都知道using(Resource res = new Resrouce){},其实相当于
Resource res = new Resrouce
try{}
catch{}
finally{res.Dispose();}
对比与现有代码的区别,无非就是资源没有调用Dispose()释放,但是CLR有强大的GC(垃圾回收)机制,方法调用完成后,方法体中创建的托管资源如果不再被使用,也一并会被GC列为可回收对象,所以就算开发人员没有手动调用Dispose,其实CLR也会帮我们做这件事情,只是时机可能会晚一些而已。
于是有了下面的测试:
1.先创建一个示例用的Class
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.IO;
using System.Xml.Serialization;
namespace Model
{
public class SampleClass
{
public string Name { set; get; }
public string ToXMLNoUsing()
{
string strXml = string.Empty;
try
{
MemoryStream ms = new MemoryStream();
XmlSerializer xml = new XmlSerializer(this.GetType());
xml.Serialize(ms, this);
byte[] arr = ms.ToArray();
strXml = Encoding.UTF8.GetString(arr, 0, arr.Length);
return strXml;
}
catch
{
return "";
}
}
public string ToXMLWithUsing()
{
string strXml = string.Empty;
try
{
using (MemoryStream ms = new MemoryStream())
{
XmlSerializer xml = new XmlSerializer(this.GetType());
xml.Serialize(ms, this);
byte[] arr = ms.ToArray();
strXml = Encoding.UTF8.GetString(arr, 0, arr.Length);
}
return strXml;
}
catch
{
return "";
}
}
}
}
这其中的ToXML为了测试方便,故意分成了二个版本(一个不用using,一个用using)
2.再创建一个Console程序(命名为WithUsing),写一段测试代码:
using System;
using System.Diagnostics;
using Model;
namespace WithUsing
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("开始折腾-WithUsing...");
Stopwatch watch = new Stopwatch();
int max = 100000;
watch.Reset();
watch.Start();
for (int i = 0; i < max; i++)
{
SampleClass c = new SampleClass() { Name = i.ToString().PadLeft(1024, '0') };
c.ToXMLWithUsing();
}
watch.Stop();
Console.WriteLine("完成,{0}次操作共耗时:{1}毫秒,平均{2}毫秒/次!", max, watch.ElapsedMilliseconds, watch.ElapsedMilliseconds /(decimal)max);
Console.ReadKey();
}
}
}
3.再创建一个Console程序(命名为NoUsing),写一段测试代码:
using System;
using System.Diagnostics;
using Model;
namespace NoUsing
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("开始折腾-NoUsing...");
Stopwatch watch = new Stopwatch();
int max = 100000;
watch.Reset();
watch.Start();
for (int i = 0; i < max; i++)
{
SampleClass c = new SampleClass() { Name = i.ToString().PadLeft(1024, '0') };
c.ToXMLNoUsing();
}
watch.Stop();
Console.WriteLine("完成,{0}次操作共耗时:{1}毫秒,平均{2}毫秒/次!", max, watch.ElapsedMilliseconds, watch.ElapsedMilliseconds / (decimal)max);
Console.ReadKey();
}
}
}
编译后,同时运行这二个程序,同时利用任务管理器观察内存使用情况:
反复多次运行比较,发现其实二者占用的内存几乎完全相同,这说明GC还是很给力的!
而且从执行时间上看,不用Using,反而更快,这也容易理解:用Using相当于每次都要调用Dispose()方法,这会带来一些系统开销;而不用Using,GC会在适当的时机批量回收资源,性能反而更好。(当然:这个结论不是要误导大家不用using,对于using还是推荐使用的!我的用意在于大家对于一些具体问题要具体分析,不可纯教条主义,一味迷信某些主流的观点)
- Vagrant使用
- java 中的异步回调
- 澳大利亚域名管理机构多年敲竹杠?
- ASP.NET中使用HttpWebRequest调用WCF
- scala 学习笔记(06) OOP(下)多重继承 及 AOP
- Angular企业级开发(4)-ngResource和REST介绍
- CSS魔法堂:"那不是bug,是你不懂我!" by inline-block
- scala 学习笔记(03) 参数缺省值、不定个数参数、类的属性(Property)、泛型初步
- Cmd Markdown编辑器简明语法手册
- 如何让spring mvc web应用启动时就执行特定处理
- CSS魔法堂:小结一下Box Model与Positioning Scheme
- jboss EAP 6.2+ 通过代码控制JNDI数据源
- jboss CLI 命令行接口学习(适用JBOSS EAP 6.2+)
- WebComponent魔法堂:深究Custom Element 之 面向痛点编程
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 手把手教你画酷炫环形图(图文详解)
- 为什么说神经网络可以逼近任意函数?
- 缩放|位移|渐变简单动画
- HTTP状态码及其含义
- 卷积神经网络中的参数共享/权重复制
- 前端面试基础题:从浏览器地址栏输入url到显示页面的步骤
- 学会这15个TS面试题,拿到更高薪的offer
- 前端面试基础题:请描述⼀下 cookies , sessionStorage 和 localStorage 的区别?
- matplotlib基础绘图命令之pie
- 【从0到1学算法】快速排序
- 每天一道前端面试题:左边宽度固定,右边⾃适应
- 几个IDEA高级调试技巧,完全是bug杀手啊
- Spring Security 实战干货:从零手写一个验证码登录
- LaTex学习笔记
- 聊聊dubbo-go的metricsFilter