在我们平常编程中,时间久了有时候会形成一种习惯性的思维方式,形成固有的编程风格,但是有些地方是需要斟酌的,即使是一个很小的错误也可能会导致昂贵的代价,要学会善于总结,从错误中汲取教训,尽量不再犯同样错误,注重编程之美,代码的优雅,总结几个平常经常犯的错误。

1、在C#编程中,字符型类型是最容易处理出错的地方,代价是非常昂贵,在.Net Framwork中,字符串是一个相当特别的引用类型,string本省就是一个不可继承的密封类,但是它具有了值类型所应用的特点,但是它在CLR中内 存还是保存于托管堆之上,也就是说,当我们每次定义一个字符串类型的时候,就在堆内存中开辟一端内存,而当我们字符串被修改之后,它会创建一个新的内存, 注意这里的内存是不连续的,而是通过修改栈内地址引用而拼凑字符串,不会改变源字符串在内存中的地址,所以有些程序员总是喜欢使用这样的方法格式化字符 串:

  1. string  SelectText="select * from "+TableName+" where UserName='"+Name+"'";

上述代码,使用了字符串拼凑的方法,因为使用了多重串联,因此会在内存中创建两个不必要的字符串垃圾副本。

其实在C#中,已经为我们提供了StringBuilder和String.Fromat来解决此问题,虽然他们可以实现同样的功能,但是他们有质 的变化,StringBuilder在内存中开辟的是一段连续内存,当增加新字符串时候,它会在栈中指向的同一个堆内存中连续存放字符,这就形成了性能的 提升。所以我们将上面代码改成:

  1. string SelectText=string.Format("select  *  from {0} where UserName={1}",TableName,Name);

2、大多数开发人员都不知道内置的验证数据类型的方法,如System.Int32,因此很多人都是自己实现的,其实这是不妥的,因为这些基本类型中都存在自己固有的类型验证方法,下面这个就是自己实现验证的一个字符串是否是数值的代码:

  1. public bool CheckIfNumeric(string value)
  2. {
  3. bool IsNumeric=true;
  4. try
  5. {
  6. int i=Convert.ToInt32(value);
  7. }
  8. catch(FormatException excepiton)
  9. {
  10. IsNumeric=false;
  11. }
  12. return IsNumeric;
  13. }

虽然使用了try catch语句,这不是最佳的做法,更好的方法是下面使用Int.TryParse;

  1. int output=0;
  2. bool IsNumeric=int.TryParse(value,out output);

int.TryParse是更快、更简洁的方法。

3、自己利用IDisposable接口手动释放内存

在.NET Framework中,对象的处理和使用一样重要,理想的方法是在使用完对象的时候,在类中实现IDisposable接口中的dispose方法进行内 存的释放,当然在.Net本身提供的垃圾回收机制(GC)中就提供了这样的功能,在我们实例化类对象时,在类本身的析构函数中会调用dispose方 法,GC在各级内存堆满的情况下,自动检查对象使用情况,去相应的释放内存,但是运行在非托管平台上的方法,需要我们自己手动释放内存,比如我们常见的 SqlConnection对象,也就有了下面的创建、使用和处理方法:

  1. public void  DALOneMethod()
  2. {
  3. SqlConnection  connection=null;
  4. try
  5. {
  6. connection =new SqlConnection("。。。。。。。。。。。");
  7. connection.Open();
  8. //sqlcommand。。run
  9. }
  10. catch(Exception exception)
  11. {
  12. // manager exception
  13. }
  14. finally
  15. {
  16. connection.Close();
  17. connection.Disopse();
  18. }
  19. }

上述代码是大部分程序员会出现的代码,乍看没啥问题,连接处理在最后一个代码中被明确调用,但是如果发生了一个异常,catch代码块就被执行,然 后再执行最后一个代码块处理连接,因此在最后一个代码块执行之前,连接将一直留在内存中,大部分我们会在此处记录错误,一般涉及到IO操作,如果延时时间 比较长的话,这个连接将在内存时间长时间停留。我们一个原则就是当对象不再使用的时候我们里面释放资源。

我们采用程序逻辑域来处理这个问题会更好:

  1. public void  DALOneMethod()
  2. {
  3. using(SqlConnction  connection=new SqlConnection("。。。。。。。"))
  4. {
  5. connction.Open();
  6. // do SUAD
  7. }
  8. }

当使用using代码快时,对象上的dispose()方法将在执行推出逻辑域的时候调用,这样就保证了SqlConnection的资源处理被尽 早释放,当然这个方法也适用于实现IDisposable接口的类,当时个人不推荐这样做,在非常有把握的情况下可以手动释放,但是没把握还是叫 给.net系统释放,因为本身类的析构函数就实现这个方法,当我们自己重写后,反而会导致系统误以为你自己定义了方法,而推迟释放资源,有兴趣可以研究下 GC运行本质,假如能在第一代被释放的内存,如果我们重写dispose方法反而推迟到第二代内存堆中释放,显然是不可取的。

4、学会合理的管理公共变量,我们在系统中经常会滥用公共变量,没有做到合适的封装好。

  1. static  void Main(string[]  args)
  2. {
  3. MyAccount  account=new MyAccount();
  4. //这地方不能随便的调用account里面的字段进行更改,但是缺改了
  5. account.AccountNumber="ddddddddd";
  6. Console.ReadKey();
  7. }
  8. public class MyAccount
  9. {
  10. public  string AccountNumber;
  11. public  MyAcctount()
  12. {
  13. AccountNumber="ssssssssssssss";
  14. }
  15. }

在上面的MyAccount类中生命了一个AccountNumber公共变量,理想情况下,AccountNumber应该是只读的,不能让外界修改,但是这里MyAccount类却没有对它做任何控制。

声明公共做法应该是使用属性,如:

  1. public  class  MyAccount
  2. {
  3. private stirng _accountNumber;
  4. public  string AccountNumber
  5. {
  6. get {  return  _accountNumber;  }
  7. }
  8. public  MyAccount()
  9. {
  10. _accountNumber="dddddddd";
  11. }
  12. }

这里我们封装了AccountNumber公共变量,它变成了只读,不能由调用者类进行修改。

5、嵌套的异常处理,有的开发人员喜欢在方法末尾加上处理的嵌套方法,如

  1. public class NestedExceptionHandling
  2. {
  3. public void MainMethod()
  4. {
  5. try
  6. {
  7. //some implementation
  8. ChildMethod1();
  9. }
  10. catch (Exception exception)
  11. {
  12. //Handle exception
  13. }
  14. }
  15. private void ChildMethod1()
  16. {
  17. try
  18. {
  19. //some implementation
  20. ChildMethod2();
  21. }
  22. catch (Exception exception)
  23. {
  24. //Handle exception
  25. throw;
  26. }
  27. }
  28. private void ChildMethod2()
  29. {
  30. try
  31. {
  32. //some implementation
  33. }
  34. catch (Exception exception)
  35. {
  36. //Handle exception
  37. throw;
  38. }
  39. }
  40. }

如果相同的异常被处理多次,性能开销将会增加。

我们的解决方法是让异常处理方法独立开来,如:

  1. public class NestedExceptionHandling
  2. {
  3. public void MainMethod()
  4. {
  5. try
  6. {
  7. //some implementation
  8. ChildMethod1();
  9. }
  10. catch(Exception exception)
  11. {
  12. //Handle exception
  13. }
  14. }
  15. private void ChildMethod1()
  16. {
  17. //some implementation
  18. ChildMethod2();
  19. }
  20. private void ChildMethod2()
  21. {
  22. //some implementation
  23. }
  24. }

6、大数据量上使用Dataset和DataReader混用,当单表数据量很大的情况,使用DataSet是一种很不明智的选择,应为 DataSet是以DataTable内存形式存放数据量,一次性将数据拖入内存,当数据很大的情况下,这种方式是很吃内存的,相比 DataSer,DataReader就显得优雅很多,它是每次读取一条数据,然后轮询调用机制,但是也有它的弊端,就是相对长连接,但是对内存消耗而言 这是有利的,当然DataSet在大部分应用场景下也是有自己的优点,充分解耦、一次性操作、领域模型操作等方面,两者分情况分场景而用,这里只是稍微提 提,根据场景分析区别。

原文链接:http://www.cnblogs.com/zhijianliutang/archive/2012/03/20/2407688.html

C#几个经常犯错误汇总的更多相关文章

  1. 编程中易犯错误汇总:一个综合案例.md

    # 11编程中易犯错误汇总:一个综合案例 在上一篇文章中,我们学习了如何区分好的代码与坏的代码,如何写好代码.所谓光说不练假把式,在这篇文章中,我们就做一件事——一起来写代码.首先,我会先列出问题,然 ...

  2. Entity Framework学习笔记——错误汇总

    之前的小项目做完了,到了总结经验和更新学习笔记的时间了.开始正题之前先啰嗦一下,对之前的学习目标进行一个调整:“根据代码生成表”与“生成数据库脚本和变更脚本”合并为“Code First模式日常使用篇 ...

  3. 李洪强iOS开发之OC常见错误汇总

    // //  main.m //  16 - 常见错误汇总 // //  Created by vic fan on 16/7/13. //  Copyright © 2016年 李洪强. All r ...

  4. scanf()常犯错误

    ------------------------------------------------------------------------ <> 本意:接收字符串. 写成代码:voi ...

  5. Python常见的错误汇总

    +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 错误: [错误分析]第二个参数必须为类,否则会报TypeError,所以正确的应 ...

  6. PHP常见错误汇总

    日常开发和调试的时候,经常会遇到一些错误,光怪陆离的不知所以,所以,特此将错误汇总一下,借鉴!!! 1. 原因分析:  一般可能是该文件出现了问题,检查一下代码和格式,是否出现开始的地方出现了空格,或 ...

  7. Quartus II 中 Verilog 常见警告/错误汇总

    Verilog 常见错误汇总 1.Found clock-sensitive change during active clock edge at time <time> on regis ...

  8. [golang 易犯错误] golang 局部变量初始化:=的陷阱

    我们知道,golang中局部变量初始化方法(使用“:=”创建并赋值),让我们在使用变量时很方便.但是,这也是易犯错误的地方之一.特别是这个初始化符还支持多个变量同时初始化,更特别的是它还支持原有变量赋 ...

  9. Python 新手常犯错误

    Python 新手常犯错误(第二部分) 转发自:http://blog.jobbole.com/43826/ 作用域 在这篇文章里,我们来关注作用域在Python被误用的地方.通常,当我们定义了一个全 ...

随机推荐

  1. poj2155 树状数组 Matrix

    Matrix Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 14826   Accepted: 5583 Descripti ...

  2. jdbc线程池

    连接oracle数据库的jdbc线程池 首先建立一个properties类型的文件存放一些信息:jdbc.properties driverClassName=oracle.jdbc.driver.O ...

  3. 很近没读书了,读书笔记之<<大道至简>>

    空闲时间不想虚度,不知道干啥的时候,就读读存在移动硬盘里的电子书吧,已经放了N久了,不知道什么时候放的,好像是大学刚毕业的时候下载的,...... 好久...... 现在才去读..是不是太晚了.... ...

  4. URAL 1784 K - Rounders 找规律

    K - RoundersTime Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hust.edu.cn/vjudge/contest/view. ...

  5. java源码部署

    环境:nginx+tomcat部署方式:源码部署源码目录 /chroot2/test/schedule 目录下面就是所有源码了tomcat 位置: /usr/local/tomcat/apache-t ...

  6. Keeplived 详解

    http://www.cnblogs.com/pricks/p/3822232.html

  7. The internals of Python string interning

    JUNE 28TH, 2014Tweet This article describes how Python string interning works in CPython 2.7.7. A fe ...

  8. Jordan Lecture Note-10: Kernel Principal Components Analysis (KPCA).

    Kernel Principal Components Analysis PCA实际上就是对原坐标进行正交变换,使得变换后的坐标之间相互无关,并且尽可能保留多的信息.但PCA所做的是线性变换,对于某些 ...

  9. 架构师书单 2nd Edition--转载

    作者:江南白衣,原文出处: http://blog.csdn.net/calvinxiu/archive/2007/03/06/1522032.aspx,转载请保留. 为了2007年的目标,列了下面待 ...

  10. SQL Insert语句数据以以unicode码存储 解决存储数据出现乱码的问题

    写了个读取原始的文本数据导入数据库的工具 ,最后发现空中有几个值是乱码 例如 原始数据是 :Bjørn 存到数据库中是 Bj?rn 研究半天发现是一直以来忽略了一个标记‘N’ 2条 Insert 语句 ...