流的本質(zhì)是數(shù)據(jù)傳輸,可以被看作一組有序的字節(jié)集合者春,跟據(jù)處理數(shù)據(jù)類(lèi)型的不同栅隐,流可以分為兩大類(lèi):字節(jié)流和字符流。字節(jié)流以字節(jié)(8bit)為單位屈糊,包含兩個(gè)抽象類(lèi):InputStream(輸入流)和OutPutStream(輸出流)的榛。字符流以字符(16bit)為單位,根據(jù)碼表映射字符逻锐,一次可以讀多個(gè)字節(jié)夫晌,它包含兩個(gè)抽象類(lèi):Reader(輸入流)和Writer(輸出流)雕薪。字節(jié)流和字符流的主要區(qū)別為:字節(jié)流在處理輸出時(shí)不會(huì)用到緩存,而字符流用到了緩存晓淀。Java類(lèi)在設(shè)計(jì)時(shí)采用了Decorator(裝飾者)模式所袁。
使用這種設(shè)計(jì)模式的好處是可以在運(yùn)行時(shí)動(dòng)態(tài)地給對(duì)象添加一些額外的職責(zé),與使用繼承的設(shè)計(jì)方法相比凶掰,該方法具有很好的靈活性燥爷。例如,要設(shè)計(jì)一個(gè)輸入流的類(lèi)懦窘,用來(lái)在讀文件時(shí)把文件中大寫(xiě)字母換成小寫(xiě)字母前翎,把小寫(xiě)字母轉(zhuǎn)換成大寫(xiě)字母。在設(shè)計(jì)時(shí)畅涂,可以通過(guò)繼承抽象類(lèi)裝飾者類(lèi)(FilterInputStream)來(lái)實(shí)現(xiàn)一個(gè)裝飾類(lèi)港华,通過(guò)調(diào)用InputStream類(lèi)或其子類(lèi)提供的一些方法再加上邏輯判斷。
class MyInputStream extends FilterInputStream{
public MyInputStream(InputStream in){
super(in);
}
public int read() throws IOException{
int c = 0;
if((c = super.read())!= -1) {
if(Character.isLowerCase((char) c)){
return Character.toUpperCase((char) c);
}
else if(Character.isLowerCase((char) c)){
return Character.toUpperCase((char) c);
}
else
return c;
} else
return -1;
}
}
public class Test{
public static void main(String []args) {
int c;
try{
InputStream is = new MyInputStream(new FileInputStream("text.txt"));
while((c = is.read()) >= 0) {
System.out.println(c);
}
}catch(IOException) {
System.out.println(e.getMessage());
}
}
}
Java Socket
網(wǎng)絡(luò)上的兩個(gè)程序通過(guò)一個(gè)雙向的通信連接實(shí)現(xiàn)數(shù)據(jù)的交換午衰,這個(gè)雙向鏈路的一端稱(chēng)為一個(gè)Socket(也稱(chēng)套接字)立宜,在Java中,Socket可以分為兩種類(lèi)型:面向連接的Socket通信協(xié)議(TCP)臊岸,和面向無(wú)連接的Socket通信協(xié)議(UDP)赘理。任何一個(gè)Socket都是有IP地址和端口號(hào)唯一確定。
基于TCP的通信過(guò)程:
- Server(服務(wù)器)Listen(監(jiān)聽(tīng))指定的某個(gè)端口(建議使用大于端口)是否有連接請(qǐng)求扇单;
- Client(客戶端)向Server端發(fā)出連接請(qǐng)求商模;
- Server端向Client端發(fā)回Accept(接受)消息。
示例:
Server端:
class Server{
public static void main(String []args){
BufferedReader br = null;
PrintWriter pw = null;
try{
ServerSocket server = new ServerSocket(2000);
Socket socket = server.accept();
br = new BufferedReader(new InputStreamReader(socket.getInputStream()));
pw = new PrintWriter(socket.getOutputStream(),true);
String s = br.readLine();
pw.println(s);
}catch(Exception e){
e.printStackTrace();
}finally{
try{
br.close();
pw.close();
}catch(Exception e){
e.printStackTrace();
}
}
}
}
客戶端程序:
class Client{
public static void main(String []args){
bufferedReader br = null;
PrintWriter pw = null;
try{
Socket socket = new Socket("localhost", 2000);
br = new BufferedReader(new InputStream(socket.getInputStream()));
pw = new PrintWriter(socket.getOutputStream(), true);
pw.println("hello"); //向服務(wù)器發(fā)送數(shù)據(jù)
String s = null;
while(true){
s = br.readLine();
if(s != null) break;
}
System.out.println(s);
}catch(Exception e){
e.printStackTrace();
}finally{
try{
br.close();
pw.close();
}catch(Exception e){
e.printStackTrace();
}
}
}
}
Java序列化
序列化石一種將對(duì)象以一連串的字符描述的過(guò)程蜘澜,用于解決在對(duì)對(duì)象流進(jìn)行讀寫(xiě)操作時(shí)所引發(fā)的問(wèn)題施流。序列化可以將對(duì)象的狀態(tài)寫(xiě)在流里進(jìn)行網(wǎng)絡(luò)傳輸,或者保存到文件鄙信、數(shù)據(jù)庫(kù)等系統(tǒng)里瞪醋,并在需要時(shí)把該流讀取出來(lái)重新構(gòu)造一個(gè)相同的對(duì)象。
序列化有以下兩個(gè)特點(diǎn):
- 如果一個(gè)類(lèi)能被序列化装诡,那么它的子類(lèi)也能被序列化银受;
- 由于static代表類(lèi)的成員,transient(Java關(guān)鍵字)代表臨時(shí)數(shù)據(jù)鸦采,因此被聲明為這兩種類(lèi)型的數(shù)據(jù)成員是不能被序列化的宾巍。
Java提供了多個(gè)對(duì)象序列化的接口,包括ObjectOutput渔伯、ObjectInput顶霞,ObjectOutputStream和ObjectInputStream。
在以下情況下需要使用該序列化:
- 需要通過(guò)網(wǎng)絡(luò)來(lái)發(fā)送對(duì)象锣吼,或?qū)ο蟮臓顟B(tài)需要被持久化到數(shù)據(jù)庫(kù)或文件中选浑。
- 序列化能實(shí)現(xiàn)深復(fù)制蓝厌,即可以復(fù)制引用的對(duì)象。