<h2 id="1">1.path</h2>
<h3 id="1.1">1.1 path 作用</h3>
设置 path 是为了方便使用 java、javac等这些命令而不用写全路径。
如:
![](https://img.haomeiwen.com/i2756784/32e5057914183224.png)
<h2 id="2">2. classpath</h2>
<h3 id="2.1">2.1 classpath 的作用</h3>
jar 包的搜索路径,告诉应用程序去哪里寻找 class。默认为当前目录。在实际的应用时,一般不需要设置 classpath。
<h3 id="2.2">2.2 classpath的使用方式</h3>
<h4 id="2.2.1">2.2.1 通配符 </h4>
- 匹配所有 .jar 或者 .JAR
如 -cp mydir/* 用来查找 mydir 下所有 .jar或者.JAR结尾的文件。仅仅是查找 mydir 下的,不会去查找 mydir 子目录
2.匹配所有 .jar或者.JAR以及 class files (如:cat.class)
-cp mydir:mydir/* or mydir/:mydir 注意它们是有顺序的,mydir:mydir/ 会先加载 classes files 再加载 jar files
<h4 id="2.2.2">2.2.2 直接通过 jar 包</h4>
-cp mydir/a.jar:mydir/b.jar:mydir/c.jar
<h4 id="2.2.3">2.2.3 加载 class files</h4>
( 可以忽略,在实际项目中根本不会用这种方式,因为有 idea 等工具的存在,所以重点来体会一下工具为我们做了什么 )
目录结构
![](https://img.haomeiwen.com/i2756784/1934e39cca204757.png)
//package test;
/**
* @author shengjk1
* @date 2020/4/8
*/
public class Test2 {
public static void main(String[] args) {
System.out.println("Hello Wrold");
}
}
没有包名的时候
package test;
/**
* @author shengjk1
* @date 2020/4/8
*/
public class Test2 {
public static void main(String[] args) {
System.out.println("Hello Wrold");
}
}
加上包名
![](https://img.haomeiwen.com/i2756784/f10d65f0cfa6b5c3.png)
我们可以发现 java 以及 java -cp ./ 不管用了。发生了什么?
关键在于 package,package就表示在这个路径下去找这个类,当执行 java Test2 时,它会在 ./test 目录下查找,但根本就没有这个目录,故
Could not find or load main class
我们来指定 classpath
![](https://img.haomeiwen.com/i2756784/6e343444b1907536.png)
接下来呢,让 Test1去引用 Test2
package test;
import java.util.ArrayList;
/**
* @author shengjk1
* @date 2020/1/19
*/
public class Test1 {
public static void main(String[] args) {
Test2.main(args);
}
}
![](https://img.haomeiwen.com/i2756784/7bba36bf44f021fc.png)
在 Test1 中引入第三方的依赖 kudu
package test;
import org.apache.kudu.client.KuduTable;
import java.lang.ref.SoftReference;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;
/**
* @author shengjk1
* @date 2020/1/19
*/
public class Test1 {
public static void main(String[] args) {
ArrayList<Integer> integers = new ArrayList<>();
integers.add(192);
integers.add(191);
Map<String, SoftReference<KuduTable>> kuduTableMap = new HashMap<>();
Test2.main(args);
}
}
<h3 id="2.3">2.3 classpath的应用场景</h3>
- 打包是没有指定主类,可以用java -cp xxx.jar 主类名称(绝对路径)
- 要引用其他的jar包,可以用java -classpath $CLASSPATH:xxxx.jar 主类名称(绝对路径)
<h2 id="3">3. idea等工具究竟为我们做了什么</h2>
当我们使用idea的时候只需要
就可以运行一个包含 main 方法的 .java 文件,简单方便。可是为什么呢?
通过上面的例子我们可以知道,要想正常的编译执行 .java 文件,需要两个条件:
1.必须的 jar 包
2.class files 的正确路径
那么我们就来看一下 idea 是否满足这两个条件。
首先必须要有必须的 jar
我们所依赖的 jar 也都被放进了 classpath
所以基本 jar 依赖的问题就解决了,
其次,我们需要正确的 class files 路径。
设置了 build 后的路径为 ${PROJECT_ROOT}/target。
设置了 target 的文件路径。通过这两个设置,保证我们自己写的.java 文件的正确路径。至此我们执行
它就可以正确的跑起来了。
<h2 id="4">4. 参考</h2>
PATH and CLASSPATH
Setting the Class Path
<h2 id="5">5. 补充</h2>
就在刚刚我还好奇为什么一些程序,比如 Flink 。竟然可以不通过 java -jar 或者 java -cp 主方法 的方式来执行( 其实是通过 java -cp 主方法 的方式来执行的)。
我们修改一下 Flink 的启动脚本
# Add HADOOP_CLASSPATH to allow the usage of Hadoop file systems
echo '========================================='
echo $JAVA_RUN
echo $JAVA_ARGS
echo ${log_setting[@]}
classpath=`manglePathList "$CC_CLASSPATH:$INTERNAL_HADOOP_CLASSPATHS"`
echo $classpath
echo '======================================='
exec $JAVA_RUN $JVM_ARGS "${log_setting[@]}" -classpath "`manglePathList "$CC_CLASSPATH:$INTERNAL_HADOOP_CLASSPATHS"`" org.apache.flink.client.cli.CliFrontend "$@"
然后我们在执行一下,看看会发生什么?
这是 JAVA_ARGS
空空如野
这是 classpath,加上了 hadoop conf
/mnt/software/flink-1.7.2/lib/flink-python_2.11-1.7.2.jar:/mnt/software/flink-1.7.2/lib/flink-shaded-hadoop2-uber-1.7.2.jar:/mnt/software/flink-1.7.2/lib/log4j-1.2.17.jar:/mnt/software/flink-1.7.2/lib/slf4j-log4j12-1.7.15.jar:/mnt/software/flink-1.7.2/lib/flink-dist_2.11-1.7.2.jar::/etc/hadoop/conf:
网友评论