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