【问题标题】:How to write a T4 template to create DTO's from Entityframework 6?如何编写 T4 模板以从 Entity Framework 6 创建 DTO?
【发布时间】:2016-02-17 21:41:00
【问题描述】:

我有一个大型数据库,我在 Entityframework 中使用数据库优先模型。它位于 Internet 服务器上并通过 WCF 进行通信。域模型使用所有小写字母作为实体、存储过程和列/属性的名称。

在我的客户端应用程序中,我希望使用标准 PascalCase 作为命名约定。

T4 模板能否使用正确的命名约定从 Entityframework 创建数据传输对象?

如果是这样,有人可以给我一个关于如何编写它的起点吗?

为了清楚起见,我不想更改 Entityframework 生成的任何代码,而是使用 Entityframework 模型添加具有适当 CamelCase 命名的简单 POCO 类作为另一个文件的输入,然后可以由WCF 服务,也许是 Automapper(或类似的东西)。

感谢您的任何建议。

  • 数据库:PostgreSQL 9.5
  • 数据库接口:Npgsql 3.0.5
  • .NET 4.5
  • 实体框架 6.0

【问题讨论】:

    标签: c# entity-framework postgresql wcf t4


    【解决方案1】:

    代替任何人回答这个问题并希望帮助像我这样的任何其他新手,这是我创建 T4 转换 DTOclasses.tt 所做的,它只产生简单的类定义。

    注意:这不是替换 .edmx 的 .tt 文件,而是在 .edmx 模板生成 .edmx 文件之后运行。 (我不想更改任何用于生成域模型的代码)。

    Visual Studio 2015 实体框架 6.0 .NET 框架 4.6.1

    ♦ Notes on Creating DTOclassess.tt
    
            This T4 transform was created by first copying the working transform used to build the entity model, MedicalOfficeModel.tt.
            Then, parts of it that were not needed for creation of POCO classes to be used for DTO's (data transfer objects) were removed.
    
        ♦ Changes made to DTOclassses.tt
    
            •   Adding "DTO" to namespace.
                    public void BeginNamespace(CodeGenerationTools code)
                    {
                        var codeNamespace = String.Format("{0}.{1}",code.VsNamespaceSuggestion(), "DTO");
                        if (!String.IsNullOrEmpty(codeNamespace))
                        {
                    #>
                    namespace <#=code.EscapeNamespace(codeNamespace)#>
                    {
                    <#+
                            PushIndent("    ");
                        }
                    }
    
            •  Put all POCO classes in single file DTOclasses.cs
    
                    <#
                    EndNamespace(code);
                }
    
                fileManager.Process(false);             <--**False stops the splitting of classes into different files. Default is true.
    
                #>
    
            •  Change the property naming code:
    
                        public string Property(EdmProperty edmProperty)
                        {
                            return string.Format(
                                CultureInfo.InvariantCulture,
                                "{0} {1} {2} {{ {3}get; {4}set; }}",
                                Accessibility.ForProperty(edmProperty),
                                _typeMapper.GetTypeName(edmProperty.TypeUsage),
                                GetPascalCase(_code.Escape(edmProperty)),
                                _code.SpaceAfter(Accessibility.ForGetter(edmProperty)),
                                _code.SpaceAfter(Accessibility.ForSetter(edmProperty)));
                        }
    
            •  Change the class naming code:
    
                        public string EntityClassOpening(EntityType entity)
                        {
                            return string.Format(
                                CultureInfo.InvariantCulture,
                                "{0} {1}partial class {2}{3}",
                                Accessibility.ForType(entity),
                                _code.SpaceAfter(_code.AbstractOption(entity)),
                                GetPascalCase(_code.Escape(entity)),
                                _code.StringBefore(" : ", _typeMapper.GetTypeName(entity.BaseType)));
                        }
    
            •  Removed all the navigational stuff. Replaced everything above the helper functions (i.e., above <#+) with:
                                <#@ template debug="false" hostspecific="true" language="C#" #>
                                <#@ assembly name="System.Core" #>
                                <#@ import namespace="System.Linq" #>
                                <#@ import namespace="System.Text" #>
                                <#@ import namespace="System.Collections.Generic" #>
                                <#@ import namespace="System.Text.RegularExpressions" #>
                                <#@ include file="EF6.Utility.CS.ttinclude" #>
                                <#@ output extension=".cs" #>
                                <#
    
                                const string inputFile = @"MedicalOfficeModel.edmx";
                                var textTransform = DynamicTextTransformation.Create(this);
                                var code = new CodeGenerationTools(this);
                                var ef = new MetadataTools(this);
                                var typeMapper = new TypeMapper(code, ef, textTransform.Errors);
                                var fileManager = EntityFrameworkTemplateFileManager.Create(this);
                                var itemCollection = new EdmMetadataLoader(textTransform.Host, textTransform.Errors).CreateEdmItemCollection(inputFile);
                                var codeStringGenerator = new CodeStringGenerator(code, typeMapper, ef);
    
                                if (!typeMapper.VerifyCaseInsensitiveTypeUniqueness(typeMapper.GetAllGlobalItems(itemCollection), inputFile))
                                {
                                    return string.Empty;
                                }
    
                                WriteHeader(codeStringGenerator, fileManager);
    
                                foreach (var entity in typeMapper.GetItemsToGenerate<EntityType>(itemCollection))
                                {
                                    fileManager.StartNewFile(entity.Name + ".cs");
                                    BeginNamespace(code);
                                #>
                                <#=codeStringGenerator.UsingDirectives(inHeader: false)#>
                                <#=codeStringGenerator.EntityClassOpening(entity)#>
                                {
                                <#
                                    var simpleProperties = typeMapper.GetSimpleProperties(entity);
                                    if (simpleProperties.Any())
                                    {
                                        foreach (var edmProperty in simpleProperties)
                                        {
                                #>
                                    <#=codeStringGenerator.Property(edmProperty)#>
                                <#
                                        }
                                    }
                                #>
                                }
                                <#
                                    EndNamespace(code);
                                }
                                fileManager.Process(false);
    
                                #>
    
            ♦  Added my helper function:
                        <#+
    
                            public static string GetPascalCase(string name)
                            {
                                return Regex.Replace(name, @"^\w|_\w",
                                    (match) => match.Value.Replace("_", "").ToUpper());
                            }
                        #>
    

    当一切都完成后,它可以完美地运行(在 VS2015 中),完全可以满足我的需要。 :)

    【讨论】:

    • 嗨,这会为每个 DTO 创建一个单独的部分类吗?我的实体来自 Sql Server,这里会有差异或任何变化。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多