以太坊C++源码解析(五)区块链同步(2)

2018-11-21  本文已影响50人  sky2016

区块链同步的核心类是BlockChainSync,在继续深入了解同步流程之前,我们还是先来了解一下这个类有哪些重要成员吧。

其中m_downloadingHeaders记录当前正在同步的块头对应的块号;m_downloadingBodies记录单曲正在同步的块体对应的块号;m_headerSyncPeersm_downloadingHeaders的基础上还记录了peer信息;m_bodySyncPeersm_downloadingBodies的基础上记录了peer信息。

注意到这里有个模板std::owner_less<>,这个模板是用来表明如何对std::weak_ptr<EthereumPeer>进行排序的,这里有一个owner-basevalue-base的概念,在一般情况下owner-basevalue-base是相同的,但是在std::shared_ptrstd::weak_ptr使用
aliasing constructor(别名构造函数)时这两者不同,需要区分。
推荐两篇文件,讲得比较详细:
C++ Memory Library - owner_less
What is shared_ptr's aliasing constructor for?

那么同步中记录这四个值有什么用呢?在这里是用来做校验的。
因为BlockChainSync类从HasInvariants类继承而来,因此继承了一个接口:

    virtual bool invariants() const = 0;

可以在BlockChainSync::invariants()中找到答案:

    bool BlockChainSync::invariants() const
    {
        if (!isSyncing() && !m_headers.empty())
            BOOST_THROW_EXCEPTION(FailedInvariant() << errinfo_comment("Got headers while not syncing"));
        if (!isSyncing() && !m_bodies.empty())
            BOOST_THROW_EXCEPTION(FailedInvariant() << errinfo_comment("Got bodies while not syncing"));
        if (isSyncing() && m_host.chain().number() > 0 && m_haveCommonHeader && m_lastImportedBlock == 0)
            BOOST_THROW_EXCEPTION(FailedInvariant() << errinfo_comment("Common block not found"));
        if (isSyncing() && !m_headers.empty() &&  m_lastImportedBlock >= m_headers.begin()->first)
            BOOST_THROW_EXCEPTION(FailedInvariant() << errinfo_comment("Header is too old"));
        if (m_headerSyncPeers.empty() != m_downloadingHeaders.empty())
            BOOST_THROW_EXCEPTION(FailedInvariant() << errinfo_comment("Header download map mismatch"));
        if (m_bodySyncPeers.empty() != m_downloadingBodies.empty() && m_downloadingBodies.size() <= m_headerIdToNumber.size())
            BOOST_THROW_EXCEPTION(FailedInvariant() << errinfo_comment("Body download map mismatch"));
        return true;
    }

那么在哪里调用这个函数呢?在InvariantChecker::checkInvariants()函数里:

    void InvariantChecker::checkInvariants(HasInvariants const* _this, char const* _fn, char const* _file, int _line, bool _pre)
    {
        if (!_this->invariants())
        {
            cwarn << (_pre ? "Pre" : "Post") << "invariant failed in" << _fn << "at" << _file << ":" << _line;
            ::boost::exception_detail::throw_exception_(FailedInvariant(), _fn, _file, _line);
        }
    }

而这个函数被定义成了两个宏:

    #if ETH_DEBUG
    #define DEV_INVARIANT_CHECK ::dev::InvariantChecker __dev_invariantCheck(this, BOOST_CURRENT_FUNCTION, __FILE__, __LINE__)
    #define DEV_INVARIANT_CHECK_HERE ::dev::InvariantChecker::checkInvariants(this, BOOST_CURRENT_FUNCTION, __FILE__, __LINE__, true)
    #else
    #define DEV_INVARIANT_CHECK (void)0;
    #define DEV_INVARIANT_CHECK_HERE (void)0;
    #endif

DEV_INVARIANT_CHECKDEV_INVARIANT_CHECK_HERE这两个宏在代码中多次调用,有兴趣可以去看看源码。

除了BlockChainSync类之外,还有一个重要类BlockQueue类也是从HasInvariants类继承而来,因而也具有check的能力。

这里使用了std::map,key表示m_headersm_bodies中连续段最低块的块号,value表示m_headersm_bodies中存在的数据。每个std::vector中保存一个连续段的数据。以m_headers为例,假如有块5,6,7,10,13,15,16,那么在m_headers中存储为:{{5, {块5,块6,块7}}, {10, {块10}}, {13, {块13}}, {15, {块15,块16}}},其中5,6,7是一个连续段,存为一个pair,key为5,value为std::vector<Header>{块5,块6,块7},块10为一个单独不连续块,那么存为一个pair,key为10,value为std::vector<Header>{块10},以此类推,m_bodies也是一样。为了在这种数据结构中方便查找,插入和删除数据,还专门定义了专有方法,比如haveItem()findItem()removeItem()removeAllStartingWith()mergeInto(),有兴趣可以自己看下,能更深入理解这种数据结构的操作。
m_headers 和 m_bodies构成了整个区块链同步的第一级缓存,存放了刚下载下来未经校验的分离的区块头和区块体。整个区块链同步的数据流程大致如下:

区块链同步模型
上一篇 下一篇

猜你喜欢

热点阅读