跳转至主要内容

针对初学者的智能合约指南

solidityhardhatalchemy智能合约入门指南部署
初学者
elanh
2021年3月31日
17 分钟阅读 minute read

如果您是区块链开发的初学者,还不知道如何开始,或者您只是想了解怎样部署智能合约并与之进行交互,这篇教程就是为您准备的。 通过使用虚拟钱包 (MetaMask(opens in a new tab))、Solidity(opens in a new tab)Hardhat(opens in a new tab)Alchemy(opens in a new tab)(如果您不理解这些名词的含义,不用担心,后续我们会进行解释),我们将演示在 Ropsten 测试网络上创建并部署一个简单的智能合约。

在本教程的第 2 部分,我们将学习在智能合约部署后如何与之进行交互,第 3 部分将学习如何在 Etherscan 上发布智能合约。

如果您有任何问题,请随时在 Alchemy Discord(opens in a new tab) 中提出!

步骤 1:连接到以太坊网络

有多种方法可以向以太坊链发起连接请求。 为了简单起见,我们将使用一个 Alchemy 上的免费帐户。 Alchemy 是一个区块链开发者平台和 API,允许我们与以太坊进行通信,而无需运行我们自己的节点。 平台还有用于监测和分析的开发者工具,我们将在本教程中利用这些工具来了解我们智能合约部署中的情况。 如果您还没有 Alchemy 账户,您可以在这里免费注册(opens in a new tab)

步骤 2:创建应用程序(和应用程序接口密钥)

创建 Alchemy 帐户后,您可以通过创建应用程序来生成应用程序接口密钥。 我们可以用它向 Ropsten 测试网发出请求。 如果您不熟悉测试网络,请查看此页面

  1. 在 Alchemy 仪表板中,将鼠标悬停在导航栏中的“应用程序”上,单击“创建应用程序”并前往此页面。

创建应用程序 Hello world

  1. 将您的应用命名为“Hello World”,提供简短的描述,选择“Staging”作为环境(用于您的应用程序簿记),选择“Ropsten”网络。

创建应用程序视图 hello world

  1. 点击“Create app”,完成! 您的应用程序应该就会出现在下面的表格中。

步骤 3:创建一个以太坊账户(地址)

我们需要一个以太坊帐户来发送和接收交易。 在本教程中,我们将使用 MetaMask,它是浏览器中的虚拟钱包,用来管理您的以太坊账户地址。 关于交易的详细信息。

您可以点击此处(opens in a new tab)免费下载并创建一个 MetaMask 账户。 创建账户时,或者如果您已经有一个账户时,确保切换到右上方的“Ropsten 测试网络”(这样我们就不会用实际货币进行交易)。

metask ropsten 示例

步骤 4:从水龙头添加以太币

为了将我们的智能合约部署到测试网络,我们需要一些虚拟以太币。 要获取以太币,您可以转到 Ropsten 水龙头(opens in a new tab)并输入您的 Ropsten 帐户地址,然后点击“Send Ropsten Eth”。 由于网络原因,您接收虚拟以太币可能需要一些时间。 您应该会很快在您的 MetaMask 帐户中看到以太币!

步骤 5:查看账户余额

为了核查我们账户中有余额,我们使用 Alchemy composer 工具(opens in a new tab)发出 eth_getBalance(opens in a new tab) 请求。 这将返回我们钱包中的以太币数量。 输入您的 MetaMask 帐户地址并点击“Send Request”后,您应该会看到这样的响应:

1{ "jsonrpc": "2.0", "id": 0, "result": "0x2B5E3AF16B1880000" }
复制

注意:结果以 wei 为单位,而非 ETH。 Wei 是以太币的最小计量单位。 将 wei 转换为 ETH 的公式为:1 eth = 1018 wei。 因此,如果我们将 0x2B5E3AF16B1880000 转换为十进制,我们会得到 5*10¹⁸,即 5 ETH。

哦! 我们的虚拟货币到账了

步骤 6:初始化我们的项目

首先,需要为我们的项目创建一个文件夹。 导航到您的命令行,然后输入:

1mkdir hello-world
2cd hello-world

现在我们进入了项目文件夹,我们将使用 npm init 来初始化项目。 如果你尚未安装 npm,请遵循这些说明(opens in a new tab)(我们还需要 Node.js,所以请一并下载!)。

1npm init

如何回答安装中的问题并不重要,以下提供一个回答的样例供参考:

1package name: (hello-world)
2version: (1.0.0)
3description: hello world smart contract
4entry point: (index.js)
5test command:
6git repository:
7keywords:
8author:
9license: (ISC)
10About to write to /Users/.../.../.../hello-world/package.json:
11
12{
13 "name": "hello-world",
14 "version": "1.0.0",
15 "description": "hello world smart contract",
16 "main": "index.js",
17 "scripts": {
18 "test": "echo \\"Error: no test specified\\" && exit 1"
19 },
20 "author": "",
21 "license": "ISC"
22}
显示全部

批准 package.json,我们就可以进行下一步了!

步骤 7:Hardhat(opens in a new tab){#step-7}

安全帽是一个用于编译、部署、测试和调试以太坊软件的开发环境。 它帮助开发者在本地构建智能合约和去中心化应用程序并部署到实时链上。

在我们的 hello-world 项目中运行:

1npm install --save-dev hardhat

查看此页面,了解更多有关安装说明(opens in a new tab)的详细信息。

步骤 8:创建安全帽项目

在我们的项目文件夹中运行:

1npx hardhat

然后应该能看到一条欢迎消息和选项,用于选择您想要做的事情。 选择“创建一个空的 hardhat.config.js”:

1888 888 888 888 888
2888 888 888 888 888
3888 888 888 888 888
48888888888 8888b. 888d888 .d88888 88888b. 8888b. 888888
5888 888 "88b 888P" d88" 888 888 "88b "88b 888
6888 888 .d888888 888 888 888 888 888 .d888888 888
7888 888 888 888 888 Y88b 888 888 888 888 888 Y88b.
8888 888 "Y888888 888 "Y88888 888 888 "Y888888 "Y888
9
10👷 Welcome to Hardhat v2.0.11 👷‍?
11
12What do you want to do? …
13Create a sample project
14❯ Create an empty hardhat.config.js
15Quit
显示全部

这将生成一个 hardhat.config.js 文件,我们将用以配置项目的所有设置(步骤 13)。

步骤 9:添加项目文件夹

为了使我们的项目有条理,我们将创建两个新的文件夹。 在您的命令行中导航到项目的根目录,然后输入:

1mkdir contracts
2mkdir scripts
  • contracts/ 是保存我们的 hello world 智能合约代码文件的位置
  • scripts/ 是我们存放脚本的位置,用于部署我们的合约和与之交互。

步骤 10:编写合约

您可能要问自己,天啊,到底什么时候才能写代码?? 答案是,就是现在,步骤 10。

在您最喜欢的编辑器中打开 hello-world 项目(我们喜欢的是 VSCode(opens in a new tab))。 智能合约是用一种叫 Solidity 的语言编写的,我们将用它来编写我们的 HelloWorld.sol 智能合约。

  1. 导航到“contracts”文件夹并创建一个名为 HelloWorld.sol 的新文件
  2. 下面是我们将用于本教程的来自以太坊基金会的 Hello World 智能合约样例。 复制下面的内容并粘贴到 HelloWorld.sol 文件,并一定要阅读注释以了解该合约的工作内容:
1// Specifies the version of Solidity, using semantic versioning.
2// Learn more: https://solidity.readthedocs.io/en/v0.5.10/layout-of-source-files.html#pragma
3pragma solidity ^0.7.0;
4
5// Defines a contract named `HelloWorld`.
6// 一个合约是函数和数据(其状态)的集合。 Once deployed, a contract resides at a specific address on the Ethereum blockchain. Learn more: https://solidity.readthedocs.io/en/v0.5.10/structure-of-a-contract.html
7contract HelloWorld {
8
9 // Declares a state variable `message` of type `string`.
10 // 状态变量是其值永久存储在合约存储中的变量。 The keyword `public` makes variables accessible from outside a contract and creates a function that other contracts or clients can call to access the value.
11 string public message;
12
13 // Similar to many class-based object-oriented languages, a constructor is a special function that is only executed upon contract creation.
14 // 构造器用于初始化合约的数据。 Learn more:https://solidity.readthedocs.io/en/v0.5.10/contracts.html#constructors
15 constructor(string memory initMessage) {
16
17 // Accepts a string argument `initMessage` and sets the value into the contract's `message` storage variable).
18 message = initMessage;
19 }
20
21 // A public function that accepts a string argument and updates the `message` storage variable.
22 function update(string memory newMessage) public {
23 message = newMessage;
24 }
25}
显示全部
复制

这是一个非常简单的智能合约,创建时存储了一条消息,而且可以通过调用 update 功能来更新消息。

步骤 11:将 MetaMask 和 Alchemy 连接至您的项目

我们创建了 MetaMask 钱包、Alchemy 帐户,并且编写了一个智能合约,现在是将这三者连起来的时候了。

从虚拟钱包发送的每笔交易都需要使用您独有的私钥签名。 为了给程序提供此项许可,我们可以安全地将私钥(和 Alchemy 应用程序接口密钥)存储在一个环境文件中。

如需了解更多关于发送交易的信息,请查看关于使用 web3 发送交易的 本教程

首先,在项目目录中安装 dotenv 软件包:

1npm install dotenv --save

然后在我们的项目根目录中创建 .env 文件,并将您的 MetaMask 私钥和超文本传输协议 Alchemy 应用程序接口网址加入其中。

获取 alchemy 应用程序接口密钥

复制 Alchemy 应用程序接口网址

您的 .env 文件应该类似:

1API_URL = "https://eth-ropsten.alchemyapi.io/v2/your-api-key"
2PRIVATE_KEY = "your-metamask-private-key"

为了将这些变量和代码连接,我们将在步骤 13 中调用 hardhat.config.js 文件中的这些变量。

不要提交 .env! 请确保永远不要与任何人共享或公开您的 .env 文件,因为这样做会泄露您的私钥。 如果您使用版本控制,请将您的 .env 添加到 gitignore(opens in a new tab) 文件中。

步骤 12:安装 Ethers.js

Ethers.js 是一个软件库,通过以更加方便用户的方法打包标准 JSON RPC 方法,从而更容易与以太坊互动,并向以太坊提出请求。

安全帽使我们更容易将插件(opens in a new tab)集成到工具和扩展功能中。 我们将利用 Ethers 插件(opens in a new tab)完成合约部署(Ethers.js(opens in a new tab) 有非常简洁的部署方法)。

在您的项目目录中输入:

1npm install --save-dev @nomiclabs/hardhat-ethers "ethers@^5.0.0"

下一步中,我们还将在 hardhat.config.js 中使用 ethers。

步骤 13:更新 hardhat.config.js

到目前为止,我们已经添加了几个依赖库和插件,现在我们需要更新 hardhat.config.js,以便项目使用所有这些新的组件。

按如下所示更新您的 hardhat.config.js 代码:

1require('dotenv').config();
2
3require("@nomiclabs/hardhat-ethers");
4const { API_URL, PRIVATE_KEY } = process.env;
5
6/**
7* @type import('hardhat/config').HardhatUserConfig
8*/
9module.exports = {
10 solidity: "0.7.3",
11 defaultNetwork: "ropsten",
12 networks: {
13 hardhat: {},
14 ropsten: {
15 url: API_URL,
16 accounts: [`0x${PRIVATE_KEY}`]
17 }
18 },
19}
显示全部

步骤 14:编译合约

为了确保一切正常,我们来编译一下合约。 compile 任务是安全帽的内部任务之一。

在命令行中运行:

1npx hardhat compile

您可能会看到关于 SPDX license identifier not provided in source file 的警告,但不用担心,希望其它的均看起来正常! 如果遇到问题,您可以随时在 Alchemy cord(opens in a new tab) 社区中发消息询问。

步骤 15:编写部署脚本

合约已经写完,配置文件也准备妥当,现在是写合约部署脚本的时候了。

转到 scripts/ 文件夹,创建一个名为 deploy.js 的新文件,在其中添加以下内容:

1async function main() {
2 const HelloWorld = await ethers.getContractFactory("HelloWorld");
3
4 // Start deployment, returning a promise that resolves to a contract object
5 const hello_world = await HelloWorld.deploy("Hello World!");
6 console.log("Contract deployed to address:", hello_world.address);}
7
8main()
9 .then(() => process.exit(0))
10 .catch(error => {
11 console.error(error);
12 process.exit(1);
13 });
显示全部

安全帽在合约教程(opens in a new tab)中对这些代码的每一行均提供了很好的解释,我们在这里直接引用他们的解释。

1const HelloWorld = await ethers.getContractFactory("HelloWorld");

ethers.js 中的 ContractFactory 是用于部署新智能合约的抽象对象。因此这里的 HelloWorld 是我们 hello world 合约实例的工厂。 使用 hardhat-ethers 插件时,ContractFactoryContract 实例默认与第一个签名账户相连。

1const hello_world = await HelloWorld.deploy();

调用 ContractFactory 代码中的 deploy() 函数会启动合约部署,然后返回解析为 ContractPromise。 这个对象包括我们智能合约中每个函数的对应调用方法。

步骤 16:部署合约

我们终于准备好部署我们的智能合约啦! 导航到命令行后运行:

1npx hardhat run scripts/deploy.js --network ropsten

您会看到类似以下所示的信息:

1Contract deployed to address: 0x6cd7d44516a20882cEa2DE9f205bF401c0d23570

如果我们访问 Ropsten etherscan(opens in a new tab) 并搜索我们的合约地址,应该能够看到它已经成功部署。 交易将类似以下:

etherscan 合约

From 地址应该和您的 MetaMask 账户地址相同,而 To 地址会显示“Contract Creation”,但如果我们点击进入交易,我们会在 To 字段看到我们的合约地址:

etherscan 交易

恭喜! 您刚刚在以太坊区块链上部署了一个智能合约 🎉

为了更深入了解到底发生了什么,我们转到 Alchemy 仪表板(opens in a new tab)中的 Explorer 选项卡。 如果您有多个 Alchemy 应用程序,请确保按应用程序筛选,然后选择“Hello World”。 hello world 浏览器

在这里您会看到一系列的 JSON-RPC 调用,都是在我们调用 .deploy() 函数时,Hardhat/Ethers 替我们在后端完成的。 这里有两项重要调用,一个是 eth_sendRawTransaction(opens in a new tab),这是实际将我们的合约写入 Ropsten 链的请求,另一个是 eth_getTransactionByHash(opens in a new tab),此为读取有关我们交易给定哈希值的请求(即 交易时的典型模式)。 如需了解更多关于发送交易的信息,请查看关于使用 Web3 发送交易的本教程

这是本教程第 1 部分的全部内容,在第 2 部分中,我们将更新我们的初始信息,从而与我们的智能合约交互(opens in a new tab);在第 3 部分,我们将在 Etherscan 上发布我们的智能合约(opens in a new tab),使得每个人都会知道如何与之交互。

想了解更多有关 Alchemy 的信息吗? 查看我们的网站(opens in a new tab)。 不想错过更新? 在这里(opens in a new tab)订阅我们的新闻通讯! 请务必关注我们的 Twitter(opens in a new tab) 并加入我们的 Discord(opens in a new tab)

上次修改时间: @pettinarip(opens in a new tab), 2023年11月23日

本教程对你有帮助吗?