簡(jiǎn)單說說Python與Go的區(qū)別

背景

工作中的主力語言是Python,今年要搞性能測(cè)試的工具,由于GIL鎖的原因,Python的性能實(shí)在是慘淡,需要學(xué)一門性能高的語言來生成性能測(cè)試的壓力端。因此我把目光放在了現(xiàn)在的新秀Go。經(jīng)過一段時(shí)間的學(xué)習(xí),也寫了一個(gè)小工具,記一下這兩個(gè)語言的區(qū)別。

需求

工具是一個(gè)小爬蟲,用來爬某網(wǎng)站的某個(gè)產(chǎn)品的迭代記錄,實(shí)現(xiàn)邏輯就是運(yùn)行腳本后,使用者從命令行輸入某些元素(產(chǎn)品ID等)后,腳本導(dǎo)出一個(gè)Excel文件出來。

最初的版本是用Python寫的,30行代碼不到就搞定了。這次用Go重寫,代碼量在110行左右。

接受輸入

第一步就是接受命令行的輸入內(nèi)容,工具要給非技術(shù)人員用的,弄一個(gè)CLI不太合適,要的效果就是一行一行的輸入內(nèi)容,用Python實(shí)現(xiàn)起來非常容易,像這樣:

app_id = raw_input('請(qǐng)輸入app_id: ')
app_analysis = raw_input('請(qǐng)輸入analysis: ')

執(zhí)行后就是一行一行的往下走,但是用Go就有點(diǎn)蛋疼了,完整的代碼如下:

func getPara() (string, string) {
    var i = 0
    var appId, analysis string
    fmt.Print("請(qǐng)輸入appId:")
    scanner := bufio.NewScanner(os.Stdin)
    for scanner.Scan() {
        text := scanner.Text()
        if i == 0 {
            appId = text
            fmt.Print("請(qǐng)輸入analysis:")
        } else if i == 1 {
            analysis = text
            fmt.Print("程序初始化數(shù)據(jù)完畢。。。。請(qǐng)按任意鍵繼續(xù)")
        } else {
            break
        }
        i++
    }
    return appId, analysis
}

Go要實(shí)現(xiàn)CLI很方便,但是涉及到這種一行一行的輸入,要一直監(jiān)聽Scan(),所以就有了上面蛋疼的循環(huán)處理,而且在必須要先打印信息,再來監(jiān)聽內(nèi)容,總體的寫的過程很惡心,也許是沒有找到更好的方法吧,實(shí)在是太逆天了。

發(fā)送請(qǐng)求

在發(fā)送請(qǐng)求方便,兩種語言倒是差別不太大,至少我寫的Get請(qǐng)求是這樣的。

Python
params = {
    "analysis": app_analysis,
    "appid": app_id,
    "country": 'cn'
}
r = requests.get(url, params)
Go
q := req.URL.Query()
q.Add("appid", appId)
q.Add("analysis", analysis)
q.Add("country", "cn")
req.URL.RawQuery = q.Encode()
var resp *http.Response
resp, _ = http.DefaultClient.Do(req)

返回結(jié)果處理

在返回結(jié)果的處理上,Python的處理方式簡(jiǎn)直是太友好了,直接調(diào)用json就處理了。

result = r.json()

但是Go就有點(diǎn)蛋疼了,由于是靜態(tài)語言,所以解包數(shù)據(jù)的時(shí)候需要先定義數(shù)據(jù)格式,比如返回的內(nèi)容必須要先做如下的結(jié)構(gòu)定義:

type ResultInfo struct {
    Code    int
    Msg     string
    Version []VersionInfo
}

type VersionInfo struct {
    Version     string `json:"version"`
    ReleaseTime string `json:"release_time"`
    ReleaseNote string `json:"release_note"`
    AppName     string `json:"app_name"`
    SubTitle    string `json:"subtitle"`
}

第一個(gè)ResultInfo是返回的數(shù)據(jù),其中的Version也是一個(gè)數(shù)組對(duì)象,所以還要再定義一個(gè)數(shù)組對(duì)象,這樣才能調(diào)用方法來解包處理。

body, _ := ioutil.ReadAll(resp.Body)
var rst = ResultInfo{}
if err := json.Unmarshal(body, &rst); err != nil {
    fmt.Println(err)
}

寫數(shù)據(jù)到Excel

這部分調(diào)用的都是第三方庫,所以沒什么可比性,代碼的實(shí)現(xiàn)完全依賴于第三方包。

無所不在的err != nil

Go的異常捕獲機(jī)制跟Python或者Java都不一樣,Python的異常捕獲使用的是try,except來包裹代碼塊,而Go用的是一個(gè)error對(duì)象,所以所有的Go代碼都會(huì)充斥著大量的

if err != nil {
        return nil, err
    }

這種鬼東西,這種異常機(jī)制在閱讀代碼的時(shí)候,非常惡心,極大的影響了閱讀體驗(yàn)。

吐槽完后

基本上從書寫代碼的過程來看,Python的編碼效率比Go高出了很多很多,Go號(hào)稱語法靈活,可以極大的提高編碼效率,實(shí)際上并沒有,受限于靜態(tài)語言,相比于Python這種動(dòng)態(tài)語言來說,編碼效率的差距還是非常大的。只能說比其他靜態(tài)語言編碼效率高。

但是?。?!

Go的效率比Python高了太多。舉個(gè)例子,有一個(gè)計(jì)算斐波那契數(shù)的算法,Go的實(shí)現(xiàn)如下:

func main() {
    const n = 40
    starttime := time.Now()
    fibN := fib(n)
    endtime := time.Now()
    cost_time := endtime.Sub(starttime)
    fmt.Println(cost_time)
    fmt.Printf("\rFibonacci(%d) = %d\n", n, fibN)
}

func fib(x int) int {
    if x < 2 {
        return x
    }
    return fib(x-1) + fib(x-2)
}

很簡(jiǎn)單的一個(gè)遞歸,當(dāng)N為40的時(shí)候,Go花了大概1秒左右的時(shí)間,執(zhí)行結(jié)果如下:

876.838ms(消耗時(shí)間)
Fibonacci(40) = 102334155

我們換成Python

def fib(x):
    if x<2:
        return x
    return fib(x-1)+fib(x-2)

if __name__ == '__main__':
    import time
    begin = time.time()
    print fib(40)
    end = time.time()
    print end-begin

一樣的執(zhí)行邏輯,執(zhí)行的結(jié)果卻是:

102334155
52.8657081127(消耗時(shí)間)

WTF!!! 用Go來處理效率是Python的50倍以上。

還沒完,工具寫完了總是要給人用的吧,Python寫完之后,如果給一個(gè)非技術(shù)人員使用,那么。。。

使用者:要怎么用?
我:你裝一下Python,然后配好環(huán)境變量,順便把requests庫和xlwt庫也裝一下。
我:要裝這兩個(gè)庫你要先裝一下pip。
使用者:黑人問號(hào)臉?。。。。?

如果你用Go來寫,打包完發(fā)過去就行了

使用者:要怎么用?
我:你雙擊一下,讓你輸入什么就輸入什么

如果使用者是用Windows系統(tǒng),那也沒問題,

CGO_ENABLED=0 GOOS=windows GOARCH=amd64 go build xxx.go

直接打包成exe文件。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,062評(píng)論 25 709
  • 本節(jié)內(nèi)容 Python介紹 發(fā)展史 Python 2 or 3? 安裝 Hello World程序 變量 用戶輸入...
    小小不懂11閱讀 3,542評(píng)論 2 30
  • 從昨晚大家的興奮勁,到遇到騎行隊(duì),來自于內(nèi)心的激情豪邁又出來了,還有三個(gè)妹子相遇相知,一切看起來都那么美好,伴著女...
    Yoga笑笑閱讀 189評(píng)論 0 0
  • 這個(gè)tableview啊, footerview有單獨(dú)的類, 代理返回的是個(gè)view, 背景色改了沒用, 得強(qiáng)轉(zhuǎn)成...
    神秘喚貓人閱讀 297評(píng)論 0 0

友情鏈接更多精彩內(nèi)容