java序列化實現
1. 什麼是序列化,在java中如何實現序列化
一、什麼是序列化:x0dx0a序列化理解成「打碎」是可以的,不過在書本上的名詞就是將對象轉換成二進制。x0dx0ax0dx0a二、在java中如何實現序列化:x0dx0a首先我們要把准備要序列化類,實現 Serializabel介面x0dx0a例如:我們要Person類里的name和age都序列化x0dx0aimport java.io.Serializable;x0dx0ax0dx0apublic class Person implements Serializable { //本類可以序列化x0dx0a private String name ;x0dx0a private int age ;x0dx0a x0dx0a public Person(String name,int age){x0dx0a this.name = name ;x0dx0a this.age = age ;x0dx0a }x0dx0a public String toString(){x0dx0a return "姓名:" + this.name + ",年齡" + this.age ;x0dx0a }x0dx0a}x0dx0ax0dx0a然後:我們將name和age序列化(也就是把這2個對象轉為二進制,統族理解為「打碎」)x0dx0apackage org.lxh.SerDemo;x0dx0ax0dx0aimport java.io.File;x0dx0aimport java.io.FileOutputStream;x0dx0aimport java.io.ObjectOutputStream ;x0dx0ax0dx0apublic class ObjectOutputStreamDemo { //序列化x0dx0ax0dx0a public static void main(String[] args) throws Exception {x0dx0a//序列化後生成指定文件路徑x0dx0a File file = new File("D:" + File.separator + "person.ser") ; ObjectOutputStream oos = null ;x0dx0a//裝飾流(流)x0dx0a oos = new ObjectOutputStream(new FileOutputStream(file)) ; x0dx0a//實例化類x0dx0a Person per = new Person("張三",30) ; oos.writeObject(per) ;//把類對象序列化x0dx0a oos.close() ;x0dx0a }x0dx0a}
2. JAVA 壓縮和序列化
壓縮和序列化主要用在數據的存儲和傳輸上,二者都是由IO流相關知識實現,這里統一介紹下。
全部章節傳送門:
Java I/O類支持讀寫壓縮格式的數據流,你可以用他們對其他的I/O流進行封裝,以提供壓縮功能。
GZIP介面比較簡單,適合對單個數據流進行壓縮,在Linux系統中使用較多。
ZIP格式可以壓縮多個文件,而且可以和壓縮工具進行協作,是經常使用的壓縮方法。
JAR(Java Archive,Java 歸檔文件)是與平台無關的文件格式,它允許將許多文件組合成一個壓縮文件。為 J2EE 應用程序創建的 JAR 文件是 EAR 文件(企業 JAR 文件)。
JAR 文件格式以流行的 ZIP 文件格式為基礎。與 ZIP 文件不同的是,JAR 文件不僅用於壓縮和發布,而且還用於部署和封裝庫、組件和插件程序,並可被像編譯器和 JVM 這樣的工具直接使用。在 JAR 中包含特殊的文件,如 manifests 和部署描述符,用來指示工具如何處理特定的 JAR。
如果一個Web應用程序的目錄和文件非常多,那麼將這個Web應用程序部署到另一台機器上,就不是很方便了,我們可以將Web應用程序打包成Web 歸檔(WAR)文件,這個過程和把Java類文件打包成JAR文件的過程類似。利用WAR文件,可以把Servlet類文件和相關的資源集中在一起進行發布。在這個過程中,Web應用程序就不是按照目錄層次結構來進行部署了,而是把WAR文件作為部署單元來使用。
一個WAR文件就是一個Web應用程序,建立WAR文件,就是把整個Web應用程序(不包括Web應用程序層次結構的根目錄)壓縮起來,指定一個.war擴展名。下面我們將第2章的Web應用程序打包成WAR文件,然後發布
要注意的是,雖然WAR文件和JAR文件的文件格式是一樣的,並且都是使用jar命令來創建,但就其應用來說,WAR文件和JAR文件是有根本區別的。JAR文件的目的是把類和相關的資源封裝到壓縮的歸檔文件中,而對於WAR文件來說,一個WAR文件代表了一個Web應用程序,它可以包含 Servlet、HTML頁面、Java類、圖像文件,以及組成Web應用程序的其他資源,而不僅僅是類的歸檔文件。
在命令行輸入jar即可查看jar命令的使用方法。
把對象轉換為位元組序列的過程稱為對象的序列化。把位元組序列恢復為對象的過程稱為對象的反序列化。
對象的序列化主要有兩種用途:
java.io.ObjectOutputStream代表對象輸出流,它的writeObject(Object obj)方法可對參數指定的obj對象進行序列化,把得到的位元組序列寫到一個目標輸出流中。
java.io.ObjectInputStream代表對象輸入流,它的readObject()方法從一個源輸入流中讀取位元組序列,再把它們反序列化為一個對象,並將其返回。
只有實現了Serializable的對象才能被序列化。對象序列化包括如下步驟:
對象反序列化的步驟如下:
創建一個可以可以序列化的對象。
然後進行序列化和反序列化測試。
serialVersionUID: 字面意思上是序列化的版本號,凡是實現Serializable介面的類都有一個表示序列化版本標識符的靜態變數。
JAVA序列化的機制是通過判斷類的serialVersionUID來驗證的版本一致的。在進行反序列化時,JVM會把傳來的位元組流中的serialVersionUID於本地相應實體類的serialVersionUID進行比較。如果相同說明是一致的,可以進行反序列化,否則會出現反序列化版本一致的異常,即是InvalidCastException。
為了提高serialVersionUID的獨立性和確定性,強烈建議在一個可序列化類中顯示的定義serialVersionUID,為它賦予明確的值。
控制序列化欄位還可以使用Externalizable介面替代Serializable借口。此時需要定義一個默認構造器,否則將為得到一個異常(java.io.InvalidClassException: Person; Person; no valid constructor);還需要定義兩個方法(writeExternal()和readExternal())來控制要序列化的欄位。
如下為將Person類修改為使用Externalizable介面。
transient修飾符僅適用於變數,不適用於方法和類。在序列化時,如果我們不想序列化特定變數以滿足安全約束,那麼我們應該將該變數聲明為transient。執行序列化時,JVM會忽略transient變數的原始值並將默認值(引用類型就是null,數字就是0)保存到文件中。因此,transient意味著不要序列化。
靜態變數不是對象狀態的一部分,因此它不參與序列化。所以將靜態變數聲明為transient變數是沒有用處的。
3. 什麼是java序列化,如何實現java序列化
序 列 化: 指把堆內存中的Java對象數據,通過某種方式把對象存儲到磁碟文件中或者傳遞給其他網路的節點(在網路上傳輸).我們把這個過程稱之為序列化.
反序列化:把磁碟文件中的對象數據或者把網路節點上的對象數據,恢復成Java對象的過程.
為什麼要做序列化:
1):在分布式系統中,需要共享的數據的JavaBean對象,都得做序列化,此時需要把對象再網路上傳輸,此時就得把對象數據轉換為二進制形式.以後存儲在HttpSession中的對象,都應該實現序列化介面(只有實現序列化介面的類,才能做序列化操作).
2):服務鈍化:如果服務發現某些對象好久都沒有活動了,此時伺服器就會把這些內存中的對象,持久化在本地磁碟文件中(Java對象-->二進制文件).如果某些對象需要活動的時候,現在內存中去尋找,找到就使用,找不到再去磁碟文件中,反序列化我們得對象數據,恢復成Java對象.
需要做序列化的對象的類,必須實現序列化介面:java.io.Serializable介面(標志介面[沒有抽象方法]).
底層會判斷,如果當前對象是Serializable的實例,才允許做序列化. boolean ret = Java對象 instanceof Serializable;
在Java中大多數類都已經實現Serializable介面.
4. 什麼是java序列化,如何實現序列化
簡單來說,如果你只是自己同一台機器的同一個環境下使用同一個JVM來操作,序列化和反序列化是沒必要的,當需要進行數據傳輸的時候就顯得十分必要。比如你的數據寫到文件里要被其他人的電腦的程序使用,或者你電腦上的數據需要通過網路傳輸給其他人的程序使用,像伺服器客戶端的這種模型就是一種應用,這個時候,大家想想,每個人的電腦配置可能不同,運行環境可能也不同,位元組序可能也不同,總之很多地方都不能保證一致,所以為了統一起見,我們傳輸的數據或者經過文件保存的數據需要經過序列化和編碼等操作,相當於交互雙方有一個公共的標准,按照這種標准來做,不管各自的環境是否有差異,各自都可以根據這種標准來翻譯出自己能理解的正確的數據。
除了在持久化對象時會用到對象序列化之外,當使用RMI(遠程方法調用),或在網路中傳遞對象時,都會用到對象序列化。實現序列化:
1 首先要把准備要序列化類,實現 Serializabel介面
importjava.io.Serializable;
{
//本類可以序列化
privateStringname;
privateintage;
publicPerson(Stringname,intage)
{
this.name=name;this.age=age;
}
publicStringtoString()
{
return"姓名:"+this.name+",年齡"+this.age;
}
}
2 序列化
packageorg.lxh.SerDemo;
importjava.io.File;
importjava.io.FileOutputStream;
importjava.io.ObjectOutputStream;
{
//序列化
publicstaticvoidmain(String[]args)throwsException
{
//序列化後生成指定文件路徑
Filefile=newFile("D:"+File.separator+"person.ser");
ObjectOutputStreamoos=null;
//裝飾流(流)
oos=newObjectOutputStream(newFileOutputStream(file));
//實例化類
Personper=newPerson("張三",30);
oos.writeObject(per);
//把類對象序列化
oos.close();
}
}
需要注意的是:
靜態成員是不能被序列化的,因為靜態成員是隨著類的載入而載入的,與類共存亡,並且靜態成員的默認初始值都是0;就是說靜態成員屬於類級別的,所以不能序列化,序列化只是序列化了對象而已。