4

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

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

当我从 Ant build 运行 scalac 时,出现以下错误:

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

因为我希望 Scala 和 Java 类在我的项目中相互依赖,所以我首先运行 scalac 任务,然后在 Ant 构建中运行 javac。我的 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
  } 
}

构建.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>

从项目根目录中的命令行 - 执行“ant build”给出以下 -

[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
4

2 回答 2

2

SI-4744PR提到了一个可能有助于解决此问题的实验标志。-Ybreak-cycles

于 2015-01-28T00:50:50.997 回答
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

或者类似蚂蚁的东西:

<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>
于 2015-01-27T01:20:03.087 回答