以太坊,作为全球领先的智能合约平台,其核心魅力在于允许开发者编写和部署自动执行的程序——智能合约,而理解以太坊合约的结构,是掌握智能合约开发、安全审计以及与以太坊交互的基础,本文将深入剖析以太坊合约的内部结构,揭示其如何实现去中心化应用的逻辑。

合约的顶层:Solidity源代码与编译单元

以太坊智能合约通常使用Solidity语言编写,一个Solidity源文件(.sol)可以包含多个合约(contract)、库(library)、接口(interface)和结构体(struct)等,从编译的角度看,每个独立的合约(或接口、库)是一个编译单元

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
// 这是一个编译单元,包含一个名为SimpleStorage的合约
contract SimpleStorage {
    // 合约内容将在下文详述
}
  • pragma:指令,用于指定编译器版本(如^0.8.0表示兼容0.8.0及以上但低于0.9.0的版本)。
  • contract关键字:定义一个合约,合约名(如SimpleStorage)是必需的。

合约的核心构成:状态变量与函数

合约的主体由两部分核心内容组成:状态变量(State Variables)函数(Functions)

状态变量 (State Variables)

状态变量是数据存储在合约中的持久化字段,它们被永久存储在以太坊区块链的特定地址(合约地址)下,每个状态变量都有一个类型(如uint256, address, bool, string, 或自定义的struct/array等)和访问修饰符(如public, private, internal, external)。

contract SimpleStorage {
    uint256 public storedData; // 状态变量,类型为uint256,默认private,public会自动生成getter函数
    string public contractName = "My First Contract";
    address private owner; // 私有状态变量,仅合约内部可访问
}
  • 存储位置:状态变量默认存储在存储(Storage)中,这是区块链上最昂贵但持久的存储。
  • public修饰符:编译器会自动为public状态变量生成一个getter函数,允许其他合约或外部账户读取其值,但不能直接修改(除非有相应的函数)。

函数 (Functions)

函数是合约中定义的执行逻辑单元,用于读取、修改状态变量,或者执行其他计算,函数由以下关键部分组成:

  • 函数修饰符(Function Modifiers):如public, external, internal, private(控制访问权限),view(不修改状态),pure(不访问也不修改状态),payable(允许接收以太币),以及自定义修饰符(如onlyOwner)。
  • 参数(Parameters):函数输入,每个参数有类型和名称。
  • 返回值(Return Values):函数输出,指定返回值的类型。
  • 函数体(Body):由包围的Solidity代码块,包含实现逻辑。
contract SimpleStorage {
    uint256 public storedData;
    // 函数:设置storedData的值
    function set(uint256 x) public {
        storedData = x;
    }
    // 函数:获取storedData的值(由于storedData是public,编译器已自动生成此函数)
    // function get() public view returns (uint256) {
    //     return storedData;
    // }
}
  • public函数:可以从合约内部或其他合约调用。
  • external函数:只能从合约外部调用(或通过this.f()),在合约内部调用效率较低。
  • internal函数:只能从当前合约或继承它的合约内部调用。
  • private函数:只能从定义它的合约内部调用,不能被继承。
  • view函数:承诺不修改状态变量,调用时不会消耗Gas(如果是外部调用)。
  • pure函数:承诺不访问也不修改状态变量,调用时不会消耗Gas(如果是外部调用)。
  • payable函数:允许在调用时向合约发送以太币。

合约的“基因”:继承与接口

为了代码复用和模块化,以太坊合约支持继承(Inheritance)

继承 (Inheritance)

合约可以使用is关键字继承其他合约,继承的合约可以获得父合约的状态变量和函数(根据访问修饰符)。

contract Ownable {
    address public owner;
    constructor() {
        owner = msg.sender;
    }
    modifier onlyOwner() {
        require(msg.sender == owner, "Not the owner");
        _;
    }
}
contract MyContract is Ownable {
    // MyContract拥有owner状态变量和onlyOwner修饰符
    function doSomethingOnlyOwner() public onlyOwner {
        // 只有owner可以调用
    }
}
  • 多重继承:合约可以继承多个父合约,使用逗号分隔,Solidity使用C3线性化算法解决菱形继承问题。
  • 修饰符继承:子合约可以使用父合约的修饰符。

接口 (Interfaces)

接口定义了一组函数签名,但不包含实现,它类似于抽象合约(所有函数都是external且没有函数体),接口用于定义合约之间的标准交互方式。随机配图