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

Fork me on GitHub
隨筆 - 215  文章 - 13  trackbacks - 0
<2016年8月>
31123456
78910111213
14151617181920
21222324252627
28293031123
45678910


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

常用鏈接

留言簿(1)

隨筆分類

隨筆檔案

相冊(cè)

Awesome

Blog

Book

GitHub

Link

搜索

  •  

積分與排名

  • 積分 - 219730
  • 排名 - 117

最新評(píng)論

閱讀排行榜

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

相對(duì)于C語言,golang是類型安全的語言。但是安全的代價(jià)就是性能的妥協(xié)。 
下面我們通過Golang中的“黑科技”來一窺Golang不想讓我們看到的“秘密”——string的底層數(shù)據(jù)。 
通過reflect包,我們可以知道,在Golang底層,string和slice其實(shí)都是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是一個(gè)指針,指向?qū)嶋H的數(shù)據(jù)地址,Len表示數(shù)據(jù)長度。 
但是,在string和[]byte轉(zhuǎn)換過程中,Golang究竟悄悄幫我們做了什么,來達(dá)到安全的目的? 
在Golang語言規(guī)范里面,string數(shù)據(jù)是禁止修改的,試圖通過&s[0], &b[0]取得string和slice數(shù)據(jù)指針地址也是不能通過編譯的。 
下面,我們就通過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個(gè)函數(shù)的神奇之處在于,通過unsafe.Pointer和reflect.XXXHeader取到了數(shù)據(jù)首地址,并實(shí)現(xiàn)了string和[]byte的直接轉(zhuǎn)換(這些操作在語言層面是禁止的)。 
下面我們就通過這幾個(gè)“黑科技”來測(cè)試一下語言底層的秘密:

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

結(jié)論如下: 
1.string常量會(huì)在編譯期分配到只讀段,對(duì)應(yīng)數(shù)據(jù)地址不可寫入,并且相同的string常量不會(huì)重復(fù)存儲(chǔ)。 
2.fmt.Sprintf生成的字符串分配在堆上,對(duì)應(yīng)數(shù)據(jù)地址可修改。 
3.常量空字符串有數(shù)據(jù)地址,動(dòng)態(tài)生成的字符串沒有設(shè)置數(shù)據(jù)地址 
4.Golang string和[]byte轉(zhuǎn)換,會(huì)將數(shù)據(jù)復(fù)制到堆上,返回?cái)?shù)據(jù)指向復(fù)制的數(shù)據(jù) 
5.動(dòng)態(tài)生成的字符串,即使內(nèi)容一樣,數(shù)據(jù)也是在不同的空間 
6.只有動(dòng)態(tài)生成的string,數(shù)據(jù)可以被黑科技修改 
8.string和[]byte通過復(fù)制轉(zhuǎn)換,性能損失接近4倍

我將測(cè)試代碼放在這里,歡迎參考: 
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 思月行云 閱讀(592) 評(píng)論(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>
            亚洲精品少妇网址| 欧美午夜精品久久久| 在线观看日韩精品| 欧美96在线丨欧| 米奇777超碰欧美日韩亚洲| 亚洲国产岛国毛片在线| 亚洲电影免费观看高清完整版| 欧美一区二区成人6969| 一区二区在线观看av| 欧美激情成人在线| 欧美日韩一区二区三区在线 | 免费精品99久久国产综合精品| 精品成人免费| 亚洲精品国产精品国产自| 欧美视频中文字幕| 久久精品亚洲国产奇米99| 免费亚洲视频| 亚洲综合视频1区| 亚洲成人自拍视频| 一本色道久久综合一区| 亚洲网在线观看| 精品999在线播放| 夜夜嗨av一区二区三区| 国产自产v一区二区三区c| 亚洲国产视频一区二区| 国产精品丝袜久久久久久app | 亚洲第一在线| 一区二区高清视频在线观看| 韩国美女久久| 亚洲一区二区三区中文字幕 | 国产偷国产偷亚洲高清97cao| 亚洲福利视频免费观看| 国产精品毛片在线看| 欧美大尺度在线| 国产精品专区一| 亚洲人成在线播放网站岛国| 国产一区二区三区在线观看精品 | 久久久亚洲国产天美传媒修理工| 亚洲午夜精品久久| 免费观看成人鲁鲁鲁鲁鲁视频| 久久xxxx精品视频| 欧美日韩国产片| 亚洲国产va精品久久久不卡综合| 国产欧美日韩一区二区三区| 日韩视频在线一区二区三区| 亚洲狠狠丁香婷婷综合久久久| 午夜伦理片一区| 午夜在线视频一区二区区别| 欧美精品免费在线| 欧美成人综合网站| 国产在线观看精品一区二区三区| 亚洲一区二区三区在线视频| 亚洲一区999| 欧美日韩卡一卡二| 亚洲精品网址在线观看| 亚洲精品网站在线播放gif| 久久亚洲视频| 美日韩在线观看| 好吊一区二区三区| 久久久噜噜噜久久中文字幕色伊伊 | 一区二区在线观看视频| 欧美一区二区精品| 久久精品人人做人人爽电影蜜月| 国产精品亚洲综合天堂夜夜| 亚洲一区二区视频在线| 亚洲欧美在线免费观看| 国产精品入口日韩视频大尺度| 中文精品一区二区三区 | 亚洲精品国产精品乱码不99按摩| 老司机久久99久久精品播放免费| 免费在线看一区| 亚洲国产精品女人久久久| 另类天堂av| 国产日韩精品一区二区| 久久蜜桃精品| 在线观看91精品国产麻豆| 久久久.com| 亚洲国产免费| 一区二区三区视频在线观看| 欧美性猛交xxxx乱大交蜜桃| 亚洲欧美日韩在线高清直播| 久久免费精品视频| 91久久精品一区二区三区| 欧美激情五月| 亚洲欧美日韩综合| 美女精品在线观看| 亚洲美女视频在线免费观看| 欧美视频中文一区二区三区在线观看| 亚洲永久精品大片| 免费看的黄色欧美网站| 一区二区三区国产精品| 国产欧美精品日韩精品| 免费成人性网站| 亚洲天堂网在线观看| 免费一级欧美在线大片| 这里只有精品电影| 狠狠干综合网| 欧美三级第一页| 久久天堂精品| 亚洲视频香蕉人妖| 欧美大秀在线观看| 亚洲欧美日韩成人高清在线一区| 激情综合网激情| 欧美午夜精品久久久久久浪潮| 久久久久久久久久久久久9999| 一本一道久久综合狠狠老精东影业 | 亚洲高清免费视频| 亚洲欧美日本精品| 亚洲经典视频在线观看| 国产欧美欧美| 欧美日韩一区二区三区免费| 久久综合九色综合久99| 亚洲一区二区三区精品视频| 亚洲韩国青草视频| 久久永久免费| 欧美一区二区国产| 中日韩高清电影网| 91久久久一线二线三线品牌| 国产视频在线观看一区二区| 欧美三级电影精品| 欧美福利影院| 久热精品在线| 久久久九九九九| 欧美一区久久| 亚洲一区二区精品视频| 亚洲美女色禁图| 亚洲激情午夜| 亚洲电影第1页| 美女主播一区| 免费欧美在线| 免费不卡视频| 免费亚洲一区二区| 美女国产精品| 免费视频一区二区三区在线观看| 欧美在线视频a| 午夜精品一区二区三区在线| 亚洲欧美日韩高清| 亚洲免费视频中文字幕| 中文欧美日韩| 亚洲一区二区三区四区五区黄 | 国产一级久久| 狠狠色噜噜狠狠狠狠色吗综合| 宅男66日本亚洲欧美视频| 亚洲一区二区黄色| 夜色激情一区二区| 妖精成人www高清在线观看| 亚洲精品1区| 99热精品在线观看| 99精品视频免费在线观看| 一区二区三区三区在线| 亚洲视频一区二区| 午夜精品久久久久影视 | 欧美一区二区三区视频| 欧美一区日本一区韩国一区| 久久精品成人欧美大片古装| 欧美自拍偷拍| 榴莲视频成人在线观看| 欧美高清视频一区二区| 亚洲国产精品va在线观看黑人| 亚洲精品国产视频| 中文成人激情娱乐网| 亚洲欧美一区二区在线观看| 久久久久国产精品www| 麻豆freexxxx性91精品| 欧美日韩一区二区在线观看视频| 国产精品美女一区二区| 精品动漫3d一区二区三区免费| 亚洲国产成人精品久久久国产成人一区| 亚洲精品精选| 欧美一区二区三区电影在线观看| 久久在线视频在线| 99国产成+人+综合+亚洲欧美| 亚洲综合国产激情另类一区| 鲁大师成人一区二区三区 | 久久精品道一区二区三区| 蜜臀91精品一区二区三区| 国产精品va在线播放我和闺蜜| 国产日韩欧美三级| 亚洲精品综合| 久久香蕉国产线看观看网| 亚洲精品国产精品乱码不99按摩 | 欧美不卡视频| 亚洲一区二区三区四区中文 | 国产亚洲精品v| 一本大道久久精品懂色aⅴ | 午夜一区在线| 亚洲国产欧美久久| 午夜免费日韩视频| 欧美日韩在线高清| 亚洲激情一区二区| 久久国产精品99国产精| 亚洲另类黄色| 美腿丝袜亚洲色图| 狠狠色综合一区二区| 亚洲一区欧美二区| 亚洲福利av| 久久综合网络一区二区| 国产欧美一区二区精品性色| 亚洲午夜电影| 亚洲国产另类久久久精品极度|