每天写代码,偶尔就会有让你抓狂的时候:代码改了千百遍,蓦然回首,Bug就在灯火阑珊处……这里就列举一些容易犯错的几个小地方,以后遇到了其他的,再慢慢添加。

  1. 获取程序当前运行路径

  情景复现:WPF客户端程序,开机自启动后无法进入主界面,卡在初始屏(Splash Screen)

  处理问题:通过日志发现加载一个icon的时候,跳了一个Bug。初始代码如下:

var icon = new Icon("Images\\xxx.ico");

    很简单,貌似不会有问题,相对目录且正确。直接双击程序启动完全正常,Debug启动同样完全正常,否则早就发现这个Bug了。开机自启动时日志中的错误是:找不到“C:\Windows\System32\Images\xxx.ico”这个文件 ??? 这很让人摸不着头脑,程序中的相对目录怎么会跑到sysem32里面了?目录不对导致文件找不到,当然就进入到Exception里面了。

    第一反应是相对目录可能不带靠谱,就改成了下面的代码:

var icon = new Icon(Directory.GetCurrentDirectory() + "\\Images\\xxx.ico");
//var icon = new Icon(Environment.CurrentDirectory + "\\Images\\xxx.ico");

    呵呵,还是不起作用,换一种写法(被注释的第二句),报的错是一样的。两个方法返回的都是“C:\Windows\System32”这个路径,在程序开机自启动的时候。其实Environment.CurrentDirectory内部调用的也是Directory.GetCurrentDirectory()方法。

   解决方案StackOverflow上面关于这个问题有个讨论,WinForm中Application.StartupPath也会有相同的问题,下面的是获取当前目录的推荐写法: 

var path = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);

  2. IEnumerable之LINQ 表达式

    软件设计有个很重要的原则,就是高内聚,低耦合,于是我们经常的会面向接口编程,并且尽可能开放底层接口。所以IEnumerable就会经常用到,因为Linq操作中很多方法的返回值都是IEnumerable<T>。这一个陷阱就是关于IEnumberable,先看下面的代码,看看返回值和你想的是不是一样:

 1   internal class Program
 2   {
 3     private static void Main(string[] args)
 4     {
 5       //...
 6       var students = GetStudents();
 7       foreach (var student in students)
 8       {
 9         student.IsActived = true;
10         Console.WriteLine(student.IsActived);
11       }
12 
13       foreach (var student in students)
14       {
15         Console.WriteLine(student.IsActived);
16       }
17 
18     }
19 
20     private static IEnumerable<string> GetNames()
21     {
22       //....
23       return new[] { "AAA""BBB""CCC" };
24     }
25 
26     private static IEnumerable<Student> GetStudents()
27     {
28       //...
29       return GetNames().Select(s => new Student(s));
30     }
31 
32     public class Student
33     {
34       public string Name { getset; }
35       public bool IsActived { getset; }
36 
37       public Student(string name)
38       {
39         Name = name;
40       }
41     }
42 
43   }
View Code

    第一个foreach里面会输出3个true,这毫无疑问,第二个foreach里面任然会输出3个true?

    如果你理解这样的结果,那么这个“陷阱”对你无效,你可以跳过这一条了……

    继续看下面的代码:

1       var studentList = GetStudents().ToList();
2       studentList[0].IsActived = true;
3       var studentsActived = studentList.Where(s => s.IsActived);
4 
5       Console.WriteLine(studentsActived.Count());
6 
7       studentList[0].IsActived = false;
8 
9       Console.WriteLine(studentsActived.Count());
View Code

    这次会输出什么?

    If(IsUnderstandingAgain) return; else……这里先不解释,我们用代码说话,继续看代码,修改一下GetStudents()方法:

 1     private static IEnumerable<Student> GetStudents()
 2     {
 3       //...
 4       return GetNames().Select(s =>
 5       {
 6         var stu = new Student(s);
 7         Console.WriteLine(s + "" + stu.GetHashCode());
 8         return stu;
 9       });
10     }

     在上面的代码中,GetStudent()方法被调用了2次,你觉得现在HashCode会输入几次?

 

 

     输出结果是9次。区域3里面的3次是由于调用GetStudents().ToList()方法,区域1和2则是由前面的两个foreach运行时输出的,而且每一次HashCode都不一样,说明每一个都是不同的实例。再联想一想Entity Framewor里面是不是有一个Lazy Loading,每一次使用集合中的某个对象,就会执行一次SQL,从数据库中查找该对象。 真相就在这里,Llinq只是表达式(这里用的都是lambda写法),可以这么理解:每个表达式它会自动生成一个匿名方法,只有在需要结果的时候这个匿名方法才会去执行,这也就是为什么它的返回值是IEnmerable<T>而不是一个具体的类。 所以在需要全部所需集合时,最好先执行ToList(),ToDictionary()这类方法,生成真正的结果。

 

 

 

 

posted @ 2014-07-17 17:09 Hans Huang 阅读(2074) 评论(16) 推荐(5) Edit
一、 让文本变成声音 .Net里面自带了一个语音类库:System.Speech,调用系统的语音功能,就能实现string到语音的转换,很简单(记得先把电脑音量调大一下)://AddSystem.SpeechreferencefirstusingSystem.Speech.Synthesis;varreader=newSpeechSynthesizer();reader.SpeakAsync("I'm a programer. Hello, world! "); Hello, world! 你听到了……这里我用了SpeakAsync方法,也就是异步执行,不会阻塞主线 Read More
posted @ 2013-11-13 22:27 Hans Huang 阅读(6628) 评论(40) 推荐(38) Edit
Visual Studio 号称有史以来最强大的IDE,确实如此。创建单元测试也是一键完成:在方法的代码块中右键“Create Unit Test…”,勾选测试项,填项目名,完成。VS就会自动帮你创建好了一个工程,直接写测试方法就好。这创建出来的工程也就是一个普通的类库(class library),所以今天我就打算手动创建一个单元测试工程,但是最后却发现总是无法显示代码覆盖率。 发现问题,解决问题:反复对比手动创建和自动创建的工程,所有的属性都没区别。但是有一点引起了我的注意,右键工程名-Add,区别出来了:自动创建的Unit Test类库好像被打上了标签,能够被VS识别为单元测试工程。.. Read More
posted @ 2013-10-24 15:12 Hans Huang 阅读(1532) 评论(2) 推荐(1) Edit
下周要在Ubuntu中调试程序,所以今天抽空先配置好Myeclipse环境。 准备: JDK 下载地址: http://www.oracle.com/technetwork/java/javase/downloads/jdk7-downloads-1880260.html MyEclipse 下载地址: http://www.myeclipseide.com/module-htmlpages-display-pid-4.html一、启用root账户 1.Ubuntu出于安全考虑默认关闭root账户,所以启用root账户之前要先为它设置密码,Ctrl+Al... Read More
posted @ 2013-06-29 20:02 Hans Huang 阅读(1329) 评论(0) 推荐(1) Edit
在我的上一篇随笔里介绍了Jquery文件上传插件Uploadify的一些基本配置参数、事件和方法,都是一些死的东西,这篇随笔里就介绍一些服务器端的配置和Demo。 用VS2010写Demo的时候,发现一个奇怪的现象,在FF、Chrome、IE6-8浏览器中文件上传一切正常,唯独在IE9中,Uploadify按钮点不动,无法打开选择文件对话框,郁闷啊,找原因:更新Flash Player、调整CSS样式,都不行……最后的解决办法竟然是在IIS中为demo单独配置一个站点,问题完美解决了!我也不知道Uploadify为什么在VS2010自带的调试服务器里面会出现这样的问题,而且只针对IE9…….. Read More
posted @ 2012-09-25 01:03 Hans Huang 阅读(2644) 评论(0) 推荐(0) Edit
上传文件的时候打算用Ajax方法,这样用户体验好一些,找来找去还是Uploadify好些,可惜相关的帮助文档不好找,园里有一篇可惜是2年前2.1版本的,和现在最新的3.2版差别很大,很多Option都已经取消了。唯一可用的就是官网(http://www.uploadify.com/),里面的资料还是很详细的,所以打算把它整理成中文以便于以后查阅(Flash版Uploadify)。 Options(选项、参数): Option Inpu... Read More
posted @ 2012-09-10 01:36 Hans Huang 阅读(3322) 评论(1) 推荐(0) Edit
原型(Prototype)模式:用原型实例指定创建对象的种类,并通过复制这些原型创建新的对象。其实就是从一个对象再创建另外一个可定制的对象,而且可以不需要知道创建的细节。当一个对象生成不是通过New而是通过复制旧对象的时候,可以考虑使用原型模式。 Read More
posted @ 2012-09-07 00:28 Hans Huang 阅读(562) 评论(0) 推荐(0) Edit
NaN:非数值(Not a Number),用于表示一个本来要返回数值的操作数未返回数值的情况。 Read More
posted @ 2012-09-06 14:13 Hans Huang 阅读(931) 评论(0) 推荐(0) Edit
创建者(Builder)模式适用于产品局部加工变化较大,但是组装过程性对固定的场景。创建者模式和工厂模式有点类似,不过关注点不同。工厂模式往往只关心你要的是什么,而创建模式则关心的是这个东西的具体细节的创建。 Read More
posted @ 2012-09-03 17:49 Hans Huang 阅读(458) 评论(0) 推荐(0) Edit
点击右上角即可分享
微信分享提示