• Silverlight陷阱:XAML中不能使用自定义字典 AG_E_PARSER_BAD_PROPERTY_VALUE


    我们知道,XAML中实际上是可以放置任何对象的,而系统将按照如下的规则管理嵌套的内容:

    1. 如果对象实现了IList,那么嵌套内容将通过IList.Add添加到父对象;

    2. 如果对象实现了IDictionary,并且元素用x:Key指定了键值,那么嵌套内容将通过IDictionary.Add添加到父对象;

    3. 如果只有父对象用ContentPropertyAttribute声明了内容属性,那么嵌套内容将被赋值为到该属性。

    我们在目前的项目中使用了很多XAML声明来减少编码量,但是在使用中我们发现,第2条对于Silverlight是不适用的,Silverlight的XAML只支持对Resources属性用字典方式来声明,对于自定义的字典内容,即使是ResourceDictionary也无法读取,否则运行时就会抛出异常。因为同样的方法在服务端已经普遍使用,所以我们把代码应用到Silverlight工程中的时候,根本没有想到这方面会出问题。从而花了很长时间、走了很多弯路去查找自己程序中的Bug,反复作了大量实验后,终于确定:这个问题来源于Silverlight和WPF读取XAML时的表现不同。

    还是用代码来说明。首先我们看看在WPF中使用自定义字典是否可行:

    publicclass BaseWindow : Window {     public BaseWindow()     {         Dict =new MyDict();     }
       
    public MyDict Dict { get; set; } }
    publicclass MyDict : Dictionary<string, Brush> { }

    然后在窗体中添加字典数据:

    <local:BaseWindow x:Class="TestWPF2.Window1"     xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"     xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"     Title="Window1"     xmlns:local="clr-namespace:TestWPF2">
       
    <local:BaseWindow.Dict>             <SolidColorBrush x:Key="1" Color="Blue"/>             <SolidColorBrush x:Key="2" Color="Black"/>     </local:BaseWindow.Dict></local:BaseWindow>

    最后检查一下声明的字典是否正确设置了:

    public Window1() {     InitializeComponent();     this.Background = Dict["1"]; }

    运行结果完全正确(如图) ,表明自定义字典在WPF中是可行的。

    然后我们再如法炮制一个Silverlight工程,运行之,出现异常:

    AG_E_PARSER_BAD_PROPERTY_VALUE [Line: 10 Position: 42]Type: XamlParseExceptionStackTrace:   

    位于 System.Windows.Application.LoadComponent(Object component, Uri resourceLocator)  

    位于 TestSL2.MainPage.InitializeComponent()  

    位于 TestSL2.MainPage..ctor()  

    位于 TestSL2.App.App_Startup(Object sender, StartupEventArgs e)  

    位于 System.Windows.CoreInvokeHandler.InvokeEventHandler(Int32 typeIndex, Delegate handlerDelegate, Object sender, Object args)  

    位于 MS.Internal.JoltHelper.FireEvent(IntPtr unmanagedObj, IntPtr unmanagedObjArgs, Int32 argsTypeIndex, String eventName)

    此实验可以说明WPF和Silverlight 在处理XAML时候的不同行为。你可以把MyDict换成ResourceDictionary试试,结果也是一样的。

    为了绕过这个问题,我们不得不在Silverlight工程中去掉字典,把内容重新组织成列表,然后在程序中将内容重新组织成字典——等于服务端已经调试好的方法在客户端又重写了一遍。非常恼人,但是没有办法。。

    这个问题告诉我们,尽管Silverlight 源出WPF,但处理细节上还是存在微妙的差别。将WPF的经验应用到Silverlight上,不见得会奏效。最好是先作一些小实验,确信方法是可行的,再添加到工程里,否则在大的项目里调试此类问题(尤其是XAML这类既缺乏编译器检查,又没有调用堆栈可查的东西)真够杀死你几万个脑细胞的。

  • 相关阅读:
    [转载]Back up all of your mysql databases nightly
    编写windows7 bat运行脚本
    Windows7开通Internet信息服务
    【LeetCode题解】7_反转整数
    【LeetCode题解】350_两个数组的交集Ⅱ
    【LeetCode题解】349_两个数组的交集
    【LeetCode题解】94_二叉树的中序遍历
    【LeetCode题解】144_二叉树的前序遍历
    【LeetCode题解】2_两数相加
    【LeetCode题解】530_二分搜索树的最小绝对值差
  • 原文地址:https://www.cnblogs.com/suizhikuo/p/2255855.html
Copyright © 2020-2023  润新知