Sağlamlık - Hızlı Kılavuz
Solidity, akıllı sözleşmeleri uygulamak için sözleşme odaklı, üst düzey bir programlama dilidir. Solidity, C ++, Python ve JavaScript'ten oldukça etkilenir ve Ethereum Sanal Makinesi'ni (EVM) hedeflemek için tasarlanmıştır.
Solidity statik olarak yazılmıştır, kalıtımı, kitaplıkları ve karmaşık kullanıcı tanımlı tür programlama dilini destekler.
Solidity'yi oylama, kitle fonlaması, kör müzayedeler ve çoklu imzalı cüzdanlar gibi kullanımlara yönelik sözleşmeler oluşturmak için kullanabilirsiniz.
Ethereum nedir?
Ethereum, merkezi olmayan bir ie'dir. Akıllı sözleşmeleri çalıştıran blockchain platformu, yani herhangi bir kesinti, sansür, dolandırıcılık veya üçüncü taraf müdahalesi olasılığı olmadan tam olarak programlandığı gibi çalışan uygulamalar.
Ethereum Sanal Makinesi (EVM)
EVM olarak da bilinen Ethereum Sanal Makinesi, Ethereum'daki akıllı sözleşmeler için çalışma zamanı ortamıdır. Ethereum Sanal Makinesi, tüm dünyadaki bilgisayarlar tarafından güvenlik sağlamaya ve güvenilmeyen kod çalıştırmaya odaklanır.
Hizmet reddi saldırılarını önlemede uzmanlaşan EVM, programların birbirlerinin durumuna erişmemesini sağlayarak, herhangi bir potansiyel müdahale olmaksızın iletişim kurulmasını sağlar.
Ethereum Sanal Makinesi, Ethereum tabanlı akıllı sözleşmeler için bir çalışma zamanı ortamı olarak hizmet verecek şekilde tasarlanmıştır.
Akıllı Sözleşme nedir?
Akıllı sözleşme, bir sözleşmenin müzakeresini veya performansını dijital olarak kolaylaştırmayı, doğrulamayı veya yürürlüğe koymayı amaçlayan bir bilgisayar protokolüdür. Akıllı sözleşmeler, üçüncü şahıslar olmadan güvenilir işlemlerin yapılmasına izin verir. Bu işlemler izlenebilir ve geri alınamaz.
Akıllı sözleşmeler kavramı ilk olarak 1994 yılında Nick Szabo tarafından önerildi. Szabo, dijital para birimi için zemin hazırlamasıyla tanınan hukuk bilimcisi ve kriptografıdır.
Şu anda Akıllı Sözleşmeyi anlamıyorsanız sorun değil, daha sonra ayrıntılara gireceğiz.
Bu bölüm, CentOS makinesinde Solidity derleyicisini nasıl kurabileceğimizi açıklar. Bir Linux makineniz yoksa, Çevrimiçi Derleyicimizi küçük sözleşmeler ve Solidity'yi hızlıca öğrenmek için kullanabilirsiniz.
Yöntem 1 - npm / Node.js
Bu, Solidity derleyicisini CentoS Makinenize kurmanın en hızlı yoludur. Solidity Compiler'ı kurmak için aşağıdaki adımları uyguluyoruz -
Node.js'yi yükleyin
Öncelikle, CentOS makinenizde node.js'nin bulunduğundan emin olun. Mevcut değilse, aşağıdaki komutları kullanarak kurun -
# 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
Her şey kurulmuşsa, bunun gibi bir çıktı göreceksiniz -
3.10.10
Solc'yi yükle
Node.js paket yöneticisini kurduktan sonra, Solidity derleyicisini aşağıdaki gibi kurmaya devam edebilirsiniz -
$sudonpm install -g solc
Yukarıdaki komut solcjs programını kuracak ve onu sistem üzerinden global olarak kullanıma sunacaktır. Şimdi aşağıdaki komutu vererek Solidity derleyicinizi test edebilirsiniz -
$solcjs-version
Her şey yolunda giderse, bu aşağıdaki gibi bir şey yazdıracaktır -
0.5.2+commit.1df8f40c.Emscripten.clang
Artık standart Solidity derleyicisinden daha az özelliğe sahip olan solcj'leri kullanmaya hazırsınız, ancak bu size iyi bir başlangıç noktası sağlayacaktır.
Yöntem 2 - Docker Görüntüsü
Bir Docker imajını çekip Solidity programlamaya başlamak için kullanmaya başlayabilirsiniz. Basit adımlar aşağıdadır. Aşağıda bir Solidity Docker Görüntüsü çekme komutu verilmiştir.
$docker pull ethereum/solc:stable
Bir docker görüntüsü indirildikten sonra, aşağıdaki komutu kullanarak onu doğrulayabiliriz.
$docker run ethereum/solc:stable-version
Bu, aşağıdaki gibi bir şey yazdıracaktır -
$ docker run ethereum/solc:stable -version
solc, the solidity compiler commandlineinterfaceVersion: 0.5.2+commit.1df8f40c.Linux.g++
Yöntem 3: İkili Paketlerin Yüklenmesi
Linux makinenize tam teşekküllü bir derleyici kurmak istiyorsanız, lütfen Solidity Derleyicisini Kurma resmi web sitesini kontrol edin.
Bir Solidity kaynak dosyaları, herhangi bir sayıda sözleşme tanımını, ithalat direktiflerini ve pragma direktiflerini içerebilir.
Solidity'nin basit bir kaynak dosyasıyla başlayalım. Aşağıda bir Solidity dosyası örneği verilmiştir -
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
İlk satır, kaynak kodun Solidity 0.4.0 sürümü için veya işlevselliği bozmayan ancak sürüm 0.6.0'ı içermeyen daha yeni bir şey için yazıldığını söyleyen bir pragma yönergesidir.
Bir pragma yönergesi her zaman bir kaynak dosyaya yereldir ve başka bir dosyayı içe aktarırsanız, o dosyadaki pragma otomatik olarak içe aktarılacak dosyaya uygulanmaz.
Dolayısıyla, 0.4.0 sürümünden daha önce derlenmeyecek ve aynı zamanda 0.5.0 sürümünden başlayan bir derleyici üzerinde çalışmayacak bir dosya için bir pragma aşağıdaki gibi yazılacaktır -
pragma solidity ^0.4.0;
Burada ikinci koşul ^ kullanılarak eklenir.
Sözleşme
Bir Solidity sözleşmesi, Ethereumblockchain üzerinde belirli bir adreste bulunan bir kod (işlevleri) ve veri (durumu) koleksiyonudur.
UintstoredData satırı, depolanmış verisi adında uint türünde bir durum değişkeni bildirir ve set ve get işlevleri değişkenin değerini değiştirmek veya almak için kullanılabilir.
Dosyaları İçe Aktarma
Yukarıdaki örnekte bir import ifadesi bulunmasa da Solidity, JavaScript'te mevcut olanlara çok benzeyen import ifadelerini destekler.
Aşağıdaki ifade, tüm global sembolleri "dosya adı" ndan içe aktarır.
import "filename";
Aşağıdaki örnek, tüm üyeleri "dosya adı" ndaki genel semboller olan yeni bir global sembol symbolName oluşturur.
import * as symbolName from "filename";
Geçerli dosyayla aynı dizinden bir x dosyasını içe aktarmak için, x; olarak import "./x" öğesini kullanın. X olarak içe aktar "x" kullanırsanız; bunun yerine, genel bir "içerme dizini" içinde farklı bir dosyaya başvurulabilir.
Ayrılmış Anahtar Kelimeler
Solidity'de ayrılmış anahtar kelimeler aşağıdadır -
Öz | sonra | takma ad | uygulamak |
Oto | durum | tutmak | kopyası |
varsayılan | tanımlamak | final | değişmez |
uygular | içinde | Çizgide | İzin Vermek |
makro | eşleşme | değişebilir | boş |
nın-nin | geçersiz kılmak | kısmi | söz vermek |
referans | yeniden yerleştirilebilir | Mühürlü | boyutu |
statik | destekler | değiştirmek | Deneyin |
typedef | bir çeşit | kontrol edilmemiş |
Solidity Code tabanımızı Derlemek ve Çalıştırmak için Remix IDE kullanıyoruz .
Step 1 - Remix IDE Kod Bölümünde verilen kodu kopyalayın.
Misal
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 - Derleme Sekmesi altında, tıklayın Start to Compile buton.
Step 3 - Run Tab altında, tıklayın Deploy buton.
Step 4 - Run Tab altında, Select SolidityTest at 0x... açılır menüde.
Step 5 - Tıklayın getResult Sonucu görüntülemek için düğme.
Çıktı
0: uint256: 3
Solidity, hem C stili hem de C ++ stili yorumları destekler, Bu nedenle -
Bir // ile bir satırın sonu arasındaki herhangi bir metin yorum olarak kabul edilir ve Solidity Compiler tarafından yok sayılır.
/ * Ve * / karakterleri arasındaki herhangi bir metin yorum olarak kabul edilir. Bu, birden çok satıra yayılabilir.
Misal
Aşağıdaki örnek, Solidity'de yorumların nasıl kullanılacağını gösterir.
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;
}
Herhangi bir dilde program yazarken, çeşitli bilgileri depolamak için çeşitli değişkenler kullanmanız gerekir. Değişkenler, değerleri saklamak için ayrılmış bellek konumlarından başka bir şey değildir. Bu, bir değişken oluşturduğunuzda bellekte biraz yer ayırdığınız anlamına gelir.
Karakter, geniş karakter, tam sayı, kayan nokta, çift kayan nokta, boole vb. Gibi çeşitli veri türlerinin bilgilerini depolamak isteyebilirsiniz. ayrılmış hafıza.
Değer Türleri
Solidity, programcıya zengin bir yerleşik ve kullanıcı tanımlı veri türleri yelpazesi sunar. Aşağıdaki tabloda yedi temel C ++ veri türü listelenmiştir -
Tür | Anahtar kelime | Değerler |
---|---|---|
Boole | bool | doğru yanlış |
Tamsayı | int / uint | Değişik boyutlarda imzalı ve işaretsiz tam sayılar. |
Tamsayı | int8 - int256 | 8 bitten 256 bit'e int imzalı. int256, int ile aynıdır. |
Tamsayı | uint8'den uint256'ya | 8 bitten 256 bite kadar işaretsiz int. uint256, uint ile aynıdır. |
Sabit Nokta Numaraları | sabit / sabitlenmemiş | Değişik boyutlarda imzalı ve imzasız sabit nokta numaraları. |
Sabit Nokta Numaraları | sabit / sabitlenmemiş | Değişik boyutlarda imzalı ve imzasız sabit nokta numaraları. |
Sabit Nokta Numaraları | sabitMxN | M tipe göre alınan bit sayısını ve N'nin ondalık noktaları temsil ettiği imzalı sabit nokta numarası. M, 8'e bölünebilir ve 8'den 256'ya gitmelidir. N, 0'dan 80'e kadar olabilir. Sabit, sabit 128x18 ile aynıdır. |
Sabit Nokta Numaraları | ufixedMxN | M tipe göre alınan bit sayısını ve N'nin ondalık noktaları temsil ettiği işaretsiz sabit nokta numarası. M, 8'e bölünebilir ve 8'den 256'ya gider. N, 0'dan 80'e kadar olabilir. Ufixed, ufixed128x18 ile aynıdır. |
adres
adres, bir Ethereum adresinin boyutunu temsil eden 20 bayt değerini tutar. Bir adres, .balance yöntemi kullanılarak bakiye elde etmek için kullanılabilir ve .transfer yöntemi kullanılarak bakiyeyi başka bir adrese transfer etmek için kullanılabilir.
address x = 0x212;
address myAddress = this;
if (x.balance < 10 && myAddress.balance >= 10) x.transfer(10);
Solidity, üç tür değişkeni destekler.
State Variables - Değerleri bir sözleşmeli depoda kalıcı olarak depolanan değişkenler.
Local Variables - Fonksiyon çalıştırılana kadar değerleri mevcut olan değişkenler.
Global Variables - Blockchain hakkında bilgi almak için kullanılan global ad alanında özel değişkenler bulunur.
Solidity, statik olarak yazılmış bir dildir; bu, durum veya yerel değişken türünün bildirim sırasında belirtilmesi gerektiği anlamına gelir. Bildirilen her değişkenin türüne bağlı olarak her zaman bir varsayılan değeri vardır. "Tanımlanmamış" veya "boş" kavramı yoktur.
Durum değişkeni
Değerleri kalıcı olarak bir sözleşmeli depoda saklanan değişkenler.
pragma solidity ^0.5.0;
contract SolidityTest {
uint storedData; // State variable
constructor() public {
storedData = 10; // Using State variable
}
}
Yerel değişken
Değerleri yalnızca tanımlandığı bir işlev içinde bulunan değişkenler. İşlev parametreleri her zaman bu işlev için yereldir.
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
}
}
Misal
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Çıktı
0: uint256: 10
Global Değişkenler
Bunlar, küresel çalışma alanında bulunan ve blok zinciri ve işlem özellikleri hakkında bilgi sağlayan özel değişkenlerdir.
İsim | İadeler |
---|---|
blockhash (uint blockNumber) döndürür (bayt32) | Verilen bloğun karması - yalnızca en son 256 için çalışır, mevcut bloklar hariç |
block.coinbase (ödenecek adres) | Mevcut blok madencisinin adresi |
block.difficulty (uint) | Mevcut blok zorluğu |
block.gaslimit (uint) | Mevcut blok gaslimit |
block.number (uint) | Mevcut blok numarası |
block.timestamp (uint) | Unix döneminden bu yana saniye cinsinden geçerli blok zaman damgası |
gasleft () döndürür (uint256) | Kalan gaz |
msg.data (bayt çağrı verileri) | Çağrı verilerini tamamla |
msg.sender (ödenecek adres) | Mesajın göndereni (mevcut arayan) |
msg.sig (bayt4) | Çağrı verilerinin ilk dört baytı (işlev tanımlayıcı) |
msg.value (uint) | Mesajla birlikte gönderilen wei sayısı |
şimdi (uint) | Mevcut blok zaman damgası |
tx.gasprice (uint) | İşlemin gaz fiyatı |
tx.origin (ödenecek adres) | İşlemin göndereni |
Katılık Değişken İsimleri
Değişkenlerinizi Solidity'de adlandırırken aşağıdaki kuralları aklınızda bulundurun.
Solidity ayrılmış anahtar sözcüklerinden hiçbirini değişken adı olarak kullanmamalısınız. Bu anahtar kelimeler bir sonraki bölümde bahsedilmektedir. Örneğin, break veya boolean değişken adları geçerli değildir.
Solidity değişken isimleri bir rakamla (0-9) başlamamalıdır. Bir harf veya alt çizgi karakteri ile başlamalıdırlar. Örneğin, 123test geçersiz bir değişken adıdır, ancak _123test geçerlidir.
Solidity değişken adları büyük / küçük harfe duyarlıdır. Örneğin, Ad ve ad iki farklı değişkendir.
Yerel değişkenlerin kapsamı, tanımlandıkları işlevle sınırlıdır, ancak Durum değişkenlerinin üç tür kapsamı olabilir.
Public- Genel durum değişkenlerine dahili olarak ve mesajlar yoluyla erişilebilir. Bir genel durum değişkeni için, otomatik bir alıcı işlevi oluşturulur.
Internal - Dahili durum değişkenlerine sadece mevcut sözleşmeden veya ondan kaynaklanan sözleşmeden bu kullanılmadan dahili olarak erişilebilir.
Private - Özel durum değişkenlerine, türetilen sözleşmede değil, tanımlandıkları mevcut sözleşmeden yalnızca dahili olarak erişilebilir.
Misal
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
}
}
Operatör nedir?
Basit bir ifade alalım 4 + 5 is equal to 9. Burada 4 ve 5 deniroperands ve "+", operator. Solidity, aşağıdaki operatör türlerini destekler.
- Aritmetik operatörler
- Karşılaştırma Operatörleri
- Mantıksal (veya İlişkisel) Operatörler
- Atama Operatörleri
- Koşullu (veya üçlü) Operatörler
Tüm operatörleri tek tek inceleyelim.
Aritmetik operatörler
Solidity aşağıdaki aritmetik operatörleri destekler -
A değişkeninin 10, B değişkeninin 20 olduğunu varsayalım, o zaman -
Örnek Göster
Sr.No. | Operatör ve Açıklama |
---|---|
1 | + (Addition) İki işlenen ekler Ex: A + B 30 verecek |
2 | - (Subtraction) İlk işlenenden ikinci işleneni çıkarır Ex: A - B -10 verecek |
3 | * (Multiplication) Her iki işlenenle çarpın Ex: A * B 200 verecek |
4 | / (Division) Payı paydaya bölün Ex: B / A 2 verecek |
5 | % (Modulus) Bir tamsayı bölümünün kalanını verir Ex: B% A 0 verir |
6 | ++ (Increment) Bir tamsayı değerini bir artırır Ex: A ++, 11 verecek |
7 | -- (Decrement) Bir tamsayı değerini bir azaltır Ex: A-- 9 verecek |
Karşılaştırma Operatörleri
Solidity, aşağıdaki karşılaştırma operatörlerini destekler -
A değişkeninin 10, B değişkeninin 20 olduğunu varsayalım, o zaman -
Örnek Göster
Sr.No. | Operatör ve Açıklama |
---|---|
1 | = = (Equal) İki işlenenin değerinin eşit olup olmadığını kontrol eder, evet ise koşul doğru olur. Ex: (A == B) doğru değil. |
2 | != (Not Equal) İki işlenenin değerinin eşit olup olmadığını kontrol eder, değerler eşit değilse, koşul doğru olur. Ex: (A! = B) doğrudur. |
3 | > (Greater than) Soldaki işlenenin değerinin sağ işlenenin değerinden büyük olup olmadığını kontrol eder, evet ise koşul doğru olur. Ex: (A> B) doğru değil. |
4 | < (Less than) Soldaki işlenenin değerinin sağ işlenenin değerinden küçük olup olmadığını kontrol eder, evet ise koşul doğru olur. Ex: (A <B) doğrudur. |
5 | >= (Greater than or Equal to) Soldaki işlenenin değerinin sağ işlenenin değerinden büyük veya ona eşit olup olmadığını kontrol eder, evet ise, koşul doğru olur. Ex: (A> = B) doğru değil. |
6 | <= (Less than or Equal to) Soldaki işlenenin değerinin sağ işlenenin değerinden küçük veya ona eşit olup olmadığını kontrol eder, eğer öyleyse, koşul doğru olur. Ex: (A <= B) doğrudur. |
Mantıksal operatörler
Solidity, aşağıdaki mantıksal operatörleri destekler -
A değişkeninin 10, B değişkeninin 20 olduğunu varsayalım, o zaman -
Örnek Göster
Sr.No. | Operatör ve Açıklama |
---|---|
1 | && (Logical AND) Her iki işlenen de sıfır değilse, koşul doğru olur. Ex: (A && B) doğrudur. |
2 | || (Logical OR) İki işlenenden herhangi biri sıfır değilse, koşul doğru olur. Ex: (A || B) doğrudur. |
3 | ! (Logical NOT) İşleneninin mantıksal durumunu tersine çevirir. Bir koşul doğruysa, Mantıksal NOT operatörü onu yanlış yapar. Ex:! (A && B) yanlıştır. |
Bitsel Operatörler
Solidity aşağıdaki bitsel operatörleri destekler -
A değişkeninin 2, B değişkeninin 3, sonra -
Örnek Göster
Sr.No. | Operatör ve Açıklama |
---|---|
1 | & (Bitwise AND) Tamsayı argümanlarının her bitinde bir Boolean AND işlemi gerçekleştirir. Ex: (A ve B) 2'dir. |
2 | | (BitWise OR) Tamsayı bağımsız değişkenlerinin her bitinde bir Boole VEYA işlemi gerçekleştirir. Ex: (A | B) 3'tür. |
3 | ^ (Bitwise XOR) Tamsayı bağımsız değişkenlerinin her bitinde bir Boole özel OR işlemi gerçekleştirir. Dışlayıcı VEYA, işlenen birin doğru olduğu veya işlenen iki'nin doğru olduğu, ancak her ikisinin birden olmadığı anlamına gelir. Ex: (A ^ B) 1'dir. |
4 | ~ (Bitwise Not) Tekli bir operatördür ve işlenendeki tüm bitleri ters çevirerek çalışır. Ex: (~ B) -4'tür. |
5 | << (Left Shift) İlk işlenendeki tüm bitleri ikinci işlenende belirtilen basamak sayısı kadar sola taşır. Yeni bitler sıfırlarla doldurulur. Bir değeri bir konumla sola kaydırmak onu 2 ile çarpmaya eşdeğerdir, iki konumu kaydırmak 4 ile çarpmaya eşdeğerdir, vb. Ex: (A << 1) 4'tür. |
6 | >> (Right Shift) İkili Sağ Shift Operatörü. Soldaki işlenenin değeri, sağ işlenen tarafından belirtilen bit sayısı kadar sağa taşınır. Ex: (A >> 1) 1'dir. |
7 | >>> (Right shift with Zero) Sola kaydırılan bitlerin her zaman sıfır olması dışında bu operatör tıpkı >> operatörü gibidir. Ex: (A >>> 1) 1'dir. |
Atama Operatörleri
Solidity, aşağıdaki atama operatörlerini destekler -
Örnek Göster
Sr.No. | Operatör ve Açıklama |
---|---|
1 | = (Simple Assignment ) Sağ taraftaki işlenenden sol taraftaki işlenene değer atar Ex: C = A + B, A + B'nin değerini C'ye atar |
2 | += (Add and Assignment) Sağ işleneni sol işlenene ekler ve sonucu sol işlenene atar. Ex: C + = A, C = C + A'ya eşdeğerdir |
3 | −= (Subtract and Assignment) Sağ işleneni sol işlenenden çıkarır ve sonucu sol işlenene atar. Ex: C - = A, C = C - A'ya eşdeğerdir |
4 | *= (Multiply and Assignment) Sağ işleneni sol işlenenle çarpar ve sonucu sol işlenene atar. Ex: C * = A, C = C * A'ya eşdeğerdir |
5 | /= (Divide and Assignment) Soldaki işleneni sağ işlenenle böler ve sonucu sol işlenene atar. Ex: C / = A, C = C / A'ya eşdeğerdir |
6 | %= (Modules and Assignment) İki işlenen kullanarak modülü alır ve sonucu soldaki işlenene atar. Ex: C% = A eşdeğerdir C = C% A |
Note - Aynı mantık Bitwise operatörleri için de geçerlidir, böylece << =, >> =, >> =, & =, | = ve ^ = gibi olurlar.
Koşullu Operatör (? :)
Koşullu operatör önce doğru veya yanlış değer için bir ifadeyi değerlendirir ve ardından değerlendirmenin sonucuna bağlı olarak verilen iki ifadeden birini yürütür.
Örnek Göster
Sr.No. | Operatör ve Açıklama |
---|---|
1 | ? : (Conditional ) Koşul doğruysa? Ardından X değeri: Aksi takdirde Y değeri |
Bir sözleşme yazarken, defalarca bir işlem yapmanız gereken bir durumla karşılaşabilirsiniz. Bu tür durumlarda, satır sayısını azaltmak için döngü ifadeleri yazmanız gerekir.
Solidity, programlama baskısını azaltmak için gerekli tüm döngüleri destekler.
Sr.No | Döngüler ve Açıklama |
---|---|
1 | Döngü sırasında Solidity'deki en temel döngü, bu bölümde tartışılacak olan while döngüsüdür. |
2 | Döngü sırasında yap Do ... while döngüsü, döngünün sonunda koşul kontrolünün gerçekleşmesi dışında while döngüsüne benzer. |
3 | Döngü için For döngüsü, en kompakt döngü biçimidir. Aşağıdaki üç önemli bölümü içerir. |
4 | Döngü Kontrolü Solidity, döngüleri işlemek ve ifadeleri değiştirmek için tam kontrol sağlar. |
Bir program yazarken, belirli bir yol kümesinden birini benimsemeniz gereken bir durum olabilir. Bu tür durumlarda, programınızın doğru kararlar vermesine ve doğru eylemler gerçekleştirmesine olanak tanıyan koşullu ifadeler kullanmanız gerekir.
Solidity, farklı koşullara göre farklı eylemleri gerçekleştirmek için kullanılan koşullu ifadeleri destekler. Burada açıklayacağızif..else Beyan.
İf-else Akış Şeması
Aşağıdaki akış şeması, if-else ifadesinin nasıl çalıştığını gösterir.
Solidity aşağıdaki formları destekler if..else ifade -
Sr.No | İfadeler ve Açıklama |
---|---|
1 | eğer ifadesi İf ifadesi, Solidity'nin koşullu olarak kararlar almasına ve ifadeleri yürütmesine izin veren temel kontrol ifadesidir. |
2 | if ... else ifadesi 'If ... else' ifadesi, Solidity'nin ifadeleri daha kontrollü bir şekilde yürütmesine izin veren bir sonraki kontrol ifadesi biçimidir. |
3 | if ... else if ... ifadesi. İf ... else if ... ifadesi, Solidity'nin çeşitli koşullardan doğru bir karar vermesine izin veren gelişmiş bir if ... else biçimidir. |
Solidity, hem çift tırnak (") hem de tek tırnak (') kullanarak String değişmezini destekler. String türünde bir değişken bildirmek için bir veri türü olarak dize sağlar.
pragma solidity ^0.5.0;
contract SolidityTest {
string data = "test";
}
Yukarıdaki örnekte, "test" bir dize değişmezidir ve veri bir dize değişkenidir. Dize işlemi bayt işlemine kıyasla daha fazla gaz gerektirdiğinden, String yerine bayt türlerini kullanmak daha çok tercih edilen yöntemdir. Solidity, baytlar arasında dizgeye ve tam tersi arasında dahili dönüşüm sağlar. Solidity'de String değişmezini bir byte32 tipi değişkene kolayca atayabiliriz. Solidity, bunu bir byte32 değişmezi olarak kabul eder.
pragma solidity ^0.5.0;
contract SolidityTest {
bytes32 data = "test";
}
Kaçış Karakterleri
Sr.No. | Karakter açıklaması |
---|---|
1 | \n Yeni bir satır başlatır. |
2 | \\ Ters eğik çizgi |
3 | \' Tek alıntı |
4 | \" Çift Alıntı |
5 | \b Geri tuşu |
6 | \f Form besleme |
7 | \r Satırbaşı |
8 | \t Sekme |
9 | \v Dikey Sekme |
10 | \xNN Hex değerini temsil eder ve uygun baytları ekler. |
11 | \uNNNN Unicode değerini temsil eder ve UTF-8 dizisi ekler. |
Bayt'tan Dize Dönüşümüne
Baytlar, string () yapıcısı kullanılarak String'e dönüştürülebilir.
bytes memory bstr = new bytes(10);
string message = string(bstr);
Misal
Solidity'de dizenin nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Çıktı
0: string: 3
Dizi, aynı türden öğelerin sabit boyutlu sıralı bir koleksiyonunu depolayan bir veri yapısıdır. Bir dizi, bir veri koleksiyonunu depolamak için kullanılır, ancak bir diziyi aynı türden değişkenlerin bir koleksiyonu olarak düşünmek genellikle daha yararlıdır.
Sayı0, sayı1, ... ve sayı99 gibi tek tek değişkenleri bildirmek yerine, sayılar gibi bir dizi değişkeni bildirir ve temsil etmek için sayılar [0], sayılar [1] ve ..., sayılar [99] kullanırsınız bireysel değişkenler. Bir dizideki belirli bir öğeye bir dizinle erişilir.
Solidity'de bir dizi, derleme zamanı sabit boyutta veya dinamik boyutta olabilir. Depolama dizisi için farklı türde öğelere de sahip olabilir. Bellek dizisi durumunda eleman tipi eşlenemez ve fonksiyon parametresi olarak kullanılması durumunda eleman tipi ABI tipi olmalıdır.
Tüm diziler bitişik bellek konumlarından oluşur. En düşük adres ilk öğeye ve en yüksek adres son öğeye karşılık gelir.
Dizileri Bildirme
Solidity'de sabit büyüklükte bir dizi bildirmek için, programcı elemanların türünü ve bir dizinin gerektirdiği eleman sayısını aşağıdaki gibi belirler -
type arrayName [ arraySize ];
Buna tek boyutlu dizi denir. arraySize sıfırdan büyük bir tamsayı sabiti olmalı ve typeherhangi bir geçerli Solidity veri türü olabilir. Örneğin, uint türünde Balance adlı 10 öğeli bir dizi bildirmek için şu ifadeyi kullanın -
uint balance[10];
Solidity'de bir dinamik boyut dizisi bildirmek için, programcı aşağıdaki gibi elemanların türünü belirler -
type[] arrayName;
Dizileri Başlatma
Solidity dizi elemanlarını tek tek veya aşağıdaki gibi tek bir ifade kullanarak başlatabilirsiniz -
uint balance[3] = [1, 2, 3];
Küme parantezleri [] arasındaki değerlerin sayısı, köşeli parantezler [] arasındaki dizi için belirttiğimiz öğelerin sayısından büyük olamaz. Aşağıda, dizinin tek bir elemanını atamak için bir örnek verilmiştir -
Dizinin boyutunu atlarsanız, başlatmayı tutacak kadar büyük bir dizi oluşturulur. Bu nedenle, yazarsanız -
uint balance[] = [1, 2, 3];
Önceki örnekte yaptığınız gibi tam olarak aynı diziyi yaratacaksınız.
balance[2] = 5;
Yukarıdaki açıklamada atar elemanı 3 numaralı üçüncü dizide 5 bir değer.
Dinamik bellek dizileri oluşturma
Dinamik bellek dizileri yeni anahtar sözcük kullanılarak oluşturulur.
uint size = 3;
uint balance[] = new uint[](size);
Dizi Öğelerine Erişim
Dizi adını indeksleyerek bir elemana erişilir. Bu, elemanın dizini dizinin isminden sonra köşeli parantez içine yerleştirilerek yapılır. Örneğin -
uint salary = balance[2];
Yukarıdaki ifade diziden 3. elemanı alacak ve değeri maaş değişkenine atayacaktır. Aşağıda, yukarıda bahsedilen üç kavramın tümünü kullanacak bir örnek verilmiştir. bildirim, atama ve dizilere erişim -
Üyeler
length- uzunluk, dizinin boyutunu döndürür. uzunluk, dinamik dizinin boyutunu değiştirmek için kullanılabilir.
push- push, sonunda dinamik bir depolama dizisine bir öğe eklemeye izin verir. Dizinin yeni uzunluğunu döndürür.
Misal
Solidity'de dizilerin nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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);
}
}
Numaralandırmalar, bir değişkeni önceden tanımlanmış birkaç değerden birine sahip olacak şekilde kısıtlar. Bu numaralandırılmış listedeki değerlere numaralandırma adı verilir.
Numaralandırmanın kullanılmasıyla kodunuzdaki hataların sayısını azaltmak mümkündür.
Örneğin, bir taze meyve suyu dükkanı için bir uygulama düşünürsek, cam boyutunu küçük, orta ve büyük olarak sınırlamak mümkün olacaktır. Bu, kimsenin küçük, orta veya büyük dışında herhangi bir boyut sipariş etmesine izin vermeyeceğinden emin olur.
Misal
Enum'un Solidity'de nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
İlk Tıklama setLarge Değeri BÜYÜK olarak ayarlamak için düğme ve ardından getChoice seçilen seçeneği almak için.
Çıktı
uint8: 2
Tıklayın getDefaultChoice Varsayılan seçeneği elde etmek için düğme.
Çıktı
uint256: 1
Struct türleri bir kaydı temsil etmek için kullanılır. Kitaplarınızı bir kütüphanede takip etmek istediğinizi varsayalım. Her kitapla ilgili aşağıdaki öznitelikleri izlemek isteyebilirsiniz -
- Title
- Author
- Subject
- Kitap kimliği
Bir Struct Tanımlama
Bir Struct tanımlamak için, structanahtar kelime. Struct anahtar sözcüğü, birden fazla üyesi olan yeni bir veri türünü tanımlar. Struct ifadesinin biçimi aşağıdaki gibidir -
struct struct_name {
type1 type_name_1;
type2 type_name_2;
type3 type_name_3;
}
Misal
struct Book {
string title;
string author;
uint book_id;
}
Bir Struct ve değişkenine erişim
Bir yapının herhangi bir üyesine erişmek için üye erişim operatörünü (.) Kullanırız. Üye erişim operatörü, yapı değişkeni adı ile erişmek istediğimiz yapı üyesi arasında bir dönem olarak kodlanır. Yapı türünün değişkenlerini tanımlamak için yapıyı kullanırsınız. Aşağıdaki örnek, bir programda bir yapının nasıl kullanılacağını gösterir.
Misal
Solidity'de yapıların nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
İlk Tıklama setBook Değeri BÜYÜK olarak ayarlamak için düğme ve ardından getBookId seçilen kitap kimliğini almak için.
Çıktı
uint256: 1
Eşleme, diziler ve yapılar olarak bir başvuru türüdür. Aşağıda bir eşleme türü bildirmek için sözdizimi verilmiştir.
mapping(_KeyType => _ValueType)
Nerede
_KeyType- herhangi bir yerleşik tür artı bayt ve dize olabilir. Referans türüne veya karmaşık nesnelere izin verilmez.
_ValueType - herhangi bir tür olabilir.
Düşünceler
Eşleme yalnızca türüne sahip olabilir storage ve genellikle durum değişkenleri için kullanılır.
Eşleme herkese açık olarak işaretlenebilir. Solidity otomatik olarak bunun için alıcı yaratır.
Misal
Solidity'de eşleme türünün nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
İlk Tıklama updateBalance Değeri 10 olarak ayarlamak için düğme, ardından kodu çözülen çıktıyı şu şekilde gösterecek olan günlüklere bakın:
Çıktı
{
"0": "uint256: 10"
}
Solidity, örtük ve açık dönüşüme izin verir. Solidity derleyicisi, örtük bir dönüştürme mümkün olmadığı ve bilgi kaybı olmadığı sürece iki veri türü arasında örtük dönüşüme izin verir. Örneğin uint8, uint16'ya dönüştürülebilir, ancak int8, uint256'da izin verilmeyen negatif bir değer içerebildiğinden, int8 uint256'ya dönüştürülebilir.
Açık Dönüşüm
Yapıcı sözdizimini kullanarak bir veri türünü açıkça diğerine dönüştürebiliriz.
int8 y = -3;
uint x = uint(y);
//Now x = 0xfffff..fd == two complement representation of -3 in 256 bit format.
Daha küçük türe dönüştürme, daha yüksek sipariş bitlerine mal olur.
uint32 a = 0x12345678;
uint16 b = uint16(a); // b = 0x5678
Daha yüksek türe dönüştürme, sola dolgu bitleri ekler.
uint16 a = 0x1234;
uint32 b = uint32(a); // b = 0x00001234
Daha küçük bayta dönüştürme, daha yüksek sipariş verilerine mal olur.
bytes2 a = 0x1234;
bytes1 b = bytes1(a); // b = 0x12
Daha büyük bayta dönüştürme, sağ tarafa dolgu bitleri ekler.
bytes2 a = 0x1234;
bytes4 b = bytes4(a); // b = 0x12340000
Sabit boyutlu bayt ve int arasındaki dönüşüm yalnızca her ikisi de aynı boyutta olduğunda mümkündür.
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
Onaltılık sayılar, kesme gerekmiyorsa herhangi bir tam sayı türüne atanabilir.
uint8 a = 12; // no error
uint32 b = 1234; // no error
uint16 c = 0x123456; // error, as truncation required to 0x3456
Sağlamlıkta wei, finney, szabo veya ether'i çeşitli eter bazlı mezhepleri dönüştürmek için kullanılacak bir literal için bir ek olarak kullanabiliriz. Düşük birim Wei ve 1e12 1 x 10 temsil eder 12 .
assert(1 wei == 1);
assert(1 szabo == 1e12);
assert(1 finney == 1e15);
assert(1 ether == 1e18);
assert(2 ether == 2000 fenny);
Zaman Birimleri
Para birimine benzer şekilde, Solidity'de en düşük birimin ikinci olduğu zaman birimleri vardır ve zamanı belirtmek için son ek olarak saniye, dakika, saat, gün ve hafta kullanabiliriz.
assert(1 seconds == 1);
assert(1 minutes == 60 seconds);
assert(1 hours == 60 minutes);
assert(1 day == 24 hours);
assert(1 week == 7 days);
Özel değişkenler, küresel olarak mevcut değişkenlerdir ve blok zinciri hakkında bilgi sağlar. Özel değişkenlerin listesi aşağıdadır -
Sr.No. | Özel Değişken ve Açıklama |
---|---|
1 | blockhash(uint blockNumber) returns (bytes32) Verilen bloğun karması - yalnızca mevcut bloklar hariç en son 256 için çalışır. |
2 | block.coinbase (address payable) Mevcut blok madencisinin adresi. |
3 | block.difficulty (uint) mevcut blok zorluğu. |
4 | block.gaslimit (uint) Mevcut blok gaz limiti. |
5 | block.number (uint) Mevcut blok numarası. |
6 | block.timestamp Unix döneminden bu yana saniye cinsinden geçerli blok zaman damgası. |
7 | gasleft() returns (uint256) Kalan gaz. |
8 | msg.data (bytes calldata) Çağrı verilerini tamamlayın. |
9 | msg.sender (address payable) Mesajı gönderen (mevcut arama). |
10 | msg.sig (bytes4) Çağrı verilerinin ilk dört baytı (yani işlev tanımlayıcı) |
11 | msg.value (uint) Mesajla birlikte gönderilen wei sayısı. |
12 | now (uint) Geçerli blok zaman damgası (block.timestamp için takma ad). |
13 | tx.gasprice (uint) İşlemin gaz fiyatı. |
14 | tx.origin (address payable) İşlemin göndericisi (tam çağrı zinciri). |
Misal
Solidity'de gönderen adresini almak için özel bir değişken olan msg'nin kullanımını görmek için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
İlk Tıklama updateBalance Değeri 10 olarak ayarlamak için düğme, ardından kodu çözülen çıktıyı şu şekilde gösterecek olan günlüklere bakın:
Çıktı
{
"0": "uint256: 10"
}
Stil Kılavuzu, kod düzeninin tutarlı olmasına ve kodu daha okunaklı hale getirmeye yardımcı olur. Solidity ile sözleşme yazarken aşağıdaki en iyi uygulamaları takip edin.
Kod Düzeni
Indentation- Girinti seviyesini korumak için sekme yerine 4 boşluk kullanın. Boşlukları sekmelerle karıştırmaktan kaçının.
Two Blank Lines Rule - İki sözleşme tanımı arasında 2 Boş satır kullanın.
pragma solidity ^0.5.0;
contract LedgerBalance {
//...
}
contract Updater {
//...
}
One Blank Line Rule- İki işlev arasında 1 Boş satır kullanın. Sadece beyanname olması durumunda satır boş bırakılmasına gerek yoktur.
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 - Okuyucuların kodu kolayca ayrıştırabilmesi için tek bir satır 79 karakteri geçmemelidir.
Wrapping rules- İlk argüman parantez açmadan yeni satırda olmalıdır. Bağımsız değişken başına tek girinti kullanın. Sonlandırma öğesi); sonuncusu olmalı.
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 veya ASCII kodlaması tercihen kullanılacaktır.
Imports - Import ifadeleri pragma bildiriminden hemen sonra dosyanın en üstüne yerleştirilmelidir.
Order of Functions - Fonksiyonlar, görünürlüklerine göre gruplandırılmalıdır.
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 - Parantez, köşeli parantez veya parantez içinde boşluk bırakmayın.
Control structures- Kaşlı ayraçlar beyannameyle aynı satırda açılmalıdır. Aynı girintiyi koruyarak kendi satırlarını kapatın. Açma parantezli bir boşluk kullanın.
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- Diş telleri için yukarıdaki kuralı kullanın. Her zaman bir görünürlük etiketi ekleyin. Görünürlük etiketi, herhangi bir özel değiştiriciden önce gelmelidir.
function kill() public onlyowner {
selfdestruct(owner);
}
Mappings - Eşleme değişkenlerini bildirirken beyaz boşluklardan kaçının.
mapping(uint => uint) map;
mapping(address => bool) registeredAddresses;
mapping(uint => mapping(bool => Data[])) public data;
mapping(uint => mapping(uint => s)) data;
Variable declaration - Dizi değişkenlerini bildirirken boşluklardan kaçının.
uint[] x; // not unit [] x;
String declaration - Tek tırnak yerine bir dizeyi bildirmek için çift tırnak kullanın.
str = "foo";
str = "Hamlet says, 'To be or not to be...'";
Düzen Sırası
Öğeler aşağıdaki sırayla düzenlenmelidir.
Pragma ifadeleri
İçe aktarma beyanları
Interfaces
Libraries
Contracts
Arayüzler, kitaplıklar veya sözleşmelerde sıra şu şekilde olmalıdır:
Tip bildirimleri
Durum değişkenleri
Events
Functions
Adlandırma kuralları
Sözleşme ve Kitaplık, CapWords Stili kullanılarak adlandırılmalıdır. Örneğin, SmartContract, Owner vb.
Sözleşme ve Kitaplık adı, dosya adlarıyla eşleşmelidir.
Bir dosyada birden fazla sözleşme / kitaplık olması durumunda, ana sözleşme / kitaplık adını kullanın.
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 {
//...
}
}
Congress.sol
pragma solidity ^0.5.0;
// Congress.sol
import "./Owned.sol";
contract Congress is Owned, TokenRecipient {
//...
}
Struct İsimleri
- SmartCoin gibi CapWords Stilini kullanın.Etkinlik İsimleri
- Deposit, AfterTransfer gibi CapWords Stillerini kullanın.İşlev Adları
- initialSupply gibi mixedCase Stilini kullanın.Yerel ve Eyalet değişkenleri
- creatorAddress, supply gibi mixedCase Style kullanın.Sabitler
- MAX_BLOCKS gibi kelimeleri ayırmak için tüm büyük harfleri alt çizgi ile kullanın.Değiştirici Adları
- mixCase Stilini onlyAfter gibi kullanın.Enum Adları
- TokenGroup gibi CapWords Stilini kullanın.
Bir işlev, programınızın herhangi bir yerinde çağrılabilen yeniden kullanılabilir bir kod grubudur. Bu, aynı kodu tekrar tekrar yazma ihtiyacını ortadan kaldırır. Programcılara modüler kodlar yazmada yardımcı olur. İşlevler, bir programcının büyük bir programı birkaç küçük ve yönetilebilir işleve bölmesine izin verir.
Diğer gelişmiş programlama dilleri gibi Solidity de fonksiyonları kullanarak modüler kod yazmak için gerekli tüm özellikleri destekler. Bu bölüm Solidity'de kendi işlevlerinizi nasıl yazacağınızı açıklar.
İşlev Tanımı
Bir işlevi kullanmadan önce onu tanımlamamız gerekir. Solidity'de bir işlevi tanımlamanın en yaygın yolu,function anahtar kelime, ardından benzersiz bir işlev adı, bir parametre listesi (boş olabilir) ve küme parantezleri ile çevrili bir ifade bloğu gelir.
Sözdizimi
Temel sözdizimi burada gösterilmektedir.
function function-name(parameter-list) scope returns() {
//statements
}
Misal
Aşağıdaki örneği deneyin. GetResult adında parametre almayan bir işlevi tanımlar -
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;
}
}
Bir Fonksiyon Çağırma
Sözleşmede daha sonra bir yerde bir işlevi çağırmak için, aşağıdaki kodda gösterildiği gibi bu işlevin adını yazmanız yeterlidir.
Solidity'de dizenin nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Çıktı
0: string: 3
Fonksiyon Parametreleri
Şimdiye kadar parametresiz fonksiyonlar gördük. Ancak bir işlevi çağırırken farklı parametreleri geçirme olanağı vardır. Aktarılan bu parametreler işlevin içinde yakalanabilir ve bu parametreler üzerinde herhangi bir işlem yapılabilir. Bir işlev, virgülle ayrılmış birden çok parametre alabilir.
Misal
Aşağıdaki örneği deneyin. Bir kullandıkuint2strburada işlev. Bir parametre alır.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Çıktı
0: string: 3
Dönüş Bildirimi
Bir Solidity işlevinin isteğe bağlı bir returnBeyan. Bir işlevden bir değer döndürmek istiyorsanız bu gereklidir. Bu ifade, bir fonksiyondaki son ifade olmalıdır.
Yukarıdaki örnekte olduğu gibi, bir dizge döndürmek için uint2str işlevini kullanıyoruz.
Solidity'de bir işlev birden çok değer döndürebilir. Aşağıdaki örneğe bakın -
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Çıktı
0: uint256: product 2
1: uint256: sum 3
İşlev Değiştiriciler, bir işlevin davranışını değiştirmek için kullanılır. Örneğin, bir işleve bir ön koşul eklemek için.
İlk önce parametresiz veya parametresiz bir değiştirici oluşturuyoruz.
contract Owner {
modifier onlyOwner {
require(msg.sender == owner);
_;
}
modifier costs(uint price) {
if (msg.value >= price) {
_;
}
}
}
İşlev gövdesi, "_;" özel sembolünün bulunduğu yere eklenir. bir değiştiricinin tanımında görünür. Bu nedenle, bu işlevi çağırırken değiştiricinin koşulu sağlanırsa, işlev çalıştırılır ve aksi takdirde bir istisna atılır.
Aşağıdaki örneğe bakın -
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;
}
}
Görüntüleme işlevleri, durumu değiştirmemelerini sağlar. Bir işlev şu şekilde bildirilebilir:view. İşlevde varsa aşağıdaki ifadelerin durumu değiştirdiği kabul edilir ve derleyici bu gibi durumlarda uyarı verir.
Durum değişkenlerini değiştirme.
Olayları yayan.
Başka sözleşmeler yaratmak.
Kendi kendini yok etme kullanma.
Çağrı yoluyla Ether gönderme.
Görünüş veya saf olarak işaretlenmemiş herhangi bir işlevi çağırmak.
Düşük seviyeli aramaları kullanma.
Belirli işlem kodlarını içeren satır içi derlemeyi kullanma.
Alıcı yöntemi varsayılan olarak görünüm işlevleridir.
Bir görünüm işlevi kullanarak aşağıdaki örneğe bakın.
Misal
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Çıktı
0: uint256: product 2
1: uint256: sum 3
Saf işlevler, durumu okumamalarını veya değiştirmemelerini sağlar. Bir işlev şu şekilde bildirilebilir:pure. İşlevde varsa aşağıdaki ifadeler durumu okumak olarak kabul edilir ve derleyici bu gibi durumlarda uyarı verir.
Durum değişkenlerini okuma.
Adres (bu) .balance veya <address> .balance'a erişiliyor.
Block, tx, msg'nin özel değişkenlerinden herhangi birine erişim (msg.sig ve msg.data okunabilir).
Saf olarak işaretlenmemiş herhangi bir işlevi çağırmak.
Belirli işlem kodlarını içeren satır içi derlemeyi kullanma.
Saf işlevler, bir hata oluşursa olası durum değişikliklerini geri döndürmek için revert () ve gerektirir () işlevlerini kullanabilir.
Bir görünüm işlevi kullanarak aşağıdaki örneğe bakın.
Misal
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Çıktı
0: uint256: product 2
1: uint256: sum 3
Geri dönüş işlevi, bir sözleşmede kullanılabilen özel bir işlevdir. Aşağıdaki özelliklere sahiptir -
Sözleşmede var olmayan bir işlev çağrıldığında çağrılır.
Harici olarak işaretlenmesi gerekir.
Adı yok.
Hiçbir argümanı yok
Hiçbir şey iade edemez.
Kontrat başına bir tane tanımlanabilir.
Ödenecek olarak işaretlenmemişse, sözleşme veri olmadan sade eter alırsa istisna atar.
Aşağıdaki örnek, sözleşme başına bir geri dönüş işlevi kavramını göstermektedir.
Misal
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));
}
}
Aynı kapsamda aynı işlev adı için birden fazla tanımınız olabilir. İşlevin tanımı, türler ve / veya bağımsız değişken listesindeki bağımsız değişkenlerin sayısı açısından birbirinden farklı olmalıdır. Yalnızca dönüş türüne göre farklılık gösteren işlev bildirimlerini aşırı yükleyemezsiniz.
Aşağıdaki örnek, Solidity'de aşırı yüklenen bir fonksiyon kavramını göstermektedir.
Misal
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Önce callSumWithTwoArguments düğmesini ve ardından sonucu görmek için callSumWithThreeArguments düğmesini tıklayın.
Çıktı
0: uint256: 3
0: uint256: 6
Solidity, dahili matematiksel işlevler de sağlar. Aşağıdakiler yoğun olarak kullanılan yöntemlerdir -
addmod(uint x, uint y, uint k) returns (uint)- (x + y)% k değerini hesaplar, burada toplama keyfi bir hassasiyetle gerçekleştirilir ve 2 256'da sarmaz .
mulmod(uint x, uint y, uint k) returns (uint)- (x * y)% k'yi hesaplar, burada toplama keyfi bir hassasiyetle gerçekleştirilir ve 2 256'da sarmaz .
Aşağıdaki örnek, Solidity'deki matematiksel fonksiyonların kullanımını göstermektedir.
Misal
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Önce callAddMod düğmesine tıklayın ve ardından sonucu görmek için CallMulMod düğmesine tıklayın.
Çıktı
0: uint256: 0
0: uint256: 2
Solidity, dahili kriptografik işlevler de sağlar. Aşağıdakiler önemli yöntemlerdir -
keccak256(bytes memory) returns (bytes32) - girişin Keccak-256 hash değerini hesaplar.
sha256(bytes memory) returns (bytes32) - girişin SHA-256 hash değerini hesaplar.
ripemd160(bytes memory) returns (bytes20) - girdinin RIPEMD-160 karmasını hesaplayın.
sha256(bytes memory) returns (bytes32) - girişin SHA-256 hash değerini hesaplar.
ecrecover(bytes32 hash, uint8 v, bytes32 r, bytes32 s) returns (address)- genel anahtarla ilişkili adresi eliptik eğri imzasından kurtarmak veya hata durumunda sıfır döndürmek. Fonksiyon parametreleri imzanın ECDSA değerlerine karşılık gelir: r - imzanın ilk 32 baytı; s: ikinci 32 bayt imza; v: son 1 bayt imza. Bu yöntem bir adres döndürür.
Aşağıdaki örnek, Solidity'de kriptografik fonksiyonun kullanımını göstermektedir.
Misal
pragma solidity ^0.5.0;
contract Test {
function callKeccak256() public pure returns(bytes32 result){
return keccak256("ABC");
}
}
Solidity First Application bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Çıktı
0: bytes32: result 0xe1629b9dda060bb30c7908346f6af189c16773fa148d3366701fbaa35d54f3c8
Para çekme modeli, bir güvenlik tehdidi oluşturan doğrudan transfer çağrısının yapılmamasını sağlar. Aşağıdaki sözleşme, ether göndermek için transfer çağrısının güvensiz kullanımını gösteriyor.
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;
}
}
}
Yukarıdaki sözleşme, en zengin olanın başarısız bir geri dönüş işlevi sözleşmesi olmasına neden olarak kullanılamaz hale getirilebilir. Geri dönüş işlevi başarısız olduğunda, haleRichest () işlevi de başarısız olur ve sözleşme sonsuza kadar takılı kalır. Bu sorunu hafifletmek için Para Çekme Modelini kullanabiliriz.
Para çekme modelinde, bekleyen tutarı her transferden önce sıfırlayacağız. Yalnızca arayan sözleşmesinin başarısız olmasını sağlayacaktır.
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);
}
}
Bir Sözleşmeye Sınırlı Erişim yaygın bir uygulamadır. Varsayılan olarak, bir sözleşme durumu, genel olarak belirtilmediği sürece salt okunurdur.
Sözleşmenin durumunu kimin değiştirebileceğini veya değiştiriciler kullanarak bir sözleşmenin işlevlerini kimin çağırabileceğini kısıtlayabiliriz. Aşağıda açıklandığı gibi birden çok değiştirici oluşturup kullanacağız -
onlyBy - bir işlevde kullanıldığında, yalnızca belirtilen arayan bu işlevi arayabilir.
onlyAfter - bir işlevde kullanıldığında, bu işlev belirli bir süre sonra çağrılabilir.
costs - Bir işlevde kullanıldığında, arayan kişi bu işlevi yalnızca belirli bir değer sağlanmışsa çağırabilir.
Misal
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;
}
}
Solidity'deki Sözleşme, C ++ 'daki bir Sınıfa benzer. Bir Sözleşmenin aşağıdaki özellikleri vardır.
Constructor - Oluşturucu anahtar kelimesi ile bildirilen, sözleşme başına bir kez çalıştırılacak ve bir sözleşme oluşturulduğunda çağrılan özel bir işlev.
State Variables - Sözleşmenin durumunu saklamak için Sözleşme başına Değişkenler.
Functions - Bir sözleşmenin durumunu değiştirmek için durum değişkenlerini değiştirebilen Sözleşme Başına İşlevler.
Görünürlük Niceleyiciler
Aşağıda, bir sözleşmenin işlevleri / durum değişkenleri için çeşitli görünürlük nicelikleri verilmiştir.
external- Harici işlevler başka sözleşmelerle çağrılmak içindir. Dahili arama için kullanılamazlar. Sözleşme içinde harici işlevi çağırmak için this.function_name () çağrısı gereklidir. Durum değişkenleri harici olarak işaretlenemez.
public- Genel işlevler / Değişkenler hem harici hem de dahili olarak kullanılabilir. Genel durum değişkeni için Solidity otomatik olarak bir alıcı işlevi oluşturur.
internal - Dahili fonksiyonlar / Değişkenler yalnızca dahili olarak veya türetilmiş sözleşmelerle kullanılabilir.
private - Özel işlevler / Değişkenler yalnızca dahili olarak kullanılabilir ve türetilmiş sözleşmelerle bile kullanılamaz.
Misal
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın . Çeşitli Sözleşmeler yöntemini çalıştırın. E.getComputedResult () ve ardından E.getResult () şunu gösterir -
Çıktı
0: uint256: 8
Miras, bir sözleşmenin işlevselliğini genişletmenin bir yoludur. Solidity, hem tek hem de çoklu mirası destekler. Aşağıdakiler, önemli vurgulamalar.
Türetilmiş bir sözleşme, dahili yöntemler ve durum değişkenleri dahil olmak üzere tüm özel olmayan üyelere erişebilir. Ancak bunun kullanılmasına izin verilmiyor.
İşlev imzasının aynı kalması koşuluyla işlev geçersiz kılmaya izin verilir. Çıktı parametrelerinin farklı olması durumunda derleme başarısız olacaktır.
Süper sözleşmesi veya süper sözleşme adı kullanarak bir süper sözleşmenin işlevini çağırabiliriz.
Çoklu miras durumunda, super kullanan işlev çağrısı, türetilen sözleşmelerin çoğunu tercih eder.
Misal
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın . Çeşitli Sözleşmeler yöntemini çalıştırın. E.getComputedResult () ve ardından E.getResult () şunu gösterir -
Çıktı
0: uint256: 8
Yapıcı, kullanılarak bildirilen özel bir işlevdir constructoranahtar kelime. İsteğe bağlı bir işlevdir ve bir sözleşmenin durum değişkenlerini başlatmak için kullanılır. Bir kurucunun temel özellikleri aşağıdadır.
Bir sözleşmede yalnızca bir kurucu olabilir.
Bir sözleşme oluşturulduğunda bir kurucu kodu çalıştırılır ve sözleşme durumunu başlatmak için kullanılır.
Bir yapıcı kodu çalıştırıldıktan sonra, son kod blok zincirine dağıtılır. Bu kod, genel işlevleri ve genel işlevler aracılığıyla erişilebilen kodu içerir. Oluşturucu kodu veya yalnızca kurucu tarafından kullanılan herhangi bir dahili yöntem son koda dahil edilmez.
Bir kurucu genel veya dahili olabilir.
Dahili bir kurucu, sözleşmeyi soyut olarak işaretler.
Herhangi bir kurucu tanımlanmamışsa, sözleşmede varsayılan bir kurucu bulunur.
pragma solidity ^0.5.0;
contract Test {
constructor() public {}
}
Temel sözleşmenin bağımsız değişkenlere sahip kurucuya sahip olması durumunda, türetilen her sözleşme bunları geçmelidir.
Temel oluşturucu, aşağıdaki şekilde doğrudan başlatılabilir -
pragma solidity ^0.5.0;
contract Base {
uint data;
constructor(uint _data) public {
data = _data;
}
}
contract Derived is Base (5) {
constructor() public {}
}
Temel oluşturucu, aşağıdaki şekilde dolaylı olarak başlatılabilir -
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 {}
}
Temel sözleşme kurucusunu başlatmanın doğrudan ve dolaylı yollarına izin verilmez.
Türetilen sözleşme, bağımsız değişkenleri temel sözleşme yapıcısına iletmiyorsa, türetilen sözleşme soyut hale gelecektir.
Özet Sözleşme, herhangi bir uygulama olmaksızın en az bir işlev içeren bir sözleşmedir. Böyle bir sözleşme, temel sözleşme olarak kullanılır. Genel olarak soyut bir sözleşme, hem uygulanan hem de soyut işlevleri içerir. Türetilmiş sözleşme, soyut işlevi uygulayacak ve gerektiğinde mevcut işlevleri kullanacaktır.
Türetilmiş bir sözleşmenin soyut işlevi uygulamaması durumunda, bu türetilmiş sözleşme soyut olarak işaretlenecektir.
Misal
Solidity'de soyut sözleşmelerin nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Çıktı
0: uint256: 3
Arayüzler soyut sözleşmelere benzer ve interfaceanahtar kelime. Bir arayüzün temel özellikleri aşağıdadır.
Arayüz uygulama ile herhangi bir işleve sahip olamaz.
Bir arayüzün işlevleri yalnızca harici tipte olabilir.
Arayüz kurucusuna sahip olamaz.
Arayüzde durum değişkenleri olamaz.
Arabirim, arabirim adı nokta gösterimi kullanılarak erişilebilen enum, yapılara sahip olabilir.
Misal
Solidity'de arayüzün nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Note - Dağıt düğmesine tıklamadan önce açılır menüden Test'i seçin.
Çıktı
0: uint256: 3
Kitaplıklar Sözleşmelere benzer ancak esas olarak yeniden kullanım amaçlıdır. Bir Kitaplık, diğer sözleşmelerin çağırabileceği işlevleri içerir. Solidity'nin bir Kütüphane kullanımıyla ilgili belirli kısıtlamaları vardır. Bir Solidity Kitaplığının temel özellikleri aşağıdadır.
Durumu değiştirmezlerse kütüphane işlevleri doğrudan çağrılabilir. Bu, saf veya görünüm işlevlerinin yalnızca kitaplık dışından çağrılabileceği anlamına gelir.
Vatansız olduğu varsayıldığı için kütüphane imha edilemez.
Bir Kitaplığın durum değişkenleri olamaz.
Bir Kitaplık herhangi bir öğeyi devralamaz.
Bir Kitaplık miras alınamaz.
Misal
Solidity'de bir Kitaplığın nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Note - Dağıt düğmesine tıklamadan önce açılır menüden Test'i seçin.
Çıktı
0: uint256: 3
İçin kullanılır
Direktif using A for B; A kitaplığının kitaplık işlevlerini belirli bir B türüne eklemek için kullanılabilir. Bu işlevler, arayan türünü ilk parametreleri olarak kullanır (self kullanılarak tanımlanır).
Misal
Solidity'de bir Kitaplığın nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Note - Dağıt düğmesine tıklamadan önce açılır menüden Test'i seçin.
Çıktı
0: uint256: 3
Solidity, Solidity kaynak kodu içinde satır içi montajı yazmak için montaj dilini kullanma seçeneği sunar. Ayrıca, daha sonra bayt koduna dönüştürülebilecek bağımsız bir montaj kodu da yazabiliriz. Bağımsız Derleme, bir Solidity derleyicisi için ara dildir ve Solidity kodunu Tek Başına Bir Derlemeye ve ardından bayt koduna dönüştürür. Bağımsız bir derlemede kod yazmak için Satır İçi Derlemede kullanılanla aynı dili kullanabiliriz.
Satır İçi Montaj
Satır içi montaj kodu, EVM üzerinde daha ince taneli kontrole sahip olmak için Solidity kod tabanı içinde serpiştirilebilir ve özellikle kütüphane fonksiyonlarını yazarken kullanılır.
Bir montaj kodu yazılır assembly { ... } blok.
Misal
Solidity'de bir Kitaplığın nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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 bölümünde verilen adımları kullanarak yukarıdaki programı çalıştırın .
Note - Dağıt düğmesine tıklamadan önce açılır menüden Test'i seçin.
Çıktı
0: uint256: 15
Olay, bir sözleşmenin miras alınabilen bir üyesidir. Bir olay yayınlanır, aktarılan argümanları işlem günlüklerinde saklar. Bu günlükler blok zincirinde saklanır ve sözleşme blok zincirinde mevcut olana kadar sözleşmenin adresi kullanılarak erişilebilir. Oluşturulan bir olaya, onları yaratan ve yayan olandan bile sözleşmelerden erişilemez.
Bir olay, event anahtar sözcüğü kullanılarak bildirilebilir.
//Declare an Event
event Deposit(address indexed _from, bytes32 indexed _id, uint _value);
//Emit an event
emit Deposit(msg.sender, _id, msg.value);
Misal
Solidity'de bir olayın nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
Önce bir sözleşme oluşturun ve bir etkinlik yayınlayın.
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);
}
}
Ardından, sözleşmenin etkinliğine JavaScript kodunda erişin.
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);
});
Aşağıdakine benzer ayrıntıları yazdırmalıdır -
Çıktı
{
"returnValues": {
"_from": "0x1111...FFFFCCCC",
"_id": "0x50...sd5adb20",
"_value": "0x420042"
},
"raw": {
"data": "0x7f...91385",
"topics": ["0xfd4...b4ead7", "0x7f...1a91385"]
}
}
Solidity, hata işleme için çeşitli işlevler sağlar. Genellikle bir hata meydana geldiğinde, durum orijinal durumuna geri döndürülür. Diğer kontroller, yetkisiz kod erişimini önlemek içindir. Hata işlemede kullanılan önemli yöntemlerden bazıları aşağıdadır -
assert(bool condition)- Koşulun karşılanmaması durumunda, bu yöntem çağrısı geçersiz bir işlem koduna neden olur ve duruma yapılan herhangi bir değişiklik geri alınır. Bu yöntem, dahili hatalar için kullanılacaktır.
require(bool condition)- Koşulun karşılanmaması durumunda, bu yöntem çağrısı orijinal durumuna geri döner. - Bu yöntem, girişlerdeki veya harici bileşenlerdeki hatalar için kullanılacaktır.
require(bool condition, string memory message)- Koşulun karşılanmaması durumunda, bu yöntem çağrısı orijinal durumuna geri döner. - Bu yöntem, girişlerdeki veya harici bileşenlerdeki hatalar için kullanılacaktır. Özel bir mesaj sağlama seçeneği sunar.
revert() - Bu yöntem, yürütmeyi durdurur ve duruma yapılan değişiklikleri geri döndürür.
revert(string memory reason)- Bu yöntem, yürütmeyi durdurur ve duruma yapılan değişiklikleri geri döndürür. Özel bir mesaj sağlama seçeneği sunar.
Misal
Solidity'de hata işlemenin nasıl çalıştığını anlamak için aşağıdaki kodu deneyin.
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.
}
}
Geri dönüş çağrıldığında, aşağıdaki gibi onaltılık verileri döndürür.
Çıktı
0x08c379a0 // Function selector for Error(string)
0x0000000000000000000000000000000000000000000000000000000000000020 // Data offset
0x000000000000000000000000000000000000000000000000000000000000001a // String length
0x4e6f7420656e6f7567682045746865722070726f76696465642e000000000000 // String data