成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

一些我認(rèn)為有用有趣的JDK方法

lx1036 / 2756人閱讀

摘要:在學(xué)習(xí)的源碼過程中我遇到了一些有趣有用的方法,在此之前如果要使用這些工具方法,我首先會(huì)想到的是和這樣的語言擴(kuò)展包,但現(xiàn)在如果是寫一些,使用原生即可達(dá)到目的。

在學(xué)習(xí)JDK的源碼過程中我遇到了一些有趣有用的方法,在此之前如果要使用這些工具方法,我首先會(huì)想到的是commons-langguava這樣的語言擴(kuò)展包,但現(xiàn)在如果是寫一些demo,使用原生即可達(dá)到目的。當(dāng)然我們也不能否認(rèn)它們的作用,在平時(shí)的工作項(xiàng)目中幾乎都會(huì)引入這些語言擴(kuò)展包,直接使用他們也使得編程風(fēng)格統(tǒng)一,而且還能夠?qū)Φ桶姹镜腏DK提供支持。
以下收集的代碼片段可能會(huì)逐漸增加,也可能不會(huì)。

java.util.Objects

java.util.Objects工具類,我覺得好用的幾個(gè)方法

    public static boolean equals(Object var0, Object var1) {
        return var0 == var1 || var0 != null && var0.equals(var1);
    }
    public static int hashCode(Object var0) {
        return var0 != null ? var0.hashCode() : 0;
    }
    public static  T requireNonNull(T var0) {
        if (var0 == null) {
            throw new NullPointerException();
        } else {
            return var0;
        }
    }

    public static  T requireNonNull(T var0, String var1) {
        if (var0 == null) {
            throw new NullPointerException(var1);
        } else {
            return var0;
        }
    }        

除此之外還應(yīng)該從Objects學(xué)習(xí)到編寫工具類的正確的規(guī)范,

定義為final class

只定義一個(gè)無參的構(gòu)造函數(shù)且拋出斷言錯(cuò)誤,防止被反射調(diào)用

工具方法都是靜態(tài)方法

靜態(tài)方法中只拋出unchecked異常

java.lang.System

這個(gè)最早應(yīng)該是在Hello World程序中見到的,推薦它的一個(gè)方法

    /**
     * Returns the same hash code for the given object as
     * would be returned by the default method hashCode(),
     * whether or not the given object"s class overrides
     * hashCode().
     * The hash code for the null reference is zero.
     *
     * @param x object for which the hashCode is to be calculated
     * @return  the hashCode
     * @since   JDK1.1
     */
    public static native int identityHashCode(Object x);

注釋寫得很明白了,不管一個(gè)對象實(shí)例的class有沒有覆蓋Object的hashCode方法,都能使用這個(gè)方法獲得hash值。

獲取泛型類的類型參數(shù)

我們可以從以下代碼獲得提示,代碼來自HashMap,

    /**
     * Returns x"s Class if it is of the form "class C implements
     * Comparable", else null.
     */
    static Class comparableClassFor(Object x) {
        if (x instanceof Comparable) {
            Class c; Type[] ts, as; Type t; ParameterizedType p;
            if ((c = x.getClass()) == String.class) // bypass checks
                return c;
            if ((ts = c.getGenericInterfaces()) != null) {
                for (int i = 0; i < ts.length; ++i) {
                    if (((t = ts[i]) instanceof ParameterizedType) &&
                        ((p = (ParameterizedType)t).getRawType() ==
                         Comparable.class) &&
                        (as = p.getActualTypeArguments()) != null &&
                        as.length == 1 && as[0] == c) // type arg is c
                        return c;
                }
            }
        }
        return null;
    }

這里的邏輯是獲得類C,然后獲取它實(shí)現(xiàn)的接口Comparable,然后從這個(gè)Comparable中獲得類型參數(shù)C,然后比較這兩個(gè)類型是否相等。雖然我們一直聽說Java的泛型是類型擦除式,但是在這里我們是可以獲得泛型的參數(shù)類型的。照例用一段demo測試一下,

public class ParameterApp {
    public static void main(String[] args) {
        StringList list = new StringList();
        Class clazz = getTypeArgument(list);
        System.out.println(clazz.getName());
    }

    static Class getTypeArgument(Object x) {
        if (x instanceof Collection) {
            Class c = x.getClass();
            Type[] ts, as; Type t; ParameterizedType p;
            if ((ts = c.getGenericInterfaces()) != null) {
                for (int i = 0; i < ts.length; ++i) {
                    if (((t = ts[i]) instanceof ParameterizedType) &&
                            ((as  = ((ParameterizedType)t).getActualTypeArguments()) != null)
                             &&
                            as.length == 1) // type arg is c
                        return (Class) as[0];
                }
            }
        }
        return null;
    }

    static class StringList extends AbstractList implements List {

        @Override
        public String get(int i) {
            return null;
        }

        @Override
        public int size() {
            return 0;
        }
    }
}
sun.reflect.Reflection

這個(gè)工具類是和反射相關(guān)的,讓大家知道有這么一個(gè)方法

    @CallerSensitive
    public static native Class getCallerClass();

我第一次見到這個(gè)方法是在java.sql.DriverManager中的getConnection方法中見到的

    @CallerSensitive
    public static Connection getConnection(String url,
        String user, String password) throws SQLException {
        java.util.Properties info = new java.util.Properties();

        if (user != null) {
            info.put("user", user);
        }
        if (password != null) {
            info.put("password", password);
        }

        return (getConnection(url, info, Reflection.getCallerClass()));
    }

Reflection.getCallerClass()是一個(gè)native方法,返回的是Class類型,在DriverManager中使用它的目的是為了獲得相應(yīng)的ClassLoader,上面的代碼是在Java 8中見到的。其中在Java 7中為獲得ClassLoader,DriverManager就直接提供了native的方法

/* Returns the caller"s class loader, or null if none */
private static native ClassLoader getCallerClassLoader();

我們用一段代碼嘗試調(diào)用這個(gè)方法

public class CalleeApp {

    public void call() {
        Class clazz = Reflection.getCallerClass();
        System.out.println("Hello " + clazz);
    }
}
public class CallerApp {

    public static void main(String[] args) {
        CalleeApp app = new CalleeApp();
        Caller1 c1 = new Caller1();
        c1.run(app);
    }

    static class Caller1 {
        void run(CalleeApp calleeApp) {
            if (calleeApp == null) {
                throw new IllegalArgumentException("callee can not be null");
            }
            calleeApp.call();
        }
    }

}

執(zhí)行main方法會(huì)拋出異常

Exception in thread "main" java.lang.InternalError: CallerSensitive annotation expected at frame 1

這個(gè)錯(cuò)誤信息說的是我們?nèi)鄙僭诤瘮?shù)調(diào)用棧開始位置添加CallerSensitive注解,觀察DriverManagergetConnection方法確實(shí)是有這么個(gè)注解的。
那如果給CalleeAppcall加上注解,那結(jié)果又會(huì)怎樣呢?

Object.wait(long timeout, int nanos)

這個(gè)方法是來賣萌,它的本義在注釋是這樣子寫的,

    /*
     * 

* This method is similar to the {@code wait} method of one * argument, but it allows finer control over the amount of time to * wait for a notification before giving up. The amount of real time, * measured in nanoseconds, is given by: *

*
     * 1000000*timeout+nanos
*

*/

意思是提供精細(xì)化的時(shí)間衡量,nano可是納秒單位啊?。?!
而它的實(shí)現(xiàn)卻是這樣的,

    public final void wait(long timeout, int nanos) throws InterruptedException {
        if (timeout < 0) {
            throw new IllegalArgumentException("timeout value is negative");
        }

        if (nanos < 0 || nanos > 999999) {
            throw new IllegalArgumentException(
                                "nanosecond timeout value out of range");
        }

        if (nanos > 0) {
            timeout++;
        }

        wait(timeout);
    }

除了對傳入?yún)?shù)的數(shù)值范圍校驗(yàn)外,對nano的使用緊緊是判斷這個(gè)變量是否大于0,是則給timeout加1,這只是增加了1毫秒的時(shí)間,并沒有體現(xiàn)出了精細(xì)化的地方。

you-dont-need-serial

Reflection.getCallerClass()使用的問題

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/69032.html

相關(guān)文章

  • 源碼|jdk源碼之Object及裝箱類型分析

    摘要:作為條件變量的的不僅可以認(rèn)為內(nèi)嵌了一把鎖,還內(nèi)嵌了一個(gè)條件變量。操作條件變量的函數(shù)將當(dāng)前線程在條件變量上阻塞,一般是為了等待其他線程的某件事情執(zhí)行完成。其它裝箱類其它裝箱類的代碼這里就不分析了。重點(diǎn)關(guān)注下各裝箱類的緩存范圍。 jdk源碼讀到現(xiàn)在這里,重要的集合類也讀了一部分了。集合類再往下讀的話,就要涉及到兩個(gè)方向。第一,是比較典型的但是不常用的數(shù)據(jù)結(jié)構(gòu),這部分我準(zhǔn)備將數(shù)據(jù)結(jié)構(gòu)復(fù)習(xí)、回...

    VioletJack 評論0 收藏0
  • 前端學(xué)習(xí)資源

    摘要:掘金日報(bào)第四期使用怎么能不知道這些插件合集掘金日報(bào)主打分享優(yōu)質(zhì)深度技術(shù)內(nèi)容,技術(shù)內(nèi)容分前端后端產(chǎn)品設(shè)計(jì)工具資源和一些有趣的東西。目前已經(jīng)涵蓋了的相關(guān)資源鏈接,供大家參考與學(xué)習(xí)。 【掘金日報(bào)】第四期 使用Sublime?怎么能不知道這些 Sublime 插件合集! 掘金日報(bào)主打分享優(yōu)質(zhì)深度技術(shù)內(nèi)容,技術(shù)內(nèi)容分:前端、后端、Android、iOS、產(chǎn)品設(shè)計(jì)、工具資源和一些有趣的東西。 前端...

    xzavier 評論0 收藏0
  • 前端學(xué)習(xí)資源

    摘要:掘金日報(bào)第四期使用怎么能不知道這些插件合集掘金日報(bào)主打分享優(yōu)質(zhì)深度技術(shù)內(nèi)容,技術(shù)內(nèi)容分前端后端產(chǎn)品設(shè)計(jì)工具資源和一些有趣的東西。目前已經(jīng)涵蓋了的相關(guān)資源鏈接,供大家參考與學(xué)習(xí)。 【掘金日報(bào)】第四期 使用Sublime?怎么能不知道這些 Sublime 插件合集! 掘金日報(bào)主打分享優(yōu)質(zhì)深度技術(shù)內(nèi)容,技術(shù)內(nèi)容分:前端、后端、Android、iOS、產(chǎn)品設(shè)計(jì)、工具資源和一些有趣的東西。 前端...

    weij 評論0 收藏0
  • LSTM入門必讀:從基礎(chǔ)知識到工作方式詳解

    摘要:意味著完全保持,意味著完全丟棄??ū全F寫這篇博文的時(shí)間我本可以抓一百只,請看下面的漫畫。神經(jīng)網(wǎng)絡(luò)神經(jīng)網(wǎng)絡(luò)會(huì)以的概率判定輸入圖片中的卡比獸正在淋浴,以的概率判定卡比獸正在喝水,以的概率判定卡比獸正在遭遇襲擊。最終結(jié)果是卡比獸正在遭遇襲擊 我第一次學(xué)習(xí) LSTM 的時(shí)候,它就吸引了我的眼球。事實(shí)證明 LSTM 是對神經(jīng)網(wǎng)絡(luò)的一個(gè)相當(dāng)簡單的擴(kuò)展,而且在最近幾年里深度學(xué)習(xí)所實(shí)現(xiàn)的驚人成就背后都有它們...

    alanoddsoff 評論0 收藏0
  • 21 個(gè)VSCode 快捷鍵,讓代碼更快,更有趣

    摘要:在這篇文章中,我將列出我最喜歡的快捷鍵,這些快捷鍵讓我更快的編寫代碼,也讓編碼變得更有趣,以下是個(gè)快捷鍵,分享給你。打開鍵盤快捷鍵或,搜索。在中,啟動(dòng)性能是很重要的。逐個(gè)選擇文本可以通過快捷鍵右箭頭右箭頭和左箭頭左箭頭逐個(gè)選擇文本。 為了保證的可讀性,本文采用意譯而非直譯。 想閱讀更多優(yōu)質(zhì)文章請猛戳GitHub博客,一年百來篇優(yōu)質(zhì)文章等著你! 注意:自己嘗試的時(shí)候,Mac(17, p...

    elina 評論0 收藏0

發(fā)表評論

0條評論

閱讀需要支付1元查看
<