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

Fork me on GitHub
隨筆 - 215  文章 - 13  trackbacks - 0
<2016年11月>
303112345
6789101112
13141516171819
20212223242526
27282930123
45678910


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

常用鏈接

留言簿(1)

隨筆分類

隨筆檔案

相冊

Awesome

Blog

Book

GitHub

Link

搜索

  •  

積分與排名

  • 積分 - 221316
  • 排名 - 117

最新評論

閱讀排行榜

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

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

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常量會在編譯期分配到只讀段,對應數(shù)據(jù)地址不可寫入,并且相同的string常量不會重復存儲。 
2.fmt.Sprintf生成的字符串分配在堆上,對應數(shù)據(jù)地址可修改。 
3.常量空字符串有數(shù)據(jù)地址,動態(tài)生成的字符串沒有設置數(shù)據(jù)地址 
4.Golang string和[]byte轉(zhuǎn)換,會將數(shù)據(jù)復制到堆上,返回數(shù)據(jù)指向復制的數(shù)據(jù) 
5.動態(tài)生成的字符串,即使內(nèi)容一樣,數(shù)據(jù)也是在不同的空間 
6.只有動態(tài)生成的string,數(shù)據(jù)可以被黑科技修改 
8.string和[]byte通過復制轉(zhuǎn)換,性能損失接近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 思月行云 閱讀(601) 評論(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资源网| 欧美77777| 中文在线不卡| 欧美在线免费视频| 亚洲美女性视频| 亚洲午夜在线观看视频在线| 国产一区二区三区精品久久久| 欧美激情按摩| 欧美日韩中文另类| 久久久久欧美精品| 欧美日韩mp4| 久久精品国产亚洲5555| 美国成人直播| 销魂美女一区二区三区视频在线| 久久久久久尹人网香蕉| 在线中文字幕日韩| 欧美中文字幕在线| 亚洲天堂视频在线观看| 久久久久久电影| 亚洲综合激情| 美女脱光内衣内裤视频久久网站| 亚洲视频中文| 久久三级视频| 久久不射中文字幕| 欧美美女福利视频| 美女脱光内衣内裤视频久久影院| 欧美日韩另类字幕中文| 久久久激情视频| 欧美日韩你懂的| 欧美成人视屏| 国产日韩一区二区| 日韩一二三在线视频播| 亚洲成色www久久网站| 中日韩在线视频| 亚洲乱码国产乱码精品精天堂| 欧美一区二区三区四区视频| 亚洲午夜精品久久久久久浪潮| 久久久亚洲人| 久久久久久久综合狠狠综合| 国产精品爱久久久久久久| 亚洲国产视频a| 红桃视频国产精品| 亚洲永久视频| 亚洲综合99| 欧美日韩一级片在线观看| 亚洲国产成人精品久久| 亚洲高清一二三区| 久久久精品性| 免费观看成人www动漫视频| 国产亚洲一区二区在线观看| 亚洲一区日韩在线| 亚洲综合色在线| 欧美日韩国产一中文字不卡| 亚洲国产精品久久久久秋霞蜜臀| 樱桃成人精品视频在线播放| 欧美在线黄色| 麻豆av一区二区三区| 国产亚洲成年网址在线观看| 新片速递亚洲合集欧美合集| 久久狠狠亚洲综合| 黄色精品一二区| 久热re这里精品视频在线6| 欧美高清视频www夜色资源网| 在线日韩欧美视频| 免费成人性网站| 亚洲啪啪91| 日韩一区二区久久| 欧美视频一区二区| 亚洲特级毛片| 久久久久久久成人| 有坂深雪在线一区| 欧美激情精品久久久久久久变态 | 国产日韩欧美另类| 欧美在线一二三| 欧美va日韩va| 日韩亚洲不卡在线| 国产精品乱看| 欧美在线观看视频在线| 欧美/亚洲一区| 一本久久知道综合久久| 国产精品日韩一区二区| 久久久久久久综合| 亚洲激情一区| 久久aⅴ国产欧美74aaa| 在线免费观看日本一区| 欧美精品色综合| 午夜精品福利在线| 亚洲电影免费在线| 午夜精品福利视频| 最新国产成人av网站网址麻豆| 欧美精品在线一区二区| 午夜视频在线观看一区| 亚洲国产成人精品久久久国产成人一区| 一区二区精品在线| 国内精品嫩模av私拍在线观看| 欧美电影打屁股sp| 欧美中文字幕精品| 日韩系列在线| 美日韩免费视频| 亚洲欧美日韩一区二区在线| 亚洲高清在线观看一区| 国产精品夜夜夜一区二区三区尤| 久久精品人人| 亚洲一区精品视频| 亚洲狼人精品一区二区三区| 久久久国产亚洲精品| 中文国产成人精品久久一| 18成人免费观看视频| 国产麻豆一精品一av一免费| 欧美激情在线狂野欧美精品| 久久精品中文字幕免费mv| 亚洲视频电影图片偷拍一区| 亚洲福利视频一区二区| 久久亚洲一区| 欧美中文字幕在线播放| 亚洲美女精品一区| 亚洲国产高清高潮精品美女| 国产一区二区三区在线观看视频| 欧美片在线观看| 欧美电影免费观看网站| 久久久青草青青国产亚洲免观| 午夜视频一区| 亚洲淫性视频| 亚洲一级高清| 中文成人激情娱乐网| 亚洲人成亚洲人成在线观看| 欧美黄色成人网| 欧美v日韩v国产v| 毛片基地黄久久久久久天堂| 久久久久久婷| 久久免费99精品久久久久久| 久久精品久久综合| 欧美在线一级视频| 欧美在线视频播放| 久久精品亚洲热| 久久激情视频久久| 久久精品一区二区国产| 久久久久久久久综合| 久久一日本道色综合久久| 久久网站热最新地址| 看欧美日韩国产| 欧美成人在线免费观看| 免费观看久久久4p| 亚洲电影有码| 亚洲乱码精品一二三四区日韩在线 | 国产一区日韩欧美| 国内精品免费在线观看| 黄色成人av网| 91久久亚洲| 亚洲一区二区免费视频| 亚洲欧美另类中文字幕| 午夜久久99| 久久一区二区三区国产精品| 免费日韩视频| 日韩视频免费大全中文字幕| 日韩亚洲在线观看| 香蕉成人久久| 毛片基地黄久久久久久天堂| 欧美日韩国产一区二区三区| 国产精品入口日韩视频大尺度| 国产欧美日韩综合| 亚洲欧洲日本国产| 亚洲欧美一区二区精品久久久| 久久久久久久欧美精品| 亚洲第一久久影院| 夜夜嗨av色一区二区不卡| 欧美一区二区三区精品| 嫩草成人www欧美| 国产精品一区二区久久久久| 亚洲成人自拍视频| 亚洲男女自偷自拍| 欧美va亚洲va日韩∨a综合色| 亚洲精品国产精品久久清纯直播 | 欧美伊久线香蕉线新在线| 看欧美日韩国产| 一区二区欧美激情| 久久亚洲图片| 国产精品美女视频网站| 亚洲经典自拍| 久久国产精品久久精品国产| 亚洲激情一区二区| 欧美中在线观看| 国产精品久久久久久久第一福利| 一区二区在线不卡|