13

我正在寻找一个可以获取实例的序列化程序,并将其序列化为一个包含 c# 代码的字符串,表示图形的内容。该类的功能类似于SerializeObjectJSON.NET 中的功能。

我知道只有一组非常狭窄的结构会起作用,但我感兴趣的结构非常简单,而且它们会起作用。

如果有人知道具有类似功能的 Visual Studio Visualizer,则可以加分。

编辑: 输出将在编译时用于不同的应用程序。我不需要在运行时反序列化输出(c# 代码),它会保存到文件中进行分析。

var foo = new Foo() { Number = 1, Bar = new Bar() { Str = "Bar"}};
string sourceCode = Magic.SerializeObject(foo);

输出:

Foo obj = new Foo();
obj.Number = 1;
obj.RefType = null; // infer this
obj.Bar = new Bar();
obj.Bar.Str = "Bar";
4

2 回答 2

6

是的,不...

最接近的解决方案称为CodeDOM,它是大多数 Visual Studio 设计人员和向导用来生成代码的方法。


在仔细查看您的评论后,我认为您应该通读 MSDN 中的二进制序列化部分,它将满足您的所有要求。

您正在寻找的是序列化,您不想每次都重新编译第二个项目,对吗?您只需要验证和向前\向后兼容性

于 2013-04-17T11:03:28.337 回答
3

使用 CodeDom 编写这样的内容实际上相对简单:

class CSharpSerializer
{
    private readonly Dictionary<string, int> m_locals =
        new Dictionary<string, int>();

    private readonly List<CodeStatement> m_statements =
        new List<CodeStatement>();

    private string GetVariableName(string suggestedName)
    {
        suggestedName = suggestedName.TrimEnd("0123456789".ToCharArray());

        int n;
        if (m_locals.TryGetValue(suggestedName, out n))
        {
            n++;
            m_locals[suggestedName] = n;
            return suggestedName + n;
        }

        m_locals[suggestedName] = 1;
        return suggestedName;
    }

    public string SerializeObject(object obj)
    {
        m_statements.Clear();

        // dynamic used to make the code simpler
        GetExpression((dynamic)obj);

        var compiler = new CSharpCodeProvider();
        var writer = new StringWriter();
        foreach (var statement in m_statements)
        {
            compiler.GenerateCodeFromStatement(
                statement, writer, new CodeGeneratorOptions());
        }
        return writer.ToString();
    }

    private static CodeExpression GetExpression(int i)
    {
        return new CodePrimitiveExpression(i);
    }

    private static CodeExpression GetExpression(string s)
    {
        return new CodePrimitiveExpression(s);
    }

    private static CodeExpression GetExpression(DateTime dateTime)
    {
        // TODO: handle culture and milliseconds
        return new CodeMethodInvokeExpression(
            new CodeTypeReferenceExpression(typeof(Convert)), "ToDateTime",
            new CodePrimitiveExpression(Convert.ToString(dateTime)));
    }

    // and so on for other primitive types
    // and types that require special handling (including arrays)

    private CodeExpression GetExpression(object obj)
    {
        if (obj == null)
            return new CodePrimitiveExpression(null);

        var type = obj.GetType();
        string typeName = type.Name;

        string variable = GetVariableName(
            typeName[0].ToString().ToLower() + typeName.Substring(1));

        m_statements.Add(
            new CodeVariableDeclarationStatement(
                typeName, variable, new CodeObjectCreateExpression(typeName)));

        foreach (var property in type.GetProperties(
            BindingFlags.Public | BindingFlags.Instance))
        {
            var expression = GetExpression((dynamic)property.GetValue(obj));
            m_statements.Add(
                new CodeAssignStatement(
                    new CodePropertyReferenceExpression(
                        new CodeVariableReferenceExpression(variable),
                        property.Name),
                    expression));
        }

        return new CodeVariableReferenceExpression(variable);
    }
}

有很多情况下这段代码会失败,但是对于行为良好的类型(公共默认构造函数、所有公共实例属性都有设置器、字段或非公共属性中没有隐藏重要状态、没有循环引用、没有命名空间冲突等)。 ),它会工作。

于 2013-04-17T18:46:42.163 回答