DEV Community

Cover image for Day 28 - Abstract Contract and Interface
Vedant Chainani
Vedant Chainani

Posted on

Day 28 - Abstract Contract and Interface

GitHub logo Envoy-VC / 30-Days-of-Solidity

30 Days of Solidity step-by-step guide to learn Smart Contract Development.

This is Day 28 of 30 in Solidity Series
Today I Learned About Abstract Contract and Interfaces in Solidity.

Abstract Contract

Abstract Contract is one which contains at least one function without any implementation. Such a contract is used as a base contract. Generally an abstract contract contains both implemented as well as abstract functions. Derived contract will implement the abstract function and use the existing functions as and when required.

Example-

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.7;

abstract contract Calculator {
   function getResult() virtual public returns(uint);
}
contract Test is Calculator {
   function getResult() override public  pure returns(uint) {
      uint a = 1;
      uint b = 2;
      uint result = a + b;
      return result;
   }
}
Enter fullscreen mode Exit fullscreen mode

Interface

When it comes to smart contracts, interfaces are their skeleton or backbone. It defines the contract’s functionalities and how to trigger them.

This way, Dapps or other smart contracts can know how to interact with any smart contract implementing a specific interface since the interface is known.

The most relevant example is by far the popular ERC20 token standard. This interface standard defines standard functions for a contract that represent a token. The Solidity code taken from OpenZeppelin explains pretty well what these functions are supposed to do.

There are so many ERC20 tokens out there! And all of them can be interacted with in the same way.

  • want to know how many tokens X you own? Call the balanceOf(...)function.
  • want to transfer some tokens X to someone? Call the transfer(...)function.

Knowing that a smart contract is based on a specific interface makes it easy:

  • understand what the contract can do, and
  • know how to interact with the contract.

It removes the need of understanding the complexity behind the actual functions themselves. In a nutshell, remember that:

Interface contracts do not focus on process or behaviour (how a function does a specific job). > But instead, focus on “what the contract can do”.

Example for a Counter Interface ICounter.sol

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.7;

interface ICounter {
    function decrement() external;
    function increment() external;
}
Enter fullscreen mode Exit fullscreen mode

and the main contract Counter.sol

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.13;

import "./ICounter.sol";

contract Counter is ICounter {
    uint private count;

    function increment() public virtual override {
        count += 1;
    }

    function decrement();() public virtual override {
        count -= 1;
    }
}
Enter fullscreen mode Exit fullscreen mode

If the contract does not implement all functions (in other words, if some functions implementations are missing), it can only be used as an abstract contract and cannot be deployed directly.

The other contract that will derive from it will have to implement the functions missing the implementations.

All functions declared inside an interface are implicitly virtual . They are virtual by default, even if they do not include the virtual keyword in their definition inside the interface.

Any function that overrides them (= the actual function implementation) do not need the override keyword. This specificity was introduced in Solidity 0.8.8, as the Solidity docs state:

Starting from Solidity 0.8.8, the override keyword is not required when overriding an interface function…

However, this is not always the case. Exceptions exist.

…except for the case where the function is defined in multiple bases.

Another rule about these function implementations is that “it is not because a function overrides the function defined in an interface that it can automatically be overridden again in a future inherited contract”. This is only possible when the overriding function is marked virtual in turn.

Another misconception is to define functions definitions only for functions that are state changing. ❓

Functions part of an interface can also be view functions. In fac, the highly popular IERC20 interface defines two view functions: totalSupply(...) and balanceOf(...).


Top comments (1)

Collapse
 
anarepp profile image
AnaRepp

This post abstract contract is one which contains at least one function without any implementation. Hire someone to break up a couple