使用Ant构建具有混合Scala和Java源文件的项目 - 非法循环引用错误

时间:2015-01-26 20:28:06

标签: java scala ant reference cyclic

目前我有一个使用Ant构建的现有Java项目,我正在尝试使用混合源(Java和Scala)进行构建。

我有一个定义为 - public interface One<T extends Two<? super T>>

的Java接口

当我从Ant构建运行scalac时,我收到以下错误:

错误:涉及类型T的非法循环引用。

由于我希望Scala和Java类在我的项目中是相互依赖的,所以我首先在Ant构建中运行javac之前运行scalac任务。我的scalac块看起来如下:

<scalac srcdir="${src.path}" destdir="${build.classes}" encoding="UTF-8" >
        <classpath refid="compile.classpath" />
        <include name="**/*.scala"/>
        <include name="**/*.java"/>
</scalac> 

有没有办法解决这个问题?我无法修改Java代码,因为有大量其他代码使用它。

我正在使用Java 7和Scala 2.11.5


这就是我在最后看到的内容:

com.arif.One.java

package com.arif.java;
public interface One<T extends One<? super T>> {
    void test1();
}

com.arif.Two.java

package com.arif.java;
public class Two implements One<Two> {
    @Override
    public void test1() {
    } 
}

com.arif.scala.Main.scala

package com.arif.scala
import com.arif.java.Two

object Main {
  def main(args:Array[String]) {
    var b:Two = null
  } 
}

的build.xml

<property name="scala.home" value="/Users/arif/software/scala/scala-2.11.5"/>
<property name="build.classes" value="./build/classes"/>

<property name="src.path" value="src"/>
<property name="javac.target" value="1.7"/>
<property name="compile.debug" value="true" />
<property name="compile.deprecation" value="false" />
<property name="compile.optimize" value="false" />

<path id="compile.classpath">
    <fileset dir="${scala.home}/lib" includes="*.jar" />
</path>

<path id="scala.classpath">
    <pathelement location="${scala.home}/lib/scala-compiler.jar" />
    <pathelement location="${scala.home}/lib/scala-library.jar" />
    <pathelement location="${scala.home}/lib/scala-reflect.jar" />
</path>

<target name="init">
    <taskdef resource="scala/tools/ant/antlib.xml">
        <classpath refid="scala.classpath" />
    </taskdef>
</target>

<target name="build" description="build" depends="init">
    <scalac srcdir="${src.path}" destdir="${build.classes}" encoding="UTF-8" >
        <classpath refid="compile.classpath" />
        <include name="**/*.scala"/>
        <include name="**/*.java"/>
    </scalac>
    <javac srcdir="${src.path}" destdir="${build.classes}" memoryinitialsize="512m" target="${javac.target}" source="${javac.target}" 
        memorymaximumsize="1024m" nowarn="true" debug="${compile.debug}" deprecation="${compile.deprecation}" 
        optimize="${compile.optimize}" fork="true" encoding="iso-8859-1" includeantruntime="false">
        <classpath refid="compile.classpath" />
    </javac>
</target>

从项目根目录中的命令行 - 执行&#34; ant build&#34;给出以下内容 -

[scalac] Compiling 1 scala and 2 java source files to /Users/arif/rbc/scala-java/build/classes
[scalac] /Users/arif/rbc/scala-java/src/com/arif/java/One.java:3: error: illegal cyclic reference involving type T
[scalac] public interface One<T extends One<? super T>> {
[scalac]                                    ^
[scalac] one error found

2 个答案:

答案 0 :(得分:2)

PRSI-4744提到可能有助于解决此问题的实验标记-Ybreak-cycles

答案 1 :(得分:1)

假设您在scala代码中执行类似的操作(因为您的java代码通过scalac编译得很好):

scala> class OneImpl[T <: Two[_ >: T]] extends One[T]
<console>:15: error: illegal cyclic reference involving type T
   class OneImpl[T <: Two[_ >: T]] extends One[T]

这实际上会导致您的Java代码(和Ant本身)独立出错。我建议将其更改为:

scala> class OneImpl[Z >: T, T <: Two[Z]] extends One[T] 
defined class OneImpl

无论如何,Scala不支持循环引用。你也可以解决它(如果你真的需要存在类型):

scala> class OneImpl[Z >: T, T <: Two[Z], K <: Two[_ >: Z]](implicit  ev: K =:= T) extends One[T]
defined class OneImpl

请参阅http://www.scala-lang.org/old/node/6358

<强>更新即可。 scalac SI-4744存在问题。它影响的不是所有版本,所以从一开始就找不到它。解决方案(解决方法)是仅由javac本身编译此接口:

javac One.java

或类似于ant的东西:

<target name="build" description="build" depends="init">
    <javac srcdir="${src.path}" destdir="${build.classes}" encoding="UTF-8">
        <classpath refid="compile.classpath" />
        <include name="**/One.java"/>
    </javac> 
    <scalac srcdir="${src.path}" destdir="${build.classes}" encoding="UTF-8" >
        <classpath refid="compile.classpath" />
        <include name="**/*.scala"/>
        <include name="**/*.java"/>
        <exclude name="**/One.java"/>
    </scalac>
    <javac>...</javac>
</target>