UML软件开发与建模工具Enterprise Architect教程:创建项目特定的代码生成器(上)

如何轻松为Enterprise Architect创建项目特定的代码生成器。

您是否曾经想过从Enterprise Architect UML或SysML模型生成代码是否尝试过自定义Enterprise Architect的代码模板框架要放弃特定于项目的代码生成器的梦想,而要阅读它们实现起来的难易程度。

需要代码生成器

与实现相比,好的软件或系统架构具有更高的抽象级别。它应该是一个一致的模型,用于记录决策并忽略不必要的(通常是技术上的)细节。考虑一下图1中的类图。它显示了一个域模型,该模型定义了商店允许客户订购商品所需的数据结构。每个类的属性都进行了详细建模,但是省略了其他不必要的方面,例如访问属性的操作。

UML软件开发与建模工具Enterprise Architect教程:创建项目特定的代码生成器(上)

图1:示例UML模型

如果在开始实施之前先对软件体系结构/设计进行了准备,那么代码生成可以避免很多繁琐且容易出错的工作。商业开箱即用的代码生成器通常不会更改抽象程度。这就是为什么它们通常不符合项目需求的原因。

可以根据项目特定需求定制Enterprise Architect的代码模板框架。但这需要一些初步培训。通常,如基于Eclipse的Enterprise Architect Models的代码生成中所述,很难达到预期的结果。

一个简单的项目专用代码生成器

我更喜欢使用Java或Xtend之类的通用编程语言来实现代码生成器。由于Xtend具有模板表达,因此特别适合于实现模板。它们允许将可执行代码嵌入要生成的文本中。感觉就像在编程PHP,JSP或JSX。在代码清单1只显示了写的Xtend代码生成模板。它为图1的类图中声明的类生成Java类。

package com.yakindu.ea.examples.orderingsoftware.templateimport com.yakindu.bridges.ea.examples.runtime.codegen.EACodegenimport org.eclipse.uml2.uml.Classimport org.eclipse.uml2.uml.NamedElementclass ClassTemplate {   @EACodegen("Java")   def String generate(Class element) '''      package lement.package.javaQualifiedName       al superType = element.generals.findFirst[true]name     al extends = '''F !superType.isNullOrEmptyextends uperTypeENDIF''     publicF element.isAbstractabstractNDIFclass lement.nameextends{          OR attribute : element.ownedAttributes SEPARATOR System.lineSeparator           al type = attribute.typejavaQualifiedName           F 1 != attribute.upper              al defaultValue = '''new java.util.LinkedList<ypegt;()'''              private final java.util.List<ypegt; ttribute.name= efaultValue            LSE              private ypettribute.name            NDIF        NDFOR         OR attribute : element.ownedAttributes SEPARATOR System.lineSeparator           al type = attribute.typejavaQualifiedName           F 1 != attribute.upper              public List<ypegt; getttribute.name.toFirstUpper) {                  return ttribute.name               }            LSE              public ypegetttribute.name.toFirstUpper) {                  return ttribute.name               }            NDIF           F 1 == attribute.upper                     al params = '''ypettribute.name''              public void setttribute.name.toFirstUpperarams {                  this.ttribute.name= ttribute.name               }            NDIF        NDFOR     }   '''   protected def String getJavaQualifiedName(NamedElement element) {      element.qualifiedName.replace("::", ".")   }}

清单1:用Xtend编写的示例代码生成模板

在显示生成的Java代码清单2,  3  和  4看起来并不像手写的,因为合格的名称来代替进口。稍后将在图4中通过方法collectImports和进行改进printImports。

package com.example.orderingsoftware;public abstract class AbstractIDObject {   private java.util.UUID id;    public java.util.UUID getId() {      return id;   }   public void setId(java.util.UUID id) {      this.id = id;   }}

清单2:清单1中的代码生成模板生成的类AbstractIDObject的Java代码

package com.example.orderingsoftware;public class OrderItem extends AbstractIDObject {   private java.math.BigInteger amount;   private com.example.orderingsoftware.Article article;    public java.math.BigInteger getAmount() {      return amount;   }   public void setAmount(java.math.BigInteger amount) {      this.amount = amount;   }   public com.example.orderingsoftware.Article getArticle() {      return article;   }   public void setArticle(com.example.orderingsoftware.Article article) {      this.article = article;   }}

清单3:清单1的代码生成模板生成的OrderItem类的Java代码

package com.example.orderingsoftware;public class Order extends AbstractIDObject {   private java.util.Date date;   private com.example.orderingsoftware.Customer customer;   private final java.util.List<OrderItem> items = new java.util.LinkedList<OrderItem>();    public java.util.Date getDate() {      return date;   }   public void setDate(java.util.Date date) {      this.date = date;   }   public com.example.orderingsoftware.Customer getCustomer() {      return customer;   }   public void setCustomer(com.example.orderingsoftware.Customer customer) {      this.customer = customer;   }   public java.util.List<com.example.orderingsoftware.OrderItem> getItems() {      return items;   }}

清单4:由清单1中的代码生成模板生成的Order类的Java代码

如果仔细查看清单1中的模板,您将意识到它对Enterprise Architect一无所知。取而代之的是,它处理UML元模型的实例,这要归功于Eclipse UML 2项目,它在Eclipse中可用。YAKINDU EA-Bridge是Enterprise Architect和UML之间缺少的连接。它是一个API,可提供对Enterprise Architect UML和SysML模型的符合UML的读写权限。Enterprise Architect项目背后的数据库会自动转换为UML元模型的实例。作为开发人员,这具有三大优势: 

  • 您的代码与基于UML 2项目的其他工具(如Papyrus)兼容。
  • 对Enterprise Architect模型的高性能读写访问,而无需对Enterprise Architect的数据库架构进行反向工程。
  • 您无需了解有关YAKINDU EA-Bridge API的任何知识。作为开发人员,它完全是隐藏的,因为YAKINDU EA-Bridge会将自身集成到Eclipse Modeling Framework(EMF)的生态系统中。

YAKINDU EA-Bridge带有可选的Eclipse IDE集成,该集成允许一个人实现特定于项目的代码生成器。这些代码生成器通常是原型开发的,并且仅在特定上下文中执行。因此,至关重要的是,与手动编码相比,减少开发工作量。要实现特定于项目的代码生成器,您要做的就是将EAP文件放置在Eclipse项目中,并使用注释代码生成模板中的方法@EACodegen。带注释的方法应接受应为其生成代码的UML元素作为唯一参数,并返回生成的文本。如果您的Enterprise Architect模型由远程数据库(例如Microsoft SQL Server)托管,则可以使用快捷方式文件 而不是EAP文件。

例如,通过主菜单项“ Project,Clean …”自动或手动构建项目时,将为所有EAP文件中声明的所有UML类启动模板。当然,仅考虑模板项目中存储的EAP文件。生成的代码保存在类的限定名称指定的文件中。文件扩展名由@EACodegen注释的参数指定。Eclipse项目的结构如图2所示。

UML软件开发与建模工具Enterprise Architect教程:创建项目特定的代码生成器(上)

图2:Eclipse中的示例项目结构

请注意,YAKINDU EA-Bridge是一个API。它允许您以任何方式处理Enterprise Architect模型。实际上,最初的用例是全面的代码生成器,例如基于UML架构的Autosar RTE生成器。 

=====================================================

想要了解或购买Enterprise Architect正版版权,请咨询官方客服

UML软件开发与建模工具Enterprise Architect教程:创建项目特定的代码生成器(上)
标签:

声明:本站部分文章及图片源自用户投稿,如本站任何资料有侵权请您尽早请联系jinwei@zod.com.cn进行处理,非常感谢!

上一篇 2020年1月27日
下一篇 2020年1月27日

相关推荐

发表回复

登录后才能评论