Web3.js 实战:基于 Hardhat 的以太坊投票 DApp 开发全流程

本文以一个完整的投票 DApp 项目为例,带你从合约开发、部署到前端集成 Web3.js,体验 Web3 全栈开发流程。


一、项目结构与部署流程时序图

sequenceDiagram participant D as 开发者 participant H as Hardhat participant F as 前端 participant U as 用户 D->>H: 编写 Voting.sol 合约 D->>H: npx hardhat compile D->>H: npx hardhat node D->>H: npx hardhat ignition deploy H->>F: 生成 ABI/地址 (contractInfo.json) U->>F: 打开 vote.html F->>F: 读取 contractInfo.json F->>H: 通过 Web3.js 连接本地节点 U->>F: 投票/查询操作 F->>H: 调用合约方法 H-->>F: 返回结果 F-->>U: 展示投票结果
  • contracts/:Solidity 智能合约源码
  • ignition/modules/:合约部署脚本
  • artifacts/:合约 ABI、字节码等编译产物
  • frontend/:前端页面与合约信息

二、合约开发与部署

1. 编写 Voting 合约

solidity 复制代码
// contracts/Voting.sol
pragma solidity ^0.7.4;

contract Voting {
    struct Candidate { uint id; string name; uint voteCount; }
    mapping(uint => Candidate) public candidates;
    mapping(address => bool) public voters;
    uint public candidatesCount;
    event CandidateAdded(uint id, string name);
    event Voted(uint indexed candidateId);

    constructor() {
        addCandidate("Alice");
        addCandidate("Bob");
    }
    function addCandidate(string memory _name) private {
        candidatesCount++;
        candidates[candidatesCount] = Candidate(candidatesCount, _name, 0);
        emit CandidateAdded(candidatesCount, _name);
    }
    function vote(uint _candidateId) public {
        require(!voters[msg.sender], "You have already voted.");
        require(_candidateId > 0 && _candidateId <= candidatesCount, "Invalid candidate ID.");
        voters[msg.sender] = true;
        candidates[_candidateId].voteCount++;
        emit Voted(_candidateId);
    }
    function getResult() public view returns (string memory winnerName, uint winnerVoteCount) {
        uint winningVoteCount = 0;
        for (uint i = 1; i <= candidatesCount; i++) {
            if (candidates[i].voteCount > winningVoteCount) {
                winningVoteCount = candidates[i].voteCount;
                winnerName = candidates[i].name;
            }
        }
        winnerVoteCount = winningVoteCount;
    }
}

2. Hardhat 编译与本地链部署

bash 复制代码
npm install
npx hardhat compile
npx hardhat node

部署合约:

bash 复制代码
npx hardhat ignition deploy ./ignition/modules/Voting.js --network localhost

部署成功后,终端会输出合约地址:


三、前端集成 Web3.js

1. 前端页面结构

html 复制代码
<!-- frontend/vote.html -->
<script src="https://cdn.jsdelivr.net/npm/web3/dist/web3.min.js"></script>
<script>
  async function loadContractInfo() {
    const res = await fetch('contractInfo.json');
    return await res.json();
  }
  let contract, web3;
  loadContractInfo().then(({ abi, address }) => {
    web3 = new Web3(Web3.givenProvider || 'http://localhost:8545');
    contract = new web3.eth.Contract(abi, address);
    loadCandidates();
  });

  async function loadCandidates() {
    const candidatesCount = await contract.methods.candidatesCount().call();
    // ...渲染候选人
  }
  async function vote() {
    const candidateId = document.getElementById('candidateId').value;
    const accounts = await web3.eth.getAccounts();
    await contract.methods.vote(candidateId).send({ from: accounts[0] });
    alert('投票成功!');
    loadCandidates();
  }
  async function getResults() {
    const result = await contract.methods.getResult().call();
    // ...显示结果
  }
</script>

2. 合约 ABI 和地址自动同步 (这里需要手动修改json文件)

json 复制代码
// frontend/contractInfo.json
{
  "abi": [ ... ],
  "address": "0x1234567890abcdef..."
}

3. 启动本地前端服务

bash 复制代码
cd frontend
python3 -m http.server 8080

浏览器访问 http://localhost:8080/vote.html


四、Web3.js 交互流程图

sequenceDiagram participant U as 用户 participant F as 前端(vote.html) participant W as Web3.js participant C as 合约(Voting) U->>F: 输入候选人ID点击投票 F->>W: 调用 contract.methods.vote(id).send() W->>C: 发送交易到 Voting 合约 C-->>W: 交易回执/事件 W-->>F: 返回投票结果 F-->>U: 页面提示"投票成功"

五、常见问题与调试

  • web3.js 加载慢/超时:可切换为本地 web3.min.js 文件
  • 合约地址不对/前端报错:请确保 contractInfo.json 里的地址和最新部署一致
  • 投票失败:请确保钱包连接本地节点,或本地节点已启动

六、参考资料


建议:

在实际项目中,建议将合约 ABI/地址自动同步到前端,并用脚本化方式管理部署和前端集成,提升开发效率。

相关推荐
暴躁小师兄数据学院10 小时前
【WEB3.0零基础转行笔记】Rust编程篇-第一讲:课程简介
rust·web3·区块链·智能合约
devmoon10 小时前
在 Paseo 测试网上获取 Coretime:On-demand 与 Bulk 的完整实操指南
开发语言·web3·区块链·测试用例·智能合约·solidity
devmoon16 小时前
在 Polkadot Runtime 中添加多个 Pallet 实例实战指南
java·开发语言·数据库·web3·区块链·波卡
Web3VentureView17 小时前
SYNBO Protocol AMA回顾:下一个起点——什么将真正推动比特币重返10万美元?
大数据·人工智能·金融·web3·区块链
REDcker20 小时前
Web1 到 Web3 技术演进详解
web3
devmoon1 天前
Polkadot SDK 平行链模板搭建全流程指南
web3·区块链·sdk·比特币·波卡
China_Yanhy1 天前
入职 Web3 运维日记 · 第 8 日:黑暗森林 —— 对抗 MEV 机器人的“三明治攻击”
运维·机器人·web3
devmoon2 天前
使用 Hardhat 在 Polkadot Hub 测试网部署基础 Solidity 合约(完整实战指南)
web3·区块链·智能合约·波卡·hardhat
devmoon3 天前
用Remix IDE在Polkadot Hub部署一个最基础的Solidity 合约(新手友好)
web3·区块链·智能合约·编译·remix·polkadot
暴躁小师兄数据学院3 天前
【WEB3.0零基础转行笔记】Golang编程篇-第4讲:Go语言中的流程控制
开发语言·后端·golang·web3·区块链