1 JAVA程序改錯 2 1. 3 abstract class Name { 4 private String name; 5 public abstract boolean isStupidName(String name) { 6 } 7 } 8 答案: 錯。abstract method必須以分號結尾,且不帶花括號。 9 2. 10 public class Something { 11 void doSomething () { 12 private String s = ""; 13 int l = s.length(); 14 } 15 } 16 答案: 錯。局部變量前不能放置任何訪問修飾符 (private,public,和protected)。final可以用來修飾局部變量 17 3. 18 abstract class Something { 19 private abstract String doSomething (); 20 21 } 22 答案: 錯。abstract的methods不能以private修飾。abstract的methods就是讓子類implement(實現)具體細節的,怎么可以用private把abstract method封鎖起來呢?
(同理,abstract method前不能加final)。 23 4. 24 public class Something { 25 public int addOne(final int x) { 26 return ++x; 27 } 28 29 } 30 答案: 錯。int x被修飾成final,意味着x不能在addOne method中被修改。 31 5. 32 public class Something { 33 public static void main(String[] args) { 34 Other o = new Other(); 35 new Something().addOne(o); 36 37 } 38 public void addOne(final Other o) { 39 o.i++; 40 41 wk_ad_begin({pid : 21}); 42 wk_ad_after(21, function(){ 43 $('.ad-hidden').hide(); 44 }, function(){ 45 $('.ad-hidden').show();}); 46 47 2 48 } } 49 class Other { 50 public int i; 51 } 52 和上面的很相似,都是關於final的問題,這有錯嗎? 53 答案: 正確。在addOne method中,參數o被修飾成final。如果在addOne method里我們修改了o的reference(比如: o = new Other();),那么如同上例這題也是錯的。
但這里修改的是o的member vairable(成員變量),而o的reference並沒有改變。 54 6. 55 class Something { 56 int i; 57 public void doSomething() { 58 System.out.println("i = " + i); 59 } 60 } 61 有什么錯呢? 62 答案: 正確。輸出的是"i = 0"。int i屬於instant variable (實例變量,或叫成員變量)。instant variable有default value。int的default value是0。 63 7. 64 class Something { 65 final int i; 66 public void doSomething() { 67 System.out.println("i = " + i); 68 69 } 70 71 } 72 和上面一題只有一個地方不同,就是多了一個final。這難道就錯了嗎? 73 74 75 答案: 錯。final int i是個final的instant variable (實例變量,或叫成員變量)。
final的instant variable沒有default value,必須在constructor (構造器)結束之前被賦予一個明確的值。可以修改為"final int i = 0;"。 76 8. 77 public class Something { 78 public static void main(String[] args) { 79 Something s = new Something(); 80 System.out.println("s.doSomething() returns " + 81 doSomething()); 82 83 } 84 public String doSomething() { 85 86 return "Do something ..."; 87 88 } 89 90 3 91 } 92 看上去很完美。 93 94 答案: 錯。看上去在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。 95 96 9. 97 此處,Something類的文件名叫OtherThing.java class Something { 98 private static void main(String[] something_to_do) {
System.out.println("Do something ..."); 99 100 } 101 } 102 這個好像很明顯。 103 答案: 正確。從來沒有人說過Java的Class名字必須和其文件名相同。但public class的名字必須和文件名相同。 104 105 10. 106 interface A{ int x = 0; } 107 class B{ int x =1; } 108 class C extends B implements A { 109 110 public void pX(){ 111 112 System.out.println(x); 113 114 } 115 116 public static void main(String[] args) { 117 118 new C().pX(); 119 120 } 121 122 } 123 答案:錯誤。在編譯時會發生錯誤(錯誤描述不同的JVM有不同的信息,意思就是未明確的x調用,兩個x都匹配(就像在同時import java.util和java.sql兩個包時直接聲明Date一樣)。
對於父類的變量,可以用super.x來明確,而接口的屬性默認隱含為 public static final.所以可以通過A.x來明確。 124 125 11. 126 interface Playable { 127 void play(); 128 129 } 130 interface Bounceable { 131 132 4 133 void play(); 134 135 } 136 137 interface Rollable extends Playable, Bounceable { 138 139 Ball ball = new Ball("PingPang"); 140 141 } 142 class Ball implements Rollable { 143 144 private String name; 145 146 public String getName() { 147 148 return name; 149 150 } 151 public Ball(String name) { 152 153 this.name = name; 154 155 } 156 public void play() { 157 158 ball = new Ball("Football"); 159 160 System.out.println(ball.getName()); 161 162 } 163 164 } 165 這個錯誤不容易發現。 166 答案: 錯。"interface Rollable extends Playable, Bounceable"沒有問題。interface可繼承多個interfaces,所以這里沒錯。問題出在: 167 interface Rollable里的"Ball ball = new Ball("PingPang");"。 168 任何在interface里聲明的interface variable (接口變量,也可稱成員變量),默認為public static final。也就是說: 169 Ball ball = new Ball("PingPang");實際上是 170 public static final Ball ball = new Ball("PingPang");。 171 在Ball類的Play()方法中,"ball = new Ball("Football");"
改變了ball的reference,而這里的ball來自Rollable interface,Rollable interface里的ball是public static final的,final的object是不能被改變reference的。
因此編譯器將在 ball = new Ball("Football"); 這里顯示有錯。