gpt4 book ai didi

c# - ApplicationSettingsBase 为自定义集合写入空标签

转载 作者:太空宇宙 更新时间:2023-11-03 14:17:14 25 4
gpt4 key购买 nike

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

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

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

我错过了什么?

更新:我已经改变了我的方法,暂时将数据写入 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;
}
}
}

Foo 类

[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
}

和 FooCollection 类

[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作为我的一个类,它作为公共(public)成员包含:string、int 和 bool,它们都是 xmlserializable。 List<> , 因为它实现了 IEnumerable也是XMLSerializable .

在自定义 Foo 类中您唯一需要注意的事情是:您必须让成员为公共(public)成员、无参数构造函数并且类本身必须是公共(public)的。您不需要添加 [Serializable]用于 xml 序列化的标记。

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

另一件事:

  • 派生自 ApplicationSettingsBase 的类可以内部密封——不需要公开。
  • 在 list prop 之上添加它是 xml 可序列化的事实:

    [全局::System.Configuration.UserScopedSettingAttribute()][SettingsSerializeAs(SettingsSerializeAs.Xml)][全局::System.Configuration.DefaultSettingValueAttribute("")]公共(public)列表列{ 得到 { return ((List)this["Columns"]); } 放 { 这个[“列”] =(列表)值; }

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

关于c# - ApplicationSettingsBase 为自定义集合写入空标签,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6309026/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com