青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Fork me on GitHub
隨筆 - 215  文章 - 13  trackbacks - 0
<2017年5月>
30123456
78910111213
14151617181920
21222324252627
28293031123
45678910


專注即時通訊及網游服務端編程
------------------------------------
Openresty 官方模塊
Openresty 標準模塊(Opm)
Openresty 三方模塊
------------------------------------
本博收藏大部分文章為轉載,并在文章開頭給出了原文出處,如有再轉,敬請保留相關信息,這是大家對原創作者勞動成果的自覺尊重!!如為您帶來不便,請于本博下留言,謝謝配合。

常用鏈接

留言簿(1)

隨筆分類

隨筆檔案

相冊

Awesome

Blog

Book

GitHub

Link

搜索

  •  

積分與排名

  • 積分 - 220946
  • 排名 - 117

最新評論

閱讀排行榜

http://blog.csdn.net/vipally/article/details/52940119

相對于C語言,golang是類型安全的語言。但是安全的代價就是性能的妥協。 
下面我們通過Golang中的“黑科技”來一窺Golang不想讓我們看到的“秘密”——string的底層數據。 
通過reflect包,我們可以知道,在Golang底層,string和slice其實都是struct:

type SliceHeader struct {
    Data uintptr
    Len  int
    Cap  int
}
type StringHeader struct {
    Data uintptr
    Len  int
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

其中Data是一個指針,指向實際的數據地址,Len表示數據長度。 
但是,在string和[]byte轉換過程中,Golang究竟悄悄幫我們做了什么,來達到安全的目的? 
在Golang語言規范里面,string數據是禁止修改的,試圖通過&s[0], &b[0]取得string和slice數據指針地址也是不能通過編譯的。 
下面,我們就通過Golang的“黑科技”來一窺Golang背后的“秘密”。

//return GoString's buffer slice(enable modify string)
func StringBytes(s string) Bytes {
    return *(*Bytes)(unsafe.Pointer(&s))
}

// convert b to string without copy
func BytesString(b []byte) String {
    return *(*String)(unsafe.Pointer(&b))
}

// returns &s[0], which is not allowed in go
func StringPointer(s string) unsafe.Pointer {
    p := (*reflect.StringHeader)(unsafe.Pointer(&s))
    return unsafe.Pointer(p.Data)
}

// returns &b[0], which is not allowed in go
func BytesPointer(b []byte) unsafe.Pointer {
    p := (*reflect.SliceHeader)(unsafe.Pointer(&b))
    return unsafe.Pointer(p.Data)
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21

以上4個函數的神奇之處在于,通過unsafe.Pointer和reflect.XXXHeader取到了數據首地址,并實現了string和[]byte的直接轉換(這些操作在語言層面是禁止的)。 
下面我們就通過這幾個“黑科技”來測試一下語言底層的秘密:

func TestPointer(t *testing.T) {
    s := []string{
        "",
        "",
        "hello",
        "hello",
        fmt.Sprintf(""),
        fmt.Sprintf(""),
        fmt.Sprintf("hello"),
        fmt.Sprintf("hello"),
    }
    fmt.Println("String to bytes:")
    for i, v := range s {
        b := unsafe.StringBytes(v)
        b2 := []byte(v)
        if b.Writeable() {
            b[0] = 'x'
        }
        fmt.Printf("%d\ts=%5s\tptr(v)=%-12v\tptr(StringBytes(v)=%-12v\tptr([]byte(v)=%-12v\n",
            i, v, unsafe.StringPointer(v), b.Pointer(), unsafe.BytesPointer(b2))
    }

    b := [][]byte{
        []byte{},
        []byte{'h', 'e', 'l', 'l', 'o'},
    }
    fmt.Println("Bytes to string:")
    for i, v := range b {
        s1 := unsafe.BytesString(v)
        s2 := string(v)
        fmt.Printf("%d\ts=%5s\tptr(v)=%-12v\tptr(StringBytes(v)=%-12v\tptr(string(v)=%-12v\n",
            i, s1, unsafe.BytesPointer(v), s1.Pointer(), unsafe.StringPointer(s2))
    }

}

const N = 3000000

func Benchmark_Normal(b *testing.B) {
    for i := 1; i < N; i++ {
        s := fmt.Sprintf("12345678901234567890123456789012345678901234567890")
        bb := []byte(s)
        bb[0] = 'x'
        s = string(bb)
        s = s
    }
}
func Benchmark_Direct(b *testing.B) {
    for i := 1; i < N; i++ {
        s := fmt.Sprintf("12345678901234567890123456789012345678901234567890")
        bb := unsafe.StringBytes(s)
        bb[0] = 'x'
        s = s
    }
}

//test result
//String to bytes:
//0 s=      ptr(v)=0x51bd70     ptr(StringBytes(v)=0x51bd70     ptr([]byte(v)=0xc042021c58
//1 s=      ptr(v)=0x51bd70     ptr(StringBytes(v)=0x51bd70     ptr([]byte(v)=0xc042021c58
//2 s=hello ptr(v)=0x51c2fa     ptr(StringBytes(v)=0x51c2fa     ptr([]byte(v)=0xc042021c58
//3 s=hello ptr(v)=0x51c2fa     ptr(StringBytes(v)=0x51c2fa     ptr([]byte(v)=0xc042021c58
//4 s=      ptr(v)=<nil>        ptr(StringBytes(v)=<nil>        ptr([]byte(v)=0xc042021c58
//5 s=      ptr(v)=<nil>        ptr(StringBytes(v)=<nil>        ptr([]byte(v)=0xc042021c58
//6 s=xello ptr(v)=0xc0420444b5 ptr(StringBytes(v)=0xc0420444b5 ptr([]byte(v)=0xc042021c58
//7 s=xello ptr(v)=0xc0420444ba ptr(StringBytes(v)=0xc0420444ba ptr([]byte(v)=0xc042021c58
//Bytes to string:
//0 s=      ptr(v)=0x5c38b8     ptr(StringBytes(v)=0x5c38b8     ptr(string(v)=<nil>
//1 s=hello ptr(v)=0xc0420445e0 ptr(StringBytes(v)=0xc0420445e0 ptr(string(v)=0xc042021c38
//Benchmark_Normal-4    1000000000           0.87 ns/op
//Benchmark_Direct-4    2000000000           0.24 ns/op
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71

結論如下: 
1.string常量會在編譯期分配到只讀段,對應數據地址不可寫入,并且相同的string常量不會重復存儲。 
2.fmt.Sprintf生成的字符串分配在堆上,對應數據地址可修改。 
3.常量空字符串有數據地址,動態生成的字符串沒有設置數據地址 
4.Golang string和[]byte轉換,會將數據復制到堆上,返回數據指向復制的數據 
5.動態生成的字符串,即使內容一樣,數據也是在不同的空間 
6.只有動態生成的string,數據可以被黑科技修改 
8.string和[]byte通過復制轉換,性能損失接近4倍

我將測試代碼放在這里,歡迎參考: 
https://github.com/vipally/gx/blob/master/unsafe/string_test.go

參考資料: 
[1] Go語言黑魔法 http://studygolang.com/articles/2909

posted on 2017-05-04 10:32 思月行云 閱讀(599) 評論(0)  編輯 收藏 引用 所屬分類: Golang
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲人成在线免费观看| 亚洲第一精品久久忘忧草社区| 日韩网站在线观看| 亚洲午夜一区二区| 国产一区二区看久久| 久久人体大胆视频| 一区二区三区蜜桃网| 免费在线观看成人av| 一本色道久久综合| 在线成人av| 国产欧美日韩一区二区三区在线观看 | 欧美欧美天天天天操| 亚洲午夜激情在线| 亚洲大片一区二区三区| 欧美一区二区成人6969| 99国内精品久久| 在线观看欧美日本| 国产丝袜美腿一区二区三区| 欧美一级网站| 亚洲美女精品成人在线视频| 免费久久99精品国产自| 久久国产一区二区三区| 亚洲一区二区三区免费观看| 99在线热播精品免费| 亚洲国产福利在线| 国产在线视频欧美| 国产麻豆精品theporn| 欧美性事免费在线观看| 欧美日韩在线大尺度| 欧美啪啪成人vr| 亚洲国产精品高清久久久| 久久综合狠狠| 毛片一区二区三区| 免费在线观看成人av| 久久精品国产久精国产思思| 亚洲欧美另类中文字幕| 亚洲永久视频| 香蕉成人久久| 久久精品国产欧美激情| 久久嫩草精品久久久久| 亚洲精品一区二区在线| 国自产拍偷拍福利精品免费一| 国产精品自拍一区| 欧美视频三区在线播放| 亚洲一二三区精品| 亚洲免费在线播放| 午夜日韩在线| 欧美一区视频| 久久久久久久久久久久久女国产乱 | 欧美在线二区| 久久精品中文字幕免费mv| 欧美在线观看视频在线| 久久亚洲精品一区| 老鸭窝亚洲一区二区三区| 久久综合久久久久88| 久久躁日日躁aaaaxxxx| 美女网站久久| 午夜精品久久久久久久久久久| 欧美一级专区| 久久精品99无色码中文字幕| 亚洲精品孕妇| 欧美一区二区啪啪| 亚洲国产精品久久久久秋霞蜜臀 | 性18欧美另类| 蜜桃av一区| 国产精品99一区| 国产一区二区久久久| 99视频一区| 久久久国产一区二区| 欧美护士18xxxxhd| 久久久久久久久综合| 亚洲欧洲在线视频| 亚洲欧美在线高清| 欧美福利视频| 国产一区二区三区免费不卡 | 久久久久中文| 欧美日韩一区在线播放| 在线观看欧美成人| 午夜在线观看免费一区| 亚洲第一搞黄网站| 午夜久久久久久| 欧美日韩国产精品专区| 伊大人香蕉综合8在线视| 欧美网站在线观看| 久久精品观看| 国产精品theporn| 影音先锋中文字幕一区| 一区二区三区**美女毛片| 久久综合九色99| 免费观看国产成人| 在线一区二区三区做爰视频网站| 毛片精品免费在线观看| 国语精品中文字幕| 羞羞视频在线观看欧美| 99视频精品免费观看| 一卡二卡3卡四卡高清精品视频| 久久综合九色综合欧美就去吻| 国产精品自在线| 99re热精品| 最新日韩在线| 欧美成人一二三| 精品99一区二区| 久久久.com| 欧美专区日韩视频| 国产嫩草影院久久久久| 亚洲欧美视频在线| 亚洲一二三区精品| 国产精品免费观看视频| 亚洲综合二区| 亚洲午夜在线| 国产无一区二区| 激情久久久久久| 久久婷婷亚洲| 久久综合九色| 一本色道久久综合亚洲精品小说 | 亚洲伊人一本大道中文字幕| 亚洲一区日本| 欧美日韩黄视频| 羞羞答答国产精品www一本 | 亚洲精品国产欧美| 午夜精品福利一区二区蜜股av| 91久久久久| 久久www成人_看片免费不卡| 亚洲视频狠狠| 男人插女人欧美| 久久亚洲综合| 国产视频自拍一区| 亚洲性夜色噜噜噜7777| 中日韩午夜理伦电影免费| 裸体一区二区| 免费观看成人| 海角社区69精品视频| 亚洲影院免费观看| 牛牛影视久久网| 欧美激情91| 欧美有码在线视频| 欧美午夜寂寞影院| 亚洲人体1000| 激情自拍一区| 久久精品30| 玖玖精品视频| 影音先锋亚洲视频| 久久久久九九九| 免费av成人在线| 在线视频国产日韩| 久久综合久久综合久久综合| 蜜臀av国产精品久久久久| 国产女主播视频一区二区| 亚洲欧美激情视频| 欧美亚洲一区在线| 国产手机视频精品| 久久精品视频在线免费观看| 老色鬼精品视频在线观看播放| 伊人成年综合电影网| 久久亚洲欧洲| 亚洲欧洲精品一区二区三区 | 亚洲欧美日本另类| 欧美午夜精品久久久| 亚洲一区二区综合| 久久久久国产成人精品亚洲午夜| 国产综合久久久久影院| 久久久久久久一区二区三区| 欧美激情一区二区在线| 99伊人成综合| 国产欧美日韩视频| 另类天堂视频在线观看| 亚洲精品国产精品久久清纯直播 | 欧美日韩一区二区三区在线观看免| 日韩亚洲欧美综合| 午夜精品国产| 影音先锋日韩精品| 欧美三级乱人伦电影| 欧美一区二区三区视频免费播放| 麻豆成人小视频| 一本色道久久综合亚洲精品不卡| 国产精品一区二区久激情瑜伽| 久久久精品一区| 一本色道久久综合精品竹菊 | 亚洲欧洲综合另类在线| 亚洲欧美韩国| 亚洲视频成人| 制服丝袜亚洲播放| 麻豆久久精品| 香蕉久久夜色精品国产| 亚洲第一黄色网| 国产精品一区二区久久久| 久久人人97超碰精品888| 亚洲视频网站在线观看| 亚洲大胆av| 久久精品日韩| 中文欧美在线视频| 亚洲国产精品成人| 国产精品主播| 欧美涩涩网站| 免费不卡中文字幕视频| 欧美一区二区三区四区视频| 亚洲影院免费观看| 亚洲经典三级| 欧美精品一区二区高清在线观看| 在线视频日韩精品|