1.psvm定義的意義
public:保證了方法的訪問許可權
static:保證在類未被例項化的時候就能呼叫(載入的時機)
void:不需要返回值
main:約定俗成的名字
String[] args:提供控制檯傳入的引數
2.程式碼塊
程式碼塊分為構造程式碼塊和靜態(類)程式碼塊、區域性程式碼塊。
構造程式碼塊(初始化塊,方法塊)隨著物件的建立而執行,在每次例項化物件時執行,且載入時機優先於建構函式。可以在構造程式碼塊中初始化成員變數和常量。(常量不賦初值會報錯,JVM不會給他賦初值,但是可以在構造程式碼塊中賦初值。)常量的宣告可以放在構造程式碼塊的後面。
靜態程式碼塊隨類載入而載入,順序上靜態程式碼塊優於構造程式碼塊,優於建構函式。
在類方法/靜態程式碼塊中不能使用this,因為static先載入,不可能有物件存在,所以也不可能有物件呼叫方法。靜態程式碼塊中可以初始化靜態變數和靜態常量。常量的初始化可以放在靜態構造塊下面。
區域性程式碼塊:在方法中出現。限定變數生命週期,及早釋放,提高記憶體利用率。
順序練習:
在沒有建構函式的情況下,靜態變數是按照語句順序執行賦值的。
在有建構函式的情況下,訪問物件的靜態變數,結果一定是建構函式中傳入的值。因為建構函式的順序在最後。
3.常用API
Math.random():返回【0.0,1.0)之間的浮點數。
System.currentTimeMillis():返回當前時間(距1970.01.01 0點的毫秒數)。經常用來計算一個方法的執行時間。
System.arraycopy(Object[] src,int srcPos,Object[] dest,int destPos,int length):從第一個陣列的起始位置複製到第二個陣列的起始位置,一共複製給定長度個元素。
Date date = new Date(); :返回當前時間的Date()物件。
date.getTime():返回毫秒數
SimpleDateFormat formatter = new SimpleDateFormat(格式);:建立一個格式器。
formatter.format(date);:格式化Date物件,返回一個格式化的字串。
formatter.parse(dateString); :解析字串,返回一個符合格式的Date。如果大於等於格式,就不會報錯。如果小於格式,就會報ParseException。
Calendar.getInstance() :返回一個Calendar物件。
calendar.get(欄位名);:返回日曆物件對應的欄位值。欄位名是Calandar類中定義的常量,注意Calandar.MONTH返回的是0-11月份,Calandar.HOUR_OF_DAY返回的是24小時制。
BigDecimal():建構函式的引數可以是int,字串,浮點數。BigDecimal可以對超過16位有效位的數進行精確運算。但是儲存浮點數時還是非精確的,所以建議用字串儲存。
bigDecimal.加減乘除();:devide()如果除以0會報ArithmeticException。
4.包裝類(八種基本資料型別的包裝類)
常用方法:
- 建構函式(已棄用)
- 基本型別Value() :返回基本型別
- 兩個數的最大值/最小值
- parse基本型別:將字串解析為基本型別,常用方法,如果含有不符合的符號會報NumberFormatException。
- toString():
- valueOf(基本型別/字串):返回包裝型別,和parse基本型別方法都是可以將字串轉換為基本型別的方法。
5.自動裝箱和自動拆箱(java5之後出現)
自動裝箱是指基本型別可以直接賦值為封裝型別。JVM自動完成型別轉換。自動裝箱的過程實際是底層呼叫量valueOf()這個方法。
自動拆箱指封裝型別可以直接賦值為基本型別。
6.包裝類的快取問題
以Integer類為例,valueOf()方法返回的是包裝類,而底層實現採用了快取機制。如果這個簡單型別在[-128,127]之間,就會使用IntegerCache的cache陣列中的物件進行返回(快取陣列,在[0,255]的下標中存放了每個物件)。
而Float和Double類的valueOf()方法沒有使用快取,直接new 了物件。 Integer s = new Integer(9) ;//分配堆記憶體,地址。Java不推薦,推薦使用自動裝箱的方法。
Integer t = new Integer(9) ;//分配堆記憶體,地址。 Long u = new Long(9) ;//分配堆記憶體,地址 // System.out.println(s==u);//Operator '==' cannot be applied to 'java.lang.Integer', 'java.lang.Long' System.out.println(s==t);//false,兩個不同地址的比較。 System.out.println(s.equals(t));//true System.out.println(s.equals(9));//true System.out.println(s.equals(new Integer(9)));
Integer a = 9;//相等於Integer a = Integer.valueOf(9); Integer b = 9; System.out.println(a==b);//true,兩個都是cache陣列的下標地址 a= 128; b= 128; System.out.println(a==b);//false,超過了快取範圍,new的新物件
Character c = 128;
Character d = 128;
System.out.println(c==d);//false,超過快取範圍
Character e = -1;//注意char類的範圍是0~2^16-1。
7.異常
異常是程式執行過程中出現的不正常情況。(開發中的語法錯誤和邏輯錯誤不屬於異常。)
異常分為:
- Error:JVM無法處理的嚴重問題。如記憶體錯誤,資源耗盡。
- Exception:因為程式設計錯誤和偶然原因出現的一般性問題。一般使用try-catch塊或throw、throws關鍵字處理。如果不處理異常,JVM會在控制檯列印堆疊資訊,並且程式會自動終止。
Exception分為:執行時異常和檢查時異常。(只有RunTimeException子類,沒有CheckedException子類)。
執行時異常是編譯器不要求強制處理的異常,通常指程式設計錯誤。有常見的ArithmeticException、ClassCastException(not instanceof時)、IndexOutOfBoundsException、NullPointerException。
編譯器異常是編譯器要求處理的異常,即一般性異常,如果不處理則程式不允許執行。
8.try-catch-finally 和return順序:
public class test { public int add(int a,int b) { try { return a+b; }catch(Exception e){ System.out.println("catch語句塊"); }finally { System.out.println("finally語句塊"); } return 0; } public static void main(String[] args) { test t=new test(); System.out.println("和是"+t.add(9, 34));//finally語句塊,和是43 } }
當try塊中有return語句,又有finally塊時,會先把try塊中的return 返回值儲存到一個棧中。當finally塊執行完時,再調出這個棧的內容返回。
public class test { public int add(int a,int b) { try { return a+b; }catch(Exception e){ System.out.println("catch語句塊"); }finally { System.out.println("finally語句塊"); a=1; } return 0; } public static void main(String[] args) { test t=new test(); System.out.println("和是"+t.add(9, 34));//finally語句塊 和是43 } }
在finally塊中又對a進行賦值,但是並沒有影響到棧中的內容,只改變了a的值,返回值沒有變。
如果catch塊中有return語句,finally塊中沒有return語句,情況也是類似的。
如果finally中也有return語句,最終會返回finally的返回語句。
try-catch塊中,可以有多個catch塊,但是隻能進入一個catch塊,並列catch塊可以是同級型別,如果有父類異常應該放在最後。catch塊捕捉的是異常物件。
9.throws/throw
throws丟擲的是異常的型別,丟擲異常可以是多個型別。
throws和throw的區別:
- 編寫的位置:throw在方法體中,throws在方法宣告上。
- 丟擲的型別:throw丟擲一個物件,throws丟擲的是異常的型別。
- 丟擲的個數:throw丟擲一個物件,throws可以丟擲多個型別。
10.自定義異常
自定義異常需要繼承Exception類,在有異常的方法體中寫throw 異常物件語句,並在這個方法的宣告上標註throws 異常型別。(標註的地方是在引數列表後面)。
自定義異常需要重寫無參構造和有參構造(String 異常資訊)。
11.練習
javac.exe:編譯.java原始檔為.class位元組碼檔案
java.exe:直譯器,通過java虛擬機器來裝載和執行編譯檔案(class檔案)的。Java直譯器是JVM的一部分。Java直譯器用來解釋執行Java編譯器編譯後的程式。java.exe可以簡單看成是Java直譯器。
javap.exe 類分析器 javap命令反彙編一個java位元組程式碼檔案, 返回有關可變部分和成員函式的資訊
javadoc.exe 是java文件生成器
形參/區域性變數不能用修飾符修飾的原因:java修飾符分為兩種,訪問控制修飾符和其他修飾符。
訪問修飾符是用來供其他物件/方法呼叫時用的,但是區域性變數只會有方法本身呼叫,已經設定了訪問許可權,所以不需要。
其他修飾符中,不能修飾變數的修飾符有abstract(方法和類)、synchronized(方法)。
對於static,被static修飾的東西的生命週期是和類一致,而區域性變數的宣告週期是和方法一致。
對於final,可以被final修飾,表示這個值是常量,只能被賦值一次。但是必須要賦初值,因為形參是不會被自動賦初值的。
(Java設定成員變數會被自動賦初值,因為成員變數的使用順序是不確定的,可以在方法後被賦值並使用,但是區域性變數的執行順序是確定的(僅在方法體內),所以java為了避免程式設計師出錯,必須給區域性變數賦初值。)
(總結來說,成員變數在程式中從來沒有手動賦值,程式不會出錯,而final常量和區域性變數從來沒有手動賦值,語法就會報錯,常量可以在方法塊、建構函式這些類載入時能執行到的地方賦值,區域性變數可以在訪問變數前賦值)
對於transient,是用來在序列化中修飾不需要被序列化的變數,使其不持久化。transient不能用來修飾區域性變數,因為它不是成員變數。
對於volatile,是用來在多執行緒中保證變數的值是最新的值。區域性變數不存在可見性問題。
對於形參是引用變數型別,我們首先知道雖然形參不能改變實參,但是通過可以修改地址中的物件。形參列表中三個變數都是引用型別。
對於String s,操作是s = s+"t"。對於變數和常量的拼接,返回的是StringBuilder.append()後,new String()的地址,所以改變了形參的值,沒有改變原來的物件(字串物件是不可變的)。因此實參s沒有發生改變。
對於StringBuffer s1,操作是s1.append("1")。StringBuffer物件是可變的,所以修改的是s1物件。實參s1發生了改變。
對於StringBuffer s2,操作時s2= new StringBuffer("C1");。沒有改變s2物件的值,修改的是形參的地址。所以實參s2沒有改變。
違反了區域性變數不能被生命週期為類的static修飾的道理。
類的初始化順序:
1. 父類靜態程式碼塊
2. 子類靜態程式碼塊
3. 父類普通程式碼塊
4. 父類建構函式
5. 子類普通程式碼塊
6. 子類建構函式
Parent x =new Child();是多型中向上轉型的體現,此時x.i是Parent的變數,x可以呼叫f()。x.f()是Child的方法,但是方法呼叫中用到的變數i,首先會查詢方法的區域性變數,然後是Child的成員變數,最後是Parent的成員變數。因此x.i=20,x.f()顯示30。Child x1 = (Child)x,是向下轉型的體現。x1可以呼叫f()和g()。編譯期就會確定x1的型別是Child,所以x1.i=30,x1.f()也是就近原則,顯示30。所以最終結果是20 30 20 20。
如果改為Parent x1 = (Child)x;儘管此時也完成了向下轉型,但是又向上轉為Parent。此時x1沒有g()方法,最後結果是20 30 20 30。
類方法中不可以使用this,因為類方法載入的時期中沒有物件存在;類方法中可以呼叫本類和其他類的類方法;類方法中可以建立物件,並呼叫這個的物件的例項方法。