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

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在线| 亚洲性感美女99在线| 久久综合给合| 亚洲激情电影在线| 亚洲欧美日韩精品久久亚洲区 | 亚洲人成在线观看一区二区| 欧美在线观看视频| 99精品免费视频| 免费观看日韩av| 黄色日韩在线| 香港久久久电影| av成人黄色| 欧美精品粉嫩高潮一区二区| 黑人巨大精品欧美一区二区小视频| 午夜精品久久久久| 99精品视频免费全部在线| 狂野欧美激情性xxxx欧美| 国产欧美日本| 亚洲性线免费观看视频成熟| 亚洲黄色成人网| 免费看精品久久片| 在线观看视频一区| 久久综合久久美利坚合众国| 欧美一区日韩一区| 国产欧美欧洲在线观看| 午夜精品剧场| 亚洲午夜久久久| 国产精品第13页| 在线午夜精品| 亚洲精品一区二| 欧美日本韩国一区| 日韩一级二级三级| 亚洲国产综合视频在线观看| 六月丁香综合| 亚洲激情电影在线| 亚洲激情不卡| 欧美日韩国产探花| 亚洲视频视频在线| 一本色道久久加勒比精品 | 久久久久久久久综合| 亚洲欧美日韩精品一区二区| 国产精品日本精品| 先锋影音久久| 亚洲欧美日韩精品久久奇米色影视| 国产精品久久久久久模特| 亚洲欧美美女| 亚洲欧美综合v| 国产一区二区精品在线观看| 久久免费偷拍视频| 久久综合网hezyo| 亚洲黄色在线观看| 亚洲区一区二| 欧美色区777第一页| 亚洲综合欧美| 午夜国产精品视频免费体验区| 国产欧美日韩一区二区三区在线观看 | 亚洲精华国产欧美| 欧美精品播放| 亚洲一区二区在线| 亚洲欧美日韩爽爽影院| 国产一区日韩一区| 欧美成人精品一区二区| 欧美mv日韩mv国产网站app| 99v久久综合狠狠综合久久| 日韩一区二区精品葵司在线| 国产精品超碰97尤物18| 欧美一区日韩一区| 久久深夜福利| 一本一本久久| 亚洲一区欧美二区| 伊人久久大香线蕉综合热线| 亚洲国产精品电影| 国产精品久久久999| 久久久久久一区二区| 麻豆成人av| 亚洲小视频在线| 午夜精品视频在线观看| …久久精品99久久香蕉国产| 亚洲经典一区| 国产女主播一区| 欧美国产大片| 国产精品电影观看| 久久综合久久综合久久综合| 欧美精品电影| 欧美亚洲视频一区二区| 裸体丰满少妇做受久久99精品| 一区二区激情视频| 欧美在线视频一区二区| 日韩视频一区二区三区在线播放| 亚洲永久字幕| 亚洲国产一区二区三区在线播| 在线综合亚洲| 亚洲电影第三页| 国产精品99久久久久久人| 在线观看免费视频综合| 日韩亚洲在线| 一区二区亚洲精品| 日韩亚洲欧美一区| 在线免费观看视频一区| 亚洲一区二区三区三| 亚洲国产精品久久久久秋霞不卡| 亚洲新中文字幕| 91久久久国产精品| 亚洲欧美一区二区三区极速播放 | 国产精品免费一区豆花| 欧美大胆成人| 国产精品一页| 亚洲精品久久久久久久久久久久久 | 亚洲欧美一区二区原创| 日韩视频在线观看国产| 欧美一区二区免费视频| 中文亚洲免费| 免费91麻豆精品国产自产在线观看| 午夜精品视频一区| 欧美精品自拍偷拍动漫精品| 久久手机免费观看| 欧美三区在线视频| 欧美激情免费在线| 狠狠入ady亚洲精品| 亚洲影视中文字幕| 亚洲美女在线一区| 久热精品视频在线| 久久久久久久久久久久久9999| 国产精品久久久久久av下载红粉 | 亚洲国产精品v| 欧美一区二区在线观看| 午夜免费久久久久| 欧美日韩大片| 亚洲福利视频免费观看| 雨宫琴音一区二区在线| 亚洲欧美日韩国产综合精品二区| 亚洲深爱激情| 欧美黄色影院| 欧美激情视频网站| 在线看成人片| 久久免费视频一区| 久热re这里精品视频在线6| 国产日韩欧美在线观看| 亚洲综合精品| 亚洲嫩草精品久久| 欧美午夜免费| 一区电影在线观看| 宅男噜噜噜66一区二区| 欧美日韩国产综合视频在线观看| 免费久久99精品国产自| 伊人精品久久久久7777| 久久狠狠亚洲综合| 久久久久久久综合狠狠综合| 国产性做久久久久久| 亚洲欧美日本另类| 欧美一级二区| 国产精品免费观看在线| 亚洲无亚洲人成网站77777 | 久久综合伊人| 欧美韩日一区二区| 最新日韩精品| 欧美激情亚洲综合一区| 亚洲欧洲一区二区三区在线观看 | 欧美一二三区精品| 久久福利电影| 国产一区在线视频| 久久久久国产免费免费| 老司机午夜精品视频| 亚洲高清久久网| 久久香蕉国产线看观看网| 欧美 日韩 国产在线| 亚洲电影免费观看高清完整版| 久久精品成人欧美大片古装| 蜜桃av噜噜一区| 91久久精品国产91性色tv| 欧美成人国产| 日韩图片一区| 性视频1819p久久| 国产亚洲欧美另类一区二区三区| 欧美一区二区久久久| 欧美成人免费在线视频| 日韩一级精品视频在线观看| 欧美午夜视频在线| 一本一本久久a久久精品综合妖精 一本一本久久a久久精品综合麻豆 | 在线免费不卡视频| 欧美激情国产高清| 一区二区激情| 久久精品国产77777蜜臀| 狠狠综合久久av一区二区老牛| 老妇喷水一区二区三区| 亚洲精品视频一区二区三区| 亚洲欧美日韩精品久久| 国产一区再线| 另类成人小视频在线| 亚洲卡通欧美制服中文| 亚洲欧美日韩精品一区二区| 国产一区久久| 欧美高清视频| 亚洲资源av| 欧美1区2区视频| 一本色道88久久加勒比精品| 国产精品一二| 美女啪啪无遮挡免费久久网站| 日韩视频国产视频|