电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

Java String不可變性實現原理解析

瀏覽:137日期:2022-09-03 10:27:36

一、原理

1、不變模式(不可變對象)

在并行軟件開發過程中,同步操作似乎是必不可少的。當多線程對同一個對象進行讀寫操作時,為了保證對象數據的一致性和正確性,有必要對對象進行同步。而同步操作對系統性能是相當的損耗。為了能盡可能的去除這些同步操作,提高并行程序性能,可以使用一種不可改變的對象,依靠對象的不變性,可以確保其在沒有同步操作的多線程環境中依然始終保持內部狀態的一致性和正確性。這就是不變模式。

不變模式天生就是多線程友好的,它的核心思想是,一個對象一旦被創建,則它的內部狀態將永遠不會發生改變。所以,沒有一個線程可以修改其內部狀態和數據,同時其內部狀態也絕不會自行發生改變。基于這些特性,對不變對象的多線程操作不需要進行同步控制。

同時還需要注意,不變模式和只讀屬性是有一定的區別的,不變模式是比讀屬性具有更強的一致性和不變性。對只讀屬性的對象而言,對象本身不能被其他線程修改,但是對象身狀態卻可能自行修改比如,一個對象的存活時間(對象創建時間和當前時間的時間差)是只讀的,因為任何個第三方線程都不能修改這個屬性,但是這是一個可變的屬性,因為隨著時間的推移,存活時司時刻都在發生變化。而不變模式則要求,無論出于什么原因,對象自創建后,其內部狀態和數據保持絕對的穩定。

2、怎么實現不可變對象

在Java語言中,不變模式的實現很簡單。為確保對象被創建后,不發生任何改變,并保證不變模式正常工作,只需要注意以下4點:

去除 setter方法以及所有修改自身屬性的方法。 將所有屬性設置為私有,并用final標記,確保其不可修改 確保沒有子類可以重載修改它的行為。 有一個可以創建完整對象的構造函數。

是不是和final的功能很吻合。我們復習一下java中final的作用。

final修飾類,表示該類不能被繼承,俗稱斷子絕孫類,該類的所有方法自動地成為final方法 final修飾方法,表示子類不可重寫該方法 final修飾基本數據類型變量,表示該變量為常量,值不能再修改 final修飾引用類型變量,表示該引用在構造對象之后不能指向其他的對象,但該引用指向的對象的狀態可以改變

這里需要說明的是:當使用final修飾基本類型變量時,不能對基本類型變量重新賦值,因此基本類型變量不能被改變。但對于引用類型變量而言,它保存的僅僅是一個引用,final只保證這個引用變量所引用的地址不會改變,即一直引用同一個對象,但這個對象完全可以發生改變。例如某個指向數組的final引用,它必須從此至終指向初始化時指向的數組,但是這個數組的內容完全可以改變。

二、String源碼分析

以下是jdk1.8中String類的部分源碼。 

public final class String implements java.io.Serializable, Comparable<String>, CharSequence { /** The value is used for character storage. */ private final char value[]; /** Cache the hash code for the string */ private int hash; // Default to 0 /** use serialVersionUID from JDK 1.0.2 for interoperability */ private static final long serialVersionUID = -6849794470754667710L; /** ...}

首先可以看到,String類使用了final修飾符,表明String類是不可繼承的。然后,我們主要關注String類的成員變量value,value是char[]類型,因此String對象實際上是用這個字符數組進行封裝的。

再看value的修飾符,使用了private,也沒有提供setter方法,所以在String類的外部不能修改value,同時value也使用了final進行修飾,那么在String類的內部也不能修改value,也就是說value一旦賦予初始值之后,value指向的地址就不能再改變了。但是上面final修飾引用類型變量的內容提到,這只能保證value不能指向其他的對象,但value指向的對象的狀態是可以改變的。

通過查看String類源碼可以發現,String類不可變,關鍵是因為SUN公司的工程師,在后面所有String的方法里都很小心的沒有去動字符數組里的元素。所以String類不可變的關鍵都在底層的實現,而不僅僅是一個final。

三、修改String使其“可變”

雖然value是final修飾的,只是說明value不能再重新指向其他的引用。但是value指向的數組可以改變,一般情況下我們是沒有辦法訪問到這個value指向的數組的元素。But,反射,對,反射可以,牛逼吧。可以反射出String對象中的value屬性, 進而改變通過獲得的value引用改變數組的結構。

public static void main(String[] args) throws Exception { String str = 'Hello World'; System.out.println('修改前的str:' + str); System.out.println('修改前的str的內存地址' + System.identityHashCode(str)); // 獲取String類中的value字段 Field valueField = String.class.getDeclaredField('value'); // 改變value屬性的訪問權限 valueField.setAccessible(true); // 獲取str對象上value屬性的值 char[] value = (char[]) valueField.get(str); // 改變value所引用的數組中的字符 value[3] = ’?’; System.out.println('修改后的str:' + str); System.out.println('修改前的str的內存地址' + System.identityHashCode(str));}// 運行結果// 可以看到str的字符串序列已經被改變了,但是str的內存地址還是沒有改變。修改前的str:Hello World修改前的str的內存地址1922154895修改后的str:Hel?o World修改前的str的內存地址1922154895

四、String設計成不可變性的原因

在Java中,將String設計成不可變的是綜合考慮到內存、同步、數據結構及安全等各種因素的結果,下文將為各種因素做一個小結。

1、運行時常量池的需要

比如執行 String s = 'abc';執行上述代碼時,JVM首先在運行時常量池中查看是否存在String對象“abc”,如果已存在該對象,則不用創建新的String對象“abc”,而是將引用s直接指向運行時常量池中已存在的String對象“abc”;如果不存在該對象,則先在運行時常量池中創建一個新的String對象“abc”,然后將引用s指向運行時常量池中創建的新String對象。這樣在運行時常量池中只會創建一個String對象'abc',這樣就節省了內存空間。

2、同步

因為String對象是不可變的,所以是多線程安全的,同一個String實例可以被多個線程共享。這樣就不用因為線程安全問題而使用同步。

3、允許String對象緩存hashcode

查看上文JDK1.8中String類源碼,可以發現其中有一個字段hash,String類的不可變性保證了hashcode的唯一性,所以可以用hash字段對String對象的hashcode進行緩存,就不需要每次重新計算hashcode。所以Java中String對象經常被用來作為HashMap等容器的鍵。

4、安全性

如果String對象是可變的,那么會引起很嚴重的安全問題。比如,數據庫的用戶名、密碼都是以字符串的形式傳入來獲得數據庫的連接,或者在socket編程中,主機名和端口都是以字符串的形式傳入。因為String對象是不可變的,所以它的值是不可改變的,否則黑客們可以鉆到空子,改變String引用指向的對象的值,造成安全漏洞。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Java
相關文章:
主站蜘蛛池模板: 河南中整光饰机械有限公司-抛光机,去毛刺抛光机,精密镜面抛光机,全自动抛光机械设备 | 【法利莱住人集装箱厂家】—活动集装箱房,集装箱租赁_大品牌,更放心 | 德州网站制作 - 网站建设设计 - seo排名优化 -「两山建站」 | 二手Sciex液质联用仪-岛津气质联用仪-二手安捷伦气质联用仪-上海隐智科学仪器有限公司 | 真空泵厂家_真空泵机组_水环泵_旋片泵_罗茨泵_耐腐蚀防爆_中德制泵 | 热回收盐水机组-反应釜冷水机组-高低温冷水机组-北京蓝海神骏科技有限公司 | 精密钢管,冷拔精密无缝钢管,精密钢管厂,精密钢管制造厂家,精密钢管生产厂家,山东精密钢管厂家 | 幂简集成 - 品种超全的API接口平台, 一站搜索、试用、集成国内外API接口 | 金属波纹补偿器厂家_不锈钢膨胀节价格_非金属伸缩节定制-庆达补偿器 | 阜阳成人高考_阜阳成考报名时间_安徽省成人高考网 | 二手回收公司_销毁处理公司_设备回收公司-找回收信息网 | 北京易通慧公司从事北京网站优化,北京网络推广、网站建设一站式服务商-北京网站优化公司 | 珠海冷却塔降噪维修_冷却塔改造报价_凉水塔风机维修厂家- 广东康明节能空调有限公司 | 运动木地板_体育木地板_篮球馆木地板_舞台木地板-实木运动地板厂家 | 武汉天安盾电子设备有限公司 - 安盾安检,武汉安检门,武汉安检机,武汉金属探测器,武汉测温安检门,武汉X光行李安检机,武汉防爆罐,武汉车底安全检查,武汉液体探测仪,武汉安检防爆设备 | 喷砂机厂家_自动喷砂机生产_新瑞自动化喷砂除锈设备 | 浙江华锤电器有限公司_地磅称重设备_防作弊地磅_浙江地磅售后维修_无人值守扫码过磅系统_浙江源头地磅厂家_浙江工厂直营地磅 | 工控机,嵌入式主板,工业主板,arm主板,图像采集卡,poe网卡,朗锐智科 | 缝纫客| 郑州大巴车出租|中巴车租赁|旅游大巴租车|包车|郑州旅游大巴车租赁有限公司 | 铝扣板-铝方通-铝格栅-铝条扣板-铝单板幕墙-佳得利吊顶天花厂家 elisa试剂盒价格-酶联免疫试剂盒-猪elisa试剂盒-上海恒远生物科技有限公司 | 制样机-密封锤式破碎机-粉碎机-智能马弗炉-南昌科鑫制样 | 工业设计,人工智能,体验式3D展示的智能技术交流服务平台-纳金网 J.S.Bach 圣巴赫_高端背景音乐系统_官网 | 防伪溯源|防窜货|微信二维码营销|兆信_行业内领先的防伪防窜货数字化营销解决方案供应商 | 移动厕所租赁|移动卫生间|上海移动厕所租赁-家瑞租赁 | _网名词典_网名大全_qq网名_情侣网名_个性网名 | 锡膏喷印机-全自动涂覆机厂家-全自动点胶机-视觉点胶机-深圳市博明智控科技有限公司 | 百方网-百方电气网,电工电气行业专业的B2B电子商务平台 | 薄壁轴承-等截面薄壁轴承生产厂家-洛阳薄壁精密轴承有限公司 | 合肥钣金加工-安徽激光切割加工-机箱机柜加工厂家-合肥通快 | 断桥铝破碎机_铝合金破碎机_废铁金属破碎机-河南鑫世昌机械制造有限公司 | 卷筒电缆-拖链电缆-特种柔性扁平电缆定制厂家「上海缆胜」 | 酒精检测棒,数显温湿度计,酒安酒精测试仪,酒精检测仪,呼气式酒精检测仪-郑州欧诺仪器有限公司 | Safety light curtain|Belt Sway Switches|Pull Rope Switch|ultrasonic flaw detector-Shandong Zhuoxin Machinery Co., Ltd | 土壤养分检测仪|土壤水分|土壤紧实度测定仪|土壤墒情监测系统-土壤仪器网 | 注浆压力变送器-高温熔体传感器-矿用压力传感器|ZHYQ朝辉 | 二手电脑回收_二手打印机回收_二手复印机回_硒鼓墨盒回收-广州益美二手电脑回收公司 | 电杆荷载挠度测试仪-电杆荷载位移-管桩测试仪-北京绿野创能机电设备有限公司 | 电缆接头_防水接头_电缆防水接头 - 乐清市新豪电气有限公司 | 抖音短视频运营_企业网站建设_网络推广_全网自媒体营销-东莞市凌天信息科技有限公司 | 杭州高温泵_热水泵_高温油泵|昆山奥兰克泵业制造有限公司 |