[WPF 容易忽视的细节] —— Exception in WPF's Converter
前言:
在WPF中,Converter是我们经常要用到的一个工具,因为XAML上绑定的数据不一定是我们需要的数据。
问题:
在Converter中抛出一个异常导致程序崩溃,而且是在对未捕获异常进行集中处理的情况。
补充:错误场景。
<Window x:Class="TestProject.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:TestProject"
Title="MainWindow"
Width="525"
Height="350"
DataContext="{Binding RelativeSource={RelativeSource Self}}">
<Window.Resources>
<local:ErrorConverter x:Key="ErrorConverter" />
</Window.Resources>
<Grid>
<TextBlock Text="{Binding Content, Converter={StaticResource ErrorConverter}}" />
</Grid>
</Window>
namespace TestProject
{
using System;
using System.Globalization;
using System.Windows.Data; /// <summary>
/// 转换过程中跑出异常的Converter。
/// </summary>
public class ErrorConverter : IValueConverter
{
/// <summary>
/// 只抛出错误,不做任何错误。
/// </summary>
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
throw new Exception("Just for test! - In ErrorConverter - Convert");
} /// <summary>
/// 只抛出错误,不做任何错误。
/// </summary>
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
throw new Exception("Just for test! - In ErrorConverter - ConvertBack");
}
}
}
上面的场景中,当TextBlock绑定到Content时,便会触发ErrorConverter的Convert方法,
但是Convert方法因为抛出了异常,导致整个程序挂掉。
虽然,在App.xaml.cs中集中对未捕获异常进行处理,但是却无法捕获这个异常。
public App()
{
this.DispatcherUnhandledException +=new DispatcherUnhandledExceptionEventHandler(Application_DispatcherUnhandledException);
AppDomain.CurrentDomain.UnhandledException +=new UnhandledExceptionEventHandler(CurrentDomain_UnhandledException);
}
探索过程:
1、UnhandledException
因为程序中,难免会出现没有处理的异常,这些异常如果没有被处理就会先被抛给我们的主程序,
如果还未被处理就会被抛给.Net Framework,最终导致程序挂掉。
所以,我时常集中处理未被处理的这部分异常,以防止程序挂掉。
相关内容可以参考:CSharp UnhandledException。
当捕获到未被处理的异常时,我就会弹出一个消息框,然后打印到日志中。
理论上来说,这是可以捕获所有程序中抛出的异常的,但是当Converter中抛出异常时,却导致程序崩溃。
通过Debug,导致崩溃异常为StackOverflowException。
2、StackOverflowException
一方面这个异常的名字,是我已经去的一个网站;另外一方面这个异常是无法被捕捉的。
这是MSDN上的解释:StackOverflowException 。
而且,我的经验是一般出现这个异常就是说明程序中出现了死循环。
但是,当我检查程序时发现,我并没有写此类代码。
而且错误出现在MessageBox.Show("");这行。(就是我在主程序中处理未捕获的异常时,抛出消息框)。
// 当捕获未处理异常时
Exception ex = e.Exception;
const string errorMsg = "UI Thread Exception : \n\n";
MessageBox.Show("An unhandled UI Thread exception occurred");
Logger.Error(errorMsg + ex.Message + Environment.NewLine + ex.StackTrace);
e.Handled = true;
将此行注释掉,发现一切正常,程序不再挂掉。说明是MessageBox引起StackOverflowException。
3、 MessageBox
当MessageBox.Show("");时,会阻塞当前线程,所以理论上来说每次只能显示一个MessageBox。
// 点击按钮执行以下代码
// 只有点击MessageBox上的确认按钮才会显示下一个,不会一次全部显示。
for (int i = ; i < ; i++)
{
MessageBox.Show("Just For test");
}
说明应该是MessageBox.Show被多次调用,造成StackOverflowException。
4、 Logger
再去检查日志,发现没有日志信息打印出来,说明应该是一直停留在MessageBox.Show这里,
所以日志还没来得及打印程序便挂掉了。
5、Trace Converter
有一个猜想,就是Converter被调用了多次:当xaml解析器发现Converter抛出异常时,
便再次调用Converter,导致又一次抛出异常,主程序捕获后,又执行MessageBox.Show,
所以程序挂掉了。
验证猜想:
public class TestConverter : IValueConverter
{
// 临时变量,用于记录被调用的次数
private int temp = ; public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
// 打印出次数
using (var writer = File.AppendText("D:\\Log.txt"))
{
writer.WriteLine(DateTime.Now + " - " + temp++);
} throw new Exception("Just for test!");
} public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
throw new NotImplementedException();
}
}
再次运行错误的程序,不例外程序崩溃,但是我也得到了想要的数据。
9/12/2013 3:01:42 PM - 0
9/12/2013 3:01:42 PM - 1
9/12/2013 3:01:42 PM - 2
9/12/2013 3:01:42 PM - 3
9/12/2013 3:01:42 PM - 4
9/12/2013 3:01:42 PM - 5
9/12/2013 3:01:42 PM - 6
9/12/2013 3:01:42 PM - 7
9/12/2013 3:01:42 PM - 8
9/12/2013 3:01:42 PM - 9
9/12/2013 3:01:43 PM - 10
9/12/2013 3:01:43 PM - 11
9/12/2013 3:01:43 PM - 12
9/12/2013 3:01:43 PM - 13
9/12/2013 3:01:43 PM - 14
跟我猜想的一样,但是Converter确被调用了15次,这也是导致程序崩溃的原因。
结论:
1、 当MessageBox.Show被同时多次调用时,会出现StackOverflowException的异常。
这个我不知道如何验证,但是貌似是这样的。
2、当xaml发现Converter抛出异常时,会继续执行Convert方法,最多15次。
完全不知道这个有什么意义,已经抛出异常了,再次调用就不会有异常?
3、尽量在Converter中将异常处理好,否则容易引起不必要的麻烦。
如果条件错误、或者转换失败,则返回null。而不是抛出异常。
写的时间比研究的时间长多了,希望对大家有帮助,希望大神能够回答我的疑问。
引用请注明出处:Exception in WPF's Converter
[WPF 容易忽视的细节] —— Exception in WPF's Converter的更多相关文章
- [WPF 容易忽视的细节] —— x:Name与Name属性
一.前言 WPF使用XAML来对界面进行编写,界面与后台逻辑分离.我们也可以写Style.Trigger来实现一些界面效果, 这些都是通过Name来定位控件的,例如Setter.TargetName. ...
- WPF Set connectionId threw an exception异常 以及重复dll的问题
1.DataOutputWPF 在显示norlib.Basic.UserConfigControl时 抛出异常 xmlparsingException : WPF Set connectionId t ...
- “WPF老矣,尚能饭否”—且说说WPF今生未来(上):担心
近日微软公布了最新的WPF路线图,一片热议:对于老牌控件提供商葡萄城来说,这是WPF系列控件一个重要的机遇,因此,Spread Studio for WPF产品做了一次重要更新,并随着Spread S ...
- WPF老矣,尚能饭否——且说说WPF今生未来(中):策略
本文接上文<WPF老矣,尚能饭否——且说说WPF今生未来(上):担心>继续. “上篇”中部分精彩的点评: 虽然WPF不再更新了,但是基于WPF的技术还是在发展着,就比如现在的WinRT,只 ...
- WPF老矣,尚能饭否——且说说WPF今生未来(下):安心
在前面的上.中篇中,我们已经可以看到园子里朋友的点评“后山见! WPF就比winform好! 激情对决”.看到大家热情洋溢的点评,做技术的我也很受感动.老实说,如何在本文收笔--WPF系列文章,我很紧 ...
- [WPF系列]基础学习(一) WPF是什么?
引言 学习之前,我们首先大概了解下WPF诞生的背景以及它所能解决的问题或者新颖之处.WPF作为微软新一代的用户界面技术, WPF简介 WPF的全称是WindowsPresentationFound ...
- WPF快速入门系列(5)——深入解析WPF命令
一.引言 WPF命令相对来说是一个崭新的概念,因为命令对于之前的WinForm根本没有实现这个概念,但是这并不影响我们学习WPF命令,因为设计模式中有命令模式,关于命令模式可以参考我设计模式的博文:h ...
- WPF快速入门系列(4)——深入解析WPF绑定
一.引言 WPF绑定使得原本需要多行代码实现的功能,现在只需要简单的XAML代码就可以完成之前多行后台代码实现的功能.WPF绑定可以理解为一种关系,该关系告诉WPF从一个源对象提取一些信息,并将这些信 ...
- wpfのuri(让你完全明白wpf的图片加载方式以及URI写法)
绝对 pack WPF URI pack://application:,,,/是协议:“,,,”是“///”的变体 1.资源文件 — 本地程序集 Uri uri = new Uri("pac ...
随机推荐
- ASP.NET Core 中使用 Hangfire 定时启动 Scrapyd 爬虫
用 Scrapy 做好的爬虫使用 Scrapyd 来管理发布启动等工作,每次手动执行也很繁琐;考虑可以使用 Hangfire 集成在 web 工程里. Scrapyd 中启动爬虫的请求如下: curl ...
- Enterprise Library 6 学习笔记
今天是2014年上班第一天,想着今年要做点与往年不同的事情,就从写博客开始吧. 公司的项目一般都用微软的企业库,一直没时间好好研究,第一天上班还不忙,就抽空研究了下.搜索一下,发现这个企业库已经到了6 ...
- filebeat-2-通过kafka队列链接logstash
filebeat 直接到logstash, 由于logstash的设计问题, 可能会出现阻塞问题, 因为中间使用消息队列分开 可以使用redis, 或者kafka, 这儿使用的是kafka 1, 安装 ...
- C/C++编程GUI库比较
转自:http://blog.csdn.net/lostown/article/details/658654 最强的GUI库当属Qt,毕竟是商业化的东西,功能最完整,什么都好,包括类似java代码风格 ...
- Charles在Mac中抓包使用说明
在工作期间,经过同事介绍,发现一款很强大的抓包工具Charles,现在记录下来分享给大家.常用的有以下几款功能: 1.支持配置抓取定向地址的网络请求 打开charles,打开Proxy->Rec ...
- .Net Core MVC实现自己的AllowAnonymous
全局过滤,在Startup中ConfigureServices里面添加如下代码 services.AddMvc(options => { options.Filters.Add(typeof(M ...
- JS forEach()与map() 用法(转载)
JavaScript中的数组遍历forEach()与map()方法以及兼容写法 原理: 高级浏览器支持forEach方法语法:forEach和map都支持2个参数:一个是回调函数(item,ind ...
- Linux安装redis和部署
第一步:下载安装包 访问https://redis.io/download 到官网进行下载.这里下载最新的4.0版本. 第二步:安装 1.通过远程管理工具,将压缩包拷贝到Linux服务器中,执行解压 ...
- 积分之迷-2015决赛C语言B组第一题
标题:积分之迷 小明开了个网上商店,卖风铃.共有3个品牌:A,B,C. 为了促销,每件商品都会返固定的积分. 小明开业第一天收到了三笔订单: 第一笔:3个A + 7个B + 1个C,共返积分:315 ...
- EF框架的三种模式
Database First就是先建数据库或使用已有的数据库.然后在vs中添加ADO.Net实体数据模型,设置连接并且选择需要的数据库和表.它是以数据库设计为基础的,并根据数据库自动生成实体数据模型, ...