golang调用智能合约,获取合约函数的返回值

如果不是只读取数据的合约函数,需要异步的执行,因此并不能直接获取到合约函数的返回值,需要等到交易执行完毕,得到确认后才能获取到合约函数的返回值。而且合约函数返回值一般是通过事件日志获取到的。

这里给出一个例子来展示我是如何获取合约函数返回值的。

我使用的以太坊版本为:github.com/ethereum/go-ethereum v1.13.0

solidity合约:

javascript 复制代码
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

contract StoreString {
  event ItemSetStr(bytes32 indexed hash,string key,string value);
  
  mapping (string => string) public itemstr;

  function setItemstr(string memory _key, string memory _value) external returns(bytes32){
    itemstr[_key] = _value;
    bytes32 hash=sha256(abi.encodePacked(_key,_value));
    emit ItemSetStr(hash,_key,_value);
    return hash;
  }
}

这是一个存储键值对的合约,利用函数setItemstr向map中存储键值对,并通过key,value计算哈希值,要求返回哈希值。通过事件ItemSetStr将哈希值(返回值)、key、value记录下来。

将合约的abi编码通过abigen工具生成go代码后。

以下是测试获取合约函数返回值的代码:

go 复制代码
func TestSetStoreString(t *testing.T) {
	url="" //节点链接
	client, err := ethclient.Dial(url)
	if err != nil {
		log.Fatal(err)
	}

	//合约地址
	contractAddress := common.HexToAddress("0xd9Ed5E352F84E182eB499ae1b5F9935C06d78Ddb")

	privateKeyStr := "" //私钥字符串

	auth := InitAuth(privateKeyStr, client, nil)
	
	//得到合约实例
	storeString, err := NewStoreString(contractAddress, client)
	if err != nil {
		log.Fatal("NewStoreString: ", err)
	}
	
	tx, err := storeString.SetItemstr(auth, "key10", "9900")
	if err != nil {
		log.Fatal("SetItemstr: ", err)
	}
	
	//等待交易确认,获取到交易的收据
	receipt, err := bind.WaitMined(context.Background(), client, tx)
	if err != nil {
		log.Fatal("WaitMined: ", err)
	}
	
	//如果交易成功
	if receipt.Status == 1 {
		//解析该交易收据里包含的日志
		eventItemSetStr, err := storeString.ParseItemSetStr(*receipt.Logs[0])
		if err != nil {
			log.Fatal("ParseItemSetStr: ", err)
		}
		fmt.Println("交易成功!")
		fmt.Println("txHash:", tx.Hash().Hex())
		fmt.Println("返回值:", eventItemSetStr.Hash)
		fmt.Println("日志Key:", eventItemSetStr.Key)
		fmt.Println("日志Value:", eventItemSetStr.Value)
	} else {
		t.Error("交易失败")
	}

}


func InitAuth(privateKeyStr string, client *ethclient.Client, value *big.Int) *bind.TransactOpts {
	privateKeyECDSA, err := crypto.HexToECDSA(privateKeyStr)
	if err != nil {
		log.Fatal("crypto.HexToECDSA: ", err)
	}

	chainID, err := client.ChainID(context.Background())
	if err != nil {
		log.Fatal("ChainID: ", err)
	}

	auth, err := bind.NewKeyedTransactorWithChainID(privateKeyECDSA, chainID)
	if err != nil {
		log.Fatal("NewKeyedTransactorWithChainID: ", err)
	}

	//设置交易参数,例如gas费等,这里全部由区块链系统评估决定
	auth.GasFeeCap = big.NewInt(1).Mul(big.NewInt(10), big.NewInt(1000000000)) // maxFee Per Gas:100gwei

	//获取平均小费
	gasTipCap, err := client.SuggestGasTipCap(context.Background())
	if err != nil {
		log.Fatal("SuggestGasTipCap: ", err)
	}
	auth.GasTipCap = gasTipCap //priorityfee Per Gas:
	auth.GasLimit = uint64(3000000)

	//还需要设置锁定金额
	auth.Value = value
	return auth
}

返回值并不能立马获得,需要等待交易成功上链。再从事件日志中解析出返回值。因此这个过程中用到了两个重要代码:

bind包下的WaitMined:等待tx交易的确认,并获取其收据,该收据中必然包含我们想要的日志

go 复制代码
func WaitMined(ctx context.Context, b DeployBackend, tx *types.Transaction) (*types.Receipt, error)

合约代码自动生成的go文件中的ParseItemSetStr,传入日志即可解析出信息。

go 复制代码
func (_StoreString *StoreStringFilterer) ParseItemSetStr(log types.Log) (*StoreStringItemSetStr, error)
相关推荐
007php0074 小时前
Git 操作偏门指南:常用和隐藏命令与问题解决
java·git·面试·职场和发展·golang·jenkins·php
lsrsyx4 小时前
TEBBIT 交易所:在数字资产时代构筑信任与性能的新标杆
区块链
源代码•宸4 小时前
goframe框架签到系统项目开发(每日签到添加积分和积分记录、获取当月最大连续签到天数、发放连续签到奖励积分、实现签到日历详情接口)
数据库·经验分享·redis·中间件·golang·dao·goframe
梦想的旅途25 小时前
企业微信外部群消息推送实战指南
java·golang·企业微信
古城小栈5 小时前
go-zero 从入门到实战 全指南(包的)
开发语言·后端·golang
Clarence Liu6 小时前
用 Go 从 100 亿个数中找到最小的 100 个数 —— 实战与原理
开发语言·后端·golang
小明的小名叫小明7 小时前
Aave协议(3)
区块链·defi
Zoey的笔记本7 小时前
构建去中心化协作引擎:基于开源框架的Web3团队项目管理实践
开源·web3·区块链
Sui_Network9 小时前
Sui 2025 年终回顾:Sui 技术栈篇
大数据·人工智能·科技·去中心化·区块链
007php0079 小时前
通过程序对接地图api展示旅游数据列表
java·数据库·面试·职场和发展·golang·php·旅游