Java的package和import机制

来源:互联网 发布:linux grub 引导 修复 编辑:程序博客网 时间:2024/05/16 05:41

======================================================
注:本文源代码点此下载
======================================================

一些人用了一阵子的java,可是对于 java 的 package 跟 import 还是不太了解。很多人以为原始码 .java 文件中的 import 会让编译器把所 import 的程序通通写到编译好的 .class 档案中,或是认为 import 跟 c/c++ 的 #include 相似,实际上,这是错误的观念。

让我们先了解一下,java 的 package 到底有何用处。

其实,package 名称就像是我们的姓,而 class 名称就像是我们的名字。package 名称有很多 . 的,就好像是复姓。比如说 java.lang.string,就是复姓 java.lang,名字为 string 的类别;java.io.inputstream 则是复姓java.io,名字为 inputstream 的类别。

java 会使用 package 这种机制的原因也非常明显,就像我们取姓名一样,光是一间学校的同一届同学中,就有可能会出现不少同名的同学,如果不取姓的话,那学校在处理学生数据,或是同学彼此之间的称呼,就会发生很大的困扰。相同的,全世界的 java 类别数量,恐怕比台湾人口还多,而且还不断的在成长当中,如果类别不使用套件名称,那在用到相同名称的不同类别时,就会产生极大的困扰。幸运的是,java 的套件名称我们可以自己取,不像人的姓没有太大的选择 ( 所以有很多同名同姓的 ),如果依照 sun 的规范来取套件名称,那理论上不同人所取的套件名称不会相同 ( 请参阅 "命名惯例"的相关文章 ),也就不会发生名称冲突的情况。

可是问题来了,因为很多套件的名称非常的长,在写程序时,会多打好多字,花费不少时间,比如说在a.b.c文件下有point和circle两个类,现在在程序中要调用:

a.b.c.pointp1=new a.b.c.point();

a.b.c.circlec1=new a.b.c.circle();

实在是不美观又麻烦.于是,sun 想了一个办法,就是 import. 就是在程序一开头的时候,说明程序中会用到那些类的路径.首先,在档案开头写:

import a.b.c.point;

import a.b.c.circle;

这两行说明了类的路径,所以当程序中提到point就是指a.b.c.point,而circle就是指a.b.c.circle,依此类推。于是原来的程序就变成:

pointp1=new point();

circlec1=new circle();

这样看起来是不是清爽多了呢?如果这些类别用的次数很多,那就更能体会到import 的好处了。可是这样还是不够,因为懒是人的天性,还是会有人觉得打太多 import 了也很浪费时间,于是 sun 又提供了一个方法:

import a.b.c.*; 意思就是,等一下程序中提到的没有姓名的类别,全都包含在a.b.c这个目录中。

注意点:但我们在程序中经常使用system.out这个类,为什么没有import system.out呢,因为java.lang 这个套件实在是太常太常太常用到了,几乎没有程序不用它的,所以不管你有没有写 import java.lang;,编译器都会自动帮你补上,也就是说编译器只要看到没有姓的类别,它就会自动去 java.lang 里面找找看,看这个类别是不是属于这个套件的。所以我们就不用特别去import java.lang 了。

为甚么我一开始说 import 跟 #include 不同呢?因为 import 的功能到此为止,它不像 #include 一样,会将档案内容载入进来。import 只是请编译器帮你打字,让编译器把没有姓的类别加上姓,并不会把别的文件的程式码写进来。如果你想练习打字,可以不要使用 import,只要在用到类别的时候,用它的全部姓名来称呼它就行了(就像例子一开始那样),跟使用 import 完全没有甚么两样。

先介绍java的package机制

基本原则:需要将类文件切实安置到其所归属之package所对应的相对路径下。

例如:以下面程序为例:假设此hello.java文件在d:\java\下

packagea;

public class hello{

public static void main(string args[]){

system.out.println("hello world!");

}

}

d:\java>javachello.java此程序可以编译通过.接着执行。

d:\java>javahello但是执行时,却提示以下错误!

exception in thread "main" java.lang.noclassdeffounderror: hello (wrong name: a/hello)

at java.lang.classloader.defineclass0(native method)

at java.lang.classloader.defineclass(classloader.java:537)

at java.security.secureclassloader.defineclass(secureclassloader.java:123)

at java.net.urlclassloader.defineclass(urlclassloader.java:251)

at java.net.urlclassloader.access$100(urlclassloader.java:55)

at java.net.urlclassloader$1.run(urlclassloader.java:194)

at java.security.accesscontroller.doprivileged(native method)

at java.net.urlclassloader.findclass(urlclassloader.java:187)

at java.lang.classloader.loadclass(classloader.java:289)

at sun.misc.launcher$appclassloader.loadclass(launcher.java:274)

at java.lang.classloader.loadclass(classloader.java:235)

at java.lang.classloader.loadclassinternal(classloader.java:302)

原因是我们把生成的hello.class规定打包在d:\java\a文件中,必须在a文件中才能去运行。所以应该在d:\java目录下建立一个a目录,然后把把hello.class放在它下面,执行时,可正常通过!

d:\java\>java a.hello就会输出:hello world!

现在介绍java的import机制

我们在d:\java目录下建立一个jintian.java文件,其内容如下:

importa.hello;

public class jintian{

public static void main(string[] args){

hellohello1=new hello();

}

}

d:\java\>javac jintian.java编译成功!

d:\java\>javajintian运行成功!

也就是你在jintian.class中成功的引用了hello.class这个类,是通过import a.hello来实现的,如果你没有这句话,就会提示不能找到hello这个类。

提示1:如果你在d:\java目录下仍保留一个hello.java文件的话,执行对主程序的编译命令时仍会报错!你自己可以试试呀!

提示2:如果你删除d:\java\a\hello.java文件的话,只留hello.class文件,执行对主程序的编译命令时是可以通过,此时可以不需要子程序的源代码。

提出一个问题:如果把目录a剪切到其它目录,如d盘根目录下,在a目录如果执行编译和执行命令呢?

很明显,会报以下错误!当然了,前提条件是你没有设置classpath路径,其实只要没把类搜索路径设置到我这个位置就会出错的!你试试吧!


======================================================
在最后,我邀请大家参加新浪APP,就是新浪免费送大家的一个空间,支持PHP+MySql,免费二级域名,免费域名绑定 这个是我邀请的地址,您通过这个链接注册即为我的好友,并获赠云豆500个,价值5元哦!短网址是http://t.cn/SXOiLh我创建的小站每天访客已经达到2000+了,每天挂广告赚50+元哦,呵呵,饭钱不愁了,\(^o^)/
原创粉丝点击