Java命令行编译文件时出现的错误,找不到符号或软件包不存在等
标签(空格分隔): Java
参考http://stackoverflow.com/questions/6665420/package-does-not-exist-error
参考http://docs.oracle.com/javase/7/docs/technotes/tools/windows/javac.html
配套代码上传在:
习惯了eclipse的自动编译,Java命令行编译、执行文件只会最基础的部分,就是对单文件的编译和执行,而且不包含任何外部JAR包。但有时候你还非得用命令行,会碰到一些问题,博主这里给出几种常见的问题及解决方法。
实验环境介绍
eclipse 的工作目录在D:\myeclipse\Workspaces
,里面有我们的一个工程example
,点进去,eclipse为我们设置好的几个文件.settings
、bin
、src
、.classpath
、project
,src
中存放的是源代码,结构就如上图所示。在eclipse导入后显示的结构如下图所示,
小提示:包名com.example.controller
指的是在src/com/example/controller
文件夹下有个controller.java文件
model.java
是一个独立的类,不导入任何外部包 controller.java
导入了com.example.model.model
view.java
导入了com.example.controller.controller
和调用了一个外部Jar包
在观察案例代码时,需要特别注意此时命令行的当前目录,选项参数和Java文件,会有不同的结果
使用绝对路径
绝对路径就是从根目录开始的路径,Windows下就是以盘符开头,如”C:\”,Linux下以斜杠开头/home。
使用绝对路径的好处就是不管命令行的当前路径在哪里,都能正确的进行编译
案例一
D:\myeclipse\Workspaces\example>javac D:\myeclipse\Workspaces\example\src\com\example\model\*.java
案例二
D:\myeclipse\Workspaces\example\src>javac D:\myeclipse\Workspaces\example\src\com\example\model\*.java
使用相对路径
- 相对路径就是相对当前上下文环境,在命令行中就是相对命令行中前面的路径。
- 相对路径不以/开头,直接文件名。
如案例一,当前路径在D:\myeclipse\Workspaces\example\src
,相对这个路径,Java文件的相对路径就为com/example/model/*.java
。在案例二中当前路径有变化,相应得Java文件的相对路径也有变化。
案例一
D:\myeclipse\Workspaces\example\src>javac com/example/model/*.java
案例二
D:\myeclipse\Workspaces\example>javac src/com/example/model/*.java
基本概念已经理解了,上干货,一般命令行编译出现的问题:
问题一:找不到符号或软件包不存在
案例一
D:\myeclipse\Workspaces\example\src>javac com/example/controller/*.java
这种情况下可以正常得进行编译,因为你自己不指定classpath,一般系统默认的classpath会在当前路径中搜索。
案例二
D:\myeclipse\Workspaces\example>javac src/com/example/controller/*.java
报错
src\com\example\controller\controller.java:3: 软件包 com.example.model 不存在
我们切换目录,已上文绝对路径、相对路径的知识,正确给出Java文件的路径。这时会报出错误,软件包com.xxx
不存在。因为在controller.java
文件中有一个import com.example.model.model
,根据系统默认的classpath
会在当前路径下找,即D:\myeclipse\Workspaces\example
,很明显这个路径下没有com
文件。
解决方法,我们告诉编译器哪个目录下会有就行了,所以需要加入选项-cp xxxx 或 -classpath xxxx
D:\myeclipse\Workspaces\example>javac -cp src src/com/example/controller/*.java
加上参数-cp src
,告诉编译器可以去src中找,就不会有问题了。
同样的,classpath
可以是相对路径,也可以是绝对路径,本例中使用的是相对路径,命令行换路径后,所有参数都要换
D:\myeclipse\Workspaces>javac -cp example/src example/src/com/example/controller/*.java
下面这个是用绝对路径写的,相对路径和绝对路径以下不再叙述
javac -cp D:\myeclipse\Workspaces\example\src example/src/com/example/controller/*.java
问题二:JAR包怎么引入
首先还是在eclipse调通,建议专门建立一个lib
文件,然后把JAR包全部放在这里面,对项目转移也比较方便。
案例一
D:\myeclipse\Workspaces\example\src>javac com/example/view/view.java
错误
com\example\view\view.java:4: 软件包 com.google.gson 不存在
import com.google.gson.Gson;
^
com\example\view\view.java:14: 找不到符号
符号: 类 Gson
位置: 类 com.example.view.view
Gson gson = new Gson();
^
com\example\view\view.java:14: 找不到符号
符号: 类 Gson
位置: 类 com.example.view.view
Gson gson = new Gson();
^
3 错误
其实就一个问题,没找到JAR包。引入JAR就是注意要把路径写到JAR包的具体位置。废话少说,给出例子:
案例二
D:\myeclipse\Workspaces\example\src>javac -cp lib/gson-2.3.1.jar com/example/view/view.java
错误
com\example\view\view.java:3: 软件包 com.example.controller 不存在
import com.example.controller.controller;
^
com\example\view\view.java:10: 找不到符号
符号: 类 controller
位置: 类 com.example.view.view
new controller();
^
2 错误
细心的你会发现,问题变了,因为我们引入了JAR包,所以因为JAR找不到的GSON问题已经没了,现在的问题就是找不到com.xxxx
,就变成问题一。这里解释一下,我们加入-cp xxxx或-classpath xxxx之后,会覆盖系统中的classpath,案例二这么写的确可以找到GSON类,但在碰到import com.example.model.model
时还在lib/gson-2.3.1.jar文件中查当然是找不到了,所以报错。方法很简单,在增加一个com.xxxx
文件夹的路径,没错就是当前路径。所以加上.;lib/gson-2.3.1.jar
。现在你是否对一开始配置Java环境时候,classpath中的路径要.;%JAVA_HOME%\lib\dt.jar;%JAVA_HOME%\lib\tools.jar
有了理解呢?
特别注意的是在Linux下用冒号:进行分隔,而在Windows用分号进行分割。
案例三
正确
D:\myeclipse\Workspaces\example\src>javac -cp .;lib/gson-2.3.1.jar com/example/view/view.java
问题三:指定class文件存放位置
默认的情况下,class文件和java文件存放在一个目录下,但你有没有发现,eclipse编译后会有两个目录,一个是src,存放的是用户编辑的源代码,定一个是bin文件,是编译后的字节码。它是加了-d
选项。
D:\myeclipse\Workspaces\example\src>javac -d ../bin com/example/model/model.java
这里bin文件和src文件是同级(看实验环境介绍),也是使用的是相对路径。打开文件后可以发现在bin文件中生成了com/example/model/model.class
具有层次结构的目录。
到此我相信你已经能基本上解决编译时的问题了。
总结
本文给出了几个常见问题的解决方案,以上代码都经过实验过,不会有问题。命令行中我们要注意相对路径和绝对路径的差别,已经Linux下和Windows下分隔符的差别(Linux用冒号,Windows下用分号)。另外设置classpath时,一般都加上当前路径.。