不要將任何敏感數(shù)據(jù)存放在合約中谋竖,因?yàn)楹霞s中的任何數(shù)據(jù)都可被讀取宰译,包括private 定義私有數(shù)據(jù)不皆。
在 solidity 中,有四種可見(jiàn)性關(guān)鍵字:external加缘,public,internal 和 private。默認(rèn)時(shí)函數(shù)可見(jiàn)性為 public。對(duì)狀態(tài)變量而言绅你,除了不能用 external 來(lái)定義,其它三個(gè)都可以來(lái)定義變量,狀態(tài)變量默認(rèn)的可見(jiàn)性為 internal。
- external 關(guān)鍵字
external 定義的外部函數(shù)可以被其它合約調(diào)用蛛芥。用 external 修飾的外部函數(shù) function() 不能作為內(nèi)部函數(shù)直接調(diào)用漓糙,也就是說(shuō) function() 的調(diào)用方式必須用 this.function() 。
- public 關(guān)鍵字
public 定義的函數(shù)可以被內(nèi)部函數(shù)或外部消息調(diào)用壮韭。對(duì)用 public 定義的狀態(tài)變量屯曹,系統(tǒng)會(huì)自動(dòng)生成一個(gè) getter 函數(shù)偷俭。
- internal 用關(guān)鍵字
internal 定義的函數(shù)和狀態(tài)變量只能在(當(dāng)前合約或當(dāng)前合約派生的合約)內(nèi)部進(jìn)行訪問(wèn)形葬。
- private 關(guān)鍵字
private 定義的函數(shù)和狀態(tài)變量只對(duì)定義它的合約可見(jiàn)猖腕,該合約派生的合約都不能調(diào)用和訪問(wèn)該函數(shù)及狀態(tài)變量。
綜上可知,合約中修飾變量存儲(chǔ)的關(guān)鍵字僅僅限制了其調(diào)用的范圍娇唯,并沒(méi)有限制其是否可讀留美。所以我們今天就來(lái)帶大家了解如何讀取合約中的所有數(shù)據(jù)。
solidity 中的三種數(shù)據(jù)存儲(chǔ)方式:
- storage(存儲(chǔ))
storage 中的數(shù)據(jù)被永久存儲(chǔ)妆毕。其以鍵值對(duì)的形式存儲(chǔ)在 slot 插槽中贮尖。
storage 中的數(shù)據(jù)會(huì)被寫(xiě)在區(qū)塊鏈中(因此它們會(huì)更改狀態(tài))笛粘,這就是為什么使用存儲(chǔ)非常昂貴的原因。
storage 共有 2^256 個(gè)插槽湿硝,每個(gè)插槽 32 個(gè)字節(jié)數(shù)據(jù)按聲明順序依次存儲(chǔ)薪前,數(shù)據(jù)將會(huì)從每個(gè)插槽的右邊開(kāi)始存儲(chǔ),如果相鄰變量適合單個(gè) 32 字節(jié)关斜,然后它們被打包到同一個(gè)插槽中否則將會(huì)啟用新的插槽來(lái)存儲(chǔ)示括。
storage 中的數(shù)組的存儲(chǔ)方式就比較獨(dú)特了,首先痢畜,solidity 中的數(shù)組分為兩種:
a.定長(zhǎng)數(shù)組(長(zhǎng)度固定):
定長(zhǎng)數(shù)組中的每個(gè)元素都會(huì)有一個(gè)獨(dú)立的插槽來(lái)存儲(chǔ)垛膝。以一個(gè)含有三個(gè) uint64 元素的定長(zhǎng)數(shù)組為例鳍侣,下圖可以清楚的看出其存儲(chǔ)方式:
b.變長(zhǎng)數(shù)組(長(zhǎng)度隨元素的數(shù)量而改變):
變長(zhǎng)數(shù)組的存儲(chǔ)方式就很奇特,在遇到變長(zhǎng)數(shù)組時(shí)吼拥,會(huì)先啟用一個(gè)新的插槽 slotA 用來(lái)存儲(chǔ)數(shù)組的長(zhǎng)度倚聚,其數(shù)據(jù)存儲(chǔ)在另外的編號(hào)為 slotV 的插槽中。slotA 表示變長(zhǎng)數(shù)組聲明的位置扔罪,用 length 表示變長(zhǎng)數(shù)組的長(zhǎng)度秉沼,用 slotV 表示變長(zhǎng)數(shù)組數(shù)據(jù)存儲(chǔ)的位置,用 value 表示變長(zhǎng)數(shù)組某個(gè)數(shù)據(jù)的值矿酵,用 index 表示 value 對(duì)應(yīng)的索引下標(biāo)唬复。
memory(內(nèi)存)
memory 是一個(gè)字節(jié)數(shù)組,其插槽大小為 256 位(32 個(gè)字節(jié))全肮。數(shù)據(jù)僅在函數(shù)執(zhí)行期間存儲(chǔ)敞咧,執(zhí)行完之后,將會(huì)被刪除辜腺。它們不會(huì)保存到區(qū)塊鏈中休建。
讀或?qū)懸粋€(gè)字節(jié)(256 位)需要 3 gas 。
為了避免給礦工帶來(lái)太多工作评疗,在進(jìn)行 22 次讀寫(xiě)操作后测砂,之后的讀寫(xiě)成本開(kāi)始上升。calldata(調(diào)用數(shù)據(jù))
calldata 是一個(gè)不可修改的百匆,非持久性的區(qū)域砌些,用于存儲(chǔ)函數(shù)參數(shù),并且其行為基本上類(lèi)似于 memory加匈。
調(diào)用外部函數(shù)的參數(shù)需要 calldata存璃,也可用于其他變量。
它避免了復(fù)制雕拼,并確保了數(shù)據(jù)不能被修改纵东。
帶有 calldata 數(shù)據(jù)位置的數(shù)組和結(jié)構(gòu)體也可以從函數(shù)中返回,但是不可以為這種類(lèi)型賦值啥寇。
由合約中可以看到 slot0 中只存儲(chǔ)了一個(gè) uint 類(lèi)型的數(shù)據(jù)偎球,我們讀取出來(lái)看一下: