在以太坊及众多兼容链上,智能合约是构建去中心化应用(DApps)的核心,ERC(Ethereum Request for Comments)标准为这些合约提供了统一的接口和行为规范,确保了代币的互操作性和安全性,ERC777作为对ERC20标准的增强,引入了许多先进特性,如可扩展性、内置发送钩子和操作员模式等,使其在某些场景下更具优势,本文将详细介绍如何在以太坊网络上部署一个ERC777标准的智能合约。
ERC777标准简介:为何选择它
在深入部署之前,我们先简要了解ERC777相较于ERC20的优势:
- 可扩展性与可替代性:ERC777代币可以是可替代的(如ERC20),也可以是不可替代的(如ERC721),或者两者的混合,这为更复杂的代币设计提供了可能。
- 内置发送钩子(Send Hooks):当代币被发送(包括
transfer和send)时,合约可以触发特定的回调函数(tokensToSend),这使得接收方合约能够自动处理收到的代币,无需额外的交易步骤,极大地提升了用户体验和合约间的交互效率。 - 操作员(Operators)模式:代币持有者可以批准其他地址(操作员)来管理他们的代币,而无需每次操作都进行授权,操作员可以代表持有者发送或接收代币,这在需要批量管理或授权第三方服务(如交易所、钱包)的场景下非常有用。
- 改进的事件:ERC777提供了更丰富的事件,如
Sent、Minted、Burned等,使得代币流转的追踪和监听更加方便。 - 向后兼容ERC20:ERC777合约实现了ERC20的所有接口,因此ERC20的应用程序也可以与ERC777代币交互,但反之则不然,可以通过“ERC777代币作为ERC20代理”的方式实现部分兼容。
部署ERC777智能合约的前期准备
在开始部署之前,你需要准备以下几样东西:
- 一个以太坊钱包:如MetaMask,并确保钱包中有足够的ETH用于支付 gas 费。
- 测试网ETH(可选但推荐):如果你想在以太坊测试网(如Goerli、Sepolia)上进行测试和调试,需要从对应的 Faucet 获取测试ETH。
- 开发环境:
- Node.js 和 npm/yarn:用于运行JavaScript/TypeScript代码。
- Truffle 或 Hardhat:流行的以太坊开发框架,用于编译、部署和测试智能合约。
- Solidity 编译器:确保安装了与合约代码兼容的Solidity版本。
- ERC777合约代码:
你可以编写自己的ERC777合约,但更常见的是使用经过审计的标准实现,OpenZeppelin提供了广泛使用的、安全的ERC777合约实现。
使用OpenZeppelin库部署ERC777合约
OpenZeppelin是一个提供安全、社区审计的智能合约标准库的开源项目,使用其ERC777实现可以大大降低开发风险。
步骤1:初始化项目
如果你还没有项目,可以创建一个新的目录并初始化:
mkdir my-erc777-project cd my-erc777-project npm init -y
步骤2:安装依赖
安装OpenZeppelin合约库和开发框架(以Hardhat为例):
npm install @openzeppelin/contracts hardhat
然后初始化Hardhat项目:
npx hardhat init
按照提示选择配置(Create a basic sample project)。
步骤3:编写ERC777合约
在contracts目录下,创建一个新的合约文件,例如MyERC777Token.sol:
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/token/ERC777/ERC777.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
/**MyERC777Token
* @dev 一个简单的ERC777代币合约,继承自OpenZeppelin的ERC777和Ownable。
*/
contract MyERC777Token is ERC777, Ownable {
constructor(
string memory name,
string memory symbol,
address[] memory defaultOperators
) ERC777(name, symbol, defaultOperators) {
// 通常在构造函数中铸造初始代币给自己或指定地址
// _mint(address to, uint256 amount, bytes memory userData, bytes memory operatorData)
_mint(msg.sender, 1000000 * 10**decimals(), "Initial mint", "0x");
}
// 你可以在这里添加自定义的逻辑,例如额外的铸造、销毁函数等
// 但要注意,ERC777本身没有标准的mint/burn函数,需要由合约实现者提供
// 或者
使用ERC777PresetMinterPauser(如果需要预铸造者和暂停功能)
}
注意:上面的例子是一个基础的ERC777合约,OpenZeppelin还提供了ERC777PresetMinterPauser预设合约,它已经包含了铸造者角色和暂停功能,更适合直接使用或在此基础上修改,你可以参考OpenZeppelin文档。
步骤4:配置部署脚本
在scripts目录下,创建一个部署脚本,例如deploy.js:
async function main() {
// 获取合约工厂
const MyERC777Token = await ethers.getContractFactory("MyERC777Token");
// 合约构造参数
const name = "My Awesome ERC777 Token";
const symbol = "MAET7";
const defaultOperators = []; // 可以传入默认操作员地址数组,或为空
// 部署合约
const myERC777Token = await MyERC777Token.deploy(name, symbol, defaultOperators);
await myERC777Token.deployed();
console.log("MyERC777Token deployed to:", myERC777Token.address);
}
main()
.then(() => process.exit(0))
.catch((error) => {
console.error(error);
process.exit(1);
});
步骤5:配置网络(如果部署到测试网或主网)
在hardhat.config.js中,你需要配置你要部署的网络信息,包括RPC URL和私钥(或使用HD Wallet Provider,如MetaMask)。
添加Goerli测试网配置:
require("@nomicfoundation/hardhat-toolbox");
require('dotenv').config(); // 使用dotenv管理环境变量
const GOERLI_RPC_URL = process.env.GOERLI_RPC_URL;
const PRIVATE_KEY = process.env.PRIVATE_KEY;
/** @type import('hardhat/config').HardhatUserConfig */
module.exports = {
solidity: "0.8.17",
networks: {
goerli: {
url: GOERLI_RPC_URL,
accounts: [PRIVATE_KEY],
chainId: 5,
},
// 可以添加其他网络,如主网、Sepolia等
},
etherscan: {
apiKey: process.env.ETHERSCAN_API_KEY, // 用于验证合约
},
};
创建.env文件存储敏感信息:
GOERLI_RPC_URL=https://goerli.infura.io/v3/YOUR_INFURA_PROJECT_ID
PRIVATE_KEY=你的钱包私钥(不要泄露!)
ETHERSCAN_API_KEY=你的Etherscan API Key
步骤6:编译和部署合约
在终端中运行以下命令编译合约:
npx hardhat compile
然后部署合约(以Goerli测试网为例):
npx hardhat run scripts/deploy.js --network goerli
如果部署成功,你会在终端看到合约的地址,类似:MyERC777Token deployed to: 0x1234567890123456789012345678901234567890
验证合约(可选但推荐)
为了增加合约的可信度和透明度,你可以将合约代码验证到以太坊浏览器(如Etherscan)上。
- 确保你在
.env文件中配置了ETHERSCAN_API_KEY。 - 在部署成功后,Hardhat会提示你如何验证,或者你可以手动运行:
npx hardhat verify --network goerli "0x你的合约地址" "参数1" "参数2" "参数3"
对于MyERC777Token,参数就是部署时的name, symbol, defaultOperators。
与部署的ERC777合约交互
部署完成后,你可以:
- 在以太坊浏览器中查看合约:输入合约地址,可以看到合约代码、事件、代币余额等。
- 使用钱包测试:将合约添加到MetaMask(通过添加自定义代币