Go语言如何发送以太坊代币

hubwiz01 · · 3152 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

在本文中,你将学习如何将ETH从一个帐户转移到另一个帐户。如果你已经熟悉以太坊,那么你就知道交易包括你发送的以太币量,gas限额,gas价格,随机数,接收地址以及可选数据。在将他被广播到网络之前,必须使用发送方的私钥对该交易进行签名。 假设你已经连接了客户端,下一步就是加载你的私钥。 ``` privateKey, err := crypto.HexToECDSA("fad9c8855b740a0b7ed4c221dbad0f33a83a49cad6b3fe8d5817ac83d38b6a19") if err != nil { log.Fatal(err) } ``` 之后我们需要获得帐户`nonce`。每笔交易都需要一个`nonce`。根据定义,`nonce`是仅使用一次的数字。如果是发送交易的新帐户,则`nonce`将为0。来自帐户的每个新交易必须具有前一个`nonce`增加1的`nonce`。很难保持所有`nonce`的手动跟踪,以便ethereum客户端提供辅助方法`PendingNonceAt`将返回你应该使用的下一个`nonce`。 该函数需要我们发送的帐户的公共地址,我们可以从私钥中获取该帐户。 ``` publicKey := privateKey.Public() publicKeyECDSA, ok := publicKey.(*ecdsa.PublicKey) if !ok { log.Fatal("cannot assert type: publicKey is not of type *ecdsa.PublicKey") } fromAddress := crypto.PubkeyToAddress(*publicKeyECDSA) ``` 这里,`privateKey.Public()`返回包含我们的公钥的接口。我们使用`publicKey`。`(<expectedType>)`执行类型断言,以明确设置`publicKey`变量的类型,并将其分配给`publicKeyECDSA`。这允许我们在程序期望输入`*ecdsa.PublicKey`类型的地方使用它。 现在我们可以读取我们应该用于帐户交易的随机数。 ``` nonce, err := client.PendingNonceAt(context.Background(), fromAddress) if err != nil { log.Fatal(err) } ``` 下一步是设置我们将要转移的ETH数量。但是我们必须将以太网转换为wei,因为这是以太坊区块链使用的。以太网支持最多18个小数位,因此1个ETH为1加18个零。这里有一个小工具可以帮助你在ETH和wei之间进行转换:[https://etherconverter.online](https://etherconverter.online) ``` value := big.NewInt(1000000000000000000) // in wei (1 eth) ``` 标准ETH转移的gas限制为`21000`单位。 ``` gasLimit := uint64(21000) // in units ``` gas价格必须在wei中设定。在撰写本文时,将在一个区块中包含相当快的交易的gas价格为30gwei。 ``` gasPrice := big.NewInt(30000000000) // in wei (30 gwei) ``` 然而,gas价格总是根据市场需求和用户愿意支付的价格而波动,因此对gas价格进行硬编码有时并不理想。go-ethereum客户提供`SuggestGasPrice`函数,用于根据先前块的x个数来获取平均gas价格。 ``` gasPrice, err := client.SuggestGasPrice(context.Background()) if err != nil { log.Fatal(err) } ``` 我们弄清楚我们将ETH发送给谁。 ``` toAddress := common.HexToAddress("0x4592d8f8d7b001e72cb26a73e4fa1806a51ac79d") ``` 现在我们可以通过导入`go-ethereum core/types`包并调用`NewTransaction`来生成我们的无符号以太坊交易,`NewTransaction`接收nonce,地址,价值,gas限价,gas价格和可选数据。仅发送ETH的数据字段为零。在与智能合约进行交互时,我们将使用数据字段。 ``` tx := types.NewTransaction(nonce, toAddress, value, gasLimit, gasPrice, nil) ``` 下一步是使用发件人的私钥对交易进行签名。为此,我们调用`SignTx`方法,该方法接受无符号交易和我们之前构造的私钥。`SignTx`方法需要EIP155签名者,我们从客户端派生链ID。 ``` chainID, err := client.NetworkID(context.Background()) if err != nil { log.Fatal(err) } signedTx, err := types.SignTx(tx, types.NewEIP155Signer(chainID), privateKey) if err != nil { log.Fatal(err) } ``` 现在,我们终于准备通过调用接收已签名交易的客户端上的`SendTransaction`将交易广播到整个网络。 ``` err = client.SendTransaction(context.Background(), signedTx) if err != nil { log.Fatal(err) } fmt.Printf("tx sent: %s", signedTx.Hash().Hex()) // tx sent: 0x77006fcb3938f648e2cc65bafd27dec30b9bfbe9df41f78498b9c8b7322a249e ``` 之后,你可以检查块浏览器上的进度,例如[https://rinkeby.etherscan.io/tx/0x77006fcb3938f648e2cc65bafd27dec30b9bfbe9df41f78498b9c8b7322a249e](https://rinkeby.etherscan.io/tx/0x77006fcb3938f648e2cc65bafd27dec30b9bfbe9df41f78498b9c8b7322a249e) ### Full code **transfer_eth.go** ``` package main import ( "context" "crypto/ecdsa" "fmt" "log" "math/big" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/core/types" "github.com/ethereum/go-ethereum/crypto" "github.com/ethereum/go-ethereum/ethclient" ) func main() { client, err := ethclient.Dial("https://rinkeby.infura.io") if err != nil { log.Fatal(err) } privateKey, err := crypto.HexToECDSA("fad9c8855b740a0b7ed4c221dbad0f33a83a49cad6b3fe8d5817ac83d38b6a19") if err != nil { log.Fatal(err) } publicKey := privateKey.Public() publicKeyECDSA, ok := publicKey.(*ecdsa.PublicKey) if !ok { log.Fatal("cannot assert type: publicKey is not of type *ecdsa.PublicKey") } fromAddress := crypto.PubkeyToAddress(*publicKeyECDSA) nonce, err := client.PendingNonceAt(context.Background(), fromAddress) if err != nil { log.Fatal(err) } value := big.NewInt(1000000000000000000) // in wei (1 eth) gasLimit := uint64(21000) // in units gasPrice, err := client.SuggestGasPrice(context.Background()) if err != nil { log.Fatal(err) } toAddress := common.HexToAddress("0x4592d8f8d7b001e72cb26a73e4fa1806a51ac79d") var data []byte tx := types.NewTransaction(nonce, toAddress, value, gasLimit, gasPrice, data) chainID, err := client.NetworkID(context.Background()) if err != nil { log.Fatal(err) } signedTx, err := types.SignTx(tx, types.NewEIP155Signer(chainID), privateKey) if err != nil { log.Fatal(err) } err = client.SendTransaction(context.Background(), signedTx) if err != nil { log.Fatal(err) } fmt.Printf("tx sent: %s", signedTx.Hash().Hex()) } ``` ====================================================================== 分享一些以太坊区块链相关的交互式在线编程实战教程: > - [java以太坊开发教程](http://xc.hubwiz.com/course/5b2b6e82c02e6b6a59171de2?affid=514studygolang),主要是针对java和android程序员进行区块链以太坊开发的web3j详解。 > - [python以太坊](http://xc.hubwiz.com/course/5b40462cc02e6b6a59171de4?affid=514studygolang),主要是针对python工程师使用web3.py进行区块链以太坊开发的详解。 > - [php以太坊](http://xc.hubwiz.com/course/5b36629bc02e6b6a59171de3?affid=514studygolang),主要是介绍使用php进行智能合约开发交互,进行账号创建、交易、转账、代币开发以及过滤器和交易等内容。 > - [以太坊入门教程](http://xc.hubwiz.com/course/5a952991adb3847553d205d1?affid=514studygolang),主要介绍智能合约与dapp应用开发,适合入门。 > - [以太坊开发进阶教程](http://xc.hubwiz.com/course/5abbb7acc02e6b6a59171dd6?affid=514studygolang),主要是介绍使用node.js、mongodb、区块链、ipfs实现去中心化电商DApp实战,适合进阶。 > - [ERC721以太坊通证实战](http://xc.hubwiz.com/course/5c6ed395070c379b559a813a?affid=514studygolang),课程以一个数字艺术品创作与分享DApp的实战开发为主线,深入讲解以太坊非同质化通证的概念、标准与开发方案。内容包含ERC-721标准的自主实现,讲解OpenZeppelin合约代码库二次开发,实战项目采用Truffle,IPFS,实现了通证以及去中心化的通证交易所。 > - [C#以太坊](http://xc.hubwiz.com/course/5b6048c3c02e6b6a59171dee?affid=514studygolang),主要讲解如何使用C#开发基于.Net的以太坊应用,包括账户管理、状态与交易、智能合约开发与交互、过滤器和交易等。 汇智网原创翻译,转载请标明出处。这里是[Go语言如何发送以太坊代币](https://goethereumbook.org/en/transfer-eth/)

有疑问加站长微信联系(非本文作者))

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

3152 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传