我从各种客户端获得多个XSD,我需要向他们提供符合他们提供的XSD的XML格式的数据。我已经编写了一个代码,使用codedom,System.Reflection和codeprovider从XSD动态创建和编译类。 现在我的计划是通过多个查询从数据库中获取数据,并将字段映射到创建的动态类的属性并将其序列化。我正在寻找映射这些字段的通用方法,可以用于任何类型的xsd,只需映射它将序列化的字段并提供XML文件。至于我将它们放在配置文件中的查询。通用解决方案是否可行?关于如何去做的任何想法或指示?
答案 0 :(得分:1)
我能够解决这个问题。以下是我使用的步骤:我首先使用带有可序列化属性的反射和encodeom.compiler从xsd创建内存运行时程序集。使用反射我在该程序集中创建了类的实例,并从我从数据库获得的数据中分配了属性。这个类转发给另一个序列化的方法,它接受一个对象并将其序列化为xml。
就映射而言,我确保数据库列名称需要与xml属性名称匹配,并且我能够映射并调用它们。
以下是摘录:
object fxClass = myAssembly.CreateInstance(cls.FullName);
Type t = fxClass.GetType();
var arrRates = Array.CreateInstance(t, tab.Rows.Count);
int i =0;
foreach (DataRow dr in tab.Rows)
{
fxClass = myAssembly.CreateInstance(cls.FullName);
PropertyInfo[] fxRateProperties = t.GetProperties();
foreach (PropertyInfo prop in fxRateProperties)
{
string rowVal = dr[prop.Name].ToString();
if (prop.PropertyType == typeof(DateTime))
{
prop.SetValue(fxClass, util.convertToDate(rowVal), null);
}
else if (prop.PropertyType == typeof(bool))
{
prop.SetValue(fxClass, util.convertToBoolean(rowVal), null);
}
else if (prop.PropertyType == typeof(decimal))
{
prop.SetValue(fxClass, util.convertToDecimal(rowVal), null);
}
else prop.SetValue(fxClass, rowVal, null);
}
arrRates.SetValue(fxClass,i);
i++;
}
myClass.GetType().GetProperty("ForexRates").SetValue(myClass, arrRates, null);
然后将myClass对象传递给serialize接受对象类型的方法,就是它
public void serializeXML(object portfolio, string xmlPath)
{
XmlSerializer serial = new XmlSerializer(portfolio.GetType());
XmlSerializerNamespaces ns = new XmlSerializerNamespaces();
ns.Add("", "");
try
{
using (FileStream fs = new FileStream(xmlPath, FileMode.Create, FileAccess.Write))
{
using (XmlTextWriter tw = new XmlTextWriter(fs, Encoding.UTF8))
{
tw.Formatting = Formatting.Indented;
serial.Serialize(tw, portfolio, ns);
}
}
}
}
为此,现在我计划添加一个UI片段,其中像“ForexRates”这样的映射被保存在数据库中,然后它将打开到任何xsd类型。
非常感谢你的回复。
答案 1 :(得分:0)
您不能序列化即时创建的类 - 至少不是正常方式。即使您找到了附加Serializable
属性的方法,也不会产生任何影响:不会生成序列化程序集。
理论上,你可以在运行时发出一个全新的类并发送它,但这绝对是一种痛苦。如果你这样做,我建议在.NET Framework的默认IL发射器上使用Mono.Cecil。