Используйте Reflection для создания класса (для создания динамического класса FileHelper) - PullRequest
6 голосов
/ 23 февраля 2012

Могу ли я построить класс, как показано ниже, динамически, используя отражение?Нет методов, только публичные переменные, некоторые имеют пользовательские атрибуты.

Требуется ли метод .Emit (из того, что я видел, "Emit" выглядит немного сложным).

Я использую программное обеспечение от www.FileHelpers.net , и для него требуется класс.Все мои определения файлов находятся в таблице базы данных, и я хотел бы сделать все более динамичным (т.е. никакой код не изменяется, когда в файле появляется новый столбец).

[FileHelpers.DelimitedRecord(",")]
public class FileRow
{
    [FileHelpers.FieldQuoted('"', QuoteMode.OptionalForBoth)] 
    public string Borrower_First_Name;
    [FileHelpers.FieldQuoted('"', QuoteMode.OptionalForBoth)] 
    public string Borrower_Last_Name;
    public string Borrower_Email;
}

Обновление 1: на основании приведенного ниже ответа Влада мне нужно было сослаться на DLL, вот как я это сделал:

    // need to reference the FileHelpers.dll from our own .exe directory 
    string diskFilenameFileHelpersDLL = 
        System.IO.Path.GetDirectoryName(
           System.Reflection.Assembly.GetExecutingAssembly().Location) + 
           @"\FileHelpers.dll";

Обновление 2: Кроме того, после выполнения того, что Влад предложилВот как я вызываю FileHelper и перебираю результаты.Я, вероятно, перенесу данные в список.

    Assembly assembly = compiledResult.CompiledAssembly;

    // Simple Data Test 
    lineContents = "John,Doe,jd123456@yahoo.com";
    FileHelperEngine engine = new FileHelperEngine(assembly.GetType("FileRow"));
    // FileRow[] FileRowArray = (FileRow[])engine.ReadString(lineContents);
    Object[] FileRowArray = engine.ReadString(lineContents);
    Object myObject = FileRowArray[0];  // only 1 row of data in this example 

    // Get the type handle of a specified class.
    Type myType = assembly.GetType("FileRow");
    // Get the fields of the specified class.
    FieldInfo[] myField = myType.GetFields();

    Console.WriteLine("\nDisplaying fields values:\n");
    for (int i = 0; i < myField.Length; i++)
    {
        Object objTest = myField.GetValue(i);

        string tempName = myField[i].Name;
        Object objTempValue = myField[i].GetValue(myObject);
        string tempValue = System.Convert.ToString(objTempValue);

        Console.WriteLine("The value of {0} is: {1}",
                            tempName, tempValue);

    }

Ответы [ 2 ]

2 голосов
/ 23 февраля 2012

Если ваш код хранится в базе данных в виде строки, вы можете сделать что-то подобное для создания сборки:

Причина, по которой я закомментировал атрибуты, потому что у меня нет пространства имён.Я предполагаю, что у вас есть пространство имен, и вам нужно будет добавить его в код для компиляции.

Код работает в LINQPad, поэтому вы можете просто скопировать и вставить.

using System;
using System.Reflection;
using System.CodeDom.Compiler;
using Microsoft.CSharp;

void Main()
{
    StringBuilder dc = new StringBuilder(512);
    dc.Append("public class FileRow");
    dc.Append("{");
    //dc.Append("[FileHelpers.FieldQuoted('\"', QuoteMode.OptionalForBoth)]");
    dc.Append("public string Borrower_First_Name;");
    //dc.Append("[FileHelpers.FieldQuoted('\"', QuoteMode.OptionalForBoth)]");
    dc.Append("public string Borrower_Last_Name;");
    dc.Append("public string Borrower_Email;");
    dc.Append("}");

    CompilerResults compiledResult = CompileScript(dc.ToString());

    if (compiledResult.Errors.HasErrors)
    {
        Console.WriteLine (compiledResult.Errors[0].ErrorText);
        throw new InvalidOperationException("Invalid Expression syntax");
    }

    Assembly assembly = compiledResult.CompiledAssembly;

    // This is just for testing purposes.
    FieldInfo field = assembly.GetType("FileRow").GetField("Borrower_First_Name");          
    Console.WriteLine (field.Name);         
    Console.WriteLine (field.FieldType);
}

public static CompilerResults CompileScript(string source) 
{ 
    CompilerParameters parms = new CompilerParameters(); 

    parms.GenerateExecutable = false; 
    parms.GenerateInMemory = true; 
    parms.IncludeDebugInformation = false; 

    CodeDomProvider compiler = CSharpCodeProvider.CreateProvider("CSharp"); 

    return compiler.CompileAssemblyFromSource(parms, source); 
} 
2 голосов
/ 23 февраля 2012

Вы также можете использовать CodeDOM для динамической генерации вашего класса.Для получения дополнительной информации посетите

http://msdn.microsoft.com/en-us/library/ms404245.aspx

...