溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Solidity錯(cuò)誤處理是什么

發(fā)布時(shí)間:2021-12-07 15:41:25 來源:億速云 閱讀:136 作者:iii 欄目:互聯(lián)網(wǎng)科技

本篇內(nèi)容介紹了“Solidity錯(cuò)誤處理是什么”的有關(guān)知識(shí),在實(shí)際案例的操作過程中,不少人都會(huì)遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

什么是錯(cuò)誤處理

錯(cuò)誤處理是指在程序發(fā)生錯(cuò)誤時(shí)的處理方式,Solidity處理錯(cuò)誤和我們常見的語言不一樣,Solidity是通過回退狀態(tài)的方式來處理錯(cuò)誤。發(fā)生異常時(shí)會(huì)撤消當(dāng)前調(diào)用(及其所有子調(diào)用)所改變的狀態(tài),同時(shí)給調(diào)用者返回一個(gè)錯(cuò)誤標(biāo)識(shí)。注意捕捉異常是不可能的,因此沒有try ... catch...。

為什么Solidity處理錯(cuò)誤要這樣設(shè)計(jì)呢? 我們可以把區(qū)塊鏈理解為是全球共享的分布式事務(wù)性數(shù)據(jù)庫(kù)。全球共享意味著參與這個(gè)網(wǎng)絡(luò)的每一個(gè)人都可以讀寫其中的記錄。如果想修改這個(gè)數(shù)據(jù)庫(kù)中的內(nèi)容,就必須創(chuàng)建一個(gè)事務(wù),事務(wù)意味著要做的修改(假如我們想同時(shí)修改兩個(gè)值)只能被完全的應(yīng)用或者一點(diǎn)都沒有進(jìn)行。 學(xué)習(xí)過數(shù)據(jù)庫(kù)的同學(xué),應(yīng)該理解事務(wù)的含義,如果你對(duì)事務(wù)一詞不是很理解,建議你搜索一下“數(shù)據(jù)庫(kù)事務(wù)“。 Solidity錯(cuò)誤處理就是要保證每次調(diào)用都是事務(wù)性的。

如何處理

Solidity提供了兩個(gè)函數(shù)assert和require來進(jìn)行條件檢查,如果條件不滿足則拋出異常。assert函數(shù)通常用來檢查(測(cè)試)內(nèi)部錯(cuò)誤,而require函數(shù)來檢查輸入變量或合同狀態(tài)變量是否滿足條件以及驗(yàn)證調(diào)用外部合約返回值。 另外,如果我們正確使用assert,有一個(gè)Solidity分析工具就可以幫我們分析出智能合約中的錯(cuò)誤,幫助我們發(fā)現(xiàn)合約中有邏輯錯(cuò)誤的bug。

除了可以兩個(gè)函數(shù)assert和require來進(jìn)行條件檢查,另外還有兩種方式來觸發(fā)異常:

  1. revert函數(shù)可以用來標(biāo)記錯(cuò)誤并回退當(dāng)前調(diào)用

  2. 使用throw關(guān)鍵字拋出異常(從0.4.13版本,throw關(guān)鍵字已被棄用,將來會(huì)被淘汰。)

當(dāng)子調(diào)用中發(fā)生異常時(shí),異常會(huì)自動(dòng)向上“冒泡”。 不過也有一些例外:send,和底層的函數(shù)調(diào)用call, delegatecall,callcode,當(dāng)發(fā)生異常時(shí),這些函數(shù)返回false。

注意:在一個(gè)不存在的地址上調(diào)用底層的函數(shù)call,delegatecall,callcode 也會(huì)返回成功,所以我們?cè)谶M(jìn)行調(diào)用時(shí),應(yīng)該總是優(yōu)先進(jìn)行函數(shù)存在性檢查。

在下面通過一個(gè)示例來說明如何使用require來檢查輸入條件,以及assert用于內(nèi)部錯(cuò)誤檢查:

pragma solidity ^0.4.0;

contract Sharer {
    function sendHalf(address addr) public payable returns (uint balance) {
        require(msg.value % 2 == 0); // 僅允許偶數(shù)
        uint balanceBeforeTransfer = this.balance;
        addr.transfer(msg.value / 2);  // 如果失敗,會(huì)拋出異常,下面的代碼就不是執(zhí)行
        assert(this.balance == balanceBeforeTransfer - msg.value / 2);
        return this.balance;
    }
}

我們實(shí)際運(yùn)行下,看看異常是如何發(fā)生的:

  1. 首先打開Remix,貼入代碼,點(diǎn)擊創(chuàng)建合約。如下圖: Solidity錯(cuò)誤處理是什么

  2. 運(yùn)行測(cè)試1:附加1wei (奇數(shù))去調(diào)用sendHalf,這時(shí)會(huì)發(fā)生異常,如下圖:

Solidity錯(cuò)誤處理是什么

  1. 運(yùn)行測(cè)試2:附加2wei 去調(diào)用sendHalf,運(yùn)行正常。

  2. 運(yùn)行測(cè)試3:附加2wei以及sendHalf參數(shù)為當(dāng)前合約本身,在轉(zhuǎn)賬是發(fā)生異常,因?yàn)楹霞s無法接收轉(zhuǎn)賬,錯(cuò)誤提示上圖類似。

assert類型異常

在下述場(chǎng)景中自動(dòng)產(chǎn)生assert類型的異常:

  1. 如果越界,或負(fù)的序號(hào)值訪問數(shù)組,如i >= x.length 或 i < 0時(shí)訪問x[i]

  2. 如果序號(hào)越界,或負(fù)的序號(hào)值時(shí)訪問一個(gè)定長(zhǎng)的bytesN。

  3. 被除數(shù)為0, 如5/0 或 23 % 0。

  4. 對(duì)一個(gè)二進(jìn)制移動(dòng)一個(gè)負(fù)的值。如:5<<i; i為-1時(shí)。

  5. 整數(shù)進(jìn)行可以顯式轉(zhuǎn)換為枚舉時(shí),如果將過大值,負(fù)值轉(zhuǎn)為枚舉類型則拋出異常

  6. 如果調(diào)用未初始化內(nèi)部函數(shù)類型的變量。

  7. 如果調(diào)用assert的參數(shù)為false

require類型異常

在下述場(chǎng)景中自動(dòng)產(chǎn)生require類型的異常:

  1. 調(diào)用throw

  2. 如果調(diào)用require的參數(shù)為false

  3. 如果你通過消息調(diào)用一個(gè)函數(shù),但在調(diào)用的過程中,并沒有正確結(jié)束(gas不足,沒有匹配到對(duì)應(yīng)的函數(shù),或被調(diào)用的函數(shù)出現(xiàn)異常)。底層操作如call,send,delegatecall或callcode除外,它們不會(huì)拋出異常,但它們會(huì)通過返回false來表示失敗。

  4. 如果在使用new創(chuàng)建一個(gè)新合約時(shí)出現(xiàn)第3條的原因沒有正常完成。

  5. 如果調(diào)用外部函數(shù)調(diào)用時(shí),被調(diào)用的對(duì)象不包含代碼。

  6. 如果合約沒有payable修飾符的public的函數(shù)在接收以太幣時(shí)(包括構(gòu)造函數(shù),和回退函數(shù))。

  7. 如果合約通過一個(gè)public的getter函數(shù)(public getter funciton)接收以太幣。

  8. 如果**.transfer()**執(zhí)行失敗

當(dāng)發(fā)生require類型的異常時(shí),Solidity會(huì)執(zhí)行一個(gè)回退操作(指令0xfd)。 當(dāng)發(fā)生assert類型的異常時(shí),Solidity會(huì)執(zhí)行一個(gè)無效操作(指令0xfe)。 在上述的兩種情況下,EVM都會(huì)撤回所有的狀態(tài)改變。是因?yàn)槠谕慕Y(jié)果沒有發(fā)生,就沒法繼續(xù)安全執(zhí)行。必須保證交易的原子性(一致性,要么全部執(zhí)行,要么一點(diǎn)改變都沒有,不能只改變一部分),所以需要撤銷所有操作,讓整個(gè)交易沒有任何影響。

注意assert類型的異常會(huì)消耗掉所有的gas, 而require從大都會(huì)版本(Metropolis, 即目前主網(wǎng)所在的版本)起不會(huì)消耗gas。

“Solidity錯(cuò)誤處理是什么”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI