我有一个类,该类具有几个包含OCL表达式的字符串属性。我需要在OCL中查询这些对象,在该OCL中,我需要在其中一个属性中评估OCL表达式。
可以做到吗?
我想我正在寻找类似<<object>>.eval(<<string>>)
的OCL操作,其中<<string>>
将在<<object>>
的上下文中求值,最好在周围的OCL中使用任何OCL变量继承,因此<<string>>
中的OCL表达式可以引用它们。
作为一种解决方法,我想我可以添加一个代码派生的额外属性,该属性在对象的上下文中评估属性的OCL(可能就足够了)。
答案 0 :(得分:2)
您可以编写自定义OCL操作。这里是一个例子:
[EcoOclOperation(typeof(OclEcoId), true)]
public class OclEcoId : OclOperationBase
{
/// <summary>
/// Initialisation of the OCL funktion
/// </summary>
protected override void Init()
{
var parameter = new IOclType[1]
{
this.Support.ObjectType,
};
this.InternalInit("EcoId", parameter, this.Support.IntegerType);
}
/// <summary>
/// Implementation of the OCL function
/// </summary>
/// <param name="parameters">OCL parameters</param>
public override void Evaluate(IOclOperationParameters parameters)
{
var iObject = parameters.Values[0].Element as IObject;
if (iObject != null && iObject.AsObject != null)
{
var ecoid = iObject.EcoID();
this.Support.MakeNewInteger(parameters.Result, int.Parse(ecoid));
}
}
}
此类操作必须像这样安装到EcoSpace中:
public static class OclInstaller
{
/// <summary>
/// InstallOclOperations
/// </summary>
/// <param name="ecoSpace">The EcoSpaceparam>
public static void InstallOclOperations(EcoSpace ecoSpace)
{
if (ecoSpace != null)
{
var ocls = ecoSpace.GetEcoService<IOclService>();
var eals = ecoSpace.GetEcoService<IActionLanguageService>();
// Install all available operations in current assembly
foreach (Type t in System.Reflection.Assembly.GetAssembly(typeof(OclInstaller)).GetTypes())
{
if (t.BaseType == typeof(OclOperationBase))
{
var op = (OclOperationBase)Activator.CreateInstance(t);
eals.InstallOperation(op);
object[] attributes = t.GetCustomAttributes(typeof(EcoOclOperationAttribute), false);
if (attributes.Length == 1)
{
if ((attributes[0] as EcoOclOperationAttribute).IsQuery)
{
ocls.InstallOperation(op);
}
}
else
{
ocls.InstallOperation(op);
}
}
}
}
else
{
throw new NullReferenceException("Error in [InstallOclOperations]: EcoSpace is null.");
}
}
}
在ecospace的构造函数中调用此安装程序:
OclInstaller.InstallOclOperations(this);
使用MDriven添加新的自定义OCL操作如此简单,真是令人惊讶。正如Hans所说,不可能开箱即用地评估OCL,但是您可以编写自己的评估器。但是请退后一步,就实现功能的方式做出适当的架构决定。
您还可以添加自定义EcoService,但这是一个完全不同的故事。
答案 1 :(得分:1)
无法从ocl / eal本身内部评估ocl / eal。有人建议这样做,但由于安全隐患和自我修改的代码确实很奇怪,因此被否决了。
MDriven for Visual Studio中的简单变通方法是在您的超类上添加一个Eval方法-并在c#中实现它-并具有调用ocl或eal服务的功能。
如果您要对结果采取行动,我想您可以让Eval方法返回Collection(SuperClass)或SuperClass