Java学习从入门到精通(附FAQ)2

发表于:2007-06-22来源:作者:点击数: 标签:
附:java入门与加深FAQ 一、准备篇 1 什么是 Java 、Java2、JDK?JDK后面的1.3、1.4.2版本号又是怎么回事? 答:Java是一种通用的,并发的,强类型的, 面向对象 的编程语言(摘自Java规范第二版) JDK是Sun公司分发的免费Java 开发 工具,正式名称为J2SDK(Ja

   

附:java入门与加深FAQ

一、准备篇

1 什么是Java、Java2、JDK?JDK后面的1.3、1.4.2版本号又是怎么回事?

答:Java是一种通用的,并发的,强类型的,面向对象的编程语言(摘自Java规范第二版) JDK是Sun公司分发的免费Java开发工具,正式名称为J2SDK(Java2 Software Develop Kit)。

2 什么是JRE/J2RE?

答:J2RE是Java2 Runtime Environment,即Java运行环境,有时简称JRE。如果你只需要运行Java程序或Applet,下载并安装它即可。如果你要自行开发Java软件,请下载JDK。在JDK中附带有J2RE。

注意:由于Microsoft对Java的支持不完全,请不要使用IE自带的虚拟机来运行Applet,务必安装一个J2RE或JDK。

3 学习Java用什么工具比较好?

答:作者建议首先使用JDK+文本编辑器,这有助你理解下列几个基础概念:path,classpath,package并熟悉基本命令:javac和java。并且下载和你的JDK版本一致的API帮助。

如果你不确定类或函数的用法,请先查阅API而不是发贴求助。当你熟悉Java之后,你可以考虑换一个IDE。很多人推荐JCreator,实际上JCreator的功能还 是很弱的。作者推荐eclipse,下载网http://www.eclipse.org因eclispe是免费的.

4 学习Java有哪些好的参考书?

答:作者首先推荐Thinking in Java,中文名《Java编程思想》,有中文版。该书第一章介绍了很多面向对象的编程思想,作为新手应当认真阅读。除此以外,O′relly出版社和Wrox出版社的书也不错。作者本人不喜欢大陆作者的书。也许你觉得英文太难,但是网上大多数资料都是英文的。另外,你需要经常查阅API,而那也是英文的。

5 Java和C++哪个更好?

答:这个问题是一个很不恰当的问题。你应该问:Java和C++哪个更适用于我的项目?如果你不需要跨平台,不需要分布式,要强调程序的运行速度,C++更为适用。反之?你应当考虑Java。

6 什么是J2SE/J2EE/J2ME?

答:J2SE就是一般的Java。

J2ME是针对嵌入式设备的,比如Java手机,它有自己的SDK。而J2EE使用J2SE的SDK。

J2EE规范更多的是对J2EE服务器的要求和开发人员的约束。详情见后继"J2EE FAQ"。

二、命令篇

7 我写了第一个Java程序,应该如何编译/运行?

答:首先请将程序保存为xxx.java文件,然后在dos窗口下使用javac xxx.java命令,你会发现该目录下多了一个xxx.class文件,再使用java xxx命令,你的java程序就开始运行了。

8 我照你说的做了,但是出现什么"′javac′ 不是内部或外部命令,也不是可运行的程序或批处理文件。"。

答:你遇到了path问题。操作系统在一定的范围(path)内搜索javac.exe,但是没能找到。请编辑你的操作系统环境变量,新增一个JAVA_HOME变量,设为你JDK的安装目录,再编辑Path变量,加上一项 %JAVA_HOME%\bin。然后关掉并新开一个dos窗口,你就可以使用javac和java命令了。

9 环境变量怎么设置?

答:请向身边会设的人咨询。

10 javac xxx.java顺利通过了,但是java xxx的时候显示什么“NoClassDefFoundError”。

答:你遇到了classpath问题。java命令在一定的范围(classpath)内搜索你要用的class文件,但是未能找到。首先请确认你没有错敲成java xxx.class,其实你并不需要设置该变量,但如果你设置了该变量又没有包含.(代表当前目录)的项,你就会遇到这个问题。请在你的CLASSPATH环境变量中加入一项. 或干脆删掉这个变量。如果你使用了并非JDK自带的标准包,比如javax.servlet.*包,也会遇到这个问题,请将相应的jar文件加入classpath。如果你在java源文件中定义了package,请参见15。

11 我在java xxx的时候显示"Exception in thread "main" java.lang.NoSuchMethodError: main"。

答:首先,在你的程序中每个java文件有且只能有一个public类,这个类的类名必须和文件名的大小写完全一样。其次,在你要运行的类中有且只能有一个public static void main(String[] args)方法,这个方法就是你的主程序。

12 package是什么意思?怎么用?

答:为了唯一标识每个类并分组,java使用了package的概念。每个类都有一个全名,例如String的全名是java.lang.String,其中java.lang是包名,String是短名。这样,如果你也定义了String,你可以把它放在mypackage中,通过使用全名mypackage.String和java.lang.String来区分这两个类。同时,将逻辑上相关的类放在同一个包中,可以使程序结构更为清楚。

你要做的就是在java文件开头加一行"package mypackage;"。注意包没有嵌套或包含关系,A包和A.B包对java命令来说是并列的两个包。

13 我没有声明任何package会怎么样?

答:你的类被认为放在默认包中。这时全名和短名是一致的。

14 在一个类中怎么使用其他类?

答:如果你使用java.lang包中的类,不用做任何事。如果你使用其他包中的类,使用import package1.class1; 或 import package2.*;这里.*表示引入这个包中的所有类。然后在程序中你可以使用其他类的短名。如果短名有冲突,使用全名来区分。

15 我用了package的时候显示"NoClassDefFoundError",但是我把所有package去掉的时候能正常运行。

答:将你的java文件按包名存放。

比如你的工作目录是/work,你的类是package1.class1,那么将它存放为/work/package1/class1.java。如果没有声明包,那么直接放在/work下。在/work下执行javac package1/class1.java,再执行java package1.class1,你会发现一切正常。另外,你可以考虑开始使用IDE。

16 我想把java编译成exe文件,该怎么做?

答:JDK只能将java源文件编译为class文件。class文件是一种跨平台的字节码,必须依赖平台相关的JRE来运行。Java以此来实现跨平台,有些开发工具可以将java文件编译为exe文件。作者反对这种做法,因为这样就取消了跨平台性。如果你确信你的软件只在Windows平台上运行,你可以考虑使用C++/C#来编程。

17 我在编译的时候遇到什么"deprecated API",是什么意思?

答:所谓deprecated是指已经?时,但是为了向前兼容起见仍然保留的方法,这些方法可能会在以后取消支持。你应当改用较新的方法。一般在API里面会说明你应当用什么方法来代替之。

三、I/O篇

18 我怎么给java程序加启动参数,就像dir /p/w那样?

答:还记得public static void main(String[] args)吗?这里的args就是你的启动参数。

在运行时你输入java package1.class1 -arg1 -arg2,args中就会有两个String,一个是arg1,另一个是arg2。

19 我怎么从键盘输入一个int/double/字符串?

答:java的I/O操作比C++要复杂一点。如果要从键盘输入,样例代码如下:

BufferedReader cin = new BufferedReader( new InputStreamReader( System.in ) )

;

String s = cin.readLine();

这样你就获得了一个字符串,如果你需要数字的话再加上:

int n = Integer.parseInt( s );

或者

double d = Double.parseDouble( s );

20 我怎么输出一个int/double/字符串?

答:在程序开始写:

PrintWriter cout = new PrintWriter( System.out );

需要时写:

cout.print(n);

或者

cout.println("hello")

等等。

21 我发现有些书上直接用System.in和System.out输入输出,比你要简单得多。

答:java使用unicode,是双字节。而System.in和System.out是单字节的stream。如果你要输入输出双字节文字比如中文,请使用作者的做法。

22 我怎么从文件输入一个int/double/字符串?

答:类似于从键盘输入,只不过换成

BufferedReader fin = new BufferedReader( new FileReader(" myFileName " ) );

PrintWriter fout = new PrintWriter( new FileWriter(" myFileName " ) );

另外如果你还没下载API,请开始下载并阅读java.io包中的内容。

23 我想读写文件的指定位置,该怎么办?

答:你肯定没有认真看API。java.io.RandomAclearcase/" target="_blank" >ccessFile可以满足你的需要。

24 怎么判断要读的文件已经到了尽头?

答:你肯定没有认真看API。在Reaer的read方法中明确说明返回-1表示流的结尾。
四、 关键字篇

25 java里面怎么定义宏?

答:java不支持宏,因为宏代换不能保证类型安全。如果你需要定义常量,可以将它定义为某个类的static final成员。参见26和30。

26 java里面没法用const。

答:你可以用final关键字。例如 final int m = 9。被声明为final的变量不能被再次赋值。也可以用于声明方法或类,被声明为final的方法或类不能被继承。注意const是java的保留字以备扩充。

27 java里面也不能用goto。

答:甚至在面向过程的语言中你也可以完全不用goto。请检查你的程序流程是否合理。如果你需要从多层循环中迅速跳出,java增强了(和C++相比)break和continue的功能。

例如:

outer :

while( ... )

{

inner :

for( ... )

{

... break inner; ...

... continue outer; ...

}

}

和const一样,goto也是java的保留字以备扩充。

28 java里面能不能重载操作符?

答:不能。String的+号是唯一一个内置的重载操作符。你可以通过定义接口和方法来实现类似功能。

29 我new了一个对象,但是没法delete掉它。

答:java有自动内存回收机制,即所谓Garbarge Collector。你再也不用担心指针错误。

30 我想知道为什么main方法必须被声明为public static?

答:声明为public是为了这个方法可以被外部调用,详情见面向对象篇37。

static是为了将某个成员变量/方法关联到类(class)而非实例(instance)。你不需要创建一个对象就可以直接使用这个类的static成员,在A类中调用B类的static成员可以使用B.staticMember的写法。注意一个类的static成员变量是唯一的,被所有该类对象所共享的。

31 throw和throws有什么不同?

答:throws用于声明一个方法会抛出哪些异常。而throw是在方法体中实际执行抛出异常的动作。如果你在方法中throw一个异常,却没有在方法声明中声明之,编译器会报错。注意Error和RuntimeException的子类是例外,无需特别声明。

32 什么是异常?

答:异常最早在Ada语言中引入,用于在程序中动态处理错误并恢复。你可以在方法中拦截底层异常并处理之,也可以抛给更高层的模块去处理。你也可以抛出自己的异常指示发生了某些不正常情况。常见的拦截处理代码如下:

try

{

...... //以下是可能发生异常的代码

...... //异常被抛出,执行流程中断并转向拦截代码。

......

}

catch(Exception1 e) //如果Exception1是Exception2的子类并要做特别处理,应排在前面

{

//发生Exception1时被该段拦截

}

catch(Exception2 e)

{

//发生Exception2时被该段拦截

}

finally //这是可选的

{

//无论异常是否发生,均执行此段代码

}

33 final和finally有什么不同?

答:final请见26。finally用于异常机制,参见32。

五、 面向对象篇

34 extends和implements有什么不同?

答:extends用于(单)继承一个类(class),而implements用于实现一个接口(interface)。interface的引入是为了部分地提供多继承的功能。

在interface中只需声明方法头,而将方法体留给实现的class来做。这些实现的class的实例完全可以当作interface的实例来对待。有趣的是在interface之间也可以声明为extends(单继承)的关系。

35 java怎么实现多继承?

答:java不支持显式的多继承。因为在显式多继承的语言例如c++中,会出现子类被迫声明祖先虚基类构造函数的问题,而这是违反面向对象的封装性原则的。java提供了interface和implements关键字来部分地实现多继承。参见34。

36 abstract是什么?

答:被声明为abstract的方法无需给出方法体,留给子类来实现。而如果一个类中有abstract方法,那么这个类也必须声明为abstract。被声明为abstract的类无法实例化,尽管它可以定义构造方法供子类使用。

37 public,protected,private有什么不同?

答:这些关键字用于声明类和成员的可见性。

public成员可以被任何类访问,

protected成员限于自己和子类访问,

private成员限于自己访问。

Java还提供了第四种的默认可见性,一般称为package private,当没有任何public,protected,private修饰符时,成员是同一包内可见。类可以用public或默认来修饰。

38 Override和Overload有什么不同?

答:Override是指父类和子类之间方法的继承关系,这些方法有着相同的名称和参数类型。Overload是指同一个类中不同方法(可以在子类也可以在父类中定义)间的关系,这些方法有着相同的名称和不同的参数类型。

39 我继承了一个方法,但现在我想调用在父类中定义的方法。

答:用super.xxx()可以在子类中调用父类方法。

40 我想在子类的构造方法中调用父类的构造方法,该怎么办?

答:在子类构造方法的第一行调用super(...)即可。

41 我在同一个类中定义了好几个构造方法并且想在一个构造方法中调用另一个。

答:在构造方法第一行调用this(...)。

42 我没有定义构造方法会怎么样?

答:自动获得一个无参数的构造方法。

43 我调用无参数的构造方法失败了。

答:如果你至少定义了一个构造方法,就不再有自动提供的无参数的构造方法了。你需要显式定义一个无参数的构造方法。

44 我该怎么定义类似于C++中的析构方法(destructor)?

答:提供一个void finalize()方法。在Garbarge Collector回收该对象时会调用该方法。注意实际上你很难判断一个对象会在什么时候被回收。作者从未感到需要提供该方法。

45 我想将一个父类对象转换成一个子类对象该怎么做?

答:强制类型转换。如

public void meth(A a)

{

B b = (B)a;

}

如果a实际上并不是B的实例,会抛出ClassCastException。所以请确保a确实是B的实例。

46 其实我不确定a是不是B的实例,能不能分情况处理?

答:可以使用instanceof操作符。例如

if( a instanceof B )

{

B b = (B)a;

}

else

{

...

}

47 我在方法里修改了一个对象的值,但是退出方法后我发现这个对象的值没变!

答:很可能你把传入参数重赋了一个新对象,例如下列代码就会造成这种错误:

public void fun1(A a) //a是局部参数,指向了一个外在对象。

{

a = new A(); //a指向了一个新对象,和外在对象脱钩了。如果你要让a作为传出变量,不要写这一句。

a.setAttr(attr);//修改了新对象的值,外在对象没有被修改。

}

基本类型也会出现这种情况。例如:

public void fun2(int a)

{

a = 10;//只作用于本方法,外面的变量不会变化。

}

六、java.util篇

48 java能动态分配数组吗?

答:可以。例如int n = 3; Language[] myLanguages = new Language[n];

49 我怎么知道数组的长度?

答:用length属性。如上例中的 myLanguages.length 就为 3。

50 我还想让数组的长度能自动改变,能够增加/删除元素。

答:用顺序表--java.util.List接口。你可以选择用ArrayList或是LinkedList,前者是数组实现,后者是链表实现。例如:

List list = new ArrayList();

或是

List list = new LinkedList();

51 什么是链表?为什么要有两种实现?

答:请补习数据结构。

52 我想用队列/栈。

答:用java.util.LinkedList。

53 我希望不要有重复的元素。

答:用集合--java.util.Set接口。例如:Set set = new HashSet()。

54 我想遍历集合/Map。

答:用java.util.Iterator。参见API。

55 我还要能够排序。

答:用java.util.TreeSet。例如:Set set = new TreeSet()。放进去的元素会自动排序。你需要为元素实现Comparable接口,还可能需要提供equals()方法,compareTo()方法,hashCode()方法。

56 但是我想给数组排序。

答:java.util.Arrays类包含了sort等实用方法。

57 我想按不同方法排序。

答:为每种方法定义一个实现了接口Comparator的类并和Arrays综合运用。

58 Map有什么用?

答:存储key-value的关键字-值对,你可以通过关键字来快速存取相应的值。

59 set方法没问题,但是get方法返回的是Object。

答:强制类型转换成你需要的类型。参见45。

60 我要获得一个随机数。

答:使用java.util.Random类。

61 我比较两个String总是false,但是它们明明都是"abc" !

答:比较String一定要使用equals或equalsIgnoreCase方法,不要使用 == !

==比较的是两个引用(变量)是否指向了同一个对象,而不是比较其内容。

62 我想修改一个String但是在String类中没找到编辑方法。

答:使用StringBuffer类。

String str = "......."; //待处理的字符串

StringBuffer buffer = new StringBuffer(str); //使用该字符串初始化一个

StringBuf

fer

buffer.append("..."); //调用StringBuffer的相关API来编辑字符串

String str2 = buffer.toString(); //获得编辑后的字符串

另外,如果你需要将多个字符串连接起来,请尽量避免使用+号直接连接,而是使用StringBuffer.append()方法。

63 我想处理日期/时间。

答:使用java.util.Date类。你可以使用java.text.SimpleDateFormat类来在String和Date间互相转换。

SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); //规定日期格式

Date date = formatter.parse("2003-07-26 18:30:35"); //将符合格式的String转换为Date

String s = formatter.format(date); //将Date转换为符合格式的String

关于定义日期格式的详细信息请参见API。

原文转自:http://www.ltesting.net