注意我正在使用 ant 1.7。借助 Ant 1.8 中的本地作用域,有一些额外的选项,因此不变性不是一个很大的挑战,但其中一些其他技巧仍然会有所帮助。
首先,关于根据宏中的更新结果选择性地执行任务的问题 - 这意味着您不需要 2 个目标。为此,请使用条件标签。如果第一个条件失败,<or> 标记将使其仅执行第二个条件。<scriptcondition> 标签允许使用 javascript 执行其他 ant 任务。这是一个示例(@ 标记表示宏定义属性):
<condition property="whatever" value="false">
<or>
<uptodate>
<srcfiles dir="@{srcdir}" includes="@{srcincludes}" excludes="@{srcexcludes}"/>
<mapper><chainedmapper>
<flattenmapper/><!-- use any mappers you need to match source to target files-->
<globmapper from="*.jxw" to="@{targetdir}\*W.java"/>
</chainedmapper></mapper>
</uptodate>
<!-- w/ java 1.6 or later, you get a rhino javascript interpreter included w/ java-->
<scriptcondition language="javascript" value="true">
self.setValue(true);
echo = project.createTask("echo");
myArg1="@{myArg1}";
myArg2="@{myArg2}";
// need to create a reference from a classpath refid
myReference = new org.apache.tools.ant.types.Reference(project,"@{my.classpath.id.string}");
// get a handle to the ant java task, which we will use to execute a java program
javaTask = project.createTask("java");
javaTask.setFork(true);
javaTask.setFailonerror(true);
javaTask.setClassname("com.mycompany.mypackage.MySpecialClass");
javaTask.setClasspathRef(myReference);
javaTask.createArg().setValue(myArg1);
javaTask.createArg().setValue(myArg2);
//output the command line to standard out, for reference
echo.setMessage(javaTask.getCommandLine());
echo.perform();
javaTask.perform();
</scriptcondition>
</or>
</condition>
现在,如果您像我一样,您可能希望对作为宏定义输入的属性进行一些处理,并生成一些可以在宏定义脚本中引用的派生值。如果您正在处理仅涉及属性和字符串的连接,则可以在块中执行此操作,如果您指定属性,则指定第二组属性,其中包含连接步骤的默认设置。但是,如果您需要做一些无法插入到属性默认值的事情,则需要将其放入属性中。因为属性是不可变的,所以您需要采取一些额外的步骤来为您的属性提供唯一的名称。tstamp 可以帮助解决这个问题。通常,传递给宏的参数的某些组合将是唯一的,但如果这种唯一组合包括反斜杠,您需要使用 tstamp 标记派生辅助唯一标识符,以便在您想要使用这些派生属性时不会在 javascript 中遇到反斜杠问题。以下是创建您可以在脚本中轻松引用的独特属性的方法。
<macrodef name="public.macro.example">
<attribute name="srcpath"/>
<sequential>
<tstamp prefix="@{srcpath}"><format pattern="ddhhmmssSSS" property="time"/></tstamp>
<private.macro.example srcpath="@{srcpath}" propertyPrefix="prop${@{srcpath}.time}"/>
</sequential>
</macrodef>
<macrodef name="private.macro.example">
<attribute name="srcpath"/>
<attribute name="prefix"/>
<sequential>
<pathconvert property="@{prefix}.src"/>
<!-- now you can do special things with ${@{prefix}.src}, even in javascript -->
<script language="javascript">
self.setValue(true);
echo = project.createTask("echo");
myPrefix="@{prefix}";
mySpecialPropertyKey=myPrefix+".src";
//if your special property contains backslashes or other special js characters
// you need to use project.getProperty instead of a string literal to get the value
mySpecialPropertyVal=project.getProperty(mySpecialPropertyKey);
// do something with this derived value in javascript
echo.setMessage("my special property = "+mySpecialPropertyVal);
echo.perform();
</script>
</sequential>
</macrodef>
在提出上述解决方案之前,我提出了一种 hack 风格的解决方案,用于用新值覆盖 ant 属性。虽然您可能会发现这对覆盖属性很有用,因为我直接调用了一个 ant 类,但在未来的 ant 版本中,这可能不会同样有效。因为这看起来更像是一种 hack,所以我的意图是尽可能使用上面列出的 2 macrodef 方法而不是这种方法。请注意,此特定变体不支持反斜杠字符,因为属性在 javascript 字符串文字中直接引用。我最初使用这个简单的变体来创建我的唯一前缀,这避免了使用 2 宏定义方法来解决属性不变性的需要。
<macrodef name="public.canova.setproperty">
<attribute name="name"/>
<attribute name="value"/>
<sequential>
<script language="javascript">
project.setUserProperty("@{name}","@{value}");
</script>
<sequential>
</macrodef>
乍一看,其中一些可能看起来有点复杂,但如果你的 macrodef 工作正确并构建组件风格的宏(即不要在你的宏中放入意大利面条代码),你的 ant 脚本实际上应该变得更短,更容易理解并且更易于维护,并且日志将更易于跟踪。提示 - 仅在必要时使用 javascript,当您使用它时,最好在宏中使用它,以便将其封装并隐藏在 ant 脚本的主要“逻辑”之外,从而有助于自我记录和可读性你的主要“逻辑”。当事情不明显时使用注释。