将字符串索引器添加到 XmlSerializer

Adding a String Indexer to XmlSerializer

我有一个无法更改的 XML 来源,我想使用 XmlSerializer 对其进行反序列化。

我可以做到这一点,但是有一些自定义数组 class 我想通过字符串而不是整数来访问数组。

我知道我可以使用

public ClassName this[string index] 

但我不知道在哪里可以将它添加到我的 Class。

我希望能够打电话给

Object.Transaction["TransactionTypeName"] 

而不是

Object.Transaction[0]

这是 class 的精简版。

public partial class Configuration
{
    private ConfigurationTransaction[] transactionsField;

    [System.Xml.Serialization.XmlArrayItemAttribute("Transaction", IsNullable = false)]
    public List<ConfigurationTransaction> Transactions
    {
        get
        {
            return this.transactionsField;
        }
        set
        {
            this.transactionsField = value;
        }
    }
}

public partial class ConfigurationTransaction
{
    private string typeField;

    [System.Xml.Serialization.XmlAttributeAttribute()]
    public string type
    {
        get
        {
            return this.typeField;
        }
        set
        {
            this.typeField = value;
        }
    }

}

带索引器的 classes 的简化示例代码。 虽然我们需要保存值以访问索引器,但必须创建额外的 class.

public class Configuration
{
    public ConfigurationTransaction this[string transactionName]
    {
        get
        {
            return Transactions.First(tran => tran.TransactionName == transactionName).ConfigurationTransaction;
        }
        set
        {
            int index = Transactions.FindIndex(tran => tran.TransactionName == transactionName);
            if (index >= 0)
                Transactions[index] = new PairHelper { TransactionName = transactionName, ConfigurationTransaction = value };
            else
                Transactions.Add(new PairHelper { TransactionName = transactionName, ConfigurationTransaction = value });
        }
    }
    [EditorBrowsable(EditorBrowsableState.Never)]
    public List<PairHelper> Transactions { get; set; }
}

public class ConfigurationTransaction
{
    [XmlAttribute()]
    public string Type { get; set; }
}

public class PairHelper
{
    public string TransactionName { get; set; }
    public ConfigurationTransaction ConfigurationTransaction { get; set; }
}

成功了:

Configuration conf = new Configuration();
conf.Transactions = new List<PairHelper>();
conf["fooTran"] = new ConfigurationTransaction { Type = "foo" };
conf["barTran"] = new ConfigurationTransaction { Type = "bar" };

var xs = new XmlSerializer(typeof(Configuration));
using (var fs = new FileStream("test.txt", FileMode.Create))
{
    xs.Serialize(fs, conf);
}

Configuration conf2;
using (var fs = new FileStream("test.txt", FileMode.Open))
{
    conf2 = (Configuration)xs.Deserialize(fs);
}

foreach (var tran in conf2.Transactions)
    Console.WriteLine(tran.TransactionName + " : " + tran.ConfigurationTransaction);
Console.WriteLine(conf2["fooTran"].Type);
Console.WriteLine(conf2["barTran"].Type);

XML 将如下所示:

<?xml version="1.0"?>
<Configuration xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Transactions>
    <PairHelper>
      <TransactionName>fooTran</TransactionName>
      <ConfigurationTransaction Type="foo" />
    </PairHelper>
    <PairHelper>
      <TransactionName>barTran</TransactionName>
      <ConfigurationTransaction Type="bar" />
    </PairHelper>
  </Transactions>
</Configuration>

我能够将其添加到配置中 class。

public ConfigurationTransaction this[String index]
{
    get
    {
        foreach (var item in Transactions)
        {
            if (item.type.ToLower().Trim() == index.ToLower().Trim())
            {
                return item;
            }
        }
        return null;
    }
}    

并称它为

Object["TransactionTypeName"] 

我想这就是我所做的,我最终手动将所有数据导入 SQL 数据库并创建一个 Entity Framework class 来访问它并 depricated XML解决方案。