Solidity - คู่มือฉบับย่อ
Solidity เป็นภาษาการเขียนโปรแกรมระดับสูงที่มุ่งเน้นตามสัญญาสำหรับการใช้งานสัญญาอัจฉริยะ Solidity ได้รับอิทธิพลอย่างมากจาก C ++, Python และ JavaScript และได้รับการออกแบบมาเพื่อกำหนดเป้าหมายไปที่ Ethereum Virtual Machine (EVM)
Solidity ถูกพิมพ์แบบคงที่รองรับการสืบทอดไลบรารีและภาษาการเขียนโปรแกรมประเภทที่ผู้ใช้กำหนดเองที่ซับซ้อน
คุณสามารถใช้ Solidity เพื่อสร้างสัญญาสำหรับการใช้งานเช่นการลงคะแนนการระดมทุนการประมูลแบบตาบอดและกระเป๋าเงินหลายลายเซ็น
Ethereum คืออะไร?
Ethereum คือการกระจายอำนาจเช่น แพลตฟอร์มบล็อกเชนที่รันสัญญาอัจฉริยะเช่นแอปพลิเคชันที่ทำงานตรงตามโปรแกรมโดยไม่มีความเป็นไปได้ของการหยุดทำงานการเซ็นเซอร์การฉ้อโกงหรือการแทรกแซงจากบุคคลที่สาม
เครื่องเสมือน Ethereum (EVM)
Ethereum Virtual Machine หรือที่เรียกว่า EVM เป็นสภาพแวดล้อมรันไทม์สำหรับสัญญาอัจฉริยะใน Ethereum Ethereum Virtual Machine มุ่งเน้นไปที่การรักษาความปลอดภัยและการรันโค้ดที่ไม่น่าเชื่อถือโดยคอมพิวเตอร์ทั่วโลก
EVM มีความเชี่ยวชาญในการป้องกันการโจมตีแบบปฏิเสธการให้บริการและทำให้แน่ใจว่าโปรแกรมต่างๆไม่สามารถเข้าถึงสถานะของกันและกันได้ทำให้มั่นใจได้ว่าจะสามารถสื่อสารได้โดยปราศจากการรบกวนที่อาจเกิดขึ้น
Ethereum Virtual Machine ได้รับการออกแบบมาเพื่อใช้เป็นสภาพแวดล้อมรันไทม์สำหรับสัญญาอัจฉริยะบนพื้นฐานของ Ethereum
Smart Contract คืออะไร?
สัญญาอัจฉริยะคือโปรโตคอลคอมพิวเตอร์ที่มีวัตถุประสงค์เพื่ออำนวยความสะดวกตรวจสอบหรือบังคับใช้การเจรจาหรือการปฏิบัติตามสัญญาแบบดิจิทัล สัญญาอัจฉริยะช่วยให้สามารถทำธุรกรรมได้อย่างน่าเชื่อถือโดยไม่มีบุคคลที่สาม ธุรกรรมเหล่านี้สามารถติดตามได้และไม่สามารถย้อนกลับได้
แนวคิดของสัญญาอัจฉริยะถูกเสนอครั้งแรกโดย Nick Szabo ในปี 1994 Szabo เป็นนักวิชาการด้านกฎหมายและนักเข้ารหัสที่รู้จักกันดีในการวางรากฐานสำหรับสกุลเงินดิจิทัล
เป็นเรื่องปกติถ้าคุณไม่เข้าใจ Smart Contract ในตอนนี้เราจะลงรายละเอียดเพิ่มเติมในภายหลัง
บทนี้อธิบายถึงวิธีการตั้งค่าคอมไพเลอร์ Solidity บนเครื่อง CentOS หากคุณไม่มีเครื่อง Linux คุณสามารถใช้ Online Compiler ของเราสำหรับสัญญาขนาดเล็กและเพื่อการเรียนรู้ Solidity อย่างรวดเร็ว
วิธีที่ 1 - npm / Node.js
นี่เป็นวิธีที่เร็วที่สุดในการติดตั้งคอมไพเลอร์ Solidity บนเครื่อง CentoS ของคุณ เรามีขั้นตอนต่อไปนี้เพื่อติดตั้ง Solidity Compiler -
ติดตั้ง Node.js
ขั้นแรกตรวจสอบให้แน่ใจว่าคุณมี node.js บนเครื่อง CentOS ของคุณ หากไม่มีให้ติดตั้งโดยใช้คำสั่งต่อไปนี้ -
# First install epel-release
$sudo yum install epel-release
# Now install nodejs
$sudo yum install nodejs
# Next install npm (Nodejs Package Manager )
$sudo yum install npm
# Finally verify installation
$npm --version
หากทุกอย่างได้รับการติดตั้งแล้วคุณจะเห็นผลลัพธ์ดังนี้ -
3.10.10
ติดตั้ง solc
เมื่อคุณติดตั้งตัวจัดการแพ็คเกจ Node.js แล้วคุณสามารถดำเนินการติดตั้งคอมไพเลอร์ Solidity ได้ดังนี้ -
$sudonpm install -g solc
คำสั่งดังกล่าวจะติดตั้งโปรแกรม solcjs และจะทำให้สามารถใช้งานได้ทั่วโลกผ่านระบบ ตอนนี้คุณสามารถทดสอบคอมไพเลอร์ Solidity ของคุณได้โดยออกคำสั่งต่อไปนี้ -
$solcjs-version
หากทุกอย่างเป็นไปด้วยดีสิ่งนี้จะพิมพ์ข้อความดังนี้ -
0.5.2+commit.1df8f40c.Emscripten.clang
ตอนนี้คุณพร้อมที่จะใช้ solcjs ซึ่งมีคุณสมบัติน้อยกว่าคอมไพเลอร์ Solidity มาตรฐาน แต่จะทำให้คุณเป็นจุดเริ่มต้นที่ดี
วิธีที่ 2 - Docker Image
คุณสามารถดึงอิมเมจ Docker และเริ่มใช้เพื่อเริ่มการเขียนโปรแกรม Solidity ต่อไปนี้เป็นขั้นตอนง่ายๆ คำสั่งต่อไปนี้คือการดึง Solidity Docker Image
$docker pull ethereum/solc:stable
เมื่อดาวน์โหลดอิมเมจนักเทียบท่าแล้วเราสามารถตรวจสอบได้โดยใช้คำสั่งต่อไปนี้
$docker run ethereum/solc:stable-version
สิ่งนี้จะพิมพ์บางสิ่งดังนี้ -
$ docker run ethereum/solc:stable -version
solc, the solidity compiler commandlineinterfaceVersion: 0.5.2+commit.1df8f40c.Linux.g++
วิธีที่ 3: การติดตั้งแพ็คเกจไบนารี
หากคุณยินดีที่จะติดตั้งคอมไพเลอร์เต็มรูปแบบบนเครื่อง Linux ของคุณโปรดตรวจสอบเว็บไซต์อย่างเป็นทางการการติดตั้ง Solidity Compiler
ไฟล์ซอร์ส Solidity สามารถมีคำจำกัดความสัญญาคำสั่งการนำเข้าและคำสั่ง pragma จำนวนเท่าใดก็ได้
เริ่มต้นด้วยซอร์สไฟล์อย่างง่ายของ Solidity ต่อไปนี้เป็นตัวอย่างของไฟล์ Solidity -
pragma solidity >=0.4.0 <0.6.0;
contract SimpleStorage {
uint storedData;
function set(uint x) public {
storedData = x;
}
function get() public view returns (uint) {
return storedData;
}
}
Pragma
บรรทัดแรกคือคำสั่ง pragma ซึ่งบอกว่าซอร์สโค้ดเขียนขึ้นสำหรับ Solidity เวอร์ชัน 0.4.0 หรืออะไรที่ใหม่กว่าที่ไม่ทำลายฟังก์ชันการทำงาน แต่ไม่รวมถึงเวอร์ชัน 0.6.0
คำสั่ง pragma อยู่ในเครื่องของไฟล์ต้นฉบับเสมอและหากคุณนำเข้าไฟล์อื่น pragma จากไฟล์นั้นจะไม่ใช้กับไฟล์ที่นำเข้าโดยอัตโนมัติ
ดังนั้น pragma สำหรับไฟล์ที่จะไม่คอมไพล์ก่อนหน้าเวอร์ชัน 0.4.0 และจะไม่ทำงานบนคอมไพเลอร์ที่เริ่มจากเวอร์ชัน 0.5.0 จะเขียนดังนี้ -
pragma solidity ^0.4.0;
นี่คือเงื่อนไขที่สองถูกเพิ่มโดยใช้ ^
สัญญา
สัญญา Solidity คือชุดของรหัส (ฟังก์ชัน) และข้อมูล (สถานะของมัน) ที่อยู่ตามที่อยู่เฉพาะบน Ethereumblockchain
บรรทัด uintstoredData ประกาศตัวแปรสถานะที่เรียกว่าorageDataของประเภท uint และฟังก์ชันที่ตั้งค่าและรับสามารถใช้เพื่อแก้ไขหรือดึงค่าของตัวแปร
การนำเข้าไฟล์
แม้ว่าตัวอย่างข้างต้นจะไม่มีคำสั่งนำเข้า แต่ Solidity สนับสนุนคำสั่งนำเข้าที่คล้ายกับที่มีอยู่ใน JavaScript
คำสั่งต่อไปนี้นำเข้าสัญลักษณ์ส่วนกลางทั้งหมดจาก "filename"
import "filename";
ตัวอย่างต่อไปนี้สร้างสัญลักษณ์สากลใหม่ symbolName ซึ่งสมาชิกเป็นสัญลักษณ์ส่วนกลางทั้งหมดจาก "ชื่อไฟล์"
import * as symbolName from "filename";
ในการนำเข้าไฟล์ x จากไดเร็กทอรีเดียวกันกับไฟล์ปัจจุบันให้ใช้ import "./x" เป็น x; หากคุณใช้การนำเข้า "x" เป็น x; แต่อาจมีการอ้างอิงไฟล์อื่นใน "รวมไดเรกทอรี" ทั่วโลก
คำหลักที่สงวนไว้
ต่อไปนี้เป็นคำหลักที่สงวนไว้ใน Solidity -
บทคัดย่อ | หลังจาก | นามแฝง | สมัคร |
อัตโนมัติ | กรณี | จับ | สำเนา |
ค่าเริ่มต้น | กำหนด | สุดท้าย | ไม่เปลี่ยนรูป |
ดำเนินการ | ใน | อินไลน์ | ปล่อย |
มาโคร | การแข่งขัน | ไม่แน่นอน | โมฆะ |
ของ | แทนที่ | บางส่วน | สัญญา |
เอกสารอ้างอิง | เคลื่อนย้ายได้ | ปิดผนึก | ขนาดของ |
คงที่ | รองรับ | สวิตซ์ | ลอง |
typedef | ประเภทของ | ไม่เลือก |
เรากำลังใช้Remix IDEเพื่อคอมไพล์และรันฐาน Solidity Code ของเรา
Step 1 - คัดลอกรหัสที่กำหนดในส่วนรหัส Remix IDE
ตัวอย่าง
pragma solidity ^0.5.0;
contract SolidityTest {
constructor() public{
}
function getResult() public view returns(uint){
uint a = 1;
uint b = 2;
uint result = a + b;
return result;
}
}
Step 2 - ภายใต้แท็บคอมไพล์คลิก Start to Compile ปุ่ม.
Step 3 - ภายใต้ Run Tab คลิก Deploy ปุ่ม.
Step 4 - ภายใต้แท็บเรียกใช้เลือก SolidityTest at 0x... ในเมนูแบบเลื่อนลง
Step 5 - คลิก getResult ปุ่มเพื่อแสดงผลลัพธ์
เอาต์พุต
0: uint256: 3
Solidity รองรับทั้งความคิดเห็นสไตล์ C และ C ++ ด้วยประการฉะนี้ -
ข้อความใด ๆ ระหว่าง a // และท้ายบรรทัดจะถือว่าเป็นข้อคิดเห็นและ Solidity Compiler จะถูกละเว้น
ข้อความใด ๆ ระหว่างอักขระ / * และ * / จะถือว่าเป็นความคิดเห็น ซึ่งอาจครอบคลุมหลายบรรทัด
ตัวอย่าง
ตัวอย่างต่อไปนี้แสดงวิธีใช้ข้อคิดเห็นใน Solidity
function getResult() public view returns(uint){
// This is a comment. It is similar to comments in C++
/*
* This is a multi-line comment in solidity
* It is very similar to comments in C Programming
*/
uint a = 1;
uint b = 2;
uint result = a + b;
return result;
}
ในขณะที่เขียนโปรแกรมในภาษาใด ๆ คุณจำเป็นต้องใช้ตัวแปรต่างๆเพื่อจัดเก็บข้อมูลต่างๆ ตัวแปรเป็นเพียงตำแหน่งหน่วยความจำที่สงวนไว้เพื่อเก็บค่า ซึ่งหมายความว่าเมื่อคุณสร้างตัวแปรคุณจะสงวนพื้นที่ในหน่วยความจำไว้
คุณอาจต้องการจัดเก็บข้อมูลประเภทข้อมูลต่างๆเช่นอักขระอักขระแบบกว้างจำนวนเต็มจุดลอยตัวจุดลอยตัวคู่บูลีนเป็นต้นตามประเภทข้อมูลของตัวแปรระบบปฏิบัติการจะจัดสรรหน่วยความจำและตัดสินใจว่าจะจัดเก็บอะไรใน หน่วยความจำที่สงวนไว้
ประเภทมูลค่า
Solidity ช่วยให้โปรแกรมเมอร์มีประเภทข้อมูลในตัวและประเภทข้อมูลที่กำหนดโดยผู้ใช้ ตารางต่อไปนี้แสดงประเภทข้อมูล C ++ พื้นฐานเจ็ดประเภท -
ประเภท | คำสำคัญ | ค่า |
---|---|---|
บูลีน | บูล | ถูกผิด |
จำนวนเต็ม | int / uint | จำนวนเต็มที่ลงชื่อและไม่ได้ลงชื่อซึ่งมีขนาดต่างกัน |
จำนวนเต็ม | int8 ถึง int256 | ลงนาม int จาก 8 บิตเป็น 256 บิต int256 เหมือนกับ int |
จำนวนเต็ม | uint8 ถึง uint256 | int ที่ไม่ได้ลงนามจาก 8 บิตเป็น 256 บิต uint256 เหมือนกับ uint |
หมายเลขจุดคงที่ | คงที่ / ไม่ได้รับการแก้ไข | หมายเลขจุดคงที่ที่ลงชื่อและไม่ได้ลงชื่อซึ่งมีขนาดแตกต่างกัน |
หมายเลขจุดคงที่ | คงที่ / ไม่ได้รับการแก้ไข | หมายเลขจุดคงที่ที่ลงชื่อและไม่ได้ลงชื่อซึ่งมีขนาดแตกต่างกัน |
หมายเลขจุดคงที่ | คงที่ | หมายเลขจุดคงที่ที่ลงนามโดยที่ M แทนจำนวนบิตที่นำมาตามประเภทและ N แทนจุดทศนิยม M ควรหารด้วย 8 และไปจาก 8 เป็น 256 N สามารถเป็น 0 ถึง 80 คงที่จะเหมือนกับ fixed128x18 |
หมายเลขจุดคงที่ | ufixedMxN | หมายเลขจุดคงที่ที่ไม่ได้ลงชื่อโดยที่ M แทนจำนวนบิตที่นำมาตามประเภทและ N แทนจุดทศนิยม M ควรหารด้วย 8 และไปจาก 8 เป็น 256 N สามารถเป็น 0 ถึง 80 ufixed เหมือนกับ ufixed128x18 |
ที่อยู่
แอดเดรสเก็บค่า 20 ไบต์แทนขนาดของแอดเดรส Ethereum สามารถใช้ที่อยู่เพื่อรับยอดคงเหลือโดยใช้วิธีการ. balance และสามารถใช้เพื่อโอนยอดคงเหลือไปยังที่อยู่อื่นโดยใช้วิธีการโอน
address x = 0x212;
address myAddress = this;
if (x.balance < 10 && myAddress.balance >= 10) x.transfer(10);
Solidity รองรับตัวแปรสามประเภท
State Variables - ตัวแปรที่มีการจัดเก็บค่าอย่างถาวรในที่เก็บสัญญา
Local Variables - ตัวแปรที่มีค่าอยู่จนถึงฟังก์ชันกำลังดำเนินการ
Global Variables - ตัวแปรพิเศษมีอยู่ในเนมสเปซทั่วโลกที่ใช้รับข้อมูลเกี่ยวกับบล็อกเชน
Solidity เป็นภาษาที่พิมพ์แบบคงที่ซึ่งหมายความว่าต้องระบุสถานะหรือประเภทตัวแปรโลคัลระหว่างการประกาศ ตัวแปรที่ประกาศแต่ละตัวจะมีค่าเริ่มต้นตามประเภทเสมอ ไม่มีแนวคิด "ไม่ได้กำหนด" หรือ "ว่าง"
ตัวแปรสถานะ
ตัวแปรที่มีค่าถูกเก็บไว้อย่างถาวรในที่เก็บสัญญา
pragma solidity ^0.5.0;
contract SolidityTest {
uint storedData; // State variable
constructor() public {
storedData = 10; // Using State variable
}
}
ตัวแปรท้องถิ่น
ตัวแปรที่มีค่าพร้อมใช้งานภายในฟังก์ชันที่กำหนดไว้เท่านั้น พารามิเตอร์ของฟังก์ชันจะอยู่ในพื้นที่ของฟังก์ชันนั้นเสมอ
pragma solidity ^0.5.0;
contract SolidityTest {
uint storedData; // State variable
constructor() public {
storedData = 10;
}
function getResult() public view returns(uint){
uint a = 1; // local variable
uint b = 2;
uint result = a + b;
return result; //access the local variable
}
}
ตัวอย่าง
pragma solidity ^0.5.0;
contract SolidityTest {
uint storedData; // State variable
constructor() public {
storedData = 10;
}
function getResult() public view returns(uint){
uint a = 1; // local variable
uint b = 2;
uint result = a + b;
return storedData; //access the state variable
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
เอาต์พุต
0: uint256: 10
ตัวแปรส่วนกลาง
ตัวแปรเหล่านี้เป็นตัวแปรพิเศษที่มีอยู่ในพื้นที่ทำงานทั่วโลกและให้ข้อมูลเกี่ยวกับคุณสมบัติบล็อกเชนและธุรกรรม
ชื่อ | ผลตอบแทน |
---|---|
blockhash (uint blockNumber) ส่งกลับ (bytes32) | แฮชของบล็อกที่กำหนด - ใช้ได้เฉพาะกับ 256 บล็อกล่าสุดไม่รวมบล็อกปัจจุบัน |
block.coinbase (ที่อยู่ที่ต้องชำระ) | ที่อยู่ของคนงานเหมืองบล็อกปัจจุบัน |
block.difficulty (uint) | ความยากของบล็อกปัจจุบัน |
block.gaslimit (uint) | แก๊สลิมิตบล็อกปัจจุบัน |
block.number (uint) | หมายเลขบล็อกปัจจุบัน |
block.timestamp (uint) | การประทับเวลาบล็อกปัจจุบันเป็นวินาทีนับตั้งแต่ยุคยูนิกซ์ |
gasleft () ส่งคืน (uint256) | ก๊าซที่เหลืออยู่ |
msg.data (ไบต์ calldata) | กรอก calldata |
msg.sender (ที่อยู่เจ้าหนี้) | ผู้ส่งข้อความ (ผู้โทรปัจจุบัน) |
msg.sig (ไบต์ 4) | สี่ไบต์แรกของ calldata (ตัวระบุฟังก์ชัน) |
msg.value (uint) | จำนวน wei ที่ส่งมาพร้อมข้อความ |
ตอนนี้ (uint) | การประทับเวลาบล็อกปัจจุบัน |
tx.gasprice (uint) | ราคาก๊าซของธุรกรรม |
tx.origin (ที่อยู่ที่ต้องชำระ) | ผู้ส่งธุรกรรม |
ชื่อตัวแปร Solidity
ในขณะที่ตั้งชื่อตัวแปรของคุณใน Solidity โปรดคำนึงถึงกฎต่อไปนี้
คุณไม่ควรใช้คีย์เวิร์ดที่สงวนไว้สำหรับ Solidity เป็นชื่อตัวแปร คำหลักเหล่านี้จะกล่าวถึงในส่วนถัดไป ตัวอย่างเช่นชื่อตัวแปรเบรกหรือบูลีนไม่ถูกต้อง
ชื่อตัวแปร Solidity ไม่ควรขึ้นต้นด้วยตัวเลข (0-9) ต้องขึ้นต้นด้วยตัวอักษรหรืออักขระขีดล่าง ตัวอย่างเช่น 123test เป็นชื่อตัวแปรที่ไม่ถูกต้อง แต่ _123test เป็นชื่อที่ถูกต้อง
ชื่อตัวแปร Solidity คำนึงถึงขนาดตัวพิมพ์ ตัวอย่างเช่นชื่อและชื่อเป็นตัวแปรสองตัวที่แตกต่างกัน
ขอบเขตของตัวแปรในพื้นที่ จำกัด เฉพาะฟังก์ชันที่กำหนดไว้ แต่ตัวแปรสถานะสามารถมีขอบเขตได้สามประเภท
Public- ตัวแปรสาธารณะสามารถเข้าถึงได้ทั้งภายในและผ่านทางข้อความ สำหรับตัวแปร public state ฟังก์ชัน getter อัตโนมัติจะถูกสร้างขึ้น
Internal - ตัวแปรสถานะภายในสามารถเข้าถึงได้ภายในจากสัญญาปัจจุบันหรือสัญญาที่ได้รับจากมันโดยไม่ต้องใช้สิ่งนี้
Private - ตัวแปรของรัฐเอกชนสามารถเข้าถึงได้ภายในจากสัญญาปัจจุบันเท่านั้นที่ไม่ได้กำหนดไว้ในสัญญาที่ได้รับจากมัน
ตัวอย่าง
pragma solidity ^0.5.0;
contract C {
uint public data = 30;
uint internal iData= 10;
function x() public returns (uint) {
data = 3; // internal access
return data;
}
}
contract Caller {
C c = new C();
function f() public view returns (uint) {
return c.data(); //external access
}
}
contract D is C {
function y() public returns (uint) {
iData = 3; // internal access
return iData;
}
function getResult() public view returns(uint){
uint a = 1; // local variable
uint b = 2;
uint result = a + b;
return storedData; //access the state variable
}
}
Operator คืออะไร?
ให้เราใช้สำนวนง่ายๆ 4 + 5 is equal to 9. ที่นี่เรียกว่า 4 และ 5operands และ '+' เรียกว่า operator. Solidity รองรับตัวดำเนินการประเภทต่อไปนี้
- ตัวดำเนินการเลขคณิต
- ตัวดำเนินการเปรียบเทียบ
- ตัวดำเนินการเชิงตรรกะ (หรือเชิงสัมพันธ์)
- ผู้ดำเนินการมอบหมาย
- ตัวดำเนินการตามเงื่อนไข (หรือตามเงื่อนไข)
มาดูตัวดำเนินการทั้งหมดทีละคน
ตัวดำเนินการเลขคณิต
Solidity รองรับตัวดำเนินการเลขคณิตต่อไปนี้ -
สมมติว่าตัวแปร A ถือ 10 และตัวแปร B ถือ 20 จากนั้น -
แสดงตัวอย่าง
ซีเนียร์ | ตัวดำเนินการและคำอธิบาย |
---|---|
1 | + (Addition) เพิ่มสองตัวถูกดำเนินการ Ex: A + B จะให้ 30 |
2 | - (Subtraction) ลบตัวถูกดำเนินการที่สองจากตัวแรก Ex: A - B จะให้ -10 |
3 | * (Multiplication) คูณตัวถูกดำเนินการทั้งสอง Ex: A * B จะให้ 200 |
4 | / (Division) หารตัวเศษด้วยตัวส่วน Ex: B / A จะให้ 2 |
5 | % (Modulus) แสดงผลส่วนที่เหลือของการหารจำนวนเต็ม Ex: B% A จะให้ 0 |
6 | ++ (Increment) เพิ่มค่าจำนวนเต็มขึ้นหนึ่ง Ex: A ++ จะให้ 11 |
7 | -- (Decrement) ลดค่าจำนวนเต็มลงหนึ่ง Ex: A-- จะให้ 9 |
ตัวดำเนินการเปรียบเทียบ
Solidity รองรับตัวดำเนินการเปรียบเทียบต่อไปนี้ -
สมมติว่าตัวแปร A ถือ 10 และตัวแปร B ถือ 20 จากนั้น -
แสดงตัวอย่าง
ซีเนียร์ | ตัวดำเนินการและคำอธิบาย |
---|---|
1 | = = (Equal) ตรวจสอบว่าค่าของตัวถูกดำเนินการสองตัวเท่ากันหรือไม่ถ้าใช่เงื่อนไขจะกลายเป็นจริง Ex: (A == B) ไม่เป็นความจริง |
2 | != (Not Equal) ตรวจสอบว่าค่าของตัวถูกดำเนินการสองตัวเท่ากันหรือไม่หากค่าไม่เท่ากันเงื่อนไขจะกลายเป็นจริง Ex: (A! = B) เป็นจริง |
3 | > (Greater than) ตรวจสอบว่าค่าของตัวถูกดำเนินการด้านซ้ายมากกว่าค่าของตัวถูกดำเนินการด้านขวาหรือไม่ถ้าใช่เงื่อนไขจะกลายเป็นจริง Ex: (A> B) ไม่เป็นความจริง |
4 | < (Less than) ตรวจสอบว่าค่าของตัวถูกดำเนินการด้านซ้ายน้อยกว่าค่าของตัวถูกดำเนินการด้านขวาหรือไม่ถ้าใช่เงื่อนไขจะกลายเป็นจริง Ex: (A <B) เป็นจริง |
5 | >= (Greater than or Equal to) ตรวจสอบว่าค่าของตัวถูกดำเนินการด้านซ้ายมากกว่าหรือเท่ากับค่าของตัวถูกดำเนินการด้านขวาหรือไม่ถ้าใช่เงื่อนไขจะกลายเป็นจริง Ex: (A> = B) ไม่เป็นความจริง |
6 | <= (Less than or Equal to) ตรวจสอบว่าค่าของตัวถูกดำเนินการด้านซ้ายน้อยกว่าหรือเท่ากับค่าของตัวถูกดำเนินการด้านขวาหรือไม่ถ้าใช่เงื่อนไขจะกลายเป็นจริง Ex: (A <= B) เป็นจริง |
ตัวดำเนินการทางตรรกะ
Solidity สนับสนุนตัวดำเนินการเชิงตรรกะต่อไปนี้ -
สมมติว่าตัวแปร A ถือ 10 และตัวแปร B ถือ 20 จากนั้น -
แสดงตัวอย่าง
ซีเนียร์ | ตัวดำเนินการและคำอธิบาย |
---|---|
1 | && (Logical AND) ถ้าตัวถูกดำเนินการทั้งสองไม่ใช่ศูนย์เงื่อนไขจะกลายเป็นจริง Ex: (A && B) เป็นเรื่องจริง |
2 | || (Logical OR) หากตัวถูกดำเนินการสองตัวใดตัวหนึ่งไม่เป็นศูนย์เงื่อนไขจะกลายเป็นจริง Ex: (A || B) เป็นจริง |
3 | ! (Logical NOT) กลับสถานะตรรกะของตัวถูกดำเนินการ หากเงื่อนไขเป็นจริงตัวดำเนินการ Logical NOT จะทำให้เป็นเท็จ Ex:! (A && B) เป็นเท็จ |
ตัวดำเนินการ Bitwise
Solidity รองรับตัวดำเนินการระดับบิตต่อไปนี้ -
สมมติว่าตัวแปร A ถือ 2 และตัวแปร B ถือ 3 จากนั้น -
แสดงตัวอย่าง
ซีเนียร์ | ตัวดำเนินการและคำอธิบาย |
---|---|
1 | & (Bitwise AND) ดำเนินการบูลีน AND กับอาร์กิวเมนต์จำนวนเต็มแต่ละบิต Ex: (A & B) คือ 2 |
2 | | (BitWise OR) ดำเนินการบูลีนหรือการดำเนินการกับอาร์กิวเมนต์จำนวนเต็มแต่ละบิต Ex: (A | B) คือ 3 |
3 | ^ (Bitwise XOR) ดำเนินการบูลีนเอกสิทธิ์เฉพาะหรือการดำเนินการกับอาร์กิวเมนต์จำนวนเต็มแต่ละบิต Exclusive OR หมายความว่าตัวถูกดำเนินการตัวใดตัวหนึ่งเป็นจริงหรือตัวถูกดำเนินการสองเป็นจริง แต่ไม่ใช่ทั้งสองอย่าง Ex: (A ^ B) คือ 1. |
4 | ~ (Bitwise Not) เป็นตัวดำเนินการยูนารีและดำเนินการโดยการย้อนกลับบิตทั้งหมดในตัวถูกดำเนินการ Ex: (~ B) คือ -4 |
5 | << (Left Shift) มันย้ายบิตทั้งหมดในตัวถูกดำเนินการตัวแรกไปทางซ้ายตามจำนวนตำแหน่งที่ระบุในตัวถูกดำเนินการที่สอง บิตใหม่เต็มไปด้วยศูนย์ การเลื่อนค่าไปทางซ้ายหนึ่งตำแหน่งจะเทียบเท่ากับการคูณด้วย 2 การเลื่อนสองตำแหน่งจะเทียบเท่ากับการคูณด้วย 4 และอื่น ๆ Ex: (A << 1) คือ 4 |
6 | >> (Right Shift) ตัวดำเนินการกะไบนารีขวา ค่าของตัวถูกดำเนินการด้านซ้ายจะถูกย้ายไปทางขวาตามจำนวนบิตที่ระบุโดยตัวถูกดำเนินการด้านขวา Ex: (A >> 1) คือ 1 |
7 | >>> (Right shift with Zero) ตัวดำเนินการนี้เหมือนกับตัวดำเนินการ >> ยกเว้นว่าบิตที่เลื่อนไปทางซ้ายจะเป็นศูนย์เสมอ Ex: (A >>> 1) คือ 1 |
ผู้ดำเนินการมอบหมาย
Solidity สนับสนุนตัวดำเนินการกำหนดดังต่อไปนี้ -
แสดงตัวอย่าง
ซีเนียร์ | ตัวดำเนินการและคำอธิบาย |
---|---|
1 | = (Simple Assignment ) กำหนดค่าจากตัวถูกดำเนินการด้านขวาไปยังตัวถูกดำเนินการด้านซ้าย Ex: C = A + B จะกำหนดค่าของ A + B ให้เป็น C |
2 | += (Add and Assignment) เพิ่มตัวถูกดำเนินการด้านขวาให้กับตัวถูกดำเนินการด้านซ้ายและกำหนดผลลัพธ์ให้กับตัวถูกดำเนินการด้านซ้าย Ex: C + = A เทียบเท่ากับ C = C + A |
3 | −= (Subtract and Assignment) มันจะลบตัวถูกดำเนินการด้านขวาออกจากตัวถูกดำเนินการด้านซ้ายและกำหนดผลลัพธ์ให้กับตัวถูกดำเนินการด้านซ้าย Ex: C - = A เทียบเท่ากับ C = C - A |
4 | *= (Multiply and Assignment) จะคูณตัวถูกดำเนินการด้านขวาด้วยตัวถูกดำเนินการด้านซ้ายและกำหนดผลลัพธ์ให้กับตัวถูกดำเนินการด้านซ้าย Ex: C * = A เทียบเท่ากับ C = C * A |
5 | /= (Divide and Assignment) มันแบ่งตัวถูกดำเนินการด้านซ้ายด้วยตัวถูกดำเนินการด้านขวาและกำหนดผลลัพธ์ให้กับตัวถูกดำเนินการด้านซ้าย Ex: C / = A เทียบเท่ากับ C = C / A |
6 | %= (Modules and Assignment) ใช้โมดูลัสโดยใช้ตัวถูกดำเนินการสองตัวและกำหนดผลลัพธ์ให้กับตัวถูกดำเนินการด้านซ้าย Ex: C% = A เทียบเท่ากับ C = C% A |
Note - ตรรกะเดียวกันนี้ใช้กับตัวดำเนินการ Bitwise ดังนั้นจึงกลายเป็น << =, >> =, >> =, & =, | = และ ^ =
ตัวดำเนินการตามเงื่อนไข (? :)
ตัวดำเนินการตามเงื่อนไขจะประเมินนิพจน์สำหรับค่าจริงหรือเท็จก่อนจากนั้นเรียกใช้หนึ่งในสองคำสั่งที่กำหนดขึ้นอยู่กับผลลัพธ์ของการประเมิน
แสดงตัวอย่าง
ซีเนียร์ | ตัวดำเนินการและคำอธิบาย |
---|---|
1 | ? : (Conditional ) ถ้า Condition เป็นจริง? จากนั้นค่า X: หรือค่า Y |
ในขณะที่เขียนสัญญาคุณอาจต้องเผชิญกับสถานการณ์ที่คุณต้องดำเนินการซ้ำแล้วซ้ำเล่า ในสถานการณ์เช่นนี้คุณจะต้องเขียนคำสั่งแบบวนซ้ำเพื่อลดจำนวนบรรทัด
Solidity รองรับลูปที่จำเป็นทั้งหมดเพื่อลดความกดดันในการเขียนโปรแกรม
ซีเนียร์ No | ลูปและคำอธิบาย |
---|---|
1 | ในขณะที่วนซ้ำ ลูปพื้นฐานที่สุดใน Solidity คือ while loop ซึ่งจะกล่าวถึงในบทนี้ |
2 | ทำ ... ในขณะที่วนซ้ำ do ... while loop คล้ายกับ while loop ยกเว้นว่าการตรวจสอบเงื่อนไขจะเกิดขึ้นที่ส่วนท้ายของลูป |
3 | สำหรับ Loop for loop เป็นรูปแบบการวนซ้ำที่กะทัดรัดที่สุด ประกอบด้วยสามส่วนที่สำคัญต่อไปนี้ |
4 | การควบคุมลูป Solidity ให้การควบคุมเต็มรูปแบบเพื่อจัดการกับลูปและเปลี่ยนคำสั่ง |
ในขณะที่เขียนโปรแกรมอาจมีสถานการณ์ที่คุณจำเป็นต้องใช้หนึ่งจากชุดของเส้นทางที่กำหนด ในกรณีเช่นนี้คุณต้องใช้ข้อความที่มีเงื่อนไขเพื่อให้โปรแกรมของคุณตัดสินใจได้อย่างถูกต้องและดำเนินการอย่างถูกต้อง
Solidity สนับสนุนคำสั่งเงื่อนไขซึ่งใช้เพื่อดำเนินการต่างๆตามเงื่อนไขที่แตกต่างกัน ที่นี่เราจะอธิบายไฟล์if..else คำให้การ.
ผังงานของ if-else
ผังงานต่อไปนี้แสดงวิธีการทำงานของคำสั่ง if-else
Solidity รองรับรูปแบบ if..else คำสั่ง -
ซีเนียร์ No | คำชี้แจงและคำอธิบาย |
---|---|
1 | ถ้าคำสั่ง คำสั่ง if เป็นคำสั่งควบคุมพื้นฐานที่ช่วยให้ Solidity สามารถตัดสินใจและดำเนินการตามเงื่อนไข |
2 | if ... else คำสั่ง คำสั่ง 'if ... else' เป็นคำสั่งควบคุมรูปแบบถัดไปที่ช่วยให้ Solidity ดำเนินการคำสั่งด้วยวิธีที่ควบคุมได้มากขึ้น |
3 | if ... else if ... คำสั่ง คำสั่ง if ... else if ... เป็นรูปแบบขั้นสูงของ if ... else ที่ช่วยให้ Solidity สามารถตัดสินใจได้อย่างถูกต้องจากเงื่อนไขต่างๆ |
Solidity รองรับ String literal โดยใช้ทั้ง double quote (") และ single quote (') ซึ่งจัดเตรียมสตริงเป็นชนิดข้อมูลเพื่อประกาศตัวแปรประเภท String
pragma solidity ^0.5.0;
contract SolidityTest {
string data = "test";
}
ในตัวอย่างข้างต้น "test" คือสตริงลิเทอรัลและข้อมูลคือตัวแปรสตริง วิธีที่ต้องการมากกว่าคือการใช้ไบต์ชนิดแทน String เนื่องจากการทำงานของสตริงต้องใช้ก๊าซมากขึ้นเมื่อเทียบกับการทำงานแบบไบต์ Solidity ให้การแปลงแบบ inbuilt ระหว่างไบต์เป็นสตริงและในทางกลับกัน ใน Solidity เราสามารถกำหนด String literal ให้กับตัวแปรประเภท byte32 ได้อย่างง่ายดาย Solidity ถือว่าเป็นไบต์ 32 ลิเทอรัล
pragma solidity ^0.5.0;
contract SolidityTest {
bytes32 data = "test";
}
อักขระหลบหนี
ซีเนียร์ | ตัวละครและคำอธิบาย |
---|---|
1 | \n เริ่มบรรทัดใหม่ |
2 | \\ แบ็กสแลช |
3 | \' ใบเสนอราคาเดียว |
4 | \" อ้างสองครั้ง |
5 | \b Backspace |
6 | \f ฟีดรูปแบบ |
7 | \r การกลับรถ |
8 | \t แท็บ |
9 | \v แท็บแนวตั้ง |
10 | \xNN แสดงค่า Hex และแทรกไบต์ที่เหมาะสม |
11 | \uNNNN แสดงค่า Unicode และแทรกลำดับ UTF-8 |
ไบต์ในการแปลงสตริง
ไบต์สามารถแปลงเป็น String โดยใช้ตัวสร้าง string ()
bytes memory bstr = new bytes(10);
string message = string(bstr);
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าสตริงทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
contract SolidityTest {
constructor() public{
}
function getResult() public view returns(string memory){
uint a = 1;
uint b = 2;
uint result = a + b;
return integerToString(result);
}
function integerToString(uint _i) internal pure
returns (string memory) {
if (_i == 0) {
return "0";
}
uint j = _i;
uint len;
while (j != 0) {
len++;
j /= 10;
}
bytes memory bstr = new bytes(len);
uint k = len - 1;
while (_i != 0) {
bstr[k--] = byte(uint8(48 + _i % 10));
_i /= 10;
}
return string(bstr);
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
เอาต์พุต
0: string: 3
Array คือโครงสร้างข้อมูลที่จัดเก็บคอลเลกชันตามลำดับที่มีขนาดคงที่ขององค์ประกอบประเภทเดียวกัน อาร์เรย์ใช้ในการจัดเก็บชุดข้อมูล แต่มักจะมีประโยชน์มากกว่าที่จะคิดว่าอาร์เรย์เป็นชุดของตัวแปรประเภทเดียวกัน
แทนที่จะประกาศตัวแปรเดี่ยวเช่น number0, number1, ... และ number99 คุณจะประกาศตัวแปรอาร์เรย์หนึ่งตัวเช่นตัวเลขและใช้ตัวเลข [0], ตัวเลข [1] และ ... , ตัวเลข [99] เพื่อแสดง ตัวแปรแต่ละตัว องค์ประกอบเฉพาะในอาร์เรย์ถูกเข้าถึงโดยดัชนี
ใน Solidity อาร์เรย์อาจมีขนาดคงที่ของเวลาคอมไพล์หรือขนาดไดนามิก สำหรับอาร์เรย์พื้นที่เก็บข้อมูลสามารถมีองค์ประกอบประเภทต่างๆได้เช่นกัน ในกรณีของอาร์เรย์หน่วยความจำประเภทองค์ประกอบไม่สามารถทำการแม็ปได้และในกรณีที่จะใช้เป็นพารามิเตอร์ฟังก์ชันประเภทองค์ประกอบควรเป็นประเภท ABI
อาร์เรย์ทั้งหมดประกอบด้วยตำแหน่งหน่วยความจำที่อยู่ติดกัน ที่อยู่ต่ำสุดสอดคล้องกับองค์ประกอบแรกและที่อยู่สูงสุดขององค์ประกอบสุดท้าย
การประกาศอาร์เรย์
ในการประกาศอาร์เรย์ที่มีขนาดคงที่ใน Solidity โปรแกรมเมอร์จะระบุประเภทขององค์ประกอบและจำนวนองค์ประกอบที่อาร์เรย์ต้องการดังนี้ -
type arrayName [ arraySize ];
สิ่งนี้เรียกว่าอาร์เรย์มิติเดียว arraySize ต้องเป็นค่าคงที่จำนวนเต็มมากกว่าศูนย์และ typeสามารถเป็นชนิดข้อมูล Solidity ที่ถูกต้องได้ ตัวอย่างเช่นในการประกาศอาร์เรย์ 10 องค์ประกอบที่เรียกว่า balance of type uint ให้ใช้คำสั่งนี้ -
uint balance[10];
ในการประกาศอาร์เรย์ของขนาดไดนามิกใน Solidity โปรแกรมเมอร์ระบุประเภทขององค์ประกอบดังต่อไปนี้ -
type[] arrayName;
การเริ่มต้นอาร์เรย์
คุณสามารถเริ่มต้นองค์ประกอบ Solidity array ทีละรายการหรือใช้คำสั่งเดียวดังนี้ -
uint balance[3] = [1, 2, 3];
จำนวนค่าระหว่างวงเล็บปีกกา [] ต้องไม่เกินจำนวนองค์ประกอบที่เราประกาศสำหรับอาร์เรย์ระหว่างวงเล็บเหลี่ยม [] ต่อไปนี้เป็นตัวอย่างในการกำหนดองค์ประกอบเดียวของอาร์เรย์ -
หากคุณละเว้นขนาดของอาร์เรย์อาร์เรย์ที่ใหญ่พอที่จะรองรับการเริ่มต้นจะถูกสร้างขึ้น ดังนั้นถ้าคุณเขียน -
uint balance[] = [1, 2, 3];
คุณจะสร้างอาร์เรย์เหมือนกับที่คุณทำในตัวอย่างก่อนหน้านี้
balance[2] = 5;
คำสั่งข้างต้นกำหนดองค์ประกอบหมายเลข 3 rdในอาร์เรย์เป็นค่า 5
การสร้างอาร์เรย์หน่วยความจำแบบไดนามิก
อาร์เรย์หน่วยความจำแบบไดนามิกถูกสร้างขึ้นโดยใช้คีย์เวิร์ดใหม่
uint size = 3;
uint balance[] = new uint[](size);
การเข้าถึงองค์ประกอบอาร์เรย์
องค์ประกอบถูกเข้าถึงโดยการสร้างดัชนีชื่ออาร์เรย์ ทำได้โดยการวางดัชนีขององค์ประกอบไว้ในวงเล็บเหลี่ยมหลังชื่อของอาร์เรย์ ตัวอย่างเช่น -
uint salary = balance[2];
ข้อความข้างต้นจะใช้องค์ประกอบ3 rdจากอาร์เรย์และกำหนดค่าให้กับตัวแปรเงินเดือน ต่อไปนี้เป็นตัวอย่างซึ่งจะใช้แนวคิดทั้งสามที่กล่าวถึงข้างต้น ได้แก่ การประกาศการมอบหมายและการเข้าถึงอาร์เรย์ -
สมาชิก
length- ความยาวส่งกลับขนาดของอาร์เรย์ ความยาวสามารถใช้เพื่อเปลี่ยนขนาดของอาร์เรย์แบบไดนามิกที่ตั้งค่าได้
push- push อนุญาตให้ผนวกองค์ประกอบเข้ากับอาร์เรย์จัดเก็บข้อมูลแบบไดนามิกในตอนท้าย จะส่งกลับความยาวใหม่ของอาร์เรย์
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าอาร์เรย์ทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
contract test {
function testArray() public pure{
uint len = 7;
//dynamic array
uint[] memory a = new uint[](7);
//bytes is same as byte[]
bytes memory b = new bytes(len);
assert(a.length == 7);
assert(b.length == len);
//access array variable
a[6] = 8;
//test array variable
assert(a[6] == 8);
//static array
uint[3] memory c = [uint(1) , 2, 3];
assert(c.length == 3);
}
}
Enums จำกัด ตัวแปรให้มีค่าที่กำหนดไว้ล่วงหน้าเพียงไม่กี่ค่า ค่าในรายการแจกแจงนี้เรียกว่า enums
ด้วยการใช้ enums คุณสามารถลดจำนวนข้อบกพร่องในโค้ดของคุณได้
ตัวอย่างเช่นหากเราพิจารณาใบสมัครสำหรับร้านขายน้ำผลไม้สดก็สามารถ จำกัด ขนาดแก้วให้เล็กกลางและใหญ่ได้ สิ่งนี้จะทำให้แน่ใจได้ว่าจะไม่อนุญาตให้ใครสั่งขนาดอื่นใดนอกจากขนาดเล็กกลางหรือใหญ่
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่า enum ทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
contract test {
enum FreshJuiceSize{ SMALL, MEDIUM, LARGE }
FreshJuiceSize choice;
FreshJuiceSize constant defaultChoice = FreshJuiceSize.MEDIUM;
function setLarge() public {
choice = FreshJuiceSize.LARGE;
}
function getChoice() public view returns (FreshJuiceSize) {
return choice;
}
function getDefaultChoice() public pure returns (uint) {
return uint(defaultChoice);
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
คลิกครั้งแรก setLarge ปุ่มเพื่อกำหนดค่าเป็น LARGE จากนั้นคลิก getChoice เพื่อรับตัวเลือกที่เลือก
เอาต์พุต
uint8: 2
คลิก getDefaultChoice ปุ่มเพื่อรับตัวเลือกเริ่มต้น
เอาต์พุต
uint256: 1
ประเภทโครงสร้างใช้เพื่อแสดงเรกคอร์ด สมมติว่าคุณต้องการติดตามหนังสือของคุณในห้องสมุด คุณอาจต้องการติดตามคุณลักษณะต่อไปนี้เกี่ยวกับหนังสือแต่ละเล่ม -
- Title
- Author
- Subject
- รหัสหนังสือ
การกำหนดโครงสร้าง
ในการกำหนดโครงสร้างคุณต้องใช้ structคำสำคัญ. คีย์เวิร์ด struct กำหนดชนิดข้อมูลใหม่โดยมีสมาชิกมากกว่าหนึ่งคน รูปแบบของคำสั่ง struct มีดังนี้ -
struct struct_name {
type1 type_name_1;
type2 type_name_2;
type3 type_name_3;
}
ตัวอย่าง
struct Book {
string title;
string author;
uint book_id;
}
การเข้าถึงโครงสร้างและตัวแปร
ในการเข้าถึงสมาชิกของโครงสร้างใด ๆ เราใช้ตัวดำเนินการเข้าถึงสมาชิก (.) ตัวดำเนินการเข้าถึงสมาชิกถูกเข้ารหัสเป็นช่วงเวลาระหว่างชื่อตัวแปรโครงสร้างและสมาชิกโครงสร้างที่เราต้องการเข้าถึง คุณจะใช้โครงสร้างเพื่อกำหนดตัวแปรประเภทโครงสร้าง ตัวอย่างต่อไปนี้แสดงวิธีใช้โครงสร้างในโปรแกรม
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าโครงสร้างทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
contract test {
struct Book {
string title;
string author;
uint book_id;
}
Book book;
function setBook() public {
book = Book('Learn Java', 'TP', 1);
}
function getBookId() public view returns (uint) {
return book.book_id;
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
คลิกครั้งแรก setBook ปุ่มเพื่อกำหนดค่าเป็น LARGE จากนั้นคลิก getBookId เพื่อรับรหัสหนังสือที่เลือก
เอาต์พุต
uint256: 1
การแม็ปเป็นประเภทการอ้างอิงเป็นอาร์เรย์และโครงสร้าง ต่อไปนี้เป็นไวยากรณ์เพื่อประกาศประเภทการแมป
mapping(_KeyType => _ValueType)
ที่ไหน
_KeyType- สามารถเป็นชนิดที่มีอยู่แล้วภายในบวกไบต์และสตริง ไม่อนุญาตให้ใช้ประเภทการอ้างอิงหรือวัตถุที่ซับซ้อน
_ValueType - สามารถเป็นประเภทใดก็ได้
ข้อควรพิจารณา
การจับคู่สามารถมีได้เฉพาะประเภทของ storage และโดยทั่วไปจะใช้สำหรับตัวแปรสถานะ
การทำแผนที่สามารถทำเครื่องหมายเป็นสาธารณะ Solidity สร้าง getter ให้กับมันโดยอัตโนมัติ
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าประเภทการแมปทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
contract LedgerBalance {
mapping(address => uint) public balances;
function updateBalance(uint newBalance) public {
balances[msg.sender] = newBalance;
}
}
contract Updater {
function updateBalance() public returns (uint) {
LedgerBalance ledgerBalance = new LedgerBalance();
ledgerBalance.updateBalance(10);
return ledgerBalance.balances(address(this));
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
คลิกครั้งแรก updateBalance ปุ่มเพื่อตั้งค่าเป็น 10 จากนั้นดูในบันทึกซึ่งจะแสดงเอาต์พุตที่ถอดรหัสเป็น -
เอาต์พุต
{
"0": "uint256: 10"
}
Solidity ช่วยให้เกิดการแปลงโดยนัยและอย่างชัดเจน คอมไพเลอร์ Solidity ช่วยให้สามารถแปลงข้อมูลโดยนัยระหว่างสองประเภทข้อมูลได้หากไม่มีการแปลงโดยนัยเป็นไปได้และไม่มีการสูญเสียข้อมูล ตัวอย่างเช่น uint8 สามารถแปลงเป็น uint16 ได้ แต่ int8 สามารถแปลงเป็น uint256 ได้เนื่องจาก int8 สามารถมีค่าลบที่ไม่อนุญาตใน uint256
การแปลงอย่างชัดเจน
เราสามารถแปลงชนิดข้อมูลเป็นประเภทอื่นอย่างชัดเจนโดยใช้ไวยากรณ์ตัวสร้าง
int8 y = -3;
uint x = uint(y);
//Now x = 0xfffff..fd == two complement representation of -3 in 256 bit format.
การแปลงเป็นประเภทที่เล็กกว่าจะทำให้บิตคำสั่งซื้อสูง
uint32 a = 0x12345678;
uint16 b = uint16(a); // b = 0x5678
การแปลงเป็นประเภทที่สูงขึ้นจะเพิ่มบิตช่องว่างทางด้านซ้าย
uint16 a = 0x1234;
uint32 b = uint32(a); // b = 0x00001234
การแปลงเป็นไบต์ที่เล็กลงจะทำให้ข้อมูลการสั่งซื้อสูง
bytes2 a = 0x1234;
bytes1 b = bytes1(a); // b = 0x12
การแปลงเป็นไบต์ที่ใหญ่ขึ้นจะเพิ่มบิตช่องว่างทางด้านขวา
bytes2 a = 0x1234;
bytes4 b = bytes4(a); // b = 0x12340000
การแปลงระหว่างไบต์ขนาดคงที่และ int จะทำได้ก็ต่อเมื่อทั้งสองมีขนาดเท่ากัน
bytes2 a = 0x1234;
uint32 b = uint16(a); // b = 0x00001234
uint32 c = uint32(bytes4(a)); // c = 0x12340000
uint8 d = uint8(uint16(a)); // d = 0x34
uint8 e = uint8(bytes1(a)); // e = 0x12
ตัวเลขฐานสิบหกสามารถกำหนดให้กับประเภทจำนวนเต็มใดก็ได้หากไม่จำเป็นต้องมีการตัดทอน
uint8 a = 12; // no error
uint32 b = 1234; // no error
uint16 c = 0x123456; // error, as truncation required to 0x3456
ในความเป็นปึกแผ่นเราสามารถใช้ wei, finney, szabo หรือ ether เป็นส่วนต่อท้ายของตัวอักษรเพื่อใช้ในการแปลงนิกายต่างๆที่อิงกับ ether ต่อหน่วยต่ำที่สุดคือ wei และ 1e12 หมายถึง 1 x 10 12
assert(1 wei == 1);
assert(1 szabo == 1e12);
assert(1 finney == 1e15);
assert(1 ether == 1e18);
assert(2 ether == 2000 fenny);
หน่วยเวลา
เช่นเดียวกับสกุลเงิน Solidity มีหน่วยเวลาที่หน่วยต่ำสุดคือวินาทีและเราสามารถใช้วินาทีนาทีชั่วโมงวันและสัปดาห์เป็นส่วนต่อท้ายเพื่อแสดงเวลา
assert(1 seconds == 1);
assert(1 minutes == 60 seconds);
assert(1 hours == 60 minutes);
assert(1 day == 24 hours);
assert(1 week == 7 days);
ตัวแปรพิเศษเป็นตัวแปรที่มีอยู่ทั่วโลกและให้ข้อมูลเกี่ยวกับบล็อกเชน ต่อไปนี้เป็นรายการตัวแปรพิเศษ -
ซีเนียร์ | ตัวแปรพิเศษและคำอธิบาย |
---|---|
1 | blockhash(uint blockNumber) returns (bytes32) แฮชของบล็อกที่กำหนด - ใช้ได้เฉพาะกับ 256 บล็อกล่าสุดไม่รวมบล็อกปัจจุบัน |
2 | block.coinbase (address payable) ที่อยู่ของคนงานเหมืองบล็อกปัจจุบัน |
3 | block.difficulty (uint) ความยากของบล็อกปัจจุบัน |
4 | block.gaslimit (uint) แก๊สลิมิตบล็อกปัจจุบัน |
5 | block.number (uint) หมายเลขบล็อกปัจจุบัน |
6 | block.timestamp การประทับเวลาบล็อกปัจจุบันเป็นวินาทีนับตั้งแต่ยุคยูนิกซ์ |
7 | gasleft() returns (uint256) ก๊าซที่เหลืออยู่ |
8 | msg.data (bytes calldata) กรอก calldata |
9 | msg.sender (address payable) ผู้ส่งข้อความ (สายปัจจุบัน) |
10 | msg.sig (bytes4) สี่ไบต์แรกของ calldata (เช่นตัวระบุฟังก์ชัน) |
11 | msg.value (uint) จำนวน wei ที่ส่งมาพร้อมข้อความ |
12 | now (uint) การประทับเวลาบล็อกปัจจุบัน (นามแฝงสำหรับ block.timestamp) |
13 | tx.gasprice (uint) ราคาก๊าซของธุรกรรม |
14 | tx.origin (address payable) ผู้ส่งธุรกรรม (เครือข่ายการโทรแบบเต็ม) |
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อดูการใช้ msg ซึ่งเป็นตัวแปรพิเศษเพื่อรับที่อยู่ผู้ส่งใน Solidity
pragma solidity ^0.5.0;
contract LedgerBalance {
mapping(address => uint) public balances;
function updateBalance(uint newBalance) public {
balances[msg.sender] = newBalance;
}
}
contract Updater {
function updateBalance() public returns (uint) {
LedgerBalance ledgerBalance = new LedgerBalance();
ledgerBalance.updateBalance(10);
return ledgerBalance.balances(address(this));
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
คลิกครั้งแรก updateBalance ปุ่มเพื่อตั้งค่าเป็น 10 จากนั้นดูในบันทึกซึ่งจะแสดงเอาต์พุตที่ถอดรหัสเป็น -
เอาต์พุต
{
"0": "uint256: 10"
}
Style Guide ช่วยรักษารูปแบบโค้ดให้สอดคล้องกันและทำให้โค้ดอ่านง่ายขึ้น ต่อไปนี้เป็นแนวทางปฏิบัติที่ดีที่สุดต่อไปนี้ในขณะที่เขียนสัญญากับ Solidity
เค้าโครงรหัส
Indentation- ใช้ช่องว่าง 4 ช่องแทนแท็บเพื่อรักษาระดับการเยื้อง หลีกเลี่ยงการผสมช่องว่างกับแท็บ
Two Blank Lines Rule - ใช้ 2 บรรทัดว่างระหว่างสองคำจำกัดความของสัญญา
pragma solidity ^0.5.0;
contract LedgerBalance {
//...
}
contract Updater {
//...
}
One Blank Line Rule- ใช้ 1 บรรทัดว่างระหว่างสองฟังก์ชัน ในกรณีที่ประกาศอย่างเดียวไม่จำเป็นต้องมีบรรทัดว่าง
pragma solidity ^0.5.0;
contract A {
function balance() public pure;
function account() public pure;
}
contract B is A {
function balance() public pure {
// ...
}
function account() public pure {
// ...
}
}
Maximum Line Length - บรรทัดเดียวไม่ควรยาวเกิน 79 อักขระเพื่อให้ผู้อ่านสามารถแยกวิเคราะห์รหัสได้ง่าย
Wrapping rules- อาร์กิวเมนต์แรกอยู่ในบรรทัดใหม่โดยไม่ต้องเปิดวงเล็บ ใช้การเยื้องเดียวต่ออาร์กิวเมนต์ องค์ประกอบสิ้นสุด); ควรเป็นคนสุดท้าย
function_with_a_long_name(
longArgument1,
longArgument2,
longArgument3
);
variable = function_with_a_long_name(
longArgument1,
longArgument2,
longArgument3
);
event multipleArguments(
address sender,
address recipient,
uint256 publicKey,
uint256 amount,
bytes32[] options
);
MultipleArguments(
sender,
recipient,
publicKey,
amount,
options
);
Source Code Encoding - ควรใช้การเข้ารหัส UTF-8 หรือ ASCII
Imports - คำสั่งนำเข้าควรอยู่ที่ด้านบนของไฟล์หลังจากการประกาศ pragma
Order of Functions - ควรจัดกลุ่มฟังก์ชันตามการมองเห็น
pragma solidity ^0.5.0;
contract A {
constructor() public {
// ...
}
function() external {
// ...
}
// External functions
// ...
// External view functions
// ...
// External pure functions
// ...
// Public functions
// ...
// Internal functions
// ...
// Private functions
// ...
}
Avoid extra whitespaces - หลีกเลี่ยงช่องว่างภายในวงเล็บวงเล็บหรือวงเล็บปีกกา
Control structures- วงเล็บควรเปิดในบรรทัดเดียวกับการประกาศ ปิดบนบรรทัดของตัวเองโดยคงการเยื้องเดียวกัน ใช้ช่องว่างที่มีวงเล็บปีกกาเปิด
pragma solidity ^0.5.0;
contract Coin {
struct Bank {
address owner;
uint balance;
}
}
if (x < 3) {
x += 1;
} else if (x > 7) {
x -= 1;
} else {
x = 5;
}
if (x < 3)
x += 1;
else
x -= 1;
Function Declaration- ใช้กฎข้างต้นในการจัดฟัน เพิ่มป้ายกำกับการมองเห็นเสมอ ป้ายกำกับการเปิดเผยควรมาก่อนตัวปรับแต่งแบบกำหนดเองใด ๆ
function kill() public onlyowner {
selfdestruct(owner);
}
Mappings - หลีกเลี่ยงช่องว่างในขณะที่ประกาศตัวแปรการทำแผนที่
mapping(uint => uint) map;
mapping(address => bool) registeredAddresses;
mapping(uint => mapping(bool => Data[])) public data;
mapping(uint => mapping(uint => s)) data;
Variable declaration - หลีกเลี่ยงช่องว่างในขณะที่ประกาศตัวแปรอาร์เรย์
uint[] x; // not unit [] x;
String declaration - ใช้เครื่องหมายคำพูดคู่เพื่อประกาศสตริงแทนคำพูดเดี่ยว
str = "foo";
str = "Hamlet says, 'To be or not to be...'";
ลำดับของการจัดวาง
องค์ประกอบควรจัดวางตามลำดับต่อไปนี้
คำสั่ง Pragma
นำเข้าคำสั่ง
Interfaces
Libraries
Contracts
ภายในอินเทอร์เฟซไลบรารีหรือสัญญาคำสั่งควรเป็น -
พิมพ์การประกาศ
ตัวแปรของรัฐ
Events
Functions
หลักการตั้งชื่อ
ควรตั้งชื่อสัญญาและห้องสมุดโดยใช้รูปแบบคำศัพท์ ตัวอย่างเช่น SmartContract, Owner เป็นต้น
ชื่อสัญญาและไลบรารีควรตรงกับชื่อไฟล์
ในกรณีที่มีหลายสัญญา / ไลบรารีในไฟล์ให้ใช้ชื่อของสัญญาหลัก / ไลบรารี
Owned.sol
pragma solidity ^0.5.0;
// Owned.sol
contract Owned {
address public owner;
constructor() public {
owner = msg.sender;
}
modifier onlyOwner {
//....
}
function transferOwnership(address newOwner) public onlyOwner {
//...
}
}
รัฐสภาโซล
pragma solidity ^0.5.0;
// Congress.sol
import "./Owned.sol";
contract Congress is Owned, TokenRecipient {
//...
}
ชื่อโครงสร้าง
- ใช้สไตล์ Capwords เช่น SmartCoinชื่อเหตุการณ์
- ใช้สไตล์ CapWords เช่นการฝากเงิน AfterTransferชื่อฟังก์ชัน
- ใช้สไตล์ mixedCase เช่น initiateSupplyตัวแปรท้องถิ่นและรัฐ
- ใช้สไตล์ mixedCase เช่นผู้สร้างที่อยู่จัดหาค่าคงที่
- ใช้ตัวพิมพ์ใหญ่ทั้งหมดพร้อมขีดล่างเพื่อแยกคำเช่น MAX_BLOCKSชื่อตัวปรับแต่ง
- ใช้สไตล์ mixCase เช่น onlyAfterชื่อ Enum
- ใช้สไตล์คำศัพท์เช่น TokenGroup
ฟังก์ชันคือกลุ่มของโค้ดที่ใช้ซ้ำได้ซึ่งสามารถเรียกใช้งานได้ทุกที่ในโปรแกรมของคุณ ทำให้ไม่ต้องเขียนโค้ดเดิมซ้ำแล้วซ้ำเล่า ช่วยโปรแกรมเมอร์ในการเขียนโค้ดโมดูลาร์ ฟังก์ชันช่วยให้โปรแกรมเมอร์สามารถแบ่งโปรแกรมขนาดใหญ่ออกเป็นฟังก์ชันขนาดเล็กและสามารถจัดการได้
เช่นเดียวกับภาษาการเขียนโปรแกรมขั้นสูงอื่น ๆ Solidity ยังสนับสนุนคุณสมบัติทั้งหมดที่จำเป็นในการเขียนโค้ดโมดูลาร์โดยใช้ฟังก์ชัน ส่วนนี้อธิบายวิธีการเขียนฟังก์ชันของคุณเองใน Solidity
นิยามฟังก์ชัน
ก่อนที่เราจะใช้ฟังก์ชันเราจำเป็นต้องกำหนดมัน วิธีทั่วไปในการกำหนดฟังก์ชันใน Solidity คือการใช้function คีย์เวิร์ดตามด้วยชื่อฟังก์ชันที่ไม่ซ้ำกันรายการพารามิเตอร์ (ที่อาจว่างเปล่า) และบล็อกคำสั่งที่ล้อมรอบด้วยวงเล็บปีกกา
ไวยากรณ์
ไวยากรณ์พื้นฐานแสดงที่นี่
function function-name(parameter-list) scope returns() {
//statements
}
ตัวอย่าง
ลองดูตัวอย่างต่อไปนี้ กำหนดฟังก์ชันที่เรียกว่า getResult ที่ไม่มีพารามิเตอร์ -
pragma solidity ^0.5.0;
contract Test {
function getResult() public view returns(uint){
uint a = 1; // local variable
uint b = 2;
uint result = a + b;
return result;
}
}
เรียกใช้ฟังก์ชัน
หากต้องการเรียกใช้ฟังก์ชันใดที่หนึ่งในสัญญาในภายหลังคุณจะต้องเขียนชื่อของฟังก์ชันนั้นตามที่แสดงในโค้ดต่อไปนี้
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าสตริงทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
contract SolidityTest {
constructor() public{
}
function getResult() public view returns(string memory){
uint a = 1;
uint b = 2;
uint result = a + b;
return integerToString(result);
}
function integerToString(uint _i) internal pure
returns (string memory) {
if (_i == 0) {
return "0";
}
uint j = _i;
uint len;
while (j != 0) {
len++;
j /= 10;
}
bytes memory bstr = new bytes(len);
uint k = len - 1;
while (_i != 0) {
bstr[k--] = byte(uint8(48 + _i % 10));
_i /= 10;
}
return string(bstr);//access local variable
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
เอาต์พุต
0: string: 3
พารามิเตอร์ฟังก์ชัน
จนถึงตอนนี้เราได้เห็นฟังก์ชันที่ไม่มีพารามิเตอร์ แต่มีสิ่งอำนวยความสะดวกในการส่งผ่านพารามิเตอร์ที่แตกต่างกันในขณะที่เรียกใช้ฟังก์ชัน พารามิเตอร์ที่ส่งผ่านเหล่านี้สามารถจับได้ภายในฟังก์ชันและการปรับแต่งใด ๆ ก็สามารถทำได้ผ่านพารามิเตอร์เหล่านั้น ฟังก์ชันสามารถใช้พารามิเตอร์หลายตัวคั่นด้วยลูกน้ำ
ตัวอย่าง
ลองดูตัวอย่างต่อไปนี้ เราได้ใช้ไฟล์uint2strฟังก์ชั่นที่นี่ ใช้เวลาหนึ่งพารามิเตอร์
pragma solidity ^0.5.0;
contract SolidityTest {
constructor() public{
}
function getResult() public view returns(string memory){
uint a = 1;
uint b = 2;
uint result = a + b;
return integerToString(result);
}
function integerToString(uint _i) internal pure
returns (string memory) {
if (_i == 0) {
return "0";
}
uint j = _i;
uint len;
while (j != 0) {
len++;
j /= 10;
}
bytes memory bstr = new bytes(len);
uint k = len - 1;
while (_i != 0) {
bstr[k--] = byte(uint8(48 + _i % 10));
_i /= 10;
}
return string(bstr);//access local variable
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
เอาต์พุต
0: string: 3
คำชี้แจงการคืนสินค้า
ฟังก์ชัน Solidity สามารถมีทางเลือกได้ returnคำให้การ. สิ่งนี้จำเป็นหากคุณต้องการส่งคืนค่าจากฟังก์ชัน คำสั่งนี้ควรเป็นคำสั่งสุดท้ายในฟังก์ชัน
ดังตัวอย่างข้างต้นเรากำลังใช้ฟังก์ชัน uint2str เพื่อส่งคืนสตริง
ใน Solidity ฟังก์ชันสามารถคืนค่าหลายค่าได้เช่นกัน ดูตัวอย่างด้านล่าง -
pragma solidity ^0.5.0;
contract Test {
function getResult() public view returns(uint product, uint sum){
uint a = 1; // local variable
uint b = 2;
product = a * b;
sum = a + b;
//alternative return statement to return
//multiple values
//return(a*b, a+b);
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
เอาต์พุต
0: uint256: product 2
1: uint256: sum 3
Function Modifiers ใช้เพื่อแก้ไขลักษณะการทำงานของฟังก์ชัน ตัวอย่างเช่นเพื่อเพิ่มข้อกำหนดเบื้องต้นให้กับฟังก์ชัน
ก่อนอื่นเราสร้างตัวปรับแต่งโดยมีหรือไม่มีพารามิเตอร์
contract Owner {
modifier onlyOwner {
require(msg.sender == owner);
_;
}
modifier costs(uint price) {
if (msg.value >= price) {
_;
}
}
}
เนื้อหาของฟังก์ชันถูกแทรกโดยที่สัญลักษณ์พิเศษ "_;" ปรากฏในคำจำกัดความของตัวปรับเปลี่ยน ดังนั้นหากพอใจเงื่อนไขของโมดิฟายเออร์ขณะเรียกใช้ฟังก์ชันนี้ฟังก์ชันจะถูกเรียกใช้งานมิฉะนั้นจะมีข้อยกเว้น
ดูตัวอย่างด้านล่าง -
pragma solidity ^0.5.0;
contract Owner {
address owner;
constructor() public {
owner = msg.sender;
}
modifier onlyOwner {
require(msg.sender == owner);
_;
}
modifier costs(uint price) {
if (msg.value >= price) {
_;
}
}
}
contract Register is Owner {
mapping (address => bool) registeredAddresses;
uint price;
constructor(uint initialPrice) public { price = initialPrice; }
function register() public payable costs(price) {
registeredAddresses[msg.sender] = true;
}
function changePrice(uint _price) public onlyOwner {
price = _price;
}
}
ดูฟังก์ชั่นให้แน่ใจว่าจะไม่แก้ไขสถานะ สามารถประกาศฟังก์ชันเป็นไฟล์view. ข้อความต่อไปนี้หากมีอยู่ในฟังก์ชันจะถือว่าเป็นการแก้ไขสถานะและคอมไพเลอร์จะส่งคำเตือนในกรณีดังกล่าว
การแก้ไขตัวแปรสถานะ
เปล่งเหตุการณ์
การสร้างสัญญาอื่น ๆ
การใช้ selfdestruct
การส่ง Ether ผ่านการโทร
เรียกใช้ฟังก์ชันใด ๆ ที่ไม่ได้ทำเครื่องหมายมุมมองหรือบริสุทธิ์
ใช้การโทรระดับต่ำ
การใช้ชุดประกอบแบบอินไลน์ที่มีรหัสบางตัว
เมธอด Getter เป็นฟังก์ชันมุมมองเริ่มต้น
ดูตัวอย่างด้านล่างโดยใช้ฟังก์ชันมุมมอง
ตัวอย่าง
pragma solidity ^0.5.0;
contract Test {
function getResult() public view returns(uint product, uint sum){
uint a = 1; // local variable
uint b = 2;
product = a * b;
sum = a + b;
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
เอาต์พุต
0: uint256: product 2
1: uint256: sum 3
ฟังก์ชั่นบริสุทธิ์ช่วยให้แน่ใจว่าพวกเขาไม่อ่านหรือแก้ไขสถานะ สามารถประกาศฟังก์ชันเป็นไฟล์pure. ข้อความต่อไปนี้หากมีอยู่ในฟังก์ชันจะถือว่าอ่านสถานะและคอมไพเลอร์จะส่งคำเตือนในกรณีดังกล่าว
การอ่านตัวแปรสถานะ
การเข้าถึงที่อยู่ (นี้) .balance หรือ <address> .balance
การเข้าถึงตัวแปรพิเศษของ block, tx, msg (สามารถอ่าน msg.sig และ msg.data ได้)
การเรียกใช้ฟังก์ชันใด ๆ ที่ไม่ได้ทำเครื่องหมายว่าบริสุทธิ์
การใช้แอสเซมบลีแบบอินไลน์ที่มี opcodes บางอย่าง
ฟังก์ชัน Pure สามารถใช้ฟังก์ชัน revert () และ require () เพื่อย้อนกลับการเปลี่ยนแปลงสถานะที่อาจเกิดขึ้นหากเกิดข้อผิดพลาด
ดูตัวอย่างด้านล่างโดยใช้ฟังก์ชันมุมมอง
ตัวอย่าง
pragma solidity ^0.5.0;
contract Test {
function getResult() public pure returns(uint product, uint sum){
uint a = 1;
uint b = 2;
product = a * b;
sum = a + b;
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
เอาต์พุต
0: uint256: product 2
1: uint256: sum 3
ฟังก์ชันทางเลือกเป็นฟังก์ชันพิเศษที่มีให้ในสัญญา มีคุณสมบัติดังต่อไปนี้ -
เรียกว่าเมื่อมีการเรียกใช้ฟังก์ชันที่ไม่มีอยู่จริงในสัญญา
จำเป็นต้องมีการทำเครื่องหมายภายนอก
มันไม่มีชื่อ
มันไม่มีข้อโต้แย้ง
มันไม่สามารถคืนสิ่งใด ๆ
สามารถกำหนดได้หนึ่งรายการต่อสัญญา
หากไม่ได้ทำเครื่องหมายว่าต้องจ่ายจะมีข้อยกเว้นหากสัญญารับอีเธอร์ธรรมดาโดยไม่มีข้อมูล
ตัวอย่างต่อไปนี้แสดงแนวคิดของฟังก์ชันทางเลือกต่อสัญญา
ตัวอย่าง
pragma solidity ^0.5.0;
contract Test {
uint public x ;
function() external { x = 1; }
}
contract Sink {
function() external payable { }
}
contract Caller {
function callTest(Test test) public returns (bool) {
(bool success,) = address(test).call(abi.encodeWithSignature("nonExistingFunction()"));
require(success);
// test.x is now 1
address payable testPayable = address(uint160(address(test)));
// Sending ether to Test contract,
// the transfer will fail, i.e. this returns false here.
return (testPayable.send(2 ether));
}
function callSink(Sink sink) public returns (bool) {
address payable sinkPayable = address(sink);
return (sinkPayable.send(2 ether));
}
}
คุณสามารถกำหนดได้หลายคำจำกัดความสำหรับชื่อฟังก์ชันเดียวกันในขอบเขตเดียวกัน นิยามของฟังก์ชันจะต้องแตกต่างกันตามประเภทและ / หรือจำนวนอาร์กิวเมนต์ในรายการอาร์กิวเมนต์ คุณไม่สามารถโอเวอร์โหลดการประกาศฟังก์ชันที่แตกต่างกันตามประเภทการส่งคืนเท่านั้น
ตัวอย่างต่อไปนี้แสดงแนวคิดของฟังก์ชันที่ทำงานหนักเกินไปใน Solidity
ตัวอย่าง
pragma solidity ^0.5.0;
contract Test {
function getSum(uint a, uint b) public pure returns(uint){
return a + b;
}
function getSum(uint a, uint b, uint c) public pure returns(uint){
return a + b + c;
}
function callSumWithTwoArguments() public pure returns(uint){
return getSum(1,2);
}
function callSumWithThreeArguments() public pure returns(uint){
return getSum(1,2,3);
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
คลิกปุ่ม callSumWithTwoArguments ก่อนจากนั้นจึงเรียกใช้ปุ่ม callSumWithThreeArguments เพื่อดูผลลัพธ์
เอาต์พุต
0: uint256: 3
0: uint256: 6
Solidity มีฟังก์ชันทางคณิตศาสตร์ในตัวเช่นกัน ต่อไปนี้เป็นวิธีการที่ใช้กันมาก -
addmod(uint x, uint y, uint k) returns (uint)- คำนวณ (x + y)% k ที่นอกจากจะดำเนินการด้วยความแม่นยำโดยพลการและไม่ได้ห่อรอบที่ 2 256
mulmod(uint x, uint y, uint k) returns (uint)- คำนวณ (x * y)% k ที่นอกจากจะดำเนินการด้วยความแม่นยำโดยพลการและไม่ได้ห่อรอบที่ 2 256
ตัวอย่างต่อไปนี้แสดงการใช้ฟังก์ชันทางคณิตศาสตร์ใน Solidity
ตัวอย่าง
pragma solidity ^0.5.0;
contract Test {
function callAddMod() public pure returns(uint){
return addmod(4, 5, 3);
}
function callMulMod() public pure returns(uint){
return mulmod(4, 5, 3);
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
คลิกปุ่ม callAddMod ก่อนจากนั้นคลิกปุ่ม callMulMod เพื่อดูผลลัพธ์
เอาต์พุต
0: uint256: 0
0: uint256: 2
Solidity มีฟังก์ชันการเข้ารหัสแบบ inbuilt เช่นกัน ต่อไปนี้เป็นวิธีการที่สำคัญ -
keccak256(bytes memory) returns (bytes32) - คำนวณแฮช Keccak-256 ของอินพุต
sha256(bytes memory) returns (bytes32) - คำนวณแฮช SHA-256 ของอินพุต
ripemd160(bytes memory) returns (bytes20) - คำนวณแฮช RIPEMD-160 ของอินพุต
sha256(bytes memory) returns (bytes32) - คำนวณแฮช SHA-256 ของอินพุต
ecrecover(bytes32 hash, uint8 v, bytes32 r, bytes32 s) returns (address)- กู้คืนที่อยู่ที่เชื่อมโยงกับคีย์สาธารณะจากลายเซ็นโค้งวงรีหรือส่งคืนศูนย์เมื่อเกิดข้อผิดพลาด พารามิเตอร์ฟังก์ชันสอดคล้องกับค่า ECDSA ของลายเซ็น: r - ลายเซ็น 32 ไบต์แรก; s: ลายเซ็น 32 ไบต์ที่สอง; v: ลายเซ็น 1 ไบต์สุดท้าย วิธีนี้ส่งคืนที่อยู่
ตัวอย่างต่อไปนี้แสดงการใช้ฟังก์ชันการเข้ารหัสใน Solidity
ตัวอย่าง
pragma solidity ^0.5.0;
contract Test {
function callKeccak256() public pure returns(bytes32 result){
return keccak256("ABC");
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
เอาต์พุต
0: bytes32: result 0xe1629b9dda060bb30c7908346f6af189c16773fa148d3366701fbaa35d54f3c8
รูปแบบการถอนทำให้มั่นใจได้ว่าจะไม่มีการโอนสายโดยตรงซึ่งก่อให้เกิดภัยคุกคามด้านความปลอดภัย สัญญาต่อไปนี้แสดงการใช้การโอนสายเพื่อส่งอีเธอร์ที่ไม่ปลอดภัย
pragma solidity ^0.5.0;
contract Test {
address payable public richest;
uint public mostSent;
constructor() public payable {
richest = msg.sender;
mostSent = msg.value;
}
function becomeRichest() public payable returns (bool) {
if (msg.value > mostSent) {
// Insecure practice
richest.transfer(msg.value);
richest = msg.sender;
mostSent = msg.value;
return true;
} else {
return false;
}
}
}
สัญญาข้างต้นสามารถแสดงผลในสถานะใช้ไม่ได้โดยทำให้คนรวยที่สุดเป็นสัญญาของฟังก์ชันทางเลือกที่ล้มเหลว เมื่อฟังก์ชันทางเลือกล้มเหลวฟังก์ชัน becomeRichest () จะล้มเหลวและสัญญาจะค้างตลอดไป เพื่อลดปัญหานี้เราสามารถใช้รูปแบบการถอน
ในรูปแบบการถอนเราจะรีเซ็ตจำนวนเงินที่รอดำเนินการก่อนการโอนแต่ละครั้ง จะทำให้มั่นใจได้ว่าสัญญาเฉพาะผู้โทรล้มเหลว
pragma solidity ^0.5.0;
contract Test {
address public richest;
uint public mostSent;
mapping (address => uint) pendingWithdrawals;
constructor() public payable {
richest = msg.sender;
mostSent = msg.value;
}
function becomeRichest() public payable returns (bool) {
if (msg.value > mostSent) {
pendingWithdrawals[richest] += msg.value;
richest = msg.sender;
mostSent = msg.value;
return true;
} else {
return false;
}
}
function withdraw() public {
uint amount = pendingWithdrawals[msg.sender];
pendingWithdrawals[msg.sender] = 0;
msg.sender.transfer(amount);
}
}
การเข้าถึงสัญญาที่ จำกัด ถือเป็นแนวทางปฏิบัติทั่วไป โดยค่าเริ่มต้นสถานะสัญญาจะเป็นแบบอ่านอย่างเดียวเว้นแต่จะระบุเป็นสาธารณะ
เราสามารถ จำกัด ผู้ที่สามารถแก้ไขสถานะของสัญญาหรือเรียกฟังก์ชันของสัญญาโดยใช้ตัวปรับเปลี่ยน เราจะสร้างและใช้ตัวดัดแปลงหลายตัวตามที่อธิบายไว้ด้านล่าง -
onlyBy - เมื่อใช้กับฟังก์ชันแล้วมีเพียงผู้โทรที่กล่าวถึงเท่านั้นที่สามารถเรียกใช้ฟังก์ชันนี้ได้
onlyAfter - เมื่อใช้กับฟังก์ชันแล้วสามารถเรียกใช้ฟังก์ชันนั้นได้หลังจากช่วงเวลาหนึ่ง
costs - เมื่อใช้กับฟังก์ชันแล้วผู้โทรจะสามารถเรียกใช้ฟังก์ชันนี้ได้ก็ต่อเมื่อมีการระบุค่าไว้
ตัวอย่าง
pragma solidity ^0.5.0;
contract Test {
address public owner = msg.sender;
uint public creationTime = now;
modifier onlyBy(address _account) {
require(
msg.sender == _account,
"Sender not authorized."
);
_;
}
function changeOwner(address _newOwner) public onlyBy(owner) {
owner = _newOwner;
}
modifier onlyAfter(uint _time) {
require(
now >= _time,
"Function called too early."
);
_;
}
function disown() public onlyBy(owner) onlyAfter(creationTime + 6 weeks) {
delete owner;
}
modifier costs(uint _amount) {
require(
msg.value >= _amount,
"Not enough Ether provided."
);
_;
if (msg.value > _amount)
msg.sender.transfer(msg.value - _amount);
}
function forceOwnerChange(address _newOwner) public payable costs(200 ether) {
owner = _newOwner;
if (uint(owner) & 0 == 1) return;
}
}
Contract in Solidity คล้ายกับ Class ใน C ++ สัญญามีคุณสมบัติดังต่อไปนี้
Constructor - ฟังก์ชั่นพิเศษที่ประกาศด้วยคีย์เวิร์ดตัวสร้างซึ่งจะดำเนินการหนึ่งครั้งต่อสัญญาและจะถูกเรียกใช้เมื่อมีการสร้างสัญญา
State Variables - ตัวแปรต่อสัญญาเพื่อจัดเก็บสถานะของสัญญา
Functions - ฟังก์ชั่นต่อสัญญาซึ่งสามารถปรับเปลี่ยนตัวแปรสถานะเพื่อเปลี่ยนสถานะของสัญญาได้
ปริมาณการมองเห็น
ต่อไปนี้เป็นตัวระบุการมองเห็นต่างๆสำหรับฟังก์ชัน / ตัวแปรสถานะของสัญญา
external- ฟังก์ชันภายนอกหมายถึงการเรียกใช้โดยสัญญาอื่น ๆ ไม่สามารถใช้สำหรับการโทรภายใน ในการเรียกใช้ฟังก์ชันภายนอกภายในสัญญานี้จำเป็นต้องมีการเรียก function_name () ไม่สามารถทำเครื่องหมายตัวแปรสถานะเป็นภายนอกได้
public- ฟังก์ชันสาธารณะ / ตัวแปรสามารถใช้ได้ทั้งภายนอกและภายใน สำหรับตัวแปร public state Solidity จะสร้างฟังก์ชัน getter โดยอัตโนมัติ
internal - ฟังก์ชันภายใน / ตัวแปรสามารถใช้ได้เฉพาะภายในหรือโดยสัญญาที่ได้รับ
private - ฟังก์ชันส่วนตัว / ตัวแปรสามารถใช้ได้เฉพาะภายในและไม่สามารถใช้ได้แม้กระทั่งสัญญาที่ได้รับ
ตัวอย่าง
pragma solidity ^0.5.0;
contract C {
//private state variable
uint private data;
//public state variable
uint public info;
//constructor
constructor() public {
info = 10;
}
//private function
function increment(uint a) private pure returns(uint) { return a + 1; }
//public function
function updateData(uint a) public { data = a; }
function getData() public view returns(uint) { return data; }
function compute(uint a, uint b) internal pure returns (uint) { return a + b; }
}
//External Contract
contract D {
function readData() public returns(uint) {
C c = new C();
c.updateData(7);
return c.getData();
}
}
//Derived Contract
contract E is C {
uint private result;
C private c;
constructor() public {
c = new C();
}
function getComputedResult() public {
result = compute(3, 5);
}
function getResult() public view returns(uint) { return result; }
function getData() public view returns(uint) { return c.info(); }
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application เรียกใช้วิธีการต่างๆของสัญญา สำหรับ E.getComputedResult () ตามด้วย E.getResult () แสดง -
เอาต์พุต
0: uint256: 8
การสืบทอดเป็นวิธีการขยายการทำงานของสัญญา Solidity รองรับการสืบทอดทั้งแบบเดี่ยวและแบบหลายรายการ ต่อไปนี้เป็นไฮไลต์สำคัญ
สัญญาที่ได้รับสามารถเข้าถึงสมาชิกที่ไม่ใช่ส่วนตัวทั้งหมดรวมถึงวิธีการภายในและตัวแปรของรัฐ แต่ไม่อนุญาตให้ใช้สิ่งนี้
อนุญาตให้ใช้การแทนที่ฟังก์ชันได้หากลายเซ็นฟังก์ชันยังคงเหมือนเดิม ในกรณีที่พารามิเตอร์เอาต์พุตแตกต่างกันการคอมไพล์จะล้มเหลว
เราสามารถเรียกฟังก์ชันของ super contract โดยใช้ super keyword หรือใช้ชื่อ super contract
ในกรณีของการสืบทอดหลายรายการการเรียกใช้ฟังก์ชันโดยใช้ super จะให้ความสำคัญกับสัญญาที่ได้รับส่วนใหญ่
ตัวอย่าง
pragma solidity ^0.5.0;
contract C {
//private state variable
uint private data;
//public state variable
uint public info;
//constructor
constructor() public {
info = 10;
}
//private function
function increment(uint a) private pure returns(uint) { return a + 1; }
//public function
function updateData(uint a) public { data = a; }
function getData() public view returns(uint) { return data; }
function compute(uint a, uint b) internal pure returns (uint) { return a + b; }
}
//Derived Contract
contract E is C {
uint private result;
C private c;
constructor() public {
c = new C();
}
function getComputedResult() public {
result = compute(3, 5);
}
function getResult() public view returns(uint) { return result; }
function getData() public view returns(uint) { return c.info(); }
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application เรียกใช้วิธีการต่างๆของสัญญา สำหรับ E.getComputedResult () ตามด้วย E.getResult () แสดง -
เอาต์พุต
0: uint256: 8
Constructor เป็นฟังก์ชันพิเศษที่ประกาศโดยใช้ constructorคำสำคัญ. เป็นฟังก์ชันเสริมและใช้เพื่อเริ่มต้นตัวแปรสถานะของสัญญา ต่อไปนี้เป็นลักษณะสำคัญของตัวสร้าง
สัญญาสามารถมีผู้สร้างได้เพียงคนเดียว
รหัสตัวสร้างจะถูกดำเนินการหนึ่งครั้งเมื่อสร้างสัญญาและใช้เพื่อเริ่มต้นสถานะสัญญา
หลังจากรันโค้ดคอนสตรัคเตอร์โค้ดสุดท้ายจะถูกปรับใช้กับบล็อคเชน รหัสนี้ประกอบด้วยฟังก์ชันสาธารณะและรหัสที่เข้าถึงได้ผ่านฟังก์ชันสาธารณะ รหัสตัวสร้างหรือวิธีการภายในใด ๆ ที่ใช้โดยตัวสร้างเท่านั้นจะไม่รวมอยู่ในโค้ดสุดท้าย
ตัวสร้างสามารถเป็นได้ทั้งแบบสาธารณะหรือแบบภายใน
ตัวสร้างภายในทำเครื่องหมายสัญญาเป็นนามธรรม
ในกรณีที่ไม่มีการกำหนดตัวสร้างมีตัวสร้างเริ่มต้นอยู่ในสัญญา
pragma solidity ^0.5.0;
contract Test {
constructor() public {}
}
ในกรณีที่สัญญาพื้นฐานมีตัวสร้างที่มีข้อโต้แย้งสัญญาที่ได้รับแต่ละสัญญาจะต้องผ่านพวกเขา
ตัวสร้างฐานสามารถเริ่มต้นได้โดยตรงโดยใช้วิธีต่อไปนี้ -
pragma solidity ^0.5.0;
contract Base {
uint data;
constructor(uint _data) public {
data = _data;
}
}
contract Derived is Base (5) {
constructor() public {}
}
ตัวสร้างฐานสามารถเริ่มต้นทางอ้อมโดยใช้วิธีต่อไปนี้ -
pragma solidity ^0.5.0;
contract Base {
uint data;
constructor(uint _data) public {
data = _data;
}
}
contract Derived is Base {
constructor(uint _info) Base(_info * _info) public {}
}
ไม่อนุญาตให้ใช้วิธีการเริ่มต้นตัวสร้างสัญญาพื้นฐานโดยตรงและโดยอ้อม
หากสัญญาที่ได้มาไม่ได้ส่งผ่านอาร์กิวเมนต์ไปยังผู้สร้างสัญญาพื้นฐานสัญญาที่ได้รับจะกลายเป็นนามธรรม
บทคัดย่อสัญญาคือสิ่งที่มีอย่างน้อยหนึ่งฟังก์ชันโดยไม่มีการใช้งานใด ๆ สัญญาดังกล่าวใช้เป็นสัญญาฐาน โดยทั่วไปแล้วสัญญาที่เป็นนามธรรมจะมีทั้งฟังก์ชันที่นำไปใช้และเป็นนามธรรม สัญญาที่ได้มาจะใช้ฟังก์ชันนามธรรมและใช้ฟังก์ชันที่มีอยู่เมื่อจำเป็น
ในกรณีที่สัญญาที่ได้มาไม่ได้ใช้ฟังก์ชันนามธรรมสัญญาที่ได้รับนี้จะถูกทำเครื่องหมายเป็นนามธรรม
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าสัญญานามธรรมทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
contract Calculator {
function getResult() public view returns(uint);
}
contract Test is Calculator {
function getResult() public view returns(uint) {
uint a = 1;
uint b = 2;
uint result = a + b;
return result;
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
เอาต์พุต
0: uint256: 3
อินเทอร์เฟซคล้ายกับสัญญานามธรรมและสร้างขึ้นโดยใช้ interfaceคำสำคัญ. ต่อไปนี้เป็นลักษณะสำคัญของอินเทอร์เฟซ
อินเทอร์เฟซไม่สามารถมีฟังก์ชันใด ๆ กับการใช้งานได้
ฟังก์ชันของอินเทอร์เฟซเป็นได้เฉพาะประเภทภายนอกเท่านั้น
อินเทอร์เฟซไม่สามารถมีตัวสร้างได้
อินเทอร์เฟซต้องไม่มีตัวแปรสถานะ
อินเทอร์เฟซสามารถมี enum, โครงสร้างที่สามารถเข้าถึงได้โดยใช้สัญกรณ์ชื่อจุดของอินเทอร์เฟซ
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าอินเทอร์เฟซทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
interface Calculator {
function getResult() external view returns(uint);
}
contract Test is Calculator {
constructor() public {}
function getResult() external view returns(uint){
uint a = 1;
uint b = 2;
uint result = a + b;
return result;
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
Note - เลือกทดสอบจากเมนูแบบเลื่อนลงก่อนคลิกปุ่มปรับใช้
เอาต์พุต
0: uint256: 3
ไลบรารีคล้ายกับสัญญา แต่ส่วนใหญ่มีไว้สำหรับการใช้ซ้ำ ไลบรารีประกอบด้วยฟังก์ชันที่สัญญาอื่น ๆ สามารถเรียกใช้ได้ Solidity มีข้อ จำกัด บางประการในการใช้ Library ต่อไปนี้เป็นลักษณะสำคัญของ Solidity Library
ฟังก์ชันไลบรารีสามารถเรียกใช้ได้โดยตรงหากไม่ได้ปรับเปลี่ยนสถานะ นั่นหมายความว่าฟังก์ชัน pure หรือ view สามารถเรียกใช้ได้จากภายนอกไลบรารีเท่านั้น
ห้องสมุดไม่สามารถถูกทำลายได้เนื่องจากถือว่าเป็นคนไร้สัญชาติ
ไลบรารีไม่สามารถมีตัวแปรสถานะ
ไลบรารีไม่สามารถสืบทอดองค์ประกอบใด ๆ
ไม่สามารถสืบทอดไลบรารีได้
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าไลบรารีทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
library Search {
function indexOf(uint[] storage self, uint value) public view returns (uint) {
for (uint i = 0; i < self.length; i++) if (self[i] == value) return i;
return uint(-1);
}
}
contract Test {
uint[] data;
constructor() public {
data.push(1);
data.push(2);
data.push(3);
data.push(4);
data.push(5);
}
function isValuePresent() external view returns(uint){
uint value = 4;
//search if value is present in the array using Library function
uint index = Search.indexOf(data, value);
return index;
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
Note - เลือกทดสอบจากเมนูแบบเลื่อนลงก่อนคลิกปุ่มปรับใช้
เอาต์พุต
0: uint256: 3
ใช้สำหรับ
คำสั่ง using A for B; สามารถใช้เพื่อแนบฟังก์ชั่นไลบรารีของไลบรารี A กับประเภทที่กำหนด B ฟังก์ชันเหล่านี้จะใช้ประเภทผู้โทรเป็นพารามิเตอร์แรก (ระบุโดยใช้ตัวเอง)
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าไลบรารีทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
library Search {
function indexOf(uint[] storage self, uint value) public view returns (uint) {
for (uint i = 0; i < self.length; i++)if (self[i] == value) return i;
return uint(-1);
}
}
contract Test {
using Search for uint[];
uint[] data;
constructor() public {
data.push(1);
data.push(2);
data.push(3);
data.push(4);
data.push(5);
}
function isValuePresent() external view returns(uint){
uint value = 4;
//Now data is representing the Library
uint index = data.indexOf(value);
return index;
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
Note - เลือกทดสอบจากเมนูแบบเลื่อนลงก่อนคลิกปุ่มปรับใช้
เอาต์พุต
0: uint256: 3
Solidity มีตัวเลือกในการใช้ภาษาแอสเซมบลีเพื่อเขียนแอสเซมบลีแบบอินไลน์ภายในซอร์สโค้ด Solidity นอกจากนี้เรายังสามารถเขียนโค้ดแอสเซมบลีแบบสแตนด์อโลนซึ่งจะถูกแปลงเป็นไบต์โค้ด Standalone Assembly เป็นภาษากลางสำหรับคอมไพเลอร์ Solidity และจะแปลงรหัส Solidity เป็น Standalone Assembly จากนั้นเป็นรหัสไบต์ เราสามารถใช้ภาษาเดียวกับที่ใช้ใน Inline Assembly เพื่อเขียนโค้ดในชุดประกอบแบบสแตนด์อโลน
การประกอบแบบอินไลน์
รหัสแอสเซมบลีแบบอินไลน์สามารถแทรกเข้าไปในฐานรหัส Solidity เพื่อให้สามารถควบคุม EVM ได้ละเอียดมากขึ้นและใช้โดยเฉพาะในขณะที่เขียนฟังก์ชันไลบรารี
รหัสแอสเซมบลีเขียนอยู่ภายใต้ assembly { ... } บล็อก.
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าไลบรารีทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
library Sum {
function sumUsingInlineAssembly(uint[] memory _data) public pure returns (uint o_sum) {
for (uint i = 0; i < _data.length; ++i) {
assembly {
o_sum := add(o_sum, mload(add(add(_data, 0x20), mul(i, 0x20))))
}
}
}
}
contract Test {
uint[] data;
constructor() public {
data.push(1);
data.push(2);
data.push(3);
data.push(4);
data.push(5);
}
function sum() external view returns(uint){
return Sum.sumUsingInlineAssembly(data);
}
}
เรียกใช้โปรแกรมข้างต้นโดยใช้ขั้นตอนที่ให้ไว้ในบทSolidity First Application
Note - เลือกทดสอบจากเมนูแบบเลื่อนลงก่อนคลิกปุ่มปรับใช้
เอาต์พุต
0: uint256: 15
เหตุการณ์เป็นสมาชิกที่สืบทอดได้ของสัญญา เหตุการณ์ถูกปล่อยออกมาจะเก็บอาร์กิวเมนต์ที่ส่งผ่านในบันทึกธุรกรรม บันทึกเหล่านี้ถูกเก็บไว้ในบล็อกเชนและสามารถเข้าถึงได้โดยใช้ที่อยู่ของสัญญาจนกว่าสัญญาจะปรากฏบนบล็อกเชน เหตุการณ์ที่สร้างขึ้นไม่สามารถเข้าถึงได้จากภายในสัญญาไม่ใช่แม้แต่เหตุการณ์ที่สร้างและปล่อยออกมา
สามารถประกาศเหตุการณ์โดยใช้คำสำคัญของเหตุการณ์
//Declare an Event
event Deposit(address indexed _from, bytes32 indexed _id, uint _value);
//Emit an event
emit Deposit(msg.sender, _id, msg.value);
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าเหตุการณ์ทำงานอย่างไรใน Solidity
ขั้นแรกสร้างสัญญาและปล่อยเหตุการณ์
pragma solidity ^0.5.0;
contract Test {
event Deposit(address indexed _from, bytes32 indexed _id, uint _value);
function deposit(bytes32 _id) public payable {
emit Deposit(msg.sender, _id, msg.value);
}
}
จากนั้นเข้าถึงเหตุการณ์ของสัญญาในโค้ด JavaScript
var abi = /* abi as generated using compiler */;
var ClientReceipt = web3.eth.contract(abi);
var clientReceiptContract = ClientReceipt.at("0x1234...ab67" /* address */);
var event = clientReceiptContract.Deposit(function(error, result) {
if (!error)console.log(result);
});
ควรพิมพ์รายละเอียดดังต่อไปนี้ -
เอาต์พุต
{
"returnValues": {
"_from": "0x1111...FFFFCCCC",
"_id": "0x50...sd5adb20",
"_value": "0x420042"
},
"raw": {
"data": "0x7f...91385",
"topics": ["0xfd4...b4ead7", "0x7f...1a91385"]
}
}
Solidity มีฟังก์ชันต่างๆสำหรับการจัดการข้อผิดพลาด โดยทั่วไปเมื่อเกิดข้อผิดพลาดสถานะจะถูกเปลี่ยนกลับสู่สถานะเดิม การตรวจสอบอื่น ๆ เพื่อป้องกันการเข้าถึงรหัสโดยไม่ได้รับอนุญาต ต่อไปนี้เป็นวิธีการสำคัญบางอย่างที่ใช้ในการจัดการข้อผิดพลาด -
assert(bool condition)- ในกรณีที่ไม่ตรงตามเงื่อนไขการเรียกใช้เมธอดนี้จะทำให้เกิด opcode ที่ไม่ถูกต้องและการเปลี่ยนแปลงใด ๆ ที่ทำกับสถานะจะถูกเปลี่ยนกลับ วิธีนี้ใช้สำหรับข้อผิดพลาดภายใน
require(bool condition)- ในกรณีที่ไม่ตรงตามเงื่อนไขการเรียกใช้เมธอดนี้จะเปลี่ยนกลับเป็นสถานะดั้งเดิม - วิธีนี้ใช้สำหรับข้อผิดพลาดในอินพุตหรือส่วนประกอบภายนอก
require(bool condition, string memory message)- ในกรณีที่ไม่ตรงตามเงื่อนไขการเรียกใช้เมธอดนี้จะเปลี่ยนกลับเป็นสถานะดั้งเดิม - วิธีนี้ใช้สำหรับข้อผิดพลาดในอินพุตหรือส่วนประกอบภายนอก มีตัวเลือกในการระบุข้อความที่กำหนดเอง
revert() - วิธีนี้จะยกเลิกการดำเนินการและยกเลิกการเปลี่ยนแปลงใด ๆ ที่ทำกับสถานะ
revert(string memory reason)- วิธีนี้จะยกเลิกการดำเนินการและยกเลิกการเปลี่ยนแปลงใด ๆ ที่ทำกับสถานะ มีตัวเลือกในการระบุข้อความที่กำหนดเอง
ตัวอย่าง
ลองใช้รหัสต่อไปนี้เพื่อทำความเข้าใจว่าการจัดการข้อผิดพลาดทำงานอย่างไรใน Solidity
pragma solidity ^0.5.0;
contract Vendor {
address public seller;
modifier onlySeller() {
require(
msg.sender == seller,
"Only seller can call this."
);
_;
}
function sell(uint amount) public payable onlySeller {
if (amount > msg.value / 2 ether)
revert("Not enough Ether provided.");
// Perform the sell operation.
}
}
เมื่อเรียกย้อนกลับก็จะส่งคืนข้อมูลเลขฐานสิบหกดังต่อไปนี้
เอาต์พุต
0x08c379a0 // Function selector for Error(string)
0x0000000000000000000000000000000000000000000000000000000000000020 // Data offset
0x000000000000000000000000000000000000000000000000000000000000001a // String length
0x4e6f7420656e6f7567682045746865722070726f76696465642e000000000000 // String data