应用程序设置库为自定义集合写入空标记

本文关键字:集合 设置 自定义 应用程序 | 更新日期: 2023-09-27 17:58:56

我已经断断续续地为此斗争了好几天了。 我需要将自定义对象的集合存储为用户设置的一部分。 基于大量的谷歌工作,似乎从ApplicationSettingsBase建立一个偏好类是一种合适的方法。 我遇到的问题是,一旦我尝试存储自定义类型的集合,就不会为该属性保存任何数据。 如果我保留基类型的集合,例如字符串,事情就会起作用。 我有一个单独的概念验证项目,过去一天我一直在工作,以隔离这个问题。

此项目由一个带有列表框和两个按钮"+"和"-"的 WPF 窗口组成。 我有一个 Prefs 类,其中包含三个定义不同类型集合的属性。 在我的窗口代码中,我将列表框绑定到其中一个列表,按钮可以在列表中添加或删除项目。 关闭窗口应将列表的内容保存到当前用户的 users.config 文件中。 重新打开它应该会显示列表的保存内容。

如果我使用 List1 ( ObservableCollection<string> ( 并单击 + 按钮几次,然后关闭窗口,数据将正确保存到 user.config。 但是,如果我更改并使用List2(ObservableCollection<Foo>(或List3(FooCollection(,我最终只会得到一个空的值标签。 我试图实现ISerializableIXmlSerializable,试图在不改变行为的情况下使其工作。 实际上,在带有断点的调试模式下运行表明,在调用 save 方法时集合包含数据,但从未调用序列化接口方法。

我错过了什么?

更新:我已经改变了我的方法,暂时将数据写到user.config旁边的另一个文件中,以便我可以在应用程序的其他部分取得一些进展。 但我仍然想知道为什么应用程序设置库无法记录我的收集数据。

这是所有相关代码,如果有人认为省略的部分很重要,我会将其添加回帖子中。

user.config 在每个列表属性中添加几个元素后

<setting name="List1" serializeAs="Xml">
    <value>
        <ArrayOfString xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
            xmlns:xsd="http://www.w3.org/2001/XMLSchema">
            <string>0</string>
            <string>1</string>
            <string>2</string>
        </ArrayOfString>
    </value>
</setting>
<setting name="List2" serializeAs="Xml">
    <value />
</setting>
<setting name="List3" serializeAs="Xml">
    <value />
</setting>

窗口 XAML

<Grid>
    <Grid.ColumnDefinitions>
        <ColumnDefinition></ColumnDefinition>
        <ColumnDefinition Width="Auto"></ColumnDefinition>
    </Grid.ColumnDefinitions>
    <ListBox Name="l1" Grid.Column="0" Grid.Row="0" ItemsSource="{Binding}"></ListBox>
    <StackPanel Grid.Column="1" Grid.Row="0">
        <Button Name="bP" Margin="5" Padding="5" Click="bP_Click">+</Button>
        <Button Name="bM" Margin="5" Padding="5" Click="bM_Click">-</Button>
    </StackPanel>
</Grid>

窗口的代码隐藏

public partial class Window1 : Window
{
    Prefs Prefs = new Prefs();
    public Window1()
    {
        InitializeComponent();
        //l1.DataContext = Prefs.List1;
        //l1.DataContext = Prefs.List2;
        l1.DataContext = Prefs.List3;
    }
    private void Window_Closed(object sender, EventArgs e)
    {
        Prefs.Save();
    }
    private void bP_Click(object sender, RoutedEventArgs e)
    {
        //Prefs.List1.Add(Prefs.List1.Count.ToString());
        //Prefs.List2.Add(new Foo(Prefs.List2.Count.ToString()));
        Prefs.List3.Add(new Foo(Prefs.List3.Count.ToString()));
    }
    private void bM_Click(object sender, RoutedEventArgs e)
    {
        //Prefs.List1.RemoveAt(Prefs.List1.Count - 1);
        //Prefs.List2.RemoveAt(Prefs.List2.Count - 1);
        Prefs.List3.RemoveAt(Prefs.List3.Count - 1);
    }
}

偏好类

class Prefs : ApplicationSettingsBase
{
    [UserScopedSettingAttribute()]
    [DefaultSettingValueAttribute(null)]
    public System.Collections.ObjectModel.ObservableCollection<string> List1
    {
        get
        {
            System.Collections.ObjectModel.ObservableCollection<string> Value = this["List1"] as System.Collections.ObjectModel.ObservableCollection<string>;
            if (Value == null)
            {
                Value = new System.Collections.ObjectModel.ObservableCollection<string>();
                this["List1"] = Value;
            }
            return Value;
        }
    }
    [UserScopedSettingAttribute()]
    [DefaultSettingValueAttribute(null)]
    public System.Collections.ObjectModel.ObservableCollection<Foo> List2
    {
        get
        {
            System.Collections.ObjectModel.ObservableCollection<Foo> Value = this["List2"] as System.Collections.ObjectModel.ObservableCollection<Foo>;
            if (Value == null)
            {
                Value = new System.Collections.ObjectModel.ObservableCollection<Foo>();
                this["List2"] = Value;
            }
            return Value;
        }
    }
    [UserScopedSettingAttribute()]
    [DefaultSettingValueAttribute(null)]
    public FooCollection List3
    {
        get
        {
            FooCollection Value = this["List3"] as FooCollection;
            if (Value == null)
            {
                Value = new FooCollection();
                this["List3"] = Value;
            }
            return Value;
        }
    }
}

福类

[Serializable()]
class Foo : System.ComponentModel.INotifyPropertyChanged, ISerializable, IXmlSerializable
{
    private string _Name;
    private const string PropName_Name = "Name";
    public string Name
    {
        get { return this._Name; }
        set
        {
            if (value != this._Name)
            {
                this._Name = value;
                RaisePropertyChanged(Foo.PropName_Name);
            }
        }
    }
    public override string ToString()
    {
        return Name;
    }
    public Foo() { }
    public Foo(string name)
    {
        this._Name = name;
    }
    #region INotifyPropertyChanged Members
/***Omitted for space***/
    #endregion
    #region ISerializable Members
    public Foo(SerializationInfo info, StreamingContext context)
    {
        this._Name = (string)info.GetValue(Foo.PropName_Name, typeof(string));
    }
    public void GetObjectData(SerializationInfo info, StreamingContext context)
    {
        info.AddValue(Foo.PropName_Name, this._Name);
    }
    #endregion
    #region IXmlSerializable Members
    public System.Xml.Schema.XmlSchema GetSchema()
    {
        return null;
    }
    public void ReadXml(System.Xml.XmlReader reader)
    {
        reader.MoveToContent();
        _Name = reader.GetAttribute(Foo.PropName_Name);
        bool Empty = reader.IsEmptyElement;
        reader.ReadStartElement();
        if (!Empty)
        {
            reader.ReadEndElement();
        }
    }
    public void WriteXml(System.Xml.XmlWriter writer)
    {
        writer.WriteAttributeString(Foo.PropName_Name, _Name);
    }
    #endregion
}

和福收藏类

[Serializable()]
class FooCollection : ICollection<Foo>, System.ComponentModel.INotifyPropertyChanged, INotifyCollectionChanged, ISerializable, IXmlSerializable
{
    List<Foo> Items;
    private const string PropName_Items = "Items";
    public FooCollection()
    {
        Items = new List<Foo>();
    }
    public Foo this[int index]
    {
/***Omitted for space***/
    }
    #region ICollection<Foo> Members
/***Omitted for space***/
    #endregion
    public void RemoveAt(int index)
    {
/***Omitted for space***/
    }
    #region IEnumerable Members
/***Omitted for space***/
    #endregion
    #region INotifyCollectionChanged Members
/***Omitted for space***/
    #endregion
    #region INotifyPropertyChanged Members
/***Omitted for space***/
    #endregion
    #region ISerializable Members
    public FooCollection(SerializationInfo info, StreamingContext context)
    {
        this.Items = (List<Foo>)info.GetValue(FooCollection.PropName_Items, typeof(List<Foo>));
    }
    public void GetObjectData(SerializationInfo info, StreamingContext context)
    {
        info.AddValue(FooCollection.PropName_Items, this.Items);
    }
    #endregion
    #region IXmlSerializable Members
    public System.Xml.Schema.XmlSchema GetSchema()
    {
        return null;
    }
    public void ReadXml(System.Xml.XmlReader reader)
    {
        XmlSerializer FooSerializer = new XmlSerializer(typeof(Foo));
        reader.MoveToContent();
        bool Empty = reader.IsEmptyElement;
        reader.ReadStartElement();
        if (!Empty)
        {
            if (reader.IsStartElement(FooCollection.PropName_Items))
            {
                reader.ReadStartElement();
                while (reader.IsStartElement("Foo"))
                {
                    this.Items.Add((Foo)FooSerializer.Deserialize(reader));
                }
                reader.ReadEndElement();
            }
            reader.ReadEndElement();
        }
    }
    public void WriteXml(System.Xml.XmlWriter writer)
    {
        XmlSerializer FooSerializer = new XmlSerializer(typeof(Foo));
        writer.WriteStartElement(FooCollection.PropName_Items);
        foreach (Foo Item in Items)
        {
            writer.WriteStartElement("Foo");
            FooSerializer.Serialize(writer, Item);
            writer.WriteEndElement();//"Foo"
        }
        writer.WriteEndElement(); //FooCollection.PropName_Items
    }
    #endregion
}

应用程序设置库为自定义集合写入空标记

我也有类似的问题,我设法解决了,但我没有使用ObservableCollection,而是正常的List<Column>Column是我的一类,包含作为公共成员:字符串、int 和布尔值,它们都是可 XML 序列化的。List<>,因为它实现了IEnumerable也是XMLSerializable

在自定义 Foo 类中,您唯一需要注意的是:您必须将成员作为公共成员,无参数构造函数并且类本身必须是公共的。您无需为 xml 序列化添加 [Serializable] 标记。

我不需要实现 FooCollection 类,因为我使用的是 List,它与 xml 序列化没有问题。

另一件事:

  • 派生自ApplicationSettingsBase的类可以是内部密封的 - 不需要它是公共的。
  • 在列表 prop 上方添加它是 XML 可序列化的事实:

    [global::System.Configuration.UserScopedSettingAttribute((][设置序列化为(设置序列化为.xml(][global::System.Configuration.DefaultSettingValueAttribute("(]公共列表列{ 获取 { 返回 ((列表(这个["列"](; } 设置 { this["列"] = (列表(值; }}

如果这不起作用,您也可以尝试实现TypeConverter

我也为

一个非常相似的问题挣扎了两天,但现在我发现一个缺失的链接可能会在这里有所帮助。如果你和我的情况实际上相似,那么类 'Foo' 和 'FooCollection' 需要明确公开!

我假设,在 List2( ObservableCollection<Foo> ( 和 List3(FooCollection) 的情况下,.NET 会遇到需要以某种方式显式公共访问(跨越自己程序集的边界(的IXmlSerializable

相反的选项列表 List1 ( ObservableCollection<string> ( 似乎在一个平面字符串类型转换上运行,它对(内部(类"Foo"感到满意......

(来自应用程序设置库文档:ApplicationSettingsBase使用两种主要机制序列化设置:1(如果存在可以与字符串相互转换的TypeConverter,我们使用它。2(如果没有,我们回退到XmlSerializer(

亲切问候鲁迪