创建实例并分配运行时程序集的属性

时间:2012-06-01 18:17:04

标签: c# system.reflection

我需要创建一个创建从xsd生成的类实例的通用解决方案。例如:

<?xml version="1.0" encoding="utf-8"?>
<xs:schema attributeFormDefault="unqualified" elementFormDefault="qualified" xmlns:xs="http://www.w3.org/2001/XMLSchema">
  <xs:element name="PS">
    <xs:complexType>
      <xs:sequence>
        <xs:element name="P" maxOccurs="unbounded">
          <xs:complexType>
            <xs:sequence>
              <xs:element name="FXs" maxOccurs="1" minOccurs="0">
                <xs:complexType>
                  <xs:sequence>
                    <xs:element maxOccurs="unbounded" name="FX" minOccurs="1">
                      <xs:complexType>
                        <xs:attribute name="asOfDate" type="xs:date" use="required" />
                        <xs:attribute name="currency" type="xs:string" use="required" />
                        <xs:attribute name="rate" type="xs:decimal" use="required" />
                      </xs:complexType>
                    </xs:element>
                  </xs:sequence>
                </xs:complexType>
              </xs:element>
 </xs:sequence>
            <xs:attribute name="PName" type="xs:string" />
            <xs:attribute name="currency" type="xs:string" />
          </xs:complexType>
        </xs:element>
      </xs:sequence>
    </xs:complexType>
  </xs:element>
</xs:schema>

当我从这个xsd运行时生成程序集时,它将具有类PS,PSP和PSFX。在这种情况下,我知道属性和期望的内容,因此createinstance和getproperties以及setvalue将起作用。我知道投资组合将有一系列投资组合资产组合。

case "PS":
                                    if (p== null)
                                        p= myAssembly.CreateInstance(cls.FullName);
                                    break;

                                case "PSP":
                                    myClass = myAssembly.CreateInstance(cls.FullName);
                                    myClass.GetType().GetProperty("PName").SetValue(myClass, dt.Rows[0]["PortfolioName"].ToString(), null);
                                    myClass.GetType().GetProperty("currency").SetValue(myClass, dt.Rows[0]["Currency"].ToString(), null);
                                    if (myClasses == null)
                                        myClasses = Array.CreateInstance(myClass.GetType(), deals.Count());
                                    myClasses.SetValue(myClass, l);
                                    l++;
                                    portfolio.GetType().GetProperty("P").SetValue(p, myClasses, null);

现在我想要一个通用的解决方案,它可以接受任何xsd(不仅仅是上面的)并生成一个程序集。我已经生成了汇编但是现在我的问题是如何为那些不知道那里存在哪些属性的类动态分配属性?我的意思是如何在创建通用解决方案时知道某个类正在使用另一个类的对象列表。因此,不是硬编码类名和属性名,而是有一种获取实例和属性的方法。正如您从上面的代码片段中看到的那样,我将数据集中的值分配给属性。计划是将值存储在数据库中作为columnname,attributename和parentnode。 因此P节点中的PName属性将来自列名Pname。我希望设计的方式是我的代码中不会有任何硬编码。能否请您提供一些示例或设计理念。

抱歉,请耐心等待,如果我没有意义,请告诉我。

2 个答案:

答案 0 :(得分:1)

长期问题值得长期回答:

我。首先,一个小凭证:我一直在使用类/程序集生成一段时间。实际上,您可以使用CodeDom(首选方法)或IL Emit执行此操作。但你可能不应该这样做。

  • 反对此的第一个论点是您的XML模型实际上并不那么有用。好的,所以你有一个想要转向类的XML。为此,您将以某种方式将XML转换为代码(C#或VB甚至IL),然后在运行时编译它以创建动态类。那么,为什么不直接使用代码而不是XML呢?
  • 上述要点有一个推论:如果你在设计时指定类,为什么不编译类(因此,在编译时生成它)?

II。您的设计驱动程序无硬代码

  • 首先进行哲学分析:你不想硬编码代码,这是一个悖论。
  • 最好尽量减少源代码中的硬编码字符串,主要是因为(1)预期这些字符串会发生变化(2)它可能会使测试变得更容易。我们可以从这个定义中提取两件事:
    • 您应该避免使用硬编码字符串,而不是程序结构
    • 您应该只编译那些预期会在编译时发生变化(“动态”)或未知的东西。数据库结构几乎不是这种情况。

现在,抽象地思考,你能避免硬编码吗?在您的示例中,您已经使用了一些字符串来引用您要处理的XML文件中的数据。如果没有关于您要处理的数据的任何信息,程序将如何工作?

(是的,有答案)

...

答案:当您只是展示数据时,即完整的数据结构与程序逻辑无关。这方面的例子是文本编辑器。文本编辑器不需要事先知道数据。您将其呈现为文件中存在的内容。

但是,在您的情况下,数据库是数据的“主”。尝试将其与您的程序分开是没有用的。

答案 1 :(得分:0)

解析XML而不是创建程序集会不会更容易?您可以使用XmlSchema及其子对象来提取数据库中所需的所有信息。您可以使用通用框架来处理XML文档,以便操作XML输出。或者,如果您确实需要程序集,则可以先更改XML,然后将其反序列化为对象。

但是,如果你的客户端代码不应该在编译时知道你的程序集的CLR类型,我认为创建程序集首先没有多大意义,你最好创建一个通用中间数据格式,您可以根据需要转换为XML或数据库列。