国产av日韩一区二区三区精品,成人性爱视频在线观看,国产,欧美,日韩,一区,www.成色av久久成人,2222eeee成人天堂

首頁(yè) Java Java入門(mén) java如何序列化

java如何序列化

Nov 12, 2019 pm 05:44 PM
java 序列化

java如何序列化

一、序列化與反序列化

序列化:指堆內(nèi)存中的java對(duì)象數(shù)據(jù),通過(guò)某種方式把對(duì)存儲(chǔ)到磁盤(pán)文件中,或者傳遞給其他網(wǎng)絡(luò)節(jié)點(diǎn)(網(wǎng)絡(luò)傳輸)。這個(gè)過(guò)程稱為序列化,通常是指將數(shù)據(jù)結(jié)構(gòu)或?qū)ο筠D(zhuǎn)化成二進(jìn)制的過(guò)程。

即將對(duì)象轉(zhuǎn)化為二進(jìn)制,用于保存,或者網(wǎng)絡(luò)傳輸。

反序列化:把磁盤(pán)文件中的對(duì)象數(shù)據(jù)或者把網(wǎng)絡(luò)節(jié)點(diǎn)上的對(duì)象數(shù)據(jù),恢復(fù)成Java對(duì)象模型的過(guò)程。也就是將在序列化過(guò)程中所生成的二進(jìn)制串轉(zhuǎn)換成數(shù)據(jù)結(jié)構(gòu)或者對(duì)象的過(guò)程

與序列化相反,將二進(jìn)制轉(zhuǎn)化成對(duì)象。

二、序列化的作用

① 想把內(nèi)存中的對(duì)象保存到一個(gè)文件中或者數(shù)據(jù)庫(kù)中時(shí)候;

② 想用套接字在網(wǎng)絡(luò)上傳送對(duì)象的時(shí)候;

③ 想通過(guò)RMI傳輸對(duì)象的時(shí)候

一些應(yīng)用場(chǎng)景,涉及到將對(duì)象轉(zhuǎn)化成二進(jìn)制,序列化保證了能夠成功讀取到保存的對(duì)象。

三、java的序列化實(shí)現(xiàn)

要實(shí)現(xiàn)對(duì)象的序列化,最直接的操作就是實(shí)現(xiàn)Serializable接口

使用IO流中的對(duì)象流可以實(shí)現(xiàn)序列化操作,將對(duì)象保存到文件,再讀取出來(lái)。

首先創(chuàng)建一個(gè)對(duì)象,并實(shí)現(xiàn)Serializable接口:

import java.io.Serializable;
public class User implements Serializable{
    private static final long serialVersionUID = 1L;
    private String name;
    private int age;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    @Override
    public String toString() {
        return "User [name=" + name + ", age=" + age + "]";
    }
}

用對(duì)象流寫(xiě)一個(gè)保存對(duì)象與讀取對(duì)象的工具類:

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
public class SerializeUtil {
    // 保存對(duì)象,序列化
    public static void saveObject(Object object) throws Exception {
        ObjectOutputStream out = null;
        FileOutputStream fout = null;
        try {
            fout = new FileOutputStream("D:/1.txt");
            out = new ObjectOutputStream(fout);
            out.writeObject(object);
        } finally {
            fout.close();
            out.close();
        }
    }
    // 讀取對(duì)象,反序列化
    public static Object readObject() throws Exception {
        ObjectInputStream in = null;
        FileInputStream fin = null;
        try {
            fin = new FileInputStream("D:/1.txt");
            in = new ObjectInputStream(fin);
            Object object = in.readObject();
            return object;
        } finally {
            fin.close();
            in.close();
        }
    }
}

測(cè)試:

public class Main {
    public static void main(String[] args) {
        User user = new User();
        user.setName("旭旭寶寶");
        user.setAge(33);
        // 保存
        try {
            SerializeUtil.saveObject(user);
        } catch (Exception e) {
            System.out.println("保存時(shí)異常:" + e.getMessage());
        }
        // 讀取
        User userObject;
        try {
            userObject = (User) SerializeUtil.readObject();
            System.out.println(userObject);
        } catch (Exception e) {
            System.out.println("讀取時(shí)異常:" + e.getMessage());
        }
    }
}

測(cè)試結(jié)果:

1d23a791ec83fe88511f5a028bec8ba.png

這里我們成功的進(jìn)行了一次將對(duì)象保存到文件中,再讀取了出來(lái)。如果此時(shí),我們不實(shí)現(xiàn)序列化接口,就會(huì)出現(xiàn)異常了。我們?nèi)∠麑?shí)現(xiàn)的Serialiable接口代碼:

public class User {
    private String name;
    private int age;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    @Override
    public String toString() {
        return "User [name=" + name + ", age=" + age + "]";
    }
}

再測(cè)試Main方法:

61ce82282a06d248ef4f6356006238d.png

可以看到此時(shí)報(bào)錯(cuò)了,此時(shí)使用 e.printStackTrace();查看異常的詳細(xì)信息:

0f19ecf6b56cf353c16a440818a5632.png

可以看到Unknown Source,因?yàn)闆](méi)有進(jìn)行序列化,所以無(wú)法進(jìn)行保存與讀取。

四、序列化ID的作用

可以看到,我們?cè)谶M(jìn)行序列化時(shí),加了一個(gè)serialVersionUID字段,這便是序列化ID

private static final long serialVersionUID = 1L;

這個(gè)序列化ID起著關(guān)鍵的作用,它決定著是否能夠成功反序列化!java的序列化機(jī)制是通過(guò)判斷運(yùn)行時(shí)類的serialVersionUID來(lái)驗(yàn)證版本一致性的,在進(jìn)行反序列化時(shí),JVM會(huì)把傳進(jìn)來(lái)的字節(jié)流中的serialVersionUID與本地實(shí)體類中的serialVersionUID進(jìn)行比較,如果相同則認(rèn)為是一致的,便可以進(jìn)行反序列化,否則就會(huì)報(bào)序列化版本不一致的異常。

即序列化ID是為了保證成功進(jìn)行反序列化

五、默認(rèn)的序列化ID

當(dāng)我們一個(gè)實(shí)體類中沒(méi)有顯式的定義一個(gè)名為“serialVersionUID”、類型為long的變量時(shí),Java序列化機(jī)制會(huì)根據(jù)編譯時(shí)的class自動(dòng)生成一個(gè)serialVersionUID作為序列化版本比較,這種情況下,只有同一次編譯生成的class才會(huì)生成相同的serialVersionUID。譬如,當(dāng)我們編寫(xiě)一個(gè)類時(shí),隨著時(shí)間的推移,我們因?yàn)樾枨蟾膭?dòng),需要在本地類中添加其他的字段,這個(gè)時(shí)候再反序列化時(shí)便會(huì)出現(xiàn)serialVersionUID不一致,導(dǎo)致反序列化失敗。那么如何解決呢?便是在本地類中添加一個(gè)“serialVersionUID”變量,值保持不變,便可以進(jìn)行序列化和反序列化。

如果沒(méi)有顯示指定serialVersionUID,會(huì)自動(dòng)生成一個(gè)。
只有同一次編譯生成的class才會(huì)生成相同的serialVersionUID。
但是如果出現(xiàn)需求變動(dòng),Bean類發(fā)生改變,則會(huì)導(dǎo)致反序列化失敗。為了不出現(xiàn)這類的問(wèn)題,所以我們最好還是顯式的指定一個(gè)
serialVersionUID。

六、序列化的其他問(wèn)題

1、靜態(tài)變量不會(huì)被序列化( static,transient)

2、當(dāng)一個(gè)父類實(shí)現(xiàn)序列化,子類自動(dòng)實(shí)現(xiàn)序列化,不需要顯式實(shí)現(xiàn)Serializable接口。

3、當(dāng)一個(gè)對(duì)象的實(shí)例變量引用其他對(duì)象,序列化該對(duì)象時(shí)也把引用對(duì)象進(jìn)行序列化。

子類序列化時(shí):
如果父類沒(méi)有實(shí)現(xiàn)Serializable接口,沒(méi)有提供默認(rèn)構(gòu)造函數(shù),那么子類的序列化會(huì)出錯(cuò);
如果父類沒(méi)有實(shí)現(xiàn)Serializable接口,提供了默認(rèn)的構(gòu)造函數(shù),那么子類可以序列化,父類的成員變量不會(huì)被序列化。如果父類
實(shí)現(xiàn)了Serializable接口,則父類和子類都可以序列化。

七、使用效率更高的序列化框架—Protostuff

其實(shí)java的原生序列化方式(通過(guò)實(shí)現(xiàn)Serialiable接口),效率并不是最高的。

github上有一個(gè)分析序列化效率的項(xiàng)目:https://github.com/eishay/jvm-serializers/wiki

0fa6fad1fded2e2d3d4e3f650d7854f.png

其中看的出來(lái)性能最優(yōu)的為google開(kāi)發(fā)的colfer ,但是由于colfer的使用難度太大,而更多的都是使用protostuff序列化框架。適用改框架要引入兩個(gè)庫(kù)(core與runtime)。

①github地址:https://github.com/protostuff/protostuff

③如果用Maven,則添加依賴:

<dependency>
  <groupId>io.protostuff</groupId>
  <artifactId>protostuff-core</artifactId>
  <version>1.5.9</version>
</dependency>
<dependency>
  <groupId>io.protostuff</groupId>
  <artifactId>protostuff-core</artifactId>
  <version>1.5.9</version>
</dependency>

修改Main代碼

import com.dyuproject.protostuff.LinkedBuffer;
import com.dyuproject.protostuff.ProtobufIOUtil;
import com.dyuproject.protostuff.ProtostuffIOUtil;
import com.dyuproject.protostuff.Schema;
import com.dyuproject.protostuff.runtime.RuntimeSchema;
public class Main {
    public static void main(String[] args) {
        User user = new User();
        user.setName("旭旭寶寶");
        user.setAge(33);
        Schema<User> schema = RuntimeSchema.getSchema(User.class);
        // 保存對(duì)象,序列化,轉(zhuǎn)化二進(jìn)制數(shù)據(jù)
        LinkedBuffer buffer = LinkedBuffer.allocate(512);
        final byte[] protostuff;
        try {
            protostuff = ProtobufIOUtil.toByteArray(user, schema, buffer);
        } finally {
            buffer.clear();
        }
        // 讀取對(duì)象,反序列化
        User userObject = schema.newMessage();
        ProtostuffIOUtil.mergeFrom(protostuff, userObject, schema);
        System.out.println(userObject);
    }
}

User類,并未實(shí)現(xiàn)Serializable接口

public class User {
    private String name;
    private int age;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    @Override
    public String toString() {
        return "User [name=" + name + ", age=" + age + "]";
    }
}

測(cè)試結(jié)果:

07078b0e6c791b4a32b1650ae800811.png

若要要整合Redis使用,也可以寫(xiě)成一個(gè)工具類:

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import com.dyuproject.protostuff.LinkedBuffer;
import com.dyuproject.protostuff.ProtobufIOUtil;
import com.dyuproject.protostuff.Schema;
import com.dyuproject.protostuff.runtime.RuntimeSchema;
public class SerializeUtil {
    private static Map<Class<?>, Schema<?>> cachedSchema = new ConcurrentHashMap<>();
    @SuppressWarnings("unchecked")
    public static <T> byte[] serializer(T obj) {
        Class<T> clazz = (Class<T>) obj.getClass();
        Schema<T> schema = getSchema(clazz);
        return ProtobufIOUtil.toByteArray(obj, schema, LinkedBuffer.allocate(256));
    }
    public static <T> T deSerializer(byte[] bytes, Class<T> clazz) {
        T message;
        try {
            message = clazz.newInstance();
        } catch (InstantiationException | IllegalAccessException e) {
            throw new RuntimeException(e);
        }
        Schema<T> schema = getSchema(clazz);
        ProtobufIOUtil.mergeFrom(bytes, message, schema);
        return message;
    }
    @SuppressWarnings("unchecked")
    public static <T> Schema<T> getSchema(Class<T> clazz) {
        Schema<T> schema = (Schema<T>) cachedSchema.get(clazz);
        if (schema == null) {
            schema = RuntimeSchema.createFrom(clazz);
            if (schema != null) {
                cachedSchema.put(clazz, schema);
            }
        }
        return schema;
    }
}

這樣即使我們的User類就不用再實(shí)現(xiàn)Serialiable接口了,同樣可以進(jìn)行序列化,效率也更高。

php中文網(wǎng),大量的免費(fèi)Java入門(mén)教程,歡迎在線學(xué)習(xí)!

以上是java如何序列化的詳細(xì)內(nèi)容。更多信息請(qǐng)關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

本站聲明
本文內(nèi)容由網(wǎng)友自發(fā)貢獻(xiàn),版權(quán)歸原作者所有,本站不承擔(dān)相應(yīng)法律責(zé)任。如您發(fā)現(xiàn)有涉嫌抄襲侵權(quán)的內(nèi)容,請(qǐng)聯(lián)系admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費(fèi)脫衣服圖片

Undresser.AI Undress

Undresser.AI Undress

人工智能驅(qū)動(dòng)的應(yīng)用程序,用于創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用于從照片中去除衣服的在線人工智能工具。

Clothoff.io

Clothoff.io

AI脫衣機(jī)

Video Face Swap

Video Face Swap

使用我們完全免費(fèi)的人工智能換臉工具輕松在任何視頻中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費(fèi)的代碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

功能強(qiáng)大的PHP集成開(kāi)發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺(jué)化網(wǎng)頁(yè)開(kāi)發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級(jí)代碼編輯軟件(SublimeText3)

熱門(mén)話題

Java中的單例設(shè)計(jì)模式是什么? Java中的單例設(shè)計(jì)模式是什么? Jul 09, 2025 am 01:32 AM

單例設(shè)計(jì)模式在Java中通過(guò)私有構(gòu)造器和靜態(tài)方法確保一個(gè)類只有一個(gè)實(shí)例并提供全局訪問(wèn)點(diǎn),適用于控制共享資源的訪問(wèn)。實(shí)現(xiàn)方式包括:1.懶加載,即首次請(qǐng)求時(shí)才創(chuàng)建實(shí)例,適用于資源消耗大且不一定需要的情況;2.線程安全處理,通過(guò)同步方法或雙重檢查鎖定確保多線程環(huán)境下只創(chuàng)建一個(gè)實(shí)例,并減少性能影響;3.餓漢式加載,在類加載時(shí)直接初始化實(shí)例,適合輕量級(jí)對(duì)象或可接受提前初始化的場(chǎng)景;4.枚舉實(shí)現(xiàn),利用Java枚舉天然支持序列化、線程安全及防止反射攻擊的特性,是推薦的簡(jiǎn)潔可靠方式。不同實(shí)現(xiàn)方式可根據(jù)具體需求選

什么是Java中的螺紋插座? 什么是Java中的螺紋插座? Jul 09, 2025 am 02:25 AM

ThreadLocal在Java中用于創(chuàng)建線程私有變量,每個(gè)線程擁有獨(dú)立副本,避免并發(fā)問(wèn)題。其通過(guò)線程內(nèi)部的ThreadLocalMap存儲(chǔ)值,使用時(shí)需注意及時(shí)清理以防止內(nèi)存泄漏。常見(jiàn)用途包括用戶會(huì)話管理、數(shù)據(jù)庫(kù)連接、事務(wù)上下文和日志追蹤。最佳實(shí)踐包括:1.使用后調(diào)用remove()清理;2.避免過(guò)度使用;3.子線程繼承需用InheritableThreadLocal;4.不存儲(chǔ)大對(duì)象??赏ㄟ^(guò)initialValue()或withInitial()設(shè)置初始值,初始化延遲到首次get()調(diào)用。

如何分析Java堆垃圾場(chǎng)? 如何分析Java堆垃圾場(chǎng)? Jul 09, 2025 am 01:25 AM

分析Java堆轉(zhuǎn)儲(chǔ)是排查內(nèi)存問(wèn)題的關(guān)鍵手段,尤其用于識(shí)別內(nèi)存泄漏和性能瓶頸。1.使用EclipseMAT或VisualVM打開(kāi).hprof文件,MAT提供Histogram和DominatorTree視圖從不同角度展示對(duì)象分布;2.在Histogram中按實(shí)例數(shù)量或占用空間排序,查找異常多或體積大的類,如byte[]、char[]或業(yè)務(wù)類;3.通過(guò)“ListObjects>withincoming/outgoingreferences”查看引用鏈,判斷是否被意外持有;4.利用“Pathto

Java可選示例 Java可選示例 Jul 12, 2025 am 02:55 AM

Optional能清晰表達(dá)意圖并減少null判斷的代碼噪音。1.Optional.ofNullable是處理可能為null對(duì)象的常用方式,如從map中取值時(shí)可結(jié)合orElse提供默認(rèn)值,邏輯更清晰簡(jiǎn)潔;2.通過(guò)鏈?zhǔn)秸{(diào)用map實(shí)現(xiàn)嵌套取值,安全地避免NPE,任一環(huán)節(jié)為null則自動(dòng)終止并返回默認(rèn)值;3.filter可用于條件篩選,滿足條件才繼續(xù)執(zhí)行后續(xù)操作,否則直接跳到orElse,適合輕量級(jí)業(yè)務(wù)判斷;4.不建議過(guò)度使用Optional,如基本類型或簡(jiǎn)單邏輯中其反而增加復(fù)雜度,部分場(chǎng)景直接返回nu

如何修復(fù)java.io.notserializable Exception? 如何修復(fù)java.io.notserializable Exception? Jul 12, 2025 am 03:07 AM

遇到j(luò)ava.io.NotSerializableException的核心解決方法是確保所有需序列化的類實(shí)現(xiàn)Serializable接口,并檢查嵌套對(duì)象的序列化支持。1.給主類添加implementsSerializable;2.確保類中自定義字段對(duì)應(yīng)的類也實(shí)現(xiàn)Serializable;3.用transient標(biāo)記不需要序列化的字段;4.檢查集合或嵌套對(duì)象中的非序列化類型;5.查看異常信息定位具體哪個(gè)類未實(shí)現(xiàn)接口;6.對(duì)無(wú)法修改的類考慮替換設(shè)計(jì),如保存關(guān)鍵數(shù)據(jù)或使用可序列化的中間結(jié)構(gòu);7.考慮改

如何在Java中實(shí)施緩存策略(例如,使用Ehcache或咖啡因)? 如何在Java中實(shí)施緩存策略(例如,使用Ehcache或咖啡因)? Jul 09, 2025 am 01:17 AM

ToimproveperformanceinJavaapplications,choosebetweenEhCacheandCaffeinebasedonyourneeds.1.Forlightweight,modernin-memorycaching,useCaffeine—setitupbyaddingthedependency,configuringacachebeanwithsizeandexpiration,andinjectingitintoservices.2.Foradvance

如何在Java解析JSON? 如何在Java解析JSON? Jul 11, 2025 am 02:18 AM

解析JSON在Java中的常見(jiàn)方式有三種:使用Jackson、Gson或org.json。1.Jackson適合大多數(shù)項(xiàng)目,性能好且功能全面,支持對(duì)象與JSON字符串之間的轉(zhuǎn)換及注解映射;2.Gson更適合Android項(xiàng)目或輕量級(jí)需求,使用簡(jiǎn)單但處理復(fù)雜結(jié)構(gòu)和高性能場(chǎng)景略遜;3.org.json適用于簡(jiǎn)單任務(wù)或小腳本,不推薦用于大型項(xiàng)目,因其靈活性和類型安全不足。選擇應(yīng)根據(jù)實(shí)際需求決定。

新電子郵件的Outlook快捷方式 新電子郵件的Outlook快捷方式 Jul 11, 2025 am 03:25 AM

在Outlook中快速新建郵件的方法如下:1.桌面版使用快捷鍵Ctrl Shift M,可直接彈出新郵件窗口;2.網(wǎng)頁(yè)版可通過(guò)創(chuàng)建包含JavaScript的書(shū)簽(如javascript:document.querySelector("divrole='button'").click())實(shí)現(xiàn)一鍵新建郵件;3.使用瀏覽器插件(如Vimium、CrxMouseGestures)自定義快捷鍵觸發(fā)“新建郵件”按鈕;4.Windows用戶還可通過(guò)右鍵任務(wù)欄Outlook圖標(biāo)選擇“新建電

See all articles