JavaIO字符操作和對象操作示例詳解
字符操作
編碼與解碼
編碼就是把字符轉(zhuǎn)換為字節(jié),而解碼是把字節(jié)重新組合成字符。
如果編碼和解碼過程使用不同的編碼方式那么就出現(xiàn)了亂碼。
- GBK 編碼中,中文字符占 2 個字節(jié),英文字符占 1 個字節(jié);
- UTF-8 編碼中,中文字符占 3 個字節(jié),英文字符占 1 個字節(jié);
- UTF-16be 編碼中,中文字符和英文字符都占 2 個字節(jié)。
UTF-16be 中的 be 指的是 Big Endian,也就是大端。相應地也有 UTF-16le,le 指的是 Little Endian,也就是小端。
Java 的內(nèi)存編碼使用雙字節(jié)編碼 UTF-16be,這不是指 Java 只支持這一種編碼方式,而是說 char 這種類型使用 UTF-16be 進行編碼。char 類型占 16 位,也就是兩個字節(jié),Java 使用這種雙字節(jié)編碼是為了讓一個中文或者一個英文都能使用一個 char 來存儲。
String 的編碼方式
String 可以看成一個字符序列,可以指定一個編碼方式將它編碼為字節(jié)序列,也可以指定一個編碼方式將一個字節(jié)序列解碼為 String。
String str1 = "中文";
byte[] bytes = str1.getBytes("UTF-8");
String str2 = new String(bytes, "UTF-8");
System.out.println(str2);
在調(diào)用無參數(shù) getBytes() 方法時,默認的編碼方式不是 UTF-16be。雙字節(jié)編碼的好處是可以使用一個 char 存儲中文和英文,而將 String 轉(zhuǎn)為 bytes[] 字節(jié)數(shù)組就不再需要這個好處,因此也就不再需要雙字節(jié)編碼。getBytes() 的默認編碼方式與平臺有關,一般為 UTF-8。
byte[] bytes = str1.getBytes();
Reader 與 Writer
不管是磁盤還是網(wǎng)絡傳輸,最小的存儲單元都是字節(jié),而不是字符。但是在程序中操作的通常是字符形式的數(shù)據(jù),因此需要提供對字符進行操作的方法。
- InputStreamReader 實現(xiàn)從字節(jié)流解碼成字符流;
- OutputStreamWriter 實現(xiàn)字符流編碼成為字節(jié)流。
實現(xiàn)逐行輸出文本文件的內(nèi)容
public static void readFileContent(String filePath) throws IOException {
FileReader fileReader = new FileReader(filePath);
BufferedReader bufferedReader = new BufferedReader(fileReader);
String line;
while ((line = bufferedReader.readLine()) != null) {
System.out.println(line);
}
// 裝飾者模式使得 BufferedReader 組合了一個 Reader 對象
// 在調(diào)用 BufferedReader 的 close() 方法時會去調(diào)用 Reader 的 close() 方法
// 因此只要一個 close() 調(diào)用即可
bufferedReader.close();
}
對象操作
序列化
序列化就是將一個對象轉(zhuǎn)換成字節(jié)序列,方便存儲和傳輸。
- 序列化:ObjectOutputStream.writeObject()
- 反序列化:ObjectInputStream.readObject()
不會對靜態(tài)變量進行序列化,因為序列化只是保存對象的狀態(tài),靜態(tài)變量屬于類的狀態(tài)。
Serializable
序列化的類需要實現(xiàn) Serializable 接口,它只是一個標準,沒有任何方法需要實現(xiàn),但是如果不去實現(xiàn)它的話而進行序列化,會拋出異常。
public static void main(String[] args) throws IOException, ClassNotFoundException {
A a1 = new A(123, "abc");
String objectFile = "file/a1";
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream(objectFile));
objectOutputStream.writeObject(a1);
objectOutputStream.close();
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(objectFile));
A a2 = (A) objectInputStream.readObject();
objectInputStream.close();
System.out.println(a2);
}
private static class A implements Serializable {
private int x;
private String y;
A(int x, String y) {
this.x = x;
this.y = y;
}
@Override
public String toString() {
return "x = " + x + " " + "y = " + y;
}
}
transient
transient 關鍵字可以使一些屬性不會被序列化。
ArrayList 中存儲數(shù)據(jù)的數(shù)組 elementData 是用 transient 修飾的,因為這個數(shù)組是動態(tài)擴展的,并不是所有的空間都被使用,因此就不需要所有的內(nèi)容都被序列化。通過重寫序列化和反序列化方法,使得可以只序列化數(shù)組中有內(nèi)容的那部分數(shù)據(jù)。
private transient Object[] elementData;
以上就是JavaIO字符操作和對象操作示例詳解的詳細內(nèi)容,更多關于JavaIO字符操作對象操作的資料請關注腳本之家其它相關文章!
相關文章
使用ScheduledThreadPoolExecutor踩過最痛的坑
這篇文章主要介紹了使用ScheduledThreadPoolExecutor踩過最痛的坑及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-08-08
try catch finally的執(zhí)行順序深入分析
首先執(zhí)行try,如果有異常執(zhí)行catch,無論如何都會執(zhí)行finally,當有return以后,函數(shù)就會把這個數(shù)據(jù)存儲在某個位置,然后告訴主函數(shù),我不執(zhí)行了,接下來你執(zhí)行吧,所以函數(shù)就會推出2013-09-09

