主頁(yè) > 知識(shí)庫(kù) > GO語(yǔ)言創(chuàng)建錢包并遍歷錢包(wallet)的實(shí)現(xiàn)代碼

GO語(yǔ)言創(chuàng)建錢包并遍歷錢包(wallet)的實(shí)現(xiàn)代碼

熱門標(biāo)簽:外呼系統(tǒng)顯本地手機(jī)號(hào) 阿克蘇地圖標(biāo)注 excel地圖標(biāo)注分布數(shù)據(jù) 電話機(jī)器人軟件免費(fèi) 評(píng)價(jià)高的400電話辦理 百度地圖標(biāo)注后傳給手機(jī) 涿州代理外呼系統(tǒng) 壽光微信地圖標(biāo)注 外呼系統(tǒng)用什么卡

基本知識(shí)

公鑰加密算法使用的是成對(duì)的密鑰:公鑰和私鑰,公鑰可以公開(kāi),私鑰不能被公開(kāi)。比特幣錢包實(shí)際上是一個(gè)密鑰對(duì),當(dāng)你安裝 一個(gè)錢包應(yīng)用,或者是使用一個(gè)比特幣客戶端來(lái)生成一個(gè)新地址是,他就會(huì)為你生成一個(gè)密鑰對(duì)。

代碼實(shí)現(xiàn)

func (cli *CLI) createWallet(nodeID string) {     //創(chuàng)建錢包的主函數(shù)
    wallets, _ := NewWallets(nodeID)   
    address := wallets.CreateWallet()
    wallets.SaveToFile(nodeID)

    fmt.Printf("Your new address: %s\n", address)
}

我們慢慢的分析這個(gè)程序,其中的NewWallets()函數(shù)如下,在這里先是定義了一個(gè)錢包集合,我們利用wallets結(jié)構(gòu)體存儲(chǔ)多個(gè)錢包,將他們保存到文件中或者從文件中進(jìn)行加載,每個(gè)錢包都保存了一堆公鑰和私鑰。創(chuàng)建出了一個(gè)空的錢包集合后,便開(kāi)始加載以前的錢包集合文件

func NewWallets(nodeID string) (*Wallets, error) {
    wallets := Wallets{}
    wallets.Wallets = make(map[string]*Wallet)
    err := wallets.LoadFromFile(nodeID)
    return wallets, err
}

type Wallets struct {
    Wallets map[string]*Wallet
}
type Wallet struct {
    PrivateKey ecdsa.PrivateKey
    PublicKey  []byte
}
func (ws *Wallets) LoadFromFile(nodeID string) error {
    walletFile := fmt.Sprintf(walletFile, nodeID)    
    if _, err := os.Stat(walletFile); os.IsNotExist(err) {   //判斷文件是否存在
        return err
    }

    fileContent, err := ioutil.ReadFile(walletFile)
    // ReadFile 讀取文件中的所有數(shù)據(jù),返回讀取的數(shù)據(jù)和遇到的錯(cuò)誤。
    //如果讀取成功,則 err 返回 nil,而不是 EOF
func ReadFile(filename string) ([]byte, error)
    if err != nil {
        log.Panic(err)
    }

    var wallets Wallets
    gob.Register(elliptic.P256())
    //gob是Golang包自帶的一個(gè)數(shù)據(jù)結(jié)構(gòu)序列化的編碼/解碼工具。
    decoder := gob.NewDecoder(bytes.NewReader(fileContent))
    err = decoder.Decode(wallets)//這里應(yīng)該是一個(gè)解碼的過(guò)程
    if err != nil {
        log.Panic(err)
    }
    ws.Wallets = wallets.Wallets
    return nil
}

再來(lái)看一看wallets.CreateWallet()方法,其中的NewWallet()如下, NewWallet()函數(shù)創(chuàng)建了一個(gè)錢包,我們可以根據(jù)公鑰打印出相應(yīng)的錢包對(duì)應(yīng)的地址,然后將錢包存儲(chǔ)到錢包集合結(jié)構(gòu)體中

unc (ws *Wallets) CreateWallet() string {
    wallet := NewWallet()
    address := fmt.Sprintf("%s", wallet.GetAddress())
    ws.Wallets[address] = wallet  //存儲(chǔ)到錢包集合中
    return address
}
func NewWallet() *Wallet {
    private, public := newKeyPair()   //得到公鑰和私鑰
    wallet := Wallet{private, public}  //存儲(chǔ)到錢包結(jié)構(gòu)體
    return wallet
}
func newKeyPair() (ecdsa.PrivateKey, []byte) {
    curve := elliptic.P256()
    private, err := ecdsa.GenerateKey(curve, rand.Reader)
    if err != nil {
        log.Panic(err)
    }
    pubKey := append(private.PublicKey.X.Bytes(), private.PublicKey.Y.Bytes()...)
    return *private, pubKey
}

//由公鑰得到地址,具體方法見(jiàn)我的博客用 [“go語(yǔ)言實(shí)現(xiàn)比特幣地址校驗(yàn)”](https://blog.csdn.net/m0_37719047/article/details/81945896) 
func (w Wallet) GetAddress() []byte {     
    pubKeyHash := HashPubKey(w.PublicKey)  
    versionedPayload := append([]byte{version}, pubKeyHash...)
    checksum := checksum(versionedPayload)
    fullPayload := append(versionedPayload, checksum...)
    address := Base58Encode(fullPayload)
    return address
}

最后將創(chuàng)建好的錢包更新到存儲(chǔ)錢包集合的文件中去

func (ws Wallets) SaveToFile(nodeID string) {
    var content bytes.Buffer     //開(kāi)辟一個(gè)內(nèi)存空間
    walletFile := fmt.Sprintf(walletFile, nodeID)
    gob.Register(elliptic.P256())
    encoder := gob.NewEncoder(content)    //序列化結(jié)構(gòu)體
    err := encoder.Encode(ws)
    if err != nil {
        log.Panic(err)
    }
    err = ioutil.WriteFile(walletFile, content.Bytes(), 0644)    //將序列化的數(shù)據(jù)寫入到文件中去
    if err != nil {
        log.Panic(err)
    }
}

如果我們需要打印錢包集合中所有錢包對(duì)應(yīng)的地址,我們可以利用以下的函數(shù)進(jìn)行遍歷。

func (cli *CLI) listAddresses(nodeID string) {
    wallets, err := NewWallets(nodeID)   //加載現(xiàn)有的錢包集合
    if err != nil {
        log.Panic(err)
    }
    addresses := wallets.GetAddresses()
    for _, address := range addresses {
        fmt.Println(address)
    }
}
func (ws *Wallets) GetAddresses() []string {
    var addresses []string
    for address := range ws.Wallets {
        addresses = append(addresses, address)
    }
    return addresses
}

通過(guò)以上的代碼,我們便完成了錢包,實(shí)現(xiàn)了 創(chuàng)建錢包和遍歷錢包的功能

參考

https://jeiwan.cc/

以上就是GO語(yǔ)言創(chuàng)建錢包并遍歷錢包(wallet)的實(shí)現(xiàn)代碼的詳細(xì)內(nèi)容,更多關(guān)于go語(yǔ)言遍歷錢包的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

您可能感興趣的文章:
  • go語(yǔ)言實(shí)現(xiàn)簡(jiǎn)易比特幣系統(tǒng)錢包的原理解析

標(biāo)簽:重慶 吐魯番 欽州 汕頭 雞西 梅河口 銅川 蘭州

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《GO語(yǔ)言創(chuàng)建錢包并遍歷錢包(wallet)的實(shí)現(xiàn)代碼》,本文關(guān)鍵詞  語(yǔ)言,創(chuàng)建,錢包,并,遍歷,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《GO語(yǔ)言創(chuàng)建錢包并遍歷錢包(wallet)的實(shí)現(xiàn)代碼》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于GO語(yǔ)言創(chuàng)建錢包并遍歷錢包(wallet)的實(shí)現(xiàn)代碼的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章