一、总结
1.引入包的概念的原因和包的作用
比如有多个人开发一个大型程序,A定义了一个Math.java类,B也定义了一个Math.java类,它们放在不同目录,使用的时候也是用目录来区分,
包实际上就是一个文件夹(可以含有多级子目录)路径名。包的作用就是用来区分同名的类(冲突问题)。
2.使用步骤
package:
Java源文件中:package 包名.子包名;eg: package a.b.c.d;
编译命令:javac -d <dir> <file.java> 表示在<dir>目录下生成包
执行方法:java 包名.子包名.类名
import:
使用不同的包时要import
public class可以被外包访问,class只能在本包中访问 ###???###
格式:java文件中 import 包名.子包名.类名; 这是手工导入这个类,也可以 import 包名.子包名.*;让JVM自动加载需要的类。
jar:
生成:jar -cvf <file.jar> <dir>
查看:jar -tvf <file.dir>
解压:tar -xvf <file.dir>
3.包就是一个文件夹路径,所以也可以在包下创建类
4.把程序交给用户的时候需要把Pack.class和整个包(a目录)交给他,这样比较麻烦,可以使用jar打包一下. jar -help 查看帮助。
5.压缩成.jar文件后删除a目录后java Pack运行报错“NoClassDefFoundError”,用户找不到这个类了,我们需要设置一个环境变量CLASSPATH,
它有两个作用,一个是在编译的时候来查找源代码,另一个作用是在执行时查找类的路径。
$ export CLASSPATH=.:my.jar //使用‘:’控制若在当前目录下找不到就去my.jar下面找
然后删除包目录再编译javac Pack.java就可以成功了,执行javac Pack也是成功的。
此时$export CLASSPATH=.将其设置为默认值,然后重新执行java Pack时就又报错了,说明是实时更加CLASS_PATH的值查询的。
如果以后遇到报找不到类的时候就可以考虑是否要设置CLASSPATH。它可以指定目录,也可以指定压缩包。
6.javac -help指明使用-verbose可以在编译时输出更多信息,加上它后可以看它去哪里查找那些源代码[search path for source files: .,my.jar]。
显示出的那些目录就是通过CLASS_PATH指定的。
$ javac Pack.java -verbose
7. 多个Java源文件可以 package 在一起,但是一个源文件使用另一个源文件中定义的类还是需要 import <包名>.<类名>
二、例子
1. 编译生成包
/*file: Pack.java*/ package a.b.c.d; //要把编译生成的class文件放在指定目录下的a/b/c/d/目录下 public class Pack { public static void main(String args[]) { System.out.println("Hello, world!"); } } /* 编译执行: $ javac -d . Pack.java $ java a.b.c.d.Pack */
2.区分使用不同包中的函数
$ tree
.
├── lisi
│ └── Math.java
├── Pack.java
└── zhangsan
├── Math.java
└── Print.java
zhangsan/Math.java
package a.b.c.d2; public class Math { public static int sub(int x, int y) { return x - y; } public static int add(int x, int y) { return x + y + 2; } }
zhangsan/Print.java
package a.b.c.d2; public class Print { public static void printInfo() { System.out.println("package: a.b.c.d2"); } }
lisi/Math.java
package a.b.c.d1; public class Math { public static int add(int x, int y) { return x + y; } }
Pack.java
import a.b.c.d1.*; //a.b.c.d1.Math import a.b.c.d2.*; //若这里都直接写a.b.c.d2.Math会报错:Math already defined in a single-type import public class Pack { public static void main(String args[]) { /* add */ System.out.println(a.b.c.d1.Math.add(1,2)); //调用指定包的函数 System.out.println(a.b.c.d2.Math.add(1,2)); /* sub */ System.out.println(a.b.c.d2.Math.sub(1,2)); a.b.c.d2.Print.printInfo(); //调用包下面的Print类的printInfo方法 Print.printInfo(); //也可以这样调用,如果没有同名的类的话就不需要使用包来限制它 } }
编译执行
javac -d . lisi/Math.java javac -d . zhangsan/Math.java javac -d . zhangsan/Print.java javac Pack.java javac Pack
打包的例子略。
2.第二个例子
/*A.java*/ package a.b.c; public class A { public int count; protected void print_a() { System.out.println("print_a"); } }
/*B.java*/ package d.e.f; /*package必须要在import前面*/ import a.b.c.A; public class B extends A { public int count; public void print_b() { super.print_a(); /*处于不同的包中子类也可以访问父类的protected成员*/ System.out.println("print_b"); } }
/*Test.java*/ package m.n; import a.b.c.A; import d.e.f.B; public class Test { public static void main(String args[]) { B b = new B(); b.print_b(); } }
/*编译步骤*/ export export CLASSPATH=./ javac -d ./ A.java javac -d ./ B.java javac -d ./ Test.java $ tree . ├── a │ └── b │ └── c │ └── A.class ├── A.java ├── B.java ├── d │ └── e │ └── f │ └── B.class ├── m │ └── n │ └── Test.class └── Test.java /*运行和结果*/ $ java m.n.Test print_a print_b
本文参考链接:https://www.cnblogs.com/hellokitty2/p/10398883.html