Java中的常量避免反模式的方法
在應(yīng)用中,我們往往需要一個(gè)常量文件,用于存儲(chǔ)被多個(gè)地方引用的共享常量。在設(shè)計(jì)應(yīng)用時(shí),我也遇到了類(lèi)似的情況,很多地方都需要各種各樣的常量。
我確定需要一個(gè)單獨(dú)的文件來(lái)存儲(chǔ)這些靜態(tài)公共常量。但是我不是特別確定是應(yīng)該用接口還是類(lèi)(枚舉不滿足我的需求)。我有兩種選擇:
使用接口,如:
package one; public interface Constants { String NAME="name1"; int MAX_VAL=25; } 或 package two; public class Constants { public static final String NAME="name1"; public static final int MAX_VAL=25; }
我的觀點(diǎn)是使用接口。因?yàn)榻涌跁?huì)自動(dòng)將成員變量設(shè)置為靜態(tài)的(static)、不可變的(final),這一點(diǎn)可以防止某些情況下錯(cuò)誤地添加新的常量。這也使得代碼看起來(lái)更簡(jiǎn)單和清晰。
同時(shí),一個(gè)的簡(jiǎn)單測(cè)試顯示,同樣的接口(字節(jié)碼文件)占用的空間是209個(gè)字節(jié)(ubuntu 14.04機(jī)器上),而類(lèi)(字節(jié)碼文件)占用的空間是366個(gè)字節(jié)(同樣的操作系統(tǒng))。更少的字節(jié)碼文件意味著加載和維護(hù)的成本更低。此外,JVM 加載接口的時(shí)候,不需要擔(dān)心類(lèi)提供的額外特征(如重載、方法的動(dòng)態(tài)綁定等),因此加載更快。
這看起來(lái)非常好,但是這是一個(gè)典型反模式的例子。雖然使用接口來(lái)保存常量看起很有幫助,但是這給應(yīng)用后期的擴(kuò)展留下一個(gè)漏洞。
假設(shè)存在在一個(gè)類(lèi),緊密】依賴(lài)于這些常量。開(kāi)發(fā)者在該類(lèi)中寫(xiě)滿了通過(guò)接口對(duì)常量的引用。如:
所以,為了“清理”這段代碼,他可能想實(shí)現(xiàn)該接口,這樣他就不需要到處寫(xiě)“packagename.Constants”,所有的常量可以直接訪問(wèn)。
但是,一旦他實(shí)現(xiàn)了該接口,所有的常量就都變成“契約”(因?yàn)樗械某A慷际枪驳摹㈧o態(tài)的)的一部分。這導(dǎo)致為這個(gè)類(lèi)增加了不必要的常量。這會(huì)動(dòng)搖整個(gè)基礎(chǔ),并引起混亂。Java 中沒(méi)有一種方式可以阻止類(lèi)實(shí)現(xiàn)接口。
而另一種方式,我們可以將類(lèi)設(shè)置為final,這樣就不能擴(kuò)展。甚至,我們可以將構(gòu)造器設(shè)置為私有的,以防止對(duì)這個(gè)類(lèi)實(shí)例化,這樣就永遠(yuǎn)不會(huì)破壞約定。此外,如果一個(gè)特殊的常量在同一個(gè)類(lèi)中被多次使用,則開(kāi)發(fā)者可以使用靜態(tài)引入。
所有對(duì)于常量類(lèi),比較好的設(shè)計(jì)應(yīng)該是:
package three; //make the class non-extendable by adding final 增加final關(guān)鍵字來(lái)避免繼承 public final class Constants { //Hide the constructor 隱藏構(gòu)造器 private Constants(){} public static String NAME="name"; }
靜態(tài)引入的例子:
import static three.Constants.NAME; public class UseConstants { public static void main(String[] args) { System.out.println("the value of constants is"+NAME); } }
這個(gè)設(shè)計(jì)問(wèn)題也稱(chēng)為接口常量反模式(Constant Interface Anti-pattern)。
以上就是java常量在使用過(guò)程中如何避免反模式的解決方法,希望對(duì)大家的學(xué)習(xí)有所幫助。
相關(guān)文章
詳解SpringBoot初始教程之Tomcat、Https配置以及Jetty優(yōu)化
本篇文章主要介紹了詳解SpringBoot初始教程之Tomcat、Https配置以及Jetty優(yōu)化,具有一定的參考價(jià)值,有興趣的可以了解一下2017-09-09servlet實(shí)現(xiàn)簡(jiǎn)單的權(quán)限管理和敏感詞過(guò)濾功能
JavaEE課要求用servlet和過(guò)濾器實(shí)現(xiàn)權(quán)限管理和敏感詞過(guò)濾功能,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05SpringBoot web開(kāi)發(fā)源碼深入分析
Web開(kāi)發(fā)的核心內(nèi)容主要包括內(nèi)嵌的Servlet容器和SpringMVCSpringBoot使用起來(lái)非常簡(jiǎn)潔,大部分配置都有SpringBoot自動(dòng)裝配,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2022-10-10Java數(shù)據(jù)結(jié)構(gòu)之List的使用總結(jié)
List是Java中比較常用的集合類(lèi),指一系列存儲(chǔ)數(shù)據(jù)的接口和類(lèi),可以解決復(fù)雜的數(shù)據(jù)存儲(chǔ)問(wèn)題,本文就來(lái)拿實(shí)際案例總結(jié)介紹一下List的使用方法,感興趣的朋友快來(lái)看看吧2021-11-11spring學(xué)習(xí)之@SessionAttributes實(shí)例解析
這篇文章主要介紹了spring學(xué)習(xí)之@SessionAttributes實(shí)例解析,分享了相關(guān)代碼示例,小編覺(jué)得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-02-02Triple協(xié)議支持Java異?;貍髟O(shè)計(jì)實(shí)現(xiàn)詳解
這篇文章主要為大家介紹了Triple協(xié)議支持Java異常回傳設(shè)計(jì)實(shí)現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12