位置: 东方标准 >> 技术专题 >> 软件专区 >> 正文

用C#+XMI技术进行UML模型捕获

[ 作者:东方标准     来源:东方标准     文章录入:东方标准 ]

有许多不同的方法可用于捕获xml模型数据并且把它放到一个数据存储中,正如你所期望的,包括使用xslt技术。但是我想使用一种不同的方法-使用c#语言。xslt是一个用于改变xml文件的好选择,但是对于更广阔的不仅仅是转变数据的应用软件来说,c#或者另外的象java这样的高级语言提供了更大的灵活性。

  在本文中,我将展示如何通过使用xmi和c#来剖析一个uml发布图。首先,我展示一个该方法的简单的发布图,所在环境为一个虚构的汽车出租公司并使用c#来捕获一些数据。这种数据可以被容易地加到一个数据库(它已经是ado调用的相当容易的一部分)上去或者作为更大些的一个基于资产的管理系统的一部分。在每一个示例中,我将逐渐增加原始图和能够获取的信息的复杂性。

  一、 示例1-查找方法的名字

  我要做的第一件事情是捕获来自于汽车出租公司的所有方法的名字。共有五个方法,它们是以uml发布图形描述的(见图1)。在上一篇文章中,我展示了如何用一个xml文档来描述一个发布图。为此,首先要创建该图,然后把它输出到一个xmi文件(企业架构)或解压(magicdraw)。(在这篇文章中,我再次使用了sparxsystems.com提供的企业架构。)


图1.五个方法:图中结点描述了汽车出租公司的方法。每一个结点为一个方法名字。

  我将首先开发一简单main程序-它提示用户输入描述一个有效的xmi文档的xml文档的名字。一旦输入文件名字,它就被传递到一个分析xmi文档的对象并打印输出方法名字。

using system;
using system.text;
namespace xmi_1{
public class consoleutils{
 public static string readstring(string msg) {
  console.write(msg); return system.console.readline();
 }
 public static void writestring(string msg)
 { system.console.writeline(msg); }
 public static void main(string[] args) {
  string name = readstring("please enter the xmi filename : ");
  nodeparse np = new nodeparse(name);
  system.console.readline();
 }
}
}

  下一步是分析xmi文档。为此,我必须创建一个xmltextreader的实例并循环操作直到没有结点为止。xmlnodetype.element检查每一个xml元素并且在元素是一个结点的情况下把它打印输出到控制台。该元素名字资格为"uml:node",然而通过使用_readxmi.localname只有元素名字的"node"被读取。详细代码参见列表1。

  二、 示例2-增加模板

  第二个示例重构了第一个示例来捕获方法的名字和"pc server"模板-这是一个用来决定每个结点是一个方法的uml方法。图3显示增加了模板后的方法的发布图。

  第一步是修改前面的代码以识别新的模板来操纵addnode()方法。

图3.增加了模板之后:每个模板描述了结点的类型-不管它是一个pc服务器,pc客户端或者是另外一些描述该结点的目标。

private void addnode(xmltextreader p_readxmi)
{
 string nodeid;
 string nodename;
 string stereotypename;
 nodeid = p_readxmi.getattribute("name");
 nodename = p_readxmi.getattribute("xmi.id");
 console.write(nodeid);
 console.write(" -> ");
 console.writeline(nodename);
 while (p_readxmi.read() && (p_readxmi.nodetype == xmlnodetype.element ||
p_readxmi.nodetype== xmlnodetype.whitespace))
 {
  switch (p_readxmi.localname)
  {
   case "modelelement.stereotype":
    while (p_readxmi.read() && (p_readxmi.nodetype == xmlnodetype.element ||
p_readxmi.nodetype == xmlnodetype.whitespace))
    {
     if (p_readxmi.localname == "stereotype")
     {
      stereotypename = p_readxmi.getattribute("name");
      console.writeline("stereotype = " + stereotypename);
     }
    }
   break;
  }
 }
}

  特别要注意的是一个idiom(对某个语言特有的一个低级别的模式)-我用来查找在另外一些元素内部的xml元素。我在p_readxmi.read循环中增加了代码来检查元素或者空格(在xml文档中,这是些空格字符)。只有每个元素结点在xml文档中适当的层次上处理时,这个idiom才工作。

  用这种方式对元素进行封装,结果不很理想。例如,在下面的uml中使用idiom代码将产生所不希望的结果:

<uml:node name="leasing">
<uml:tagname name="test">
</uml:node name="smallleasing">
</uml:tagname>
</uml:node>


图4.示例2的输出结果:在示例1中的结点名字/id之后,列出每个模板。

  第一个元素被作为一个结点处理,第二个元素被忽略并在初始循环中退出,而第三个元素被作为在同一级上的第一个正常结点处理。为了修正这个问题,你可以使用xmlnodetype.endelement来检查元素的结束标签。在本文中,这不是个问题而保存额外的代码要求检查结尾元素。
xmi中的模板被封装在结点元素中。该示例中的代码检查名字为"modelelement.stereotype"的元素并且使用刚才讨论的相同的idiom来处理一个"模板"类型的封装元素,然后把它打印输出到屏幕上(见图4)。(列表3列出了示例2的完全的源码。)

 

(只显示最新5条。评论内容只代表网友观点,与本站立场无关!) 发表评论