1

我将发布一个我在其他地方看到过变体的问题,但我还没有完全看到我想出的答案。我随后会发布我的答案。

为了使用宏模块化我的构建脚本,我想将 updatetask 和我想要执行的任务(如果它不是最新的)放在同一个宏中。
我该怎么做 - 使用脚本标签是可以的 - 如果传入的唯一唯一属性是包含反斜杠的路径,并且我需要能够在不同的调用中将不同的值传递给脚本标签。当涉及反斜杠时,我需要避免字符串文字可能发生的任何问题。使用 unique 属性,我需要解决 ant 不可变属性行为,并解决通常使用 2 个目标来处理最新处理决策的 ant 模式,并且需要解决字符串文字中反斜杠的 javascript 处理。

4

1 回答 1

2

注意我正在使用 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 脚本的主要“逻辑”之外,从而有助于自我记录和可读性你的主要“逻辑”。当事情不明显时使用注释。

于 2010-03-05T18:47:25.503 回答