JAVA程序改錯
1.
abstract class Name { private String name;
public abstract boolean isStupidName(String name) {} }
答案: 錯。abstract method必須以分號結尾,且不帶花括號。
2.
public class Something { void doSomething () { private String s = ""; int l = s.length(); } }
答案: 錯。局部變量前不能放置任何訪問修飾符 (private,public,和protected)。final可以用來修飾局部變量
3.abstract class Something {
private abstract String doSomething (); }
答案: 錯。abstract的methods不能以private修飾。abstract的methods就是讓子類implement(實現)具體細節的,怎麽可以用private把abstract method封鎖起來呢? (同理,abstract method前不能加final)。
4.
public class Something {
public int addOne(final int x) { return ++x; } }
答案: 錯。int x被修飾成final,意味著x不能在addOne method中被修改。
5.
public class Something {
public static void main(String[] args) { Other o = new Other();
new Something().addOne(o); }
public void addOne(final Other o) { o.i++;
wk_ad_begin({pid : 21});wk_ad_after(21, function(){$('.ad-hidden').hide();}, function(){$('.ad-hidden').show();});
2
} }
class Other { public int i; }
和上面的很相似,都是關於final的問題,這有錯嗎?
答案: 正確。在addOne method中,參數o被修飾成final。如果在addOne method裏我們修改了o的reference(比如: o = new Other();),那麽如同上例這題也是錯的。但這裏修改的是o的member vairable(成員變量),而o的reference並沒有改變。
6.
class Something { int i;
public void doSomething() { System.out.println("i = " + i); } }
有什麽錯呢?
答案: 正確。輸出的是"i = 0"。int i屬於instant variable (實例變量,或叫成員變量)。instant variable有default value。int的default value是0。
7.
class Something { final int i;
public void doSomething() { System.out.println("i = " + i); } }
和上面壹題只有壹個地方不同,就是多了壹個final。這難道就錯了嗎? 答案: 錯。final int i是個final的instant variable (實例變量,或叫成員變量)。final的instant variable沒有default value,必須在constructor (構造器)結束之前被賦予壹個明確的值。可以修改為"final int i = 0;"。
8.
public class Something {
public static void main(String[] args) { Something s = new Something();
System.out.println("s.doSomething() returns " +
doSomething());
}
public String doSomething() { return "Do something ..."; }
3
}
看上去很完美。 答案: 錯。看上去在main裏call doSomething沒有什麽問題,畢竟兩個methods都在同壹個class裏。但仔細看,main是static的。static method不能直接call non-static methods。可改成"System.out.println("s.doSomething() returns " + s.doSomething());"。同理,static method不能訪問non-static instant variable。
9.
此處,Something類的文件名叫OtherThing.java class Something {
private static void main(String[] something_to_do) { System.out.println("Do something ..."); } }
這個好像很明顯。
答案: 正確。從來沒有人說過Java的Class名字必須和其文件名相同。但public class的名字必須和文件名相同。
10.
interface A{ int x = 0; }
class B{ int x =1; }
class C extends B implements A { public void pX(){
System.out.println(x); }
public static void main(String[] args) { new C().pX(); } }
答案:錯誤。在編譯時會發生錯誤(錯誤描述不同的JVM有不同的信息,意思就是未明確的x調用,兩個x都匹配(就象在同時import java.util和java.sql兩個包時直接聲明Date壹樣)。對於父類的變量,可以用super.x來明確,而接口的屬性默認隱含為 public static final.所以可以通過A.x來明確。
11.
interface Playable { void play(); }
interface Bounceable {
4
void play(); }
interface Rollable extends Playable, Bounceable { Ball ball = new Ball("PingPang"); }
class Ball implements Rollable { private String name;
public String getName() { return name; }
public Ball(String name) { this.name = name; }
public void play() {
ball = new Ball("Football");
System.out.println(ball.getName()); } }
這個錯誤不容易發現。
答案: 錯。"interface Rollable extends Playable, Bounceable"沒有問題。interface可繼承多個interfaces,所以這裏沒錯。問題出在:
interface Rollable裏的"Ball ball = new Ball("PingPang");"。
任何在interface裏聲明的interface variable (接口變量,也可稱成員變量),默認為public static final。也就是說:
Ball ball = new Ball("PingPang");實際上是
public static final Ball ball = new Ball("PingPang");。
在Ball類的Play()方法中,"ball = new Ball("Football");"改變了ball的reference,而這裏的ball來自Rollable interface,Rollable interface裏的ball是public static final的,final的object是不能被改變reference的。因此編譯器將在 ball = new Ball("Football"); 這裏顯示有錯。