Maven编译项目显示 程序包com.sun.*包不存在 【原因及三种解决方案】

本文解析了Maven编译时出现的特定Sun包缺失错误,深入探讨了问题的根源在于Javac默认不链接rt.jar,而是使用特殊的symboltable。提供三种解决方案,包括修改bootclasspath、忽略symbol.file及调整maven-compiler-plugin配置。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

maven 编译时报错:

[ERROR] Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:
程序包com.sun.…… 不存在
程序包com.sun.xml.internal.ws.spi不存在

【官方解释】:javac uses a special symbol table that does not include all Sun-proprietary classes. When javac is compiling code it doesn't link against rt.jar by default. Instead it uses special symbol file lib/ct.sym with class stubs.大意是:javac在编译时,并不引用 rt.jar,用的是一个特别的symbol table(lib/ct.sym),这个symbol table不包含所有的sun包的类。

【具体原因】:J2SE中的类大致可以划分为以下的各个包:java.*,javax.*,org.*,sun.*;除了“sun”包,其它各个包都是Java平台的标准实现,并且今后也将被继续支持。一般说来,“sun”之类的包并不包含在Java平台的标准中,它与操作系统相关,在不同的操作系统(如Solaris,Windows,Linux,Mac等等)中的实现也各不相同,并且可能随着J2SE版本不定期变化。因此,直接调用“sun”包的程序代码并不是100%的Java实现。也就是说:“sun.*”包并不是API公开接口的一部分,调用“sun”包的程序并不能确保工作在所有Java平台上,事实上,这样的程序并不能工作在今后的Java平台上。

【解决方案一】:

<compilerArguments>  
    <verbose />  
    <bootclasspath>${JAVA_HOME}/jre/lib/rt.jar</bootclasspath>  
</compilerArguments>  

注意:${JAVA_HOME} 指的是你配置项目依赖的java路径(jdk版本)
1、如果使用的类,接口等在其他的jar里面(如tools.jar),则bootclasspath值需要配置成其他的jar
2、pom文件如果没有配置${JAVA_HOME} 导致会报错,也就是说这种方案行不通。

【解决方案二】(亲测有效):

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <version>3.1</version>
    <configuration>
        <source>1.7</source>
        <target>1.7</target>
        <encoding>UTF-8</encoding>
        <compilerArgs>
            <arg>-XDignore.symbol.file</arg>
        </compilerArgs>
        <fork>true</fork>
    </configuration>
</plugin>

切记:不要漏掉标签项 <fork>true</fork>

【解决方案三】:

<plugin>
  <groupId>org.apache.maven.plugins</groupId>
  <artifactId>maven-compiler-plugin</artifactId>
  <version>3.1</version>
  <configuration>
      <source>${java.version}</source>
      <target>${java.version}</target>
      <encoding>UTF-8</encoding>
      <compilerArguments>
          <bootclasspath>${java.home}/lib/rt.jar</bootclasspath>
      </compilerArguments>
  </configuration>
</plugin>

如果还是不行 调整配置如下:

<plugin>
    <groupId>org.springframework.boot</groupId>
    <groupId>***(其他)***</groupId>
    <artifactId>spring-boot-maven-plugin</artifactId>
</plugin>

将maven-compiler-plugin配置在后面,插件会覆盖springboot、其他集成的插件

 

 

### com.sun.tools.javac.code.Symbol$PackageSymbol.members() 方法详解 `com.sun.tools.javac.code.Symbol.PackageSymbol.members()` 是 Java 编译器内部使用的 API,主要用于获取指定中的成员列表。此方法返回一个 `Scope` 对象,该对象含了属于特定的所有类和接口。 #### Scope 的作用 `Scope` 表示名称空间,在这里它用于存储内的所有顶级类和接口定义。通过这个 `Scope` 可以遍历并访问这些类型的元数据[^1]。 #### 使用场景 当需要处理或分析某个下的全部类时,可以调用此方法来获得完整的类型集合。这对于构建工具、静态代码分析器或其他依赖于源码结构的应用非常有用。 #### 返回值说明 - **返回类型**: `com.sun.tools.javac.util.Scope` - **描述**: 含了当前 PackageSymbol 下所有的顶层元素(即直接位于该中的类/接口) 注意:由于这是编译期API的一部分,并是标准库公开的部分,因此建议在常规应用程序逻辑中使用此类功能;而是应该考虑采用反射机制或者其他更稳定的方式来实现类似需求[^2]。 ```java import com.sun.tools.javac.code.Symbol; import com.sun.tools.javac.code.Symbol.PackageSymbol; public class Example { public static void main(String[] args){ // 获取 java.lang 对应的 PackageSymbol 实例 PackageSymbol pkg = ... ; // 调用 members 方法得到其下所有成员组成的 Scope Symbol.Scope scope = pkg.members(); // 进一步操作... } } ```
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值