国产秋霞理论久久久电影-婷婷色九月综合激情丁香-欧美在线观看乱妇视频-精品国avA久久久久久久-国产乱码精品一区二区三区亚洲人-欧美熟妇一区二区三区蜜桃视频

2w字 詳解 String,yyds

共 42432字,需瀏覽 85分鐘

 ·

2021-07-04 20:28

關(guān)注公眾號(hào)Java后端技術(shù)全棧

回復(fù)“000”獲取程序員必備電子書(shū)

前言

大家好,我是老田,今天給大家分享java基礎(chǔ)知識(shí)之String。

String類的重要性就不必說(shuō)了,可以說(shuō)是我們后端開(kāi)發(fā)用的最多的類,所以,很有必要好好來(lái)聊聊它。

本文主要內(nèi)容如下:


String簡(jiǎn)介

我們先來(lái)說(shuō)說(shuō),java中八大數(shù)據(jù)類型,然后在說(shuō)String。

八大基本數(shù)據(jù)類型

byte:8位,最大存儲(chǔ)數(shù)據(jù)量是255,存放的數(shù)據(jù)范圍是-128~127之間。

short:16位,最大數(shù)據(jù)存儲(chǔ)量是65536,數(shù)據(jù)范圍是-32768~32767之間。

int:32位,最大數(shù)據(jù)存儲(chǔ)容量是2的32次方減1,數(shù)據(jù)范圍是負(fù)的2的31次方到正的2的31次方減1。

long:64位,最大數(shù)據(jù)存儲(chǔ)容量是2的64次方減1,數(shù)據(jù)范圍為負(fù)的2的63次方到正的2的63次方減1。

float:32位,數(shù)據(jù)范圍在3.4e-45~1.4e38,直接賦值時(shí)必須在數(shù)字后加上f或F。

double:64位,數(shù)據(jù)范圍在4.9e-324~1.8e308,賦值時(shí)可以加d或D也可以不加。

boolean:只有true和false兩個(gè)取值。

char:16位,存儲(chǔ)Unicode碼,用單引號(hào)賦值。

除了這八大數(shù)據(jù)類型以外(八大數(shù)據(jù)類型也有與之對(duì)應(yīng)的封裝類型,我相信你是知道的),Java中還有一種比較特殊的類型:String,字面意義就是字符串。

String官方介紹

英文版


地址:https://docs.oracle.com/en/java/javase/11/docs/api/java.base/java/lang/String.html

看不懂嗎?沒(méi)事,我們可以借用翻譯工具,瀏覽器自帶的,更希望的是你能看懂原版英文。

String 存在于咱們安裝的JDK目錄下rt.ar包中,全路徑名為:java.lang.String。我們java代碼中String用來(lái)表示字符串,比如:

String str = "中國(guó)夢(mèng),我的夢(mèng)";
String name = "zhangsan";

暫時(shí)先知道這些就可以了。

String使用

定義類型

在日常開(kāi)發(fā)中,使用String的地方太多了,尤其是用來(lái)定義變量、常量的類型,基本上只要你碼代碼,總是能見(jiàn)到它。

比如:用戶信息,用實(shí)體類User來(lái)表示。

public class User{
    private Long id;
    private String userName;
    private String address;
    private String password;
    ....
}

常用方法演示

String類有20多個(gè)方法,下面給出一個(gè)使用示例(這里演示大部分方法,剩下的可以自行去試試)。

//案例代碼,來(lái)源于網(wǎng)絡(luò)
public class StringDemo {
    public static void main(String[] args) throws Exception {
        String str1 = "Hello World";
        String str2 = "Hello World";
        String str3 = "hello world";
        String str4 = " hello world ";
        //返回字符串的長(zhǎng)度
        System.out.println("r1: " + str1.length());
        //比較兩個(gè)字符串的大小compareTo(返回的是int),0相等,復(fù)數(shù)小于,正數(shù)大于
        System.out.println("r2 : " + str1.compareTo(str2));
        //比較兩個(gè)字符串的大小compareTo(返回的是int),0相等,復(fù)數(shù)小于,正數(shù)大于
        System.out.println("r3 : " + str1.compareTo(str3));
        //字符串比較compareToIgnoreCase,忽略大小寫(xiě)。0相等,復(fù)數(shù)小于,正數(shù)大于
        System.out.println("r4 : " + str1.compareToIgnoreCase(str3));
        //字符串查找indexOf,返回的是找到的第一個(gè)的位置,沒(méi)找到返回-1。從0開(kāi)始
        System.out.println("r5 : " + str1.indexOf("o"));
        //查找字符串最后一次出現(xiàn)的位置lastIndexOf
        System.out.println("r6 : " + str1.lastIndexOf("o"));
        //刪除字符串中的一個(gè)字符,字符串從0開(kāi)始的 substring(a, b)
        //返回指定起始位置(含)到結(jié)束位置(不含)之間的字符串
        System.out.println("r7 : " + str1.substring(05) + str1.substring(6));

        //字符串替換,替換所有
        System.out.println("r8 : " + str1.replace("o""h"));
        //字符串替換,替換所有
        System.out.println("r9 : " + str1.replaceAll("o""h"));
        //字符串替換,替換第一個(gè)
        System.out.println("r10 : " + str1.replaceFirst("o""h"));
        //字符串反轉(zhuǎn)
        System.out.println("r11 : " + new StringBuffer(str1).reverse());
        //字符串反轉(zhuǎn)
        System.out.println("r11’: " + new StringBuilder(str1).reverse());
        //字符串分割
        String[] temp = str1.split("\\ ");
        for (String str : temp) {
            System.out.println("r12 : " + str);
        }
        //字符串轉(zhuǎn)大寫(xiě)
        System.out.println("r13 : " + str1.toUpperCase());
        //字符串轉(zhuǎn)小寫(xiě)
        System.out.println("r14 : " + str1.toLowerCase());
        //去掉首尾空格
        System.out.println("r15 : " + str4.trim());
        //是否包含,大小寫(xiě)區(qū)分
        System.out.println("r16 : " + str1.contains("World"));
        //返回指定位置字符
        System.out.println("r17 : " + str1.charAt(4));
        //測(cè)試此字符串是否以指定的后綴結(jié)束
        System.out.println("r18 : " + str1.endsWith("d"));
        //測(cè)試此字符串是否以指定的前綴開(kāi)始
        System.out.println("r19 : " + str1.startsWith("H"));
        //測(cè)試此字符串從指定索引開(kāi)始的子字符串是否以指定前綴開(kāi)始
        System.out.println("r20 : " + str1.startsWith("ll"2));
        //將指定字符串連接到此字符串的結(jié)尾。等價(jià)于用“+”
        System.out.println("r21 : " + str1.concat("haha"));
        //比較字符串的內(nèi)容是否相同
        System.out.println("r22 : " + str1.equals(str2));
        //與equals方法類似,忽略大小寫(xiě)
        System.out.println("r23 : " + str1.equalsIgnoreCase(str2));
        //判斷是否是空字符串
        System.out.println("r24:  " + str1.isEmpty());

    }
}

我們開(kāi)發(fā)中差不多也就是這么使用了,但是如果你僅僅是使用很牛了,貌似遇到面試照樣會(huì)掛。所以,學(xué)知識(shí),不能停留在使用層面,需要更深層次的學(xué)習(xí)。

下面我們就來(lái)深層次的學(xué)習(xí)String,希望大家?guī)е活w平常的心學(xué)習(xí),不要害怕什么,燈籠是張紙,捅破不值錢。

String核心部分源碼分析

備注:JDK版本為1.8+,因?yàn)镴DK9版本中和舊版本有細(xì)微差別。

String類源碼注釋

/**
 * The {@code String} class represents character strings. All
 * string literals in Java programs, such as {@code "abc"}, are
 * implemented as instances of this class.
 * 這個(gè)String類代表字符串。java編程中的所有字符串常量。
 * 比如說(shuō):"abc"就是這個(gè)String類的實(shí)例
 * <p>
 * Strings are constant; their values cannot be changed after they
 * are created. 
 * 字符串是常量,他們一旦被創(chuàng)建后,他們的值是不能被修改。(重點(diǎn))
 * String buffers support mutable strings.
 * String緩存池支持可變的字符串,
 * Because String objects are immutable they can be shared. For example:
 * 因?yàn)镾tring字符串不可變,但他們可以被共享。比如:
 * <blockquote><pre>
 *     String str = "abc";
 * </pre></blockquote><p>
 * is equivalent to:
 * <blockquote><pre>
 *     char data[] = {'a', 'b', 'c'};
 *     String str = new String(data);
 * </pre></blockquote><p>
 * Here are some more examples of how strings can be used:
 * String使用案例
 *     System.out.println("abc");
 *     String cde = "cde";
 *     System.out.println("abc" + cde);
 *     String c = "abc".substring(2,3);
 *     String d = cde.substring(1, 2);
 * <p>
 * The class {@code String} includes methods for examining
 * individual characters of the sequence, for comparing strings, for
 * searching strings, for extracting substrings, and for creating a
 * copy of a string with all characters translated to uppercase or to
 * lowercase. Case mapping is based on the Unicode Standard version
 * specified by the {@link java.lang.Character Character} class.
 * 這個(gè)String類包含了一些測(cè)評(píng)單個(gè)字符序列的方法,比如字符串比較,查找字符串,
 * 提取字符串,和拷貝一個(gè)字符串的大小寫(xiě)副本。
 * 大小寫(xiě)映射的是基于Character類支持的Unicode的字符集標(biāo)準(zhǔn)版本。
 * <p>
 * The Java language provides special support for the string
 * concatenation operator (&nbsp;+&nbsp;), and for conversion of
 * other objects to strings. 
 * java語(yǔ)言提供了對(duì)字符串的特殊支持,如:可以通過(guò)"+"號(hào)來(lái)進(jìn)行字符串的拼接操作,
 * 為其他類提供了與字符串轉(zhuǎn)換的操作
 * String concatenation is implemented
 * through the {@code StringBuilder}(or {@code StringBuffer})
 * class and its {@code append} method.
 * 字符串的+號(hào)拼接操作是通過(guò)StringBuilder或者StringBuffer類的append()方法
 * 來(lái)實(shí)現(xiàn)的
 * String conversions are implemented through the method
 * {@code toString}, defined by {@code Object} and
 * inherited by all classes in Java. 
 * 對(duì)象與字符串的轉(zhuǎn)換操作是通過(guò)所有類的父類Object中定義的toString()方法來(lái)實(shí)現(xiàn)的
 * For additional information on
 * string concatenation and conversion, see Gosling, Joy, and Steele,
 * <i>The Java Language Specification</i>.
 *
 * <p> Unless otherwise noted, passing a <tt>null</tt> argument to a constructor
 * or method in this class will cause a {@link NullPointerException} to be
 * thrown.
 * 除非有特殊說(shuō)明,否則傳一個(gè)null給String的構(gòu)造方法或者put方法,會(huì)報(bào)空指針異常的
 * <p>A {@code String} represents a string in the UTF-16 format
 * in which <em>supplementary characters</em> are represented by <em>surrogate
 * pairs</em> (see the section <a href="Character.html#unicode">Unicode
 * Character Representations</a> in the {@code Character} class for
 * more information).
 * 一個(gè)String 對(duì)象代表了一個(gè)UTF-16編碼語(yǔ)法組成的字符串
 * Index values refer to {@code char} code units, so a supplementary
 * character uses two positions in a {@code String}.
 * <p>The {@code String} class provides methods for dealing with
 * Unicode code points (i.e., characters), in addition to those for
 * dealing with Unicode code units (i.e., {@code char} values).
 * 索引值指向字符碼單元,所以一個(gè)字符在一個(gè)字符串中使用兩個(gè)位置,
 * String 類提供了一些方法區(qū)處理單個(gè)Unicode編碼,除了那些處理Unicode代碼單元。
 * @since   JDK1.0
 */

以上便是String類注釋的整個(gè)片段,后面剩下的就是作者、相關(guān)類、相關(guān)方法以及從JDK哪個(gè)版本開(kāi)始有的。

String類定義

public final class String
    implements java.io.SerializableComparable<String>, CharSequence 
{
 ....   
 }

類圖


String類被final修飾,表示String不可以被繼承。下面我們來(lái)說(shuō)說(shuō)String實(shí)現(xiàn)三個(gè)接口有什么用處:

  • 實(shí)現(xiàn)Serializable,可以被序列化
  • 實(shí)現(xiàn)Comparable,可以用于比較大?。ò错樞虮容^單個(gè)字符的ASCII碼)
  • 實(shí)現(xiàn)CharSequence,表示是一個(gè)有序字符的序列,(因?yàn)镾tring的本質(zhì)是一個(gè)char類型數(shù)組)

簡(jiǎn)單介紹final

修飾類:類不可被繼承,也就是說(shuō),String類不可被繼承了

修飾方法:把方法鎖定,以訪任何繼承類修改它的涵義

修飾遍歷:初始化后不可更改

重要成員

 /** The value is used for character storage. */
// 來(lái)用存儲(chǔ)String內(nèi)容的
private final char value[];
// 存儲(chǔ)字符串哈希值,默認(rèn)值為0
private int hash; // Default to 0
// 實(shí)現(xiàn)序列化的標(biāo)識(shí)
private static final long serialVersionUID = -6849794470754667710L;

char value[]被final修飾,說(shuō)明value[]數(shù)組是不可變的。

構(gòu)造方法

/**
 * Initializes a newly created {@code String} object so that it represents
 * an empty character sequence.  Note that use of this constructor is
 * unnecessary since Strings are immutable.
 * 初始化新創(chuàng)建的String對(duì)象,時(shí)期表示空字符串序列。
 * 注意:這個(gè)構(gòu)造方法的用法是沒(méi)必要的,因?yàn)樽址遣豢勺兊?br> */

public String() {
        this.value = "".value;
}

無(wú)參構(gòu)造方法中是將一個(gè)空字符串的value值賦給當(dāng)前value。

 /**
  * Initializes a newly created {@code String} object so that it represents
  * the same sequence of characters as the argument; in other words, the
  * newly created string is a copy of the argument string. Unless an
  * explicit copy of {@code original} is needed, use of this constructor is
  * unnecessary since Strings are immutable.
  * 初始化創(chuàng)建的String對(duì)象,時(shí)期表示與參數(shù)相同的字符串序列。
  * 換句話說(shuō):新創(chuàng)建的字符串是參數(shù)自粗糙的副本。
  * 除非,如果需要original的顯示副本,否則也是沒(méi)有必要使用此構(gòu)造方法的
  * 因?yàn)樽址遣豢勺兊?br>  * @param  original
  *         A {@code String}
  */

 public String(String original) {
     this.value = original.value;
     this.hash = original.hash;
 }
//案例:  String str=new String("abc");  

把original的value賦給當(dāng)前的value,并把original的hash賦給當(dāng)前的hash。

/**
 * Allocates a new {@code String} so that it represents the sequence of
 * characters currently contained in the character array argument. The
 * contents of the character array are copied; subsequent modification of
 * the character array does not affect the newly created string.
 * 分配一個(gè)新的{@code String},以便它表示字符數(shù)組參數(shù)中當(dāng)前包含的字符。這個(gè)
 * 復(fù)制字符數(shù)組的內(nèi)容;隨后修改字符數(shù)組不影響新創(chuàng)建的字符串。
 * @param  value
 *         The initial value of the string
 */

public String(char value[]) {
    //注:將傳過(guò)來(lái)的char數(shù)組copy到value數(shù)組里
    this.value = Arrays.copyOf(value, value.length);
}
//Arrays類中的copyOf方法
public static char[] copyOf(char[] original, int newLength) {
    //創(chuàng)建一個(gè)新的char數(shù)組
    char[] copy = new char[newLength];
    //把original數(shù)組中內(nèi)容拷貝到新建的char數(shù)組中
    System.arraycopy(original, 0, copy, 0, Math.min(original.length, newLength));
    //返回新建的char數(shù)組
    return copy;
}

使用Arrays類的copyOf方法,新建一個(gè)char數(shù)組,將original的內(nèi)容放到新建的char數(shù)組中。

然后,把新建的char數(shù)組賦給當(dāng)前的vlaue。

public String(StringBuffer buffer) {
   synchronized(buffer) {
            this.value = Arrays.copyOf(buffer.getValue(), buffer.length());
   }
}

因?yàn)镾tringBuffer是線程安全類,所以,這里加了同步鎖,保證線程安全。

public String(StringBuilder builder) {
        this.value = Arrays.copyOf(builder.getValue(), builder.length());
}

StringBuilder是非線程安全的,這里也就沒(méi)有做線程安全處理,其他內(nèi)容和前面一樣。

注:很多時(shí)候我們不會(huì)這么去構(gòu)造,因?yàn)镾tringBuilder跟StringBuffer有toString方法如果不考慮線程安全,優(yōu)先選擇StringBuilder

這里就講這么多構(gòu)造方法,其他很復(fù)雜,也基本不用,所以,了解這些就夠了。如果對(duì)其他感興趣的,可以自行去研究研究。

常用方法分析

前面的使用案例中,我們已經(jīng)對(duì)String的大部分方法進(jìn)行演示一波,這里我們就挑幾個(gè)相對(duì)重要的方法進(jìn)行深度解析。

hashCode方法

hashCode()方法是在Object類中定義的,String對(duì)其進(jìn)行了重寫(xiě)。

public int hashCode() {
        int h = hash;
        if (h == 0 && value.length > 0) {
            char val[] = value;
            //hash算法,s[0]*31^(n-1) + s[1]*31^(n-2) + ... + s[n-1]
            //使用{@codeint}算法,其中{@codes[i]}是<i> i</i>字符串的第個(gè)字符,
            //{@code n}是字符串,{@code^}表示指數(shù)運(yùn)算。
            for (int i = 0; i < value.length; i++) {
                h = 31 * h + val[i];
            }
            hash = h;
        }
        return h;
}

hashCode的一個(gè)具體實(shí)現(xiàn),由于java體系中每個(gè)對(duì)象都可以轉(zhuǎn)換成String,因此他們應(yīng)該都是通過(guò)這個(gè)hash來(lái)實(shí)現(xiàn)的

接著,我們看看equals()方法;

equals()方法

equals()方法也是Object類中定義的,String類對(duì)其進(jìn)行了重寫(xiě)。

public boolean equals(Object anObject) {
    //首先會(huì)判斷是否是同一個(gè)對(duì)象
     if (this == anObject) {
         return true;
     }
    //判斷是否為String類型
     if (anObject instanceof String) {
         String anotherString = (String)anObject;
         int n = value.length;
         //長(zhǎng)度是否相同
         if (n == anotherString.value.length) {
             char v1[] = value;
             char v2[] = anotherString.value;
             int i = 0;
             //逐個(gè)遍歷判斷是否相等
             //從后往前單個(gè)字符判斷,如果有不相等,返回假
             while (n-- != 0) {
                 //不相等,直接返回false
                 if (v1[i] != v2[i])
                     return false;
                 i++;
             }
             return true;
         }
     }
     return false;
}

補(bǔ)充:==比較

==比較基本數(shù)據(jù)類型,比較的是值
==比較引用數(shù)據(jù)類型,比較的是地址值

substring()方法

substring方法在工作使用的也是相當(dāng)?shù)亩?,作用就是截取一段字符串?/p>

public String substring(int beginIndex) {
    if (beginIndex < 0) {
        throw new StringIndexOutOfBoundsException(beginIndex);
    }
    int subLen = value.length - beginIndex;
    if (subLen < 0) {
        throw new StringIndexOutOfBoundsException(subLen);
    }
    //如果beginIndex==0,返回的是當(dāng)前對(duì)象,
    //否則這里是new的一個(gè)新對(duì)象,其實(shí)String中的很多函數(shù)都是這樣的操作
    return (beginIndex == 0) ? this : new String(value, beginIndex, subLen);
}

intern()方法

intern()方法是native修飾的方法,表示該方法為本地方法。

/*
 * When the intern method is invoked, if the pool already contains a
 * string equal to this {@code String} object as determined by
 * the {@link #equals(Object)} method, then the string from the pool is
 * returned. Otherwise, this {@code String} object is added to the
 * pool and a reference to this {@code String} object is returned.
 */

public native String intern();

方法注釋會(huì)有寫(xiě)到,意思就是調(diào)用方法時(shí),如果常量池有當(dāng)前String的值,就返回這個(gè)值,沒(méi)有就加進(jìn)去,返回這個(gè)值的引用。

案例如下

public class StringDemo {
    public static void main(String[] args) throws Exception {
        String str1 = "a";
        String str2 = "b";
        String str3 = "ab";
        String str4 = str1 + str2;
        String str5 = new String("ab");

        System.out.println(str5 == str3);//堆內(nèi)存比較字符串池
        //intern如果常量池有當(dāng)前String的值,就返回這個(gè)值,沒(méi)有就加進(jìn)去,返回這個(gè)值的引用
        System.out.println(str5.intern() == str3);//引用的是同一個(gè)字符串池里的
        System.out.println(str5.intern() == str4);//變量相加給一個(gè)新值,所以str4引用的是個(gè)新的
        System.out.println(str4 == str3);//變量相加給一個(gè)新值,所以str4引用的是個(gè)新的

    }
}

運(yùn)行結(jié)果

false
true
false
false

length()方法

獲取字符串長(zhǎng)度,實(shí)際上是獲取字符數(shù)組長(zhǎng)度 ,源碼就非常簡(jiǎn)單了,沒(méi)什么好說(shuō)的。

public int length() {
    return value.length;
}

isEmpty() 方法

判斷字符串是否為空,實(shí)際上是盼復(fù)字符數(shù)組長(zhǎng)度是否為0 ,源碼也是非常簡(jiǎn)單,沒(méi)什么好說(shuō)的。

public boolean isEmpty() {
    return value.length == 0;
}

charAt(int index) 方法

根據(jù)索引參數(shù)獲取字符 。

public char charAt(int index) {
    //索引小于0或者索引大于字符數(shù)組長(zhǎng)度,則拋出越界異常
    if ((index < 0) || (index >= value.length)) {
        throw new StringIndexOutOfBoundsException(index);
    }
    //返回字符數(shù)組指定位置字符
    return value[index];
}

getBytes()方法

獲取字符串的字節(jié)數(shù)組,按照系統(tǒng)默認(rèn)字符編碼將字符串解碼為字節(jié)數(shù)組 。

public byte[] getBytes() {
    return StringCoding.encode(value, 0, value.length);
}

compareTo()方法

這個(gè)方法寫(xiě)的很巧妙,先從0開(kāi)始判斷字符大小。如果兩個(gè)對(duì)象能比較字符的地方比較完了還相等,就直接返回自身長(zhǎng)度減被比較對(duì)象長(zhǎng)度,如果兩個(gè)字符串長(zhǎng)度相等,則返回的是0,巧妙地判斷了三種情況。

public int compareTo(String anotherString) {
    //自身對(duì)象字符串長(zhǎng)度len1
    int len1 = value.length;
    //被比較對(duì)象字符串長(zhǎng)度len2
    int len2 = anotherString.value.length;
    //取兩個(gè)字符串長(zhǎng)度的最小值lim
    int lim = Math.min(len1, len2);
    char v1[] = value;
    char v2[] = anotherString.value;
 
    int k = 0;
    //從value的第一個(gè)字符開(kāi)始到最小長(zhǎng)度lim處為止,如果字符不相等,
    //返回自身(對(duì)象不相等處字符-被比較對(duì)象不相等字符)
    while (k < lim) {
        char c1 = v1[k];
        char c2 = v2[k];
        if (c1 != c2) {
            return c1 - c2;
        }
        k++;
    }
    //如果前面都相等,則返回(自身長(zhǎng)度-被比較對(duì)象長(zhǎng)度)
    return len1 - len2;
}

startsWith()方法

public boolean startsWith(String prefix, int toffset) {
    char ta[] = value;
    int to = toffset;
    char pa[] = prefix.value;
    int po = 0;
    int pc = prefix.value.length;
    // Note: toffset might be near -1>>>1.
    //如果起始地址小于0或者(起始地址+所比較對(duì)象長(zhǎng)度)大于自身對(duì)象長(zhǎng)度,返回假
    if ((toffset < 0) || (toffset > value.length - pc)) {
        return false;
    }
    //從所比較對(duì)象的末尾開(kāi)始比較
    while (--pc >= 0) {
        if (ta[to++] != pa[po++]) {
            return false;
        }
    }
    return true;
}
 
public boolean startsWith(String prefix) {
    return startsWith(prefix, 0);
}
 
public boolean endsWith(String suffix) {
    return startsWith(suffix, value.length - suffix.value.length);
}

起始比較和末尾比較都是比較經(jīng)常用得到的方法,例如:在判斷一個(gè)字符串是不是http協(xié)議的,或者初步判斷一個(gè)文件是不是mp3文件,都可以采用這個(gè)方法進(jìn)行比較。

concat()方法

public String concat(String str) {
    int otherLen = str.length();
    //如果被添加的字符串為空,返回對(duì)象本身
    if (otherLen == 0) {
        return this;
    }
    int len = value.length;
    char buf[] = Arrays.copyOf(value, len + otherLen);
    str.getChars(buf, len);
    return new String(buf, true);
}

concat方法也是經(jīng)常用的方法之一,它先判斷被添加字符串是否為空來(lái)決定要不要?jiǎng)?chuàng)建新的對(duì)象。

replace()方法

public String replace(char oldChar, char newChar) {
    //新舊值先對(duì)比
    if (oldChar != newChar) {
        int len = value.length;
        int i = -1;
        char[] val = value; 
 
        //找到舊值最開(kāi)始出現(xiàn)的位置
        while (++i < len) {
            if (val[i] == oldChar) {
                break;
            }
        }
        //從那個(gè)位置開(kāi)始,直到末尾,用新值代替出現(xiàn)的舊值
        if (i < len) {
            char buf[] = new char[len];
            for (int j = 0; j < i; j++) {
                buf[j] = val[j];
            }
            while (i < len) {
                char c = val[i];
                buf[i] = (c == oldChar) ? newChar : c;
                i++;
            }
            return new String(buf, true);
        }
    }
    return this;
}

這個(gè)方法也有討巧的地方,例如最開(kāi)始先找出舊值出現(xiàn)的位置,這樣節(jié)省了一部分對(duì)比的時(shí)間。replace(String oldStr,String newStr)方法通過(guò)正則表達(dá)式來(lái)判斷。

trim()方法

public String trim() {
    int len = value.length;
    int st = 0;
    char[] val = value;    /* avoid getfield opcode */
 
    //找到字符串前段沒(méi)有空格的位置
    while ((st < len) && (val[st] <= ' ')) {
        st++;
    }
    //找到字符串末尾沒(méi)有空格的位置
    while ((st < len) && (val[len - 1] <= ' ')) {
        len--;
    }
    //如果前后都沒(méi)有出現(xiàn)空格,返回字符串本身
    return ((st > 0) || (len < value.length)) ? substring(st, len) : this;
}

trim方法就是將字符串中的空白字符串刪掉。

valueOf()方法

public static String valueOf(boolean b) {
   //如果b為true就返回"true"否則返回"false"
   return b ? "true" : "false";
}
public static String valueOf(char c) {
    //創(chuàng)建data[]數(shù)組 并把c添加進(jìn)去
    char data[] = {c};        
     //創(chuàng)建一個(gè)新的String對(duì)象并進(jìn)行返回
    return new String(data, true); 
}
public static String valueOf(int i) {
    //調(diào)用Integer對(duì)象的toString()方法并進(jìn)行返回
    return Integer.toString(i);  
}
//Integer類中的toString(i)方法
public static String toString(int i) {
    //是否為Integer最小數(shù),是直接返回
    if (i == Integer.MIN_VALUE)
       return "-2147483648";
    //這個(gè)i有多少位
    int size = (i < 0) ? stringSize(-i) + 1 : stringSize(i);
    //創(chuàng)建一個(gè)char數(shù)組
    char[] buf = new char[size];
    //把i內(nèi)容方法char數(shù)組中區(qū)
    getChars(i, size, buf);
    //返回一個(gè)String對(duì)象
    return new String(buf, true);
}

split() 方法

public String[] split(String regex) {
    return split(regex, 0);
}
//使用到了正則表達(dá)式
public String[] split(String regex, int limit) {
      //....
    //源碼有點(diǎn)多了,反正就是里面使用到了正則表達(dá)式,進(jìn)行切分
    }

split() 方法用于把一個(gè)字符串分割成字符串?dāng)?shù)組,返回一個(gè)字符串?dāng)?shù)組返回的數(shù)組中的字串不包括 regex自身??蛇x的“limit”是一個(gè)整數(shù),第一個(gè)方法中默認(rèn)是0,允許各位指定要返回的最大數(shù)組的元素個(gè)數(shù)。

常見(jiàn)方法源碼分析就這么多了,下面我們?cè)倩仡櫟绞褂脠?chǎng)景中來(lái),尤其是面試中。

String在面試中常見(jiàn)問(wèn)題

如何比較字符串相同?

在java中比較對(duì)象是否相同,通常有兩種方法:

  • ==
  • equals方法

注意==用于基本數(shù)據(jù)類型的比較和用于引用類型的比較的區(qū)別。

==比較基本數(shù)據(jù)類型,比較的是值

==比較引用數(shù)據(jù)類型,比較的是地址值

另外,String對(duì)equals方法進(jìn)行了重寫(xiě),所以比較字符串咱們還是要使用equals方法來(lái)比較。主要是Stringequals方法里包含了==的判斷(請(qǐng)看前面源碼分析部分)。

案例

public class StringDemo {
   public static void main(String[] args) {
     String st1 = "abc";
     String st2 = "abc";
     System.out.println(st1 == st2);
     System.out.println(st1.equals(st2)); 
   }
}

輸出

true
true

String str=new String("abc");這行代碼創(chuàng)建了幾個(gè)對(duì)象?

看下面這段代碼:

String str1 = "abc";  // 在常量池中
String str2 = new String("abc"); // 在堆上

關(guān)于這段代碼,創(chuàng)建了幾個(gè)對(duì)象,網(wǎng)上答案有多重,1個(gè),2個(gè)還有3個(gè)的。下面我們就來(lái)聊聊到底是幾個(gè)?

首先,我們需要明確的是;不管是str1還是str2,他們都是String類型的變量,不是對(duì)象,平時(shí),可能我們會(huì)叫str2對(duì)象,那只是為了便于理解,本質(zhì)上來(lái)說(shuō)str2、str1都不是對(duì)象。

其次,String str="abc";的時(shí)候,字符串“abc”會(huì)被存儲(chǔ)在字符串常量池中,只有1份,此時(shí)的賦值操作等于是創(chuàng)建0個(gè)或1個(gè)對(duì)象。如果常量池中已經(jīng)存在了“abc”,那么不會(huì)再創(chuàng)建對(duì)象,直接將引用賦值給str1;如果常量池中沒(méi)有“abc”,那么創(chuàng)建一個(gè)對(duì)象,并將引用賦值給str1。

那么,通過(guò)new String("abc");的形式又是如何呢?

答案是1個(gè)或2個(gè)。

當(dāng)JVM遇到上述代碼時(shí),會(huì)先檢索常量池中是否存在“abc”,如果不存在“abc”這個(gè)字符串,則會(huì)先在常量池中創(chuàng)建這個(gè)一個(gè)字符串。然后再執(zhí)行new操作,會(huì)在堆內(nèi)存中創(chuàng)建一個(gè)存儲(chǔ)“abc”的String對(duì)象,對(duì)象的引用賦值給str2。此過(guò)程創(chuàng)建了2個(gè)對(duì)象。

當(dāng)然,如果檢索常量池時(shí)發(fā)現(xiàn)已經(jīng)存在了對(duì)應(yīng)的字符串,那么只會(huì)在堆內(nèi)創(chuàng)建一個(gè)新的String對(duì)象,此過(guò)程只創(chuàng)建了1個(gè)對(duì)象。

最后,如果單獨(dú)問(wèn)String str=new String("abc");創(chuàng)建了幾個(gè)對(duì)象,切記:常量池中是否存在"abc",存在,創(chuàng)建一個(gè)對(duì)象;不存在創(chuàng)建兩個(gè)對(duì)象。

String 和 StringBuilder、StringBuffer 的區(qū)別

線程安全性

String 中的對(duì)象是不可變的,也就可以理解為常量,線程安全。AbstractStringBuilder 是 StringBuilder 與 StringBuffer 的公共父類,定義了一些字符串的基本操作,如 expandCapacity、append、insert、indexOf 等公共方法。StringBuffer 對(duì)方法加了同步鎖或者對(duì)調(diào)用的方法加了同步鎖,所以是線程安全的。StringBuilder 并沒(méi)有對(duì)方法進(jìn)行加同步鎖,所以是非線程安全的。

性能

每次對(duì) String 類型進(jìn)行改變的時(shí)候,都會(huì)生成一個(gè)新的 String 對(duì)象,然后將 指針指向新的 String 對(duì)象。StringBuffer 每次都會(huì)對(duì) StringBuffer 對(duì)象本身進(jìn)行操作,而不是生成新的對(duì)象并改變對(duì)象引用。相同情況下使用 StringBuilder 相比使用 StringBuffer 僅能獲得 10%~15% 左右的性能提升,但卻要冒多線程不安全的風(fēng)險(xiǎn)。

對(duì)于三者使用的總結(jié):

  • 操作少量的數(shù)據(jù) ,推薦使用String
  • 單線程操作字符串緩沖區(qū)下操作大量數(shù)據(jù),推薦使用 StringBuilder
  • 多線程操作字符串緩沖區(qū)下操作大量數(shù)據(jù) ,推薦使用 StringBuffer

String 和 JVM有什么關(guān)系?

String 常見(jiàn)的創(chuàng)建方式有兩種,new String() 的方式和直接賦值的方式,直接賦值的方式會(huì)先去字符串常量池中查找是否已經(jīng)有此值,如果有則把引用地址直接指向此值,否則會(huì)先在常量池中創(chuàng)建,然后再把引用指向此值;而 new String() 的方式一定會(huì)先在堆上創(chuàng)建一個(gè)字符串對(duì)象,然后再去常量池中查詢此字符串的值是否已經(jīng)存在,如果不存在會(huì)先在常量池中創(chuàng)建此字符串,然后把引用的值指向此字符串。

JVM中的常量池


字面量—文本字符串,也就是我們舉例中的 public String s = " abc "; 中的 "abc"。

用 final 修飾的成員變量,包括靜態(tài)變量、實(shí)例變量和局部變量。

請(qǐng)看下面這段代碼:

String s1 = new String("Java");
String s2 = s1.intern();
String s3 = "Java";
System.out.println(s1 == s2); // false
System.out.println(s2 == s3); // true

它們?cè)?JVM 存儲(chǔ)的位置,如下圖所示:


注意:JDK 1.7 之后把永生代換成的元空間,把字符串常量池從方法區(qū)移到了 Java 堆上。

除此之外編譯器還會(huì)對(duì) String 字符串做一些優(yōu)化,例如以下代碼:

String s1 = "Ja" + "va";
String s2 = "Java";
System.out.println(s1 == s2);

雖然 s1 拼接了多個(gè)字符串,但對(duì)比的結(jié)果卻是 true,我們使用反編譯工具,看到的結(jié)果如下:

Compiled from "StringExample.java"
public class com.lagou.interview.StringExample {
  public com.lagou.interview.StringExample();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return
    LineNumberTable:
      line 3: 0
  public static void main(java.lang.String[]);
    Code:
       0: ldc           #2                  // String Java
       2: astore_1
       3: ldc           #2                  // String Java
       5: astore_2
       6: getstatic     #3                  // Field java/lang/System.out:Ljava/io/PrintStream;
       9: aload_1
      10: aload_2
      11: if_acmpne     18
      14: iconst_1
      15: goto          19
      18: iconst_0
      19: invokevirtual #4                  // Method java/io/PrintStream.println:(Z)V
      22: return
    LineNumberTable:
      line 5: 0
      line 6: 3
      line 7: 6
      line 8: 22
}

從編譯代碼 #2 可以看出,代碼 "Ja"+"va" 被直接編譯成了 "Java" ,因此 s1==s2 的結(jié)果才是 true,這就是編譯器對(duì)字符串優(yōu)化的結(jié)果。

如何判斷兩個(gè)字符串中含有幾個(gè)相同字符

  • 將字符串轉(zhuǎn)化成數(shù)組
  • HashMap 方法
  • 字符串直接進(jìn)行比較
  • 正則表達(dá)式
  • HashSet 方法

String有沒(méi)有長(zhǎng)度限制?是多少?為什么?

下面先看看length方法源碼:

private final char value[];
public int length() {
        return value.length;
}

length()方法返回的是int類型,那可以得知String類型的長(zhǎng)度肯定不能超過(guò)Integer.MAX_VALUE的。

答:首先字符串的內(nèi)容是由一個(gè)字符數(shù)組 char[] 來(lái)存儲(chǔ)的,由于數(shù)組的長(zhǎng)度及索引是整數(shù),且String類中返回字符串長(zhǎng)度的方法length() 的返回值也是int ,所以通過(guò)查看java源碼中的類Integer我們可以看到Integer的最大范圍是2^31 -1,由于數(shù)組是從0開(kāi)始的,所以**數(shù)組的最大長(zhǎng)度可以使【0~2^31】**通過(guò)計(jì)算是大概4GB。

但是通過(guò)翻閱java虛擬機(jī)手冊(cè)對(duì)class文件格式的定義以及常量池中對(duì)String類型的結(jié)構(gòu)體定義我們可以知道對(duì)于索引定義了u2,就是無(wú)符號(hào)占2個(gè)字節(jié),2個(gè)字節(jié)可以表示的最大范圍是2^16 -1 = 65535

但是由于JVM需要1個(gè)字節(jié)表示結(jié)束指令,所以這個(gè)范圍就為65534了。超出這個(gè)范圍在編譯時(shí)期是會(huì)報(bào)錯(cuò)的,但是運(yùn)行時(shí)拼接或者賦值的話范圍是在整形的最大范圍。

字符串對(duì)象能否用在switch表達(dá)式中?

JDK7開(kāi)始的話,我們就可以在switch條件表達(dá)式中使用字符串了,也就是說(shuō)7之前的版本是不可以的。

switch (str.toLowerCase()) {
      case "tian":
           value = 1;
           break;
      case "jiang":
           value = 2;
           break;
}

說(shuō)說(shuō)String中intern方法

JDK7之前的版本,調(diào)用這個(gè)方法的時(shí)候,會(huì)去常量池中查看是否已經(jīng)存在這個(gè)常量了,如果已經(jīng)存在,那么直接返回這個(gè)常量在常量池中的地址值,如果不存在,則在常量池中創(chuàng)建一個(gè),并返回其地址值。

但是在JDK7以及之后的版本中,常量池從perm區(qū)搬到了heap區(qū)。intern檢測(cè)到這個(gè)常量在常量池中不存在的時(shí)候,不會(huì)直接在常量池中創(chuàng)建該對(duì)象了,而是將堆中的這個(gè)對(duì)象的引用直接存到常量池中,減少內(nèi)存開(kāi)銷。

下面的案例

public class InternTest {
  
  public static void main(String[] args) {
    String str1 = new String("hello") + new String("world");
    str1.intern();
    String str2 = "helloworld";
    System.out.println(str1 == str2);//true
    System.out.println(str1.intern() == str2);//true
  }
}

好了,關(guān)于Stirng類的分享就到此,歡迎找我探討更多技術(shù)問(wèn)題。

另外一篇關(guān)于String類的文章:美團(tuán)面試題:String s = new String("111")會(huì)創(chuàng)建幾個(gè)對(duì)象?

參考

http://prren.cn/vWv68 、http://prren.cn/7ICgG、http://dsblb.cn/ku78V、王磊《 Java 源碼剖析 34 講》

 

推薦閱讀:
億級(jí)系統(tǒng)的Redis緩存如何設(shè)計(jì)
學(xué)會(huì)這10個(gè)設(shè)計(jì)原則,離架構(gòu)師又進(jìn)了一步
Spring Boot 集成 Kafka

關(guān)號(hào)互聯(lián)網(wǎng)全棧架構(gòu),價(jià)。

瀏覽 62
點(diǎn)贊
評(píng)論
收藏
分享

手機(jī)掃一掃分享

分享
舉報(bào)
評(píng)論
圖片
表情
推薦
點(diǎn)贊
評(píng)論
收藏
分享

手機(jī)掃一掃分享

分享
舉報(bào)

感谢您访问我们的网站,您可能还对以下资源感兴趣:

国产秋霞理论久久久电影-婷婷色九月综合激情丁香-欧美在线观看乱妇视频-精品国avA久久久久久久-国产乱码精品一区二区三区亚洲人-欧美熟妇一区二区三区蜜桃视频 日韩精品极品视频在线观看免费| 日韩成人精品视频| 国产A片网站| 亚洲婷婷在线观看| 黄片www.| 无码成人视频| 久久精品免费| 日韩精品人妻中文字幕蜜乳| 日韩欧美高清第一期| 无码AV免费观看| 国产麻豆一区二区三区| 日韩无码你懂的| 国产青娱乐在线视频| 亚洲秘无码一区二区三区欧美| 久久成人片| 操比片| 91人妻综合| 久久久xxx| 亚洲精品成人网站| 久久亚洲AV无码午夜麻豆| 怡春院熟女精品AV| 亚洲品久久久蜜| 俺去俺来也www色视频| 亚洲精品456| 久久亭亭| 色999日韩| 成人精品秘免费波多野结衣| 男人操女人免费网站| 一级片操逼| 俺也去电影| 中国无码专区| 日韩欧美综合| 国产黄色视频在线| 大地影院在线资源观看| 白浆AV| 亚洲综合另类| 国产一级黄色录像| 日韩aaaaaa| 91精品在线观看视频| 国产精品秘ThePorn| 国产中文字幕在线免费观看| 操逼爆奶网站| 三级高清无码视频| 亚洲国精产品| 色av影音先锋无吗一区| 91大长腿美女花外围在线观看| 黄色成人在线观看| 国产AV天堂| 中文字幕人妻无码| 午夜成人在线| 久久久免费| 日韩成人无码电影| 日韩精品一二三| 久久精品中文| 高潮国产视频| 蜜臀精品一区二区三区| 亚洲成人影片在线观看| 亚洲成人AV无码| 青草久久视频| 男人的天堂在线播放| 亚洲AV无码专区在线播放中文| 免费无码婬片AAAA片在线蜜芽 | 男人亚洲天堂| 国产伊人在线| 熟妇在线| 18禁黄色免费网站| 色婷婷AV一区二区三区之e本道| 一级黄色在线| 囯产精品久久久久久久久免费无码 | 亚洲青娱乐在线| 国产婷婷五月| 7777精品伊人久久7777| 亚洲一区视频| 人人操人人干人人摸| 青草无码视频| 亚洲va视频| 国产做爱| www.黄色电影| 成人在线毛片| 熟女人妻ThePorn| 韩国免费一级a一片在线播放| 免费看黄色片视频| 九九九九九九精品视频| 色悠悠中文字幕| 国产性播放| 日本老妇操屄视频| 国产福利视频| 日本免费黄色视频| 国产无码一区二区| 免费v片| 免费黄片在线看| 久色| 日韩激情网| 亚洲成人在线免费观看| 无码av在线观看| 日本AⅤ在线| 国产一级a爱做片免费☆观看| 日韩免费在线观看视频| 黄色小视频免费观看| 午夜a片| 欧美淫乱视频| 欧美日韩精品一区二区| 婷婷午夜精品久久久久久性色| 日本免费不卡视频| 麻豆91精品91久久久| 欧洲美一区二区三区亚洲| 午夜福利成人网站| а天堂中文在线资源| 日本熟妇一区二区三区| 翔田千里AV在线| 日韩午夜| 天天综合久久| 国产性爱精品影片免费看| 久草视频99| 亚洲精品无码人妻| 超碰91免费在线观看| 大香蕉玖玖| 中文字幕乱码在线| 四虎福利| 黄色无码av| 免费看一级黄色片| 国产天堂网| 嫩草在线观看| 91精品电影18| 欧美三级在线| 黄色激情AV| 国产成人精品国内自产拍免费看| 人人摸人人操人人爱| 人妻HDHDHD96XXXX| 中字无码制服| 日皮视频在线| 一区二区三区免费看| 国产区av| 99re在线观看| 98在线++传媒麻豆的视频| 亚洲AV色香蕉一区二区三区| 国产精品九九视频| 国产精品资源在线观看| 波多野结衣大战黑人| 内射视频在线免费观看| 中文在线高清字幕| 91在线无码精品秘入口| 91啪啪| 粉嫩一区| 欧美日韩视频免费观看| 北条麻妃AV观看| 3d动漫精品一区二区三区在线观看 | 欧美性爱高清| 999福利视频| 欧美久久久久久久| 青青草中文字幕| 日韩熟妇无码中文字幕| 18禁网站在线| 色色激情视频| 中文字幕高清视频| 久久精彩免费视频| 成人在线毛片| 黑人一级| 美女被操网站| 曰本精品综合网在线| 黄色激情av| 一级特黄毛片| 久久新视频| 91AV| 亚洲性爱AV| 无套影院| 精品一本道| 午夜福利电影AV| 色色播播| 你懂的在线观看| 人人妻人人摸| 亚洲黄色视频免费| 亚洲日逼网站| 亚洲在线大香蕉| 国产欧美综合一区二区三区| 99在线免费观看| 国产欧美在线综合| 欧美系列在线| 欧美国产综合| 大荫蒂hd大荫蒂视频| 超碰成人在线观看| 欧美伊人网在线观看| 久久艹精品视频| 99久久婷婷国产综合精品| 国产在线免费视频| 影音先锋成人av| 亚洲激情在线| 亚洲天堂AV2025| 欧美成人网站免费在线观看| 中文字幕乱码中文字幕| 三级网站免费| 中文字幕一区二区蜜桃| 一级a一级a爰片免费免免中国A片| 亚洲国际中文字幕在线| 大香蕉88| 亚洲秘无码一区二区三区| 日韩在线播放视频| 免费看A| 怡春院中文字幕| 成人动漫免费观看| 日韩久久中文字幕| 亚洲色图一区二区三区| 亚洲成人自拍| 99久久久久久久久久| 日韩一区二区三区精品| 福利导航在线| 嫩草在线观看| 黄色免费毛片| 日本乱伦网站| 五月丁香天堂网| 精品人妻一区二区三区在| 亚洲天堂福利| 韩国人妻无码| 少妇搡BBBB搡BBB搡视频一级| AV在线资源| 日韩干网| 亚洲秘av无码一区二区| 丁香五月婷婷综合| 国产成人宗合| 特级特黄AAAA免费看| 欧美婷婷综合| 爱爱毛片| 青青草做爱视频| 精品人妻一区二区免费蜜桃 | 人妻丝袜无码视频专区| 影音先锋天堂| 四虎在线视频观看96| 在线视频A| 日本中文字幕在线播放| 天天肏| 亚洲日日夜夜| 亚洲天堂精品在线| 亚洲午夜视频在线观看| 成人A电影| 日韩三级在线播放| 青青无码| 肉片无遮挡一区二区三区免费观看视频| www.jiujiujiu| 成人电影无码| 91无码精品| 一道本不卡视频| 影音先锋乱伦| 色色欧美| 麻豆国产成人AV一区二区三区| 日韩精品在线视频观看| 中文日韩字幕| 久久中文字幕综合| 九九九九综合| 日本十八禁网站| BBWBBw嫩| 亚洲成人精品AV| 乱伦无码视频| 亚洲精品成人| 超碰综合| 欧美日韩精品一区二区| 女人的天堂AV在线观看| av一区在线观看| 干屄网| 亚洲字幕在线播放| Av高清无码| 色丁香婷婷| 欧美色成人免费在线视频| 97久久综合| 在线有区别亚洲| 色妞视频精品一区| 手机看片国产| 91无码精品国产| 亚洲无码日| 欧美极品另类| 爱草视频| 亚洲99热| 国产在线网址| 91精品在线播放| 人人舔人人草| 成人在线h| 久久精品免费看| 黃色一級片黃色一級片尖叫声-百度-百 | 人妻公日日澡久久久| 日韩色婷婷| 97在线免费视频| 欧美性爱日韩| 男人日女人视频| 久久国产乱子伦精品免费女,网站| 人人操人人上| 西西4444WWW无码精品| 亚洲国产高清在线观看视频| 翔田千里无码播放| 波多野结衣无码AV在线| 中文字幕无码精品| 亚洲成人五月天| 黄色视频| 无码中文一区| 日韩欧美色图| 性爱xxxxx| 国产精品在线观看视频| 欧美成人一区免费视频| 开心激情婷婷| 久久性| 亚洲视频天天射| 国产17c精品视频一二三区 | 中文字幕精品在线免费视频观看视频 | 欧美AAA视频| 五月婷婷性爱| www.俺去了| 91在线观看| 欧美性交网| 欧美日韩国产一区二区三区| 亚洲无码一区二区三| 8x8x黄色| 欧美性爱天天| 欧美a区| 人人人操| 操美女视频网站| 色欲av在线| 少妇人妻一区二区三区| 国产黄色视频在线免费观看| 一区二区三区麻豆| 亚洲性爱影院| 在线观看视频日韩| 中文字幕欧美视频| 中文字幕在线看成人电影| 亚洲一区二区三区在线播放| 亚洲国产成人精品女人| 亚洲无码中文视频| 99在线视频免费观看| 99热青青| 一级a一级a免费观看视频Al明星 | 51午夜| 国产成人自拍视频在线| 在线看片你懂的| 91白丝在线观看| 日本一区免费| 青青青青操| 亚洲成人影片在线观看| 米奇7777狠狠狠狠| 国产黄色视频在线观看免费| 国产中文字幕亚洲综合欧美| 中文一区在线| 国产香蕉视频| 操逼逼网站| 日韩欧美国产| 成人AV一区二区三区| 欧美色图15p| 黄色在线视频观看| 久久高清免费视频| 91西安站街老熟女露脸| 国产无码专区| 亚洲尤物在线| 欧美第一页| 人人做人人做人人做,人人做全句下一 | 北条麻妃波多波多野结衣| 日本无码在线| www俺来也com| 久久草| 国产精品theporn| 三级片AAA成人免费| 久久大鸡巴| 日本欧美在线观看高清| 亚洲一区欧美| 国产精品HongKong麻豆| 五月天亚洲色图| 91狠狠爱| 怮交小拗女小嫩苞视频| 欧美理伦| 中文字幕2018第一页| 91一区二区在线播放精品| 欧美囗交荫蒂AAAA| 国产女人水真多18毛片18精品| 亚洲日韩在线观看视频| 97亚洲国产| 一级免费黄片| 大香蕉免费| 中文字幕综合网| 五月激情六月| 色五月婷婷AV| 欧美激情另类| 蜜桃av久久久亚洲精品| 操人妻| AV国产精品| 欧美性网| 蜜臀久久99精品久久久久久婷婷| 国产一级黄片| 亚洲天堂精品视频| 水蜜桃视频网站在线观看| 亚洲精品成a人在线观看| 91视频首页| 影音先锋三级片| 人人搞人人操| 人人操人人| 国产精品免费人成人网站酒店| 欧美亚洲综合在线观看| 四虎精品成人无码A片| 国产一精品一aⅴ一免费| 免费在线观看毛片| 一区二区A片| 欧美三级网站| 久热精品免费| 91在线无码精品秘软件| 亚洲天堂无码在线观看| 日本一区免费观看| 91青青草| 国产乱子伦精品免费,| 亚洲无码午夜| 高清视频一区二区| 色哟哟无码精品一区二区三区| 亚洲天天干| 日日操夜夜爽| 青青草国产在线视频| 成人区精品一区二区婷婷| 在线观看无码高清视频| 在线观看无码视频| 99性爱| 97色色婷婷五月天| 亚洲性爱视频在线观看| 九九人妻| 肏婷婷| 国产福利一区二区| 2018天天日天天操| 久操| 伊人九九热| 日韩AV无码高清| 成人午夜免费视频| AV乱伦小说| 国产一级免费在线观看| 草久在线视频| 操逼视频网站免费| 日韩黄色免费视频| 韩日精品视频| 青青操视频在线| 欧美激情色色| 一级在线| 欧美成人免费在线| 午色婷婷国产无码| 在线播放JUY-925被丈夫上司侵犯的第7天 | 鸡巴网站| 国产精品午夜福利视频| 九月婷婷综合| 国产91无码网站在线观看| 国产学生妹| 好爽~要尿了~要喷了~同桌 | 极品无码| av在线一区二区三区| 性做久久久久久| 在线免费观看黄色电影| 国产成人综合在线| 精品国产乱子伦一区二区三区,小小扐 | 国产人人爽| 国产高清视频在线| 欧美性性生交XXXXX无码| 亚洲国产精品成人综合色在线婷婷| 激情视频网站| 开心五月激情婷婷| 夜夜夜夜撸| 日本无码一区二区三三| 中文字幕在线国产| 精品一区二区免费视频| 99视频在线| 日韩性爱AV| 青娱乐精品在线视频| 丁香五月激情中文字幕| 少妇嫩搡BBBB搡BBBB| 不卡成人| 成人国产精品秘欧美高清| 污视频在线免费观看| 伊人精品A片一区二区三区| 蜜桃无码一区| 高清无码黄片| 91在线| 国产最新av| 白丝自慰网站| 日韩中文字幕av| 欧美中文字幕在线播放| 欧美午夜电影| 亚洲第一网无码性色| 亚洲国产成人综合| 日本AA片视频| 欧美一级黄色片| 亚洲中文字幕影院| 学生妹一级片内射视频| 大地8免费高清视频观看大全| 成年人观看视频| 国产精品不卡一区二区三区| 在线观看亚洲无码视频| 99久久精品国产毛片| 国产色哟哟| 天天撸在线| 精品亚洲一区二区三区| 黄色成人18| 波多野结衣国产| 黄色免费av| 99成人电影| 丰滿老婦BBwBBwBBw| 操操干| 国产免费av在线| 免费av一区二区| 天天爽天天干| 日本AⅤ在线观看| 精品无码一区二区三区在线| 狠狠大香蕉| 99久久综合| 俺来也俺也啪WWW色| www.199麻豆在线观看网站| 特级毛片av| 国产福利视频在线| 日本欧美视频| 日韩大鸡巴| aaa精品| 狠狠躁夜夜躁人人爽人妻| 一区二区三区四区久久| 玖玖资源站中文字幕| 爱爱视频天天操| 国产欧美二区综合中文字幕精品一 | www.日批| 日韩欧美高清在线| 久久无码免费| 人妻大香蕉| 大香蕉尹在线| 日本一级视频| 午夜老司机福利一二三区| 欧美爱| 波多野结衣视频网站| 国产天堂网| 一区二区三区免费播放| 91精品午夜少妇| 菊花插综合网| 尤物av| 久操免费在线视频| 美女操逼图| 国产又粗又大又长| 天天夜夜爽| 成人A毛片| 国产无码成人免费| 欧美激情无码一区二区三区张丽| 超碰在线人妻| 一级片AA| 欧美综合亚洲图片综合区| 久久精品国产视频| 亚洲无码成人片| 大香蕉伊人婷婷| 中文字幕日韩有码| 人人操成人| 国产无套进入免费| 日韩无码一级片| 高清AV在线| 乱子伦国产精品视频一级毛| 欧美韩日一区二区| 久久久久三级| 一区二区三区精品| 天天噜| 成人无码人妻| 亚洲日本中文| 翔田千里无码A片| 苍井空一区二区三区四区| 日本久久高清| 国产三级毛片| 欧美日韩免费在线观看| 免费视频一区二区三区四区 | 激情五月激情综合网| 无码激情18激情视频| 成人婷婷五月| 青青操在线视频| 黄色电影A片| 日韩成人黄色电影| 成人区人妻精品一| 五月丁香电影| 中文无码AV在线| 色综合五月婷婷| 婷婷开心色四房播播在线| www.99| 亚洲影院在线观看| 91超碰免费| 九九综合伊人7777777| 无套内射无码| 午夜久久福利| 欧美一级无码| 天天天天毛片| 娇小,学生,高潮,videos| 日韩三级在线| 五月天社区| 久久99久久99久久99| 久热精品在线观看视频| 国内自拍视频网| 极品久久久久| 亚洲中文免费| 黄色一级爱爱| 色色色色色欧美| 精品内射| 亚洲精品秘一区二区三线观看 | 亚洲综合天堂| 久操不卡| 午夜激情免费| 人妻无码一区二区三区| 91精品丝袜久久久久久| 精品玖玖| 操逼网视频| 成人性爱毛片| 亚洲成人少妇老妇a视频在线| 草逼com| 日韩中文字幕av| 91久久国产性奴调教| 五月天伊人| 欧美高潮喷水| 国产精品不卡在线观看| 欧美日韩不卡视频| 狠狠干狠狠艹| 在线黄色视频网站| 国精品无码A区一区二区| www.黄色大片| 91精品国产人妻| 免费V片在线观看| 久久嫩草精品久久久久精| 人人摸人人草| 国产一级a毛一级a| 你懂的视频在线| 91蝌蚪| 日韩午夜| 操啊操| 伊人久久免费视频| 极品久久久| 九色PORNY蝌蚪视频| 久久不卡视频| 美女操逼网站| 色呦呦在线| 日韩av在线电影| 人人操人人搞| 伊人国产视频| 在线网址你懂的| 国产亲子乱XXXXimim/| 中文字幕第23页| 成人免费操| 六月激情丁香| 久久亚洲精品视频| P站免费版-永久免费的福利视频平台 | 特级西西444www大胆免费看| 中文字幕+乱码+中文乱码91| 亚洲资源站| 大香蕉久久草| 国产123区| 97人妻精品一区二区三区| 国内自拍第一页| 黄页网站在线免费观看| 亚洲天堂免费| 先锋影音资源av| 无码免费一区二区| 东北嫖老熟女一区二区视频网站| 在线免费观看黄| 豆花视频成人网站入口免费观看 | 午夜a片| 色综合一区| 国产乱伦自拍| 俺来也俺就去www色情网| 国产精品秘麻豆免费版现看视频| 日逼黄色| 高清无码一区二区三区四区| 国产一区二| 欧美精品网站| 日本A级视频| 成人精品免费| 99热最新网址| 日本人妻中文字幕| 尤物av| 亚洲成人黄色电影| 色男人天堂| 在线不卡免费Av| 色丁香视频在线观看的| 操久久久久久| 一本久久A精品一合区久久久| www五月天com| 天天干天天在线观看| 中文字幕成人网站中文字幕| 国产在线接入| 日韩午夜片| 国产日韩二区| 伊人青青操| 激情无码一区二区三区| 抽插免费视频| 日本女人牲交视频| 五月婷婷综合在线| 2025av天堂| 天天操夜夜撸| 一道本AV| 欧美成人激情| 成人超碰在线| 99九九网| 翔田千里与黑人50分钟| 国产人成视频免费观看| 亚洲高清无码在线观看视频| 天天爱夜夜爱| 蜜臀久久99久久久久久宅男| 成人精品一区二区无码| 久久天天拍| 亚洲免费视频网| 99热在线观看免费精品| 日韩A∨视频| 日韩欧美中文| 国产精品欧美7777777| 日韩欧美高清视频| 免费一区二区三区四区| 日本久久久久久久久视频在线观看| 中文字幕高清无码免费视频| 亚洲va欧美va天堂v国产综合| 苏妲己一级婬片A片| 国产女人18水真多18精品| 黑人乱伦| а中文在线天堂精品| 操操色| 丝袜乱伦| 成人乱无码AV在线观看| 亚洲综合中文| 99久久精品国产毛片| 成人社区视频| 精品人妻一区二区三区日产乱码| 天天看高清无码| 激情五月俺也去| 国产精品不卡一区二区三区| 国产小电影在线| 国产一区二| 日韩无码国产精品| 91熟女丰满原味| 国产在线久久久| 麻酥酥在线视频| 北条麻妃99精品青青久久| 高清无码爱爱| 欧美成人福利| 六月婷婷久久| 夜夜操天天干| 91青青草| 四虎在线观看视频| 丁香六月婷| 五月激情六月婷婷| 日啪| 国产综合精品久久久久成人AV| 欧美一级特黄A片免费看视频小说 东北嫖老熟女一区二区视频网站 国产丨熟女丨国产熟女视频 | 欧美一级视频| 麻豆视频在线看| 久久国产一区二区| 夜色88V精品国产亚洲| 小h片| 无码av免费精品一区二区三区| 91免费小视频| AV解说| 欧美不卡在线视频| 骚BBBB槡BBB槡BBB| 懂色AV成人| 午夜福利成人网站| 欧美大香蕉在线视频| 中文字幕在线免费视频| 欧美三级长视频| 欧美日韩一区二区三区四区五区六区 | 洞av | 亚洲中文字幕网| 免费性爱视频| 苍井空中文字幕在线观看| 无码人妻AⅤ一区二区三区 | 亚洲第一中文字幕| 人人看人人摸人人搞| 青青草激情视频| 插逼网站| 97人妻碰碰中文无码久热丝袜| 中文字幕在线播放av| 伊大香蕉| 热的无码| 久草手机在线视频| 国产黄色大片| 精品成人Av一区二区三区| 97超碰碰碰| 91丝袜一区在线观看| 午夜无码AV| 青青草免费在线视频| 又大又长又粗91| 艹美女视频| 性无码一区二区三区在线观看| 麻豆视频在线| 亚洲影音| 操综合| 免费无码婬片aaaa| 一起草在线视频| 免费一级黄色视频| 久久91| 夜夜操狠狠操| 尤物网站在线播放| 国产又爽又黄免费网站在线看| 亚洲vs无码秘蜜桃少妇| 91在线无码精品国产三年| 激情综合网五月婷婷| 乱子伦国产精品视频一级毛| 亚洲视频在线免费播放| 欧美A级视频| 99久久精品一区二区成人| 黑人一区二区三区四区| 黄网免费| 久久久久大香蕉| 韩国AV三级| 人妻丰满熟妇av无码| 欧美日韩中文字幕在线视频| 亚洲网站视频| 高清无码免费在线观看| 国产一级a毛一级a做免费的视频| 黄色一级片在线| 在线无码一区二区三区| 香蕉久草| 你懂的久久| 亚洲精品成人AV| 99久久婷婷国产精品2020| 久久久久久久国产精品| 午夜午夜福利理论片在线播放| 成人精品一区二区三区电影| 超碰人人爱国产视| EEUSS| 国产手机AV在线| www,久久久| 欧美A黄| 3D动漫精品啪啪一区二区免费| 久久久国产精品人人片| 国产91在线一区| 欧美黄色毛片| 一级性爱视频| 白嫩无码| 五月丁香激情婷婷| 精品一区二区视频| 美国操逼片| 秋霞丝鲁片一区二区三区手机在绒免| 欧美成人在线视频网站| 激情婷婷五月| 精品国产一二三| 99自拍网| 一区二区网站| 中国熟女网站| 欧美大屌网站| 日韩伊人网| 激情视频网址| 精国产品一区二区三区A片| 日韩精品丰满无码一级A片∴| 日韩高清av| 久久大香蕉视频| 国产主播AV| 亚洲五月天婷婷| 日韩免费视频在线观看| 久久精品性爱| 亚洲欧美卡通| 黄片久久| 天天色天天干天天| 黄片免费高清| 免费高清无码在线观看| 国产免费一区二区三区免费视频 | 小草久久95| 亚洲黄色在线观看视频| 被黑人猛躁10次高潮视频| 成人无码在线观看免费视频| 无码人妻一区二区三区三| 亚洲日韩欧美一区二区天天天 | 久久久天堂| 最新日韩中文字幕| 国产av地址| 玖玖资源在线| 四川揉BBB搡BBB| 亚洲1区| 日韩免费一级片| 国产福利美女网站| 国产高清A片| 91视频一区二区| 二区三区无码| 黑人巨粗进入疼哭A片| 最新中文字幕777私人在线| 成人免费A片视频| 午夜在线观看视频| 日韩视频一区二区三区| 国产极品久久久| 蜜桃传媒av| 江苏妇搡BBBB搡BBB| 深爱开心激情| 天天色色色| 国产激情久久| 99色国产| 激情五月天丁香| 操逼逼一区二区三区| 免费一级AAAAA片在线播放| 亚洲性爱无码| 91久久久裸身美女| 六月婷婷网| 青青伊人网| 欧美日韩国产成人在线观看| 777视频在线观看| 国产精品乱伦| 91在线一区| 老司机av| 成人乱码一区二区三区| av中文无码| 69av在线播放| 国产女人水真多18毛片18精品| 欧美日韩国产精品成人| 人人看人人摸| 在线免费中文字幕| 日本人人操| 一区二区成人免费视频| 国产精品AV在线观看| 黄色片免费观看| 中文字幕精品一级A片| 一级欧美视频| 日韩高清无码免费看|