Solidity là gì? Solidity là ngôn ngữ lập trình hướng contract, được sử dụng để viết smart contract trong hệ sinh thái Ethereum. Các tài liệu chính về lập trình bằng solidity có thể tìm thấy tại Solidity documentation. Lập trình Blockchain với Solidity 1. Kiểu dữ liệu bool ví dụ:
Solidity là gì?
Solidity là ngôn ngữ lập trình hướng contract, được sử dụng để viết smart contract trong hệ sinh thái Ethereum. Các tài liệu chính về lập trình bằng solidity có thể tìm thấy tại Solidity documentation.
Lập trình Blockchain với Solidity
1. Kiểu dữ liệu
bool
ví dụ: bool public finished = trueint/uint
(int8
–int256
/uint8
–uint256
) (step by 8 bit) ví dụ: uint public age = 20address
: ~20 byte ví dụ: address public payable owner = 0xe291CA329864b53F44964592B3BC52E6a8894b2Cstruct
ví dụ: struct Student { uint256 age; string name; }- string ví dụ: string public name = 'test';
mapping
: tương tự hashtable. Cấu trúcmapping(_KeyType => _ValueType)
với_KeyType
có thể là bất kì kiểu gì ngoại trừmapping
,dynamic-sized array
,contract
,enum
,struct
._ValueType
có thể là bất kì kiểu dữ liệu nào ví dụ: mapping(address=>Student) public listStudentarray
:- Fixed-sized array:
bytesI
(với0 < I <= 32
): alias forbyte[I]
,[]
- Dynamic-sized array:
string
,bytes
,[]
- Fixed-sized array:
2. Contract
Solidity xây dựng xoay quanh thành phần chính là contract
. Về cơ bản, contract tương tự class trong OOP với các thuộc tính (state variables
) và phương thức (functions
). Các khái niệm abstract contract (contract với ít nhất 1 phương thức chưa được thực thi), interface (chỉ gồm chữ ký thao tác) cũng tương tự OOP.
Contract trong solidity cho phép đa kế thừa. Việc này dễ đến nhiều vấn đề, trong số đó có Diamond Problem. Solidity sử dụng thuật toán C3 Linearzation tương tự python để xử lí đa kế thừa. Do vậy thứ tự khai báo kế thừa sau từ khóa is
là rất quan trọng.
Ví dụ: contract tạo token demo
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.1;
import "@openzeppelin/contracts/token/ERC20/extensions/ERC20Burnable.sol";
contract ContractToken is ERC20Burnable {
constructor() ERC20("Token Name", "SYMBOL"){
_mint(msg.sender, 1000 * 10 ** 18);
}
}
Ví dụ: tạo token NFT
// SPDX-License-Identifier: MIT
pragma solidity 0.8.13;
import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
import "@openzeppelin/contracts/utils/Strings.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
contract NftErc721 is ERC721, Ownable {
using Strings for uint256;
string public baseURI = "https://raw.githubusercontent.com/TranTien139/nft-demo/master/cards/";
uint256 public nftId = 0;
address public operator;
event Initialized(address indexed executor, uint256 at);
constructor() ERC721("Name Token", "SYMBOL") {
operator == msg.sender;
emit Initialized(msg.sender, block.number);
}
function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token");
string memory buri = baseURI;
return bytes(buri).length > 0
? string(abi.encodePacked(buri, tokenId.toString(), ".json"))
: '';
}
function mint(address to) public onlyOwner {
_mint(to, nftId);
nftId++;
}
function getTotalNft() view public returns(uint){
return nftId;
}
function setBaseUri(string memory tokenUri) public onlyOwner {
require(bytes(tokenUri).length > 0, "tokenUri required");
baseURI = tokenUri;
}
}
3. Phương thức
function ()
{internal|external|public|private}
[pure|constant|view|payable]
[returns ()]
Có 2 cách gọi 1 phương thức:
- internal calling: con trỏ
instruction
nhảy đến vị trí function trong bộ nhớ để thực thi - external calling: EVM thực hiện lệnh
call
Ứng với 2 cách gọi này có 4 mức visibility với phương thức: internal
, external
, public
và private
. Mặc định, phương thức sẽ có mức visibility là public
. Mô tả khái quát 4 mức này như sau:
internal
: chỉ có thể truy cập từ các phương thức bên trong contract hoặc từ contract con (kế thừa) (internal calling
)external
: là 1 phần của giao diện của contract, do đó có thể được truy cập từ contract khác. Bản thân contract chứa phương thức cũng có thể gọi nó bằng cách sử dụng từ khóathis
. Ví dụ ta có thể gọi phương thức được khai báofunction extFunc() external
bằng cáchthis.extFunc()
(external calling
)public
: là 1 phần của giao diện của contract, có thể được gọi từ trong contract (mà không cần từ khóathis
) hoặc từ contract khác (internal calling/external calling
)private
: chỉ có thể truy cập từ các phương thức bên trong contract (internal calling
)
Note
public
vsexternal
: phương thứcpublic
cần sao chép tham số vàomemory
trước khi thực thi (để có thể gọi từ cả trong và ngoài contract), trong khi đó phương thứcexternal
có thể đọc trực tiếp từ vùng dữ liệucalldata
. Đối với các kiểu dữ liệu phức tạp (array/struct), việc sao chép và cấp pháp bộ nhớ là tốn kém hơn so với đọc trực tiếp từcalldata
(tốn gas hơn)private
vsinternal
: phương thứcprivate
chỉ có thể truy cập từ trong chính contract của nó, trong khi phương thứcinternal
có thể được gọi từ cả contract con của nócall
vsdelegatecall
:call
sử dụng context (storage) của contract được gọi. trong khi đó,delegatecall
sử dụng context của contract gọi lệnhdelegatecall
Function modifier
Modifier được sử dụng để kiểm soát ngữ cảnh của phương thức. Một số modifier mặc định có thể kể đến đó là:
pure
: không truy cập/thay đổi các thuộc tính của contractview
: không thay đổi thuộc tính của contractconstant
: dùng cho khai báo biến hằng số không thay đổi giá trịpayable
: phải có với các phương thức sử dụngmsg.value
/ nếu muốn nhận Ether
modifier dùng cho các trường hợp mình sử dụng hàm dùng đến vài lần chung nên mình sẽ khai báo chung lại code ví dụ
modifier isAdmin(address _add) {
require(msg.address == _add, "Not an admin");
_;
}
Fallback function
Mỗi contract có duy nhất 1 phương thức không có tên (gọi là fallback-function). Phương thức này không có tham số (tuy nhiên vẫn có thể sử dụng msg.data
để lấy dữ liệu truyền theo lời gọi), cũng không có giá trị trả về. Phương thức này chỉ được gọi khi không có bất kì phương thức nào của contract khớp với lời gọi
fallback() external
{
// xử lý logic trong này
}
4. Thuộc tính
Khác với phương thức, thuộc tính của contract chỉ có 3 mức visibility (mặc định là internal
):
public
: các contract khác có thể truy cập thuộc tínhpublic
thông quagetter function
.
Ex: Đối với mapping
:
contract Complex {
mapping (uint => address) public data;
}
thì getter function
tương ứng: function data(uint key) public returns (address)
private
,internal
: tương tự visibility của phương thức
Data location: memory/storage/calldata
Trong EVM, có 3 dạng lưu trữ: memory
(lưu trữ không bền lâu), storage
(lưu trữ bền vững), calldata
(lưu trữ không bên vững, chỉ áp dụng để lưu trữ tham số của phương thức external
).
- Forced data location:
- Tham số của phương thức
external
:calldata
- Thuộc tính:
storage
- Tham số của phương thức
- Default data location:
- Thuộc tính và giá trị trả về của phương thức:
memory
- Biến cục bộ: storage
- Thuộc tính và giá trị trả về của phương thức:
Đối với các kiểu dữ liệu phức tạp (struct, array), phép gán giữa vùng lưu trữ memory
và storage
(giữa tham số và thuộc tính) luôn tạo ra 1 bản sao chép độc lập. Phép gán vào 1 biến cục bộ chỉ là phép gán 1 con trỏ, trỏ vào thuộc tính của contract. Hiểu được cơ chế này để kiểm soát việc gán và sao chép dữ liệu phức tạp (do chi phí cấp phát và sao chép dữ liệu khá cao) để kiểm soát gas cần cho mỗi transaction.
5. triển khai smart contract
- Sử dụng IDE https://remix.ethereum.org/ Hoặc bạn có thể dùng framework truffle hoặc hardhat
- code mẫu có thế tham khảo tại đây