在区块链应用的开发与测试阶段,我们往往需要一个安全、可控且成本极低的网络环境,公共以太坊主网因其交易费用高昂、网络拥堵且不可逆,显然不适合作为日常开发的“沙盒”,这时,以太坊私有链 便成为了开发者的首选,本文将详细介绍如何从零开始,搭建一个功能完备的以太坊私有链,并重点讲解如何管理其中的多账户,以满足不同角色(如开发者、测试用户、合约部署者等)的需求。
在深入技术细节前,我们先明确使用以太坊私有链的核心优势:
在开始之前,请确保你的电脑已经安装了以下工具:
geth 和 solc)依赖于 Node.js 环境。你可以通过以下命令安装 Geth(以 macOS/Linux 为例):

# 或者从官方 GitHub 下载二进制文件
创建私有链的核心是创世区块,创世区块是区块链的“起点”,它定义了网络的初始规则和状态,我们需要创建一个自定义的 JSON 文件来定义它。
创建创世配置文件 genesis.json
在你的工作目录下,创建一个名为 genesis.json 的文件,并填入以下内容:
{
"config": {
"chainId": 15, // 私有链的唯一ID,与主网(1)、Ropsten(3)等不同即可
"homesteadBlock": 0,
"eip150Block": 0,
"eip155Block": 0,
"eip158Block": 0
},
"alloc": {}, // 预先分配账户的地方,我们暂时为空
"coinbase": "0x0000000000000000000000000000000000000000",
"difficulty": "0x4000", // 初始难度,较低值可以让出块更快
"extraData": "",
"gasLimit": "0xffffffff",
"nonce": "0x0000000000000042",
"mixhash": "0x0000000000000000000000000000000000000000000000000000000000000000",
"parentHash": "0x0000000000000000000000000000000000000000000000000000000000000000",
"timestamp": "0x00"
} 注意:chainId 是私有链的身份标识,确保它是一个独一无二的数字。
初始化并启动私有链节点
打开终端,进入 genesis.json 所在的目录,执行以下命令来初始化数据目录:
geth --datadir "./my-private-chain" init genesis.json
这条命令会在当前目录下创建一个名为 my-private-chain 的文件夹,用于存储区块链数据。
启动节点:
geth --datadir "./my-private-chain" --networkid 15 console
--datadir: 指定数据目录。--networkid: 指定网络ID,必须与 genesis.json 中的 chainId 保持一致。console: 启动一个交互式 JavaScript 控制台,方便我们直接操作。成功启动后,你将看到类似 Welcome to the Geth JavaScript console! 的提示,这意味着你已经连接到了你的私有链节点。

让我们来管理私有链中的账户,在 geth console 中,我们可以使用一系列内置命令。
创建新账户
使用 personal.newAccount() 命令创建新账户,系统会提示你输入两次密码来加密账户的私钥。
personal.newAccount() // 提示输入密码: "123456" // 输出: "0x5e3b9d5b5d5b5d5b5d5b5d5b5d5b5d5b5d5b5d5b"
我们再创建一个账户:
personal.newAccount() // 提示输入密码: "654321" // 输出: "0x7f8a8d8c8c8c8c8c8c8c8c8c8c8c8c8c8c8c8c8c8"
我们拥有了两个账户:0x5e3b... 和 0x7f8a...。
查看所有账户
使用 eth.accounts 可以列出当前节点中所有已解锁的账户地址:
eth.accounts // 输出: ["0x5e3b9d5b5d5b5d5b5d5b5d5b5d5b5d5b5d5b5d5b", "0x7f8a8d8c8c8c8c8c8c8c8c8c8c8c8c8c8c8c8c8c"]
解锁账户
在发送交易或部署合约之前,必须先解锁账户。personal.unlockAccount() 命令用于此目的。

personal.unlockAccount(eth.accounts[0], "123456") // 输出: true
给账户分配初始以太币
我们的私有链还没有挖矿,所以账户余额为0,我们需要手动给一些账户分配“创世币”,在 alloc 字段中分配是方法之一,但更简单的方法是在节点启动后,通过“挖矿”来产生币。
启动挖矿:
miner.start(1) // 1 表示使用1个CPU线程进行挖矿
挖矿开始后,你可以查看账户余额变化:
eth.getBalance(eth.accounts[0]) // 输出: "0" (可能需要等待几秒钟才会看到增长)
等待一段时间后,再次查看:
eth.getBalance(eth.accounts[0]) // 输出: "1000000000000000000" (即 1 ETH)
默认情况下,每挖到一个区块,矿工(coinbase)会获得 5 ETH,你可以通过 miner.setEtherbase(eth.accounts[1]) 将矿工地址切换到第二个账户,然后继续挖矿为它分配币。
当不需要挖矿时,记得停止它:
miner.stop() // 输出: true
账户间转账
让我们从账户0向账户1转账0.1 ETH,转账是一个交易,需要构造一个交易对象并发送。
检查余额:
// 账户0余额 eth.getBalance(eth.accounts[0]) // 输出: "1000000000000000000" // 账户1余额 eth.getBalance(eth.accounts[1]) // 输出: "0" (假设之前没有给它挖矿)
构造并发送交易:
// 解锁账户1,因为它将接收交易
personal.unlockAccount(eth.accounts[1], "654321")
// 发送交易
eth.sendTransaction({
from: eth.accounts[0],
to: eth.accounts[1],
value: web3.toWei(0.1, "ether"),
gas: 21000 // 转账的最低Gas限制
})
// 输出: