總結Java對象被序列化的兩種方法
Java對象為什麼需要被序列化
- 序列化能夠將對象轉為二進制流,對象就可以方便的在網絡中被傳輸和保存。
實現序列化的方式
- 實現Serializable接口
- 實現Externalizable接口
**這兩個接口的區別是:**Serializable接口會自動給對象的所有屬性標記為可被序列化。而Externalizable接口默認不給任何屬性標記可被序列化,如果需要序列化,需要重寫兩個方法,分別是writeExternal()和readExternal(),然後在這兩個方法中標記需要被序列化的對象屬性。
實現這兩個接口,隻是表示該對象可以被序列化,真正的做序列化操作,需要ObjectOutputStream對象操作。接下來就用編碼的方式體現序列化。
先寫個序列化操作的工具類,用於實現序列化和反序列化。
import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; /** * 序列化操作工具類 * @author 楊33 * @date 2020/6/21 15:22 */ public class SerializeUtil { /** * 將對象轉成字節數組 * @param object 需要序列化的對象 * @return * @throws IOException */ public static byte[] serialize(Object object) throws IOException{ if(object == null){ return null; } ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(); ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteArrayOutputStream); objectOutputStream.writeObject(object); return byteArrayOutputStream.toByteArray(); } /** * 反序列化 * @param bytes 對象字節數組 * @throws IOException * @throws ClassNotFoundException */ public static Object unserialize(byte[] bytes) throws IOException, ClassNotFoundException{ ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(bytes); ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream); return objectInputStream.readObject(); } }
先來實現一個Serializable接口
/** * @author 楊33 * @date 2020/6/21 14:20 */ public class Owner implements Serializable { private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } }
測試下:
import java.io.IOException; /** * @author 楊33 * @date 2020/6/21 14:54 */ public class Demo { public static void main(String[] args) throws IOException, ClassNotFoundException { Owner owner = new Owner(); owner.setName("李四"); //序列化 byte[] serialize = SerializeUtil.serialize(owner); System.out.println("序列化的效果:" + serialize); //反序列化 owner = (Owner)SerializeUtil.unserialize(serialize); System.out.println("反序列化的效果:" + owner.getName()); } }
控制臺打印結果:
序列化的效果:[B@58ca40be
反序列化的效果:李四
如果這個name字段不需要被序列化,可以使用關鍵字transient修飾,比如:
private transient String name;
此時測試一下,name字段就不會被序列化,反序列化後拿到的值就為null。
序列化的效果:[B@4ca49360
反序列化的效果:null
再實現一個Externalizable接口
/** * @author 楊33 * @date 2020/6/21 14:20 */ public class Medium implements Externalizable { private String name; private String sex; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getSex() { return sex; } public void setSex(String sex) { this.sex = sex; } public void writeExternal(ObjectOutput out) throws IOException { out.writeObject(name); out.writeObject(sex); } public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException { name = (String) in.readObject(); sex = (String) in.readObject(); } }
測試下:
import java.io.IOException; /** * @author 楊33 * @date 2020/6/21 14:54 */ public class Demo { public static void main(String[] args) throws IOException, ClassNotFoundException { Medium medium = new Medium(); medium.setName("李四"); medium.setSex("女"); //序列化 byte[] serialize = SerializeUtil.serialize(medium); System.out.println("序列化的效果:" + serialize); //反序列化 medium = (Medium)SerializeUtil.unserialize(serialize); System.out.println("反序列化的效果:" + medium.getName()); System.out.println("反序列化的效果:" + medium.getSex()); } }
控制臺打印結果:
序列化的效果:[B@71d9a2ab
反序列化的效果:李四
反序列化的效果:女
如果字段sex不需要被序列化,那麼可以在方法writeExternal()和readExternal()中去掉設置sex字段的代碼。最後測試,sex字段不會被序列化,反序列化後拿到的值就為null。
序列化的效果:[B@746c2f2
反序列化的效果:李四
反序列化的效果:null
到此這篇關於總結Java對象被序列化的兩種方法的文章就介紹到這瞭,更多相關Java對象被序列化內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- java Object轉byte與byte轉Object方式
- Java設計模式之單例模式示例詳解
- Java基礎知識之ByteArrayOutputStream流的使用
- Java設計模式之原型模式詳解
- Java設計模式之java原型模式詳解