diff --git a/404.html b/404.html index b731a35a..93fb4fd3 100644 --- a/404.html +++ b/404.html @@ -10,13 +10,13 @@ - +
Skip to main content

Page Not Found

We could not find what you were looking for.

Please contact the owner of the site that linked you to the original URL and let them know their link is broken.

- + \ No newline at end of file diff --git a/CONTRIBUTING.html b/CONTRIBUTING.html index 8529517b..5a544e4b 100644 --- a/CONTRIBUTING.html +++ b/CONTRIBUTING.html @@ -8,13 +8,13 @@ - + - +
-
Skip to main content

Contribution & Tech Stack

Ebook Dasar Pemrograman Rust adalah project open source. Siapapun bebas untuk berkontribusi di sini, bisa dalam bentuk perbaikan typo, update kalimat, maupun submit tulisan baru.

Bagi kawan-kawan yang berminat untuk berkontribusi, silakan fork github.com/novalagung/dasarpemrogramanrust, kemudian langsung saja cek/buat issue kemudian submit relevan pull request untuk issue tersebut 😊

Checkout project

git clone https://github.com/novalagung/dasarpemrogramanrust.git
git submodule update --init --recursive --remote

Maintainer

E-book ini di-inisialisasi dan di-maintain oleh Noval Agung Prayogo.

Contributors

Berikut merupakan hall of fame kontributor yang sudah berbaik hati menyisihkan waktunya untuk membantu pengembangan e-book ini.

  1. Acep Saepudin
  2. Afifurrohman
  3. Fal
  4. Fredianto
  5. I Putu Saputrayana
  6. Khisby Al Ghofari
  7. Lori
  8. M. Zakiyuddin Munziri
  9. Opa Kholis Majid
  10. Rian
  11. Shandy Siswandi
  12. Theis Andatu
  13. ... anda :-)

Ebook/webhook techstack & architecture

Jika pembaca tertarik untuk membuat konten ebook berbasis web sekaligus versi PDF file-nya, bisa menggunakan link berikut sebagai referensi:

https://medium.com/geekculture/serverless-e-book-web-book-using-docusaurus-v2-github-pages-actions-and-pdf-tools-4fef54847b85

Tools yang digunakan:

Desain arsitektur:

Dasar pemrograman python

- +
Skip to main content

Contribution & Tech Stack

Ebook Dasar Pemrograman Rust adalah project open source. Siapapun bebas untuk berkontribusi di sini, bisa dalam bentuk perbaikan typo, update kalimat, maupun submit tulisan baru.

Bagi kawan-kawan yang berminat untuk berkontribusi, silakan fork github.com/novalagung/dasarpemrogramanrust, kemudian langsung saja cek/buat issue kemudian submit relevan pull request untuk issue tersebut 😊

Checkout project

git clone https://github.com/novalagung/dasarpemrogramanrust.git
git submodule update --init --recursive --remote

Maintainer

E-book ini di-inisialisasi dan di-maintain oleh Noval Agung Prayogo.

Contributors

Berikut merupakan hall of fame kontributor yang sudah berbaik hati menyisihkan waktunya untuk membantu pengembangan e-book ini.

  1. Acep Saepudin
  2. Afifurrohman
  3. ardiantovn
  4. Fal
  5. Fredianto
  6. I Putu Saputrayana
  7. Khisby Al Ghofari
  8. Lori
  9. M. Zakiyuddin Munziri
  10. Opa Kholis Majid
  11. Rian
  12. Shandy Siswandi
  13. Theis Andatu
  14. ... anda :-)

Ebook/webhook techstack & architecture

Jika pembaca tertarik untuk membuat konten ebook berbasis web sekaligus versi PDF file-nya, bisa menggunakan link berikut sebagai referensi:

https://medium.com/geekculture/serverless-e-book-web-book-using-docusaurus-v2-github-pages-actions-and-pdf-tools-4fef54847b85

Tools yang digunakan:

Desain arsitektur:

Dasar pemrograman python

+ \ No newline at end of file diff --git a/LICENSE.html b/LICENSE.html index e95dc8e8..0dd57244 100644 --- a/LICENSE.html +++ b/LICENSE.html @@ -8,13 +8,13 @@ - + - +
Skip to main content

Lisensi & Distribusi Konten

Ebook Dasar Pemrograman Rust gratis untuk disebarluaskan secara bebas, dengan catatan sesuai dengan aturan lisensi CC BY-SA 4.0 yang kurang lebih sebagai berikut:

  • Diperbolehkan menyebar, mencetak, dan menduplikasi material dalam konten ini ke siapapun.
  • Diperbolehkan memodifikasi, mengubah, atau membuat konten baru menggunakan material yang ada dalam ebook ini untuk keperluan komersil maupun tidak.

Dengan catatan:

  • Harus ada credit sumber aslinya, yaitu Dasar Pemrograman Rust atau novalagung
  • Tidak mengubah lisensi aslinya, yaitu CC BY-SA 4.0
  • Tidak ditambahi restrictions baru
  • Lebih jelasnya silakan cek https://creativecommons.org/licenses/by-sa/4.0/.

FOSSA Status

- + \ No newline at end of file diff --git a/assets/js/016a0787.0ab1d3ab.js b/assets/js/016a0787.0ab1d3ab.js deleted file mode 100644 index a78136a1..00000000 --- a/assets/js/016a0787.0ab1d3ab.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[274],{3905:(a,e,t)=>{t.d(e,{Zo:()=>o,kt:()=>u});var n=t(7294);function r(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function i(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function m(a){for(var e=1;e=0||(r[t]=a[t]);return r}(a,e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(r[t]=a[t])}return r}var p=n.createContext({}),k=function(a){var e=n.useContext(p),t=e;return a&&(t="function"==typeof a?a(e):m(m({},e),a)),t},o=function(a){var e=k(a.components);return n.createElement(p.Provider,{value:e},a.children)},d={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},s=n.forwardRef((function(a,e){var t=a.components,r=a.mdxType,i=a.originalType,p=a.parentName,o=l(a,["components","mdxType","originalType","parentName"]),s=k(t),u=r,g=s["".concat(p,".").concat(u)]||s[u]||d[u]||i;return t?n.createElement(g,m(m({ref:e},o),{},{components:t})):n.createElement(g,m({ref:e},o))}));function u(a,e){var t=arguments,r=e&&e.mdxType;if("string"==typeof a||r){var i=t.length,m=new Array(i);m[0]=s;var l={};for(var p in e)hasOwnProperty.call(e,p)&&(l[p]=e[p]);l.originalType=a,l.mdxType="string"==typeof a?a:r,m[1]=l;for(var k=2;k{t.r(e),t.d(e,{assets:()=>p,contentTitle:()=>m,default:()=>d,frontMatter:()=>i,metadata:()=>l,toc:()=>k});var n=t(7462),r=(t(7294),t(3905));const i={sidebar_position:32,title:"A.32. Basic Memory Management",sidebar_label:"A.32. Basic Memory Management"},m=void 0,l={unversionedId:"basic/basic-memory-management",id:"basic/basic-memory-management",title:"A.32. Basic Memory Management",description:"Pada chapter ini kita akan belajar tentang salah satu hal penting dalam topik pemrograman secara general, yaitu memory management yang mencakup pembahasan tentang alamat memori, pointer, heap, stack, dan juga error-error yang terjadi karena kesalahan dalam manajemen memori.",source:"@site/docs/basic/basic-memory-management.md",sourceDirName:"basic",slug:"/basic/basic-memory-management",permalink:"/basic/basic-memory-management",draft:!1,tags:[],version:"current",sidebarPosition:32,frontMatter:{sidebar_position:32,title:"A.32. Basic Memory Management",sidebar_label:"A.32. Basic Memory Management"},sidebar:"tutorialSidebar",previous:{title:"A.31. Shadowing",permalink:"/basic/shadowing"},next:{title:"A.33. Pointer & References",permalink:"/basic/pointer-references"}},p={},k=[{value:"A.32.1. Memory management",id:"a321-memory-management",level:2},{value:"\u25c9 Garbage collection (GC)",id:"-garbage-collection-gc",level:3},{value:"\u25c9 Automatic reference counting (ARC)",id:"-automatic-reference-counting-arc",level:3},{value:"\u25c9 Manual memory management",id:"-manual-memory-management",level:3},{value:"\u25c9 Ownership rules",id:"-ownership-rules",level:3},{value:"A.32.2. Memory Address",id:"a322-memory-address",level:2},{value:"A.32.3. Stack memory",id:"a323-stack-memory",level:2},{value:"\u25c9 Contoh ke-1",id:"-contoh-ke-1",level:3},{value:"\u25c9 Contoh ke-2",id:"-contoh-ke-2",level:3},{value:"A.32.4. Heap memory",id:"a324-heap-memory",level:2},{value:"A.32.5. Error memory management",id:"a325-error-memory-management",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:k};function d(a){let{components:e,...i}=a;return(0,r.kt)("wrapper",(0,n.Z)({},o,i,{components:e,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Pada chapter ini kita akan belajar tentang salah satu hal penting dalam topik pemrograman secara general, yaitu ",(0,r.kt)("em",{parentName:"p"},"memory management")," yang mencakup pembahasan tentang alamat memori, pointer, heap, stack, dan juga error-error yang terjadi karena kesalahan dalam manajemen memori."),(0,r.kt)("p",null,"Di bahasa pemrograman high-level biasanya topik tersebut jarang disentuh, tetapi di Rust yang notabene adalah system programming, hal di atas wajib untuk dipelajari."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Penulis tekankan bahwa mungkin pembelajaran pada bagian ini akan terasa seperti ",(0,r.kt)("em",{parentName:"p"},"oversimplified")," karena tujuannya adalah untuk para pembaca yang masih dalam proses belajar atau malah belum mengenal sama sekali tentang manajemen memori.")),(0,r.kt)("h2",{id:"a321-memory-management"},"A.32.1. Memory management"),(0,r.kt)("p",null,"Semua bahasa pemrograman memiliki caranya sendiri dalam melakukan pengelolaan memory atau memory management. Ada beberapa macam metode manajemen memori yang diterapkan pada bahasa pemrograman, di antaranya adalah berikut:"),(0,r.kt)("h3",{id:"-garbage-collection-gc"},"\u25c9 Garbage collection (GC)"),(0,r.kt)("p",null,"GC adalah metode manajemen memori otomatis pada bahasa pemrograman. GC memiliki suatu unit yang disebut dengan ",(0,r.kt)("em",{parentName:"p"},"garbage collector"),". Collector tersebut aktif memonitor program, dan pada periode atau event tertentu ia akan berusaha untuk mengambil kembali (reclaim) memory yang sebelumnya telah dialokasikan dengan catatan memori tersebut sudah tidak lagi digunakan. Proses ini disebut dengan dealokasi memory."),(0,r.kt)("p",null,"Proses dealokasi pada GC terjadi di belakang layar secara asynchronous."),(0,r.kt)("p",null,"Beberapa bahasa pemrograman yang menerapkan GC di antara adalah Java, C#, Go, Lisp, dan banyak bahasa lainnya."),(0,r.kt)("h3",{id:"-automatic-reference-counting-arc"},"\u25c9 Automatic reference counting (ARC)"),(0,r.kt)("p",null,"ARC adalah metode manajemen memori yang diterapkan pada bahasa Objective-C dan Swift. Cara ARC me-manage memory adalah dengan mencatat ",(0,r.kt)("em",{parentName:"p"},"reference")," object dan segala aktifitas yang terjadi pada object tersebut."),(0,r.kt)("p",null,"Di ARC, ada satuan yang disebut dengan ",(0,r.kt)("em",{parentName:"p"},"retain count")," yang merupakan representasi jumlah banyaknya variabel atau object yang memegang suatu ",(0,r.kt)("em",{parentName:"p"},"reference"),". Ketika ",(0,r.kt)("em",{parentName:"p"},"reference")," sudah pindah ke luar scope atau dihapus isinya dan dilihat pada catatan rupanya tidak ada variabel yang memegang ",(0,r.kt)("em",{parentName:"p"},"reference")," tersebut, maka dilakukan proses dealokasi memory."),(0,r.kt)("p",null,"Dalam bahasa yang menerapkan ARC, programmer dianjurkan untuk perhatian dan bijak dalam pengalokasian variabel beserta nilainya. Mana data yang diperlukan untuk di-retain secara ",(0,r.kt)("em",{parentName:"p"},"strong")," dan mana yang tidak, harus pas sesuai dengan kebutuhan. Jika tidak hati-hati maka program mempunyai resiko lebih tinggi untuk menemui error ",(0,r.kt)("em",{parentName:"p"},"deadlocks")," ataupun ",(0,r.kt)("em",{parentName:"p"},"memory leaks")," (yang juga akan dibahas pada chapter ini)."),(0,r.kt)("h3",{id:"-manual-memory-management"},"\u25c9 Manual memory management"),(0,r.kt)("p",null,"Manual memory management berarti programmer dibebani secara penuh dalam hal manajemen memori, mengharuskan programmer untuk super hati-hati dalam pengalokasian memory, kapan waktunya, dimana alokasinya (apakah ",(0,r.kt)("em",{parentName:"p"},"heap")," atau ",(0,r.kt)("em",{parentName:"p"},"stack"),"), dan kapan harus melakukan operasi dealokasi memory."),(0,r.kt)("p",null,"Metode manajemen memori ini dipakai dalam system programming contohnya bahasa C dan C++."),(0,r.kt)("h3",{id:"-ownership-rules"},"\u25c9 Ownership rules"),(0,r.kt)("p",null,"Manajemen memori yang dilakukan dengan menerapkan konsep ",(0,r.kt)("em",{parentName:"p"},"ownership")," beserta aturan-aturannya. Metode manajemen memori ini adalah yang digunakan di Rust."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai ownership rules pada Rust dibahas pada chapter selanjutnya, yaitu ",(0,r.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership"),". Untuk sekarang silakan selesaikan terlebih dahulu pembahasan chapter ini.")),(0,r.kt)("h2",{id:"a322-memory-address"},"A.32.2. Memory Address"),(0,r.kt)("p",null,"Memory address atau alamat memori adalah sebuah lokasi spesifik di memori yang digunakan oleh software maupun hardware untuk menyimpan suatu data."),(0,r.kt)("p",null,"Pembahasan mengenai memory address ini sangatlah luas. Pada chapter ini kita tidak akan membahasnya secara menyeluruh, melainkan hanya poin penting yang perlu diketahui dan dibutuhkan dalam proses pembelajaran."),(0,r.kt)("p",null,"Ok lanjut ke contoh agar lebih jelas. Dimisalkan ada sebuah variabel bertipe data numerik ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", variabel tersebut akan membutuhkan sejumlah bit alokasi alamat memori untuk bisa menyimpan value-nya yang pada contoh ini adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"32 bits")," (karena tipenya ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),") atau jika dikonversi ke bentuk ",(0,r.kt)("em",{parentName:"p"},"bytes")," adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"4 bytes"),"."),(0,r.kt)("p",null,"Alokasi memory address mengacu ke tipe data (bukan value), sebagai contoh pada data bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"i32")," maka berapapun value data tersebut (entah ",(0,r.kt)("inlineCode",{parentName:"p"},"1"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"2"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"1999999"),", atau lainnya) tetap membutuhkan ",(0,r.kt)("inlineCode",{parentName:"p"},"32 bits")," alokasi alamat memori untuk menyimpan data tersebut."),(0,r.kt)("p",null,"Silakan perhatikan ilustrasi berikut agar lebih jelas. Gambar diambil dari post ",(0,r.kt)("a",{parentName:"p",href:"https://medium.com/@luischaparroc/integer-numbers-storage-in-computer-memory-47af4b59009"},"medium.com/@luischaparroc"),"."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Memory address",src:t(4275).Z,width:"875",height:"477"})),(0,r.kt)("p",null,"Pada contoh di atas, variabel adalah bertipe data ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", maka di memory dialokasikanlah alamat memory dengan lebar 32 bit."),(0,r.kt)("p",null,"Nilai variabel tersebut adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"25"),", yang jika dikonversi ke bentuk binary adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"11001"),". Dengan ini maka pada 32 bit yang sudah dialokasikan, akan terisi dengan nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"11001"),". Penulisannya dari kanan dan jika ada slot kosong sebelah kiri maka terisi dengan ",(0,r.kt)("inlineCode",{parentName:"p"},"0"),"."),(0,r.kt)("p",null,"Hasilnya adalah angka biner berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"00000000000000000000000000011001\n")),(0,r.kt)("h2",{id:"a323-stack-memory"},"A.32.3. Stack memory"),(0,r.kt)("p",null,"Masih dalam topik manajemen memori, ada dua hal lagi yang sangat penting untuk diketahui, yaitu ",(0,r.kt)("em",{parentName:"p"},"stack")," dan ",(0,r.kt)("em",{parentName:"p"},"heap"),". Keduanya adalah bagian dari memory, tempat dimana alokasi dilakukan."),(0,r.kt)("p",null,"Data disimpan dalam stack memory dalam bentuk stack. Karakteristik dari stack:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Di Rust programming, stack digunakan sebagai default tempat alokasi memori"),(0,r.kt)("li",{parentName:"ul"},"Data yang terakhir masuk adalah yang pertama akan keluar (LIFO)"),(0,r.kt)("li",{parentName:"ul"},"Data yang disimpan diketahui size/ukurannya, dan memiliki batas"),(0,r.kt)("li",{parentName:"ul"},"Alokasi bersifat lokal terhadap pemanggilan fungsi"),(0,r.kt)("li",{parentName:"ul"},"Kecepatan pengaksesan data sangat tinggi")),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Stack memory",src:t(9929).Z,width:"500",height:"345"})),(0,r.kt)("p",null,"Data untuk tipe primitif (seperti ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"bool"),", dll) disimpan di stack."),(0,r.kt)("h3",{id:"-contoh-ke-1"},"\u25c9 Contoh ke-1"),(0,r.kt)("p",null,"Selanjutnya kita akan pelajari secara garis besar tentang bagaimana sebuah data dialokasikan di stack. Silakan mulai dengan mempelajari kode sederhana berikut (tanpa perlu dipraktikkan), kemudian ikuti pembahasan setelahnya."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n let x = 24;\n}\n")),(0,r.kt)("p",null,"Fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," hanya berisi 1 buah data, yaitu variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," dengan nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),". Ketika fungsi tersebut dipanggil, maka data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," disimpan di stack (karena di Rust by default semua data disimpan di stack). Nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," adalah numerik bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", maka compiler akan mengalokasikan 32 bits di stack memory untuk menyimpan nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),"."),(0,r.kt)("p",null,"Pada catatan karakteristik stack di atas, telah disinggung bahwa alokasi data di stack bersifat lokal terhadap pemanggilan fungsi, artinya apa? \u279c Semua data dalam suatu blok fungsi akan disimpan dalam sebuah group yang disebut dengan ",(0,r.kt)("em",{parentName:"p"},"stack frame"),"."),(0,r.kt)("p",null,"Pada contoh di atas, ketika fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," dipanggil, sebuah stack frame terbuat, dan data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," disimpan dalam stack frame tersebut."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Kolom ",(0,r.kt)("inlineCode",{parentName:"p"},"No.")," di atas merepresentasikan urutan data dalam stack agar lebih mudah dipahami")),(0,r.kt)("h3",{id:"-contoh-ke-2"},"\u25c9 Contoh ke-2"),(0,r.kt)("p",null,"Ok, sekarang mari lanjut contoh ke-2 berikut ini agar makin jelas."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something() {\n let y = 13;\n let z = 11;\n}\n\nfn main() {\n let x = 24;\n do_something();\n let a = 4;\n let b = 18;\n}\n")),(0,r.kt)("p",null,"Program sederhana di atas memiliki dua buah fungsi, ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," dan ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something"),". Saat program dijalankan, lebih tepatnya saat fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," dipanggil maka sebuah stack frame dibuat dan data dalam fungsi tersebut dialokasikan pada stack frame."),(0,r.kt)("p",null,"Perlu diketahui bahwa eksekusi statement dalam fungsi adalah per baris, dimulai dari atas. Dengan ini maka data yang pertama dialokasikan ke memory adalah variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"x"),"."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Setelah itu, lanjut ke statement ke-2 yaitu pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something"),"."),(0,r.kt)("p",null,"Kembali ke teori, bahwa alokasi data stack adalah bersifat lokal terhadap pemanggilan fungsi. Maka dibuatlah stack frame baru untuk pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something")," dengan isi adalah alokasi data ",(0,r.kt)("inlineCode",{parentName:"p"},"y")," dan ",(0,r.kt)("inlineCode",{parentName:"p"},"z"),"."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"2"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"z"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"11"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"strong"},"do_something()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"1"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"y"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"13"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"strong"},"do_something()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Setelah eksekusi blok kode do_something selesai maka stack frame pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something")," akan di-dealokasi, dikosongkan, dihapus."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Sebenarnya tidak se-sederhana itu proses dealokasi memori, ada pengecekan ownership yang harus dilakukan terlebih dahulu. Namun agar makin tidak bingung, untuk sementara kita gunakan penjelasan di atas."),(0,r.kt)("p",{parentName:"blockquote"},"Silakan selesaikan dulu pembahasan chapter ini, kemudian pada chapter selanjutnya kita akan bahas tentang apa itu ",(0,r.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership"),".")),(0,r.kt)("p",null,"Ok, maka dari 2 stack yang sebelumnya ada, sekarang tinggal 1 stack frame saja yaitu milik pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,r.kt)("p",null,"Dari yang sebelumnya ..."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"2"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"z"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"11"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"del"},"do_something()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"1"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"y"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"13"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"del"},"do_something()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"... sekarang menjadi ..."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Ok, lanjut ke block fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," berikutnya, yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"let a = 4"),". Saat dipanggil maka ada penambahan data baru pada stack frame pertama."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"3"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"a"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"4"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"strong"},"main()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Kemudian sampai di statement terakhir fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),". Ketika ",(0,r.kt)("inlineCode",{parentName:"p"},"let b = 14")," dipanggil maka ada penambahan data baru pada stack frame pertama."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"4"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"b"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"18"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"strong"},"main()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"3"),(0,r.kt)("td",{parentName:"tr",align:"center"},"a"),(0,r.kt)("td",{parentName:"tr",align:"center"},"4"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Setelah fungsi selesai dieksekusi, stack frame di-dealokasi."),(0,r.kt)("p",null,"Kurang lebih seperti itu sekilas peran dari stack dalam Rust programming. Selanjutnya kita bahas tentang heap memory."),(0,r.kt)("h2",{id:"a324-heap-memory"},"A.32.4. Heap memory"),(0,r.kt)("p",null,"Heap adalah salah satu tempat alokasi memory selain stack. Karakteristik dari heap:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Heap digunakan untuk alokasi data yang sifatnya dinamis, tidak diketahui size-nya, atau bisa berubah size-nya"),(0,r.kt)("li",{parentName:"ul"},"Data di heap tidak memiliki pattern tertentu"),(0,r.kt)("li",{parentName:"ul"},"Alokasi dan dealokasi data di heap bisa dilakukan kapanpun"),(0,r.kt)("li",{parentName:"ul"},"Kecepatan pengaksesan data di heap lebih lambat dibanding stack")),(0,r.kt)("p",null,"Ilustrasi perbandingan stack dan heap:"),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Heap memory",src:t(6907).Z,width:"414",height:"236"})),(0,r.kt)("p",null,"Tipe data non-primitive di Rust data-nya disimpan di heap, contohnya seperti Vector, ",(0,r.kt)("inlineCode",{parentName:"p"},"String"),", dan beberapa lainnya. Penulis tekankan, bahwa ",(0,r.kt)("strong",{parentName:"p"},"yang disimpan di heap adalah data-nya saja, sedangkan atribut lainnya (seperti ",(0,r.kt)("inlineCode",{parentName:"strong"},"length")," dan ",(0,r.kt)("inlineCode",{parentName:"strong"},"capacity"),") tetap disimpan disimpan di stack"),". Lebih jelasnya silakan lihat ilustrasi berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"let numbers = vec![12, 24, 7];\n")),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Heap memory",src:t(6033).Z,width:"420",height:"184"})),(0,r.kt)("p",null,"Heap, selain digunakan sebagai penyimpanan data dinamis, kita juga bisa manfaatkan sebagai penyimpanan data jenis lainnya (secara eksplisit) menggunakan tipe data ",(0,r.kt)("inlineCode",{parentName:"p"},"Box"),"."),(0,r.kt)("p",null,"Pembahasan Lebih jelasnya mengenai heap allocation ada pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/string-slice-vs-string-literal"},"String Custom Type vs ",(0,r.kt)("inlineCode",{parentName:"a"},"&str"))," dan juga ",(0,r.kt)("a",{parentName:"p",href:"#/wip/box"},"Box"),". Namun untuk sekarang, penulis anjurkan untuk lanjut ke pembahasan berikutnya terlebih dahulu."),(0,r.kt)("h2",{id:"a325-error-memory-management"},"A.32.5. Error memory management"),(0,r.kt)("p",null,"Rust sangat disiplin dalam hal penulisan source code, terutama untuk kode-kode yang berhubungan dengan memory management. Hal ini dilakukan oleh Rust untuk meminimalisir munculnya error seperti memory leak dan sejenisnya."),(0,r.kt)("p",null,"Namun meski demikian, potensi error memory tetap ada, dan kita akan bahas itu nantinya setelah masuk chapter ",(0,r.kt)("a",{parentName:"p",href:"#/wip/safe-unsafe"},"Safe & Unsafe"),"."),(0,r.kt)("p",null,"Nantinya akan dibahas juga tentang beberapa error saat compile time yang error tersebut berhubungan dengan memory management, yaitu pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership")," dan ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),". Untuk sekarang, mari lanjut ke chapter berikutnya terlebih dahulu."),(0,r.kt)("hr",null),(0,r.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,r.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,r.kt)("pre",null,(0,r.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/memory_management"},"github.com/novalagung/dasarpemrogramanrust-example/../memory_management")),(0,r.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/nomicon/ownership.html"},"https://doc.rust-lang.org/nomicon/ownership.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-01-what-is-ownership.html"},"https://doc.rust-lang.org/book/ch04-01-what-is-ownership.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/1.22.0/book/first-edition/the-stack-and-the-heap.html"},"https://doc.rust-lang.org/1.22.0/book/first-edition/the-stack-and-the-heap.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/std/box.html"},"https://doc.rust-lang.org/rust-by-example/std/box.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://docs.swift.org/swift-book/LanguageGuide/AutomaticReferenceCounting.html"},"https://docs.swift.org/swift-book/LanguageGuide/AutomaticReferenceCounting.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://docs.elementscompiler.com/Concepts/ARC/"},"https://docs.elementscompiler.com/Concepts/ARC/")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Memory_address"},"https://en.wikipedia.org/wiki/Memory_address")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Code_segment"},"https://en.wikipedia.org/wiki/Code_segment")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Automatic_Reference_Counting"},"https://en.wikipedia.org/wiki/Automatic_Reference_Counting")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Garbage_collection_(computer_science)"},"https://en.wikipedia.org/wiki/Garbage_collection_(computer_science)")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://log2base2.com/C/pointer/computer-memory-address-basics.html"},"https://log2base2.com/C/pointer/computer-memory-address-basics.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://quora.com/How-does-memory-management-work-in-Rust"},"https://quora.com/How-does-memory-management-work-in-Rust")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://medium.com/the-legend/management-memori-pada-bahasa-pemrograman-a33772635aa5"},"https://medium.com/the-legend/management-memori-pada-bahasa-pemrograman-a33772635aa5")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://imam.digmi.id/post/belajar-rust-memahami-stack-dan-heap/"},"https://imam.digmi.id/post/belajar-rust-memahami-stack-dan-heap/")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://javatpoint.com/rust-ownership"},"https://javatpoint.com/rust-ownership")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/24158114"},"https://stackoverflow.com/questions/24158114"))))}d.isMDXComponent=!0},4275:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/basic-memory-management-1-ac98989ab48847e58470100c02455dab.png"},9929:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/basic-memory-management-2-114833fdf1f1ac3114d03494e6603ff1.jpg"},6907:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},6033:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/016a0787.ddd4552f.js b/assets/js/016a0787.ddd4552f.js new file mode 100644 index 00000000..ed6c3df1 --- /dev/null +++ b/assets/js/016a0787.ddd4552f.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[274],{3905:(a,e,t)=>{t.d(e,{Zo:()=>o,kt:()=>u});var n=t(7294);function r(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function i(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function m(a){for(var e=1;e=0||(r[t]=a[t]);return r}(a,e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(r[t]=a[t])}return r}var p=n.createContext({}),k=function(a){var e=n.useContext(p),t=e;return a&&(t="function"==typeof a?a(e):m(m({},e),a)),t},o=function(a){var e=k(a.components);return n.createElement(p.Provider,{value:e},a.children)},d={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},s=n.forwardRef((function(a,e){var t=a.components,r=a.mdxType,i=a.originalType,p=a.parentName,o=l(a,["components","mdxType","originalType","parentName"]),s=k(t),u=r,g=s["".concat(p,".").concat(u)]||s[u]||d[u]||i;return t?n.createElement(g,m(m({ref:e},o),{},{components:t})):n.createElement(g,m({ref:e},o))}));function u(a,e){var t=arguments,r=e&&e.mdxType;if("string"==typeof a||r){var i=t.length,m=new Array(i);m[0]=s;var l={};for(var p in e)hasOwnProperty.call(e,p)&&(l[p]=e[p]);l.originalType=a,l.mdxType="string"==typeof a?a:r,m[1]=l;for(var k=2;k{t.r(e),t.d(e,{assets:()=>p,contentTitle:()=>m,default:()=>d,frontMatter:()=>i,metadata:()=>l,toc:()=>k});var n=t(7462),r=(t(7294),t(3905));const i={sidebar_position:32,title:"A.32. Basic Memory Management",sidebar_label:"A.32. Basic Memory Management"},m=void 0,l={unversionedId:"basic/basic-memory-management",id:"basic/basic-memory-management",title:"A.32. Basic Memory Management",description:"Pada chapter ini kita akan belajar tentang salah satu hal penting dalam topik pemrograman secara general, yaitu memory management yang mencakup pembahasan tentang alamat memori, pointer, heap, stack, dan juga error-error yang terjadi karena kesalahan dalam manajemen memori.",source:"@site/docs/basic/basic-memory-management.md",sourceDirName:"basic",slug:"/basic/basic-memory-management",permalink:"/basic/basic-memory-management",draft:!1,tags:[],version:"current",sidebarPosition:32,frontMatter:{sidebar_position:32,title:"A.32. Basic Memory Management",sidebar_label:"A.32. Basic Memory Management"},sidebar:"tutorialSidebar",previous:{title:"A.31. Shadowing",permalink:"/basic/shadowing"},next:{title:"A.33. Pointer & References",permalink:"/basic/pointer-references"}},p={},k=[{value:"A.32.1. Memory management",id:"a321-memory-management",level:2},{value:"\u25c9 Garbage collection (GC)",id:"-garbage-collection-gc",level:3},{value:"\u25c9 Automatic reference counting (ARC)",id:"-automatic-reference-counting-arc",level:3},{value:"\u25c9 Manual memory management",id:"-manual-memory-management",level:3},{value:"\u25c9 Ownership rules",id:"-ownership-rules",level:3},{value:"A.32.2. Memory Address",id:"a322-memory-address",level:2},{value:"A.32.3. Stack memory",id:"a323-stack-memory",level:2},{value:"\u25c9 Contoh ke-1",id:"-contoh-ke-1",level:3},{value:"\u25c9 Contoh ke-2",id:"-contoh-ke-2",level:3},{value:"A.32.4. Heap memory",id:"a324-heap-memory",level:2},{value:"A.32.5. Error memory management",id:"a325-error-memory-management",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:k};function d(a){let{components:e,...i}=a;return(0,r.kt)("wrapper",(0,n.Z)({},o,i,{components:e,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Pada chapter ini kita akan belajar tentang salah satu hal penting dalam topik pemrograman secara general, yaitu ",(0,r.kt)("em",{parentName:"p"},"memory management")," yang mencakup pembahasan tentang alamat memori, pointer, heap, stack, dan juga error-error yang terjadi karena kesalahan dalam manajemen memori."),(0,r.kt)("p",null,"Di bahasa pemrograman high-level biasanya topik tersebut jarang disentuh, tetapi di Rust yang notabene adalah system programming, hal di atas wajib untuk dipelajari."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Penulis tekankan bahwa mungkin pembelajaran pada bagian ini akan terasa seperti ",(0,r.kt)("em",{parentName:"p"},"oversimplified")," karena tujuannya adalah untuk para pembaca yang masih dalam proses belajar atau malah belum mengenal sama sekali tentang manajemen memori.")),(0,r.kt)("h2",{id:"a321-memory-management"},"A.32.1. Memory management"),(0,r.kt)("p",null,"Semua bahasa pemrograman memiliki caranya sendiri dalam melakukan pengelolaan memory atau memory management. Ada beberapa macam metode manajemen memori yang diterapkan pada bahasa pemrograman, di antaranya adalah berikut:"),(0,r.kt)("h3",{id:"-garbage-collection-gc"},"\u25c9 Garbage collection (GC)"),(0,r.kt)("p",null,"GC adalah metode manajemen memori otomatis pada bahasa pemrograman. GC memiliki suatu unit yang disebut dengan ",(0,r.kt)("em",{parentName:"p"},"garbage collector"),". Collector tersebut aktif memonitor program, dan pada periode atau event tertentu ia akan berusaha untuk mengambil kembali (reclaim) memory yang sebelumnya telah dialokasikan dengan catatan memori tersebut sudah tidak lagi digunakan. Proses ini disebut dengan dealokasi memory."),(0,r.kt)("p",null,"Proses dealokasi pada GC terjadi di belakang layar secara asynchronous."),(0,r.kt)("p",null,"Beberapa bahasa pemrograman yang menerapkan GC di antara adalah Java, C#, Go, Lisp, dan banyak bahasa lainnya."),(0,r.kt)("h3",{id:"-automatic-reference-counting-arc"},"\u25c9 Automatic reference counting (ARC)"),(0,r.kt)("p",null,"ARC adalah metode manajemen memori yang diterapkan pada bahasa Objective-C dan Swift. Cara ARC me-manage memory adalah dengan mencatat ",(0,r.kt)("em",{parentName:"p"},"reference")," object dan segala aktifitas yang terjadi pada object tersebut."),(0,r.kt)("p",null,"Di ARC, ada satuan yang disebut dengan ",(0,r.kt)("em",{parentName:"p"},"retain count")," yang merupakan representasi jumlah banyaknya variabel atau object yang memegang suatu ",(0,r.kt)("em",{parentName:"p"},"reference"),". Ketika ",(0,r.kt)("em",{parentName:"p"},"reference")," sudah pindah ke luar scope atau dihapus isinya dan dilihat pada catatan rupanya tidak ada variabel yang memegang ",(0,r.kt)("em",{parentName:"p"},"reference")," tersebut, maka dilakukan proses dealokasi memory."),(0,r.kt)("p",null,"Dalam bahasa yang menerapkan ARC, programmer dianjurkan untuk perhatian dan bijak dalam pengalokasian variabel beserta nilainya. Mana data yang diperlukan untuk di-retain secara ",(0,r.kt)("em",{parentName:"p"},"strong")," dan mana yang tidak, harus pas sesuai dengan kebutuhan. Jika tidak hati-hati maka program mempunyai resiko lebih tinggi untuk menemui error ",(0,r.kt)("em",{parentName:"p"},"deadlocks")," ataupun ",(0,r.kt)("em",{parentName:"p"},"memory leaks")," (yang juga akan dibahas pada chapter ini)."),(0,r.kt)("h3",{id:"-manual-memory-management"},"\u25c9 Manual memory management"),(0,r.kt)("p",null,"Manual memory management berarti programmer dibebani secara penuh dalam hal manajemen memori, mengharuskan programmer untuk super hati-hati dalam pengalokasian memory, kapan waktunya, di mana alokasinya (apakah ",(0,r.kt)("em",{parentName:"p"},"heap")," atau ",(0,r.kt)("em",{parentName:"p"},"stack"),"), dan kapan harus melakukan operasi dealokasi memory."),(0,r.kt)("p",null,"Metode manajemen memori ini dipakai dalam system programming contohnya bahasa C dan C++."),(0,r.kt)("h3",{id:"-ownership-rules"},"\u25c9 Ownership rules"),(0,r.kt)("p",null,"Manajemen memori yang dilakukan dengan menerapkan konsep ",(0,r.kt)("em",{parentName:"p"},"ownership")," beserta aturan-aturannya. Metode manajemen memori ini adalah yang digunakan di Rust."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai ownership rules pada Rust dibahas pada chapter selanjutnya, yaitu ",(0,r.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership"),". Untuk sekarang silakan selesaikan terlebih dahulu pembahasan chapter ini.")),(0,r.kt)("h2",{id:"a322-memory-address"},"A.32.2. Memory Address"),(0,r.kt)("p",null,"Memory address atau alamat memori adalah sebuah lokasi spesifik di memori yang digunakan oleh software maupun hardware untuk menyimpan suatu data."),(0,r.kt)("p",null,"Pembahasan mengenai memory address ini sangatlah luas. Pada chapter ini kita tidak akan membahasnya secara menyeluruh, melainkan hanya poin penting yang perlu diketahui dan dibutuhkan dalam proses pembelajaran."),(0,r.kt)("p",null,"Ok lanjut ke contoh agar lebih jelas. Dimisalkan ada sebuah variabel bertipe data numerik ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", variabel tersebut akan membutuhkan sejumlah bit alokasi alamat memori untuk bisa menyimpan value-nya yang pada contoh ini adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"32 bits")," (karena tipenya ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),") atau jika dikonversi ke bentuk ",(0,r.kt)("em",{parentName:"p"},"bytes")," adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"4 bytes"),"."),(0,r.kt)("p",null,"Alokasi memory address mengacu ke tipe data (bukan value), sebagai contoh pada data bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"i32")," maka berapapun value data tersebut (entah ",(0,r.kt)("inlineCode",{parentName:"p"},"1"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"2"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"1999999"),", atau lainnya) tetap membutuhkan ",(0,r.kt)("inlineCode",{parentName:"p"},"32 bits")," alokasi alamat memori untuk menyimpan data tersebut."),(0,r.kt)("p",null,"Silakan perhatikan ilustrasi berikut agar lebih jelas. Gambar diambil dari post ",(0,r.kt)("a",{parentName:"p",href:"https://medium.com/@luischaparroc/integer-numbers-storage-in-computer-memory-47af4b59009"},"medium.com/@luischaparroc"),"."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Memory address",src:t(4275).Z,width:"875",height:"477"})),(0,r.kt)("p",null,"Pada contoh di atas, variabel adalah bertipe data ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", maka di memory dialokasikanlah alamat memory dengan lebar 32 bit."),(0,r.kt)("p",null,"Nilai variabel tersebut adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"25"),", yang jika dikonversi ke bentuk binary adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"11001"),". Dengan ini maka pada 32 bit yang sudah dialokasikan, akan terisi dengan nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"11001"),". Penulisannya dari kanan dan jika ada slot kosong sebelah kiri maka terisi dengan ",(0,r.kt)("inlineCode",{parentName:"p"},"0"),"."),(0,r.kt)("p",null,"Hasilnya adalah angka biner berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"00000000000000000000000000011001\n")),(0,r.kt)("h2",{id:"a323-stack-memory"},"A.32.3. Stack memory"),(0,r.kt)("p",null,"Masih dalam topik manajemen memori, ada dua hal lagi yang sangat penting untuk diketahui, yaitu ",(0,r.kt)("em",{parentName:"p"},"stack")," dan ",(0,r.kt)("em",{parentName:"p"},"heap"),". Keduanya adalah bagian dari memory, tempat di mana alokasi dilakukan."),(0,r.kt)("p",null,"Data disimpan dalam stack memory dalam bentuk stack. Karakteristik dari stack:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Di Rust programming, stack digunakan sebagai default tempat alokasi memori"),(0,r.kt)("li",{parentName:"ul"},"Data yang terakhir masuk adalah yang pertama akan keluar (LIFO)"),(0,r.kt)("li",{parentName:"ul"},"Data yang disimpan diketahui size/ukurannya, dan memiliki batas"),(0,r.kt)("li",{parentName:"ul"},"Alokasi bersifat lokal terhadap pemanggilan fungsi"),(0,r.kt)("li",{parentName:"ul"},"Kecepatan pengaksesan data sangat tinggi")),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Stack memory",src:t(9929).Z,width:"500",height:"345"})),(0,r.kt)("p",null,"Data untuk tipe primitif (seperti ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"bool"),", dll) disimpan di stack."),(0,r.kt)("h3",{id:"-contoh-ke-1"},"\u25c9 Contoh ke-1"),(0,r.kt)("p",null,"Selanjutnya kita akan pelajari secara garis besar tentang bagaimana sebuah data dialokasikan di stack. Silakan mulai dengan mempelajari kode sederhana berikut (tanpa perlu dipraktikkan), kemudian ikuti pembahasan setelahnya."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n let x = 24;\n}\n")),(0,r.kt)("p",null,"Fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," hanya berisi 1 buah data, yaitu variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," dengan nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),". Ketika fungsi tersebut dipanggil, maka data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," disimpan di stack (karena di Rust by default semua data disimpan di stack). Nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," adalah numerik bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", maka compiler akan mengalokasikan 32 bits di stack memory untuk menyimpan nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),"."),(0,r.kt)("p",null,"Pada catatan karakteristik stack di atas, telah disinggung bahwa alokasi data di stack bersifat lokal terhadap pemanggilan fungsi, artinya apa? \u279c Semua data dalam suatu blok fungsi akan disimpan dalam sebuah group yang disebut dengan ",(0,r.kt)("em",{parentName:"p"},"stack frame"),"."),(0,r.kt)("p",null,"Pada contoh di atas, ketika fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," dipanggil, sebuah stack frame terbuat, dan data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," disimpan dalam stack frame tersebut."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Kolom ",(0,r.kt)("inlineCode",{parentName:"p"},"No.")," di atas merepresentasikan urutan data dalam stack agar lebih mudah dipahami")),(0,r.kt)("h3",{id:"-contoh-ke-2"},"\u25c9 Contoh ke-2"),(0,r.kt)("p",null,"Ok, sekarang mari lanjut contoh ke-2 berikut ini agar makin jelas."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something() {\n let y = 13;\n let z = 11;\n}\n\nfn main() {\n let x = 24;\n do_something();\n let a = 4;\n let b = 18;\n}\n")),(0,r.kt)("p",null,"Program sederhana di atas memiliki dua buah fungsi, ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," dan ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something"),". Saat program dijalankan, lebih tepatnya saat fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," dipanggil maka sebuah stack frame dibuat dan data dalam fungsi tersebut dialokasikan pada stack frame."),(0,r.kt)("p",null,"Perlu diketahui bahwa eksekusi statement dalam fungsi adalah per baris, dimulai dari atas. Dengan ini maka data yang pertama dialokasikan ke memory adalah variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"x"),"."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Setelah itu, lanjut ke statement ke-2 yaitu pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something"),"."),(0,r.kt)("p",null,"Kembali ke teori, bahwa alokasi data stack adalah bersifat lokal terhadap pemanggilan fungsi. Maka dibuatlah stack frame baru untuk pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something")," dengan isi adalah alokasi data ",(0,r.kt)("inlineCode",{parentName:"p"},"y")," dan ",(0,r.kt)("inlineCode",{parentName:"p"},"z"),"."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"2"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"z"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"11"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"strong"},"do_something()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"1"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"y"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"13"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"strong"},"do_something()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Setelah eksekusi blok kode do_something selesai maka stack frame pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something")," akan di-dealokasi, dikosongkan, dihapus."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Sebenarnya tidak se-sederhana itu proses dealokasi memori, ada pengecekan ownership yang harus dilakukan terlebih dahulu. Namun agar makin tidak bingung, untuk sementara kita gunakan penjelasan di atas."),(0,r.kt)("p",{parentName:"blockquote"},"Silakan selesaikan dulu pembahasan chapter ini, kemudian pada chapter selanjutnya kita akan bahas tentang apa itu ",(0,r.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership"),".")),(0,r.kt)("p",null,"Ok, maka dari 2 stack yang sebelumnya ada, sekarang tinggal 1 stack frame saja yaitu milik pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,r.kt)("p",null,"Dari yang sebelumnya ..."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"2"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"z"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"11"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"del"},"do_something()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"1"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"y"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"13"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"red"}},(0,r.kt)("del",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"del"},"do_something()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"... sekarang menjadi ..."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Ok, lanjut ke block fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main")," berikutnya, yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"let a = 4"),". Saat dipanggil maka ada penambahan data baru pada stack frame pertama."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"3"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"a"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"4"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"strong"},"main()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Kemudian sampai di statement terakhir fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),". Ketika ",(0,r.kt)("inlineCode",{parentName:"p"},"let b = 14")," dipanggil maka ada penambahan data baru pada stack frame pertama."),(0,r.kt)("table",null,(0,r.kt)("thead",{parentName:"table"},(0,r.kt)("tr",{parentName:"thead"},(0,r.kt)("th",{parentName:"tr",align:"center"},"No."),(0,r.kt)("th",{parentName:"tr",align:"center"},"Variabel"),(0,r.kt)("th",{parentName:"tr",align:"center"},"Nilai"),(0,r.kt)("th",{parentName:"tr",align:"left"},(0,r.kt)("em",{parentName:"th"},"Stack frame")))),(0,r.kt)("tbody",{parentName:"table"},(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"4"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"b"))),(0,r.kt)("td",{parentName:"tr",align:"center"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"18"))),(0,r.kt)("td",{parentName:"tr",align:"left"},(0,r.kt)("span",{style:{color:"#2e8555"}},(0,r.kt)("strong",{parentName:"td"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"strong"},"main()"))))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"3"),(0,r.kt)("td",{parentName:"tr",align:"center"},"a"),(0,r.kt)("td",{parentName:"tr",align:"center"},"4"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))),(0,r.kt)("tr",{parentName:"tbody"},(0,r.kt)("td",{parentName:"tr",align:"center"},"0"),(0,r.kt)("td",{parentName:"tr",align:"center"},"x"),(0,r.kt)("td",{parentName:"tr",align:"center"},"24"),(0,r.kt)("td",{parentName:"tr",align:"left"},"milik fungsi ",(0,r.kt)("inlineCode",{parentName:"td"},"main()"))))),(0,r.kt)("p",null,"Setelah fungsi selesai dieksekusi, stack frame di-dealokasi."),(0,r.kt)("p",null,"Kurang lebih seperti itu sekilas peran dari stack dalam Rust programming. Selanjutnya kita bahas tentang heap memory."),(0,r.kt)("h2",{id:"a324-heap-memory"},"A.32.4. Heap memory"),(0,r.kt)("p",null,"Heap adalah salah satu tempat alokasi memory selain stack. Karakteristik dari heap:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Heap digunakan untuk alokasi data yang sifatnya dinamis, tidak diketahui size-nya, atau bisa berubah size-nya"),(0,r.kt)("li",{parentName:"ul"},"Data di heap tidak memiliki pattern tertentu"),(0,r.kt)("li",{parentName:"ul"},"Alokasi dan dealokasi data di heap bisa dilakukan kapanpun"),(0,r.kt)("li",{parentName:"ul"},"Kecepatan pengaksesan data di heap lebih lambat dibanding stack")),(0,r.kt)("p",null,"Ilustrasi perbandingan stack dan heap:"),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Heap memory",src:t(6907).Z,width:"414",height:"236"})),(0,r.kt)("p",null,"Tipe data non-primitive di Rust data-nya disimpan di heap, contohnya seperti Vector, ",(0,r.kt)("inlineCode",{parentName:"p"},"String"),", dan beberapa lainnya. Penulis tekankan, bahwa ",(0,r.kt)("strong",{parentName:"p"},"yang disimpan di heap adalah data-nya saja, sedangkan atribut lainnya (seperti ",(0,r.kt)("inlineCode",{parentName:"strong"},"length")," dan ",(0,r.kt)("inlineCode",{parentName:"strong"},"capacity"),") tetap disimpan disimpan di stack"),". Lebih jelasnya silakan lihat ilustrasi berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"let numbers = vec![12, 24, 7];\n")),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Heap memory",src:t(6033).Z,width:"420",height:"184"})),(0,r.kt)("p",null,"Heap, selain digunakan sebagai penyimpanan data dinamis, kita juga bisa manfaatkan sebagai penyimpanan data jenis lainnya (secara eksplisit) menggunakan tipe data ",(0,r.kt)("inlineCode",{parentName:"p"},"Box"),"."),(0,r.kt)("p",null,"Pembahasan Lebih jelasnya mengenai heap allocation ada pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/string-slice-vs-string-literal"},"String Custom Type vs ",(0,r.kt)("inlineCode",{parentName:"a"},"&str"))," dan juga ",(0,r.kt)("a",{parentName:"p",href:"#/wip/box"},"Box"),". Namun untuk sekarang, penulis anjurkan untuk lanjut ke pembahasan berikutnya terlebih dahulu."),(0,r.kt)("h2",{id:"a325-error-memory-management"},"A.32.5. Error memory management"),(0,r.kt)("p",null,"Rust sangat disiplin dalam hal penulisan source code, terutama untuk kode-kode yang berhubungan dengan memory management. Hal ini dilakukan oleh Rust untuk meminimalisir munculnya error seperti memory leak dan sejenisnya."),(0,r.kt)("p",null,"Namun meski demikian, potensi error memory tetap ada, dan kita akan bahas itu nantinya setelah masuk chapter ",(0,r.kt)("a",{parentName:"p",href:"#/wip/safe-unsafe"},"Safe & Unsafe"),"."),(0,r.kt)("p",null,"Nantinya akan dibahas juga tentang beberapa error saat compile time yang error tersebut berhubungan dengan memory management, yaitu pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership")," dan ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),". Untuk sekarang, mari lanjut ke chapter berikutnya terlebih dahulu."),(0,r.kt)("hr",null),(0,r.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,r.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,r.kt)("pre",null,(0,r.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/memory_management"},"github.com/novalagung/dasarpemrogramanrust-example/../memory_management")),(0,r.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/nomicon/ownership.html"},"https://doc.rust-lang.org/nomicon/ownership.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-01-what-is-ownership.html"},"https://doc.rust-lang.org/book/ch04-01-what-is-ownership.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/1.22.0/book/first-edition/the-stack-and-the-heap.html"},"https://doc.rust-lang.org/1.22.0/book/first-edition/the-stack-and-the-heap.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/std/box.html"},"https://doc.rust-lang.org/rust-by-example/std/box.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://docs.swift.org/swift-book/LanguageGuide/AutomaticReferenceCounting.html"},"https://docs.swift.org/swift-book/LanguageGuide/AutomaticReferenceCounting.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://docs.elementscompiler.com/Concepts/ARC/"},"https://docs.elementscompiler.com/Concepts/ARC/")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Memory_address"},"https://en.wikipedia.org/wiki/Memory_address")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Code_segment"},"https://en.wikipedia.org/wiki/Code_segment")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Automatic_Reference_Counting"},"https://en.wikipedia.org/wiki/Automatic_Reference_Counting")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Garbage_collection_(computer_science)"},"https://en.wikipedia.org/wiki/Garbage_collection_(computer_science)")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://log2base2.com/C/pointer/computer-memory-address-basics.html"},"https://log2base2.com/C/pointer/computer-memory-address-basics.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://quora.com/How-does-memory-management-work-in-Rust"},"https://quora.com/How-does-memory-management-work-in-Rust")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://medium.com/the-legend/management-memori-pada-bahasa-pemrograman-a33772635aa5"},"https://medium.com/the-legend/management-memori-pada-bahasa-pemrograman-a33772635aa5")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://imam.digmi.id/post/belajar-rust-memahami-stack-dan-heap/"},"https://imam.digmi.id/post/belajar-rust-memahami-stack-dan-heap/")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://javatpoint.com/rust-ownership"},"https://javatpoint.com/rust-ownership")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/24158114"},"https://stackoverflow.com/questions/24158114"))))}d.isMDXComponent=!0},4275:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/basic-memory-management-1-ac98989ab48847e58470100c02455dab.png"},9929:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/basic-memory-management-2-114833fdf1f1ac3114d03494e6603ff1.jpg"},6907:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},6033:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/04da11da.b3ba07e3.js b/assets/js/04da11da.b3ba07e3.js deleted file mode 100644 index 986d3192..00000000 --- a/assets/js/04da11da.b3ba07e3.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[3007],{3905:(e,a,n)=>{n.d(a,{Zo:()=>k,kt:()=>o});var A=n(7294);function l(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function t(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var A=Object.getOwnPropertySymbols(e);a&&(A=A.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,A)}return n}function i(e){for(var a=1;a=0||(l[n]=e[n]);return l}(e,a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);for(A=0;A=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(l[n]=e[n])}return l}var p=A.createContext({}),d=function(e){var a=A.useContext(p),n=a;return e&&(n="function"==typeof e?e(a):i(i({},a),e)),n},k=function(e){var a=d(e.components);return A.createElement(p.Provider,{value:a},e.children)},m={inlineCode:"code",wrapper:function(e){var a=e.children;return A.createElement(A.Fragment,{},a)}},u=A.forwardRef((function(e,a){var n=e.components,l=e.mdxType,t=e.originalType,p=e.parentName,k=r(e,["components","mdxType","originalType","parentName"]),u=d(n),o=l,c=u["".concat(p,".").concat(o)]||u[o]||m[o]||t;return n?A.createElement(c,i(i({ref:a},k),{},{components:n})):A.createElement(c,i({ref:a},k))}));function o(e,a){var n=arguments,l=a&&a.mdxType;if("string"==typeof e||l){var t=n.length,i=new Array(t);i[0]=u;var r={};for(var p in a)hasOwnProperty.call(a,p)&&(r[p]=a[p]);r.originalType=e,r.mdxType="string"==typeof e?e:l,i[1]=r;for(var d=2;d{n.r(a),n.d(a,{assets:()=>p,contentTitle:()=>i,default:()=>m,frontMatter:()=>t,metadata:()=>r,toc:()=>d});var A=n(7462),l=(n(7294),n(3905));const t={sidebar_position:11,title:"A.11. Perulangan \u279c loop, break, continue, label",sidebar_label:"A.11. Perulangan \u279c loop, break, continue, label"},i=void 0,r={unversionedId:"basic/perulangan-loop-break-continue-label",id:"basic/perulangan-loop-break-continue-label",title:"A.11. Perulangan \u279c loop, break, continue, label",description:"Selain keyword while, ada juga keyword loop yang fungsi dasarnya adalah sama, yaitu untuk perulangan.",source:"@site/docs/basic/perulangan-loop-break-continue-label.md",sourceDirName:"basic",slug:"/basic/perulangan-loop-break-continue-label",permalink:"/basic/perulangan-loop-break-continue-label",draft:!1,tags:[],version:"current",sidebarPosition:11,frontMatter:{sidebar_position:11,title:"A.11. Perulangan \u279c loop, break, continue, label",sidebar_label:"A.11. Perulangan \u279c loop, break, continue, label"},sidebar:"tutorialSidebar",previous:{title:"A.10. Perulangan \u279c while",permalink:"/basic/perulangan-while"},next:{title:"A.12. Perulangan \u279c for in",permalink:"/basic/perulangan-for-in"}},p={},d=[{value:"A.11.1. Keyword loop",id:"a111-keyword-loop",level:2},{value:"A.11.2. Keyword break",id:"a112-keyword-break",level:2},{value:"A.11.3. Nested loop",id:"a113-nested-loop",level:2},{value:"A.11.4. Keyword continue",id:"a114-keyword-continue",level:2},{value:"A.11.5. Label perulangan",id:"a115-label-perulangan",level:2},{value:"A.11.6. Returning from loop",id:"a116-returning-from-loop",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],k={toc:d};function m(e){let{components:a,...t}=e;return(0,l.kt)("wrapper",(0,A.Z)({},k,t,{components:a,mdxType:"MDXLayout"}),(0,l.kt)("p",null,"Selain keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"while"),", ada juga keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," yang fungsi dasarnya adalah sama, yaitu untuk perulangan."),(0,l.kt)("h2",{id:"a111-keyword-loop"},"A.11.1. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"loop")),(0,l.kt)("p",null,"Notasi penulisan dan cara penggunaan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," ada sedikit beda dibanding ",(0,l.kt)("inlineCode",{parentName:"p"},"while"),". Keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," tidak membutuhkan argument. Blok kode loop akan terus dieksekusi selama program tidak di-stop."),(0,l.kt)("p",null,"Silakan coba praktikkan kode berikut. Angka ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," akan ditampilkan setiap perulangan kemudian di-increment nilainya. Angka akan muncul terus sampai aplikasi di-stop."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut i = 0;\n\n loop {\n println!("nilai: {i}");\n i += 1;\n }\n}\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"loop tanpa break",src:n(3723).Z,width:"608",height:"213"})),(0,l.kt)("h2",{id:"a112-keyword-break"},"A.11.2. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"break")),(0,l.kt)("p",null,(0,l.kt)("inlineCode",{parentName:"p"},"loop")," menghasilkan perulangan tanpa henti, lalu bagaimana cara stop-nya? Di sinilah keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," berperan. ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," digunakan untuk menghentikan paksa eksekusi blok kode perulangan. Biasanya keyword ini digunakan dalam kondisi tertentu (sesuai kebutuhan), misalnya ",(0,l.kt)("inlineCode",{parentName:"p"},"perulangan harus berhenti ketika nilai i di atas max"),"."),(0,l.kt)("p",null,"Mari kita ubah kode di atas dengan menambahkan kondisi untuk menghentikan perulangan jika ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," nilainya lebih dari ",(0,l.kt)("inlineCode",{parentName:"p"},"5"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 5;\n\nloop {\n println!("nilai: {i}");\n i += 1;\n if i > max {\n break;\n }\n}\n\nprintln!("perulangan selesai");\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"keyword break",src:n(4722).Z,width:"451",height:"195"})),(0,l.kt)("blockquote",null,(0,l.kt)("p",{parentName:"blockquote"},"Penulisan ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," boleh tanpa diakhiri semicolon ",(0,l.kt)("inlineCode",{parentName:"p"},";"))),(0,l.kt)("h2",{id:"a113-nested-loop"},"A.11.3. Nested ",(0,l.kt)("inlineCode",{parentName:"h2"},"loop")),(0,l.kt)("p",null,"Cara menerapkan nested loop (atau ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," di dalam ",(0,l.kt)("inlineCode",{parentName:"p"},"loop"),"), tulis saja statement ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," di dalam ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," sesuai kebutuhan."),(0,l.kt)("p",null,"Pada kode berikut, teknik nested loop diterapkan untuk membuat bentuk segitiga menggunakan karakter ",(0,l.kt)("inlineCode",{parentName:"p"},"*"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 5;\n\nloop {\n let mut j = max;\n let max_inner = i;\n\n loop {\n print!("* ");\n j -= 1;\n if j < max_inner {\n break;\n }\n }\n \n println!();\n\n i += 1;\n if i > max {\n break;\n }\n}\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"keyword break",src:n(4237).Z,width:"454",height:"175"})),(0,l.kt)("h2",{id:"a114-keyword-continue"},"A.11.4. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"continue")),(0,l.kt)("p",null,(0,l.kt)("inlineCode",{parentName:"p"},"continue")," digunakan untuk melanjutkan paksa sebuah perulangan, kebalikan dari ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," yang fungsinya menghentikan paksa sebuah perulangan."),(0,l.kt)("p",null,"Source code berikut murapakan contoh penerapan ",(0,l.kt)("inlineCode",{parentName:"p"},"continue"),". Variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," berperan sebagai counter perulangan. Jika nilai ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," adalah ganjil, maka perulangan dipaksa lanjut ke iterasi berikutnya. Dengan ini maka macro ",(0,l.kt)("inlineCode",{parentName:"p"},"println")," hanya akan menampilkan nilai genap. Dan program akan berhenti jika ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," nilainya lebih dari ",(0,l.kt)("inlineCode",{parentName:"p"},"max"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 15;\n\nloop {\n i += 1;\n\n if i % 2 == 1 {\n continue;\n }\n\n println!("nilai i: {i}");\n\n if i > max {\n break;\n }\n}\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"keyword continue",src:n(5541).Z,width:"456",height:"230"})),(0,l.kt)("h2",{id:"a115-label-perulangan"},"A.11.5. Label perulangan"),(0,l.kt)("p",null,"Statement perulangan menggunakan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," bisa ditandai dengan label. Manfaat dari penggunaan label adalah bisa mengeksekusi ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," atau ",(0,l.kt)("inlineCode",{parentName:"p"},"continue")," ke perulangan di luar blok kode perulangan dimana statement itu berada. Umumnya label perulangan dipergunakan pada nested loop, dimana ada kebutuhan untuk menghentikan/melanjutkan paksa perulangan terluar."),(0,l.kt)("p",null,"Berikut adalah notasi penulisan loop dengan dan tanpa label. Nama label diawali dengan tanda petik ",(0,l.kt)("inlineCode",{parentName:"p"},"'"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},"// loop biasa\nloop {\n // statements\n break;\n}\n\n// loop dengan label\n'namaLabel: loop {\n // statements\n break 'namaLabel;\n}\n")),(0,l.kt)("p",null,"Mari kita pelajari dan praktikkan kode berikut ini. Dibawah ini adalah sebuah program sederhana menampilkan angka yang hasilnya bisa dilihat digambar dibawahnya. Perulangan di level 2 akan dihentikan secara paksa ketika ",(0,l.kt)("inlineCode",{parentName:"p"},"j > i"),". Sedangkan perulangan level pertama atau terluar (dengan label ",(0,l.kt)("inlineCode",{parentName:"p"},"'mainLoop"),") akan dihentikan paksa dari perulangan level 2 jika kondisi ",(0,l.kt)("inlineCode",{parentName:"p"},"i > max"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},"let mut i = 0;\nlet max = 9;\n\n'mainLoop: loop {\n i += 1;\n let mut j = 0;\n\n loop {\n if i > max {\n break 'mainLoop;\n }\n\n j += 1;\n if j > i {\n break;\n }\n\n print!(\"{i} \");\n }\n\n println!();\n}\n")),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"looping label",src:n(5302).Z,width:"514",height:"227"})),(0,l.kt)("h2",{id:"a116-returning-from-loop"},"A.11.6. Returning from ",(0,l.kt)("inlineCode",{parentName:"h2"},"loop")),(0,l.kt)("p",null,(0,l.kt)("em",{parentName:"p"},"Returning from ",(0,l.kt)("inlineCode",{parentName:"em"},"loop"))," merupakan teknik pemanfaatan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," dan ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," untuk menampung sebuah return value dari blok kode perulangan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop"),". Agar lebih jelas, silakan coba kode berikut:"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'let mut counter = 0;\n\nlet result = loop {\n counter += 1;\n\n if counter == 10 {\n break counter * 2;\n }\n};\n\nprintln!("result: {result}");\n')),(0,l.kt)("p",null,"Pada kode di atas, variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"result")," dideklarasikan dengan predefined value adalah blok kode ",(0,l.kt)("inlineCode",{parentName:"p"},"loop"),". Di sini artinya variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"result")," isinya bukan blok kode perulangan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop"),", melainkan isinya adalah apapun yang dituliskan setelah keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"break"),". Unik ya?"),(0,l.kt)("p",null,"Variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"counter")," yang nilai awalnya ",(0,l.kt)("inlineCode",{parentName:"p"},"0"),", di-increment dalam perulangan. Ketika nilai ",(0,l.kt)("inlineCode",{parentName:"p"},"counter")," adalah ",(0,l.kt)("inlineCode",{parentName:"p"},"10"),", nilai ",(0,l.kt)("inlineCode",{parentName:"p"},"counter * 2")," dijadikan sebagai return value. Dengan ini maka variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"result")," nilainya adalah ",(0,l.kt)("inlineCode",{parentName:"p"},"20"),"."),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"looping label",src:n(5249).Z,width:"553",height:"117"})),(0,l.kt)("hr",null),(0,l.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,l.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,l.kt)("pre",null,(0,l.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/perulangan_loop_break_continue_label"},"github.com/novalagung/dasarpemrogramanrust-example/../perulangan_loop_break_continue_label")),(0,l.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-05-control-flow.html"},"https://doc.rust-lang.org/book/ch03-05-control-flow.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.loop.html"},"https://doc.rust-lang.org/std/keyword.loop.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.break.html"},"https://doc.rust-lang.org/std/keyword.break.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.continue.html"},"https://doc.rust-lang.org/std/keyword.continue.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/loop.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/loop.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/loop/nested.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/loop/nested.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/loop/return.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/loop/return.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/expressions/loop-expr.html"},"https://doc.rust-lang.org/reference/expressions/loop-expr.html"))))}m.isMDXComponent=!0},3723:(e,a,n)=>{n.d(a,{Z:()=>A});const A=n.p+"assets/images/perulangan-loop-break-continue-label-1-f02ae31c07ab672a301cbf6db2f23179.png"},4722:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""},4237:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""},5541:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""},5302:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""},5249:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""}}]); \ No newline at end of file diff --git a/assets/js/04da11da.d61e5604.js b/assets/js/04da11da.d61e5604.js new file mode 100644 index 00000000..e5bc661a --- /dev/null +++ b/assets/js/04da11da.d61e5604.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[3007],{3905:(e,a,n)=>{n.d(a,{Zo:()=>k,kt:()=>o});var A=n(7294);function l(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function t(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var A=Object.getOwnPropertySymbols(e);a&&(A=A.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,A)}return n}function i(e){for(var a=1;a=0||(l[n]=e[n]);return l}(e,a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);for(A=0;A=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(l[n]=e[n])}return l}var p=A.createContext({}),d=function(e){var a=A.useContext(p),n=a;return e&&(n="function"==typeof e?e(a):i(i({},a),e)),n},k=function(e){var a=d(e.components);return A.createElement(p.Provider,{value:a},e.children)},m={inlineCode:"code",wrapper:function(e){var a=e.children;return A.createElement(A.Fragment,{},a)}},u=A.forwardRef((function(e,a){var n=e.components,l=e.mdxType,t=e.originalType,p=e.parentName,k=r(e,["components","mdxType","originalType","parentName"]),u=d(n),o=l,c=u["".concat(p,".").concat(o)]||u[o]||m[o]||t;return n?A.createElement(c,i(i({ref:a},k),{},{components:n})):A.createElement(c,i({ref:a},k))}));function o(e,a){var n=arguments,l=a&&a.mdxType;if("string"==typeof e||l){var t=n.length,i=new Array(t);i[0]=u;var r={};for(var p in a)hasOwnProperty.call(a,p)&&(r[p]=a[p]);r.originalType=e,r.mdxType="string"==typeof e?e:l,i[1]=r;for(var d=2;d{n.r(a),n.d(a,{assets:()=>p,contentTitle:()=>i,default:()=>m,frontMatter:()=>t,metadata:()=>r,toc:()=>d});var A=n(7462),l=(n(7294),n(3905));const t={sidebar_position:11,title:"A.11. Perulangan \u279c loop, break, continue, label",sidebar_label:"A.11. Perulangan \u279c loop, break, continue, label"},i=void 0,r={unversionedId:"basic/perulangan-loop-break-continue-label",id:"basic/perulangan-loop-break-continue-label",title:"A.11. Perulangan \u279c loop, break, continue, label",description:"Selain keyword while, ada juga keyword loop yang fungsi dasarnya adalah sama, yaitu untuk perulangan.",source:"@site/docs/basic/perulangan-loop-break-continue-label.md",sourceDirName:"basic",slug:"/basic/perulangan-loop-break-continue-label",permalink:"/basic/perulangan-loop-break-continue-label",draft:!1,tags:[],version:"current",sidebarPosition:11,frontMatter:{sidebar_position:11,title:"A.11. Perulangan \u279c loop, break, continue, label",sidebar_label:"A.11. Perulangan \u279c loop, break, continue, label"},sidebar:"tutorialSidebar",previous:{title:"A.10. Perulangan \u279c while",permalink:"/basic/perulangan-while"},next:{title:"A.12. Perulangan \u279c for in",permalink:"/basic/perulangan-for-in"}},p={},d=[{value:"A.11.1. Keyword loop",id:"a111-keyword-loop",level:2},{value:"A.11.2. Keyword break",id:"a112-keyword-break",level:2},{value:"A.11.3. Nested loop",id:"a113-nested-loop",level:2},{value:"A.11.4. Keyword continue",id:"a114-keyword-continue",level:2},{value:"A.11.5. Label perulangan",id:"a115-label-perulangan",level:2},{value:"A.11.6. Returning from loop",id:"a116-returning-from-loop",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],k={toc:d};function m(e){let{components:a,...t}=e;return(0,l.kt)("wrapper",(0,A.Z)({},k,t,{components:a,mdxType:"MDXLayout"}),(0,l.kt)("p",null,"Selain keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"while"),", ada juga keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," yang fungsi dasarnya adalah sama, yaitu untuk perulangan."),(0,l.kt)("h2",{id:"a111-keyword-loop"},"A.11.1. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"loop")),(0,l.kt)("p",null,"Notasi penulisan dan cara penggunaan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," ada sedikit beda dibanding ",(0,l.kt)("inlineCode",{parentName:"p"},"while"),". Keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," tidak membutuhkan argument. Blok kode loop akan terus dieksekusi selama program tidak di-stop."),(0,l.kt)("p",null,"Silakan coba praktikkan kode berikut. Angka ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," akan ditampilkan setiap perulangan kemudian di-increment nilainya. Angka akan muncul terus sampai aplikasi di-stop."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut i = 0;\n\n loop {\n println!("nilai: {i}");\n i += 1;\n }\n}\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"loop tanpa break",src:n(3723).Z,width:"608",height:"213"})),(0,l.kt)("h2",{id:"a112-keyword-break"},"A.11.2. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"break")),(0,l.kt)("p",null,(0,l.kt)("inlineCode",{parentName:"p"},"loop")," menghasilkan perulangan tanpa henti, lalu bagaimana cara stop-nya? Di sinilah keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," berperan. ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," digunakan untuk menghentikan paksa eksekusi blok kode perulangan. Biasanya keyword ini digunakan dalam kondisi tertentu (sesuai kebutuhan), misalnya ",(0,l.kt)("inlineCode",{parentName:"p"},"perulangan harus berhenti ketika nilai i di atas max"),"."),(0,l.kt)("p",null,"Mari kita ubah kode di atas dengan menambahkan kondisi untuk menghentikan perulangan jika ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," nilainya lebih dari ",(0,l.kt)("inlineCode",{parentName:"p"},"5"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 5;\n\nloop {\n println!("nilai: {i}");\n i += 1;\n if i > max {\n break;\n }\n}\n\nprintln!("perulangan selesai");\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"keyword break",src:n(4722).Z,width:"451",height:"195"})),(0,l.kt)("blockquote",null,(0,l.kt)("p",{parentName:"blockquote"},"Penulisan ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," boleh tanpa diakhiri semicolon ",(0,l.kt)("inlineCode",{parentName:"p"},";"))),(0,l.kt)("h2",{id:"a113-nested-loop"},"A.11.3. Nested ",(0,l.kt)("inlineCode",{parentName:"h2"},"loop")),(0,l.kt)("p",null,"Cara menerapkan nested loop (atau ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," di dalam ",(0,l.kt)("inlineCode",{parentName:"p"},"loop"),"), tulis saja statement ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," di dalam ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," sesuai kebutuhan."),(0,l.kt)("p",null,"Pada kode berikut, teknik nested loop diterapkan untuk membuat bentuk segitiga menggunakan karakter ",(0,l.kt)("inlineCode",{parentName:"p"},"*"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 5;\n\nloop {\n let mut j = max;\n let max_inner = i;\n\n loop {\n print!("* ");\n j -= 1;\n if j < max_inner {\n break;\n }\n }\n \n println!();\n\n i += 1;\n if i > max {\n break;\n }\n}\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"keyword break",src:n(4237).Z,width:"454",height:"175"})),(0,l.kt)("h2",{id:"a114-keyword-continue"},"A.11.4. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"continue")),(0,l.kt)("p",null,(0,l.kt)("inlineCode",{parentName:"p"},"continue")," digunakan untuk melanjutkan paksa sebuah perulangan, kebalikan dari ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," yang fungsinya menghentikan paksa sebuah perulangan."),(0,l.kt)("p",null,"Source code berikut merupakan contoh penerapan ",(0,l.kt)("inlineCode",{parentName:"p"},"continue"),". Variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," berperan sebagai counter perulangan. Jika nilai ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," adalah ganjil, maka perulangan dipaksa lanjut ke iterasi berikutnya. Dengan ini maka macro ",(0,l.kt)("inlineCode",{parentName:"p"},"println")," hanya akan menampilkan nilai genap. Dan program akan berhenti jika ",(0,l.kt)("inlineCode",{parentName:"p"},"i")," nilainya lebih dari ",(0,l.kt)("inlineCode",{parentName:"p"},"max"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 15;\n\nloop {\n i += 1;\n\n if i % 2 == 1 {\n continue;\n }\n\n println!("nilai i: {i}");\n\n if i > max {\n break;\n }\n}\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"keyword continue",src:n(5541).Z,width:"456",height:"230"})),(0,l.kt)("h2",{id:"a115-label-perulangan"},"A.11.5. Label perulangan"),(0,l.kt)("p",null,"Statement perulangan menggunakan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," bisa ditandai dengan label. Manfaat dari penggunaan label adalah bisa mengeksekusi ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," atau ",(0,l.kt)("inlineCode",{parentName:"p"},"continue")," ke perulangan di luar blok kode perulangan di mana statement itu berada. Umumnya label perulangan dipergunakan pada nested loop untuk keperluan menghentikan/melanjutkan paksa perulangan terluar."),(0,l.kt)("p",null,"Berikut adalah notasi penulisan loop dengan dan tanpa label. Nama label diawali dengan tanda petik ",(0,l.kt)("inlineCode",{parentName:"p"},"'"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},"// loop biasa\nloop {\n // statements\n break;\n}\n\n// loop dengan label\n'namaLabel: loop {\n // statements\n break 'namaLabel;\n}\n")),(0,l.kt)("p",null,"Mari kita pelajari dan praktikkan kode berikut ini. Di bawah ini adalah sebuah program sederhana menampilkan angka yang hasilnya bisa dilihat pada gambar di bawahnya. Perulangan di level 2 akan dihentikan secara paksa ketika ",(0,l.kt)("inlineCode",{parentName:"p"},"j > i"),". Sedangkan perulangan level pertama atau terluar (dengan label ",(0,l.kt)("inlineCode",{parentName:"p"},"'mainLoop"),") akan dihentikan paksa dari perulangan level 2 jika kondisi ",(0,l.kt)("inlineCode",{parentName:"p"},"i > max"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},"let mut i = 0;\nlet max = 9;\n\n'mainLoop: loop {\n i += 1;\n let mut j = 0;\n\n loop {\n if i > max {\n break 'mainLoop;\n }\n\n j += 1;\n if j > i {\n break;\n }\n\n print!(\"{i} \");\n }\n\n println!();\n}\n")),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"looping label",src:n(5302).Z,width:"514",height:"227"})),(0,l.kt)("h2",{id:"a116-returning-from-loop"},"A.11.6. Returning from ",(0,l.kt)("inlineCode",{parentName:"h2"},"loop")),(0,l.kt)("p",null,(0,l.kt)("em",{parentName:"p"},"Returning from ",(0,l.kt)("inlineCode",{parentName:"em"},"loop"))," merupakan teknik pemanfaatan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop")," dan ",(0,l.kt)("inlineCode",{parentName:"p"},"break")," untuk menampung sebuah return value dari blok kode perulangan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop"),". Agar lebih jelas, silakan coba kode berikut:"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'let mut counter = 0;\n\nlet result = loop {\n counter += 1;\n\n if counter == 10 {\n break counter * 2;\n }\n};\n\nprintln!("result: {result}");\n')),(0,l.kt)("p",null,"Pada kode di atas, variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"result")," dideklarasikan dengan predefined value adalah blok kode ",(0,l.kt)("inlineCode",{parentName:"p"},"loop"),". Di sini artinya variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"result")," isinya bukan blok kode perulangan ",(0,l.kt)("inlineCode",{parentName:"p"},"loop"),", melainkan isinya adalah apapun yang dituliskan setelah keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"break"),". Unik ya?"),(0,l.kt)("p",null,"Variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"counter")," yang nilai awalnya ",(0,l.kt)("inlineCode",{parentName:"p"},"0"),", di-increment dalam perulangan. Ketika nilai ",(0,l.kt)("inlineCode",{parentName:"p"},"counter")," adalah ",(0,l.kt)("inlineCode",{parentName:"p"},"10"),", nilai ",(0,l.kt)("inlineCode",{parentName:"p"},"counter * 2")," dijadikan sebagai return value. Dengan ini maka variabel ",(0,l.kt)("inlineCode",{parentName:"p"},"result")," nilainya adalah ",(0,l.kt)("inlineCode",{parentName:"p"},"20"),"."),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"looping label",src:n(5249).Z,width:"553",height:"117"})),(0,l.kt)("hr",null),(0,l.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,l.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,l.kt)("pre",null,(0,l.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/perulangan_loop_break_continue_label"},"github.com/novalagung/dasarpemrogramanrust-example/../perulangan_loop_break_continue_label")),(0,l.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-05-control-flow.html"},"https://doc.rust-lang.org/book/ch03-05-control-flow.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.loop.html"},"https://doc.rust-lang.org/std/keyword.loop.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.break.html"},"https://doc.rust-lang.org/std/keyword.break.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.continue.html"},"https://doc.rust-lang.org/std/keyword.continue.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/loop.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/loop.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/loop/nested.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/loop/nested.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/loop/return.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/loop/return.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/expressions/loop-expr.html"},"https://doc.rust-lang.org/reference/expressions/loop-expr.html"))))}m.isMDXComponent=!0},3723:(e,a,n)=>{n.d(a,{Z:()=>A});const A=n.p+"assets/images/perulangan-loop-break-continue-label-1-f02ae31c07ab672a301cbf6db2f23179.png"},4722:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""},4237:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""},5541:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""},5302:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""},5249:(e,a,n)=>{n.d(a,{Z:()=>A});const A=""}}]); \ No newline at end of file diff --git a/assets/js/05cc1716.272262df.js b/assets/js/05cc1716.272262df.js deleted file mode 100644 index 9166a28a..00000000 --- a/assets/js/05cc1716.272262df.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6477],{3905:(e,a,n)=>{n.d(a,{Zo:()=>p,kt:()=>u});var i=n(7294);function t(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function l(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);a&&(i=i.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,i)}return n}function r(e){for(var a=1;a=0||(t[n]=e[n]);return t}(e,a);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(t[n]=e[n])}return t}var k=i.createContext({}),A=function(e){var a=i.useContext(k),n=a;return e&&(n="function"==typeof e?e(a):r(r({},a),e)),n},p=function(e){var a=A(e.components);return i.createElement(k.Provider,{value:a},e.children)},s={inlineCode:"code",wrapper:function(e){var a=e.children;return i.createElement(i.Fragment,{},a)}},o=i.forwardRef((function(e,a){var n=e.components,t=e.mdxType,l=e.originalType,k=e.parentName,p=d(e,["components","mdxType","originalType","parentName"]),o=A(n),u=t,m=o["".concat(k,".").concat(u)]||o[u]||s[u]||l;return n?i.createElement(m,r(r({ref:a},p),{},{components:n})):i.createElement(m,r({ref:a},p))}));function u(e,a){var n=arguments,t=a&&a.mdxType;if("string"==typeof e||t){var l=n.length,r=new Array(l);r[0]=o;var d={};for(var k in a)hasOwnProperty.call(a,k)&&(d[k]=a[k]);d.originalType=e,d.mdxType="string"==typeof e?e:t,r[1]=d;for(var A=2;A{n.r(a),n.d(a,{assets:()=>k,contentTitle:()=>r,default:()=>s,frontMatter:()=>l,metadata:()=>d,toc:()=>A});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:9,title:"A.9. Seleksi Kondisi \u279c if, else if, else",sidebar_label:"A.9. Seleksi Kondisi \u279c if, else if, else"},r=void 0,d={unversionedId:"basic/seleksi-kondisi-if",id:"basic/seleksi-kondisi-if",title:"A.9. Seleksi Kondisi \u279c if, else if, else",description:"Pada chapter ini kita akan bahas tentang penerapan if untuk kontrol alur program.",source:"@site/docs/basic/seleksi-kondisi-if.md",sourceDirName:"basic",slug:"/basic/seleksi-kondisi-if",permalink:"/basic/seleksi-kondisi-if",draft:!1,tags:[],version:"current",sidebarPosition:9,frontMatter:{sidebar_position:9,title:"A.9. Seleksi Kondisi \u279c if, else if, else",sidebar_label:"A.9. Seleksi Kondisi \u279c if, else if, else"},sidebar:"tutorialSidebar",previous:{title:"A.8. Operator",permalink:"/basic/operator"},next:{title:"A.10. Perulangan \u279c while",permalink:"/basic/perulangan-while"}},k={},A=[{value:"A.9.1. Keyword if",id:"a91-keyword-if",level:2},{value:"A.9.2. Keyword if, else if, dan else",id:"a92-keyword-if-else-if-dan-else",level:2},{value:"A.9.3. Nested if",id:"a93-nested-if",level:2},{value:"A.9.4. Returning From if",id:"a94-returning-from-if",level:2},{value:"A.9.5. Kombinasi Keyword let dan if, Dengan Tipe Data Eksplisit",id:"a95-kombinasi-keyword-let-dan-if-dengan-tipe-data-eksplisit",level:2},{value:"A.9.6. Keyword if let",id:"a96-keyword-if-let",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],p={toc:A};function s(e){let{components:a,...l}=e;return(0,t.kt)("wrapper",(0,i.Z)({},p,l,{components:a,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Pada chapter ini kita akan bahas tentang penerapan ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," untuk kontrol alur program."),(0,t.kt)("p",null,"Seleksi kondisi sendiri merupakan teknik untuk grouping blok kode yang eksekusinya tergantung hasil ekspresi seleksi kondisi. Analoginya mirip seperti fungsi rambu lalu lintas di jalan raya. Kapan kendaraan diperbolehkan melaju dan kapan harus berhenti diatur oleh rambu tersebut. Seleksi kondisi pada program juga kurang lebih sama, kapan sebuah blok kode akan dieksekusi dikontrol."),(0,t.kt)("h2",{id:"a91-keyword-if"},"A.9.1. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"if")),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"if")," adalah salah satu keyword untuk seleksi kondisi di Rust, penggunaannya sangat mudah, yaitu dengan cukup tulis keyword tersebut diikuti dengan data boolean (atau ekspresi logika yang hasilnya boolean), lalu diikuti blok kode. Notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," seperti berikut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"if operasiLogika {\n // blok kode\n}\n")),(0,t.kt)("p",null,"Pada notasi di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"operasiLogika")," bisa diisi dengan variabel yang bertipe ",(0,t.kt)("inlineCode",{parentName:"p"},"bool"),", atau statement ekspresi perbandingan seperti ",(0,t.kt)("inlineCode",{parentName:"p"},"a == b"),". Lebih jelasnya sekarang silakan perhatikan dan coba kode berikut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_a = 3;\nif number_a < 5 {\n println!("number_a adalah dibawah 5");\n}\n\nlet result_a = number_a >= 5;\nif result_a {\n println!("result_a adalah di atas atau sama dengan 5");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"seleksi kondisi",src:n(5632).Z,width:"446",height:"110"})),(0,t.kt)("p",null,"Pada kode di atas ada dua buah blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Yang pertama mengecek hasil ekspresi logika ",(0,t.kt)("inlineCode",{parentName:"p"},"apakah variabel number_a dibawah 5?"),". Jika hasilnya benar atau ",(0,t.kt)("inlineCode",{parentName:"p"},"true")," maka blok kode setelahnya yang diapit tanda kurung kurawal akan dieksekusi, hasilnya menampilkan tulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"angka adalah dibawah 5"),"."),(0,t.kt)("p",null,"Blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," kedua adalah mengecek nilai ",(0,t.kt)("inlineCode",{parentName:"p"},"bool")," variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_a"),". Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_a")," sendiri isinya berasal dari ekspresi logika ",(0,t.kt)("inlineCode",{parentName:"p"},"apakah variabel number_a lebih besar atau sama dengan 5?"),". Jika hasilnya ",(0,t.kt)("inlineCode",{parentName:"p"},"true")," maka blok kode setelahnya (yang diapit tanda kurung kurawal) dieksekusi. Namun, pada contoh di atas, hasilnya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"false"),", karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"number_a")," nilainya adalah tidak lebih besar atau sama dengan 5, dengan demikian blok kode tidak dieksekusi."),(0,t.kt)("h2",{id:"a92-keyword-if-else-if-dan-else"},"A.9.2. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"if"),", ",(0,t.kt)("inlineCode",{parentName:"h2"},"else if"),", dan ",(0,t.kt)("inlineCode",{parentName:"h2"},"else")),(0,t.kt)("p",null,"Jika seleksi kondisi lebih dari satu, gunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"else if")," dan/atau ",(0,t.kt)("inlineCode",{parentName:"p"},"else"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_b = 3;\nif number_b == 2 {\n println!("number_b adalah 2");\n} else if number_b < 2 {\n println!("number_b adalah dibawah 2");\n} else {\n println!("number_b adalah di atas 2");\n}\n')),(0,t.kt)("p",null,"Pada contoh di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"number_b")," yang nilainya ",(0,t.kt)("inlineCode",{parentName:"p"},"3"),", match dengan statement seleksi kondisi ",(0,t.kt)("inlineCode",{parentName:"p"},"else"),", dengan ini maka statement dalam blok kode tersebut dieksekusi, text ",(0,t.kt)("inlineCode",{parentName:"p"},"number_b adalah di atas 2")," muncul."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"seleksi kondisi",src:n(3732).Z,width:"447",height:"96"})),(0,t.kt)("p",null,"Silakan bermain dengan nilai variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"number_b")," untuk coba-coba."),(0,t.kt)("h2",{id:"a93-nested-if"},"A.9.3. Nested ",(0,t.kt)("inlineCode",{parentName:"h2"},"if")),(0,t.kt)("p",null,"Sebuah blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," bisa saja berada di dalam sebuah ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),", dan seperti ini umum terjadi di bahasa pemrograman. Di Rust penerapan nested ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," sama seperti pada bahasa lainnya, yaitu dengan langsung tuliskan saja blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," ke dalam blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_c = 10;\nif number_c > 6 {\n println!("selamat, anda lulus");\n\n if number_c == 10 {\n println!("dengan nilai sempurna");\n } else if number_c > 7 {\n println!("dengan nilai baik");\n } else {\n println!("dengan nilai cukup");\n }\n} else {\n println!("anda tidak lulus");\n\n if number_c < 4 {\n println!("belajar lagi ya");\n } else {\n println!("jangan malas belajar!");\n }\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"nested if",src:n(1669).Z,width:"448",height:"128"})),(0,t.kt)("h2",{id:"a94-returning-from-if"},"A.9.4. Returning From ",(0,t.kt)("inlineCode",{parentName:"h2"},"if")),(0,t.kt)("p",null,(0,t.kt)("em",{parentName:"p"},"Returning from ",(0,t.kt)("inlineCode",{parentName:"em"},"if"))," adalah salah satu cara unik penerapan ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Yang biasanya ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," digunakan untuk eksekusi statements dengan kondisi tertentu, pada case ini statement yang ada pada blok kode if ditampung sebagai ",(0,t.kt)("em",{parentName:"p"},"return value")," atau nilai balik. Teknik ini mirip seperti operasi ",(0,t.kt)("em",{parentName:"p"},"ternary")," hanya saja jumlah kondisinya bisa sebanyak yang kita inginkan."),(0,t.kt)("p",null,"Agar lebih jelas, silakan pelajari dulu kode berikut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_d = 3;\nlet result_d: bool;\n\nif number_d == 2 {\n result_d = true\n} else {\n result_d = false\n}\n\nprintln!("result_d adalah {result_d}");\n')),(0,t.kt)("p",null,"Blok seleksi kondisi pada contoh di atas menjadi penentu nilai variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_d"),". Dengan kebutuhan tersebut, kita bisa juga memanfaatkan ",(0,t.kt)("inlineCode",{parentName:"p"},"let if")," untuk mendapatkan hasil yang ekuivalen."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Pada kode di atas, deklarasi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_d")," dan pengisian nilainya adalah ditulis dalam statement terpisah. Cara ini diperbolehkan pada variable immutable (tanpa perlu membuatnya mutable) selama operasi assignment hanya dilakukan sekali saja setelah deklarasi."),(0,t.kt)("p",{parentName:"blockquote"},"Lebih detailnya dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/variabel#a45-deklarasi-variabel-tanpa-predefined-value"},"Variabel section A.4.5"),".")),(0,t.kt)("p",null,"Kode di atas jika diubah ke bentuk ",(0,t.kt)("inlineCode",{parentName:"p"},"let if")," hasilnya menjadi seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_d = 3;\nlet result_d = if number_d == 2 {\n true\n} else {\n false\n};\nprintln!("result_d adalah {result_d}");\n')),(0,t.kt)("p",null,"Penulisannya cukup unik. Blok kode seleksi kondisi dituliskan sebagai value dari statement ",(0,t.kt)("inlineCode",{parentName:"p"},"let result_d"),". Dan isi blok kode nantinya akan menjadi value untuk variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_d"),", tergantung kondisi mana yang match."),(0,t.kt)("p",null,"Pada contoh di atas, karena ",(0,t.kt)("inlineCode",{parentName:"p"},"number_d")," nilainya 3, maka ",(0,t.kt)("inlineCode",{parentName:"p"},"result_d")," bernilai false. Blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"else")," adalah yang dieksekusi."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"seleksi kondisi",src:n(5146).Z,width:"447",height:"112"})),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Dalam penggunakan kombinasi keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"let")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),", pastikan di akhir blok kode seleksi kondisi dituliskan tanda ",(0,t.kt)("inlineCode",{parentName:"p"},";"),".")),(0,t.kt)("p",null,"O iya, beberapa orang lebih memilih memanfaatkan indentasi untuk mempermudah memahami statement ",(0,t.kt)("inlineCode",{parentName:"p"},"let if"),". Contohnya seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_d = 3;\nlet result_d = \n if number_d == 2 {\n true\n } else {\n false\n };\nprintln!("result_d adalah {result_d}");\n')),(0,t.kt)("h2",{id:"a95-kombinasi-keyword-let-dan-if-dengan-tipe-data-eksplisit"},"A.9.5. Kombinasi Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"let")," dan ",(0,t.kt)("inlineCode",{parentName:"h2"},"if"),", Dengan Tipe Data Eksplisit"),(0,t.kt)("p",null,"Ada situasi dimana dalam pemanfaatan ",(0,t.kt)("inlineCode",{parentName:"p"},"let if")," kita perlu men-specify secara eksplisit tipe data variabel penampung. Caranya sama seperti statement deklarasi variabel beserta tipe data, langsung tulis saja tipe data yang diinginkan setelah nama variabel dan sebelum operator ",(0,t.kt)("inlineCode",{parentName:"p"},"="),"."),(0,t.kt)("p",null,"Pada contoh berikut, variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_e")," saya definisikan tipenya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"string literal &str"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_e = 3;\nlet result_e: &str = if number_e == 2 {\n "angka adalah 2"\n} else if number_e < 2 {\n "angka adalah dibawah 2"\n} else {\n "angka adalah di atas 2"\n};\nprintln!("angka adalah {result_e}");\n')),(0,t.kt)("p",null,"Contoh lain:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let max = 100.0;\nlet string_f = "nilai minimum kelulusan";\nlet result_f: f64 = if string_f == "nilai maksimum kelulusan" {\n max\n} else {\n max * 3.0 / 4.0\n};\nprintln!("angka adalah {result_f}");\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"seleksi kondisi",src:n(3853).Z,width:"450",height:"94"})),(0,t.kt)("h2",{id:"a96-keyword-if-let"},"A.9.6. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"if let")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"if let")," berbeda dengan kombinasi ",(0,t.kt)("inlineCode",{parentName:"p"},"let")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Kita akan bahas topik ini secara terpisah pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"),"."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/seleksi_kondisi_if"},"github.com/novalagung/dasarpemrogramanrust-example/../seleksi_kondisi_if")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-05-control-flow.html"},"https://doc.rust-lang.org/book/ch03-05-control-flow.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.if.html"},"https://doc.rust-lang.org/std/keyword.if.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.else.html"},"https://doc.rust-lang.org/std/keyword.else.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/if_else.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/if_else.html"))))}s.isMDXComponent=!0},5632:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""},3732:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""},1669:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""},5146:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""},3853:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/05cc1716.bb8cde02.js b/assets/js/05cc1716.bb8cde02.js new file mode 100644 index 00000000..4357bd00 --- /dev/null +++ b/assets/js/05cc1716.bb8cde02.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6477],{3905:(e,a,n)=>{n.d(a,{Zo:()=>p,kt:()=>u});var i=n(7294);function t(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function l(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);a&&(i=i.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,i)}return n}function r(e){for(var a=1;a=0||(t[n]=e[n]);return t}(e,a);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(t[n]=e[n])}return t}var k=i.createContext({}),A=function(e){var a=i.useContext(k),n=a;return e&&(n="function"==typeof e?e(a):r(r({},a),e)),n},p=function(e){var a=A(e.components);return i.createElement(k.Provider,{value:a},e.children)},s={inlineCode:"code",wrapper:function(e){var a=e.children;return i.createElement(i.Fragment,{},a)}},o=i.forwardRef((function(e,a){var n=e.components,t=e.mdxType,l=e.originalType,k=e.parentName,p=d(e,["components","mdxType","originalType","parentName"]),o=A(n),u=t,m=o["".concat(k,".").concat(u)]||o[u]||s[u]||l;return n?i.createElement(m,r(r({ref:a},p),{},{components:n})):i.createElement(m,r({ref:a},p))}));function u(e,a){var n=arguments,t=a&&a.mdxType;if("string"==typeof e||t){var l=n.length,r=new Array(l);r[0]=o;var d={};for(var k in a)hasOwnProperty.call(a,k)&&(d[k]=a[k]);d.originalType=e,d.mdxType="string"==typeof e?e:t,r[1]=d;for(var A=2;A{n.r(a),n.d(a,{assets:()=>k,contentTitle:()=>r,default:()=>s,frontMatter:()=>l,metadata:()=>d,toc:()=>A});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:9,title:"A.9. Seleksi Kondisi \u279c if, else if, else",sidebar_label:"A.9. Seleksi Kondisi \u279c if, else if, else"},r=void 0,d={unversionedId:"basic/seleksi-kondisi-if",id:"basic/seleksi-kondisi-if",title:"A.9. Seleksi Kondisi \u279c if, else if, else",description:"Pada chapter ini kita akan bahas tentang penerapan if untuk kontrol alur program.",source:"@site/docs/basic/seleksi-kondisi-if.md",sourceDirName:"basic",slug:"/basic/seleksi-kondisi-if",permalink:"/basic/seleksi-kondisi-if",draft:!1,tags:[],version:"current",sidebarPosition:9,frontMatter:{sidebar_position:9,title:"A.9. Seleksi Kondisi \u279c if, else if, else",sidebar_label:"A.9. Seleksi Kondisi \u279c if, else if, else"},sidebar:"tutorialSidebar",previous:{title:"A.8. Operator",permalink:"/basic/operator"},next:{title:"A.10. Perulangan \u279c while",permalink:"/basic/perulangan-while"}},k={},A=[{value:"A.9.1. Keyword if",id:"a91-keyword-if",level:2},{value:"A.9.2. Keyword if, else if, dan else",id:"a92-keyword-if-else-if-dan-else",level:2},{value:"A.9.3. Nested if",id:"a93-nested-if",level:2},{value:"A.9.4. Returning From if",id:"a94-returning-from-if",level:2},{value:"A.9.5. Kombinasi Keyword let dan if, Dengan Tipe Data Eksplisit",id:"a95-kombinasi-keyword-let-dan-if-dengan-tipe-data-eksplisit",level:2},{value:"A.9.6. Keyword if let",id:"a96-keyword-if-let",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],p={toc:A};function s(e){let{components:a,...l}=e;return(0,t.kt)("wrapper",(0,i.Z)({},p,l,{components:a,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Pada chapter ini kita akan bahas tentang penerapan ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," untuk kontrol alur program."),(0,t.kt)("p",null,"Seleksi kondisi sendiri merupakan teknik untuk grouping blok kode yang eksekusinya tergantung hasil ekspresi seleksi kondisi. Analoginya mirip seperti fungsi rambu lalu lintas di jalan raya. Kapan kendaraan diperbolehkan melaju dan kapan harus berhenti diatur oleh rambu tersebut. Seleksi kondisi pada program juga kurang lebih sama, kapan sebuah blok kode akan dieksekusi dikontrol."),(0,t.kt)("h2",{id:"a91-keyword-if"},"A.9.1. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"if")),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"if")," adalah salah satu keyword untuk seleksi kondisi di Rust, penggunaannya sangat mudah, yaitu dengan cukup tulis keyword tersebut diikuti dengan data boolean (atau ekspresi logika yang hasilnya boolean), lalu diikuti blok kode. Notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," seperti berikut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"if operasiLogika {\n // blok kode\n}\n")),(0,t.kt)("p",null,"Pada notasi di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"operasiLogika")," bisa diisi dengan variabel yang bertipe ",(0,t.kt)("inlineCode",{parentName:"p"},"bool"),", atau statement ekspresi perbandingan seperti ",(0,t.kt)("inlineCode",{parentName:"p"},"a == b"),". Lebih jelasnya sekarang silakan perhatikan dan coba kode berikut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_a = 3;\nif number_a < 5 {\n println!("number_a adalah di bawah 5");\n}\n\nlet result_a = number_a >= 5;\nif result_a {\n println!("result_a adalah di atas atau sama dengan 5");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"seleksi kondisi",src:n(5632).Z,width:"446",height:"110"})),(0,t.kt)("p",null,"Pada kode di atas ada dua buah blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Yang pertama mengecek hasil ekspresi logika ",(0,t.kt)("inlineCode",{parentName:"p"},"apakah variabel number_a di bawah 5?"),". Jika hasilnya benar atau ",(0,t.kt)("inlineCode",{parentName:"p"},"true")," maka blok kode setelahnya yang diapit tanda kurung kurawal akan dieksekusi, hasilnya menampilkan tulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"angka adalah di bawah 5"),"."),(0,t.kt)("p",null,"Blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," kedua adalah mengecek nilai ",(0,t.kt)("inlineCode",{parentName:"p"},"bool")," variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_a"),". Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_a")," sendiri isinya berasal dari ekspresi logika ",(0,t.kt)("inlineCode",{parentName:"p"},"apakah variabel number_a lebih besar atau sama dengan 5?"),". Jika hasilnya ",(0,t.kt)("inlineCode",{parentName:"p"},"true")," maka blok kode setelahnya (yang diapit tanda kurung kurawal) dieksekusi. Namun, pada contoh di atas, hasilnya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"false"),", karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"number_a")," nilainya adalah tidak lebih besar atau sama dengan 5, dengan demikian blok kode tidak dieksekusi."),(0,t.kt)("h2",{id:"a92-keyword-if-else-if-dan-else"},"A.9.2. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"if"),", ",(0,t.kt)("inlineCode",{parentName:"h2"},"else if"),", dan ",(0,t.kt)("inlineCode",{parentName:"h2"},"else")),(0,t.kt)("p",null,"Jika seleksi kondisi lebih dari satu, gunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"else if")," dan/atau ",(0,t.kt)("inlineCode",{parentName:"p"},"else"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_b = 3;\nif number_b == 2 {\n println!("number_b adalah 2");\n} else if number_b < 2 {\n println!("number_b adalah di bawah 2");\n} else {\n println!("number_b adalah di atas 2");\n}\n')),(0,t.kt)("p",null,"Pada contoh di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"number_b")," yang nilainya ",(0,t.kt)("inlineCode",{parentName:"p"},"3"),", match dengan statement seleksi kondisi ",(0,t.kt)("inlineCode",{parentName:"p"},"else"),", dengan ini maka statement dalam blok kode tersebut dieksekusi, text ",(0,t.kt)("inlineCode",{parentName:"p"},"number_b adalah di atas 2")," muncul."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"seleksi kondisi",src:n(3732).Z,width:"447",height:"96"})),(0,t.kt)("p",null,"Silakan bermain dengan nilai variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"number_b")," untuk coba-coba."),(0,t.kt)("h2",{id:"a93-nested-if"},"A.9.3. Nested ",(0,t.kt)("inlineCode",{parentName:"h2"},"if")),(0,t.kt)("p",null,"Sebuah blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," bisa saja berada di dalam sebuah ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),", dan seperti ini umum terjadi di bahasa pemrograman. Di Rust penerapan nested ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," sama seperti pada bahasa lainnya, yaitu dengan langsung tuliskan saja blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," ke dalam blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_c = 10;\nif number_c > 6 {\n println!("selamat, anda lulus");\n\n if number_c == 10 {\n println!("dengan nilai sempurna");\n } else if number_c > 7 {\n println!("dengan nilai baik");\n } else {\n println!("dengan nilai cukup");\n }\n} else {\n println!("anda tidak lulus");\n\n if number_c < 4 {\n println!("belajar lagi ya");\n } else {\n println!("jangan malas belajar!");\n }\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"nested if",src:n(1669).Z,width:"448",height:"128"})),(0,t.kt)("h2",{id:"a94-returning-from-if"},"A.9.4. Returning From ",(0,t.kt)("inlineCode",{parentName:"h2"},"if")),(0,t.kt)("p",null,(0,t.kt)("em",{parentName:"p"},"Returning from ",(0,t.kt)("inlineCode",{parentName:"em"},"if"))," adalah salah satu cara unik penerapan ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Yang biasanya ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," digunakan untuk eksekusi statements dengan kondisi tertentu, pada case ini statement yang ada pada blok kode if ditampung sebagai ",(0,t.kt)("em",{parentName:"p"},"return value")," atau nilai balik. Teknik ini mirip seperti operasi ",(0,t.kt)("em",{parentName:"p"},"ternary")," hanya saja jumlah kondisinya bisa sebanyak yang kita inginkan."),(0,t.kt)("p",null,"Agar lebih jelas, silakan pelajari dulu kode berikut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_d = 3;\nlet result_d: bool;\n\nif number_d == 2 {\n result_d = true\n} else {\n result_d = false\n}\n\nprintln!("result_d adalah {result_d}");\n')),(0,t.kt)("p",null,"Blok seleksi kondisi pada contoh di atas menjadi penentu nilai variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_d"),". Dengan kebutuhan tersebut, kita bisa juga memanfaatkan ",(0,t.kt)("inlineCode",{parentName:"p"},"let if")," untuk mendapatkan hasil yang ekuivalen."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Pada kode di atas, deklarasi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_d")," dan pengisian nilainya adalah ditulis dalam statement terpisah. Cara ini diperbolehkan pada variable immutable (tanpa perlu membuatnya mutable) selama operasi assignment hanya dilakukan sekali saja setelah deklarasi."),(0,t.kt)("p",{parentName:"blockquote"},"Lebih detailnya dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/variabel#a45-deklarasi-variabel-tanpa-predefined-value"},"Variabel section A.4.5"),".")),(0,t.kt)("p",null,"Kode di atas jika diubah ke bentuk ",(0,t.kt)("inlineCode",{parentName:"p"},"let if")," hasilnya menjadi seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_d = 3;\nlet result_d = if number_d == 2 {\n true\n} else {\n false\n};\nprintln!("result_d adalah {result_d}");\n')),(0,t.kt)("p",null,"Penulisannya cukup unik. Blok kode seleksi kondisi dituliskan sebagai value dari statement ",(0,t.kt)("inlineCode",{parentName:"p"},"let result_d"),". Dan isi blok kode nantinya akan menjadi value untuk variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_d"),", tergantung kondisi mana yang match."),(0,t.kt)("p",null,"Pada contoh di atas, karena ",(0,t.kt)("inlineCode",{parentName:"p"},"number_d")," nilainya 3, maka ",(0,t.kt)("inlineCode",{parentName:"p"},"result_d")," bernilai false. Blok kode ",(0,t.kt)("inlineCode",{parentName:"p"},"else")," adalah yang dieksekusi."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"seleksi kondisi",src:n(5146).Z,width:"447",height:"112"})),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Dalam penggunakan kombinasi keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"let")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),", pastikan di akhir blok kode seleksi kondisi dituliskan tanda ",(0,t.kt)("inlineCode",{parentName:"p"},";"),".")),(0,t.kt)("p",null,"O iya, beberapa orang lebih memilih memanfaatkan indentasi untuk mempermudah memahami statement ",(0,t.kt)("inlineCode",{parentName:"p"},"let if"),". Contohnya seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_d = 3;\nlet result_d = \n if number_d == 2 {\n true\n } else {\n false\n };\nprintln!("result_d adalah {result_d}");\n')),(0,t.kt)("h2",{id:"a95-kombinasi-keyword-let-dan-if-dengan-tipe-data-eksplisit"},"A.9.5. Kombinasi Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"let")," dan ",(0,t.kt)("inlineCode",{parentName:"h2"},"if"),", Dengan Tipe Data Eksplisit"),(0,t.kt)("p",null,"Ada situasi di mana dalam pemanfaatan ",(0,t.kt)("inlineCode",{parentName:"p"},"let if")," kita perlu men-specify secara eksplisit tipe data variabel penampung. Caranya sama seperti statement deklarasi variabel beserta tipe data, langsung tulis saja tipe data yang diinginkan setelah nama variabel dan sebelum operator ",(0,t.kt)("inlineCode",{parentName:"p"},"="),"."),(0,t.kt)("p",null,"Pada contoh berikut, variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"result_e")," saya definisikan tipenya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"string literal &str"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let number_e = 3;\nlet result_e: &str = if number_e == 2 {\n "angka adalah 2"\n} else if number_e < 2 {\n "angka adalah di bawah 2"\n} else {\n "angka adalah di atas 2"\n};\nprintln!("angka adalah {result_e}");\n')),(0,t.kt)("p",null,"Contoh lain:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let max = 100.0;\nlet string_f = "nilai minimum kelulusan";\nlet result_f: f64 = if string_f == "nilai maksimum kelulusan" {\n max\n} else {\n max * 3.0 / 4.0\n};\nprintln!("angka adalah {result_f}");\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"seleksi kondisi",src:n(3853).Z,width:"450",height:"94"})),(0,t.kt)("h2",{id:"a96-keyword-if-let"},"A.9.6. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"if let")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"if let")," berbeda dengan kombinasi ",(0,t.kt)("inlineCode",{parentName:"p"},"let")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Kita akan bahas topik ini secara terpisah pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"),"."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/seleksi_kondisi_if"},"github.com/novalagung/dasarpemrogramanrust-example/../seleksi_kondisi_if")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-05-control-flow.html"},"https://doc.rust-lang.org/book/ch03-05-control-flow.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.if.html"},"https://doc.rust-lang.org/std/keyword.if.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.else.html"},"https://doc.rust-lang.org/std/keyword.else.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/if_else.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/if_else.html"))))}s.isMDXComponent=!0},5632:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""},3732:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""},1669:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""},5146:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""},3853:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/0ae704f7.12a11a9d.js b/assets/js/0ae704f7.12a11a9d.js new file mode 100644 index 00000000..998d058c --- /dev/null +++ b/assets/js/0ae704f7.12a11a9d.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[2089],{3905:(e,a,n)=>{n.d(a,{Zo:()=>o,kt:()=>k});var t=n(7294);function r(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function i(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function l(e){for(var a=1;a=0||(r[n]=e[n]);return r}(e,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var m=t.createContext({}),u=function(e){var a=t.useContext(m),n=a;return e&&(n="function"==typeof e?e(a):l(l({},a),e)),n},o=function(e){var a=u(e.components);return t.createElement(m.Provider,{value:a},e.children)},d={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},b=t.forwardRef((function(e,a){var n=e.components,r=e.mdxType,i=e.originalType,m=e.parentName,o=p(e,["components","mdxType","originalType","parentName"]),b=u(n),k=r,s=b["".concat(m,".").concat(k)]||b[k]||d[k]||i;return n?t.createElement(s,l(l({ref:a},o),{},{components:n})):t.createElement(s,l({ref:a},o))}));function k(e,a){var n=arguments,r=a&&a.mdxType;if("string"==typeof e||r){var i=n.length,l=new Array(i);l[0]=b;var p={};for(var m in a)hasOwnProperty.call(a,m)&&(p[m]=a[m]);p.originalType=e,p.mdxType="string"==typeof e?e:r,l[1]=p;for(var u=2;u{n.r(a),n.d(a,{assets:()=>m,contentTitle:()=>l,default:()=>d,frontMatter:()=>i,metadata:()=>p,toc:()=>u});var t=n(7462),r=(n(7294),n(3905));const i={sidebar_position:33,title:"A.33. Pointer & References",sidebar_label:"A.33. Pointer & References"},l=void 0,p={unversionedId:"basic/pointer-references",id:"basic/pointer-references",title:"A.33. Pointer & References",description:"Chapter ini membahas tentang apa itu pointer, references, dan dereferences. Pembelajaran dimulai tentang konsep dan praktik dasar tentang ketiga topik tersebut, kemudian diikuti dengan pembahasan tentang karakteristik pointer & reference.",source:"@site/docs/basic/pointer-references.md",sourceDirName:"basic",slug:"/basic/pointer-references",permalink:"/basic/pointer-references",draft:!1,tags:[],version:"current",sidebarPosition:33,frontMatter:{sidebar_position:33,title:"A.33. Pointer & References",sidebar_label:"A.33. Pointer & References"},sidebar:"tutorialSidebar",previous:{title:"A.32. Basic Memory Management",permalink:"/basic/basic-memory-management"},next:{title:"A.34. Ownership",permalink:"/basic/ownership"}},m={},u=[{value:"A.33.1. Pointer",id:"a331-pointer",level:2},{value:"A.33.2. Reference (operator &)",id:"a332-reference-operator-",level:2},{value:"A.33.3. Dereference (operator *)",id:"a333-dereference-operator-",level:2},{value:"A.33.4. Mutable References (operator &mut)",id:"a334-mutable-references-operator-mut",level:2},{value:"A.33.5. Aturan Reference",id:"a335-aturan-reference",level:2},{value:"A.33.6. Karakteristik pointer & reference",id:"a336-karakteristik-pointer--reference",level:2},{value:"\u25c9 Contoh ke-1",id:"-contoh-ke-1",level:3},{value:"\u25c9 Contoh ke-2",id:"-contoh-ke-2",level:3},{value:"A.33.7. Reference & borrowing",id:"a337-reference--borrowing",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:u};function d(e){let{components:a,...i}=e;return(0,r.kt)("wrapper",(0,t.Z)({},o,i,{components:a,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Chapter ini membahas tentang apa itu pointer, references, dan dereferences. Pembelajaran dimulai tentang konsep dan praktik dasar tentang ketiga topik tersebut, kemudian diikuti dengan pembahasan tentang karakteristik pointer & reference."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Penulis tekankan bahwa ",(0,r.kt)("strong",{parentName:"p"},"pada chapter ini, konsep reference yang dibahas adalah dalam konteks programming secara general.")," Tidak dari sudut pandang ownership.")),(0,r.kt)("p",null,"Reference pada Rust memiliki keunikan, tapi kita akan bahas itu secara terpisah pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),". Untuk sekarang silakan ikuti terlebih dahulu pembahasan chapter-per-chapter secara berurutan."),(0,r.kt)("h2",{id:"a331-pointer"},"A.33.1. Pointer"),(0,r.kt)("p",null,"Pointer artinya adalah alamat memori. Variabel pointer artinya adalah variabel yang berisi alamat memory (hanya alamat memory-nya saja, bukan value yang sebenarnya)."),(0,r.kt)("p",null,"Di Rust, variabel pointer ditandai dengan adanya karakter ",(0,r.kt)("inlineCode",{parentName:"p"},"&")," pada tipe data. Sebagai contoh ",(0,r.kt)("inlineCode",{parentName:"p"},"&i32")," artinya adalah tipe data pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),". Contoh lain adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"&bool")," yang merupakan tipe data pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"bool"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Ada dua jenis tipe pointer di Rust programming, yaitu smart pointer dan raw pointer. Pada chapter ini kita tidak membahasnya karena termasuk topik yang cukup advance. Pembahasan akan ada pada chapter terpisah ",(0,r.kt)("a",{parentName:"p",href:"#/wip/smart-pointer-raw-pointer"},"Smart Pointer vs Raw Pointer"),".")),(0,r.kt)("h2",{id:"a332-reference-operator-"},"A.33.2. Reference (operator ",(0,r.kt)("inlineCode",{parentName:"h2"},"&"),")"),(0,r.kt)("p",null,"Ok, selanjutnya apa itu ",(0,r.kt)("em",{parentName:"p"},"reference"),"? Istilah ini sudah beberapa kali disinggung pada chapter sebelum-sebelumnya."),(0,r.kt)("p",null,"Reference artinya adalah pointer dari sebuah variabel atau data. Operasi pengambilan pointer dari variabel disebut dengan ",(0,r.kt)("em",{parentName:"p"},"referencing"),", caranya dilakukan dengan menggunakan karakter ",(0,r.kt)("inlineCode",{parentName:"p"},"&"),"."),(0,r.kt)("p",null,"O iya, semua jenis variabel bisa diambil nilai pointernya. Contohnya bisa dilihat berikut ini:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'let number: i32 = 24;\nprintln!("value: {:?}", number);\n\nlet pointer_number: &i32 = &number;\nprintln!("pointer: {:p}", pointer_number);\n')),(0,r.kt)("p",null,"Pada contoh di atas, sebuah variabel dideklarasikan bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," dengan tipe data adalah numerik dan value ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),". Variabel tersebut jika di-print akan muncul nilainya, yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),"."),(0,r.kt)("p",null,"Ada satu lagi variabel yang dideklarasikan yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number"),", yang nilainya adalah ",(0,r.kt)("em",{parentName:"p"},"reference")," dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Cara pengambilan reference yang dilihat pada contoh, yaitu dengan menambahkan operator ",(0,r.kt)("inlineCode",{parentName:"p"},"&")," pada variabel yang ingin diambil pointernya."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"// variabel pointer_number nilainya adalah reference variabel number.\n// contoh statement:\nlet pointer_number_1 = &number;\nlet pointer_number_2: &i32 = &number;\n")),(0,r.kt)("p",null,"By default, ketika variabel pointer di print, yang muncul adalah ",(0,r.kt)("em",{parentName:"p"},"underlying value")," atau nilai sebenarnya, yang pada contoh di atas adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),". Untuk menampilkan alamat memory gunakan formatted print ",(0,r.kt)("inlineCode",{parentName:"p"},"{:p}"),"."),(0,r.kt)("p",null,"Coba jalankan program kemudian lihat hasilnya. Value muncul sesuai dengan yang ditulis, sedangkan pointer memunculkan data alamat memory yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"0xa3bbeff6dc"),"."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"pointer & reference",src:n(1007).Z,width:"494",height:"320"})),(0,r.kt)("p",null,"Seperti yang sudah dijelaskan di awal bahwa variabel pointer isinya adalah alamat memory. Penulisannya dalam notasi heksadesimal diawali dengan karakter ",(0,r.kt)("inlineCode",{parentName:"p"},"0x"),". Contohnya adalah nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"0xa3bbeff6dc")," yang merupakan alamat memory, yang alamat tersebut adalah reference ke pemilik data sebenarnya (yaitu variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),")."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Di environment lokal masing-masing, alamat memory yang muncul sangat mungkin berbeda karena alokasi adalah random")),(0,r.kt)("h2",{id:"a333-dereference-operator-"},"A.33.3. Dereference (operator ",(0,r.kt)("inlineCode",{parentName:"h2"},"*"),")"),(0,r.kt)("p",null,"Kita sudah belajar cara mengambil data pointer dari sebuah variabel. Pada bagian ini kita belajar cara mengambil nilai sebenarnya atau underlying value dari sebuah pointer (istilahnya adalah ",(0,r.kt)("em",{parentName:"p"},"dereference"),")."),(0,r.kt)("p",null,"Cara melakukan operasi ",(0,r.kt)("em",{parentName:"p"},"dereferencing")," adalah dengan menambahkan karakter ",(0,r.kt)("inlineCode",{parentName:"p"},"*")," pada variabel pointer. Contohnya seperti berikut, silakan tambahkan kode ini ke program yang sudah ditulis."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'// variabel underlying_value nilainya adalah nilai sebenarnya dari pointer pointer_number\nlet underlying_value = *pointer_number;\nprintln!("value: {:}", underlying_value);\n')),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Dereference",src:n(6356).Z,width:"467",height:"412"})),(0,r.kt)("p",null,"Bisa dilihat, nilai dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"underlying_value")," adalah nilai sebenarnya dari pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number"),", yang pastinya adalah sama dengan nilai variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),"."),(0,r.kt)("h2",{id:"a334-mutable-references-operator-mut"},"A.33.4. Mutable References (operator ",(0,r.kt)("inlineCode",{parentName:"h2"},"&mut"),")"),(0,r.kt)("p",null,"By default, reference sifatnya ",(0,r.kt)("em",{parentName:"p"},"read-only")," atau ",(0,r.kt)("em",{parentName:"p"},"immutable"),", artinya tidak bisa diubah ",(0,r.kt)("em",{parentName:"p"},"underlying-value"),"-nya."),(0,r.kt)("p",null,"Jika ada variabel (sebut saja X) yang merupakan underlying value dari reference variabel lain, maka perubahan value pada variabel X tersebut ",(0,r.kt)("em",{parentName:"p"},"default"),"-nya menghasilkan error."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'let mut number = 24;\nprintln!("number: {}", number);\n\nlet pointer_number = &number;\nprintln!("pointer_number: {:p}", pointer_number);\n\n*pointer_number = 12;\n\nprintln!("*pointer_number: {}", *pointer_number);\nprintln!("number: {}", number);\n')),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Pointer & reference",src:n(8992).Z,width:"715",height:"343"})),(0,r.kt)("p",null,"Pada gambar di atas bisa dilihat, variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," merupakan reference dari variabel mutable ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Setelahnya ada statement ",(0,r.kt)("inlineCode",{parentName:"p"},"*pointer_number = 12"),", yang artinya adalah underlying value dari variabel pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," diakses kemudian di-isi nilainya dengan angka ",(0,r.kt)("inlineCode",{parentName:"p"},"12"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Statement ",(0,r.kt)("inlineCode",{parentName:"p"},"*pointer_number = 12")," tidak menghasilkan error, dan statement ini berbeda dengan ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number = 12"),"."),(0,r.kt)("p",{parentName:"blockquote"},"Statement ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number = 12")," menghasilkan error karena ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," adalah variabel bertipe ",(0,r.kt)("strong",{parentName:"p"},"pointer ",(0,r.kt)("inlineCode",{parentName:"strong"},"i32")," atau ",(0,r.kt)("inlineCode",{parentName:"strong"},"&i32")," (bukan ",(0,r.kt)("inlineCode",{parentName:"strong"},"i32"),")"),". Jika ingin mengubah nilainya perlu mengakses dulu underlying value-nya menggunakan ",(0,r.kt)("inlineCode",{parentName:"p"},"*pointer_number"),".")),(0,r.kt)("p",null,"Ok, lalu kenapa muncul error? Di gambar terlihat ada garis merah dan popup pesan error muncul, padahal tidak ada yang salah dengan statement-nya."),(0,r.kt)("p",null,"Penyebab erronya bukan dari statement tersebut, tetapi pada baris statement pengambilan reference variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Statement ",(0,r.kt)("inlineCode",{parentName:"p"},"&number")," artinya adalah mengambil reference dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Di atas sempat kita bahas bahwa ",(0,r.kt)("em",{parentName:"p"},"by default")," sebuah reference tidak bisa diubah nilainya (",(0,r.kt)("em",{parentName:"p"},"immutable"),"), dan ini adalah penyebab error yang dialami."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Silakan perhatikan pesan di popup error message agar mudah untuk tau di mana sumber masalahnya.")),(0,r.kt)("p",null,"Perubahan isi nilai variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," tidak menghasilkan error, hal ini karena number adalah variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," adalah mutable. Sedangkan operasi perubahan nilai variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"*pointer_number")," pada contoh di atas, dianggap sebagai error karena variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," reference-nya adalah bukan mutable (meskipun reference diperoleh dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," yang notabene mutable)."),(0,r.kt)("p",null,"Solusi dari error di atas adalah menerapkan ",(0,r.kt)("strong",{parentName:"p"},"mutable reference"),". Mutable reference sama seperti reference biasa tetapi nilainya diperbolehkan untuk diubah (mutable). Caranya pengaksesan mutable reference dilakukan menggunakan operator ",(0,r.kt)("inlineCode",{parentName:"p"},"&mut"),"."),(0,r.kt)("p",null,"Ok, sekarang ubah statement berikut ini, lalu jalankan program:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"// before\nlet pointer_number = &number;\n\n// after\nlet pointer_number = &mut number;\n")),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Pointer & reference",src:n(5127).Z,width:"488",height:"417"})),(0,r.kt)("p",null,"Deklarasi variable pointer yang menampung mutable reference bisa via metode penulisan ",(0,r.kt)("em",{parentName:"p"},"manifest typing")," atau ",(0,r.kt)("em",{parentName:"p"},"type inference"),". Pada contoh berikut, variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," tipe data-nya adalah mutable reference ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", maka penulisan tipe data adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"&mut i32"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"// type inference\nlet pointer_number = &mut number;\n\n// manifest typing\nlet pointer_number: &mut i32 = &mut number;\n")),(0,r.kt)("p",null,"Pada contoh ini kita menggunakan tipe data primitif, lalu bagaimana dengan tipe data non-primitive seperti custom type struct atau ",(0,r.kt)("inlineCode",{parentName:"p"},"String"),", apakah penerapan mutable reference juga sama? Jawabannya tidak. Ada beberapa perbedaan dan nantinya kita akan bahas secara detail pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),". Untuk sekarang, dianjurkan mengikuti pembahasan chapter per chapter secara urut."),(0,r.kt)("h2",{id:"a335-aturan-reference"},"A.33.5. Aturan Reference"),(0,r.kt)("p",null,"Ada dua aturan penting yang harus dipatuhi dalam penerapan reference baik mutable atau immutable reference."),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Dalam waktu yang sama, hanya boleh ada satu mutable reference atau banyak immutable reference (keduanya tidak bisa bersamaan, harus salah satu)."),(0,r.kt)("li",{parentName:"ul"},"Reference harus selalu valid.")),(0,r.kt)("p",null,"Mengenai penjelasan tentang dua aturan penting di atas akan bahas pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),"."),(0,r.kt)("h2",{id:"a336-karakteristik-pointer--reference"},"A.33.6. Karakteristik pointer & reference"),(0,r.kt)("p",null,"Kita sudah belajar tentang definisi beserta cara penerapan pointer, reference, dereference, dan mutable reference. Sekarang lanjut ke pembahasan tentang karakteristik dari pointer & reference."),(0,r.kt)("p",null,"Pointer merupakan variabel yang isinya adalah alamat memory (bukan nilai sebenarnya). Dan reference adalah alamat memory suatu data/variabel. Dengan menerapkan keduanya kita bisa menerapkan efisiensi penggunaan memory yang dampaknya sangat positif terhadap performa program."),(0,r.kt)("p",null,"Sekarang perhatikan statement berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"let number_one = 24;\nlet number_two = number_one;\n")),(0,r.kt)("p",null,"Variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one")," adalah data numerik bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", eksekusi statement tersebut menghasilkan alokasi memori selebar 32-bit."),(0,r.kt)("p",null,"Kemudian ada lagi statement ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two"),". Meskipun nilainya didapat dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one"),", yang terjadi di balik layar adalah Rust akan mengalokasikan lagi alamat memory selebar 32-bit untuk menampung data ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two")," yang didapat dari hasil operasi ",(0,r.kt)("strong",{parentName:"p"},"copy")," dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Semua variabel primitif di Rust mengadopsi ",(0,r.kt)("em",{parentName:"p"},"copy semantics"),", yang artinya jika variabel tersebut digunakan dalam statement assignment, maka nilai akan di-duplikasi untuk kemudian ditampung pada variabel baru."),(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai ",(0,r.kt)("em",{parentName:"p"},"copy semantics")," dibahas pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/ownership#a334-copy-semantics-vs-move-semantics"},"Ownership"),".")),(0,r.kt)("p",null,"Selanjutnya, bandingkan dengan statement berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"let number_one = 24;\nlet number_two = &number_one;\n")),(0,r.kt)("p",null,"Pada contoh di atas, variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one")," datanya disimpan di memory dengan lebar 32-bit (masih sama seperti contoh sebelumnya). Namun ada yang berbeda dengan dengan variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two"),", variabel ini adalah variabel pointer yang nilainya reference ke alamat memory data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one"),". Yang terjadi di belakang layar, Rust tidak mengalokasikan lagi memory selebar 32-bit untuk menampung data ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two"),", melainkan menggunakan alamat memory data ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one")," sebagai reference, menjadikannya sebagai alamat tujuan variabel pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two"),"."),(0,r.kt)("p",null,"Ok, lanjut. Per sekarang, reference variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one")," dan ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two")," adalah sama. Dengan ini, jika dicontohkan isi data satu variabel diubah, maka variabel lainnya juga akan berubah, hal karena reference-nya adalah sama."),(0,r.kt)("h3",{id:"-contoh-ke-1"},"\u25c9 Contoh ke-1"),(0,r.kt)("p",null,"Contoh penerapannya bisa kita lihat pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/pointer-references#a315-karakteristik-pointer--reference"},"Pointer & References")," ini, di situ bisa dilihat ada variabel mutable ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," dan ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," yang reference-nya adalah sama dengan variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Ketika underlying value ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," diubah (dari ",(0,r.kt)("inlineCode",{parentName:"p"},"24")," ke ",(0,r.kt)("inlineCode",{parentName:"p"},"12"),"), isi data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," juga berubah."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Pointer & reference",src:n(5127).Z,width:"488",height:"417"})),(0,r.kt)("h3",{id:"-contoh-ke-2"},"\u25c9 Contoh ke-2"),(0,r.kt)("p",null,"Agar makin familiar dengan cara penerapan pointer & reference, silakan lanjut dengan praktik program berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="Cargo.toml"',title:'"Cargo.toml"'},'# ...\n\n[dependencies]\nrand = "0.8.5"\n')),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="main.rs"',title:'"main.rs"'},'use rand::Rng;\n\nfn main() {\n let mut number = 24;\n println!("number: {}", number);\n\n for _ in 0..=5 {\n change_value(&mut number);\n println!("number: {}", number);\n }\n}\n\nfn change_value(n: &mut i32) {\n *n = generate_random_number()\n}\n\nfn generate_random_number() -> i32 {\n rand::thread_rng().gen_range(0..100)\n}\n')),(0,r.kt)("p",null,"Pada sederhana di atas, ada sebuah variable mutable bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," yang dideklarasikan. Variabel tersebut kemudian diakses ",(0,r.kt)("em",{parentName:"p"},"mutable reference"),"-nya untuk dijadikan argumen statement pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"change_value"),"."),(0,r.kt)("p",null,"Di dalam fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"change_value"),", disiapkan 1 buah angka random hasil pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"generate_random_number"),", yang angka tersebut kemudian dijadikan sebagai nilai baru variable mutable reference yang mengarah ke ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Perubahan dalam fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"change_value")," akan mempengaruhi data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),", karena reference-nya adalah sama."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Pointer & reference",src:n(3319).Z,width:"457",height:"204"})),(0,r.kt)("h2",{id:"a337-reference--borrowing"},"A.33.7. Reference & borrowing"),(0,r.kt)("p",null,"Di Rust, reference (atau pengaksesan alamat memory suatu data) memiliki hubungan yang sangat erat dengan konsep ",(0,r.kt)("strong",{parentName:"p"},"borrowing"),"."),(0,r.kt)("p",null,"Ketika kita mengambil reference suatu data, yang terjadi sebenarnya adalah kita meminjam data tersebut dari owner/pemilik sebenarnya."),(0,r.kt)("p",null,"Pada chapter ini kita tidak akan membahasnya lebih jauh lagi, karena akan ada chapter lain yang fokus mengulas topik tersebut secara detail, yaitu chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),"."),(0,r.kt)("p",null,"Untuk sekarang penulis anjurkan untuk lanjut ke chapter berikutnya secara urut."),(0,r.kt)("hr",null),(0,r.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,r.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,r.kt)("pre",null,(0,r.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/pointer_references"},"github.com/novalagung/dasarpemrogramanrust-example/../pointer_references")),(0,r.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-02-references-and-borrowing.html"},"https://doc.rust-lang.org/book/ch04-02-references-and-borrowing.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/marker/trait.Copy.html"},"https://doc.rust-lang.org/std/marker/trait.Copy.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Reference_(computer_science)"},"https://en.wikipedia.org/wiki/Reference_(computer_science)")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://progressivecoder.com/understanding-rust-ownership-and-borrowing-with-examples/"},"https://progressivecoder.com/understanding-rust-ownership-and-borrowing-with-examples/")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://os.phil-opp.com/heap-allocation/"},"https://os.phil-opp.com/heap-allocation/"))))}d.isMDXComponent=!0},1007:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/pointer-references-1-f103aebf5517db2d9de53c1fe39be319.png"},6356:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/pointer-references-2-8fdb7f20f15717b7d97311d81ffe005d.png"},8992:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/pointer-references-3-7b60c4bdc0392eb962187ee76ac2ab54.png"},5127:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/pointer-references-4-ba178a14defb70eb04b45ce050af0e6e.png"},3319:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/0ae704f7.5940e697.js b/assets/js/0ae704f7.5940e697.js deleted file mode 100644 index 850fceca..00000000 --- a/assets/js/0ae704f7.5940e697.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[2089],{3905:(e,a,n)=>{n.d(a,{Zo:()=>o,kt:()=>k});var t=n(7294);function r(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function i(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function l(e){for(var a=1;a=0||(r[n]=e[n]);return r}(e,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var m=t.createContext({}),u=function(e){var a=t.useContext(m),n=a;return e&&(n="function"==typeof e?e(a):l(l({},a),e)),n},o=function(e){var a=u(e.components);return t.createElement(m.Provider,{value:a},e.children)},d={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},b=t.forwardRef((function(e,a){var n=e.components,r=e.mdxType,i=e.originalType,m=e.parentName,o=p(e,["components","mdxType","originalType","parentName"]),b=u(n),k=r,s=b["".concat(m,".").concat(k)]||b[k]||d[k]||i;return n?t.createElement(s,l(l({ref:a},o),{},{components:n})):t.createElement(s,l({ref:a},o))}));function k(e,a){var n=arguments,r=a&&a.mdxType;if("string"==typeof e||r){var i=n.length,l=new Array(i);l[0]=b;var p={};for(var m in a)hasOwnProperty.call(a,m)&&(p[m]=a[m]);p.originalType=e,p.mdxType="string"==typeof e?e:r,l[1]=p;for(var u=2;u{n.r(a),n.d(a,{assets:()=>m,contentTitle:()=>l,default:()=>d,frontMatter:()=>i,metadata:()=>p,toc:()=>u});var t=n(7462),r=(n(7294),n(3905));const i={sidebar_position:33,title:"A.33. Pointer & References",sidebar_label:"A.33. Pointer & References"},l=void 0,p={unversionedId:"basic/pointer-references",id:"basic/pointer-references",title:"A.33. Pointer & References",description:"Chapter ini membahas tentang apa itu pointer, references, dan dereferences. Pembelajaran dimulai tentang konsep dan praktik dasar tentang ketiga topik tersebut, kemudian diikuti dengan pembahasan tentang karakteristik pointer & reference.",source:"@site/docs/basic/pointer-references.md",sourceDirName:"basic",slug:"/basic/pointer-references",permalink:"/basic/pointer-references",draft:!1,tags:[],version:"current",sidebarPosition:33,frontMatter:{sidebar_position:33,title:"A.33. Pointer & References",sidebar_label:"A.33. Pointer & References"},sidebar:"tutorialSidebar",previous:{title:"A.32. Basic Memory Management",permalink:"/basic/basic-memory-management"},next:{title:"A.34. Ownership",permalink:"/basic/ownership"}},m={},u=[{value:"A.33.1. Pointer",id:"a331-pointer",level:2},{value:"A.33.2. Reference (operator &)",id:"a332-reference-operator-",level:2},{value:"A.33.3. Dereference (operator *)",id:"a333-dereference-operator-",level:2},{value:"A.33.4. Mutable References (operator &mut)",id:"a334-mutable-references-operator-mut",level:2},{value:"A.33.5. Aturan Reference",id:"a335-aturan-reference",level:2},{value:"A.33.6. Karakteristik pointer & reference",id:"a336-karakteristik-pointer--reference",level:2},{value:"\u25c9 Contoh ke-1",id:"-contoh-ke-1",level:3},{value:"\u25c9 Contoh ke-2",id:"-contoh-ke-2",level:3},{value:"A.33.7. Reference & borrowing",id:"a337-reference--borrowing",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:u};function d(e){let{components:a,...i}=e;return(0,r.kt)("wrapper",(0,t.Z)({},o,i,{components:a,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Chapter ini membahas tentang apa itu pointer, references, dan dereferences. Pembelajaran dimulai tentang konsep dan praktik dasar tentang ketiga topik tersebut, kemudian diikuti dengan pembahasan tentang karakteristik pointer & reference."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Penulis tekankan bahwa ",(0,r.kt)("strong",{parentName:"p"},"pada chapter ini, konsep reference yang dibahas adalah dalam konteks programming secara general.")," Tidak dari sudut pandang ownership.")),(0,r.kt)("p",null,"Reference pada Rust memiliki keunikan, tapi kita akan bahas itu secara terpisah pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),". Untuk sekarang silakan ikuti terlebih dahulu pembahasan chapter-per-chapter secara berurutan."),(0,r.kt)("h2",{id:"a331-pointer"},"A.33.1. Pointer"),(0,r.kt)("p",null,"Pointer artinya adalah alamat memori. Variabel pointer artinya adalah variabel yang berisi alamat memory (hanya alamat memory-nya saja, bukan value yang sebenarnya)."),(0,r.kt)("p",null,"Di Rust, variabel pointer ditandai dengan adanya karakter ",(0,r.kt)("inlineCode",{parentName:"p"},"&")," pada tipe data. Sebagai contoh ",(0,r.kt)("inlineCode",{parentName:"p"},"&i32")," artinya adalah tipe data pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),". Contoh lain adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"&bool")," yang merupakan tipe data pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"bool"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Ada dua jenis tipe pointer di Rust programming, yaitu smart pointer dan raw pointer. Pada chapter ini kita tidak membahasnya karena termasuk topik yang cukup advance. Pembahasan akan ada pada chapter terpisah ",(0,r.kt)("a",{parentName:"p",href:"#/wip/smart-pointer-raw-pointer"},"Smart Pointer vs Raw Pointer"),".")),(0,r.kt)("h2",{id:"a332-reference-operator-"},"A.33.2. Reference (operator ",(0,r.kt)("inlineCode",{parentName:"h2"},"&"),")"),(0,r.kt)("p",null,"Ok, selanjutnya apa itu ",(0,r.kt)("em",{parentName:"p"},"reference"),"? Istilah ini sudah beberapa kali disinggung pada chapter sebelum-sebelumnya."),(0,r.kt)("p",null,"Reference artinya adalah pointer dari sebuah variabel atau data. Operasi pengambilan pointer dari variabel disebut dengan ",(0,r.kt)("em",{parentName:"p"},"referencing"),", caranya dilakukan dengan menggunakan karakter ",(0,r.kt)("inlineCode",{parentName:"p"},"&"),"."),(0,r.kt)("p",null,"O iya, semua jenis variabel bisa diambil nilai pointernya. Contohnya bisa dilihat berikut ini:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'let number: i32 = 24;\nprintln!("value: {:?}", number);\n\nlet pointer_number: &i32 = &number;\nprintln!("pointer: {:p}", pointer_number);\n')),(0,r.kt)("p",null,"Pada contoh di atas, sebuah variabel dideklarasikan bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," dengan tipe data adalah numerik dan value ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),". Variabel tersebut jika di-print akan muncul nilainya, yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),"."),(0,r.kt)("p",null,"Ada satu lagi variabel yang dideklarasikan yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number"),", yang nilainya adalah ",(0,r.kt)("em",{parentName:"p"},"reference")," dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Cara pengambilan reference yang dilihat pada contoh, yaitu dengan menambahkan operator ",(0,r.kt)("inlineCode",{parentName:"p"},"&")," pada variabel yang ingin dimabil pointernya."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"// variabel pointer_number nilainya adalah reference variabel number.\n// contoh statement:\nlet pointer_number_1 = &number;\nlet pointer_number_2: &i32 = &number;\n")),(0,r.kt)("p",null,"By default, ketika variabel pointer di print, yang muncul adalah ",(0,r.kt)("em",{parentName:"p"},"underlying value")," atau nilai sebenarnya, yang pada contoh di atas adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"24"),". Untuk menampilkan alamat memory gunakan formatted print ",(0,r.kt)("inlineCode",{parentName:"p"},"{:p}"),"."),(0,r.kt)("p",null,"Coba jalankan program kemudian lihat hasilnya. Value muncul sesuai dengan yang ditulis, sedangkan pointer memunculkan data alamat memory yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"0xa3bbeff6dc"),"."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"pointer & reference",src:n(1007).Z,width:"494",height:"320"})),(0,r.kt)("p",null,"Seperti yang sudah dijelaskan di awal bahwa variabel pointer isinya adalah alamat memory. Penulisannya dalam notasi heksadesimal diawali dengan karakter ",(0,r.kt)("inlineCode",{parentName:"p"},"0x"),". Contohnya adalah nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"0xa3bbeff6dc")," yang merupakan alamat memory, yang alamat tersebut adalah reference ke pemilik data sebenarnya (yaitu variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),")."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Di environment lokal masing-masing, alamat memory yang muncul sangat mungkin berbeda karena alokasi adalah random")),(0,r.kt)("h2",{id:"a333-dereference-operator-"},"A.33.3. Dereference (operator ",(0,r.kt)("inlineCode",{parentName:"h2"},"*"),")"),(0,r.kt)("p",null,"Kita sudah belajar cara mengambil data pointer dari sebuah variabel. Pada bagian ini kita belajar cara mengambil nilai sebenarnya atau underlying value dari sebuah pointer (istilahnya adalah ",(0,r.kt)("em",{parentName:"p"},"dereference"),")."),(0,r.kt)("p",null,"Cara melakukan operasi ",(0,r.kt)("em",{parentName:"p"},"dereferencing")," adalah dengan menambahkan karakter ",(0,r.kt)("inlineCode",{parentName:"p"},"*")," pada variabel pointer. Contohnya seperti berikut, silakan tambahkan kode ini ke program yang sudah ditulis."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'// variabel underlying_value nilainya adalah nilai sebenarnya dari pointer pointer_number\nlet underlying_value = *pointer_number;\nprintln!("value: {:}", underlying_value);\n')),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Dereference",src:n(6356).Z,width:"467",height:"412"})),(0,r.kt)("p",null,"Bisa dilihat, nilai dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"underlying_value")," adalah nilai sebenarnya dari pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number"),", yang pastinya adalah sama dengan nilai variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),"."),(0,r.kt)("h2",{id:"a334-mutable-references-operator-mut"},"A.33.4. Mutable References (operator ",(0,r.kt)("inlineCode",{parentName:"h2"},"&mut"),")"),(0,r.kt)("p",null,"By default, reference sifatnya ",(0,r.kt)("em",{parentName:"p"},"read-only")," atau ",(0,r.kt)("em",{parentName:"p"},"immutable"),", artinya tidak bisa diubah ",(0,r.kt)("em",{parentName:"p"},"underlying-value"),"-nya."),(0,r.kt)("p",null,"Jika ada variabel (sebut saja X) yang merupakan underlying value dari reference variabel lain, maka perubahan value pada variabel X tersebut ",(0,r.kt)("em",{parentName:"p"},"default"),"-nya menghasilkan error."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'let mut number = 24;\nprintln!("number: {}", number);\n\nlet pointer_number = &number;\nprintln!("pointer_number: {:p}", pointer_number);\n\n*pointer_number = 12;\n\nprintln!("*pointer_number: {}", *pointer_number);\nprintln!("number: {}", number);\n')),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Pointer & reference",src:n(8992).Z,width:"715",height:"343"})),(0,r.kt)("p",null,"Pada gambar di atas bisa dilihat, variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," merupakan reference dari variabel mutable ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Setelahnya ada statement ",(0,r.kt)("inlineCode",{parentName:"p"},"*pointer_number = 12"),", yang artinya adalah underlying value dari variabel pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," diakses kemudian di-isi nilainya dengan angka ",(0,r.kt)("inlineCode",{parentName:"p"},"12"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Statement ",(0,r.kt)("inlineCode",{parentName:"p"},"*pointer_number = 12")," tidak menghasilkan error, dan statement ini berbeda dengan ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number = 12"),"."),(0,r.kt)("p",{parentName:"blockquote"},"Statement ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number = 12")," menghasilkan error karena ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," adalah variabel bertipe ",(0,r.kt)("strong",{parentName:"p"},"pointer ",(0,r.kt)("inlineCode",{parentName:"strong"},"i32")," atau ",(0,r.kt)("inlineCode",{parentName:"strong"},"&i32")," (bukan ",(0,r.kt)("inlineCode",{parentName:"strong"},"i32"),")"),". Jika ingin mengubah nilainya perlu mengakses dulu underlying value-nya menggunakan ",(0,r.kt)("inlineCode",{parentName:"p"},"*pointer_number"),".")),(0,r.kt)("p",null,"Ok, lalu kenapa muncul error? Di gambar terlihat ada garis merah dan popup pesan error muncul, padahal tidak ada yang salah dengan statement-nya."),(0,r.kt)("p",null,"Penyebab erronya bukan dari statement tersebut, tetapi pada baris statement pengambilan reference variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Statement ",(0,r.kt)("inlineCode",{parentName:"p"},"&number")," artinya adalah mengambil reference dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Di atas sempat kita bahas bahwa ",(0,r.kt)("em",{parentName:"p"},"by default")," sebuah reference tidak bisa diubah nilainya (",(0,r.kt)("em",{parentName:"p"},"immutable"),"), dan ini adalah penyebab error yang dialami."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Silakan perhatikan pesan di popup error message agar mudah untuk tau dimana sumber masalahnya.")),(0,r.kt)("p",null,"Perubahan isi nilai variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," tidak menghasilkan error, hal ini karena number adalah variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," adalah mutable. Sedangkan operasi perubahan nilai variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"*pointer_number")," pada contoh di atas, dianggap sebagai error karena variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," reference-nya adalah bukan mutable (meskipun reference diperoleh dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," yang notabene mutable)."),(0,r.kt)("p",null,"Solusi dari error di atas adalah menerapkan ",(0,r.kt)("strong",{parentName:"p"},"mutable reference"),". Mutable reference sama seperti reference biasa tetapi nilainya diperbolehkan untuk diubah (mutable). Caranya pengaksesan mutable reference dilakukan menggunakan operator ",(0,r.kt)("inlineCode",{parentName:"p"},"&mut"),"."),(0,r.kt)("p",null,"Ok, sekarang ubah statement berikut ini, lalu jalankan program:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"// before\nlet pointer_number = &number;\n\n// after\nlet pointer_number = &mut number;\n")),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Pointer & reference",src:n(5127).Z,width:"488",height:"417"})),(0,r.kt)("p",null,"Deklarasi variable pointer yang menampung mutable reference bisa via metode penulisan ",(0,r.kt)("em",{parentName:"p"},"manifest typing")," atau ",(0,r.kt)("em",{parentName:"p"},"type inference"),". Pada contoh berikut, variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," tipe data-nya adalah mutable reference ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", maka penulisan tipe data adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"&mut i32"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"// type inference\nlet pointer_number = &mut number;\n\n// manifest typing\nlet pointer_number: &mut i32 = &mut number;\n")),(0,r.kt)("p",null,"Pada contoh ini kita menggunakan tipe data primitif, lalu bagaimana dengan tipe data non-primitive seperti custom type struct atau ",(0,r.kt)("inlineCode",{parentName:"p"},"String"),", apakah penerapan mutable reference juga sama? Jawabannya tidak. Ada beberapa perbedaan dan nantinya kita akan bahas secara detail pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),". Untuk sekarang, dianjurkan mengikuti pembahasan chapter per chapter secara urut."),(0,r.kt)("h2",{id:"a335-aturan-reference"},"A.33.5. Aturan Reference"),(0,r.kt)("p",null,"Ada dua aturan penting yang harus dipatuhi dalam penerapan reference baik mutable atau immutable reference."),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Dalam waktu yang sama, hanya boleh ada satu mutable reference atau banyak immutable reference (keduanya tidak bisa bersamaan, harus salah satu)."),(0,r.kt)("li",{parentName:"ul"},"Reference harus selalu valid.")),(0,r.kt)("p",null,"Mengenai penjelasan tentang dua aturan penting di atas akan bahas pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),"."),(0,r.kt)("h2",{id:"a336-karakteristik-pointer--reference"},"A.33.6. Karakteristik pointer & reference"),(0,r.kt)("p",null,"Kita sudah belajar tentang definisi beserta cara penerapan pointer, reference, dereference, dan mutable reference. Sekarang lanjut ke pembahasan tentang karakteristik dari pointer & reference."),(0,r.kt)("p",null,"Pointer merupakan variabel yang isinya adalah alamat memory (bukan nilai sebenarnya). Dan reference adalah alamat memory suatu data/variabel. Dengan menerapkan keduanya kita bisa menerapkan efisiensi penggunaan memory yang dampaknya sangat positif terhadap performa program."),(0,r.kt)("p",null,"Sekarang perhatikan statement berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"let number_one = 24;\nlet number_two = number_one;\n")),(0,r.kt)("p",null,"Variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one")," adalah data numerik bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"i32"),", eksekusi statement tersebut menghasilkan alokasi memori selebar 32-bit."),(0,r.kt)("p",null,"Kemudian ada lagi statement ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two"),". Meskipun nilainya didapat dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one"),", yang terjadi di balik layar adalah Rust akan mengalokasikan lagi alamat memory selebar 32-bit untuk menampung data ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two")," yang didapat dari hasil operasi ",(0,r.kt)("strong",{parentName:"p"},"copy")," dari variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Semua variabel primitif di Rust mengadopsi ",(0,r.kt)("em",{parentName:"p"},"copy semantics"),", yang artinya jika variabel tersebut digunakan dalam statement assignment, maka nilai akan di-duplikasi untuk kemudian ditampung pada variabel baru."),(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai ",(0,r.kt)("em",{parentName:"p"},"copy semantics")," dibahas pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/ownership#a334-copy-semantics-vs-move-semantics"},"Ownership"),".")),(0,r.kt)("p",null,"Selanjutnya, bandingkan dengan statement berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"let number_one = 24;\nlet number_two = &number_one;\n")),(0,r.kt)("p",null,"Pada contoh di atas, variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one")," datanya disimpan di memory dengan lebar 32-bit (masih sama seperti contoh sebelumnya). Namun ada yang berbeda dengan dengan variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two"),", variabel ini adalah variabel pointer yang nilainya reference ke alamat memory data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one"),". Yang terjadi di belakang layar, Rust tidak mengalokasikan lagi memory selebar 32-bit untuk menampung data ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two"),", melainkan menggunakan alamat memory data ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one")," sebagai reference, menjadikannya sebagai alamat tujuan variabel pointer ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two"),"."),(0,r.kt)("p",null,"Ok, lanjut. Per sekarang, reference variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number_one")," dan ",(0,r.kt)("inlineCode",{parentName:"p"},"number_two")," adalah sama. Dengan ini, jika dicontohkan isi data satu variabel diubah, maka variabel lainnya juga akan berubah, hal karena reference-nya adalah sama."),(0,r.kt)("h3",{id:"-contoh-ke-1"},"\u25c9 Contoh ke-1"),(0,r.kt)("p",null,"Contoh penerapannya bisa kita lihat pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/pointer-references#a315-karakteristik-pointer--reference"},"Pointer & References")," chapter ini, disitu bisa dilihat ada variabel mutable ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," dan ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," yang reference-nya adalah sama dengan variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Ketika underlying value ",(0,r.kt)("inlineCode",{parentName:"p"},"pointer_number")," diubah (dari ",(0,r.kt)("inlineCode",{parentName:"p"},"24")," ke ",(0,r.kt)("inlineCode",{parentName:"p"},"12"),"), isi data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," juga berubah."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Pointer & reference",src:n(5127).Z,width:"488",height:"417"})),(0,r.kt)("h3",{id:"-contoh-ke-2"},"\u25c9 Contoh ke-2"),(0,r.kt)("p",null,"Agar makin familiar dengan cara penerapan pointer & reference, silakan lanjut dengan praktik program berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="Cargo.toml"',title:'"Cargo.toml"'},'# ...\n\n[dependencies]\nrand = "0.8.5"\n')),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="main.rs"',title:'"main.rs"'},'use rand::Rng;\n\nfn main() {\n let mut number = 24;\n println!("number: {}", number);\n\n for _ in 0..=5 {\n change_value(&mut number);\n println!("number: {}", number);\n }\n}\n\nfn change_value(n: &mut i32) {\n *n = generate_random_number()\n}\n\nfn generate_random_number() -> i32 {\n rand::thread_rng().gen_range(0..100)\n}\n')),(0,r.kt)("p",null,"Pada sederhana di atas, ada sebuah variable mutable bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," yang dideklarasikan. Variabel tersebut kemudian diakses ",(0,r.kt)("em",{parentName:"p"},"mutable reference"),"-nya untuk dijadikan argumen statement pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"change_value"),"."),(0,r.kt)("p",null,"Di dalam fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"change_value"),", disiapkan 1 buah angka random hasil pemanggilan fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"generate_random_number"),", yang angka tersebut kemudian dijadikan sebagai nilai baru variable mutable reference yang mengarah ke ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),". Perubahan dalam fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"change_value")," akan mempengaruhi data variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),", karena reference-nya adalah sama."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Pointer & reference",src:n(3319).Z,width:"457",height:"204"})),(0,r.kt)("h2",{id:"a337-reference--borrowing"},"A.33.7. Reference & borrowing"),(0,r.kt)("p",null,"Di Rust, reference (atau pengaksesan alamat memory suatu data) memiliki hubungan yang sangat erat dengan konsep ",(0,r.kt)("strong",{parentName:"p"},"borrowing"),"."),(0,r.kt)("p",null,"Ketika kita mengambil reference suatu data, yang terjadi sebenarnya adalah kita meminjam data tersebut dari owner/pemilik sebenarnya."),(0,r.kt)("p",null,"Pada chapter ini kita tidak akan membahasnya lebih jauh lagi, karena akan ada chapter lain yang fokus mengulas topik tersebut secara detail, yaitu chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),"."),(0,r.kt)("p",null,"Untuk sekarang penulis anjurkan untuk lanjut ke chapter berikutnya secara urut."),(0,r.kt)("hr",null),(0,r.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,r.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,r.kt)("pre",null,(0,r.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/pointer_references"},"github.com/novalagung/dasarpemrogramanrust-example/../pointer_references")),(0,r.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-02-references-and-borrowing.html"},"https://doc.rust-lang.org/book/ch04-02-references-and-borrowing.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/marker/trait.Copy.html"},"https://doc.rust-lang.org/std/marker/trait.Copy.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Reference_(computer_science)"},"https://en.wikipedia.org/wiki/Reference_(computer_science)")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://progressivecoder.com/understanding-rust-ownership-and-borrowing-with-examples/"},"https://progressivecoder.com/understanding-rust-ownership-and-borrowing-with-examples/")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://os.phil-opp.com/heap-allocation/"},"https://os.phil-opp.com/heap-allocation/"))))}d.isMDXComponent=!0},1007:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/pointer-references-1-f103aebf5517db2d9de53c1fe39be319.png"},6356:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/pointer-references-2-8fdb7f20f15717b7d97311d81ffe005d.png"},8992:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/pointer-references-3-7b60c4bdc0392eb962187ee76ac2ab54.png"},5127:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/pointer-references-4-ba178a14defb70eb04b45ce050af0e6e.png"},3319:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/3aa512bf.35e7bf91.js b/assets/js/3aa512bf.35e7bf91.js new file mode 100644 index 00000000..14d7ec3c --- /dev/null +++ b/assets/js/3aa512bf.35e7bf91.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7386],{3905:(a,e,n)=>{n.d(e,{Zo:()=>m,kt:()=>o});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function r(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var d=i.createContext({}),u=function(a){var e=i.useContext(d),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},m=function(a){var e=u(a.components);return i.createElement(d.Provider,{value:e},a.children)},p={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},s=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,l=a.originalType,d=a.parentName,m=A(a,["components","mdxType","originalType","parentName"]),s=u(n),o=t,k=s["".concat(d,".").concat(o)]||s[o]||p[o]||l;return n?i.createElement(k,r(r({ref:e},m),{},{components:n})):i.createElement(k,r({ref:e},m))}));function o(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var l=n.length,r=new Array(l);r[0]=s;var A={};for(var d in e)hasOwnProperty.call(e,d)&&(A[d]=e[d]);A.originalType=a,A.mdxType="string"==typeof a?a:t,r[1]=A;for(var u=2;u{n.r(e),n.d(e,{assets:()=>d,contentTitle:()=>r,default:()=>p,frontMatter:()=>l,metadata:()=>A,toc:()=>u});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:21,title:"A.21. Module System \u279c Inline Module",sidebar_label:"A.21. Module System \u279c Inline Module"},r=void 0,A={unversionedId:"basic/module-inline",id:"basic/module-inline",title:"A.21. Module System \u279c Inline Module",description:"Pada chapter sebelumnya kita telah belajar cara manajemen module. Pembahasan tersebut dilanjut pada chapter ini, yaitu tentang inline module.",source:"@site/docs/basic/module-inline.md",sourceDirName:"basic",slug:"/basic/module-inline",permalink:"/basic/module-inline",draft:!1,tags:[],version:"current",sidebarPosition:21,frontMatter:{sidebar_position:21,title:"A.21. Module System \u279c Inline Module",sidebar_label:"A.21. Module System \u279c Inline Module"},sidebar:"tutorialSidebar",previous:{title:"A.20. Module System \u279c Module",permalink:"/basic/module-basic"},next:{title:"A.22. Module System \u279c Scope & Akses Item",permalink:"/basic/module-scope-item-access"}},d={},u=[{value:"A.21.1. Inline Module",id:"a211-inline-module",level:2},{value:"A.21.2. Praktik inline module",id:"a212-praktik-inline-module",level:2},{value:"A.21.3. Inline module item",id:"a213-inline-module-item",level:2},{value:"A.21.4. Inline module file nama_module.rs atau nama_module/mod.rs",id:"a214-inline-module-file-nama_modulers-atau-nama_modulemodrs",level:2},{value:"A.21.5. Inline module path attribute",id:"a215-inline-module-path-attribute",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:u};function p(a){let{components:e,...l}=a;return(0,t.kt)("wrapper",(0,i.Z)({},m,l,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Pada chapter sebelumnya kita telah belajar cara manajemen module. Pembahasan tersebut dilanjut pada chapter ini, yaitu tentang inline module."),(0,t.kt)("h2",{id:"a211-inline-module"},"A.21.1. Inline Module"),(0,t.kt)("p",null,"Inline module adalah cara lain dalam pendefinisian module beserta item-nya, caranya dengan tetap menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mod")," hanya saja isi/item ditulis di dalam blok kode keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mod")," tersebut. Agar lebih jelas silakan lihat penerapan inline module berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"mod module_a {\n\n pub fn fungsi_satu() {\n // ...\n }\n\n pub mod submodule_b {\n\n pub const PI: u32 = 3.14;\n\n pub fn fungsi_dua() {\n // ...\n }\n }\n}\n")),(0,t.kt)("p",null,"Inline module cukup mudah diterapkan, yang sebelumnya module dan submodule di-manage dengan mengacu ke penamaan file, dalam inline module cukup tulis saja sebagai blok kode definisi module."),(0,t.kt)("p",null,"Pada contoh di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"module_a")," memiliki beberapa item:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Item berupa fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"module_a::fungsi_satu")),(0,t.kt)("li",{parentName:"ul"},"Item berupa submodule ",(0,t.kt)("inlineCode",{parentName:"li"},"module_a::submodule_b")),(0,t.kt)("li",{parentName:"ul"},"Item berupa konstanta ",(0,t.kt)("inlineCode",{parentName:"li"},"module_a::submodule_b::PI")),(0,t.kt)("li",{parentName:"ul"},"Item berupa fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"module_a::submodule_b::fungsi_dua"))),(0,t.kt)("p",null,"Aturan definisi submodule dan item pada inline module masih sama seperti pada ",(0,t.kt)("em",{parentName:"p"},"normal module"),", salah satunya adalah agar item bisa diakses dari luar module maka perlu menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),"."),(0,t.kt)("h2",{id:"a212-praktik-inline-module"},"A.21.2. Praktik inline module"),(0,t.kt)("p",null,"Mari lanjut proses pembalajaran dengan praktik. Kita akan buat program sederhana yang di dalamnya ada proses generate random string, yang kemudian di-hash."),(0,t.kt)("p",null,"Silakan buat package baru menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"cargo new"),". Penulis di sini memilih nama ",(0,t.kt)("inlineCode",{parentName:"p"},"inline_module_1")," sebagai nama package. Setelah itu, siapkan fungsi main dan juga module bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," dengan penulisan kode menerapkan inline module. O iya, tulis keduanya (",(0,t.kt)("inlineCode",{parentName:"p"},"module utilities")," dan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),") dalam satu file yang sama yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod utilities {\n\n pub mod random {\n\n pub fn string(length: u32) -> String {\n // ... isi fungsi string milik module utilities::random\n }\n }\n\n pub mod password {\n\n pub fn hash(text: &str) -> String {\n // ... isi fungsi hash milik module utilities::password\n }\n\n pub fn is_valid(plain: &str, hashed: &str) -> bool {\n // ... isi fungsi is_valid milik module utilities::password\n }\n }\n}\n\nfn main() {\n // ... isi fungsi main\n}\n")),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::random::string")," adalah fungsi nantinya digunakan untuk generate data random string"),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password::hash")," adalah fungsi untuk melakukan proses hashing password menggunakan ",(0,t.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Bcrypt"},"bcrypt")),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password::is_valid")," adalah fungsi untuk pengecekan apakah password sama dengan data setelah di-hash")),(0,t.kt)("p",null,"Ketiga fungsi tersebut kemudian diterapkan pada program kecil yang logic-nya ditulis dalam fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Pada contoh di atas, semua item dan sub-item milik ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," didefinisikan public menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," agar bisa diakses dari module."),(0,t.kt)("p",{parentName:"blockquote"},"Sedangkan module ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," sendiri tidak wajib di-expose menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),", karena definisi module-nya berada dalam satu file yang sama dengan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", yaitu file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,t.kt)("p",null,"Sebelum lanjut proses koding, silakan tambahkan dulu beberapa dependencies berikut pada ",(0,t.kt)("inlineCode",{parentName:"p"},"Cargo.toml"),", kemudian jalankan ",(0,t.kt)("inlineCode",{parentName:"p"},"cargo build")," di-download."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="Cargo.toml"',title:'"Cargo.toml"'},'[package]\nname = "module_inline_1"\nversion = "0.1.0"\nedition = "2021"\n\n[dependencies]\nbcrypt = "0.13"\nrand = "0.8.5"\n')),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Package ",(0,t.kt)("a",{parentName:"li",href:"https://crates.io/keywords/bcrypt"},(0,t.kt)("inlineCode",{parentName:"a"},"bcrypt"))," adalah crate yang isinya banyak fungsi untuk kebutuhan enkripsi bcrypt"),(0,t.kt)("li",{parentName:"ul"},"Package ",(0,t.kt)("a",{parentName:"li",href:"https://crates.io/keywords/rand"},(0,t.kt)("inlineCode",{parentName:"a"},"rand"))," berisi item untuk keperluan generate data random. Kita sudah beberapa kali menggunakan crate ini.")),(0,t.kt)("p",null,"Ok, sekarang kembali ke source code. Tulis isi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities::random::string")," berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod utilities {\n\n pub mod random {\n\n pub fn string(length: u32) -> String {\n use rand::Rng;\n \n const CHARSET: &[u8] = "abcdefghijklmnopqrstuvwxyz".as_bytes();\n let mut arr = Vec::new();\n for _ in 0..=length {\n let n = rand::thread_rng().gen_range(0..(CHARSET.len()));\n let char = CHARSET[n];\n arr.push(char);\n }\n\n std::str::from_utf8(&arr[..]).unwrap().to_string()\n }\n }\n\n // ...\n}\n')),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Di atas adalah salah satu contoh penerapan generate random string dengan lebar ",(0,t.kt)("em",{parentName:"p"},"parameterized"),"."),(0,t.kt)("p",{parentName:"blockquote"},"Cara tersebut sebenarnya bukan yang paling efisien, namun karena jumlah topik yang kita pelajari masih belum terlalu banyak, penulis menghindari penerapan beberapa hal yang sifatnya baru dan bikin tambah bingung.")),(0,t.kt)("p",null,"Lanjut, silakan tulis kode untuk hashing password dan juga untuk validasi password berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod utilities {\n\n // ...\n\n pub mod password {\n\n pub fn hash(text: &str) -> String {\n let result = bcrypt::hash(text, bcrypt::DEFAULT_COST).unwrap();\n result\n }\n\n pub fn is_valid(plain: &str, hashed: &str) -> bool {\n let valid = bcrypt::verify(plain, hashed).unwrap();\n valid\n }\n }\n}\n")),(0,t.kt)("p",null,"Dalam enkripsi menggunakan bcrypt ada dua hal yang penting diketahui, yang pertama adalah data yang akan di-hash (pada contoh di atas adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"text"),"), dan ",(0,t.kt)("em",{parentName:"p"},"cost")," atau biaya enkripsi dalam bentuk numerik. Pada praktik ini, ",(0,t.kt)("inlineCode",{parentName:"p"},"bcrypt::DEFAULT_COST")," digunakan sebagai cost enkripsi."),(0,t.kt)("p",null,"Pengecekan apakah data terenkripsi adalah sama dengan data asli dilakukan menggunakan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"bcrypt::verify"),"."),(0,t.kt)("p",null,"Kedua fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"bcrypt::hash")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"bcrypt::verify")," menghasilkan data bertipe ",(0,t.kt)("em",{parentName:"p"},"generic result type")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"Result"),". Tipe ini memiliki method bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"unwrap")," yang berguna untuk pengambilan result atau nilai."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang result type dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,t.kt)("p",null,"Oke, sekarang fungsi utility sudah siap, mari implementasikan semua fungsi tersebut pada ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,t.kt)("p",null,"Silakan tulis kode berikut. Sebuah variabel bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"password")," dibuat, diisi dengan kombinasi dari string dan juga random string. Lalu data password tersebut di-hash kemudian dicek nilainya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'fn main() {\n let password = format!("zereth mortis {}", utilities::random::string(10));\n println!("raw password: {}", password);\n\n let hashed = utilities::password::hash(&password);\n println!("hashed password: {}", hashed);\n\n let is_valid = utilities::password::is_valid(&password, &hashed);\n println!("is password matched? {}", is_valid);\n}\n')),(0,t.kt)("p",null,"Jalankan program untuk melihat hasilnya."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust Inline Module",src:n(9632).Z,width:"605",height:"125"})),(0,t.kt)("p",null,"Statement ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities::random::string(10)")," menghasilkan random string dengan lebar ",(0,t.kt)("inlineCode",{parentName:"p"},"10"),", sesuai kebutuhan. Hasil dari pemanggilan fungsi tersebut di-concat dengan text kemudian ditampung variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"password"),"."),(0,t.kt)("p",null,"Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"password")," kemudian di-hash menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities::password::hash"),", kemudian dibandingkan hasil hash-nya apakah cocok dengan text aslinya menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities::password::is_valid"),"."),(0,t.kt)("h2",{id:"a213-inline-module-item"},"A.21.3. Inline module item"),(0,t.kt)("p",null,"Module item dalam inline module masih sama seperti pada module normal, yaitu adalah apapun yang didefinisikan di dalam sebuah module. Bisa berupa fungsi, konstanta, submodule, dan lainnya. Pada contoh di atas ada beberapa item didefinisikan:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::random")," yang merupakan sebuah submodule (inline module)."),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::random::string")," yang merupakan sebuah fungsi."),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password")," yang merupakan sebuah submodule (inline module)."),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password::hash")," yang merupakan sebuah fungsi."),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password::is_valid")," yang merupakan sebuah fungsi.")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang macam-macam item dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/path-item"},"Module System \u279c Path & Item"))),(0,t.kt)("h2",{id:"a214-inline-module-file-nama_modulers-atau-nama_modulemodrs"},"A.21.4. Inline module file ",(0,t.kt)("inlineCode",{parentName:"h2"},"nama_module.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"h2"},"nama_module/mod.rs")),(0,t.kt)("p",null,"Dalam ",(0,t.kt)("em",{parentName:"p"},"real project")," hampir tidak mungkin semua kode ditulis di file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),". Pastinya kode akan di-split menjadi banyak module sesuai kebutuhan."),(0,t.kt)("p",null,"Kode sebelumya, akan kita refactor. Module ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," yang berada di ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," perlu dipindah ke file baru yang sesuai dengan aturan penulisan module, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs"),". Di sini penulis memilih ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs"),", jadi silakan buat file bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities.rs"),", lalu pindah isi itemnya kesana (statement definisi inline module ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," tidak perlu ikut dipindah, hanya isinya saja, seperti normalnya definisi module)."),(0,t.kt)("p",null,"O iya, jangan lupa untuk menambahkan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"mod utilities")," pada ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),", agar module terdaftar dan bisa digunakan."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust Inline Module",src:n(3009).Z,width:"775",height:"963"})),(0,t.kt)("p",null,"Jalankan program untuk melihat hasilnya."),(0,t.kt)("p",null,"Pada contoh ke-2 ini modul ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," didefinisikan menggunakan cara biasa, sedang isi dari module tersebut didefinisikan menggunakan teknik inline module."),(0,t.kt)("h2",{id:"a215-inline-module-path-attribute"},"A.21.5. Inline module ",(0,t.kt)("inlineCode",{parentName:"h2"},"path")," attribute"),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute bisa digunakan dalam manajemen inline module. Silakan utak-atik kode yang sudah dibuat dengan mengikuti step berikut:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},(0,t.kt)("p",{parentName:"li"},"Buat file baru bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/password.rs"),", kemudian pindah isi definisi module ",(0,t.kt)("inlineCode",{parentName:"p"},"password")," ke file tersebut.")),(0,t.kt)("li",{parentName:"ol"},(0,t.kt)("p",{parentName:"li"},"Buat file baru bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/random.rs"),", kemudian pindah isi definisi module ",(0,t.kt)("inlineCode",{parentName:"p"},"random")," ke file tersebut.")),(0,t.kt)("li",{parentName:"ol"},(0,t.kt)("p",{parentName:"li"},"Hapus file ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities.rs"),", pindah definisi module ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," ke file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),", hasilnya kurang lebih seperti ini:"),(0,t.kt)("pre",{parentName:"li"},(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod utilities {\n #[path = "random.rs"]\n pub mod random;\n\n #[path = "password.rs"]\n pub mod password;\n}\n\nfn main() {\n // ...\n}\n')))),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust Inline Module",src:n(6633).Z,width:"920",height:"954"})),(0,t.kt)("p",null,"Jalankan program untuk melihat hasilnya."),(0,t.kt)("p",null,"Pada kode di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," yang digunakan bukan ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/random.rs")," melainkan ",(0,t.kt)("inlineCode",{parentName:"p"},"random.rs"),", hail ini dikarenakan ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute dipanggil ",(0,t.kt)("strong",{parentName:"p"},"di dalam module ",(0,t.kt)("inlineCode",{parentName:"strong"},"utilities")),", menjadikan current path pada blok kode tersebut menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/"),"."),(0,t.kt)("p",null,"Silakan coba ubah isi ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/random.rs"),", hasilnya adalah error. Rust akan menggunakan gabungan dari current path (",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/"),") dan path pada ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute (",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/random.rs"),") dalam lookup, jadinya yang di-lookup adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/utilities/random.rs"),", dan hasilnya error karena tidak ada file di sana."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust Inline Module",src:n(1081).Z,width:"772",height:"188"})),(0,t.kt)("p",null,"Module system di Rust merupakan topik yang cukup membingungkan (menurut penulis) karena desain-nya yang cukup unik jika dibandingkan dengan bagaimana bahasa pemrograman lain mengelola module-nya. Penulis anjurkan untuk mencoba praktik membuat program lainnya yang menerapkan module system, agar cepat terbiasa."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/module_inline"},"github.com/novalagung/dasarpemrogramanrust-example/../module_inline")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html"},"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.mod.html"},"https://doc.rust-lang.org/std/keyword.mod.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/mod.html"},"https://doc.rust-lang.org/rust-by-example/mod.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/69275034/what-is-the-difference-between-use-and-pub-use"},"https://stackoverflow.com/questions/69275034/what-is-the-difference-between-use-and-pub-use"))))}p.isMDXComponent=!0},9632:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},3009:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/module-inline-2-c6d08bd6d0d03ddba99457fd3c0bdc5d.png"},6633:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/module-inline-3-9234cea75bd46ea1eea49d0320315320.png"},1081:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/3aa512bf.a1c7a02b.js b/assets/js/3aa512bf.a1c7a02b.js deleted file mode 100644 index f74c9470..00000000 --- a/assets/js/3aa512bf.a1c7a02b.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7386],{3905:(a,e,n)=>{n.d(e,{Zo:()=>m,kt:()=>o});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function r(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var d=i.createContext({}),u=function(a){var e=i.useContext(d),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},m=function(a){var e=u(a.components);return i.createElement(d.Provider,{value:e},a.children)},p={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},s=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,l=a.originalType,d=a.parentName,m=A(a,["components","mdxType","originalType","parentName"]),s=u(n),o=t,k=s["".concat(d,".").concat(o)]||s[o]||p[o]||l;return n?i.createElement(k,r(r({ref:e},m),{},{components:n})):i.createElement(k,r({ref:e},m))}));function o(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var l=n.length,r=new Array(l);r[0]=s;var A={};for(var d in e)hasOwnProperty.call(e,d)&&(A[d]=e[d]);A.originalType=a,A.mdxType="string"==typeof a?a:t,r[1]=A;for(var u=2;u{n.r(e),n.d(e,{assets:()=>d,contentTitle:()=>r,default:()=>p,frontMatter:()=>l,metadata:()=>A,toc:()=>u});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:21,title:"A.21. Module System \u279c Inline Module",sidebar_label:"A.21. Module System \u279c Inline Module"},r=void 0,A={unversionedId:"basic/module-inline",id:"basic/module-inline",title:"A.21. Module System \u279c Inline Module",description:"Pada chapter sebelumnya kita telah belajar cara manajemen module. Pembahasan tersebut dilanjut pada chapter ini, yaitu tentang inline module.",source:"@site/docs/basic/module-inline.md",sourceDirName:"basic",slug:"/basic/module-inline",permalink:"/basic/module-inline",draft:!1,tags:[],version:"current",sidebarPosition:21,frontMatter:{sidebar_position:21,title:"A.21. Module System \u279c Inline Module",sidebar_label:"A.21. Module System \u279c Inline Module"},sidebar:"tutorialSidebar",previous:{title:"A.20. Module System \u279c Module",permalink:"/basic/module-basic"},next:{title:"A.22. Module System \u279c Scope & Akses Item",permalink:"/basic/module-scope-item-access"}},d={},u=[{value:"A.21.1. Inline Module",id:"a211-inline-module",level:2},{value:"A.21.2. Praktik inline module",id:"a212-praktik-inline-module",level:2},{value:"A.21.3. Inline module item",id:"a213-inline-module-item",level:2},{value:"A.21.4. Inline module file nama_module.rs atau nama_module/mod.rs",id:"a214-inline-module-file-nama_modulers-atau-nama_modulemodrs",level:2},{value:"A.21.5. Inline module path attribute",id:"a215-inline-module-path-attribute",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:u};function p(a){let{components:e,...l}=a;return(0,t.kt)("wrapper",(0,i.Z)({},m,l,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Pada chapter sebelumnya kita telah belajar cara manajemen module. Pembahasan tersebut dilanjut pada chapter ini, yaitu tentang inline module."),(0,t.kt)("h2",{id:"a211-inline-module"},"A.21.1. Inline Module"),(0,t.kt)("p",null,"Inline module adalah cara lain dalam pendefinisian module beserta item-nya, caranya dengan tetap menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mod")," hanya saja isi/item ditulis di dalam blok kode keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mod")," tersebut. Agar lebih jelas silakan lihat penerapan inline module berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"mod module_a {\n\n pub fn fungsi_satu() {\n // ...\n }\n\n pub mod submodule_b {\n\n pub const PI: u32 = 3.14;\n\n pub fn fungsi_dua() {\n // ...\n }\n }\n}\n")),(0,t.kt)("p",null,"Inline module cukup mudah diterapkan, yang sebelumnya module dan submodule di-manage dengan mengacu ke penamaan file, dalam inline module cukup tulis saja sebagai blok kode definisi module."),(0,t.kt)("p",null,"Pada contoh di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"module_a")," memiliki beberapa item:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Item berupa fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"module_a::fungsi_satu")),(0,t.kt)("li",{parentName:"ul"},"Item berupa submodule ",(0,t.kt)("inlineCode",{parentName:"li"},"module_a::submodule_b")),(0,t.kt)("li",{parentName:"ul"},"Item berupa konstanta ",(0,t.kt)("inlineCode",{parentName:"li"},"module_a::submodule_b::PI")),(0,t.kt)("li",{parentName:"ul"},"Item berupa fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"module_a::submodule_b::fungsi_dua"))),(0,t.kt)("p",null,"Aturan definisi submodule dan item pada inline module masih sama seperti pada ",(0,t.kt)("em",{parentName:"p"},"normal module"),", salah satunya adalah agar item bisa diakses dari luar module maka perlu menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),"."),(0,t.kt)("h2",{id:"a212-praktik-inline-module"},"A.21.2. Praktik inline module"),(0,t.kt)("p",null,"Mari lanjut proses pembalajaran dengan praktik. Kita akan buat program sederhana yang di dalamnya ada proses generate random string, yang kemudian di-hash."),(0,t.kt)("p",null,"Silakan buat package baru menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"cargo new"),". Penulis di sini memilih nama ",(0,t.kt)("inlineCode",{parentName:"p"},"inline_module_1")," sebagai nama package. Setelah itu, siapkan fungsi main dan juga module bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," dengan penulisan kode menerapkan inline module. O iya, tulis keduanya (",(0,t.kt)("inlineCode",{parentName:"p"},"module utilities")," dan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),") dalam satu file yang sama yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod utilities {\n\n pub mod random {\n\n pub fn string(length: u32) -> String {\n // ... isi fungsi string milik module utilities::random\n }\n }\n\n pub mod password {\n\n pub fn hash(text: &str) -> String {\n // ... isi fungsi hash milik module utilities::password\n }\n\n pub fn is_valid(plain: &str, hashed: &str) -> bool {\n // ... isi fungsi is_valid milik module utilities::password\n }\n }\n}\n\nfn main() {\n // ... isi fungsi main\n}\n")),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::random::string")," adalah fungsi nantinya digunakan untuk generate data random string"),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password::hash")," adalah fungsi untuk melakukan proses hashing password menggunakan ",(0,t.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Bcrypt"},"bcrypt")),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password::is_valid")," adalah fungsi untuk pengecekan apakah password sama dengan data setelah di-hash")),(0,t.kt)("p",null,"Ketiga fungsi tersebut kemudian diterapkan pada program kecil yang logic-nya ditulis dalam fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Pada contoh di atas, semua item dan sub-item milik ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," didefinisikan public menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," agar bisa diakses dari module."),(0,t.kt)("p",{parentName:"blockquote"},"Sedangkan module ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," sendiri tidak wajib di-expose menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),", karena definisi module-nya berada dalam satu file yang sama dengan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", yaitu file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,t.kt)("p",null,"Sebelum lanjut proses koding, silakan tambahkan dulu beberapa dependencies berikut pada ",(0,t.kt)("inlineCode",{parentName:"p"},"Cargo.toml"),", kemudian jalankan ",(0,t.kt)("inlineCode",{parentName:"p"},"cargo build")," di-download."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="Cargo.toml"',title:'"Cargo.toml"'},'[package]\nname = "module_inline_1"\nversion = "0.1.0"\nedition = "2021"\n\n[dependencies]\nbcrypt = "0.13"\nrand = "0.8.5"\n')),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Package ",(0,t.kt)("a",{parentName:"li",href:"https://crates.io/keywords/bcrypt"},(0,t.kt)("inlineCode",{parentName:"a"},"bcrypt"))," adalah crate yang isinya banyak fungsi untuk kebutuhan enkripsi bcrypt"),(0,t.kt)("li",{parentName:"ul"},"Package ",(0,t.kt)("a",{parentName:"li",href:"https://crates.io/keywords/rand"},(0,t.kt)("inlineCode",{parentName:"a"},"rand"))," berisi item untuk keperluan generate data random. Kita sudah beberapa kali menggunakan crate ini.")),(0,t.kt)("p",null,"Ok, sekarang kembali ke source code. Tulis isi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities::random::string")," berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod utilities {\n\n pub mod random {\n\n pub fn string(length: u32) -> String {\n use rand::Rng;\n \n const CHARSET: &[u8] = "abcdefghijklmnopqrstuvwxyz".as_bytes();\n let mut arr = Vec::new();\n for _ in 0..=length {\n let n = rand::thread_rng().gen_range(0..(CHARSET.len()));\n let char = CHARSET[n];\n arr.push(char);\n }\n\n std::str::from_utf8(&arr[..]).unwrap().to_string()\n }\n }\n\n // ...\n}\n')),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Di atas adalah salah satu contoh penerapan generate random string dengan lebar ",(0,t.kt)("em",{parentName:"p"},"parameterized"),"."),(0,t.kt)("p",{parentName:"blockquote"},"Cara tersebut sebenarnya bukan yang paling efisien, namun karena jumlah topik yang kita pelajari masih belum terlalu banyak, penulis menghindari penerapan beberapa hal yang sifatnya baru dan bikin tambah bingung.")),(0,t.kt)("p",null,"Lanjut, silakan tulis kode untuk hashing password dan juga untuk validasi password berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod utilities {\n\n // ...\n\n pub mod password {\n\n pub fn hash(text: &str) -> String {\n let result = bcrypt::hash(text, bcrypt::DEFAULT_COST).unwrap();\n result\n }\n\n pub fn is_valid(plain: &str, hashed: &str) -> bool {\n let valid = bcrypt::verify(plain, hashed).unwrap();\n valid\n }\n }\n}\n")),(0,t.kt)("p",null,"Dalam enkripsi menggunakan bcrypt ada dua hal yang penting diketahui, yang pertama adalah data yang akan di-hash (pada contoh di atas adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"text"),"), dan ",(0,t.kt)("em",{parentName:"p"},"cost")," atau biaya enkripsi dalam bentuk numerik. Pada praktik ini, ",(0,t.kt)("inlineCode",{parentName:"p"},"bcrypt::DEFAULT_COST")," digunakan sebagai cost enkripsi."),(0,t.kt)("p",null,"Pengecekan apakah data terenkripsi adalah sama dengan data asli dilakukan menggunakan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"bcrypt::verify"),"."),(0,t.kt)("p",null,"Kedua fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"bcrypt::hash")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"bcrypt::verify")," menghasilkan data bertipe ",(0,t.kt)("em",{parentName:"p"},"generic result type")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"Result"),". Tipe ini memiliki method bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"unwrap")," yang berguna untuk pengambilan result atau nilai."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang result type dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,t.kt)("p",null,"Oke, sekarang fungsi utility sudah siap, mari implementasikan semua fungsi tersebut pada ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,t.kt)("p",null,"Silakan tulis kode berikut. Sebuah variabel bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"password")," dibuat, diisi dengan kombinasi dari string dan juga random string. Lalu data password tersebut di-hash kemudian dicek nilainya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'fn main() {\n let password = format!("zereth mortis {}", utilities::random::string(10));\n println!("raw password: {}", password);\n\n let hashed = utilities::password::hash(&password);\n println!("hashed password: {}", hashed);\n\n let is_valid = utilities::password::is_valid(&password, &hashed);\n println!("is password matched? {}", is_valid);\n}\n')),(0,t.kt)("p",null,"Jalankan program untuk melihat hasilnya."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust Inline Module",src:n(9632).Z,width:"605",height:"125"})),(0,t.kt)("p",null,"Statement ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities::random::string(10)")," menghasilkan random string dengan lebar ",(0,t.kt)("inlineCode",{parentName:"p"},"10"),", sesuai kebutuhan. Hasil dari pemanggilan fungsi tersebut di-concat dengan text kemudian ditampung variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"password"),"."),(0,t.kt)("p",null,"Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"password")," kemudian di-hash menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities::password::hash"),", kemudian dibandingkan hasil hash-nya apakah cocok dengan text aslinya menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities::password::is_valid"),"."),(0,t.kt)("h2",{id:"a213-inline-module-item"},"A.21.3. Inline module item"),(0,t.kt)("p",null,"Module item dalam inline module masih sama seperti pada module normal, yaitu adalah apapun yang didefinisikan di dalam sebuah module. Bisa berupa fungsi, konstanta, submodule, dan lainnya. Pada contoh di atas ada beberapa item didefinisikan:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::random")," yang merupakan sebuah submodule (inline module)."),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::random::string")," yang merupakan sebuah fungsi."),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password")," yang merupakan sebuah submodule (inline module)."),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password::hash")," yang merupakan sebuah fungsi."),(0,t.kt)("li",{parentName:"ul"},"Item ",(0,t.kt)("inlineCode",{parentName:"li"},"utilities::password::is_valid")," yang merupakan sebuah fungsi.")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang macam-macam item dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/path-item"},"Module System \u279c Path & Item"))),(0,t.kt)("h2",{id:"a214-inline-module-file-nama_modulers-atau-nama_modulemodrs"},"A.21.4. Inline module file ",(0,t.kt)("inlineCode",{parentName:"h2"},"nama_module.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"h2"},"nama_module/mod.rs")),(0,t.kt)("p",null,"Dalam ",(0,t.kt)("em",{parentName:"p"},"real project")," hampir tidak mungkin semua kode ditulis di file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),". Pastinya kode akan di-split menjadi banyak module sesuai kebutuhan."),(0,t.kt)("p",null,"Kode sebelumya, akan kita refactor. Module ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," yang berada di ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," perlu dipindah ke file baru yang sesuai dengan aturan penulisan module, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs"),". Di sini penulis memilih ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs"),", jadi silakan buat file bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities.rs"),", lalu pindah isi itemnya kesana (statement definisi inline module ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," tidak perlu ikut dipindah, hanya isinya saja, seperti normalnya definisi module)."),(0,t.kt)("p",null,"O iya, jangan lupa untuk menambahkan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"mod utilities")," pada ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),", agar module terdaftar dan bisa digunakan."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust Inline Module",src:n(3009).Z,width:"775",height:"963"})),(0,t.kt)("p",null,"Jalankan program untuk melihat hasilnya."),(0,t.kt)("p",null,"Pada contoh ke-2 ini modul ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," didefinisikan menggunakan cara biasa, sedang isi dari module tersebut didefinisikan menggunakan teknik inline module."),(0,t.kt)("h2",{id:"a215-inline-module-path-attribute"},"A.21.5. Inline module ",(0,t.kt)("inlineCode",{parentName:"h2"},"path")," attribute"),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute bisa digunakan dalam manajemen inline module. Silakan utak-atik kode yang sudah dibuat dengan mengikuti step berikut:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},(0,t.kt)("p",{parentName:"li"},"Buat file baru bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/password.rs"),", kemudian pindah isi definisi module ",(0,t.kt)("inlineCode",{parentName:"p"},"password")," ke file tersebut.")),(0,t.kt)("li",{parentName:"ol"},(0,t.kt)("p",{parentName:"li"},"Buat file baru bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/random.rs"),", kemudian pindah isi definisi module ",(0,t.kt)("inlineCode",{parentName:"p"},"random")," ke file tersebut.")),(0,t.kt)("li",{parentName:"ol"},(0,t.kt)("p",{parentName:"li"},"Hapus file ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities.rs"),", pindah definisi module ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities")," ke file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),", hasilnya kurang lebih seperti ini:"),(0,t.kt)("pre",{parentName:"li"},(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod utilities {\n #[path = "random.rs"]\n pub mod random;\n\n #[path = "password.rs"]\n pub mod password;\n}\n\nfn main() {\n // ...\n}\n')))),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust Inline Module",src:n(6633).Z,width:"920",height:"954"})),(0,t.kt)("p",null,"Jalankan program untuk melihat hasilnya."),(0,t.kt)("p",null,"Pada kode di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," yang digunakan bukan ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/random.rs")," melainkan ",(0,t.kt)("inlineCode",{parentName:"p"},"random.rs"),", hail ini dikarenakan ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute dipanggil ",(0,t.kt)("strong",{parentName:"p"},"di dalam module ",(0,t.kt)("inlineCode",{parentName:"strong"},"utilities")),", menjadikan current path pada blok kode tersebut menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/"),"."),(0,t.kt)("p",null,"Silakan coba ubah isi ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/random.rs"),", hasilnya adalah error. Rust akan menggunakan gabungan dari current path (",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/"),") dan path pada ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute (",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/random.rs"),") dalam lookup, jadinya yang di-lookup adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"utilities/utilities/random.rs"),", dan hasilnya error karena tidak ada file disana."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust Inline Module",src:n(1081).Z,width:"772",height:"188"})),(0,t.kt)("p",null,"Module system di Rust merupakan topik yang cukup membingungkan (menurut penulis) karena desain-nya yang cukup unik jika dibandingkan dengan bagaimana bahasa pemrograman lain mengelola module-nya. Penulis anjurkan untuk mencoba praktik membuat program lainnya yang menerapkan module system, agar cepat terbiasa."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/module_inline"},"github.com/novalagung/dasarpemrogramanrust-example/../module_inline")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html"},"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.mod.html"},"https://doc.rust-lang.org/std/keyword.mod.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/mod.html"},"https://doc.rust-lang.org/rust-by-example/mod.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/69275034/what-is-the-difference-between-use-and-pub-use"},"https://stackoverflow.com/questions/69275034/what-is-the-difference-between-use-and-pub-use"))))}p.isMDXComponent=!0},9632:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},3009:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/module-inline-2-c6d08bd6d0d03ddba99457fd3c0bdc5d.png"},6633:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/module-inline-3-9234cea75bd46ea1eea49d0320315320.png"},1081:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/3b1dfe68.84da4a4d.js b/assets/js/3b1dfe68.84da4a4d.js deleted file mode 100644 index ac2eff03..00000000 --- a/assets/js/3b1dfe68.84da4a4d.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6161],{3905:(e,a,n)=>{n.d(a,{Zo:()=>u,kt:()=>k});var i=n(7294);function t(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function l(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);a&&(i=i.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,i)}return n}function r(e){for(var a=1;a=0||(t[n]=e[n]);return t}(e,a);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(t[n]=e[n])}return t}var m=i.createContext({}),p=function(e){var a=i.useContext(m),n=a;return e&&(n="function"==typeof e?e(a):r(r({},a),e)),n},u=function(e){var a=p(e.components);return i.createElement(m.Provider,{value:a},e.children)},d={inlineCode:"code",wrapper:function(e){var a=e.children;return i.createElement(i.Fragment,{},a)}},o=i.forwardRef((function(e,a){var n=e.components,t=e.mdxType,l=e.originalType,m=e.parentName,u=s(e,["components","mdxType","originalType","parentName"]),o=p(n),k=t,g=o["".concat(m,".").concat(k)]||o[k]||d[k]||l;return n?i.createElement(g,r(r({ref:a},u),{},{components:n})):i.createElement(g,r({ref:a},u))}));function k(e,a){var n=arguments,t=a&&a.mdxType;if("string"==typeof e||t){var l=n.length,r=new Array(l);r[0]=o;var s={};for(var m in a)hasOwnProperty.call(a,m)&&(s[m]=a[m]);s.originalType=e,s.mdxType="string"==typeof e?e:t,r[1]=s;for(var p=2;p{n.r(a),n.d(a,{assets:()=>m,contentTitle:()=>r,default:()=>d,frontMatter:()=>l,metadata:()=>s,toc:()=>p});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:28,title:"A.28. Module System \u279c Visibility & Privacy",sidebar_label:"A.28. Module System \u279c Visibility & Privacy"},r=void 0,s={unversionedId:"basic/visibility-privacy",id:"basic/visibility-privacy",title:"A.28. Module System \u279c Visibility & Privacy",description:"Kita sebenarnya sudah mempelajari banyak hal yang berhubungan dengan visibility & privacy pada beberapa chapter sebelumnya. Jika pembaca mempelajari ebook ini secara urut, maka pastinya sudah familiar dengan keyword pub, self, crate, dan super.",source:"@site/docs/basic/visibility-privacy.md",sourceDirName:"basic",slug:"/basic/visibility-privacy",permalink:"/basic/visibility-privacy",draft:!1,tags:[],version:"current",sidebarPosition:28,frontMatter:{sidebar_position:28,title:"A.28. Module System \u279c Visibility & Privacy",sidebar_label:"A.28. Module System \u279c Visibility & Privacy"},sidebar:"tutorialSidebar",previous:{title:"A.27. Type Alias & Casting",permalink:"/basic/type-alias-casting"},next:{title:"A.29. Module System \u279c Use, Import, Re-export",permalink:"/basic/use"}},m={},p=[{value:"A.28.1. Pembahasan module system",id:"a281-pembahasan-module-system",level:2},{value:"A.28.2. Default visibility",id:"a282-default-visibility",level:2},{value:"A.28.3. Re-export item",id:"a283-re-export-item",level:2},{value:"A.28.4. Public visibility scope",id:"a284-public-visibility-scope",level:2},{value:"\u25c9 Keyword pub",id:"-keyword-pub",level:3},{value:"\u25c9 Keyword pub(in path)",id:"-keyword-pubin-path",level:3},{value:"\u25c9 Keyword pub(crate)",id:"-keyword-pubcrate",level:3},{value:"\u25c9 Keyword pub(super)",id:"-keyword-pubsuper",level:3},{value:"\u25c9 Keyword pub(self)",id:"-keyword-pubself",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:p};function d(e){let{components:a,...l}=e;return(0,t.kt)("wrapper",(0,i.Z)({},u,l,{components:a,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Kita sebenarnya sudah mempelajari banyak hal yang berhubungan dengan visibility & privacy pada beberapa chapter sebelumnya. Jika pembaca mempelajari ebook ini secara urut, maka pastinya sudah familiar dengan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"self"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"crate"),", dan ",(0,t.kt)("inlineCode",{parentName:"p"},"super"),"."),(0,t.kt)("p",null,"Chapter ini merupakan pembahasan tambahan untuk ke-4 keyword tersebut, dan fokusnya lebih ke visibility & privacy di Rust secara general."),(0,t.kt)("p",null,"O iya, perihal ",(0,t.kt)("em",{parentName:"p"},"visbility")," dan ",(0,t.kt)("em",{parentName:"p"},"privacy")," itu sendiri, kedua istilah tersebut di sini kita maknai sama, yang artinya kurang lebih adalah tentang manajemen akses item di Rust."),(0,t.kt)("h2",{id:"a281-pembahasan-module-system"},"A.28.1. Pembahasan module system"),(0,t.kt)("p",null,"Pastikan sudah mempelajari 5 buah chapter tentang module system yang sebelumnya sudah dibahas. Kesemua chapter tersebut sangat berhubungan dengan pembahasan chapter ini."),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/path-item"},"A.18. Module System \u279c Path & Item")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/package-crate"},"A.19. Module System \u279c Package & Crate")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/module-basic"},"A.20. Module System \u279c Module")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/module-inline"},"A.21. Module System \u279c Inline Module")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/module-scope-item-access"},"A.22. Module System \u279c Scope & Akses Item"))),(0,t.kt)("h2",{id:"a282-default-visibility"},"A.28.2. Default visibility"),(0,t.kt)("p",null,"Di Rust, ",(0,t.kt)("em",{parentName:"p"},"by default"),", hampir semua item adalah private. Apa efeknya ketika item adalah private atau publik? Silakan ingat 2 aturan penting berikut:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},"Jika suatu item adalah private, maka item tersebut hanya bisa diakses dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," dan dari ",(0,t.kt)("em",{parentName:"li"},"submodules")," milik ",(0,t.kt)("em",{parentName:"li"},"current module"),"."),(0,t.kt)("li",{parentName:"ol"},"Jika suatu item adalah publik, maka dia bisa diakses dari module lain di luar ",(0,t.kt)("em",{parentName:"li"},"current module scope"),", dengan catatan parent module scope item tersebut harus publik.")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Kita sepakati di sini, pada istilah ",(0,t.kt)("strong",{parentName:"p"},"current module")," kata ",(0,t.kt)("em",{parentName:"p"},"module")," disitu bisa saja tertuju untuk module atau juga submodule")),(0,t.kt)("p",null,"Dua point di atas sangat penting untuk dipahami, karena digunakan sebagai landasan pertimbangan dalam penyusunan hirarki module. Sebagai contoh, kita bisa membuat program yang hanya meng-expose API tertentu (yang memang diperlukan untuk diakses oleh publik), tanpa perlu ikut meng-expose detail implementasinya."),(0,t.kt)("p",null,"Ok, sekarang silakan perhatikan path sederhana di bawah ini. Diasumsikan ada sebuah fungsi yang path aksesnya adalah berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash"},"messaging::service_layer::some_black_magic\n")),(0,t.kt)("p",null,"Segmen pertama yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging")," pasti adalah publik, karena di-import ke ",(0,t.kt)("em",{parentName:"p"},"crate root"),". Lalu bagaimana dengan segmen ",(0,t.kt)("inlineCode",{parentName:"p"},"service_layer")," dan juga ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic"),"?"),(0,t.kt)("p",null,"Jika item ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic")," disitu adalah publik, maka idealnya pengaksesan menggunakan path tersebut memungkinkan. Tapi kembali ke point ke-2 aturan yang sudah dibahas di atas, yaitu meskipun ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic")," adalah publik, jika parent-nya (yang pada konteks ini adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"service_layer"),") adalah private, maka pengaksesan menggunakan path tersebut menghasilkan error."),(0,t.kt)("p",null,"Intinya, ",(0,t.kt)("strong",{parentName:"p"},"sebuah item bisa diakses jika item tersebut adalah publik, dan parent item tersebut juga publik. Sedangkan default visibility untuk hampir semua item adalah private.")),(0,t.kt)("p",null,"Ok, sekarang mari lanjut ke praktik menggunakan contoh dengan pembahasan yang lebih mendetail. Silakan perhatikan dan praktikkan kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 messaging.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'const SOME_MESSAGE: &str = "hello rust";\n\nmod service_layer {\n\n pub fn some_black_magic() {\n println!("{}", crate::messaging::SOME_MESSAGE)\n }\n}\n\npub fn say_hello() {\n service_layer::some_black_magic();\n}\n')),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::say_hello();\n}\n")),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Konstanta ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::SOME_MESSAGE")," adalah ",(0,t.kt)("strong",{parentName:"p"},"private"),". Penjelasan:"),(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Konstanta ini merupakan module item milik ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Konstanta ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),")."),(0,t.kt)("li",{parentName:"ul"},"Konstanta ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari submodule milik ",(0,t.kt)("em",{parentName:"li"},"current module"),", yaitu submodule dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),". Contohnya bisa dilihat pada fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer::some_black_magic")," yang disitu ada statement pemanggilan ",(0,t.kt)("inlineCode",{parentName:"li"},"SOME_MESSAGE"),"."),(0,t.kt)("li",{parentName:"ul"},"Konstanta ini ",(0,t.kt)("strong",{parentName:"li"},"tidak bisa diakses")," dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),")."))),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Submodule ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::service_layer")," adalah ",(0,t.kt)("strong",{parentName:"p"},"private"),". Penjelasan:"),(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Submodule ini merupakan module item milik ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Submodule ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"). Contohnya bisa dilihat pada fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::say_hello")," yang disitu ada statement pemanggilan ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer"),"."),(0,t.kt)("li",{parentName:"ul"},"Submodule ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari submodule milik ",(0,t.kt)("em",{parentName:"li"},"current module"),", yaitu submodule dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Submodule ini ",(0,t.kt)("strong",{parentName:"li"},"tidak bisa diakses")," dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),")."))),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::service_layer::some_black_magic")," adalah ",(0,t.kt)("strong",{parentName:"p"},"publik"),". Penjelasan:"),(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Fungsi ini merupakan module item milik ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),"."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),")."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari submodule milik ",(0,t.kt)("em",{parentName:"li"},"current module"),", yaitu submodule dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),". Contohnya bisa dilihat pada fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::say_hello")," yang disitu ada statement pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"some_black_magic"),"."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),")."),(0,t.kt)("li",{parentName:"ul"},"Namun meskipun demikian, bisa tidaknya fungsi ini diakses dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),") juga tergantung dengan visibility dari current module itu sendiri, yaitu ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),"."),(0,t.kt)("li",{parentName:"ul"},"Karena module ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer")," adalah private, meskipun fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"some_black_magic")," di dalamnya adalah publik, pengaksesan fungsi tersebut dari luar module scope ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer")," tidak dimungkinkan.",(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Pengaksesan ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer::some_black_magic")," dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::say_hello")," tidak error karena submodule ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer")," meskipun private, posisinya adalah masih dalam satu module scope yang sama dengan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"say_hello"),"."),(0,t.kt)("li",{parentName:"ul"},"Dimisalkan jika ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer::some_black_magic")," dipaksa diakses dari ",(0,t.kt)("inlineCode",{parentName:"li"},"main"),", maka muncul error karena ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer")," adalah private dan posisinya tidak berada dalam module scope yang sama dengan crate root (",(0,t.kt)("inlineCode",{parentName:"li"},"main"),")."))))),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::say_hello")," adalah ",(0,t.kt)("strong",{parentName:"p"},"public"),". Penjelasan:"),(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Fungsi ini merupakan module item milik ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),")."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari submodule milik ",(0,t.kt)("em",{parentName:"li"},"current module"),", yaitu submodule dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"). Contohnya bisa dilihat pada crate root fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"main"),", disitu ada pemanggilan statement ",(0,t.kt)("inlineCode",{parentName:"li"},"say_hello"),".")))),(0,t.kt)("h2",{id:"a283-re-export-item"},"A.28.3. Re-export item"),(0,t.kt)("p",null,"Pada contoh, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::say_hello")," didesain sebagai media untuk mengakses fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic"),". Di situasi ",(0,t.kt)("em",{parentName:"p"},"real world")," pastinya sangat jarang terjadi sebuah fungsi isinya hanya satu baris pemanggilan fungsi lainnya. Jika memang ada situasi seperti itu, (kontekstual) lebih baik hapus saja fungsi yang jadi media pemanggilan dan langsung saja panggil fungsi di dalamnya sesuai kebutuhan."),(0,t.kt)("p",null,"Pada praktik selanjutnya ini kita misalkan bahwa fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," isinya memang hanya 1 baris, dan yang paling penting adalah isi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic")," perlu untuk bisa diakses dari ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),". Untuk kasus seperti ini ada 3 alternatif solusi:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},"Tidak perlu mengubah apapun, gunakan saja kode yang sudah ditulis di atas. Kode tersebut sudah bisa mengakomodir pemanggilan ",(0,t.kt)("inlineCode",{parentName:"li"},"some_black_magic")," via ",(0,t.kt)("inlineCode",{parentName:"li"},"say_hello"),"."),(0,t.kt)("li",{parentName:"ol"},"Atau, hapus saja fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"say_hello"),", lalu ubah visibility module ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer")," menjadi publik, dengan demikian kita bisa mengakses ",(0,t.kt)("inlineCode",{parentName:"li"},"some_black_magic")," dari ",(0,t.kt)("inlineCode",{parentName:"li"},"main")," menggunakan path ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer::some_black_magic"),"."),(0,t.kt)("li",{parentName:"ol"},"Atau, gunakan teknik ",(0,t.kt)("strong",{parentName:"li"},"re-export item"),".")),(0,t.kt)("p",null,"Re-export item adalah sebuah cara untuk mem-",(0,t.kt)("em",{parentName:"p"},"bypass")," pengaksesan item yang secara hirarki memang tidak bisa diakses dari luar module (bisa jadi karena visibility item ataupun parent module nya adalah private). Dengan teknik ini, maka item pasti bisa diakses dari luar module."),(0,t.kt)("p",null,"Item yang di-re-export akan menjadi item milik ",(0,t.kt)("em",{parentName:"p"},"current module")," dimana statement re-export tersebut ditulis."),(0,t.kt)("p",null,"Cara re-export item adalah menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub use")," kemudian diikuti dengan path yang ingin di-export dan juga nama export item dengan notasi penulisan berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"pub use the_path as exported_name;\npub use self::service_layer::some_black_magic as say_hello;\n")),(0,t.kt)("p",null,"Contoh jika diterapkan pada program yang sebelumnya sudah ditulis:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'pub use self::service_layer::some_black_magic as say_hello;\n\nconst SOME_MESSAGE: &str = "hello rust";\n\nmod service_layer {\n\n pub fn some_black_magic() {\n println!("{}", crate::messaging::SOME_MESSAGE)\n }\n}\n')),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::say_hello();\n}\n")),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Visibility & Privacy",src:n(2201).Z,width:"466",height:"109"})),(0,t.kt)("p",null,"Bisa dilihat di contoh di atas, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," dihapus, kemudian item ",(0,t.kt)("inlineCode",{parentName:"p"},"service_layer::some_black_magic")," di-re-export dengan nama ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello"),". Dengannya kita bisa mengakses ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic")," dari luar module ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging")," menggunakan path ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::say_hello"),"."),(0,t.kt)("p",null,"Jika item ingin di-re-export tanpa perubahan nama item, bisa gunakan notasi berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"pub use the_path;\npub use self::service_layer::some_black_magic;\n")),(0,t.kt)("p",null,"Jika diterapkan pada program sebelumnya, kurang lebih seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'pub use self::service_layer::some_black_magic;\n\nconst SOME_MESSAGE: &str = "hello rust";\n\nmod service_layer {\n\n pub fn some_black_magic() {\n println!("{}", crate::messaging::SOME_MESSAGE)\n }\n}\n')),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::some_black_magic();\n}\n")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"use")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/use"},"Module System \u279c Use"))),(0,t.kt)("h2",{id:"a284-public-visibility-scope"},"A.28.4. Public visibility scope"),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," digunakan untuk mengubah visibility item menjadi publik. Keyword ini bisa dikombinasikan dengan salah satu dari keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"self"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"crate"),", dan ",(0,t.kt)("inlineCode",{parentName:"p"},"super"),"; denganya kita bisa menentukan visibility sebuah publik item dengan scope yang lebih spesifik."),(0,t.kt)("h3",{id:"-keyword-pub"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub")),(0,t.kt)("p",null,"Penulis rasa untuk penerapan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," ini sudah sangat jelas, kita beberapa kali mempraktikkannya."),(0,t.kt)("p",null,"Dengan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),", sebuah item visibility-nya menjadi publik."),(0,t.kt)("h3",{id:"-keyword-pubin-path"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub(in path)")),(0,t.kt)("p",null,"Keyword ini menjadikan visibility item hanya di dalam ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," yang ditulis di ",(0,t.kt)("inlineCode",{parentName:"p"},"pub(in path)"),", dengan ketentuan ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," tersebut merupakan parent dari module item dimana keyword digunakan."),(0,t.kt)("p",null,"Contohnya bisa dilihat pada kode berikut. Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," didefinisikan publik dengan scope path ditentukan secara eksplisit adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"crate::outer_mod"),". Dengan demikian fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," hanya bisa diakses dari dalam ",(0,t.kt)("inlineCode",{parentName:"p"},"outer_mod"),"."),(0,t.kt)("p",null,"Bisa dilihat di contoh, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," diakses dari ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),". Silakan coba saja paksa untuk mengaksesnya dari fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", hasilnya pasti error."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'pub mod outer_mod {\n\n pub mod inner_mod {\n\n // fungsi say_hello berikut hanya bisa diakses dari dalam `outer_mod`.\n // pengaksesannya dari luar `outer_mod` menghasilkan error.\n pub(in crate::outer_mod) fn say_hello() {\n println!("hello rust")\n }\n }\n\n pub fn do_something() {\n inner_mod::say_hello();\n }\n}\n\nfn main() {\n outer_mod::do_something();\n}\n')),(0,t.kt)("h3",{id:"-keyword-pubcrate"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub(crate)")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub(crate)")," digunakan untuk membuat visibility item menjadi publik dengan scope akses ",(0,t.kt)("em",{parentName:"p"},"current crate"),". Dengan ini item bisa diakses dari manapun asalakan masih dalam crate yang sama."),(0,t.kt)("p",null,"Contoh penerapannya bisa dilihat berikut ini. Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," visibility scope nya ditentukan adalah ",(0,t.kt)("em",{parentName:"p"},"current crate"),". Fungsi tersebut bisa diakses dari ",(0,t.kt)("inlineCode",{parentName:"p"},"outer_mod_one::do_something"),", dari ",(0,t.kt)("inlineCode",{parentName:"p"},"outer_mod_two::do_something"),", dan juga dari fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'pub mod outer_mod_one {\n\n pub mod inner_mod {\n\n // fungsi ini visibility scope-nya di level crate\n pub(crate) fn say_hello() {\n println!("hello rust")\n }\n }\n\n pub fn do_something() {\n inner_mod::say_hello();\n }\n}\n\npub mod outer_mod_two {\n\n pub fn do_something() {\n crate::outer_mod_one::inner_mod::say_hello();\n }\n}\n\nfn main() {\n outer_mod_one::inner_mod::say_hello();\n outer_mod_one::do_something();\n outer_mod_two::do_something();\n}\n')),(0,t.kt)("h3",{id:"-keyword-pubsuper"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub(super)")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub(super)")," digunakan untuk membuat visibility item menjadi publik dengan scope akses ",(0,t.kt)("em",{parentName:"p"},"parent module"),"."),(0,t.kt)("p",null,"Pada contoh berikut, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," visibility scope nya ditentukan adalah ",(0,t.kt)("em",{parentName:"p"},"parent module"),", artinya fungsi tersebut hanya bisa diakses dari dalam ",(0,t.kt)("em",{parentName:"p"},"parent module")," (yang pada konteks ini adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"outer_mod"),")."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'pub mod outer_mod {\n\n pub mod inner_mod {\n\n // fungsi ini visibility scope-nya di parent module scope,\n // yaitu `outer_mod`\n pub(super) fn say_hello() {\n println!("hello rust")\n }\n }\n\n pub fn do_something() {\n inner_mod::say_hello();\n }\n}\n\nfn main() {\n outer_mod::do_something();\n}\n')),(0,t.kt)("h3",{id:"-keyword-pubself"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub(self)")),(0,t.kt)("p",null,"Keyword ini digunakan untuk membuat visibility item menjadi publik dengan scope akses hanya pada current module. Contohnya bisa dilihat pada kode program berikut."),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," visibility scope-nya adalah ",(0,t.kt)("em",{parentName:"p"},"current module"),". Fungsi tersebut hanya bisa diakses dari tempat yang merupakan module item dari ",(0,t.kt)("em",{parentName:"p"},"current module")," yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"inner_mod"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'pub mod outer_mod {\n\n pub mod inner_mod {\n\n // fungsi ini visibility scope-nya di current module scope,\n // yaitu `inner_mod`\n pub(self) fn say_hello() {\n println!("hello rust")\n }\n\n pub fn do_something() {\n say_hello();\n }\n }\n}\n\nfn main() {\n outer_mod::inner_mod::do_something();\n}\n')),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/visibility_privacy"},"github.com/novalagung/dasarpemrogramanrust-example/../visibility_privacy")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/visibility-and-privacy.html"},"https://doc.rust-lang.org/reference/visibility-and-privacy.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html"))))}d.isMDXComponent=!0},2201:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/3b1dfe68.a3e9cea9.js b/assets/js/3b1dfe68.a3e9cea9.js new file mode 100644 index 00000000..36b7f22d --- /dev/null +++ b/assets/js/3b1dfe68.a3e9cea9.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6161],{3905:(e,a,n)=>{n.d(a,{Zo:()=>u,kt:()=>k});var i=n(7294);function t(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function l(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);a&&(i=i.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,i)}return n}function r(e){for(var a=1;a=0||(t[n]=e[n]);return t}(e,a);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(t[n]=e[n])}return t}var m=i.createContext({}),p=function(e){var a=i.useContext(m),n=a;return e&&(n="function"==typeof e?e(a):r(r({},a),e)),n},u=function(e){var a=p(e.components);return i.createElement(m.Provider,{value:a},e.children)},d={inlineCode:"code",wrapper:function(e){var a=e.children;return i.createElement(i.Fragment,{},a)}},o=i.forwardRef((function(e,a){var n=e.components,t=e.mdxType,l=e.originalType,m=e.parentName,u=s(e,["components","mdxType","originalType","parentName"]),o=p(n),k=t,g=o["".concat(m,".").concat(k)]||o[k]||d[k]||l;return n?i.createElement(g,r(r({ref:a},u),{},{components:n})):i.createElement(g,r({ref:a},u))}));function k(e,a){var n=arguments,t=a&&a.mdxType;if("string"==typeof e||t){var l=n.length,r=new Array(l);r[0]=o;var s={};for(var m in a)hasOwnProperty.call(a,m)&&(s[m]=a[m]);s.originalType=e,s.mdxType="string"==typeof e?e:t,r[1]=s;for(var p=2;p{n.r(a),n.d(a,{assets:()=>m,contentTitle:()=>r,default:()=>d,frontMatter:()=>l,metadata:()=>s,toc:()=>p});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:28,title:"A.28. Module System \u279c Visibility & Privacy",sidebar_label:"A.28. Module System \u279c Visibility & Privacy"},r=void 0,s={unversionedId:"basic/visibility-privacy",id:"basic/visibility-privacy",title:"A.28. Module System \u279c Visibility & Privacy",description:"Kita sebenarnya sudah mempelajari banyak hal yang berhubungan dengan visibility & privacy pada beberapa chapter sebelumnya. Jika pembaca mempelajari ebook ini secara urut, maka pastinya sudah familiar dengan keyword pub, self, crate, dan super.",source:"@site/docs/basic/visibility-privacy.md",sourceDirName:"basic",slug:"/basic/visibility-privacy",permalink:"/basic/visibility-privacy",draft:!1,tags:[],version:"current",sidebarPosition:28,frontMatter:{sidebar_position:28,title:"A.28. Module System \u279c Visibility & Privacy",sidebar_label:"A.28. Module System \u279c Visibility & Privacy"},sidebar:"tutorialSidebar",previous:{title:"A.27. Type Alias & Casting",permalink:"/basic/type-alias-casting"},next:{title:"A.29. Module System \u279c Use, Import, Re-export",permalink:"/basic/use"}},m={},p=[{value:"A.28.1. Pembahasan module system",id:"a281-pembahasan-module-system",level:2},{value:"A.28.2. Default visibility",id:"a282-default-visibility",level:2},{value:"A.28.3. Re-export item",id:"a283-re-export-item",level:2},{value:"A.28.4. Public visibility scope",id:"a284-public-visibility-scope",level:2},{value:"\u25c9 Keyword pub",id:"-keyword-pub",level:3},{value:"\u25c9 Keyword pub(in path)",id:"-keyword-pubin-path",level:3},{value:"\u25c9 Keyword pub(crate)",id:"-keyword-pubcrate",level:3},{value:"\u25c9 Keyword pub(super)",id:"-keyword-pubsuper",level:3},{value:"\u25c9 Keyword pub(self)",id:"-keyword-pubself",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:p};function d(e){let{components:a,...l}=e;return(0,t.kt)("wrapper",(0,i.Z)({},u,l,{components:a,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Kita sebenarnya sudah mempelajari banyak hal yang berhubungan dengan visibility & privacy pada beberapa chapter sebelumnya. Jika pembaca mempelajari ebook ini secara urut, maka pastinya sudah familiar dengan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"self"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"crate"),", dan ",(0,t.kt)("inlineCode",{parentName:"p"},"super"),"."),(0,t.kt)("p",null,"Chapter ini merupakan pembahasan tambahan untuk ke-4 keyword tersebut, dan fokusnya lebih ke visibility & privacy di Rust secara general."),(0,t.kt)("p",null,"O iya, perihal ",(0,t.kt)("em",{parentName:"p"},"visbility")," dan ",(0,t.kt)("em",{parentName:"p"},"privacy")," itu sendiri, kedua istilah tersebut di sini kita maknai sama, yang artinya kurang lebih adalah tentang manajemen akses item di Rust."),(0,t.kt)("h2",{id:"a281-pembahasan-module-system"},"A.28.1. Pembahasan module system"),(0,t.kt)("p",null,"Pastikan sudah mempelajari 5 buah chapter tentang module system yang sebelumnya sudah dibahas. Kesemua chapter tersebut sangat berhubungan dengan pembahasan chapter ini."),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/path-item"},"A.18. Module System \u279c Path & Item")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/package-crate"},"A.19. Module System \u279c Package & Crate")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/module-basic"},"A.20. Module System \u279c Module")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/module-inline"},"A.21. Module System \u279c Inline Module")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/module-scope-item-access"},"A.22. Module System \u279c Scope & Akses Item"))),(0,t.kt)("h2",{id:"a282-default-visibility"},"A.28.2. Default visibility"),(0,t.kt)("p",null,"Di Rust, ",(0,t.kt)("em",{parentName:"p"},"by default"),", hampir semua item adalah private. Apa efeknya ketika item adalah private atau publik? Silakan ingat 2 aturan penting berikut:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},"Jika suatu item adalah private, maka item tersebut hanya bisa diakses dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," dan dari ",(0,t.kt)("em",{parentName:"li"},"submodules")," milik ",(0,t.kt)("em",{parentName:"li"},"current module"),"."),(0,t.kt)("li",{parentName:"ol"},"Jika suatu item adalah publik, maka dia bisa diakses dari module lain di luar ",(0,t.kt)("em",{parentName:"li"},"current module scope"),", dengan catatan parent module scope item tersebut harus publik.")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Kita sepakati di sini, pada istilah ",(0,t.kt)("strong",{parentName:"p"},"current module")," kata ",(0,t.kt)("em",{parentName:"p"},"module")," di situ bisa saja tertuju untuk module atau juga submodule")),(0,t.kt)("p",null,"Dua point di atas sangat penting untuk dipahami, karena digunakan sebagai landasan pertimbangan dalam penyusunan hirarki module. Sebagai contoh, kita bisa membuat program yang hanya meng-expose API tertentu (yang memang diperlukan untuk diakses oleh publik), tanpa perlu ikut meng-expose detail implementasinya."),(0,t.kt)("p",null,"Ok, sekarang silakan perhatikan path sederhana di bawah ini. Diasumsikan ada sebuah fungsi yang path aksesnya adalah berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash"},"messaging::service_layer::some_black_magic\n")),(0,t.kt)("p",null,"Segmen pertama yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging")," pasti adalah publik, karena di-import ke ",(0,t.kt)("em",{parentName:"p"},"crate root"),". Lalu bagaimana dengan segmen ",(0,t.kt)("inlineCode",{parentName:"p"},"service_layer")," dan juga ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic"),"?"),(0,t.kt)("p",null,"Jika item ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic")," di situ adalah publik, maka idealnya pengaksesan menggunakan path tersebut memungkinkan. Tapi kembali ke point ke-2 aturan yang sudah dibahas di atas, yaitu meskipun ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic")," adalah publik, jika parent-nya (yang pada konteks ini adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"service_layer"),") adalah private, maka pengaksesan menggunakan path tersebut menghasilkan error."),(0,t.kt)("p",null,"Intinya, ",(0,t.kt)("strong",{parentName:"p"},"sebuah item bisa diakses jika item tersebut adalah publik, dan parent item tersebut juga publik. Sedangkan default visibility untuk hampir semua item adalah private.")),(0,t.kt)("p",null,"Ok, sekarang mari lanjut ke praktik menggunakan contoh dengan pembahasan yang lebih mendetail. Silakan perhatikan dan praktikkan kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 messaging.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'const SOME_MESSAGE: &str = "hello rust";\n\nmod service_layer {\n\n pub fn some_black_magic() {\n println!("{}", crate::messaging::SOME_MESSAGE)\n }\n}\n\npub fn say_hello() {\n service_layer::some_black_magic();\n}\n')),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::say_hello();\n}\n")),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Konstanta ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::SOME_MESSAGE")," adalah ",(0,t.kt)("strong",{parentName:"p"},"private"),". Penjelasan:"),(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Konstanta ini merupakan module item milik ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Konstanta ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),")."),(0,t.kt)("li",{parentName:"ul"},"Konstanta ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari submodule milik ",(0,t.kt)("em",{parentName:"li"},"current module"),", yaitu submodule dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),". Contohnya bisa dilihat pada fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer::some_black_magic")," yang di situ ada statement pemanggilan ",(0,t.kt)("inlineCode",{parentName:"li"},"SOME_MESSAGE"),"."),(0,t.kt)("li",{parentName:"ul"},"Konstanta ini ",(0,t.kt)("strong",{parentName:"li"},"tidak bisa diakses")," dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),")."))),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Submodule ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::service_layer")," adalah ",(0,t.kt)("strong",{parentName:"p"},"private"),". Penjelasan:"),(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Submodule ini merupakan module item milik ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Submodule ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"). Contohnya bisa dilihat pada fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::say_hello")," yang di situ ada statement pemanggilan ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer"),"."),(0,t.kt)("li",{parentName:"ul"},"Submodule ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari submodule milik ",(0,t.kt)("em",{parentName:"li"},"current module"),", yaitu submodule dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Submodule ini ",(0,t.kt)("strong",{parentName:"li"},"tidak bisa diakses")," dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),")."))),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::service_layer::some_black_magic")," adalah ",(0,t.kt)("strong",{parentName:"p"},"publik"),". Penjelasan:"),(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Fungsi ini merupakan module item milik ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),"."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),")."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari submodule milik ",(0,t.kt)("em",{parentName:"li"},"current module"),", yaitu submodule dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),". Contohnya bisa dilihat pada fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::say_hello")," yang di situ ada statement pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"some_black_magic"),"."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),")."),(0,t.kt)("li",{parentName:"ul"},"Namun meskipun demikian, bisa tidaknya fungsi ini diakses dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),") juga tergantung dengan visibility dari current module itu sendiri, yaitu ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer"),"."),(0,t.kt)("li",{parentName:"ul"},"Karena module ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer")," adalah private, meskipun fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"some_black_magic")," di dalamnya adalah publik, pengaksesan fungsi tersebut dari luar module scope ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer")," tidak dimungkinkan.",(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Pengaksesan ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer::some_black_magic")," dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::say_hello")," tidak error karena submodule ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer")," meskipun private, posisinya adalah masih dalam satu module scope yang sama dengan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"say_hello"),"."),(0,t.kt)("li",{parentName:"ul"},"Dimisalkan jika ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer::some_black_magic")," dipaksa diakses dari ",(0,t.kt)("inlineCode",{parentName:"li"},"main"),", maka muncul error karena ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer")," adalah private dan posisinya tidak berada dalam module scope yang sama dengan crate root (",(0,t.kt)("inlineCode",{parentName:"li"},"main"),")."))))),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::say_hello")," adalah ",(0,t.kt)("strong",{parentName:"p"},"public"),". Penjelasan:"),(0,t.kt)("ul",{parentName:"li"},(0,t.kt)("li",{parentName:"ul"},"Fungsi ini merupakan module item milik ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),")."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari submodule milik ",(0,t.kt)("em",{parentName:"li"},"current module"),", yaitu submodule dari ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"."),(0,t.kt)("li",{parentName:"ul"},"Fungsi ini ",(0,t.kt)("strong",{parentName:"li"},"bisa diakses")," dari luar ",(0,t.kt)("em",{parentName:"li"},"current module scope")," (",(0,t.kt)("inlineCode",{parentName:"li"},"messaging"),"). Contohnya bisa dilihat pada crate root fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"main"),", di situ ada pemanggilan statement ",(0,t.kt)("inlineCode",{parentName:"li"},"say_hello"),".")))),(0,t.kt)("h2",{id:"a283-re-export-item"},"A.28.3. Re-export item"),(0,t.kt)("p",null,"Pada contoh, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::say_hello")," didesain sebagai media untuk mengakses fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic"),". Di situasi ",(0,t.kt)("em",{parentName:"p"},"real world")," pastinya sangat jarang terjadi sebuah fungsi isinya hanya satu baris pemanggilan fungsi lainnya. Jika memang ada situasi seperti itu, (kontekstual) lebih baik hapus saja fungsi yang jadi media pemanggilan dan langsung saja panggil fungsi di dalamnya sesuai kebutuhan."),(0,t.kt)("p",null,"Pada praktik selanjutnya ini kita misalkan bahwa fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," isinya memang hanya 1 baris, dan yang paling penting adalah isi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic")," perlu untuk bisa diakses dari ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),". Untuk kasus seperti ini ada 3 alternatif solusi:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},"Tidak perlu mengubah apapun, gunakan saja kode yang sudah ditulis di atas. Kode tersebut sudah bisa mengakomodir pemanggilan ",(0,t.kt)("inlineCode",{parentName:"li"},"some_black_magic")," via ",(0,t.kt)("inlineCode",{parentName:"li"},"say_hello"),"."),(0,t.kt)("li",{parentName:"ol"},"Atau, hapus saja fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"say_hello"),", lalu ubah visibility module ",(0,t.kt)("inlineCode",{parentName:"li"},"service_layer")," menjadi publik, dengan demikian kita bisa mengakses ",(0,t.kt)("inlineCode",{parentName:"li"},"some_black_magic")," dari ",(0,t.kt)("inlineCode",{parentName:"li"},"main")," menggunakan path ",(0,t.kt)("inlineCode",{parentName:"li"},"messaging::service_layer::some_black_magic"),"."),(0,t.kt)("li",{parentName:"ol"},"Atau, gunakan teknik ",(0,t.kt)("strong",{parentName:"li"},"re-export item"),".")),(0,t.kt)("p",null,"Re-export item adalah sebuah cara untuk mem-",(0,t.kt)("em",{parentName:"p"},"bypass")," pengaksesan item yang secara hirarki memang tidak bisa diakses dari luar module (bisa jadi karena visibility item ataupun parent module nya adalah private). Dengan teknik ini, maka item pasti bisa diakses dari luar module."),(0,t.kt)("p",null,"Item yang di-re-export akan menjadi item milik ",(0,t.kt)("em",{parentName:"p"},"current module")," di mana statement re-export tersebut ditulis."),(0,t.kt)("p",null,"Cara re-export item adalah menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub use")," kemudian diikuti dengan path yang ingin di-export dan juga nama export item dengan notasi penulisan berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"pub use the_path as exported_name;\npub use self::service_layer::some_black_magic as say_hello;\n")),(0,t.kt)("p",null,"Contoh jika diterapkan pada program yang sebelumnya sudah ditulis:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'pub use self::service_layer::some_black_magic as say_hello;\n\nconst SOME_MESSAGE: &str = "hello rust";\n\nmod service_layer {\n\n pub fn some_black_magic() {\n println!("{}", crate::messaging::SOME_MESSAGE)\n }\n}\n')),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::say_hello();\n}\n")),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Visibility & Privacy",src:n(2201).Z,width:"466",height:"109"})),(0,t.kt)("p",null,"Bisa dilihat di contoh di atas, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," dihapus, kemudian item ",(0,t.kt)("inlineCode",{parentName:"p"},"service_layer::some_black_magic")," di-re-export dengan nama ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello"),". Dengannya kita bisa mengakses ",(0,t.kt)("inlineCode",{parentName:"p"},"some_black_magic")," dari luar module ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging")," menggunakan path ",(0,t.kt)("inlineCode",{parentName:"p"},"messaging::say_hello"),"."),(0,t.kt)("p",null,"Jika item ingin di-re-export tanpa perubahan nama item, bisa gunakan notasi berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"pub use the_path;\npub use self::service_layer::some_black_magic;\n")),(0,t.kt)("p",null,"Jika diterapkan pada program sebelumnya, kurang lebih seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'pub use self::service_layer::some_black_magic;\n\nconst SOME_MESSAGE: &str = "hello rust";\n\nmod service_layer {\n\n pub fn some_black_magic() {\n println!("{}", crate::messaging::SOME_MESSAGE)\n }\n}\n')),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::some_black_magic();\n}\n")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"use")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/use"},"Module System \u279c Use"))),(0,t.kt)("h2",{id:"a284-public-visibility-scope"},"A.28.4. Public visibility scope"),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," digunakan untuk mengubah visibility item menjadi publik. Keyword ini bisa dikombinasikan dengan salah satu dari keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"self"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"crate"),", dan ",(0,t.kt)("inlineCode",{parentName:"p"},"super"),"; denganya kita bisa menentukan visibility sebuah publik item dengan scope yang lebih spesifik."),(0,t.kt)("h3",{id:"-keyword-pub"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub")),(0,t.kt)("p",null,"Penulis rasa untuk penerapan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," ini sudah sangat jelas, kita beberapa kali mempraktikkannya."),(0,t.kt)("p",null,"Dengan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),", sebuah item visibility-nya menjadi publik."),(0,t.kt)("h3",{id:"-keyword-pubin-path"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub(in path)")),(0,t.kt)("p",null,"Keyword ini menjadikan visibility item hanya di dalam ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," yang ditulis di ",(0,t.kt)("inlineCode",{parentName:"p"},"pub(in path)"),", dengan ketentuan ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," tersebut merupakan parent dari module item di mana keyword digunakan."),(0,t.kt)("p",null,"Contohnya bisa dilihat pada kode berikut. Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," didefinisikan publik dengan scope path ditentukan secara eksplisit adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"crate::outer_mod"),". Dengan demikian fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," hanya bisa diakses dari dalam ",(0,t.kt)("inlineCode",{parentName:"p"},"outer_mod"),"."),(0,t.kt)("p",null,"Bisa dilihat di contoh, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," diakses dari ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),". Silakan coba saja paksa untuk mengaksesnya dari fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", hasilnya pasti error."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'pub mod outer_mod {\n\n pub mod inner_mod {\n\n // fungsi say_hello berikut hanya bisa diakses dari dalam `outer_mod`.\n // pengaksesannya dari luar `outer_mod` menghasilkan error.\n pub(in crate::outer_mod) fn say_hello() {\n println!("hello rust")\n }\n }\n\n pub fn do_something() {\n inner_mod::say_hello();\n }\n}\n\nfn main() {\n outer_mod::do_something();\n}\n')),(0,t.kt)("h3",{id:"-keyword-pubcrate"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub(crate)")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub(crate)")," digunakan untuk membuat visibility item menjadi publik dengan scope akses ",(0,t.kt)("em",{parentName:"p"},"current crate"),". Dengan ini item bisa diakses dari manapun asalakan masih dalam crate yang sama."),(0,t.kt)("p",null,"Contoh penerapannya bisa dilihat berikut ini. Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," visibility scope nya ditentukan adalah ",(0,t.kt)("em",{parentName:"p"},"current crate"),". Fungsi tersebut bisa diakses dari ",(0,t.kt)("inlineCode",{parentName:"p"},"outer_mod_one::do_something"),", dari ",(0,t.kt)("inlineCode",{parentName:"p"},"outer_mod_two::do_something"),", dan juga dari fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'pub mod outer_mod_one {\n\n pub mod inner_mod {\n\n // fungsi ini visibility scope-nya di level crate\n pub(crate) fn say_hello() {\n println!("hello rust")\n }\n }\n\n pub fn do_something() {\n inner_mod::say_hello();\n }\n}\n\npub mod outer_mod_two {\n\n pub fn do_something() {\n crate::outer_mod_one::inner_mod::say_hello();\n }\n}\n\nfn main() {\n outer_mod_one::inner_mod::say_hello();\n outer_mod_one::do_something();\n outer_mod_two::do_something();\n}\n')),(0,t.kt)("h3",{id:"-keyword-pubsuper"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub(super)")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub(super)")," digunakan untuk membuat visibility item menjadi publik dengan scope akses ",(0,t.kt)("em",{parentName:"p"},"parent module"),"."),(0,t.kt)("p",null,"Pada contoh berikut, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," visibility scope nya ditentukan adalah ",(0,t.kt)("em",{parentName:"p"},"parent module"),", artinya fungsi tersebut hanya bisa diakses dari dalam ",(0,t.kt)("em",{parentName:"p"},"parent module")," (yang pada konteks ini adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"outer_mod"),")."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'pub mod outer_mod {\n\n pub mod inner_mod {\n\n // fungsi ini visibility scope-nya di parent module scope,\n // yaitu `outer_mod`\n pub(super) fn say_hello() {\n println!("hello rust")\n }\n }\n\n pub fn do_something() {\n inner_mod::say_hello();\n }\n}\n\nfn main() {\n outer_mod::do_something();\n}\n')),(0,t.kt)("h3",{id:"-keyword-pubself"},"\u25c9 Keyword ",(0,t.kt)("inlineCode",{parentName:"h3"},"pub(self)")),(0,t.kt)("p",null,"Keyword ini digunakan untuk membuat visibility item menjadi publik dengan scope akses hanya pada current module. Contohnya bisa dilihat pada kode program berikut."),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," visibility scope-nya adalah ",(0,t.kt)("em",{parentName:"p"},"current module"),". Fungsi tersebut hanya bisa diakses dari tempat yang merupakan module item dari ",(0,t.kt)("em",{parentName:"p"},"current module")," yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"inner_mod"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'pub mod outer_mod {\n\n pub mod inner_mod {\n\n // fungsi ini visibility scope-nya di current module scope,\n // yaitu `inner_mod`\n pub(self) fn say_hello() {\n println!("hello rust")\n }\n\n pub fn do_something() {\n say_hello();\n }\n }\n}\n\nfn main() {\n outer_mod::inner_mod::do_something();\n}\n')),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/visibility_privacy"},"github.com/novalagung/dasarpemrogramanrust-example/../visibility_privacy")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/visibility-and-privacy.html"},"https://doc.rust-lang.org/reference/visibility-and-privacy.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html"))))}d.isMDXComponent=!0},2201:(e,a,n)=>{n.d(a,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/4232c3ed.0eade53f.js b/assets/js/4232c3ed.0eade53f.js deleted file mode 100644 index 8bb5a038..00000000 --- a/assets/js/4232c3ed.0eade53f.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8308],{3905:(a,e,n)=>{n.d(e,{Zo:()=>d,kt:()=>A});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function r(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var u=t.createContext({}),m=function(a){var e=t.useContext(u),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},d=function(a){var e=m(a.components);return t.createElement(u.Provider,{value:e},a.children)},o={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},k=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,l=a.originalType,u=a.parentName,d=p(a,["components","mdxType","originalType","parentName"]),k=m(n),A=i,s=k["".concat(u,".").concat(A)]||k[A]||o[A]||l;return n?t.createElement(s,r(r({ref:e},d),{},{components:n})):t.createElement(s,r({ref:e},d))}));function A(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var l=n.length,r=new Array(l);r[0]=k;var p={};for(var u in e)hasOwnProperty.call(e,u)&&(p[u]=e[u]);p.originalType=a,p.mdxType="string"==typeof a?a:i,r[1]=p;for(var m=2;m{n.r(e),n.d(e,{assets:()=>u,contentTitle:()=>r,default:()=>o,frontMatter:()=>l,metadata:()=>p,toc:()=>m});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:26,title:"A.26. Enum",sidebar_label:"A.26. Enum"},r=void 0,p={unversionedId:"basic/enum",id:"basic/enum",title:"A.26. Enum",description:"Enum atau enumerated type adalah sebuah tipe data yang digunakan untuk menampung nilai konstan. Pada chapter ini kita akan mempelajarinya.",source:"@site/docs/basic/enum.md",sourceDirName:"basic",slug:"/basic/enum",permalink:"/basic/enum",draft:!1,tags:[],version:"current",sidebarPosition:26,frontMatter:{sidebar_position:26,title:"A.26. Enum",sidebar_label:"A.26. Enum"},sidebar:"tutorialSidebar",previous:{title:"A.25. Method",permalink:"/basic/method"},next:{title:"A.27. Type Alias & Casting",permalink:"/basic/type-alias-casting"}},u={},m=[{value:"A.26.1. Keyword enum",id:"a261-keyword-enum",level:2},{value:"A.26.2. Naming convention enum",id:"a262-naming-convention-enum",level:2},{value:"A.26.3. Seleksi kondisi enum",id:"a263-seleksi-kondisi-enum",level:2},{value:"A.26.4. Keyword match",id:"a264-keyword-match",level:2},{value:"A.26.5. Enum value \u279c tuple struct-like",id:"a265-enum-value--tuple-struct-like",level:2},{value:"A.26.6. Enum value \u279c struct-like",id:"a266-enum-value--struct-like",level:2},{value:"A.26.7. Aturan pattern matching enum",id:"a267-aturan-pattern-matching-enum",level:2},{value:"A.26.8. Enum module & visibility",id:"a268-enum-module--visibility",level:2},{value:"A.26.9. Generic pada enum",id:"a269-generic-pada-enum",level:2},{value:"A.26.10. Pembahasan lanjutan pattern matching",id:"a2610-pembahasan-lanjutan-pattern-matching",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function o(a){let{components:e,...l}=a;return(0,i.kt)("wrapper",(0,t.Z)({},d,l,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Enum atau ",(0,i.kt)("em",{parentName:"p"},"enumerated type")," adalah sebuah tipe data yang digunakan untuk menampung nilai konstan. Pada chapter ini kita akan mempelajarinya."),(0,i.kt)("p",null,"Enum ada sedikit kemiripan dengan konstanta, bedanya ada pada nilai atau ",(0,i.kt)("em",{parentName:"p"},"underlying value"),"-nya. Jika di konstanta, yang didefinisikan adalah nama beserta value-nya, di enum yang didefinisikan adalah tipe data enum dan enum value. Enum value ini bentuknya seperti variabel tanpa nilai (lebih tepatnya nama dari enum value tersebut adalah nilainya). Lebih jelasnya silakan ikut pembahasan chapter ini."),(0,i.kt)("h2",{id:"a261-keyword-enum"},"A.26.1. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"enum")),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"enum")," digunakan untuk membuat ",(0,i.kt)("em",{parentName:"p"},"enumerated type"),". Cara penulisannya seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum NamaEnum {\n NilaiEnum1,\n Nilai2,\n NilaiEnumKe3,\n // ...\n}\n")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"NamaEnum")," di atas adalah tipe data custom yang didefinisikan bertipe enum. Sedangkan ",(0,i.kt)("inlineCode",{parentName:"p"},"NilaiEnum1"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Nilai2"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"NilaiEnumKe3")," adalah yang disebut dengan enum value. Dengan itu maka ketiga enum values tersebut tipe datanya adalah sama, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"NamaEnum"),"."),(0,i.kt)("p",null,"Mari kita lanjut praktik. Berikut ini adalah definisi konstanta yang menggunakan tipe data string untuk menampung nilai konstan-nya. Lalu dibawahnya ada lagi definisi nilai konstan tetapi menggunakan enum sebagai tipe data yang digunakan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// definisi konstanta\nconst SuperheroSuperman: &str = "superman";\nconst SuperheroOmniMan: &str = "omniman";\nconst SuperheroHomelander: &str = "homelander";\nconst SuperheroHyperion: &str = "hyperion";\n\n// definisi enum\nenum Superhero {\n Superman,\n OmniMan,\n Homelander,\n Hyperion,\n};\n')),(0,i.kt)("p",null,"Di contoh bisa dilihat, ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero")," adalah tipe data enum baru. Dari tipe data tersebut dibuat 4 buah enum values, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"Superman"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"OmniMan"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Homelander"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Hyperion"),"."),(0,i.kt)("p",null,"Pada pembuatan konstanta, tipe data beserta value-nya harus ditentukan di awal. Pada enum, yang perlu didefinisikan adalah tipe data enum-nya (sebagai contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero"),") kemudian diikut dengan enum value yang dituliskan tanpa pengisian nilai."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Definisi variabel dengan isi konstanta:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let value1: &str = SuperheroSuperman;\nlet value2 = SuperheroOmniMan;\n// ...\n"))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Definisi variabel bertipe data enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero"),":"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let value3: Superhero = Superhero::Superman;\nlet value4 = Superhero::OmniMan;\n// ...\n")))),(0,i.kt)("p",null,"Notasi path digunakan dalam penulisan enum value dengan format ",(0,i.kt)("inlineCode",{parentName:"p"},"NamaEnum::EnumValue")),(0,i.kt)("h2",{id:"a262-naming-convention-enum"},"A.26.2. ",(0,i.kt)("em",{parentName:"h2"},"Naming convention")," enum"),(0,i.kt)("p",null,"Sesuai anjuran di ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", upper camel case digunakan dalam penamaan Enum beserta value-nya."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Superhero {\n Superman,\n OmniMan,\n Homelander,\n Hyperion,\n};\n")),(0,i.kt)("h2",{id:"a263-seleksi-kondisi-enum"},"A.26.3. Seleksi kondisi enum"),(0,i.kt)("p",null,"Tipe data enum biasa dipakai pada seleksi kondisi, namun caranya sedikit berbeda. Default-nya keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," tidak bisa digunakan pada tipe data enum."),(0,i.kt)("p",null,"Pada contoh berikut, statement seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"value3")," menghasilkan error:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// seleksi kondisi pada konstanta\nif value1 == SuperheroSuperman {\n println!("hello superman!");\n}\n\n// seleksi kondisi pada enum\nif value3 == Superhero::Superman {\n println!("hello superman!");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(1179).Z,width:"859",height:"378"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Error tersebut muncul karena tipe data enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero")," tidak memiliki trait ",(0,i.kt)("inlineCode",{parentName:"p"},"PartialEq"),". Lebih jelasnya mengenai trait dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),".")),(0,i.kt)("p",null,"Lalu bagaimana cara pengaplikasian seleksi kondisi pada tipe enum? Ada beberapa cara, namun yang paling praktis adalah keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),"."),(0,i.kt)("h2",{id:"a264-keyword-match"},"A.26.4. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"match")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"match")," adalah salah satu keyword untuk operasi seleksi kondisi di Rust. Penerapan keyword ini cukup luas, namun pada chapter ini hanya akan dibahas penerapannya yang relevan dengan topik enum."),(0,i.kt)("p",null,"Mari kita pelajarinya sembari praktik. Silakan buat package baru, lalu definisikan tipe enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Food")," berikut beserta 4 enum value-nya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Food {\n PenyetanTerangBulan,\n PizzaNanas,\n EsKrimIkanMujaer,\n MiGorengKuah,\n}\n")),(0,i.kt)("p",null,"Lalu buat sebuah variabel bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," untuk menampung salah satu nilai enum. Kemudian gunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk menerapkan operasi seleksi kondisi dengan aksi menampilkan sebuah pesan sesuai dengan nilai yang cocok."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let makanan_favorit: Food = Food::PenyetanTerangBulan;\n \n match makanan_favorit {\n Food::PenyetanTerangBulan => {\n println!("your food taste is quite ... unique");\n },\n Food::PizzaNanas => {\n println!("it\'s morally wrong to have pineaple on top of pizza");\n },\n Food::EsKrimIkanMujaer => {\n println!("I don\'t know what to say");\n },\n Food::MiGorengKuah => {\n println!("sometimes people do eat this, but it\'s ok");\n }\n }\n}\n')),(0,i.kt)("p",null,"Di atas bisa dilihat bagaimana cara penggunaan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk penerapan seleksi kondisi pada tipe data enum."),(0,i.kt)("p",null,"Notasi penulisannya kurang lebih seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"}," match variabel_enum {\n TipeEnum::ValueEnum1 => {\n // ...\n },\n TipeEnum::ValueEnum2 => {\n // ...\n },\n \n // ...\n}\n")),(0,i.kt)("p",null,"Kembali ke contoh program, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," dicek nilainya menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Jika nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::PenyetanTerangBulan"),", muncul pesan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'"your food taste is quite ... unique"\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Jika nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::PizzaNanas"),", muncul pesan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'"it is morally wrong to have pineaple on top of pizza"\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Jika nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::EsKrimIkanMujaer"),", muncul pesan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'"I don\'t know what to say"\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Jika nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MiGorengKuah"),", muncul pesan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'"sometimes people do eat this, but it\'s ok"\n')))),(0,i.kt)("p",null,"Jalankan program untuk melihat hasilnya:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(5543).Z,width:"450",height:"82"})),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," ini sebenarnya tidak hanya digunakan untuk seleksi kondisi saja. Di Rust ada yang disebut dengan ",(0,i.kt)("em",{parentName:"p"},"pattern matching"),". Metode ",(0,i.kt)("em",{parentName:"p"},"pattern matching")," ini memahami ",(0,i.kt)("em",{parentName:"p"},"special syntax")," yang kegunaanya lebih luas dibanding hanya sekedar seleksi kondisi biasa."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai ",(0,i.kt)("em",{parentName:"p"},"pattern matching")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"))),(0,i.kt)("h2",{id:"a265-enum-value--tuple-struct-like"},"A.26.5. Enum value \u279c ",(0,i.kt)("em",{parentName:"h2"},"tuple struct"),"-like"),(0,i.kt)("p",null,"Enum value di struct bisa juga didesain seperti ",(0,i.kt)("em",{parentName:"p"},"tuple struct"),". Sebagai contoh, enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Food")," di atas akan kita tambahi dengan satu enum value baru berbentuk ",(0,i.kt)("em",{parentName:"p"},"tuple struct"),"."),(0,i.kt)("p",null,"Silakan tambahkan enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"MakananLainnya")," berikut. Enum ini kita fungsikan untuk mengidentifikasi data makanan lainnya selain dari yang sudah ada di enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Food"),". Notasi penulisan ",(0,i.kt)("em",{parentName:"p"},"tuple struct")," ",(0,i.kt)("inlineCode",{parentName:"p"},"MakananLainnya(String)")," artinya enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"MakananLainnya")," didefinisikan untuk bisa menampung data property dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Food {\n PenyetanTerangBulan,\n PizzaNanas,\n EsKrimIkanMujaer,\n MiGorengKuah,\n MakananLainnya(String), // <---- enum value baru\n}\n")),(0,i.kt)("p",null,"Sekarang ubah isi variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," dengan enum value baru yang sudah dibuat. Syntax ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya(nasi_goreng)")," artinya enum value yang digunakan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya")," dengan isi property didapat dari variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"nasi_goreng"),"."),(0,i.kt)("p",null,"Tambahkan juga ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya")," dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n // enum value MakananLainnya digunakan\n // dengan isi property adalah string "nasi goreng"\n let nasi_goreng = String::from("nasi goreng");\n let makanan_favorit = Food::MakananLainnya(nasi_goreng);\n \n match makanan_favorit {\n Food::PenyetanTerangBulan => {\n println!("your food taste is quite ... unique");\n },\n Food::PizzaNanas => {\n println!("it\'s morally wrong to have pineaple on top of pizza");\n },\n Food::EsKrimIkanMujaer => {\n println!("I don\'t know what to say. this should be illegal");\n },\n Food::MiGorengKuah => {\n println!("sometimes people do eat this, but it\'s ok");\n },\n Food::MakananLainnya(m) => { // <---- seleksi kondisi baru\n println!("do you like {m}? nice taste!");\n }\n }\n}\n')),(0,i.kt)("p",null,"Bisa dilihat ada keunikan dalam penulisan seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya")," dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),". Disitu ada parameter bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"m")," yang parameter tersebut akan berisi data property jika memang ",(0,i.kt)("em",{parentName:"p"},"match")," dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit"),"."),(0,i.kt)("p",null,"Coba jalankan untuk melihat hasilnya:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(8424).Z,width:"462",height:"87"})),(0,i.kt)("p",null,"O iya, jumlah property value enum berbentuk ",(0,i.kt)("em",{parentName:"p"},"tuple struct")," ini tidak terbatas ya. Pada contoh di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya")," hanya memiliki 1 property. Lebih dari satu juga bisa."),(0,i.kt)("h2",{id:"a266-enum-value--struct-like"},"A.26.6. Enum value \u279c ",(0,i.kt)("em",{parentName:"h2"},"struct"),"-like"),(0,i.kt)("p",null,"Enum value bisa juga didesain memiliki property seperti ",(0,i.kt)("em",{parentName:"p"},"struct"),"."),(0,i.kt)("p",null,"Mari kita terapkan pada kode sebelumnya. Tambahkan 1 buah enum value lagi dengan nama ",(0,i.kt)("inlineCode",{parentName:"p"},"MieSetan")," yang ditulis dalam bentuk ",(0,i.kt)("em",{parentName:"p"},"struct"),"-like, dan memiliki 2 buah property."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Food {\n PenyetanTerangBulan,\n PizzaNanas,\n EsKrimIkanMujaer,\n MiGorengKuah,\n MakananLainnya(String),\n MieSetan { level_pedas: i32, pakek_piring: bool }\n}\n")),(0,i.kt)("p",null,"Setelah itu, isi variabel dengan nilai adalah enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MieSetan"),", level pedasnya 5, dan tanpa piring."),(0,i.kt)("p",null,"Tak lupa tambahkan seleksi kondisi untuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MieSetan")," pada blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let makanan_favorit = Food::MieSetan { \n level_pedas: 5,\n pakek_piring: false\n };\n \n match makanan_favorit {\n Food::PenyetanTerangBulan => {\n println!("your food taste is quite ... unique");\n },\n Food::PizzaNanas => {\n println!("it\'s morally wrong to have pineaple on top of pizza");\n },\n Food::EsKrimIkanMujaer => {\n println!("I don\'t know what to say. this should be illegal");\n },\n Food::MiGorengKuah => {\n println!("sometimes people do eat this, but it\'s ok");\n },\n Food::MakananLainnya(m) => {\n println!("do you like {m}? nice taste!");\n },\n Food::MieSetan { level_pedas, pakek_piring } => {\n if level_pedas > 3 {\n println!("mie setan lvl {} is too much!", level_pedas);\n } else {\n println!("mie setan lvl {} is perfect!", level_pedas);\n }\n\n if !pakek_piring {\n println!("how are you going to eat the food without a plate, huh?");\n }\n }\n }\n}\n')),(0,i.kt)("p",null,"Dalam seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MieSetan")," bisa dilihat ada beberapa statement. Kurang lebih jika nilai dari variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MieSetan")," maka:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Akan memunculkan pesan yang berbeda tergantung level pedasnya"),(0,i.kt)("li",{parentName:"ul"},"Dan jika terdeteksi tidak menggunakan piring, dimunculkan pesan tambahan")),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(6989).Z,width:"448",height:"98"})),(0,i.kt)("h2",{id:"a267-aturan-pattern-matching-enum"},"A.26.7. Aturan ",(0,i.kt)("em",{parentName:"h2"},"pattern matching")," enum"),(0,i.kt)("p",null,"Dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),", semua enum value harus dituliskan. Jika tidak, pasti muncul error. Contohnya bisa dilihat di gambar berikut, beberapa seleksi kondisi enum value di-remark, hasilnya ada error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(2930).Z,width:"1096",height:"309"})),(0,i.kt)("p",null,"Error tersebut sebenarnya bisa diantisipasi dengan menambahkan seleksi kondisi dengan penulisan seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'match makanan_favorit {\n Food::PenyetanTerangBulan => {\n println!("your food taste is quite ... unique");\n },\n Food::PizzaNanas => {\n println!("it\'s morally wrong to have pineaple on top of pizza");\n },\n _ => {\n println!("never heard about that food");\n }\n}\n')),(0,i.kt)("p",null,"Menggunakan blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," di atas, jika nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," adalah selain ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::PenyetanTerangBulan")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::PizzaNanas"),", maka pesan ",(0,i.kt)("inlineCode",{parentName:"p"},"never heard about that food")," adalah yang muncul di layar."),(0,i.kt)("p",null,"Selain variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"_")," bisa juga menggunakan nama variabel apapun, misalnya ",(0,i.kt)("inlineCode",{parentName:"p"},"some_var"),". Namun jika variabel tersebut tidak digunakan dalam blok kode, akan muncul warning."),(0,i.kt)("h2",{id:"a268-enum-module--visibility"},"A.26.8. Enum module & visibility"),(0,i.kt)("p",null,"Mari kita coba cek perihal visibility dari enum. Siapkan package baru dengan struktur seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 constants.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("p",null,"Pada file ",(0,i.kt)("inlineCode",{parentName:"p"},"constants.rs"),", tambahkan enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Company")," berikut. Pastikan enum adalah publik dengan menambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," pada deklarasinya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/constants.rs"',title:'"src/constants.rs"'},"pub enum Company {\n Apple,\n Microsoft,\n Google,\n Github\n}\n")),(0,i.kt)("p",null,"Kemudian tambahkan kode berikut di ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod constants;\n\nfn main() {\n let company = constants::Company::Apple;\n \n match company {\n constants::Company::Apple => {\n print!("apple")\n },\n _ => {\n print!("other than apple")\n }\n }\n}\n')),(0,i.kt)("p",null,"Jalankan, hasilnya tidak error, karena ",(0,i.kt)("inlineCode",{parentName:"p"},"Company")," didefinisikan publik."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(3101).Z,width:"457",height:"85"})),(0,i.kt)("p",null,"Coba lakukan modifikasi dengan menghilangkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," saat definisi enum, hasilnya pasti error."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Pada tipe data enum, keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," cukup ditambahkan pada definisi enum type, tidak perlu ditambahkan satu persatu di tiap enum values.")),(0,i.kt)("h2",{id:"a269-generic-pada-enum"},"A.26.9. Generic pada enum"),(0,i.kt)("p",null,"Pembahasan mengenai generic pada enum ada pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics#a368-generics-enum"},"Generics"),"."),(0,i.kt)("h2",{id:"a2610-pembahasan-lanjutan-pattern-matching"},"A.26.10. Pembahasan lanjutan ",(0,i.kt)("em",{parentName:"h2"},"pattern matching")),(0,i.kt)("p",null,"Pembahasan yang lebih mendetail tentang keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," dan ",(0,i.kt)("em",{parentName:"p"},"pattern matching")," ada pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/enum"},"github.com/novalagung/dasarpemrogramanrust-example/../enum")),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang associated function dan method pada enum")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch06-01-defining-an-enum.html"},"https://doc.rust-lang.org/book/ch06-01-defining-an-enum.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/custom_types/enum.html"},"https://doc.rust-lang.org/rust-by-example/custom_types/enum.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/items/enumerations.html"},"https://doc.rust-lang.org/reference/items/enumerations.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch06-02-match.html"},"https://doc.rust-lang.org/book/ch06-02-match.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.enum.html"},"https://doc.rust-lang.org/std/keyword.enum.html"))))}o.isMDXComponent=!0},1179:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/enum-1-15c91a97fbd3c87f01a4c4c5060b5fd8.png"},5543:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},8424:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},6989:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},2930:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/enum-5-c9da1005565fda09175ddda67f9a16b5.png"},3101:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/4232c3ed.32b73572.js b/assets/js/4232c3ed.32b73572.js new file mode 100644 index 00000000..87502edf --- /dev/null +++ b/assets/js/4232c3ed.32b73572.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8308],{3905:(a,e,n)=>{n.d(e,{Zo:()=>d,kt:()=>A});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function r(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var u=t.createContext({}),m=function(a){var e=t.useContext(u),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},d=function(a){var e=m(a.components);return t.createElement(u.Provider,{value:e},a.children)},o={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},k=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,l=a.originalType,u=a.parentName,d=p(a,["components","mdxType","originalType","parentName"]),k=m(n),A=i,s=k["".concat(u,".").concat(A)]||k[A]||o[A]||l;return n?t.createElement(s,r(r({ref:e},d),{},{components:n})):t.createElement(s,r({ref:e},d))}));function A(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var l=n.length,r=new Array(l);r[0]=k;var p={};for(var u in e)hasOwnProperty.call(e,u)&&(p[u]=e[u]);p.originalType=a,p.mdxType="string"==typeof a?a:i,r[1]=p;for(var m=2;m{n.r(e),n.d(e,{assets:()=>u,contentTitle:()=>r,default:()=>o,frontMatter:()=>l,metadata:()=>p,toc:()=>m});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:26,title:"A.26. Enum",sidebar_label:"A.26. Enum"},r=void 0,p={unversionedId:"basic/enum",id:"basic/enum",title:"A.26. Enum",description:"Enum atau enumerated type adalah sebuah tipe data yang digunakan untuk menampung nilai konstan. Pada chapter ini kita akan mempelajarinya.",source:"@site/docs/basic/enum.md",sourceDirName:"basic",slug:"/basic/enum",permalink:"/basic/enum",draft:!1,tags:[],version:"current",sidebarPosition:26,frontMatter:{sidebar_position:26,title:"A.26. Enum",sidebar_label:"A.26. Enum"},sidebar:"tutorialSidebar",previous:{title:"A.25. Method",permalink:"/basic/method"},next:{title:"A.27. Type Alias & Casting",permalink:"/basic/type-alias-casting"}},u={},m=[{value:"A.26.1. Keyword enum",id:"a261-keyword-enum",level:2},{value:"A.26.2. Naming convention enum",id:"a262-naming-convention-enum",level:2},{value:"A.26.3. Seleksi kondisi enum",id:"a263-seleksi-kondisi-enum",level:2},{value:"A.26.4. Keyword match",id:"a264-keyword-match",level:2},{value:"A.26.5. Enum value \u279c tuple struct-like",id:"a265-enum-value--tuple-struct-like",level:2},{value:"A.26.6. Enum value \u279c struct-like",id:"a266-enum-value--struct-like",level:2},{value:"A.26.7. Aturan pattern matching enum",id:"a267-aturan-pattern-matching-enum",level:2},{value:"A.26.8. Enum module & visibility",id:"a268-enum-module--visibility",level:2},{value:"A.26.9. Generic pada enum",id:"a269-generic-pada-enum",level:2},{value:"A.26.10. Pembahasan lanjutan pattern matching",id:"a2610-pembahasan-lanjutan-pattern-matching",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function o(a){let{components:e,...l}=a;return(0,i.kt)("wrapper",(0,t.Z)({},d,l,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Enum atau ",(0,i.kt)("em",{parentName:"p"},"enumerated type")," adalah sebuah tipe data yang digunakan untuk menampung nilai konstan. Pada chapter ini kita akan mempelajarinya."),(0,i.kt)("p",null,"Enum ada sedikit kemiripan dengan konstanta, bedanya ada pada nilai atau ",(0,i.kt)("em",{parentName:"p"},"underlying value"),"-nya. Jika di konstanta, yang didefinisikan adalah nama beserta value-nya, di enum yang didefinisikan adalah tipe data enum dan enum value. Enum value ini bentuknya seperti variabel tanpa nilai (lebih tepatnya nama dari enum value tersebut adalah nilainya). Lebih jelasnya silakan ikut pembahasan chapter ini."),(0,i.kt)("h2",{id:"a261-keyword-enum"},"A.26.1. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"enum")),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"enum")," digunakan untuk membuat ",(0,i.kt)("em",{parentName:"p"},"enumerated type"),". Cara penulisannya seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum NamaEnum {\n NilaiEnum1,\n Nilai2,\n NilaiEnumKe3,\n // ...\n}\n")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"NamaEnum")," di atas adalah tipe data custom yang didefinisikan bertipe enum. Sedangkan ",(0,i.kt)("inlineCode",{parentName:"p"},"NilaiEnum1"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Nilai2"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"NilaiEnumKe3")," adalah yang disebut dengan enum value. Dengan itu maka ketiga enum values tersebut tipe datanya adalah sama, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"NamaEnum"),"."),(0,i.kt)("p",null,"Mari kita lanjut praktik. Berikut ini adalah definisi konstanta yang menggunakan tipe data string untuk menampung nilai konstan-nya. Lalu di bawahnya ada lagi definisi nilai konstan tetapi menggunakan enum sebagai tipe data yang digunakan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// definisi konstanta\nconst SuperheroSuperman: &str = "superman";\nconst SuperheroOmniMan: &str = "omniman";\nconst SuperheroHomelander: &str = "homelander";\nconst SuperheroHyperion: &str = "hyperion";\n\n// definisi enum\nenum Superhero {\n Superman,\n OmniMan,\n Homelander,\n Hyperion,\n};\n')),(0,i.kt)("p",null,"Di contoh bisa dilihat, ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero")," adalah tipe data enum baru. Dari tipe data tersebut dibuat 4 buah enum values, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"Superman"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"OmniMan"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Homelander"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Hyperion"),"."),(0,i.kt)("p",null,"Pada pembuatan konstanta, tipe data beserta value-nya harus ditentukan di awal. Pada enum, yang perlu didefinisikan adalah tipe data enum-nya (sebagai contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero"),") kemudian diikut dengan enum value yang dituliskan tanpa pengisian nilai."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Definisi variabel dengan isi konstanta:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let value1: &str = SuperheroSuperman;\nlet value2 = SuperheroOmniMan;\n// ...\n"))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Definisi variabel bertipe data enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero"),":"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let value3: Superhero = Superhero::Superman;\nlet value4 = Superhero::OmniMan;\n// ...\n")))),(0,i.kt)("p",null,"Notasi path digunakan dalam penulisan enum value dengan format ",(0,i.kt)("inlineCode",{parentName:"p"},"NamaEnum::EnumValue")),(0,i.kt)("h2",{id:"a262-naming-convention-enum"},"A.26.2. ",(0,i.kt)("em",{parentName:"h2"},"Naming convention")," enum"),(0,i.kt)("p",null,"Sesuai anjuran di ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", upper camel case digunakan dalam penamaan Enum beserta value-nya."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Superhero {\n Superman,\n OmniMan,\n Homelander,\n Hyperion,\n};\n")),(0,i.kt)("h2",{id:"a263-seleksi-kondisi-enum"},"A.26.3. Seleksi kondisi enum"),(0,i.kt)("p",null,"Tipe data enum biasa dipakai pada seleksi kondisi, namun caranya sedikit berbeda. Default-nya keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," tidak bisa digunakan pada tipe data enum."),(0,i.kt)("p",null,"Pada contoh berikut, statement seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"value3")," menghasilkan error:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// seleksi kondisi pada konstanta\nif value1 == SuperheroSuperman {\n println!("hello superman!");\n}\n\n// seleksi kondisi pada enum\nif value3 == Superhero::Superman {\n println!("hello superman!");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(1179).Z,width:"859",height:"378"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Error tersebut muncul karena tipe data enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero")," tidak memiliki trait ",(0,i.kt)("inlineCode",{parentName:"p"},"PartialEq"),". Lebih jelasnya mengenai trait dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),".")),(0,i.kt)("p",null,"Lalu bagaimana cara pengaplikasian seleksi kondisi pada tipe enum? Ada beberapa cara, namun yang paling praktis adalah keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),"."),(0,i.kt)("h2",{id:"a264-keyword-match"},"A.26.4. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"match")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"match")," adalah salah satu keyword untuk operasi seleksi kondisi di Rust. Penerapan keyword ini cukup luas, namun pada chapter ini hanya akan dibahas penerapannya yang relevan dengan topik enum."),(0,i.kt)("p",null,"Mari kita pelajarinya sembari praktik. Silakan buat package baru, lalu definisikan tipe enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Food")," berikut beserta 4 enum value-nya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Food {\n PenyetanTerangBulan,\n PizzaNanas,\n EsKrimIkanMujaer,\n MiGorengKuah,\n}\n")),(0,i.kt)("p",null,"Lalu buat sebuah variabel bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," untuk menampung salah satu nilai enum. Kemudian gunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk menerapkan operasi seleksi kondisi dengan aksi menampilkan sebuah pesan sesuai dengan nilai yang cocok."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let makanan_favorit: Food = Food::PenyetanTerangBulan;\n \n match makanan_favorit {\n Food::PenyetanTerangBulan => {\n println!("your food taste is quite ... unique");\n },\n Food::PizzaNanas => {\n println!("it\'s morally wrong to have pineaple on top of pizza");\n },\n Food::EsKrimIkanMujaer => {\n println!("I don\'t know what to say");\n },\n Food::MiGorengKuah => {\n println!("sometimes people do eat this, but it\'s ok");\n }\n }\n}\n')),(0,i.kt)("p",null,"Di atas bisa dilihat bagaimana cara penggunaan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk penerapan seleksi kondisi pada tipe data enum."),(0,i.kt)("p",null,"Notasi penulisannya kurang lebih seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"}," match variabel_enum {\n TipeEnum::ValueEnum1 => {\n // ...\n },\n TipeEnum::ValueEnum2 => {\n // ...\n },\n \n // ...\n}\n")),(0,i.kt)("p",null,"Kembali ke contoh program, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," dicek nilainya menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Jika nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::PenyetanTerangBulan"),", muncul pesan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'"your food taste is quite ... unique"\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Jika nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::PizzaNanas"),", muncul pesan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'"it is morally wrong to have pineaple on top of pizza"\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Jika nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::EsKrimIkanMujaer"),", muncul pesan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'"I don\'t know what to say"\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Jika nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MiGorengKuah"),", muncul pesan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'"sometimes people do eat this, but it\'s ok"\n')))),(0,i.kt)("p",null,"Jalankan program untuk melihat hasilnya:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(5543).Z,width:"450",height:"82"})),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," ini sebenarnya tidak hanya digunakan untuk seleksi kondisi saja. Di Rust ada yang disebut dengan ",(0,i.kt)("em",{parentName:"p"},"pattern matching"),". Metode ",(0,i.kt)("em",{parentName:"p"},"pattern matching")," ini memahami ",(0,i.kt)("em",{parentName:"p"},"special syntax")," yang kegunaanya lebih luas dibanding hanya sekedar seleksi kondisi biasa."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai ",(0,i.kt)("em",{parentName:"p"},"pattern matching")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"))),(0,i.kt)("h2",{id:"a265-enum-value--tuple-struct-like"},"A.26.5. Enum value \u279c ",(0,i.kt)("em",{parentName:"h2"},"tuple struct"),"-like"),(0,i.kt)("p",null,"Enum value di struct bisa juga didesain seperti ",(0,i.kt)("em",{parentName:"p"},"tuple struct"),". Sebagai contoh, enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Food")," di atas akan kita tambahi dengan satu enum value baru berbentuk ",(0,i.kt)("em",{parentName:"p"},"tuple struct"),"."),(0,i.kt)("p",null,"Silakan tambahkan enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"MakananLainnya")," berikut. Enum ini kita fungsikan untuk mengidentifikasi data makanan lainnya selain dari yang sudah ada di enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Food"),". Notasi penulisan ",(0,i.kt)("em",{parentName:"p"},"tuple struct")," ",(0,i.kt)("inlineCode",{parentName:"p"},"MakananLainnya(String)")," artinya enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"MakananLainnya")," didefinisikan untuk bisa menampung data property dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Food {\n PenyetanTerangBulan,\n PizzaNanas,\n EsKrimIkanMujaer,\n MiGorengKuah,\n MakananLainnya(String), // <---- enum value baru\n}\n")),(0,i.kt)("p",null,"Sekarang ubah isi variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," dengan enum value baru yang sudah dibuat. Syntax ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya(nasi_goreng)")," artinya enum value yang digunakan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya")," dengan isi property didapat dari variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"nasi_goreng"),"."),(0,i.kt)("p",null,"Tambahkan juga ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya")," dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n // enum value MakananLainnya digunakan\n // dengan isi property adalah string "nasi goreng"\n let nasi_goreng = String::from("nasi goreng");\n let makanan_favorit = Food::MakananLainnya(nasi_goreng);\n \n match makanan_favorit {\n Food::PenyetanTerangBulan => {\n println!("your food taste is quite ... unique");\n },\n Food::PizzaNanas => {\n println!("it\'s morally wrong to have pineaple on top of pizza");\n },\n Food::EsKrimIkanMujaer => {\n println!("I don\'t know what to say. this should be illegal");\n },\n Food::MiGorengKuah => {\n println!("sometimes people do eat this, but it\'s ok");\n },\n Food::MakananLainnya(m) => { // <---- seleksi kondisi baru\n println!("do you like {m}? nice taste!");\n }\n }\n}\n')),(0,i.kt)("p",null,"Bisa dilihat ada keunikan dalam penulisan seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya")," dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),". Di situ ada parameter bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"m")," yang parameter tersebut akan berisi data property jika memang ",(0,i.kt)("em",{parentName:"p"},"match")," dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit"),"."),(0,i.kt)("p",null,"Coba jalankan untuk melihat hasilnya:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(8424).Z,width:"462",height:"87"})),(0,i.kt)("p",null,"O iya, jumlah property value enum berbentuk ",(0,i.kt)("em",{parentName:"p"},"tuple struct")," ini tidak terbatas ya. Pada contoh di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MakananLainnya")," hanya memiliki 1 property. Lebih dari satu juga bisa."),(0,i.kt)("h2",{id:"a266-enum-value--struct-like"},"A.26.6. Enum value \u279c ",(0,i.kt)("em",{parentName:"h2"},"struct"),"-like"),(0,i.kt)("p",null,"Enum value bisa juga didesain memiliki property seperti ",(0,i.kt)("em",{parentName:"p"},"struct"),"."),(0,i.kt)("p",null,"Mari kita terapkan pada kode sebelumnya. Tambahkan 1 buah enum value lagi dengan nama ",(0,i.kt)("inlineCode",{parentName:"p"},"MieSetan")," yang ditulis dalam bentuk ",(0,i.kt)("em",{parentName:"p"},"struct"),"-like, dan memiliki 2 buah property."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Food {\n PenyetanTerangBulan,\n PizzaNanas,\n EsKrimIkanMujaer,\n MiGorengKuah,\n MakananLainnya(String),\n MieSetan { level_pedas: i32, pakek_piring: bool }\n}\n")),(0,i.kt)("p",null,"Setelah itu, isi variabel dengan nilai adalah enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MieSetan"),", level pedasnya 5, dan tanpa piring."),(0,i.kt)("p",null,"Tak lupa tambahkan seleksi kondisi untuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MieSetan")," pada blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let makanan_favorit = Food::MieSetan { \n level_pedas: 5,\n pakek_piring: false\n };\n \n match makanan_favorit {\n Food::PenyetanTerangBulan => {\n println!("your food taste is quite ... unique");\n },\n Food::PizzaNanas => {\n println!("it\'s morally wrong to have pineaple on top of pizza");\n },\n Food::EsKrimIkanMujaer => {\n println!("I don\'t know what to say. this should be illegal");\n },\n Food::MiGorengKuah => {\n println!("sometimes people do eat this, but it\'s ok");\n },\n Food::MakananLainnya(m) => {\n println!("do you like {m}? nice taste!");\n },\n Food::MieSetan { level_pedas, pakek_piring } => {\n if level_pedas > 3 {\n println!("mie setan lvl {} is too much!", level_pedas);\n } else {\n println!("mie setan lvl {} is perfect!", level_pedas);\n }\n\n if !pakek_piring {\n println!("how are you going to eat the food without a plate, huh?");\n }\n }\n }\n}\n')),(0,i.kt)("p",null,"Dalam seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MieSetan")," bisa dilihat ada beberapa statement. Kurang lebih jika nilai dari variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::MieSetan")," maka:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Akan memunculkan pesan yang berbeda tergantung level pedasnya"),(0,i.kt)("li",{parentName:"ul"},"Dan jika terdeteksi tidak menggunakan piring, dimunculkan pesan tambahan")),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(6989).Z,width:"448",height:"98"})),(0,i.kt)("h2",{id:"a267-aturan-pattern-matching-enum"},"A.26.7. Aturan ",(0,i.kt)("em",{parentName:"h2"},"pattern matching")," enum"),(0,i.kt)("p",null,"Dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),", semua enum value harus dituliskan. Jika tidak, pasti muncul error. Contohnya bisa dilihat di gambar berikut, beberapa seleksi kondisi enum value di-remark, hasilnya ada error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(2930).Z,width:"1096",height:"309"})),(0,i.kt)("p",null,"Error tersebut sebenarnya bisa diantisipasi dengan menambahkan seleksi kondisi dengan penulisan seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'match makanan_favorit {\n Food::PenyetanTerangBulan => {\n println!("your food taste is quite ... unique");\n },\n Food::PizzaNanas => {\n println!("it\'s morally wrong to have pineaple on top of pizza");\n },\n _ => {\n println!("never heard about that food");\n }\n}\n')),(0,i.kt)("p",null,"Menggunakan blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," di atas, jika nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"makanan_favorit")," adalah selain ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::PenyetanTerangBulan")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Food::PizzaNanas"),", maka pesan ",(0,i.kt)("inlineCode",{parentName:"p"},"never heard about that food")," adalah yang muncul di layar."),(0,i.kt)("p",null,"Selain variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"_")," bisa juga menggunakan nama variabel apapun, misalnya ",(0,i.kt)("inlineCode",{parentName:"p"},"some_var"),". Namun jika variabel tersebut tidak digunakan dalam blok kode, akan muncul warning."),(0,i.kt)("h2",{id:"a268-enum-module--visibility"},"A.26.8. Enum module & visibility"),(0,i.kt)("p",null,"Mari kita coba cek perihal visibility dari enum. Siapkan package baru dengan struktur seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 constants.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("p",null,"Pada file ",(0,i.kt)("inlineCode",{parentName:"p"},"constants.rs"),", tambahkan enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Company")," berikut. Pastikan enum adalah publik dengan menambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," pada deklarasinya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/constants.rs"',title:'"src/constants.rs"'},"pub enum Company {\n Apple,\n Microsoft,\n Google,\n Github\n}\n")),(0,i.kt)("p",null,"Kemudian tambahkan kode berikut di ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod constants;\n\nfn main() {\n let company = constants::Company::Apple;\n \n match company {\n constants::Company::Apple => {\n print!("apple")\n },\n _ => {\n print!("other than apple")\n }\n }\n}\n')),(0,i.kt)("p",null,"Jalankan, hasilnya tidak error, karena ",(0,i.kt)("inlineCode",{parentName:"p"},"Company")," didefinisikan publik."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Enum",src:n(3101).Z,width:"457",height:"85"})),(0,i.kt)("p",null,"Coba lakukan modifikasi dengan menghilangkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," saat definisi enum, hasilnya pasti error."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Pada tipe data enum, keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," cukup ditambahkan pada definisi enum type, tidak perlu ditambahkan satu persatu di tiap enum values.")),(0,i.kt)("h2",{id:"a269-generic-pada-enum"},"A.26.9. Generic pada enum"),(0,i.kt)("p",null,"Pembahasan mengenai generic pada enum ada pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics#a368-generics-enum"},"Generics"),"."),(0,i.kt)("h2",{id:"a2610-pembahasan-lanjutan-pattern-matching"},"A.26.10. Pembahasan lanjutan ",(0,i.kt)("em",{parentName:"h2"},"pattern matching")),(0,i.kt)("p",null,"Pembahasan yang lebih mendetail tentang keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," dan ",(0,i.kt)("em",{parentName:"p"},"pattern matching")," ada pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/enum"},"github.com/novalagung/dasarpemrogramanrust-example/../enum")),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang associated function dan method pada enum")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch06-01-defining-an-enum.html"},"https://doc.rust-lang.org/book/ch06-01-defining-an-enum.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/custom_types/enum.html"},"https://doc.rust-lang.org/rust-by-example/custom_types/enum.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/items/enumerations.html"},"https://doc.rust-lang.org/reference/items/enumerations.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch06-02-match.html"},"https://doc.rust-lang.org/book/ch06-02-match.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.enum.html"},"https://doc.rust-lang.org/std/keyword.enum.html"))))}o.isMDXComponent=!0},1179:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/enum-1-15c91a97fbd3c87f01a4c4c5060b5fd8.png"},5543:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},8424:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},6989:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},2930:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/enum-5-c9da1005565fda09175ddda67f9a16b5.png"},3101:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/4513d9f5.6ce8fc47.js b/assets/js/4513d9f5.6ce8fc47.js deleted file mode 100644 index 1de1d2d6..00000000 --- a/assets/js/4513d9f5.6ce8fc47.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[5729],{3905:(a,e,t)=>{t.d(e,{Zo:()=>p,kt:()=>u});var i=t(7294);function n(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function l(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,i)}return t}function r(a){for(var e=1;e=0||(n[t]=a[t]);return n}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(n[t]=a[t])}return n}var c=i.createContext({}),s=function(a){var e=i.useContext(c),t=e;return a&&(t="function"==typeof a?a(e):r(r({},e),a)),t},p=function(a){var e=s(a.components);return i.createElement(c.Provider,{value:e},a.children)},d={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},o=i.forwardRef((function(a,e){var t=a.components,n=a.mdxType,l=a.originalType,c=a.parentName,p=m(a,["components","mdxType","originalType","parentName"]),o=s(t),u=n,k=o["".concat(c,".").concat(u)]||o[u]||d[u]||l;return t?i.createElement(k,r(r({ref:e},p),{},{components:t})):i.createElement(k,r({ref:e},p))}));function u(a,e){var t=arguments,n=e&&e.mdxType;if("string"==typeof a||n){var l=t.length,r=new Array(l);r[0]=o;var m={};for(var c in e)hasOwnProperty.call(e,c)&&(m[c]=e[c]);m.originalType=a,m.mdxType="string"==typeof a?a:n,r[1]=m;for(var s=2;s{t.r(e),t.d(e,{assets:()=>c,contentTitle:()=>r,default:()=>d,frontMatter:()=>l,metadata:()=>m,toc:()=>s});var i=t(7462),n=(t(7294),t(3905));const l={sidebar_position:43,title:"A.43. Slice Memory Management",sidebar_label:"A.43. Slice Memory Management"},r=void 0,m={unversionedId:"basic/slice-memory-management",id:"basic/slice-memory-management",title:"A.43. Slice Memory Management",description:"Kita telah mempelajari tipe data Array dan Vector, serta sudah beberapa kali menggunakan tipe data string slice (String). 3 tipe data itu memiliki kemiripan, yaitu kesemuanya termasuk dalam kategori tipe data slice.",source:"@site/docs/basic/slice-memory-management.md",sourceDirName:"basic",slug:"/basic/slice-memory-management",permalink:"/basic/slice-memory-management",draft:!1,tags:[],version:"current",sidebarPosition:43,frontMatter:{sidebar_position:43,title:"A.43. Slice Memory Management",sidebar_label:"A.43. Slice Memory Management"},sidebar:"tutorialSidebar",previous:{title:"A.42. Lifetime",permalink:"/basic/lifetime"},next:{title:"A.44. String Literal (&str) vs. String Custom Type",permalink:"/basic/string-slice-vs-string-literal"}},c={},s=[{value:"A.43.1. Konsep slice",id:"a431-konsep-slice",level:2},{value:"A.43.2. Memory management pada slice",id:"a432-memory-management-pada-slice",level:2},{value:"A.43.2. Mutable slice",id:"a432-mutable-slice",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],p={toc:s};function d(a){let{components:e,...l}=a;return(0,n.kt)("wrapper",(0,i.Z)({},p,l,{components:e,mdxType:"MDXLayout"}),(0,n.kt)("p",null,"Kita telah mempelajari tipe data ",(0,n.kt)("a",{parentName:"p",href:"/basic/array"},"Array")," dan ",(0,n.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector"),", serta sudah beberapa kali menggunakan tipe data string slice (",(0,n.kt)("inlineCode",{parentName:"p"},"String"),"). 3 tipe data itu memiliki kemiripan, yaitu kesemuanya termasuk dalam kategori tipe data slice."),(0,n.kt)("blockquote",null,(0,n.kt)("p",{parentName:"blockquote"},"Ciri khas dari tipe data yang termasuk dalam kategori slice adalah jika diakses reference-nya menghasilkan data bertipe ",(0,n.kt)("inlineCode",{parentName:"p"},"&[T]")," dimana ",(0,n.kt)("inlineCode",{parentName:"p"},"T")," adalah tipe data elemen.")),(0,n.kt)("p",null,"Pada chapter ini, kita akan bahas lebih dalam lagi tentang apa itu slice terutama bagian memory management-nya."),(0,n.kt)("blockquote",null,(0,n.kt)("p",{parentName:"blockquote"},"Perbedaan chapter ini dengan chapter ini dengan chapter ",(0,n.kt)("a",{parentName:"p",href:"/basic/slice"},"Slice (Basic)")," adalah di sini fokusnya lebih banyak di memory management.")),(0,n.kt)("h2",{id:"a431-konsep-slice"},"A.43.1. Konsep slice"),(0,n.kt)("p",null,"Slice adalah representasi ",(0,n.kt)("em",{parentName:"p"},"block of memory")," berbentuk pointer dan memiliki size yang dinamis, dengan isi adalah koleksi element data. Slice merupakan reference atau data pinjaman (borrow)."),(0,n.kt)("p",null,"Pada program berikut, beberapa variabel dideklarasikan menggunakan tiga tipe data di atas, kemudian masing-masing data dipinjam kemudian di-print."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-rust"},'let data_arr = [1, 2, 3];\nprintln!("data_arr: {} {:?}", data_arr.len(), data_arr);\nlet slice1 = &data_arr[1..];\nprintln!("slice1 : {} {:?}", slice1.len(), slice1);\nlet slice2 = &data_arr[..2];\nprintln!("slice2 : {} {:?}", slice2.len(), slice2);\n\nlet data_vec = vec![1, 2, 3];\nprintln!("data_vec: {} {:?}", data_vec.len(), data_vec);\nlet slice1 = &data_vec[1..];\nprintln!("slice1 : {} {:?}", slice1.len(), slice1);\nlet slice2 = &data_vec[..2];\nprintln!("slice2 : {} {:?}", slice2.len(), slice2);\n\nlet data_str = String::from("sesuk prei jarene, mosokk");\nprintln!("data_str: {} {:?}", data_str.len(), data_str);\nlet slice1 = &data_str[1..];\nprintln!("slice1 : {} {:?}", slice1.len(), slice1);\nlet slice2 = &data_str[..2];\nprintln!("slice2 : {} {:?}", slice2.len(), slice2);\n')),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Slice memory management",src:t(456).Z,width:"302",height:"205"})),(0,n.kt)("p",null,"Terlihat kemiripan-nya, slice bisa terbentuk dari ketiga jenis data di atas."),(0,n.kt)("p",null,"Penulis ingatkan lagi, bahwa slice adalah tipe data reference yang berarti isi adalah data pinjaman (borrow). Tipe data slice selalu ",(0,n.kt)("inlineCode",{parentName:"p"},"&[T]")," dimana ",(0,n.kt)("inlineCode",{parentName:"p"},"T")," adalah tipe data element."),(0,n.kt)("p",null,"Karena slice adalah data borrow, maka operasi standar borrowing termasuk mutable borrowing bisa dilakukan di slice."),(0,n.kt)("h2",{id:"a432-memory-management-pada-slice"},"A.43.2. Memory management pada slice"),(0,n.kt)("p",null,"Sekarang lanjut ke pembahasan tentang bagaimana data bertipe slice di manage di-memory. Sebagai bahan belajar, kita perlu memilih satu dari 3 tipe data slice yang ada. Bebas sebenarnya mau pilih yang mana. Penulis memilih ",(0,n.kt)("inlineCode",{parentName:"p"},"String")," untuk memulai pembahasan."),(0,n.kt)("p",null,"Silakan perhatikan statement sederhana berikut:"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-rust"},'let data_str = String::from("sesuk preiii");\n')),(0,n.kt)("p",null,"Ada sebuah data ",(0,n.kt)("inlineCode",{parentName:"p"},"String")," dideklarasikan, value-nya adalah ",(0,n.kt)("inlineCode",{parentName:"p"},"sesuk preiii"),", dan owner-nya adalah variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str"),". Data variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str")," disimpan di memory pada 2 tempat, yaitu ",(0,n.kt)("strong",{parentName:"p"},"heap")," dan ",(0,n.kt)("strong",{parentName:"p"},"stack"),"."),(0,n.kt)("p",null,"Selanjutnya, 2 buah slice tercipta hasil operasi slicing pada variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str"),"."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-rust"},'let data_str = String::from("sesuk preiii"); // "sesuk preiii"\nlet slice1 = &data_str[8..]; // "eiii"\nlet slice2 = &data_str[..5]; // "sesuk"\n')),(0,n.kt)("p",null,"Slice adalah data borrow, artinya jika ada beberapa variabel baru dibuat hasil dari operasi slicing, maka isi variabel tersebut merupakan reference yang mengarah ke data sebenarnya. Di heap memory tidak ada perubahan, namun di stack memory ada beberapa data baru. Lebih jelasnya silakan perhatikan ilustrasi berikut."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Slice memory management",src:t(2236).Z,width:"739",height:"713"})),(0,n.kt)("p",null,"Sekarang di stack memory ada 3 buah metadata informasi disimpan, yaitu ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str")," (yang merupakan owner sebenarnya data), dan ",(0,n.kt)("inlineCode",{parentName:"p"},"slice1")," & ",(0,n.kt)("inlineCode",{parentName:"p"},"slice2"),". Sedangkan untuk data-nya sendiri tetap berada di heap memory tanpa ada perubahan."),(0,n.kt)("p",null,"Tiga variabel di atas kesemuanya mengakses reference yang sama, yang membedakan adalah elemen-nya saja. Owner (yaitu ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str"),") bisa mengakses seluruh data, selain itu juga tau informasi kapasitas data. Sedangkan borrower hanya bisa mengakses data yang dia pinjam sesuai dengan operasi slicing-nya. Borrower tidak mengetahui kapasitas data, namun ia tau size dari elemen yang ia pinjam."),(0,n.kt)("h2",{id:"a432-mutable-slice"},"A.43.2. Mutable slice"),(0,n.kt)("p",null,"Bagaimana dengan ",(0,n.kt)("em",{parentName:"p"},"mutability")," pada slice, apa yang terjadi di belakang layar ketika elemen slice nilainya diubah?"),(0,n.kt)("p",null,"Ok, mari kita bahas dengan contoh. Pada kode berikut ada sebuah data mutable array bertipe ",(0,n.kt)("inlineCode",{parentName:"p"},"[i32; 6]")," dengan owner bernama ",(0,n.kt)("inlineCode",{parentName:"p"},"numbers"),". Dari variabel tersebut, dilakukan operasi mutable borrowing untuk disimpan pada variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"n1"),"."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-rust"},'let mut numbers = [24, 12, 8, 7, 9, 2];\nprintln!("numbers: {:?}", numbers);\nlet n1 = &mut numbers[2..];\nprintln!("n1 : {:?}", n1);\nn1[0] = 3022;\nprintln!("numbers: {:?}", numbers);\n')),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Slice memory management",src:t(743).Z,width:"409",height:"147"})),(0,n.kt)("p",null,"Salah satu elemen ",(0,n.kt)("inlineCode",{parentName:"p"},"n1")," diubah nilainya, maka efeknya juga berpengaruh pada owner. Ketika variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"numbers")," di-print, element indeks ",(0,n.kt)("inlineCode",{parentName:"p"},"2")," berubah nilainya dari yang sebelumnya ",(0,n.kt)("inlineCode",{parentName:"p"},"8")," sekarang ",(0,n.kt)("inlineCode",{parentName:"p"},"3022"),"."),(0,n.kt)("p",null,"Visualisasi memory management-nya kurang lebih seperti ini:"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Slice memory management",src:t(7609).Z,width:"718",height:"1088"})),(0,n.kt)("p",null,"Variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"numbers")," dan ",(0,n.kt)("inlineCode",{parentName:"p"},"n1")," sama-sama mengarah ke reference yang sama. Itulah kenapa perubahan data pada peminjam membawa efek ke owner."),(0,n.kt)("hr",null),(0,n.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,n.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,n.kt)("pre",null,(0,n.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/slice_memory_management"},"github.com/novalagung/dasarpemrogramanrust-example/../slice_memory_management")),(0,n.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/array"},"Array")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/slice"},"Slice (Basic)")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/vector"},"Vector")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/basic-memory-management"},"Basic Memory Management")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/tipe-data-custom-type-string-slice"},"Tipe Data \u279c String Custom Type")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type"))),(0,n.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-03-slices.html"},"https://doc.rust-lang.org/book/ch04-03-slices.html")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"https://users.rust-lang.org/t/why-rust-slice-has-not-ownership/27356"},"https://users.rust-lang.org/t/why-rust-slice-has-not-ownership/27356"))))}d.isMDXComponent=!0},456:(a,e,t)=>{t.d(e,{Z:()=>i});const i=""},2236:(a,e,t)=>{t.d(e,{Z:()=>i});const i=t.p+"assets/images/slice-memory-management-2-97ee1868013eb53b825e4c59489aa236.png"},743:(a,e,t)=>{t.d(e,{Z:()=>i});const i=""},7609:(a,e,t)=>{t.d(e,{Z:()=>i});const i=t.p+"assets/images/slice-memory-management-4-25a4c7b54f07cac6f773e2ff178a6d32.png"}}]); \ No newline at end of file diff --git a/assets/js/4513d9f5.f78b137f.js b/assets/js/4513d9f5.f78b137f.js new file mode 100644 index 00000000..dd98aa96 --- /dev/null +++ b/assets/js/4513d9f5.f78b137f.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[5729],{3905:(a,e,t)=>{t.d(e,{Zo:()=>p,kt:()=>u});var i=t(7294);function n(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function l(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,i)}return t}function r(a){for(var e=1;e=0||(n[t]=a[t]);return n}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(n[t]=a[t])}return n}var c=i.createContext({}),s=function(a){var e=i.useContext(c),t=e;return a&&(t="function"==typeof a?a(e):r(r({},e),a)),t},p=function(a){var e=s(a.components);return i.createElement(c.Provider,{value:e},a.children)},d={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},o=i.forwardRef((function(a,e){var t=a.components,n=a.mdxType,l=a.originalType,c=a.parentName,p=m(a,["components","mdxType","originalType","parentName"]),o=s(t),u=n,k=o["".concat(c,".").concat(u)]||o[u]||d[u]||l;return t?i.createElement(k,r(r({ref:e},p),{},{components:t})):i.createElement(k,r({ref:e},p))}));function u(a,e){var t=arguments,n=e&&e.mdxType;if("string"==typeof a||n){var l=t.length,r=new Array(l);r[0]=o;var m={};for(var c in e)hasOwnProperty.call(e,c)&&(m[c]=e[c]);m.originalType=a,m.mdxType="string"==typeof a?a:n,r[1]=m;for(var s=2;s{t.r(e),t.d(e,{assets:()=>c,contentTitle:()=>r,default:()=>d,frontMatter:()=>l,metadata:()=>m,toc:()=>s});var i=t(7462),n=(t(7294),t(3905));const l={sidebar_position:43,title:"A.43. Slice Memory Management",sidebar_label:"A.43. Slice Memory Management"},r=void 0,m={unversionedId:"basic/slice-memory-management",id:"basic/slice-memory-management",title:"A.43. Slice Memory Management",description:"Kita telah mempelajari tipe data Array dan Vector, serta sudah beberapa kali menggunakan tipe data string slice (String). 3 tipe data itu memiliki kemiripan, yaitu kesemuanya termasuk dalam kategori tipe data slice.",source:"@site/docs/basic/slice-memory-management.md",sourceDirName:"basic",slug:"/basic/slice-memory-management",permalink:"/basic/slice-memory-management",draft:!1,tags:[],version:"current",sidebarPosition:43,frontMatter:{sidebar_position:43,title:"A.43. Slice Memory Management",sidebar_label:"A.43. Slice Memory Management"},sidebar:"tutorialSidebar",previous:{title:"A.42. Lifetime",permalink:"/basic/lifetime"},next:{title:"A.44. String Literal (&str) vs. String Custom Type",permalink:"/basic/string-slice-vs-string-literal"}},c={},s=[{value:"A.43.1. Konsep slice",id:"a431-konsep-slice",level:2},{value:"A.43.2. Memory management pada slice",id:"a432-memory-management-pada-slice",level:2},{value:"A.43.2. Mutable slice",id:"a432-mutable-slice",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],p={toc:s};function d(a){let{components:e,...l}=a;return(0,n.kt)("wrapper",(0,i.Z)({},p,l,{components:e,mdxType:"MDXLayout"}),(0,n.kt)("p",null,"Kita telah mempelajari tipe data ",(0,n.kt)("a",{parentName:"p",href:"/basic/array"},"Array")," dan ",(0,n.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector"),", serta sudah beberapa kali menggunakan tipe data string slice (",(0,n.kt)("inlineCode",{parentName:"p"},"String"),"). 3 tipe data itu memiliki kemiripan, yaitu kesemuanya termasuk dalam kategori tipe data slice."),(0,n.kt)("blockquote",null,(0,n.kt)("p",{parentName:"blockquote"},"Ciri khas dari tipe data yang termasuk dalam kategori slice adalah jika diakses reference-nya menghasilkan data bertipe ",(0,n.kt)("inlineCode",{parentName:"p"},"&[T]")," yang mana ",(0,n.kt)("inlineCode",{parentName:"p"},"T")," adalah tipe data elemen.")),(0,n.kt)("p",null,"Pada chapter ini, kita akan bahas lebih dalam lagi tentang apa itu slice terutama bagian memory management-nya."),(0,n.kt)("blockquote",null,(0,n.kt)("p",{parentName:"blockquote"},"Perbedaan chapter ini dengan chapter ini dengan chapter ",(0,n.kt)("a",{parentName:"p",href:"/basic/slice"},"Slice (Basic)")," adalah di sini fokusnya lebih banyak di memory management.")),(0,n.kt)("h2",{id:"a431-konsep-slice"},"A.43.1. Konsep slice"),(0,n.kt)("p",null,"Slice adalah representasi ",(0,n.kt)("em",{parentName:"p"},"block of memory")," berbentuk pointer dan memiliki size yang dinamis, dengan isi adalah koleksi element data. Slice merupakan reference atau data pinjaman (borrow)."),(0,n.kt)("p",null,"Pada program berikut, beberapa variabel dideklarasikan menggunakan tiga tipe data di atas, kemudian masing-masing data dipinjam kemudian di-print."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-rust"},'let data_arr = [1, 2, 3];\nprintln!("data_arr: {} {:?}", data_arr.len(), data_arr);\nlet slice1 = &data_arr[1..];\nprintln!("slice1 : {} {:?}", slice1.len(), slice1);\nlet slice2 = &data_arr[..2];\nprintln!("slice2 : {} {:?}", slice2.len(), slice2);\n\nlet data_vec = vec![1, 2, 3];\nprintln!("data_vec: {} {:?}", data_vec.len(), data_vec);\nlet slice1 = &data_vec[1..];\nprintln!("slice1 : {} {:?}", slice1.len(), slice1);\nlet slice2 = &data_vec[..2];\nprintln!("slice2 : {} {:?}", slice2.len(), slice2);\n\nlet data_str = String::from("sesuk prei jarene, mosokk");\nprintln!("data_str: {} {:?}", data_str.len(), data_str);\nlet slice1 = &data_str[1..];\nprintln!("slice1 : {} {:?}", slice1.len(), slice1);\nlet slice2 = &data_str[..2];\nprintln!("slice2 : {} {:?}", slice2.len(), slice2);\n')),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Slice memory management",src:t(456).Z,width:"302",height:"205"})),(0,n.kt)("p",null,"Terlihat kemiripan-nya, slice bisa terbentuk dari ketiga jenis data di atas."),(0,n.kt)("p",null,"Penulis ingatkan lagi, bahwa slice adalah tipe data reference yang berarti isi adalah data pinjaman (borrow). Tipe data slice selalu ",(0,n.kt)("inlineCode",{parentName:"p"},"&[T]")," yang mana ",(0,n.kt)("inlineCode",{parentName:"p"},"T")," adalah tipe data element."),(0,n.kt)("p",null,"Karena slice adalah data borrow, maka operasi standar borrowing termasuk mutable borrowing bisa dilakukan di slice."),(0,n.kt)("h2",{id:"a432-memory-management-pada-slice"},"A.43.2. Memory management pada slice"),(0,n.kt)("p",null,"Sekarang lanjut ke pembahasan tentang bagaimana data bertipe slice di-manage di memory. Sebagai bahan belajar, kita perlu memilih satu dari 3 tipe data slice yang ada. Bebas sebenarnya mau pilih yang mana. Penulis memilih ",(0,n.kt)("inlineCode",{parentName:"p"},"String")," untuk memulai pembahasan."),(0,n.kt)("p",null,"Silakan perhatikan statement sederhana berikut:"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-rust"},'let data_str = String::from("sesuk preiii");\n')),(0,n.kt)("p",null,"Ada sebuah data ",(0,n.kt)("inlineCode",{parentName:"p"},"String")," dideklarasikan, value-nya adalah ",(0,n.kt)("inlineCode",{parentName:"p"},"sesuk preiii"),", dan owner-nya adalah variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str"),". Data variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str")," disimpan di memory pada 2 tempat, yaitu ",(0,n.kt)("strong",{parentName:"p"},"heap")," dan ",(0,n.kt)("strong",{parentName:"p"},"stack"),"."),(0,n.kt)("p",null,"Selanjutnya, 2 buah slice tercipta hasil operasi slicing pada variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str"),"."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-rust"},'let data_str = String::from("sesuk preiii"); // "sesuk preiii"\nlet slice1 = &data_str[8..]; // "eiii"\nlet slice2 = &data_str[..5]; // "sesuk"\n')),(0,n.kt)("p",null,"Slice adalah data borrow, artinya jika ada beberapa variabel baru dibuat hasil dari operasi slicing, maka isi variabel tersebut merupakan reference yang mengarah ke data sebenarnya. Di heap memory tidak ada perubahan, namun di stack memory ada beberapa data baru. Lebih jelasnya silakan perhatikan ilustrasi berikut."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Slice memory management",src:t(2236).Z,width:"739",height:"713"})),(0,n.kt)("p",null,"Sekarang di stack memory ada 3 buah metadata informasi disimpan, yaitu ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str")," (yang merupakan owner sebenarnya data), dan ",(0,n.kt)("inlineCode",{parentName:"p"},"slice1")," & ",(0,n.kt)("inlineCode",{parentName:"p"},"slice2"),". Sedangkan untuk data-nya sendiri tetap berada di heap memory tanpa ada perubahan."),(0,n.kt)("p",null,"Tiga variabel di atas kesemuanya mengakses reference yang sama, yang membedakan adalah elemen-nya saja. Owner (yaitu ",(0,n.kt)("inlineCode",{parentName:"p"},"data_str"),") bisa mengakses seluruh data, selain itu juga tau informasi kapasitas data. Sedangkan borrower hanya bisa mengakses data yang dia pinjam sesuai dengan operasi slicing-nya. Borrower tidak mengetahui kapasitas data, namun ia tau size dari elemen yang ia pinjam."),(0,n.kt)("h2",{id:"a432-mutable-slice"},"A.43.2. Mutable slice"),(0,n.kt)("p",null,"Bagaimana dengan ",(0,n.kt)("em",{parentName:"p"},"mutability")," pada slice, apa yang terjadi di belakang layar ketika elemen slice nilainya diubah?"),(0,n.kt)("p",null,"Ok, mari kita bahas dengan contoh. Pada kode berikut ada sebuah data mutable array bertipe ",(0,n.kt)("inlineCode",{parentName:"p"},"[i32; 6]")," dengan owner bernama ",(0,n.kt)("inlineCode",{parentName:"p"},"numbers"),". Dari variabel tersebut, dilakukan operasi mutable borrowing untuk disimpan pada variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"n1"),"."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-rust"},'let mut numbers = [24, 12, 8, 7, 9, 2];\nprintln!("numbers: {:?}", numbers);\nlet n1 = &mut numbers[2..];\nprintln!("n1 : {:?}", n1);\nn1[0] = 3022;\nprintln!("numbers: {:?}", numbers);\n')),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Slice memory management",src:t(743).Z,width:"409",height:"147"})),(0,n.kt)("p",null,"Salah satu elemen ",(0,n.kt)("inlineCode",{parentName:"p"},"n1")," diubah nilainya, maka efeknya juga berpengaruh pada owner. Ketika variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"numbers")," di-print, element indeks ",(0,n.kt)("inlineCode",{parentName:"p"},"2")," berubah nilainya dari yang sebelumnya ",(0,n.kt)("inlineCode",{parentName:"p"},"8")," sekarang ",(0,n.kt)("inlineCode",{parentName:"p"},"3022"),"."),(0,n.kt)("p",null,"Visualisasi memory management-nya kurang lebih seperti ini:"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Slice memory management",src:t(7609).Z,width:"718",height:"1088"})),(0,n.kt)("p",null,"Variabel ",(0,n.kt)("inlineCode",{parentName:"p"},"numbers")," dan ",(0,n.kt)("inlineCode",{parentName:"p"},"n1")," sama-sama mengarah ke reference yang sama. Itulah kenapa perubahan data pada peminjam membawa efek ke owner."),(0,n.kt)("hr",null),(0,n.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,n.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,n.kt)("pre",null,(0,n.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/slice_memory_management"},"github.com/novalagung/dasarpemrogramanrust-example/../slice_memory_management")),(0,n.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/array"},"Array")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/slice"},"Slice (Basic)")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/vector"},"Vector")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/basic-memory-management"},"Basic Memory Management")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/tipe-data-custom-type-string-slice"},"Tipe Data \u279c String Custom Type")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type"))),(0,n.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-03-slices.html"},"https://doc.rust-lang.org/book/ch04-03-slices.html")),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("a",{parentName:"li",href:"https://users.rust-lang.org/t/why-rust-slice-has-not-ownership/27356"},"https://users.rust-lang.org/t/why-rust-slice-has-not-ownership/27356"))))}d.isMDXComponent=!0},456:(a,e,t)=>{t.d(e,{Z:()=>i});const i=""},2236:(a,e,t)=>{t.d(e,{Z:()=>i});const i=t.p+"assets/images/slice-memory-management-2-97ee1868013eb53b825e4c59489aa236.png"},743:(a,e,t)=>{t.d(e,{Z:()=>i});const i=""},7609:(a,e,t)=>{t.d(e,{Z:()=>i});const i=t.p+"assets/images/slice-memory-management-4-25a4c7b54f07cac6f773e2ff178a6d32.png"}}]); \ No newline at end of file diff --git a/assets/js/475d163e.4cbfe654.js b/assets/js/475d163e.60520781.js similarity index 97% rename from assets/js/475d163e.4cbfe654.js rename to assets/js/475d163e.60520781.js index 076312f9..82793d6c 100644 --- a/assets/js/475d163e.4cbfe654.js +++ b/assets/js/475d163e.60520781.js @@ -1 +1 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6403],{3905:(a,e,n)=>{n.d(e,{Zo:()=>u,kt:()=>m});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function r(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var d=t.createContext({}),p=function(a){var e=t.useContext(d),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},u=function(a){var e=p(a.components);return t.createElement(d.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},o=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,l=a.originalType,d=a.parentName,u=A(a,["components","mdxType","originalType","parentName"]),o=p(n),m=i,c=o["".concat(d,".").concat(m)]||o[m]||k[m]||l;return n?t.createElement(c,r(r({ref:e},u),{},{components:n})):t.createElement(c,r({ref:e},u))}));function m(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var l=n.length,r=new Array(l);r[0]=o;var A={};for(var d in e)hasOwnProperty.call(e,d)&&(A[d]=e[d]);A.originalType=a,A.mdxType="string"==typeof a?a:i,r[1]=A;for(var p=2;p{n.r(e),n.d(e,{assets:()=>d,contentTitle:()=>r,default:()=>k,frontMatter:()=>l,metadata:()=>A,toc:()=>p});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:10,title:"A.10. Perulangan \u279c while",sidebar_label:"A.10. Perulangan \u279c while"},r=void 0,A={unversionedId:"basic/perulangan-while",id:"basic/perulangan-while",title:"A.10. Perulangan \u279c while",description:"Di Rust ada beberapa cara untuk melakukan perulangan, salah satunya adalah dengan menggunakan keyword while, dan pada chapter ini kita akan mempelajarinya.",source:"@site/docs/basic/perulangan-while.md",sourceDirName:"basic",slug:"/basic/perulangan-while",permalink:"/basic/perulangan-while",draft:!1,tags:[],version:"current",sidebarPosition:10,frontMatter:{sidebar_position:10,title:"A.10. Perulangan \u279c while",sidebar_label:"A.10. Perulangan \u279c while"},sidebar:"tutorialSidebar",previous:{title:"A.9. Seleksi Kondisi \u279c if, else if, else",permalink:"/basic/seleksi-kondisi-if"},next:{title:"A.11. Perulangan \u279c loop, break, continue, label",permalink:"/basic/perulangan-loop-break-continue-label"}},d={},p=[{value:"A.10.1. Keyword while",id:"a101-keyword-while",level:2},{value:"A.10.2. Nested while",id:"a102-nested-while",level:2},{value:"\u25c9 Macro print",id:"-macro-print",level:3},{value:"A.10.3. Keyword while let",id:"a103-keyword-while-let",level:2},{value:"A.10.4. Menambahkan delay dalam perulangan",id:"a104-menambahkan-delay-dalam-perulangan",level:2},{value:"\u25c9 Keyword use dan module dalam Rust",id:"-keyword-use-dan-module-dalam-rust",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:p};function k(a){let{components:e,...l}=a;return(0,i.kt)("wrapper",(0,t.Z)({},u,l,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Di Rust ada beberapa cara untuk melakukan perulangan, salah satunya adalah dengan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"while"),", dan pada chapter ini kita akan mempelajarinya."),(0,i.kt)("p",null,"Perulangan sendiri merupakan teknik untuk mengulang-ulang eksekusi blok kode dengan kondisi tertentu. Perulangan akan terus menerus dieksekusi ketika kondisi perulangan nilainya ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),", dan hanya akan berhenti kita nilainya ",(0,i.kt)("inlineCode",{parentName:"p"},"false"),"."),(0,i.kt)("h2",{id:"a101-keyword-while"},"A.10.1. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"while")),(0,i.kt)("p",null,"Perulangan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," mengharuskan kondisi perulangan dituliskan di awal dengan notasi penulisan sebagai berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"while kondisi {\n \n}\n")),(0,i.kt)("p",null,"Contoh berikut adalah penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," untuk operasi perulangan yang isinya menampilkan angka ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," dengan kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," dibawah ",(0,i.kt)("inlineCode",{parentName:"p"},"max"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 5;\n\nwhile i < max {\n println!("nilai: {i}");\n i += 1;\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"keyword while",src:n(4735).Z,width:"463",height:"163"})),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," pada contoh di atas menjadi penentu kapan perulangan berhenti. Di dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," (yang dimana akan dieksekusi setiap kondisi menghasilkan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),"), nilai variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," di-increment, membuat variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," nilainya selalu bertambah 1 setiap kali perulangan. Perulangan akan berhenti ketika nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," sudah tidak dibawah ",(0,i.kt)("inlineCode",{parentName:"p"},"max")," lagi."),(0,i.kt)("h2",{id:"a102-nested-while"},"A.10.2. Nested ",(0,i.kt)("inlineCode",{parentName:"h2"},"while")),(0,i.kt)("p",null,"Penerapan nested while (atau ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," di dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"while"),") cukup mudah, tulis saja statement ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," di dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," sesuai kebutuhan. Contoh bisa dilihat pada kode berikut, yaitu penerapan teknik nested while untuk print karakter ",(0,i.kt)("inlineCode",{parentName:"p"},"*")," membuat bentuk segitiga."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 5;\n\nwhile i < max {\n let mut j = 0;\n let max_inner = i;\n\n while j <= max_inner {\n print!("* ");\n j += 1;\n }\n \n println!();\n i += 1;\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"nested while",src:n(8423).Z,width:"448",height:"172"})),(0,i.kt)("h3",{id:"-macro-print"},"\u25c9 Macro ",(0,i.kt)("inlineCode",{parentName:"h3"},"print")),(0,i.kt)("p",null,"Macro ",(0,i.kt)("inlineCode",{parentName:"p"},"print")," kegunaannya mirip dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),", hanya saja tanpa ",(0,i.kt)("em",{parentName:"p"},"newline")," atau baris baru. Jadi semua string yang di-print menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"print")," akan muncul menyamping ke kanan dan tidak di baris baru. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'print!("* ");\nprint!("* ");\nprint!("* ");\nprint!("* ");\n')),(0,i.kt)("p",null,"Akan menghasilkan output:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash"},"* * * * \n")),(0,i.kt)("h2",{id:"a103-keyword-while-let"},"A.10.3. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"while let")),(0,i.kt)("p",null,"Kita akan bahas keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"while let")," pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"),"."),(0,i.kt)("h2",{id:"a104-menambahkan-delay-dalam-perulangan"},"A.10.4. Menambahkan delay dalam perulangan"),(0,i.kt)("p",null,"Eksekusi statement perulangan pada program sangatlah cepat. Bisa jadi dalam 1 detik ada ribuan (atau bahkan jauh lebih banyak) statement dalam perulangan yang dieksekusi. Di-tiap eksekusinya, delay bisa saja ditambahkan, misalnya 1 detik, dan pada contoh di bawah ini kita akan coba mengaplikasikannya."),(0,i.kt)("p",null,"Silakan tulis kode program berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use std::thread::sleep;\nuse std::time::Duration;\n\nfn main() {\n let mut i = 0;\n let max = 5;\n\n while i < max {\n println!("nilai: {i}");\n i += 1;\n\n sleep(Duration::from_secs(1));\n }\n}\n')),(0,i.kt)("p",null,"Ada beberapa keyword baru dipergunakan pada source code di atas. Untuk sekarang coba jalankan terlebih dahulu program yang baru dibuat, kemudian stop. Text ",(0,i.kt)("inlineCode",{parentName:"p"},"nilai: {i}")," akan muncul di console setiap satu detik, dan hanya akan berhenti ketika program di stop."),(0,i.kt)("p",null,"Selanjutnya kita akan beberapa sintaks baru di atas."),(0,i.kt)("h3",{id:"-keyword-use-dan-module-dalam-rust"},"\u25c9 Keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"use")," dan module dalam Rust"),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," memiliki banyak kegunaan. Pada contoh ini ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," difungsikan untuk ",(0,i.kt)("em",{parentName:"p"},"import module"),", yang di bahasa Rust dikenal dengan istilah ",(0,i.kt)("strong",{parentName:"p"},"import paths"),"."),(0,i.kt)("p",null,"Notasi penulisan ",(0,i.kt)("em",{parentName:"p"},"module path")," di Rust dinotasikan dengan separator ",(0,i.kt)("inlineCode",{parentName:"p"},"::"),", contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"std::thread::sleep")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"use std::thread::sleep")," artinya path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::thread::sleep")," digunakan dalam kode program. ",(0,i.kt)("inlineCode",{parentName:"p"},"sleep")," adalah sebuah fungsi yang gunanya untuk menambahkan jeda dengan durasi sesuai keinginan, contohnya, 1 detik, 30 menit, dst.")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"use std::time::Duration")," artinya path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration")," digunakan dalam kode program. ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration")," merupakan sebuah struct yang isinya banyak hal untuk keperluan yang berhubungan dengan waktu atau ",(0,i.kt)("em",{parentName:"p"},"duration"),"."))),(0,i.kt)("p",null,"Pada contoh di atas, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"sleep")," dipanggil dalam blok kode perulangan agar ada jeda ditiap eksekusinya. Sedangkan durasi jedanya sendiri ditentukan oleh argument ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration::from_secs(1)")," yang artinya durasi ",(0,i.kt)("inlineCode",{parentName:"p"},"1 detik"),". Dengan ini durasi delay adalah 1 detik."),(0,i.kt)("p",null,"Jika ingin jeda durasi yang lebih lama, ubah saja angka ",(0,i.kt)("inlineCode",{parentName:"p"},"1")," pada ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration::from_secs(1)"),". Contoh: ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration::from_secs(5)")," berarti jeda 5 detik."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya perihal path dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/path-item"},"Module System \u279c Pach & Item")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya perihal keyword ",(0,i.kt)("inlineCode",{parentName:"li"},"use")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/use"},"Module System \u279c Use")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya perihal fungsi dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/function"},"Function")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya perihal struct dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/struct"},"Struct")))),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/perulangan_while"},"github.com/novalagung/dasarpemrogramanrust-example/../perulangan_while")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-05-control-flow.html"},"https://doc.rust-lang.org/book/ch03-05-control-flow.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.while.html"},"https://doc.rust-lang.org/std/keyword.while.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.use.html"},"https://doc.rust-lang.org/std/keyword.use.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/while.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/while.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/stable/std/time/struct.Duration.html"},"https://doc.rust-lang.org/stable/std/time/struct.Duration.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/expressions/while-expr.html"},"https://doc.rust-lang.org/reference/expressions/while-expr.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/paths.html"},"https://doc.rust-lang.org/reference/paths.html"))))}k.isMDXComponent=!0},4735:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},8423:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6403],{3905:(a,e,n)=>{n.d(e,{Zo:()=>u,kt:()=>m});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function r(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var d=t.createContext({}),p=function(a){var e=t.useContext(d),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},u=function(a){var e=p(a.components);return t.createElement(d.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},o=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,l=a.originalType,d=a.parentName,u=A(a,["components","mdxType","originalType","parentName"]),o=p(n),m=i,c=o["".concat(d,".").concat(m)]||o[m]||k[m]||l;return n?t.createElement(c,r(r({ref:e},u),{},{components:n})):t.createElement(c,r({ref:e},u))}));function m(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var l=n.length,r=new Array(l);r[0]=o;var A={};for(var d in e)hasOwnProperty.call(e,d)&&(A[d]=e[d]);A.originalType=a,A.mdxType="string"==typeof a?a:i,r[1]=A;for(var p=2;p{n.r(e),n.d(e,{assets:()=>d,contentTitle:()=>r,default:()=>k,frontMatter:()=>l,metadata:()=>A,toc:()=>p});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:10,title:"A.10. Perulangan \u279c while",sidebar_label:"A.10. Perulangan \u279c while"},r=void 0,A={unversionedId:"basic/perulangan-while",id:"basic/perulangan-while",title:"A.10. Perulangan \u279c while",description:"Di Rust ada beberapa cara untuk melakukan perulangan, salah satunya adalah dengan menggunakan keyword while, dan pada chapter ini kita akan mempelajarinya.",source:"@site/docs/basic/perulangan-while.md",sourceDirName:"basic",slug:"/basic/perulangan-while",permalink:"/basic/perulangan-while",draft:!1,tags:[],version:"current",sidebarPosition:10,frontMatter:{sidebar_position:10,title:"A.10. Perulangan \u279c while",sidebar_label:"A.10. Perulangan \u279c while"},sidebar:"tutorialSidebar",previous:{title:"A.9. Seleksi Kondisi \u279c if, else if, else",permalink:"/basic/seleksi-kondisi-if"},next:{title:"A.11. Perulangan \u279c loop, break, continue, label",permalink:"/basic/perulangan-loop-break-continue-label"}},d={},p=[{value:"A.10.1. Keyword while",id:"a101-keyword-while",level:2},{value:"A.10.2. Nested while",id:"a102-nested-while",level:2},{value:"\u25c9 Macro print",id:"-macro-print",level:3},{value:"A.10.3. Keyword while let",id:"a103-keyword-while-let",level:2},{value:"A.10.4. Menambahkan delay dalam perulangan",id:"a104-menambahkan-delay-dalam-perulangan",level:2},{value:"\u25c9 Keyword use dan module dalam Rust",id:"-keyword-use-dan-module-dalam-rust",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:p};function k(a){let{components:e,...l}=a;return(0,i.kt)("wrapper",(0,t.Z)({},u,l,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Di Rust ada beberapa cara untuk melakukan perulangan, salah satunya adalah dengan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"while"),", dan pada chapter ini kita akan mempelajarinya."),(0,i.kt)("p",null,"Perulangan sendiri merupakan teknik untuk mengulang-ulang eksekusi blok kode dengan kondisi tertentu. Perulangan akan terus menerus dieksekusi ketika kondisi perulangan nilainya ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),", dan hanya akan berhenti kita nilainya ",(0,i.kt)("inlineCode",{parentName:"p"},"false"),"."),(0,i.kt)("h2",{id:"a101-keyword-while"},"A.10.1. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"while")),(0,i.kt)("p",null,"Perulangan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," mengharuskan kondisi perulangan dituliskan di awal dengan notasi penulisan sebagai berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"while kondisi {\n \n}\n")),(0,i.kt)("p",null,"Contoh berikut adalah penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," untuk operasi perulangan yang isinya menampilkan angka ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," dengan kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," di bawah ",(0,i.kt)("inlineCode",{parentName:"p"},"max"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 5;\n\nwhile i < max {\n println!("nilai: {i}");\n i += 1;\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"keyword while",src:n(4735).Z,width:"463",height:"163"})),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," pada contoh di atas menjadi penentu kapan perulangan berhenti. Di dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," (yang mana akan dieksekusi setiap kondisi menghasilkan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),"), nilai variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," di-increment, membuat variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," nilainya selalu bertambah 1 setiap kali perulangan. Perulangan akan berhenti ketika nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," sudah tidak di bawah ",(0,i.kt)("inlineCode",{parentName:"p"},"max")," lagi."),(0,i.kt)("h2",{id:"a102-nested-while"},"A.10.2. Nested ",(0,i.kt)("inlineCode",{parentName:"h2"},"while")),(0,i.kt)("p",null,"Penerapan nested while (atau ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," di dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"while"),") cukup mudah, tulis saja statement ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," di dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," sesuai kebutuhan. Contoh bisa dilihat pada kode berikut, yaitu penerapan teknik nested while untuk print karakter ",(0,i.kt)("inlineCode",{parentName:"p"},"*")," membuat bentuk segitiga."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut i = 0;\nlet max = 5;\n\nwhile i < max {\n let mut j = 0;\n let max_inner = i;\n\n while j <= max_inner {\n print!("* ");\n j += 1;\n }\n \n println!();\n i += 1;\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"nested while",src:n(8423).Z,width:"448",height:"172"})),(0,i.kt)("h3",{id:"-macro-print"},"\u25c9 Macro ",(0,i.kt)("inlineCode",{parentName:"h3"},"print")),(0,i.kt)("p",null,"Macro ",(0,i.kt)("inlineCode",{parentName:"p"},"print")," kegunaannya mirip dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),", hanya saja tanpa ",(0,i.kt)("em",{parentName:"p"},"newline")," atau baris baru. Jadi semua string yang di-print menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"print")," akan muncul menyamping ke kanan dan tidak di baris baru. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'print!("* ");\nprint!("* ");\nprint!("* ");\nprint!("* ");\n')),(0,i.kt)("p",null,"Akan menghasilkan output:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash"},"* * * * \n")),(0,i.kt)("h2",{id:"a103-keyword-while-let"},"A.10.3. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"while let")),(0,i.kt)("p",null,"Kita akan bahas keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"while let")," pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"),"."),(0,i.kt)("h2",{id:"a104-menambahkan-delay-dalam-perulangan"},"A.10.4. Menambahkan delay dalam perulangan"),(0,i.kt)("p",null,"Eksekusi statement perulangan pada program sangatlah cepat. Bisa jadi dalam 1 detik ada ribuan (atau bahkan jauh lebih banyak) statement dalam perulangan yang dieksekusi. Di-tiap eksekusinya, delay bisa saja ditambahkan, misalnya 1 detik, dan pada contoh di bawah ini kita akan coba mengaplikasikannya."),(0,i.kt)("p",null,"Silakan tulis kode program berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use std::thread::sleep;\nuse std::time::Duration;\n\nfn main() {\n let mut i = 0;\n let max = 5;\n\n while i < max {\n println!("nilai: {i}");\n i += 1;\n\n sleep(Duration::from_secs(1));\n }\n}\n')),(0,i.kt)("p",null,"Ada beberapa keyword baru dipergunakan pada source code di atas. Untuk sekarang coba jalankan terlebih dahulu program yang baru dibuat, kemudian stop. Text ",(0,i.kt)("inlineCode",{parentName:"p"},"nilai: {i}")," akan muncul di console setiap satu detik, dan hanya akan berhenti ketika program di stop."),(0,i.kt)("p",null,"Selanjutnya kita akan beberapa sintaks baru di atas."),(0,i.kt)("h3",{id:"-keyword-use-dan-module-dalam-rust"},"\u25c9 Keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"use")," dan module dalam Rust"),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," memiliki banyak kegunaan. Pada contoh ini ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," difungsikan untuk ",(0,i.kt)("em",{parentName:"p"},"import module"),", yang di bahasa Rust dikenal dengan istilah ",(0,i.kt)("strong",{parentName:"p"},"import paths"),"."),(0,i.kt)("p",null,"Notasi penulisan ",(0,i.kt)("em",{parentName:"p"},"module path")," di Rust dinotasikan dengan separator ",(0,i.kt)("inlineCode",{parentName:"p"},"::"),", contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"std::thread::sleep")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"use std::thread::sleep")," artinya path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::thread::sleep")," digunakan dalam kode program. ",(0,i.kt)("inlineCode",{parentName:"p"},"sleep")," adalah sebuah fungsi yang gunanya untuk menambahkan jeda dengan durasi sesuai keinginan, contohnya, 1 detik, 30 menit, dst.")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"use std::time::Duration")," artinya path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration")," digunakan dalam kode program. ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration")," merupakan sebuah struct yang isinya banyak hal untuk keperluan yang berhubungan dengan waktu atau ",(0,i.kt)("em",{parentName:"p"},"duration"),"."))),(0,i.kt)("p",null,"Pada contoh di atas, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"sleep")," dipanggil dalam blok kode perulangan agar ada jeda ditiap eksekusinya. Sedangkan durasi jedanya sendiri ditentukan oleh argument ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration::from_secs(1)")," yang artinya durasi ",(0,i.kt)("inlineCode",{parentName:"p"},"1 detik"),". Dengan ini durasi delay adalah 1 detik."),(0,i.kt)("p",null,"Jika ingin jeda durasi yang lebih lama, ubah saja angka ",(0,i.kt)("inlineCode",{parentName:"p"},"1")," pada ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration::from_secs(1)"),". Contoh: ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration::from_secs(5)")," berarti jeda 5 detik."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya perihal path dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/path-item"},"Module System \u279c Pach & Item")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya perihal keyword ",(0,i.kt)("inlineCode",{parentName:"li"},"use")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/use"},"Module System \u279c Use")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya perihal fungsi dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/function"},"Function")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya perihal struct dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/struct"},"Struct")))),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/perulangan_while"},"github.com/novalagung/dasarpemrogramanrust-example/../perulangan_while")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-05-control-flow.html"},"https://doc.rust-lang.org/book/ch03-05-control-flow.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.while.html"},"https://doc.rust-lang.org/std/keyword.while.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.use.html"},"https://doc.rust-lang.org/std/keyword.use.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/while.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/while.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/stable/std/time/struct.Duration.html"},"https://doc.rust-lang.org/stable/std/time/struct.Duration.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/expressions/while-expr.html"},"https://doc.rust-lang.org/reference/expressions/while-expr.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/paths.html"},"https://doc.rust-lang.org/reference/paths.html"))))}k.isMDXComponent=!0},4735:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},8423:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/4c7e7ada.9f072125.js b/assets/js/4c7e7ada.9f072125.js new file mode 100644 index 00000000..bcd46f84 --- /dev/null +++ b/assets/js/4c7e7ada.9f072125.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[2706],{3905:(e,a,n)=>{n.d(a,{Zo:()=>s,kt:()=>k});var i=n(7294);function t(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function r(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);a&&(i=i.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,i)}return n}function l(e){for(var a=1;a=0||(t[n]=e[n]);return t}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(t[n]=e[n])}return t}var u=i.createContext({}),d=function(e){var a=i.useContext(u),n=a;return e&&(n="function"==typeof e?e(a):l(l({},a),e)),n},s=function(e){var a=d(e.components);return i.createElement(u.Provider,{value:a},e.children)},o={inlineCode:"code",wrapper:function(e){var a=e.children;return i.createElement(i.Fragment,{},a)}},p=i.forwardRef((function(e,a){var n=e.components,t=e.mdxType,r=e.originalType,u=e.parentName,s=m(e,["components","mdxType","originalType","parentName"]),p=d(n),k=t,b=p["".concat(u,".").concat(k)]||p[k]||o[k]||r;return n?i.createElement(b,l(l({ref:a},s),{},{components:n})):i.createElement(b,l({ref:a},s))}));function k(e,a){var n=arguments,t=a&&a.mdxType;if("string"==typeof e||t){var r=n.length,l=new Array(r);l[0]=p;var m={};for(var u in a)hasOwnProperty.call(a,u)&&(m[u]=a[u]);m.originalType=e,m.mdxType="string"==typeof e?e:t,l[1]=m;for(var d=2;d{n.r(a),n.d(a,{assets:()=>u,contentTitle:()=>l,default:()=>o,frontMatter:()=>r,metadata:()=>m,toc:()=>d});var i=n(7462),t=(n(7294),n(3905));const r={sidebar_position:20,title:"A.20. Module System \u279c Module",sidebar_label:"A.20. Module System \u279c Module"},l=void 0,m={unversionedId:"basic/module-basic",id:"basic/module-basic",title:"A.20. Module System \u279c Module",description:"Module adalah salah satu chapter yang cukup penting dalam pemrograman Rust. Pada bagian ini kita akan mempelajari dasarnya.",source:"@site/docs/basic/module-basic.md",sourceDirName:"basic",slug:"/basic/module-basic",permalink:"/basic/module-basic",draft:!1,tags:[],version:"current",sidebarPosition:20,frontMatter:{sidebar_position:20,title:"A.20. Module System \u279c Module",sidebar_label:"A.20. Module System \u279c Module"},sidebar:"tutorialSidebar",previous:{title:"A.19. Module System \u279c Package & Crate",permalink:"/basic/package-crate"},next:{title:"A.21. Module System \u279c Inline Module",permalink:"/basic/module-inline"}},u={},d=[{value:"A.20.1. Rust module system",id:"a201-rust-module-system",level:2},{value:"A.20.2. Praktik #1 - nama_module.rs",id:"a202-praktik-1---nama_modulers",level:2},{value:"\u25c9 Summary praktik #1",id:"-summary-praktik-1",level:3},{value:"\u25c9 Naming convention module",id:"-naming-convention-module",level:3},{value:"\u25c9 Module item",id:"-module-item",level:3},{value:"A.20.3. Praktik #2 - nama_module/mod.rs",id:"a203-praktik-2---nama_modulemodrs",level:2},{value:"\u25c9 Summary praktik #2",id:"-summary-praktik-2",level:3},{value:"\u25c9 Konversi string ke numerik",id:"-konversi-string-ke-numerik",level:3},{value:"A.20.4. Submodules",id:"a204-submodules",level:2},{value:"\u25c9 Summary praktik #3",id:"-summary-praktik-3",level:3},{value:"A.20.5. Penerapan keyword use",id:"a205-penerapan-keyword-use",level:2},{value:"A.20.6. Module path attribute",id:"a206-module-path-attribute",level:2},{value:"\u25c9 Summary praktik #4",id:"-summary-praktik-4",level:3},{value:"A.20.7. Pembahasan lanjutan",id:"a207-pembahasan-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],s={toc:d};function o(e){let{components:a,...r}=e;return(0,t.kt)("wrapper",(0,i.Z)({},s,r,{components:a,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Module adalah salah satu chapter yang cukup penting dalam pemrograman Rust. Pada bagian ini kita akan mempelajari dasarnya."),(0,t.kt)("h2",{id:"a201-rust-module-system"},"A.20.1. Rust module system"),(0,t.kt)("p",null,"Setiap bahasa pemrograman memiliki caranya sendiri dalam hal pengelolahan struktur files dan folder dalam project. Project yang isinya ada sangat banyak hal diatur sedemikian rupa menjadi beberapa bagian dan/atau sub-bagian sesuai dengan fungsinya masing-masing."),(0,t.kt)("p",null,"Di Rust, module memiliki hirarki (biasa disebut dengan ",(0,t.kt)("em",{parentName:"p"},"module tree"),") yang ",(0,t.kt)("em",{parentName:"p"},"root"),"/akarnya adalah file entrypoint crate, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," untuk ",(0,t.kt)("em",{parentName:"p"},"binary crate")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"lib.rs")," untuk ",(0,t.kt)("em",{parentName:"p"},"library crate"),". Kedua file ini biasa disebut dengan ",(0,t.kt)("em",{parentName:"p"},"crate root file"),"."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Di sini pembahasan akan fokus pada penerapan module dalam ",(0,t.kt)("em",{parentName:"p"},"binary crate"),". Kita belum masuk ke pembahasan tentang ",(0,t.kt)("em",{parentName:"p"},"library crate"),".")),(0,t.kt)("p",null,"Rust memiliki 2 jenis modul, yaitu ",(0,t.kt)("em",{parentName:"p"},"normal module")," dan ",(0,t.kt)("em",{parentName:"p"},"inline module"),". Pembahasan dimulai dengan normal module terlebih dahulu."),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mod")," digunakan untuk mendefinisikan/mendaftarkan sebuah module. Nama module menjadi path di mana isi module atau ",(0,t.kt)("em",{parentName:"p"},"module item")," harus berada. Sebagai contoh:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"module yang didefinisikan dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number"),", maka item-nya harus berada pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"module yang didefinisikan dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", maka item-nya harus berada pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io/mod.rs"))),(0,t.kt)("p",null,"Pendefinisian nama module sendiri berada pada file entrypoint ",(0,t.kt)("em",{parentName:"p"},"crate")," yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," (atau ",(0,t.kt)("inlineCode",{parentName:"p"},"lib.rs")," untuk library crate). Jadi pendefinisian nama module dan isi/item-nya terpisah."),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Nama module ditulis di ",(0,t.kt)("inlineCode",{parentName:"li"},"main.rs")," (atau ",(0,t.kt)("inlineCode",{parentName:"li"},"lib.rs")," untuk library crate)"),(0,t.kt)("li",{parentName:"ul"},"Item atau isi module ditulis dalam file ",(0,t.kt)("inlineCode",{parentName:"li"},"nama_module.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"li"},"nama_module/mod.rs"))),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Sebenarnya ada beberapa hal lainnya lagi yang masih relevan yang perlu dibahas di-awal, yaitu perihal sub-module. Akan tetapi agar tidak makin bingung, mari kita lanjut ke praktik terlebih dahulu.")),(0,t.kt)("h2",{id:"a202-praktik-1---nama_modulers"},"A.20.2. Praktik #1 - ",(0,t.kt)("inlineCode",{parentName:"h2"},"nama_module.rs")),(0,t.kt)("p",null,"Mari buat program sederhana, yang isinya mencakup pembahasan tentang module. Pada program kecil ini, inputan user ditampung ke sebuah variabel, kemudian ditampilkan ke layar. Proses pembacaan inputan user akan di-split sebagai module."),(0,t.kt)("p",null,"Ok, langsung saja, buat package/project baru dengan nama bebas. Di sini penulis menggunakan nama package ",(0,t.kt)("inlineCode",{parentName:"p"},"module_1"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"cargo new module_1\n")),(0,t.kt)("p",null,"Setelah itu siapkan 1 buah file bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io.rs"),", letakan di dalam folder ",(0,t.kt)("inlineCode",{parentName:"p"},"src")," (1 level dengan file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"). File ini difungsikan sebagai tempat definisi ",(0,t.kt)("em",{parentName:"p"},"module item")," milik sebuah module bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," (io di sini kependekan dari ",(0,t.kt)("em",{parentName:"p"},"input output"),")."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"module_1\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2514\u2500\u2500\u2500 my_io.rs\n")),(0,t.kt)("p",null,"Lanjut, definisikan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," di file ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io.rs"),", isinya kurang lebih adalah kode untuk membaca inputan user lalu mengembalikannya dalam bentuk ",(0,t.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_io.rs"',title:'"src/my_io.rs"'},'pub fn read_entry() -> String {\n let mut message = std::string::String::new();\n let stdin_reader = std::io::stdin(); \n let reader_res = stdin_reader.read_line(&mut message);\n\n if reader_res.is_err() {\n println!("error! {:?}", reader_res.err());\n }\n\n message.trim().to_string()\n}\n')),(0,t.kt)("p",null,"Ok, kita telah menyiapkan satu item milik module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," yaitu sebuah fungsi bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry"),". 1 hal yang sedikit berbeda pada definisi fungsi di atas adalah penambahan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," yang ditulis sebagai prefix definisi fungsi."),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," digunakan untuk menjadikan suatu item menjadi ",(0,t.kt)("strong",{parentName:"p"},"public"),", agar bisa diakses dari luar module."),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," ini berada dalam module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io"),". Jika tidak ada keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," di situ, maka fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," hanya bisa diakses dari dalam ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," saja, tidak bisa diakses dari luar module contohnya seperti dari ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),". Dengan menjadikan ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," sebagai fungsi yang public, maka fungsi tersebut bisa diakses dari ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,t.kt)("p",null,"Isi module sudah siap, selanjutnya lanjut ke pendefinisian modul. Umumnya pada bahasa pemrograman, definisi module adalah ada dalam file di mana isi module berada, namun tidak untuk Rust."),(0,t.kt)("p",null,"Di Rust, definisi sebuah module (sekali lagi bukan item/isinya ya, tapi definisi dari module itu sendiri) dituliskan pada file terpisah, yaitu di file entrypoint crate, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"lib.rs"),"."),(0,t.kt)("p",null,"Lanjut, sekarang buka file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," dan tulis definisi modul ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," menggunakan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"mod my_io"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"// definisi module my_io\nmod my_io;\n\n// fungsi main\nfn main() {\n // ...\n}\n")),(0,t.kt)("p",null,"Sekarang aplikasikan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," milik module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," di fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main")," untuk membaca inputan user."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod my_io;\n\nfn main() {\n println!("enter any number:");\n let message = my_io::read_entry();\n println!("your number: {}", message);\n}\n')),(0,t.kt)("p",null,"Jalankan program, masukan sebuah angka, lalu enter. Lihat hasilnya, program berjalan sesuai harapan."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust module",src:n(5152).Z,width:"762",height:"717"})),(0,t.kt)("p",null,"Cara pengaksesan item dari sebuah module yang kita definisikan sendiri adalah sama seperti pengaksesan item dari crate lainnya, yaitu menggunakan ",(0,t.kt)("em",{parentName:"p"},"path"),". Statement ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io::read_entry()")," artinya item ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," yang merupakan fungsi dalam module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," digunakan."),(0,t.kt)("h3",{id:"-summary-praktik-1"},"\u25c9 Summary praktik #1"),(0,t.kt)("p",null,"Pada praktik pertama ini kita telah belajar penerapan module dengan mengaplikasikan beberapa hal berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Penggunaan normal module dalam binary crate"),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," untuk meng-export atau menjadikan item menjadi public, agar bisa diakses dari luar module"),(0,t.kt)("li",{parentName:"ul"},"Pengaksesan item milik module, yaitu: ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io::read_entry"))),(0,t.kt)("h3",{id:"-naming-convention-module"},"\u25c9 ",(0,t.kt)("em",{parentName:"h3"},"Naming convention")," module"),(0,t.kt)("p",null,"Sesuai anjuran di ",(0,t.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", aturan penulisan nama module adalah menggunakan snake case, contohnya ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io"),"."),(0,t.kt)("h3",{id:"-module-item"},"\u25c9 Module item"),(0,t.kt)("p",null,"Module item adalah apapun yang didefinisikan di dalam sebuah module. Pada contoh praktik ke-1 di atas, module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," memiliki 1 buah item yaitu fungsi bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry"),"."),(0,t.kt)("p",null,"Selain fungsi, module item bisa dalam bentuk lainnya, contohnya: konstanta, submodule, struct, dan lainnya."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang macam-macam item dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/path-item"},"Module System \u279c Path & Item"))),(0,t.kt)("h2",{id:"a203-praktik-2---nama_modulemodrs"},"A.20.3. Praktik #2 - ",(0,t.kt)("inlineCode",{parentName:"h2"},"nama_module/mod.rs")),(0,t.kt)("p",null,"Bagian ini merupakan kelanjutan dari praktik sebelumnya. Program sederhana yang sudah di-buat ditambahi beberapa hal. Data inputan user dikonversi ke bentuk angka untuk kemudian dicek apakah angka tersebut bilangan ganjil atau genap."),(0,t.kt)("p",null,"Fungsi untuk konversi string ke bentuk numerik dan juga untuk pengecekan bilangan ganjil genap, adalah dua buah item milik module bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," yang akan kita definisikan sebentar lagi."),(0,t.kt)("p",null,"Definisi item module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," menggunakan notasi penulisan file ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," (yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io.rs"),"). Pada module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," ini kita akan gunakan notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs")," (menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs"),") untuk menampung definisi item module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"module_1\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 my_number\n \u2502 \u2514\u2500\u2500\u2500 mod.rs\n \u2502\u2500\u2500\u2500 main.rs\n \u2514\u2500\u2500\u2500 my_io.rs\n")),(0,t.kt)("p",null,"Ok, sekarang buat saja folder dan filenya, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs"),". Lalu pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"mod.rs")," tulis 2 buah fungsi berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Fungsi untuk konversi string ke numerik ",(0,t.kt)("inlineCode",{parentName:"p"},"i32")),(0,t.kt)("pre",{parentName:"li"},(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},"pub fn string_to_number(text: String) -> i32 {\n return text.parse::().unwrap();\n}\n"))),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Fungsi untuk pengecekan bilangan ganjil"),(0,t.kt)("pre",{parentName:"li"},(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},"// ...\n\npub fn is_odd_number(number: i32) -> bool {\n number % 2 == 1\n}\n")))),(0,t.kt)("p",null,"Kemudian tulis definisi module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," dalam file entrypoint crate, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"// definisi module my_io\nmod my_io;\n\n// definisi module my_number\nmod my_number;\n\n// fungsi main\nfn main() {\n // ...\n}\n")),(0,t.kt)("p",null,"Terakhir, aplikasikan dua buah fungsi yang sudah didefinisikan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'fn main() {\n println!("enter any number:");\n let message = my_io::read_entry();\n println!("your number: {}", message);\n\n let number = my_number::string_to_number(message);\n let result = my_number::is_odd_number(number);\n println!("is odd number: {}", result);\n}\n')),(0,t.kt)("p",null,"Jalankan program, lihat hasilnya."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust module",src:n(7050).Z,width:"777",height:"872"})),(0,t.kt)("h3",{id:"-summary-praktik-2"},"\u25c9 Summary praktik #2"),(0,t.kt)("p",null,"Hingga praktik ke-2 ini, kita telah belajar penerapan module dengan mengaplikasikan beberapa hal berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Penggunaan normal module dalam binary crate"),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," untuk meng-export atau menjadikan item menjadi public, agar bisa diakses dari luar module"),(0,t.kt)("li",{parentName:"ul"},"Pengaksesan item milik module, yaitu: ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io::read_entry"),", ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::string_to_number"),", dan ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::is_odd_number"))),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Penerapan notasi penulisan isi module ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs")," bisa diterapkan dalam satu package")),(0,t.kt)("h3",{id:"-konversi-string-ke-numerik"},"\u25c9 Konversi string ke numerik"),(0,t.kt)("p",null,"Rust mengenal beberapa jenis konversi antar tipe data. Teknik konversi yang kita praktikkan pada chapter ini menggunakan method ",(0,t.kt)("inlineCode",{parentName:"p"},".parse()"),", cara ini bisa dilakukan untuk konversi tipe data dari custom types ke primitive. Contohnya seperti di atas, dari ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,t.kt)("p",null,"Penggunaannya cukup mudah, akses saja method ",(0,t.kt)("inlineCode",{parentName:"p"},".parse()")," kemudian sisipkan tipe data sebagai parameter generic. Lalu chain dengan method ",(0,t.kt)("inlineCode",{parentName:"p"},"unwrap")," (yang method tersebut merupakan item milik tipe data ",(0,t.kt)("em",{parentName:"p"},"generic result type")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"Result"),")."),(0,t.kt)("p",null,"Contoh lain untuk konversi dari tipe primitif lainnya bisa dilihat berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let data_number = "24".parse::().unwrap();\nlet data_bool = "true".parse::().unwrap();\nlet data_float = "3.14".parse::().unwrap();\n')),(0,t.kt)("p",null,"Method ",(0,t.kt)("inlineCode",{parentName:"p"},"parse")," menghasilkan data bertipe ",(0,t.kt)("em",{parentName:"p"},"generic result type")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"Result"),". Tipe ini memiliki method bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"unwrap")," yang berguna untuk mengambil nilai."),(0,t.kt)("blockquote",null,(0,t.kt)("ul",{parentName:"blockquote"},(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai casting dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/type-alias-casting"},"Type Alias & Casting")," dan ",(0,t.kt)("a",{parentName:"li",href:"#/wip/trait-conversion-from-into"},"Trait \u279c Conversion (From & Into)")),(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai generic dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai result type dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/result-type"},"Tipe Data \u279c Result")))),(0,t.kt)("h2",{id:"a204-submodules"},"A.20.4. Submodules"),(0,t.kt)("p",null,"Sampai bagian ini kita telah belajar tentang module beserta 2 macam cara penerapannya."),(0,t.kt)("p",null,"Sebuah module bisa saja memiliki module di bawahnya (biasa disebut submodule), dan hal ini adalah konsep yang umum dalam bahasa pemrograman. Di Rust, aturan dalam pembuatan submodule masih sama seperti module, perbedaannya adalah tempat di mana submodule didefinisikan. Jika pada root module definisi ada pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"lib.rs"),", maka pada submodule definisi ada pada file di mana ",(0,t.kt)("em",{parentName:"p"},"parent module")," berada."),(0,t.kt)("p",null,"Sebagai contoh jika pada program sebelumnya kita tambahkan module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," yang sudah dibuat, jika ada submodule dengan nama ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion_utility"),", maka definisi module berada di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs")," dan itemnya di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility/mod.rs"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 my_io.rs\n \u2514\u2500\u2500\u2500 my_number\n \u2502\u2500\u2500\u2500 mod.rs // <----- definisi submodule conversion_utility\n \u2514\u2500\u2500\u2500 conversion_utility\n \u2514\u2500\u2500\u2500 mod.rs // <----- definisi item/isi conversion_utility\n")),(0,t.kt)("p",null,"Atau definisi module tetap di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs")," namun itemnya di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility.rs"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 my_io.rs\n \u2514\u2500\u2500\u2500 my_number\n \u2502\u2500\u2500\u2500 mod.rs // <----- definisi submodule conversion_utility\n \u2514\u2500\u2500\u2500 conversion_utility.rs // <----- definisi item/isi conversion_utility\n")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Penerapan notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," biasanya dalam case ketika module tersebut tidak memiliki submodule."),(0,t.kt)("p",{parentName:"blockquote"},"Untuk module yang memiliki submodule, parent module harus menerapkan notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs"),", hal ini karena pendefinisian submodule berada pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"mod.rs")," dalam parent module tersebut.")),(0,t.kt)("p",null,"Lalu bagaimana jika sebuah submodule memiliki submodule yang memiliki submodule ... dst, aturannya tetap sama seperti aturan submodule."),(0,t.kt)("p",null,"Mari lanjut praktik agar tidak bingung. Kita akan modifikasi program sebelumnya. Item ",(0,t.kt)("inlineCode",{parentName:"p"},"string_to_number")," yang sebelumnya adalah item milik ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," kita pindah ke sebuah module baru bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion_utility")," yang merupakan submodule dari ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number"),". Silakan buat file baru ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility/mod.rs"),", kemudian isi dengan fungsi berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/conversion_utility/mod.rs"',title:'"src/my_number/conversion_utility/mod.rs"'},"pub fn string_to_number(text: String) -> i32 {\n return text.parse::().unwrap();\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"string_to_number")," yang sebelumnya ada di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs")," silakan dihapus."),(0,t.kt)("p",null,"Kemudian pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs"),", tambahkan definisi submodule ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion_utility"),". Isi file tersebut kurang lebih menjadi seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},"pub mod conversion_utility;\n\npub fn is_odd_number(number: i32) -> bool {\n number % 2 == 1\n}\n")),(0,t.kt)("p",null,"O iya, karena submodule merupakan sebuah item milik module, maka harus ditambahkan juga keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),", agar submodule bisa diakses dari luar scope-nya. Contoh penerapannya bisa dilihat di atas."),(0,t.kt)("p",null,"Terakhir, pada ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),", ubah pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"string_to_number")," dari ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"let number = my_number::string_to_number(message);\n")),(0,t.kt)("p",null,"... menjadi ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"let number = my_number::conversion_utility::string_to_number(message);\n")),(0,t.kt)("p",null,"Kurang lebih strukturnya menjadi seperti berikut:"),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust submodule",src:n(3749).Z,width:"940",height:"955"})),(0,t.kt)("p",null,"Jalankan program untuk mengetest hasilnya."),(0,t.kt)("h3",{id:"-summary-praktik-3"},"\u25c9 Summary praktik #3"),(0,t.kt)("p",null,"Hingga pembahasan pada praktik submodule, kita telah belajar penerapan module dengan mengaplikasikan beberapa hal berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Penggunaan normal module dalam binary crate"),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan submodule dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/conversion_utility"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/conversion.rs")," yang di-import menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"path")," attribute."),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," pada fungsi agar bisa diakses dari luar module"),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," pada submodule agar bisa diakses dari luar parent module"),(0,t.kt)("li",{parentName:"ul"},"Pengaksesan item milik module, yaitu: ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io::read_entry"),", ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::conversion_utility::string_to_number"),", dan ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::is_odd_number"))),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Penerapan notasi penulisan isi module ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs")," bisa diterapkan dalam satu package")),(0,t.kt)("h2",{id:"a205-penerapan-keyword-use"},"A.20.5. Penerapan keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"use")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"use")," bisa digunakan untuk meng-import module atau item tertentu, dan dengannya pengaksesan item menjadi lebih pendek. Contohnya:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"my_number::conversion_utility::string_to_number(message);\n")),(0,t.kt)("p",null,"... bisa dituliskan menjadi ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"use my_number::conversion_utility::string_to_number;\nstring_to_number(message);\n")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"use")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/use"},"Module System \u279c Use"))),(0,t.kt)("h2",{id:"a206-module-path-attribute"},"A.20.6. Module ",(0,t.kt)("inlineCode",{parentName:"h2"},"path")," attribute"),(0,t.kt)("p",null,"Selain menggunakan dua teknik definisi module item di atas yang fokusnya ada pada penamaan file, ada juga cara lain pendefinisian module item, yaitu dengan memanfaatkan ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute."),(0,t.kt)("p",null,"Mari kita praktikkan, silakan ubah struktur package yang sudah dibuat dari ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 my_io.rs\n \u2514\u2500\u2500\u2500 my_number\n \u2502\u2500\u2500\u2500 mod.rs // <----- definisi submodule conversion_utility\n \u2514\u2500\u2500\u2500 conversion_utility.rs // <----- definisi item/isi conversion_utility\n")),(0,t.kt)("p",null,"... menjadi ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 my_io.rs\n \u2514\u2500\u2500\u2500 my_number\n \u2502\u2500\u2500\u2500 mod.rs // <----- definisi submodule conversion_utility\n \u2514\u2500\u2500\u2500 conversion.rs // <----- definisi item/isi conversion_utility\n")),(0,t.kt)("p",null,"Yang telah kita lakukan adalah me-rename file ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion_utility.rs")," menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion.rs"),". Efeknya akan muncul error dalam pengaksesan module item, karena submodule ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility")," isi/item-nya harus berada pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility/mod.rs"),". Sedangkan file ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion.rs")," tidak memenuhi kriteria tersebut."),(0,t.kt)("p",null,"Sekarang buka isi file ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs"),", lalu ubah statement pendefinisian submodule dari ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},"pub mod conversion_utility;\n")),(0,t.kt)("p",null,"... menjadi ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},'#[path = "conversion.rs"]\npub mod conversion_utility;\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust module",src:n(6827).Z,width:"690",height:"365"})),(0,t.kt)("p",null,"Lalu run, dan program akan jalan normal tanpa error."),(0,t.kt)("p",null,"Statement ",(0,t.kt)("inlineCode",{parentName:"p"},'#[path = "conversion.rs"]')," di atas merupakan contoh penerapan dari Rust path attributes. Dengannya kita bisa menempatkan isi sebuah module pada file yang namanya bebas (pada contoh di atas, file bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion.rs"),")."),(0,t.kt)("h3",{id:"-summary-praktik-4"},"\u25c9 Summary praktik #4"),(0,t.kt)("p",null,"Hingga pembahasan pada praktik module ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute, kita telah belajar penerapan module dengan mengaplikasikan beberapa hal berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Penggunaan normal module dalam binary crate"),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan submodule dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/conversion_utility"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/conversion_utility/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," pada fungsi agar bisa diakses dari luar module"),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," pada submodule agar bisa diakses dari luar parent module"),(0,t.kt)("li",{parentName:"ul"},"Pengaksesan item milik module, yaitu: ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io::read_entry"),", ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::conversion_utility::string_to_number"),", dan ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::is_odd_number"))),(0,t.kt)("h2",{id:"a207-pembahasan-lanjutan"},"A.20.7. Pembahasan lanjutan"),(0,t.kt)("p",null,"Pembahasan topik module dilanjutkan pada beberapa chapter lain. Pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/module-inline"},"Module System \u279c Inline Module")," kita akan bahas secara mendetail mengenai apa itu inline module dan perbedaannya dibanding normal module. Lalu nantinya di cahpter ",(0,t.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy")," akan dibahas secara lengkap mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," dan kontrol privasi lainnya di pemrograman Rust."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/module"},"github.com/novalagung/dasarpemrogramanrust-example/../module")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html"},"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.pub.html"},"https://doc.rust-lang.org/std/keyword.pub.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.mod.html"},"https://doc.rust-lang.org/std/keyword.mod.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/mod.html"},"https://doc.rust-lang.org/rust-by-example/mod.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/69275034/what-is-the-difference-between-use-and-pub-use"},"https://stackoverflow.com/questions/69275034/what-is-the-difference-between-use-and-pub-use"))))}o.isMDXComponent=!0},5152:(e,a,n)=>{n.d(a,{Z:()=>i});const i=n.p+"assets/images/module-1-0eea2521d2399e73e455b90f5c2f97fa.png"},7050:(e,a,n)=>{n.d(a,{Z:()=>i});const i=n.p+"assets/images/module-2-d8f6f3862036da93c4b7920e144a3a3f.png"},3749:(e,a,n)=>{n.d(a,{Z:()=>i});const i=n.p+"assets/images/module-3-b8f34c47b905cadb9001a866ebb80809.png"},6827:(e,a,n)=>{n.d(a,{Z:()=>i});const i=n.p+"assets/images/module-4-e83e7a2e8d7d6d4d11ec121d6dc184d0.png"}}]); \ No newline at end of file diff --git a/assets/js/4c7e7ada.f4643a6f.js b/assets/js/4c7e7ada.f4643a6f.js deleted file mode 100644 index b45a3da2..00000000 --- a/assets/js/4c7e7ada.f4643a6f.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[2706],{3905:(e,a,n)=>{n.d(a,{Zo:()=>s,kt:()=>k});var i=n(7294);function t(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function r(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);a&&(i=i.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,i)}return n}function l(e){for(var a=1;a=0||(t[n]=e[n]);return t}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(t[n]=e[n])}return t}var u=i.createContext({}),d=function(e){var a=i.useContext(u),n=a;return e&&(n="function"==typeof e?e(a):l(l({},a),e)),n},s=function(e){var a=d(e.components);return i.createElement(u.Provider,{value:a},e.children)},o={inlineCode:"code",wrapper:function(e){var a=e.children;return i.createElement(i.Fragment,{},a)}},p=i.forwardRef((function(e,a){var n=e.components,t=e.mdxType,r=e.originalType,u=e.parentName,s=m(e,["components","mdxType","originalType","parentName"]),p=d(n),k=t,b=p["".concat(u,".").concat(k)]||p[k]||o[k]||r;return n?i.createElement(b,l(l({ref:a},s),{},{components:n})):i.createElement(b,l({ref:a},s))}));function k(e,a){var n=arguments,t=a&&a.mdxType;if("string"==typeof e||t){var r=n.length,l=new Array(r);l[0]=p;var m={};for(var u in a)hasOwnProperty.call(a,u)&&(m[u]=a[u]);m.originalType=e,m.mdxType="string"==typeof e?e:t,l[1]=m;for(var d=2;d{n.r(a),n.d(a,{assets:()=>u,contentTitle:()=>l,default:()=>o,frontMatter:()=>r,metadata:()=>m,toc:()=>d});var i=n(7462),t=(n(7294),n(3905));const r={sidebar_position:20,title:"A.20. Module System \u279c Module",sidebar_label:"A.20. Module System \u279c Module"},l=void 0,m={unversionedId:"basic/module-basic",id:"basic/module-basic",title:"A.20. Module System \u279c Module",description:"Module adalah salah satu chapter yang cukup penting dalam pemrograman Rust. Pada bagian ini kita akan mempelajari dasarnya.",source:"@site/docs/basic/module-basic.md",sourceDirName:"basic",slug:"/basic/module-basic",permalink:"/basic/module-basic",draft:!1,tags:[],version:"current",sidebarPosition:20,frontMatter:{sidebar_position:20,title:"A.20. Module System \u279c Module",sidebar_label:"A.20. Module System \u279c Module"},sidebar:"tutorialSidebar",previous:{title:"A.19. Module System \u279c Package & Crate",permalink:"/basic/package-crate"},next:{title:"A.21. Module System \u279c Inline Module",permalink:"/basic/module-inline"}},u={},d=[{value:"A.20.1. Rust module system",id:"a201-rust-module-system",level:2},{value:"A.20.2. Praktik #1 - nama_module.rs",id:"a202-praktik-1---nama_modulers",level:2},{value:"\u25c9 Summary praktik #1",id:"-summary-praktik-1",level:3},{value:"\u25c9 Naming convention module",id:"-naming-convention-module",level:3},{value:"\u25c9 Module item",id:"-module-item",level:3},{value:"A.20.3. Praktik #2 - nama_module/mod.rs",id:"a203-praktik-2---nama_modulemodrs",level:2},{value:"\u25c9 Summary praktik #2",id:"-summary-praktik-2",level:3},{value:"\u25c9 Konversi string ke numerik",id:"-konversi-string-ke-numerik",level:3},{value:"A.20.4. Submodules",id:"a204-submodules",level:2},{value:"\u25c9 Summary praktik #3",id:"-summary-praktik-3",level:3},{value:"A.20.5. Penerapan keyword use",id:"a205-penerapan-keyword-use",level:2},{value:"A.20.6. Module path attribute",id:"a206-module-path-attribute",level:2},{value:"\u25c9 Summary praktik #4",id:"-summary-praktik-4",level:3},{value:"A.20.7. Pembahasan lanjutan",id:"a207-pembahasan-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],s={toc:d};function o(e){let{components:a,...r}=e;return(0,t.kt)("wrapper",(0,i.Z)({},s,r,{components:a,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Module adalah salah satu chapter yang cukup penting dalam pemrograman Rust. Pada bagian ini kita akan mempelajari dasarnya."),(0,t.kt)("h2",{id:"a201-rust-module-system"},"A.20.1. Rust module system"),(0,t.kt)("p",null,"Setiap bahasa pemrograman memiliki caranya sendiri dalam hal pengelolahan struktur files dan folder dalam project. Project yang isinya ada sangat banyak hal diatur sedemikian rupa menjadi beberapa bagian dan/atau sub-bagian sesuai dengan fungsinya masing-masing."),(0,t.kt)("p",null,"Di Rust, module memiliki hirarki (biasa disebut dengan ",(0,t.kt)("em",{parentName:"p"},"module tree"),") yang ",(0,t.kt)("em",{parentName:"p"},"root"),"/akarnya adalah file entrypoint crate, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," untuk ",(0,t.kt)("em",{parentName:"p"},"binary crate")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"lib.rs")," untuk ",(0,t.kt)("em",{parentName:"p"},"library crate"),". Kedua file ini biasa disebut dengan ",(0,t.kt)("em",{parentName:"p"},"crate root file"),"."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Di sini pembahasan akan fokus pada penerapan module dalam ",(0,t.kt)("em",{parentName:"p"},"binary crate"),". Kita belum masuk ke pembahasan tentang ",(0,t.kt)("em",{parentName:"p"},"library crate"),".")),(0,t.kt)("p",null,"Rust memiliki 2 jenis modul, yaitu ",(0,t.kt)("em",{parentName:"p"},"normal module")," dan ",(0,t.kt)("em",{parentName:"p"},"inline module"),". Pembahasan dimulai dengan normal module terlebih dahulu."),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mod")," digunakan untuk mendefinisikan/mendaftarkan sebuah module. Nama module menjadi path dimana isi module atau ",(0,t.kt)("em",{parentName:"p"},"module item")," harus berada. Sebagai contoh:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"module yang didefinisikan dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number"),", maka item-nya harus berada pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"module yang didefinisikan dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", maka item-nya harus berada pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io/mod.rs"))),(0,t.kt)("p",null,"Pendefinisian nama module sendiri berada pada file entrypoint ",(0,t.kt)("em",{parentName:"p"},"crate")," yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," (atau ",(0,t.kt)("inlineCode",{parentName:"p"},"lib.rs")," untuk library crate). Jadi pendefinisian nama module dan isi/item-nya terpisah."),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Nama module ditulis di ",(0,t.kt)("inlineCode",{parentName:"li"},"main.rs")," (atau ",(0,t.kt)("inlineCode",{parentName:"li"},"lib.rs")," untuk library crate)"),(0,t.kt)("li",{parentName:"ul"},"Item atau isi module ditulis dalam file ",(0,t.kt)("inlineCode",{parentName:"li"},"nama_module.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"li"},"nama_module/mod.rs"))),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Sebenarnya ada beberapa hal lainnya lagi yang masih relevan yang perlu dibahas di-awal, yaitu perihal sub-module. Akan tetapi agar tidak makin bingung, mari kita lanjut ke praktik terlebih dahulu.")),(0,t.kt)("h2",{id:"a202-praktik-1---nama_modulers"},"A.20.2. Praktik #1 - ",(0,t.kt)("inlineCode",{parentName:"h2"},"nama_module.rs")),(0,t.kt)("p",null,"Mari buat program sederhana, yang isinya mencakup pembahasan tentang module. Pada program kecil ini, inputan user ditampung ke sebuah variabel, kemudian ditampilkan ke layar. Proses pembacaan inputan user akan di-split sebagai module."),(0,t.kt)("p",null,"Ok, langsung saja, buat package/project baru dengan nama bebas. Di sini penulis menggunakan nama package ",(0,t.kt)("inlineCode",{parentName:"p"},"module_1"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"cargo new module_1\n")),(0,t.kt)("p",null,"Setelah itu siapkan 1 buah file bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io.rs"),", letakan di dalam folder ",(0,t.kt)("inlineCode",{parentName:"p"},"src")," (1 level dengan file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"). File ini difungsikan sebagai tempat definisi ",(0,t.kt)("em",{parentName:"p"},"module item")," milik sebuah module bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," (io di sini kependekan dari ",(0,t.kt)("em",{parentName:"p"},"input output"),")."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"module_1\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2514\u2500\u2500\u2500 my_io.rs\n")),(0,t.kt)("p",null,"Lanjut, definisikan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," di file ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io.rs"),", isinya kurang lebih adalah kode untuk membaca inputan user lalu mengembalikannya dalam bentuk ",(0,t.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_io.rs"',title:'"src/my_io.rs"'},'pub fn read_entry() -> String {\n let mut message = std::string::String::new();\n let stdin_reader = std::io::stdin(); \n let reader_res = stdin_reader.read_line(&mut message);\n\n if reader_res.is_err() {\n println!("error! {:?}", reader_res.err());\n }\n\n message.trim().to_string()\n}\n')),(0,t.kt)("p",null,"Ok, kita telah menyiapkan satu item milik module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," yaitu sebuah fungsi bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry"),". 1 hal yang sedikit berbeda pada definisi fungsi di atas adalah penambahan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," yang ditulis sebagai prefix definisi fungsi."),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," digunakan untuk menjadikan suatu item menjadi ",(0,t.kt)("strong",{parentName:"p"},"public"),", agar bisa diakses dari luar module."),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," ini berada dalam module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io"),". Jika tidak ada keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," disitu, maka fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," hanya bisa diakses dari dalam ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," saja, tidak bisa diakses dari luar module contohnya seperti dari ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),". Dengan menjadikan ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," sebagai fungsi yang public, maka fungsi tersebut bisa diakses dari ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,t.kt)("p",null,"Isi module sudah siap, selanjutnya lanjut ke pendefinisian modul. Umumnya pada bahasa pemrograman, definisi module adalah ada dalam file dimana isi module berada, namun tidak untuk Rust."),(0,t.kt)("p",null,"Di Rust, definisi sebuah module (sekali lagi bukan item/isinya ya, tapi definisi dari module itu sendiri) dituliskan pada file terpisah, yaitu di file entrypoint crate, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"lib.rs"),"."),(0,t.kt)("p",null,"Lanjut, sekarang buka file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," dan tulis definisi modul ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," menggunakan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"mod my_io"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"// definisi module my_io\nmod my_io;\n\n// fungsi main\nfn main() {\n // ...\n}\n")),(0,t.kt)("p",null,"Sekarang aplikasikan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," milik module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," di fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main")," untuk membaca inputan user."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod my_io;\n\nfn main() {\n println!("enter any number:");\n let message = my_io::read_entry();\n println!("your number: {}", message);\n}\n')),(0,t.kt)("p",null,"Jalankan program, masukan sebuah angka, lalu enter. Lihat hasilnya, program berjalan sesuai harapan."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust module",src:n(5152).Z,width:"762",height:"717"})),(0,t.kt)("p",null,"Cara pengaksesan item dari sebuah module yang kita definisikan sendiri adalah sama seperti pengaksesan item dari crate lainnya, yaitu menggunakan ",(0,t.kt)("em",{parentName:"p"},"path"),". Statement ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io::read_entry()")," artinya item ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry")," yang merupakan fungsi dalam module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," digunakan."),(0,t.kt)("h3",{id:"-summary-praktik-1"},"\u25c9 Summary praktik #1"),(0,t.kt)("p",null,"Pada praktik pertama ini kita telah belajar penerapan module dengan mengaplikasikan beberapa hal berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Penggunaan normal module dalam binary crate"),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," untuk meng-export atau menjadikan item menjadi public, agar bisa diakses dari luar module"),(0,t.kt)("li",{parentName:"ul"},"Pengaksesan item milik module, yaitu: ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io::read_entry"))),(0,t.kt)("h3",{id:"-naming-convention-module"},"\u25c9 ",(0,t.kt)("em",{parentName:"h3"},"Naming convention")," module"),(0,t.kt)("p",null,"Sesuai anjuran di ",(0,t.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", aturan penulisan nama module adalah menggunakan snake case, contohnya ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io"),"."),(0,t.kt)("h3",{id:"-module-item"},"\u25c9 Module item"),(0,t.kt)("p",null,"Module item adalah apapun yang didefinisikan di dalam sebuah module. Pada contoh praktik ke-1 di atas, module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," memiliki 1 buah item yaitu fungsi bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"read_entry"),"."),(0,t.kt)("p",null,"Selain fungsi, module item bisa dalam bentuk lainnya, contohnya: konstanta, submodule, struct, dan lainnya."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang macam-macam item dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/path-item"},"Module System \u279c Path & Item"))),(0,t.kt)("h2",{id:"a203-praktik-2---nama_modulemodrs"},"A.20.3. Praktik #2 - ",(0,t.kt)("inlineCode",{parentName:"h2"},"nama_module/mod.rs")),(0,t.kt)("p",null,"Bagian ini merupakan kelanjutan dari praktik sebelumnya. Program sederhana yang sudah di-buat ditambahi beberapa hal. Data inputan user dikonversi ke bentuk angka untuk kemudian dicek apakah angka tersebut bilangan ganjil atau genap."),(0,t.kt)("p",null,"Fungsi untuk konversi string ke bentuk numerik dan juga untuk pengecekan bilangan ganjil genap, adalah dua buah item milik module bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," yang akan kita definisikan sebentar lagi."),(0,t.kt)("p",null,"Definisi item module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io")," menggunakan notasi penulisan file ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," (yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"my_io.rs"),"). Pada module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," ini kita akan gunakan notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs")," (menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs"),") untuk menampung definisi item module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"module_1\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 my_number\n \u2502 \u2514\u2500\u2500\u2500 mod.rs\n \u2502\u2500\u2500\u2500 main.rs\n \u2514\u2500\u2500\u2500 my_io.rs\n")),(0,t.kt)("p",null,"Ok, sekarang buat saja folder dan filenya, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs"),". Lalu pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"mod.rs")," tulis 2 buah fungsi berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Fungsi untuk konversi string ke numerik ",(0,t.kt)("inlineCode",{parentName:"p"},"i32")),(0,t.kt)("pre",{parentName:"li"},(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},"pub fn string_to_number(text: String) -> i32 {\n return text.parse::().unwrap();\n}\n"))),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Fungsi untuk pengecekan bilangan ganjil"),(0,t.kt)("pre",{parentName:"li"},(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},"// ...\n\npub fn is_odd_number(number: i32) -> bool {\n number % 2 == 1\n}\n")))),(0,t.kt)("p",null,"Kemudian tulis definisi module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," dalam file entrypoint crate, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"// definisi module my_io\nmod my_io;\n\n// definisi module my_number\nmod my_number;\n\n// fungsi main\nfn main() {\n // ...\n}\n")),(0,t.kt)("p",null,"Terakhir, aplikasikan dua buah fungsi yang sudah didefinisikan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'fn main() {\n println!("enter any number:");\n let message = my_io::read_entry();\n println!("your number: {}", message);\n\n let number = my_number::string_to_number(message);\n let result = my_number::is_odd_number(number);\n println!("is odd number: {}", result);\n}\n')),(0,t.kt)("p",null,"Jalankan program, lihat hasilnya."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust module",src:n(7050).Z,width:"777",height:"872"})),(0,t.kt)("h3",{id:"-summary-praktik-2"},"\u25c9 Summary praktik #2"),(0,t.kt)("p",null,"Hingga praktik ke-2 ini, kita telah belajar penerapan module dengan mengaplikasikan beberapa hal berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Penggunaan normal module dalam binary crate"),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," untuk meng-export atau menjadikan item menjadi public, agar bisa diakses dari luar module"),(0,t.kt)("li",{parentName:"ul"},"Pengaksesan item milik module, yaitu: ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io::read_entry"),", ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::string_to_number"),", dan ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::is_odd_number"))),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Penerapan notasi penulisan isi module ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs")," bisa diterapkan dalam satu package")),(0,t.kt)("h3",{id:"-konversi-string-ke-numerik"},"\u25c9 Konversi string ke numerik"),(0,t.kt)("p",null,"Rust mengenal beberapa jenis konversi antar tipe data. Teknik konversi yang kita praktikkan pada chapter ini menggunakan method ",(0,t.kt)("inlineCode",{parentName:"p"},".parse()"),", cara ini bisa dilakukan untuk konversi tipe data dari custom types ke primitive. Contohnya seperti di atas, dari ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,t.kt)("p",null,"Penggunaannya cukup mudah, akses saja method ",(0,t.kt)("inlineCode",{parentName:"p"},".parse()")," kemudian sisipkan tipe data sebagai parameter generic. Lalu chain dengan method ",(0,t.kt)("inlineCode",{parentName:"p"},"unwrap")," (yang method tersebut merupakan item milik tipe data ",(0,t.kt)("em",{parentName:"p"},"generic result type")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"Result"),")."),(0,t.kt)("p",null,"Contoh lain untuk konversi dari tipe primitif lainnya bisa dilihat berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let data_number = "24".parse::().unwrap();\nlet data_bool = "true".parse::().unwrap();\nlet data_float = "3.14".parse::().unwrap();\n')),(0,t.kt)("p",null,"Method ",(0,t.kt)("inlineCode",{parentName:"p"},"parse")," menghasilkan data bertipe ",(0,t.kt)("em",{parentName:"p"},"generic result type")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"Result"),". Tipe ini memiliki method bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"unwrap")," yang berguna untuk mengambil nilai."),(0,t.kt)("blockquote",null,(0,t.kt)("ul",{parentName:"blockquote"},(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai casting dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/type-alias-casting"},"Type Alias & Casting")," dan ",(0,t.kt)("a",{parentName:"li",href:"#/wip/trait-conversion-from-into"},"Trait \u279c Conversion (From & Into)")),(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai generic dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai result type dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/result-type"},"Tipe Data \u279c Result")))),(0,t.kt)("h2",{id:"a204-submodules"},"A.20.4. Submodules"),(0,t.kt)("p",null,"Sampai bagian ini kita telah belajar tentang module beserta 2 macam cara penerapannya."),(0,t.kt)("p",null,"Sebuah module bisa saja memiliki module dibawahnya (biasa disebut submodule), dan hal ini adalah konsep yang umum dalam bahasa pemrograman. Di Rust, aturan dalam pembuatan submodule masih sama seperti module, perbedaannya adalah tempat dimana submodule didefinisikan. Jika pada root module definisi ada pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"lib.rs"),", maka pada submodule definisi ada pada file dimana ",(0,t.kt)("em",{parentName:"p"},"parent module")," berada."),(0,t.kt)("p",null,"Sebagai contoh jika pada program sebelumnya kita tambahkan module ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," yang sudah dibuat, jika ada submodule dengan nama ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion_utility"),", maka definisi module berada di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs")," dan itemnya di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility/mod.rs"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 my_io.rs\n \u2514\u2500\u2500\u2500 my_number\n \u2502\u2500\u2500\u2500 mod.rs // <----- definisi submodule conversion_utility\n \u2514\u2500\u2500\u2500 conversion_utility\n \u2514\u2500\u2500\u2500 mod.rs // <----- definisi item/isi conversion_utility\n")),(0,t.kt)("p",null,"Atau definisi module tetap di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs")," namun itemnya di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility.rs"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 my_io.rs\n \u2514\u2500\u2500\u2500 my_number\n \u2502\u2500\u2500\u2500 mod.rs // <----- definisi submodule conversion_utility\n \u2514\u2500\u2500\u2500 conversion_utility.rs // <----- definisi item/isi conversion_utility\n")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Penerapan notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," biasanya dalam case ketika module tersebut tidak memiliki submodule."),(0,t.kt)("p",{parentName:"blockquote"},"Untuk module yang memiliki submodule, parent module harus menerapkan notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs"),", hal ini karena pendefinisian submodule berada pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"mod.rs")," dalam parent module tersebut.")),(0,t.kt)("p",null,"Lalu bagaimana jika sebuah submodule memiliki submodule yang memiliki submodule ... dst, aturannya tetap sama seperti aturan submodule."),(0,t.kt)("p",null,"Mari lanjut praktik agar tidak bingung. Kita akan modifikasi program sebelumnya. Item ",(0,t.kt)("inlineCode",{parentName:"p"},"string_to_number")," yang sebelumnya adalah item milik ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number")," kita pindah ke sebuah module baru bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion_utility")," yang merupakan submodule dari ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number"),". Silakan buat file baru ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility/mod.rs"),", kemudian isi dengan fungsi berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/conversion_utility/mod.rs"',title:'"src/my_number/conversion_utility/mod.rs"'},"pub fn string_to_number(text: String) -> i32 {\n return text.parse::().unwrap();\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"string_to_number")," yang sebelumnya ada di ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs")," silakan dihapus."),(0,t.kt)("p",null,"Kemudian pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs"),", tambahkan definisi submodule ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion_utility"),". Isi file tersebut kurang lebih menjadi seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},"pub mod conversion_utility;\n\npub fn is_odd_number(number: i32) -> bool {\n number % 2 == 1\n}\n")),(0,t.kt)("p",null,"O iya, karena submodule merupakan sebuah item milik module, maka harus ditambahkan juga keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub"),", agar submodule bisa diakses dari luar scope-nya. Contoh penerapannya bisa dilihat di atas."),(0,t.kt)("p",null,"Terakhir, pada ",(0,t.kt)("inlineCode",{parentName:"p"},"main.rs"),", ubah pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"string_to_number")," dari ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"let number = my_number::string_to_number(message);\n")),(0,t.kt)("p",null,"... menjadi ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"let number = my_number::conversion_utility::string_to_number(message);\n")),(0,t.kt)("p",null,"Kurang lebih strukturnya menjadi seperti berikut:"),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust submodule",src:n(3749).Z,width:"940",height:"955"})),(0,t.kt)("p",null,"Jalankan program untuk mengetest hasilnya."),(0,t.kt)("h3",{id:"-summary-praktik-3"},"\u25c9 Summary praktik #3"),(0,t.kt)("p",null,"Hingga pembahasan pada praktik submodule, kita telah belajar penerapan module dengan mengaplikasikan beberapa hal berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Penggunaan normal module dalam binary crate"),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan submodule dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/conversion_utility"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/conversion.rs")," yang di-import menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"path")," attribute."),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," pada fungsi agar bisa diakses dari luar module"),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," pada submodule agar bisa diakses dari luar parent module"),(0,t.kt)("li",{parentName:"ul"},"Pengaksesan item milik module, yaitu: ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io::read_entry"),", ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::conversion_utility::string_to_number"),", dan ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::is_odd_number"))),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Penerapan notasi penulisan isi module ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module.rs")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_module/mod.rs")," bisa diterapkan dalam satu package")),(0,t.kt)("h2",{id:"a205-penerapan-keyword-use"},"A.20.5. Penerapan keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"use")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"use")," bisa digunakan untuk meng-import module atau item tertentu, dan dengannya pengaksesan item menjadi lebih pendek. Contohnya:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"my_number::conversion_utility::string_to_number(message);\n")),(0,t.kt)("p",null,"... bisa dituliskan menjadi ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"use my_number::conversion_utility::string_to_number;\nstring_to_number(message);\n")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"use")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/use"},"Module System \u279c Use"))),(0,t.kt)("h2",{id:"a206-module-path-attribute"},"A.20.6. Module ",(0,t.kt)("inlineCode",{parentName:"h2"},"path")," attribute"),(0,t.kt)("p",null,"Selain menggunakan dua teknik definisi module item di atas yang fokusnya ada pada penamaan file, ada juga cara lain pendefinisian module item, yaitu dengan memanfaatkan ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute."),(0,t.kt)("p",null,"Mari kita praktikkan, silakan ubah struktur package yang sudah dibuat dari ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 my_io.rs\n \u2514\u2500\u2500\u2500 my_number\n \u2502\u2500\u2500\u2500 mod.rs // <----- definisi submodule conversion_utility\n \u2514\u2500\u2500\u2500 conversion_utility.rs // <----- definisi item/isi conversion_utility\n")),(0,t.kt)("p",null,"... menjadi ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 my_io.rs\n \u2514\u2500\u2500\u2500 my_number\n \u2502\u2500\u2500\u2500 mod.rs // <----- definisi submodule conversion_utility\n \u2514\u2500\u2500\u2500 conversion.rs // <----- definisi item/isi conversion_utility\n")),(0,t.kt)("p",null,"Yang telah kita lakukan adalah me-rename file ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion_utility.rs")," menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion.rs"),". Efeknya akan muncul error dalam pengaksesan module item, karena submodule ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility")," isi/item-nya harus berada pada file ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility.rs")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/conversion_utility/mod.rs"),". Sedangkan file ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion.rs")," tidak memenuhi kriteria tersebut."),(0,t.kt)("p",null,"Sekarang buka isi file ",(0,t.kt)("inlineCode",{parentName:"p"},"my_number/mod.rs"),", lalu ubah statement pendefinisian submodule dari ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},"pub mod conversion_utility;\n")),(0,t.kt)("p",null,"... menjadi ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/my_number/mod.rs"',title:'"src/my_number/mod.rs"'},'#[path = "conversion.rs"]\npub mod conversion_utility;\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Rust module",src:n(6827).Z,width:"690",height:"365"})),(0,t.kt)("p",null,"Lalu run, dan program akan jalan normal tanpa error."),(0,t.kt)("p",null,"Statement ",(0,t.kt)("inlineCode",{parentName:"p"},'#[path = "conversion.rs"]')," di atas merupakan contoh penerapan dari Rust path attributes. Dengannya kita bisa menempatkan isi sebuah module pada file yang namanya bebas (pada contoh di atas, file bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"conversion.rs"),")."),(0,t.kt)("h3",{id:"-summary-praktik-4"},"\u25c9 Summary praktik #4"),(0,t.kt)("p",null,"Hingga pembahasan pada praktik module ",(0,t.kt)("inlineCode",{parentName:"p"},"path")," attribute, kita telah belajar penerapan module dengan mengaplikasikan beberapa hal berikut:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Penggunaan normal module dalam binary crate"),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan module dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"Pembuatan submodule dengan nama ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/conversion_utility"),", dengan isi/item ditulis pada file ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number/conversion_utility/mod.rs")),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," pada fungsi agar bisa diakses dari luar module"),(0,t.kt)("li",{parentName:"ul"},"Penggunaan keyword ",(0,t.kt)("inlineCode",{parentName:"li"},"pub")," pada submodule agar bisa diakses dari luar parent module"),(0,t.kt)("li",{parentName:"ul"},"Pengaksesan item milik module, yaitu: ",(0,t.kt)("inlineCode",{parentName:"li"},"my_io::read_entry"),", ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::conversion_utility::string_to_number"),", dan ",(0,t.kt)("inlineCode",{parentName:"li"},"my_number::is_odd_number"))),(0,t.kt)("h2",{id:"a207-pembahasan-lanjutan"},"A.20.7. Pembahasan lanjutan"),(0,t.kt)("p",null,"Pembahasan topik module dilanjutkan pada beberapa chapter lain. Pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/module-inline"},"Module System \u279c Inline Module")," kita akan bahas secara mendetail mengenai apa itu inline module dan perbedaannya dibanding normal module. Lalu nantinya di cahpter ",(0,t.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy")," akan dibahas secara lengkap mengenai keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"pub")," dan kontrol privasi lainnya di pemrograman Rust."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/module"},"github.com/novalagung/dasarpemrogramanrust-example/../module")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html"},"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.pub.html"},"https://doc.rust-lang.org/std/keyword.pub.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.mod.html"},"https://doc.rust-lang.org/std/keyword.mod.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/mod.html"},"https://doc.rust-lang.org/rust-by-example/mod.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/69275034/what-is-the-difference-between-use-and-pub-use"},"https://stackoverflow.com/questions/69275034/what-is-the-difference-between-use-and-pub-use"))))}o.isMDXComponent=!0},5152:(e,a,n)=>{n.d(a,{Z:()=>i});const i=n.p+"assets/images/module-1-0eea2521d2399e73e455b90f5c2f97fa.png"},7050:(e,a,n)=>{n.d(a,{Z:()=>i});const i=n.p+"assets/images/module-2-d8f6f3862036da93c4b7920e144a3a3f.png"},3749:(e,a,n)=>{n.d(a,{Z:()=>i});const i=n.p+"assets/images/module-3-b8f34c47b905cadb9001a866ebb80809.png"},6827:(e,a,n)=>{n.d(a,{Z:()=>i});const i=n.p+"assets/images/module-4-e83e7a2e8d7d6d4d11ec121d6dc184d0.png"}}]); \ No newline at end of file diff --git a/assets/js/5925ce9f.9da92efc.js b/assets/js/5925ce9f.9da92efc.js new file mode 100644 index 00000000..32310f57 --- /dev/null +++ b/assets/js/5925ce9f.9da92efc.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[9831],{3905:(a,e,n)=>{n.d(e,{Zo:()=>m,kt:()=>g});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function l(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var d=t.createContext({}),s=function(a){var e=t.useContext(d),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},m=function(a){var e=s(a.components);return t.createElement(d.Provider,{value:e},a.children)},u={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},k=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,r=a.originalType,d=a.parentName,m=p(a,["components","mdxType","originalType","parentName"]),k=s(n),g=i,o=k["".concat(d,".").concat(g)]||k[g]||u[g]||r;return n?t.createElement(o,l(l({ref:e},m),{},{components:n})):t.createElement(o,l({ref:e},m))}));function g(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=n.length,l=new Array(r);l[0]=k;var p={};for(var d in e)hasOwnProperty.call(e,d)&&(p[d]=e[d]);p.originalType=a,p.mdxType="string"==typeof a?a:i,l[1]=p;for(var s=2;s{n.r(e),n.d(e,{assets:()=>d,contentTitle:()=>l,default:()=>u,frontMatter:()=>r,metadata:()=>p,toc:()=>s});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:37,title:"A.37. Generics",sidebar_label:"A.37. Generics"},l=void 0,p={unversionedId:"basic/generics",id:"basic/generics",title:"A.37. Generics",description:"Chapter ini membahas tentang generics.",source:"@site/docs/basic/generics.md",sourceDirName:"basic",slug:"/basic/generics",permalink:"/basic/generics",draft:!1,tags:[],version:"current",sidebarPosition:37,frontMatter:{sidebar_position:37,title:"A.37. Generics",sidebar_label:"A.37. Generics"},sidebar:"tutorialSidebar",previous:{title:"A.36. Traits",permalink:"/basic/traits"},next:{title:"A.38. Tipe Data \u279c Option",permalink:"/basic/option-type"}},d={},s=[{value:"A.37.1. Generics basic",id:"a371-generics-basic",level:2},{value:"A.37.2. Mengasosiasikan traits ke parameter generic",id:"a372-mengasosiasikan-traits-ke-parameter-generic",level:2},{value:"\u25c9 Contoh ke-1",id:"-contoh-ke-1",level:3},{value:"\u25c9 Contoh ke-2",id:"-contoh-ke-2",level:3},{value:"\u25c9 Contoh ke-3",id:"-contoh-ke-3",level:3},{value:"A.37.3. Multi traits pada parameter generic",id:"a373-multi-traits-pada-parameter-generic",level:2},{value:"A.37.4. Keyword where",id:"a374-keyword-where",level:2},{value:"A.37.5. Generics struct",id:"a375-generics-struct",level:2},{value:"A.37.6. Generics method",id:"a376-generics-method",level:2},{value:"A.37.7. Method khusus untuk spesifik tipe parameter generic tertentu",id:"a377-method-khusus-untuk-spesifik-tipe-parameter-generic-tertentu",level:2},{value:"A.37.8. Generics enum",id:"a378-generics-enum",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:s};function u(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,t.Z)({},m,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas tentang generics."),(0,i.kt)("p",null,"Generics sendiri merupakan salah satu fitur yang ada pada beberapa bahasa pemrograman (termasuk Rust), yang digunakan untuk menambahkan fleksibilitas dalam pemanfaatan tipe data pada suatu block kode. Dengan adanya generics, kita bisa menentukan tipe data yang digunakan pada parameter maupun return value sbuah block fungsi, method dan lainnya."),(0,i.kt)("p",null,"Generics dinotasikan dengan ",(0,i.kt)("inlineCode",{parentName:"p"},""),". Kita sempat sedikit memanfaatkan generic pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector")," di mana dalam pendefinisian tipe data harus dituliskan juga tipe data item (via generics parameter), contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec<&str>"),", dll. Kita juga sempat sedikit belajar tentang topik generic pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),"."),(0,i.kt)("h2",{id:"a371-generics-basic"},"A.37.1. Generics basic"),(0,i.kt)("p",null,"Mari mulai pembahasan dengan sebauh contoh definisi fungsi yang memiliki generics."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something(arg1: i32, arg2: T) {\n // ...\n}\n")),(0,i.kt)("p",null,"Pada contoh di atas, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something")," didefinisikan dengan 2 buah parameter argument dan 1 buah parameter generics. Parameter argument pertama, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"arg1")," bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", kemudian diikuti parameter ke-2 bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah parameter generic fungsi."),(0,i.kt)("p",null,"Dalam pemanggilan fungsi tersebut, ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," generics dan tipe data argument ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2")," harus sama. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n do_something::(24, false);\n}\n")),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something")," dipanggil dengan disisipi argument ",(0,i.kt)("inlineCode",{parentName:"p"},"24")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"false"),". Parameter ke-2 fungsi tersebut tipe data-nya jelas ",(0,i.kt)("inlineCode",{parentName:"p"},"bool")," karena pada argument-nya nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"false")," digunakan, oleh karena itu tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"bool")," disisipkan pada parameter generics saat pemanggilan fungsi; dengan notasi penulisan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"::"),", posisi penulisannya di antara nama fungsi dan tanda kurung pemanggilan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"("),"."),(0,i.kt)("p",null,"Apa yang terjadi jika nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," diisi dengan tipe data argument ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2")," berbeda? program error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(7757).Z,width:"690",height:"169"})),(0,i.kt)("p",null,"Ok sampai sini semoga jelas."),(0,i.kt)("p",null,"Khusus untuk fungsi yang generic parameter-nya dipergunakan sebagai tipe data parameter, maka boleh untuk tidak ditentukan tipe data-nya saat pemanggilan fungsi. Contohnya bisa dilihat pada kode berikut, dua statements ini adalah ekuivalen."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"do_something::(24, false);\n// T adalah bool, karena ditulis secara eksplisit\n\ndo_something(24, false);\n// T adalah bool, karena diketahui dari tipe data argument.\n// nilai `false` pasti tipe data-nya bool\n")),(0,i.kt)("p",null,"Tipe data generic boleh tidak ditulis karena nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," bisa diketahui dari argument pemanggilan fungsi. Sedangkan jika nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," tidak digunakan sebagai tipe data parameter, maka wajib untuk di-isi nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," saat pemanggilan fungsi."),(0,i.kt)("p",null,"O iya, jumlah parameter generics tidak ada batasan. Bisa saja ada lebih dari satu parameter generic yang didefinisikan, contohnya bisa dilihat pada fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something_v2")," ini yang memiliki 2 buah parameter generic, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"R")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v2(arg1: R, arg2: T) {\n // ...\n}\n")),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"R")," digunakan sebagai tipe parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"arg1")," dan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," pada parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2"),"."),(0,i.kt)("h2",{id:"a372-mengasosiasikan-traits-ke-parameter-generic"},"A.37.2. Mengasosiasikan traits ke parameter generic"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something")," yang telah dipraktikkan, susah untuk diisi dengan apapun. Memang parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"arg1")," tipe-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", dan harusnya mudah untuk bermain-main dengan parameter tersebut. Yang agak repot adalah parameter ke-2, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2")," yang bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something(arg1: i32, arg2: T) {\n // ...\n}\n")),(0,i.kt)("p",null,"Parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2")," hampir tidak bisa diapa-apakan."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Misal mau di-print, tidak bisa, karena tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," tidak implement trait ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},(0,i.kt)("inlineCode",{parentName:"a"},"std::fmt::Debug")),"."),(0,i.kt)("li",{parentName:"ul"},"Misal diisi nilai numerik kemudian dijadikan operand operasi aritmatika, juga tidak bisa karena tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," tidak implement trait ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.Add.html"},(0,i.kt)("inlineCode",{parentName:"a"},"std::ops::Add")),", dan trait operasi bilangan lainnya."),(0,i.kt)("li",{parentName:"ul"},"Misal diisi dengan nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"bool"),", tidak bisa digunakan pada seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"li"},"if")," karena tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," tidak implement trait ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/cmp/trait.PartialOrd.html"},(0,i.kt)("inlineCode",{parentName:"a"},"std::cmp::PartialOrd")),", dan trait operasi logika lainnya.")),(0,i.kt)("p",null,"Repot kan? Tapi tenang, tidak usah khawatir, ada soluasi agar tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," bisa dimanfaatkan, yaitu dengan mengasosiasikan trait ke tipe data generic (sesuai kebutuhan)."),(0,i.kt)("h3",{id:"-contoh-ke-1"},"\u25c9 Contoh ke-1"),(0,i.kt)("p",null,"Contoh pengaplikasiannya bisa dilihat pada kode berikut. Ada sebuah fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"print_x_times")," yang tugasnya adalah menampilkan data ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," sejumlah ",(0,i.kt)("inlineCode",{parentName:"p"},"x")," kali, yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah parameter generics."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n print_x_times("Hello guys", 10);\n}\n\nfn print_x_times(data: T, x: i32) {\n for _ in 0..x {\n println!("{:?}", data);\n }\n}\n')),(0,i.kt)("p",null,"Parameter generic ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," diasosiasikan dengan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug"),", dengan ini maka kita akan bisa print parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"data")," yang tipe data-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("p",null,"Untuk mengetes hasilnya, jalankan program. Bisa dilihat text ",(0,i.kt)("inlineCode",{parentName:"p"},"Hello guys")," muncul 10x."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(4541).Z,width:"315",height:"248"})),(0,i.kt)("p",null,"Cara untuk mengasosiasikan trait ke parameter generic adalah dengan menuliskannya dalam notasi berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn nama_fungsi(arg1 ...) {\n // ...\n}\n")),(0,i.kt)("h3",{id:"-contoh-ke-2"},"\u25c9 Contoh ke-2"),(0,i.kt)("p",null,"Pada contoh ke-2 ini, dideklarasikan sebuah fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"find_largest_number")," yang tugasnya adalah mencari nilai maksimum dari sebuah tipe slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn find_largest_number(list: &[T]) -> &T {\n let mut largest = &list[0];\n\n for item in list {\n if item > largest {\n largest = item;\n }\n }\n\n largest\n}\n")),(0,i.kt)("p",null,"Seperti yang sudah dibahas pada bagian sebelumnya, bahwa tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," tidak akan bisa diapa-apakan kalau tidak diasosiasikan dengan trait. Maka pada contoh ini, tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," diasosiasikan dengan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::cmp::PartialOrd"),". Benefitnya, semua data dengan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," bisa dipergunakan dalam operasi perbandingan."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tanpa adanya trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::cmp::PartialOrd"),", maka statement ",(0,i.kt)("inlineCode",{parentName:"p"},"if item > largest { ... }")," menghasilkan error.")),(0,i.kt)("p",null,"Ok, sekarang panggil fungsi tersebut 2x, yang pertama diisi dengan data slice dari sebuah array, dan yang kedua data slice dari sebuah vector."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let data_arr = [0, 1, 2, 3];\n let largest_number1 = find_largest_number(&data_arr);\n println!("largest_number1: {:?}", largest_number1);\n \n let data_vec = vec![4, 5, 6, 7];\n let largest_number2 = find_largest_number(&data_vec);\n println!("largest_number2: {:?}", largest_number2);\n}\n')),(0,i.kt)("p",null,"Hasilnya sesuai harapan."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(7811).Z,width:"461",height:"113"})),(0,i.kt)("h3",{id:"-contoh-ke-3"},"\u25c9 Contoh ke-3"),(0,i.kt)("p",null,"Trait yang bisa diasosiasikan dengan parameter generics adalah semua jenis traits (tanpa terkecuali), termasuk custom trait yang kita buat sendiri."),(0,i.kt)("p",null,"Contohnya bisa dilihat pada kode berikut. Ada custom trait bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"MyTrait")," dideklarasikan, kemudian diasosiasikan dengan parameter generic ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," milik fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something_v3"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"trait MyTrait {\n // methods declaration\n}\n\nfn do_something_v3(arg1: T) {\n // do something\n}\n")),(0,i.kt)("h2",{id:"a373-multi-traits-pada-parameter-generic"},"A.37.3. Multi traits pada parameter generic"),(0,i.kt)("p",null,"Bagaimana jika ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," perlu untuk diasosiasikan dengan banyak traits (lebih dari satu), apakah bisa? Bisa. Cara penulisannya kurang lebih seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// fn nama_fungsi(arg1 ...)\n\nfn print_largest_number(list: &[T]) {\n let largest = find_largest_number::(list);\n println!("largest number: {:?}", largest);\n}\n')),(0,i.kt)("p",null,"Fungsi di atas adalah fungsi baru, namanya ",(0,i.kt)("inlineCode",{parentName:"p"},"print_largest_number"),", tugasnya adalah mencari nilai maksimum kemudian menampilkannya."),(0,i.kt)("p",null,"Proses pencarian nilai maksimum dilakukan dengan memanfaatkan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"find_largest_number")," yang sebelumnya sudah dibuat. Fungsi tersebut memerlukan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::cmp::PartialOrd")," untuk diasosiasikan dengan tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("p",null,"Setelah nilai maksimum diketemukan, nilainya di-print ke stdout menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Nilai maksimum ditampung pada variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"largest"),", tipe data-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),". Agar bisa di-print maka perlu diasosiasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug"),"."),(0,i.kt)("p",null,"Bisa dilihat cara penulisan asosiasi multi trait ke parameter generic seperti apa. Cukup tulis saja dengan separator tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"+"),"."),(0,i.kt)("h2",{id:"a374-keyword-where"},"A.37.4. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"where")),(0,i.kt)("p",null,"Selain notasi penulisan yang sudah dipelajari di atas ada lagi alternatif lainnya, yaitu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"where"),". Kurang lebih seperti berikut penerapannya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn print_largest_number(list: &[T]) {\n let largest = find_largest_number::(list);\n println!("largest number: {:?}", largest);\n}\n\n// ... adalah ekuivalen dengan ...\n\nfn print_largest_number(list: &[T]) \nwhere\n T: std::cmp::PartialOrd + std::fmt::Debug,\n{\n let largest = find_largest_number::(list);\n println!("largest number: {:?}", largest);\n}\n')),(0,i.kt)("p",null,"Silakan gunakan sesuai preferensi dan kesepakatan tim. Kalau penulis lebih suka cara ke-2, karena terasa lebih rapi untuk penulisan fungsi yang ada banyak parameter generic beserta asosiasi traits-nya. Contohnya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something(arg1: T, arg2: U, arg3: V) \nwhere\n T: some::traits:TraitA,\n U: some::traits:TraitB + some::traits:TraitC + some::traits:TraitD,\n V: some::traits:TraitA + some::traits:TraitD,\n{\n // do something\n}\n")),(0,i.kt)("h2",{id:"a375-generics-struct"},"A.37.5. Generics struct"),(0,i.kt)("p",null,"Selain diterapkan di fungsi, generics bisa juga diterapkan di struct. Cara penulisannya, tambahkan notasi parameter generic di antara nama struct dan block struct."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct Point {\n x: T,\n y: T,\n z: U\n}\n\nfn main() {\n let num_one: Point = Point { x: 502, y: 120, z: 4.5 };\n let num_two: Point = Point { x: 1.2, y: 4.3, z: 534 };\n}\n")),(0,i.kt)("p",null,"Pada contoh di atas, struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," memiliki 2 parameter generic. Kemudian struct tersebut digunakan untuk membuat dua variabel berbeda:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"num_one"),", bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"Point"),". Tipe data property ",(0,i.kt)("inlineCode",{parentName:"li"},"x")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"y")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"i32")," sedangkan tipe data property ",(0,i.kt)("inlineCode",{parentName:"li"},"z")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"f64"),"."),(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"num_two"),", bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"Point"),". Tipe data property ",(0,i.kt)("inlineCode",{parentName:"li"},"x")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"y")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"f64")," sedangkan tipe data property ",(0,i.kt)("inlineCode",{parentName:"li"},"z")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"i32"),".")),(0,i.kt)("h2",{id:"a376-generics-method"},"A.37.6. Generics method"),(0,i.kt)("p",null,"Generic bisa diterapkan pada method. Notasi penulisannya kurang lebih sama seperti pada penulisan method, hanya saja pada syntax ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," perlu diikuti block parameter generics. Perbandingannya kurang lebih seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// method biasa\nimpl Square {\n fn x(&self) -> &i32 {\n &self.x\n }\n}\n\n// method dengan generic\nimpl Square {\n fn x(&self) -> &T {\n &self.x\n }\n}\n")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Di syntax bagian ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),", penulisan parameter generics di keduanya harus sama persis. Misal pada struct ada 3 buah parameter generic ",(0,i.kt)("inlineCode",{parentName:"p"},"Square")," maka pada syntax ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," juga harus sama, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"impl"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Jika tidak sama akan muncul error.")),(0,i.kt)("p",null,"Ok, sekarang mari kita praktikkan. Struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," yang sudah dibuat, kita siapkan method-nya. Ada 3 buah method yang akan dibuat dan kesemuanya adalah method ",(0,i.kt)("em",{parentName:"p"},"getter")," untuk masing-masing property struct (yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"x"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"y"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"z"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct Point {\n x: T,\n y: T,\n z: U\n}\n\nimpl Point {\n\n fn get_x(&self) -> &T {\n &self.x\n }\n\n fn get_y(&self) -> &T {\n &self.y\n }\n\n fn get_z(&self) -> &U {\n &self.z\n }\n}\n")),(0,i.kt)("p",null,"Bisa dilihat pada kode di atas, ada method ",(0,i.kt)("inlineCode",{parentName:"p"},"get_x")," untuk mengambil nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"x"),". Nilai baliknya bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," yang tipe tersebut juga dipakai sebagai tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"x"),"."),(0,i.kt)("p",null,"Kemudian coba gunakan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," untuk membuat satu atau dua variabel, lalu akses method-nya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let num_one: Point = Point { x: 502, y: 120, z: 4.5 };\n println!("{} {} {}", num_one.get_x(), num_one.get_y(), num_one.get_z());\n // 502 120 4.5\n\n let num_two: Point = Point { x: 1.2, y: 4.3, z: 534 };\n println!("{} {} {}", num_two.get_x(), num_two.get_y(), num_two.get_z());\n // 1.2 4.3 534\n}\n')),(0,i.kt)("p",null,"Hasilnya ketika di-run:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(5243).Z,width:"319",height:"110"})),(0,i.kt)("h2",{id:"a377-method-khusus-untuk-spesifik-tipe-parameter-generic-tertentu"},"A.37.7. Method khusus untuk spesifik tipe parameter generic tertentu"),(0,i.kt)("p",null,"Pada contoh di atas, struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," bisa digunakan dalam banyak kombinasi tipe data, misalnya: ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),", dan lainya."),(0,i.kt)("p",null,"Di luar itu, bisa lho mendefinisikan method hanya untuk tipe parameter generic. Misalnya, method hanya bisa diakses ketika ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"U")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),". Caranya kurang lebih seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl Point {\n\n fn get_x(&self) -> &i32 {\n &self.x\n }\n\n fn get_y(&self) -> &i32 {\n &self.y\n }\n\n fn get_z(&self) -> &f64 {\n &self.z\n }\n}\n")),(0,i.kt)("p",null,"Block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," tidak diterapkan dengan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"impl Point"),", melainkan ",(0,i.kt)("inlineCode",{parentName:"p"},"impl Point"),". Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," diganti dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"U")," diganti ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),". Dengan penulisan yang seperti ini, maka method dalam block kode hanya bisa diakses ketika data memiliki tipe data sesuai dengan yang dideklarasikan di block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," (yang pada contoh di atas adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),")."),(0,i.kt)("p",null,"Bisa dilihat pada gambar berikut, sekarang statement pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"num_two")," menjadi error, karena method-method tersebut hanya tersedia untuk tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," sedangkan ",(0,i.kt)("inlineCode",{parentName:"p"},"num_two")," bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(6714).Z,width:"851",height:"501"})),(0,i.kt)("p",null,"Jika ada keperluan untuk mendeklarasikan method lainnya khusus untuk tipe lainnya, cukup tulis lagi block ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," diikuti dengan tipe yang diinginkan. Misalnya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl Point {\n // method untuk tipe data Point\n}\n\nimpl Point {\n // method untuk tipe data Point\n}\n\nimpl Point {\n // method untuk tipe data Point\n}\n\n// ...\n")),(0,i.kt)("h2",{id:"a378-generics-enum"},"A.37.8. Generics enum"),(0,i.kt)("p",null,"Generic juga bisa diterapkan pada tipe enum. Caranya tulis saja deklarasi parameter generic setelah nama enum, lalu gunakan parameter generic-nya sesuai kebutuhan."),(0,i.kt)("p",null,"Contohnya pada kode berikut ini, enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Kendaraan")," memiliki parameter generic ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," yang tipe tersebut dipakai pada value enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Gledekan(T)"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'enum Kendaraan {\n Skateboard,\n SepedaPancal,\n Gledekan(T),\n}\n\nlet kendaraan1 = Kendaraan::<&str>::Skateboard;\nlet kendaraan2 = Kendaraan::<&str>::SepedaPancal;\nlet kendaraan3 = Kendaraan::<&str>::Gledekan("Artco");\n')),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/generics"},"github.com/novalagung/dasarpemrogramanrust-example/../generics")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch10-01-syntax.html"},"https://doc.rust-lang.org/book/ch10-01-syntax.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},"https://doc.rust-lang.org/std/fmt/trait.Debug.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.Add.html"},"https://doc.rust-lang.org/std/ops/trait.Add.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/cmp/trait.PartialOrd.html"},"https://doc.rust-lang.org/std/cmp/trait.PartialOrd.html"))))}u.isMDXComponent=!0},7757:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},4541:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},7811:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},5243:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},6714:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/generics-5-9f575c9b47e95f4301f0e9db22059d17.png"}}]); \ No newline at end of file diff --git a/assets/js/5925ce9f.d25861ea.js b/assets/js/5925ce9f.d25861ea.js deleted file mode 100644 index ad65a9ff..00000000 --- a/assets/js/5925ce9f.d25861ea.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[9831],{3905:(a,e,n)=>{n.d(e,{Zo:()=>m,kt:()=>g});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function l(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var p=t.createContext({}),s=function(a){var e=t.useContext(p),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},m=function(a){var e=s(a.components);return t.createElement(p.Provider,{value:e},a.children)},u={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},k=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,r=a.originalType,p=a.parentName,m=d(a,["components","mdxType","originalType","parentName"]),k=s(n),g=i,o=k["".concat(p,".").concat(g)]||k[g]||u[g]||r;return n?t.createElement(o,l(l({ref:e},m),{},{components:n})):t.createElement(o,l({ref:e},m))}));function g(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=n.length,l=new Array(r);l[0]=k;var d={};for(var p in e)hasOwnProperty.call(e,p)&&(d[p]=e[p]);d.originalType=a,d.mdxType="string"==typeof a?a:i,l[1]=d;for(var s=2;s{n.r(e),n.d(e,{assets:()=>p,contentTitle:()=>l,default:()=>u,frontMatter:()=>r,metadata:()=>d,toc:()=>s});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:37,title:"A.37. Generics",sidebar_label:"A.37. Generics"},l=void 0,d={unversionedId:"basic/generics",id:"basic/generics",title:"A.37. Generics",description:"Chapter ini membahas tentang generics.",source:"@site/docs/basic/generics.md",sourceDirName:"basic",slug:"/basic/generics",permalink:"/basic/generics",draft:!1,tags:[],version:"current",sidebarPosition:37,frontMatter:{sidebar_position:37,title:"A.37. Generics",sidebar_label:"A.37. Generics"},sidebar:"tutorialSidebar",previous:{title:"A.36. Traits",permalink:"/basic/traits"},next:{title:"A.38. Tipe Data \u279c Option",permalink:"/basic/option-type"}},p={},s=[{value:"A.37.1. Generics basic",id:"a371-generics-basic",level:2},{value:"A.37.2. Mengasosiasikan traits ke parameter generic",id:"a372-mengasosiasikan-traits-ke-parameter-generic",level:2},{value:"\u25c9 Contoh ke-1",id:"-contoh-ke-1",level:3},{value:"\u25c9 Contoh ke-2",id:"-contoh-ke-2",level:3},{value:"\u25c9 Contoh ke-3",id:"-contoh-ke-3",level:3},{value:"A.37.3. Multi traits pada parameter generic",id:"a373-multi-traits-pada-parameter-generic",level:2},{value:"A.37.4. Keyword where",id:"a374-keyword-where",level:2},{value:"A.37.5. Generics struct",id:"a375-generics-struct",level:2},{value:"A.37.6. Generics method",id:"a376-generics-method",level:2},{value:"A.37.7. Method khusus untuk spesifik tipe parameter generic tertentu",id:"a377-method-khusus-untuk-spesifik-tipe-parameter-generic-tertentu",level:2},{value:"A.37.8. Generics enum",id:"a378-generics-enum",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:s};function u(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,t.Z)({},m,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas tentang generics."),(0,i.kt)("p",null,"Generics sendiri merupakan salah satu fitur yang ada pada beberapa bahasa pemrograman (termasuk Rust), yang digunakan untuk menambahkan fleksibilitas dalam pemanfaatan tipe data pada suatu block kode. Dengan adanya generics, kita bisa menentukan tipe data yang digunakan pada parameter maupun return value sbuah block fungsi, method dan lainnya."),(0,i.kt)("p",null,"Generics dinotasikan dengan ",(0,i.kt)("inlineCode",{parentName:"p"},""),". Kita sempat sedikit memanfaatkan generic pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector")," dimana dalam pendefinisian tipe data harus dituliskan juga tipe data item (via generics parameter), contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec<&str>"),", dll. Kita juga sempat sedikit belajar tentang topik generic pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),"."),(0,i.kt)("h2",{id:"a371-generics-basic"},"A.37.1. Generics basic"),(0,i.kt)("p",null,"Mari mulai pembahasan dengan sebauh contoh definisi fungsi yang memiliki generics."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something(arg1: i32, arg2: T) {\n // ...\n}\n")),(0,i.kt)("p",null,"Pada contoh di atas, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something")," didefinisikan dengan 2 buah parameter argument dan 1 buah parameter generics. Parameter argument pertama, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"arg1")," bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", kemudian diikuti parameter ke-2 bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah parameter generic fungsi."),(0,i.kt)("p",null,"Dalam pemanggilan fungsi tersebut, ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," generics dan tipe data argument ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2")," harus sama. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n do_something::(24, false);\n}\n")),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something")," dipanggil dengan disisipi argument ",(0,i.kt)("inlineCode",{parentName:"p"},"24")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"false"),". Parameter ke-2 fungsi tersebut tipe data-nya jelas ",(0,i.kt)("inlineCode",{parentName:"p"},"bool")," karena pada argument-nya nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"false")," digunakan, oleh karena itu tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"bool")," disisipkan pada parameter generics saat pemanggilan fungsi; dengan notasi penulisan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"::"),", posisi penulisannya di antara nama fungsi dan tanda kurung pemanggilan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"("),"."),(0,i.kt)("p",null,"Apa yang terjadi jika nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," diisi dengan tipe data argument ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2")," berbeda? program error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(7757).Z,width:"690",height:"169"})),(0,i.kt)("p",null,"Ok sampai sini semoga jelas."),(0,i.kt)("p",null,"Khusus untuk fungsi yang generic parameter-nya dipergunakan sebagai tipe data parameter, maka boleh untuk tidak ditentukan tipe data-nya saat pemanggilan fungsi. Contohnya bisa dilihat pada kode berikut, dua statements ini adalah ekuivalen."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"do_something::(24, false);\n// T adalah bool, karena ditulis secara eksplisit\n\ndo_something(24, false);\n// T adalah bool, karena diketahui dari tipe data argument.\n// nilai `false` pasti tipe data-nya bool\n")),(0,i.kt)("p",null,"Tipe data generic boleh tidak ditulis karena nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," bisa diketahui dari argument pemanggilan fungsi. Sedangkan jika nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," tidak digunakan sebagai tipe data parameter, maka wajib untuk di-isi nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," saat pemanggilan fungsi."),(0,i.kt)("p",null,"O iya, jumlah parameter generics tidak ada batasan. Bisa saja ada lebih dari satu parameter generic yang didefinisikan, contohnya bisa dilihat pada fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something_v2")," ini yang memiliki 2 buah parameter generic, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"R")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v2(arg1: R, arg2: T) {\n // ...\n}\n")),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"R")," digunakan sebagai tipe parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"arg1")," dan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," pada parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2"),"."),(0,i.kt)("h2",{id:"a372-mengasosiasikan-traits-ke-parameter-generic"},"A.37.2. Mengasosiasikan traits ke parameter generic"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something")," yang telah dipraktikkan, susah untuk diisi dengan apapun. Memang parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"arg1")," tipe-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", dan harusnya mudah untuk bermain-main dengan parameter tersebut. Yang agak repot adalah parameter ke-2, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2")," yang bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something(arg1: i32, arg2: T) {\n // ...\n}\n")),(0,i.kt)("p",null,"Parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"arg2")," hampir tidak bisa diapa-apakan."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Misal mau di-print, tidak bisa, karena tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," tidak implement trait ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},(0,i.kt)("inlineCode",{parentName:"a"},"std::fmt::Debug")),"."),(0,i.kt)("li",{parentName:"ul"},"Misal diisi nilai numerik kemudian dijadikan operand operasi aritmatika, juga tidak bisa karena tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," tidak implement trait ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.Add.html"},(0,i.kt)("inlineCode",{parentName:"a"},"std::ops::Add")),", dan trait operasi bilangan lainnya."),(0,i.kt)("li",{parentName:"ul"},"Misal diisi dengan nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"bool"),", tidak bisa digunakan pada seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"li"},"if")," karena tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," tidak implement trait ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/cmp/trait.PartialOrd.html"},(0,i.kt)("inlineCode",{parentName:"a"},"std::cmp::PartialOrd")),", dan trait operasi logika lainnya.")),(0,i.kt)("p",null,"Repot kan? Tapi tenang, tidak usah khawatir, ada soluasi agar tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," bisa dimanfaatkan, yaitu dengan mengasosiasikan trait ke tipe data generic (sesuai kebutuhan)."),(0,i.kt)("h3",{id:"-contoh-ke-1"},"\u25c9 Contoh ke-1"),(0,i.kt)("p",null,"Contoh pengaplikasiannya bisa dilihat pada kode berikut. Ada sebuah fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"print_x_times")," yang tugasnya adalah menampilkan data ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," sejumlah ",(0,i.kt)("inlineCode",{parentName:"p"},"x")," kali, dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah parameter generics."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n print_x_times("Hello guys", 10);\n}\n\nfn print_x_times(data: T, x: i32) {\n for _ in 0..x {\n println!("{:?}", data);\n }\n}\n')),(0,i.kt)("p",null,"Parameter generic ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," diasosiasikan dengan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug"),", dengan ini maka kita akan bisa print parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"data")," yang tipe data-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("p",null,"Untuk mengetes hasilnya, jalankan program. Bisa dilihat text ",(0,i.kt)("inlineCode",{parentName:"p"},"Hello guys")," muncul 10x."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(4541).Z,width:"315",height:"248"})),(0,i.kt)("p",null,"Cara untuk mengasosiasikan trait ke parameter generic adalah dengan menuliskannya dalam notasi berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn nama_fungsi(arg1 ...) {\n // ...\n}\n")),(0,i.kt)("h3",{id:"-contoh-ke-2"},"\u25c9 Contoh ke-2"),(0,i.kt)("p",null,"Pada contoh ke-2 ini, dideklarasikan sebuah fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"find_largest_number")," yang tugasnya adalah mencari nilai maksimum dari sebuah tipe slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn find_largest_number(list: &[T]) -> &T {\n let mut largest = &list[0];\n\n for item in list {\n if item > largest {\n largest = item;\n }\n }\n\n largest\n}\n")),(0,i.kt)("p",null,"Seperti yang sudah dibahas pada bagian sebelumnya, bahwa tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," tidak akan bisa diapa-apakan kalau tidak diasosiasikan dengan trait. Maka pada contoh ini, tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," diasosiasikan dengan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::cmp::PartialOrd"),". Benefitnya, semua data dengan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," bisa dipergunakan dalam operasi perbandingan."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tanpa adanya trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::cmp::PartialOrd"),", maka statement ",(0,i.kt)("inlineCode",{parentName:"p"},"if item > largest { ... }")," menghasilkan error.")),(0,i.kt)("p",null,"Ok, sekarang panggil fungsi tersebut 2x, yang pertama diisi dengan data slice dari sebuah array, dan yang kedua data slice dari sebuah vector."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let data_arr = [0, 1, 2, 3];\n let largest_number1 = find_largest_number(&data_arr);\n println!("largest_number1: {:?}", largest_number1);\n \n let data_vec = vec![4, 5, 6, 7];\n let largest_number2 = find_largest_number(&data_vec);\n println!("largest_number2: {:?}", largest_number2);\n}\n')),(0,i.kt)("p",null,"Hasilnya sesuai harapan."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(7811).Z,width:"461",height:"113"})),(0,i.kt)("h3",{id:"-contoh-ke-3"},"\u25c9 Contoh ke-3"),(0,i.kt)("p",null,"Trait yang bisa diasosiasikan dengan parameter generics adalah semua jenis traits (tanpa terkecuali), termasuk custom trait yang kita buat sendiri."),(0,i.kt)("p",null,"Contohnya bisa dilihat pada kode berikut. Ada custom trait bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"MyTrait")," dideklarasikan, kemudian diasosiasikan dengan parameter generic ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," milik fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"do_something_v3"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"trait MyTrait {\n // methods declaration\n}\n\nfn do_something_v3(arg1: T) {\n // do something\n}\n")),(0,i.kt)("h2",{id:"a373-multi-traits-pada-parameter-generic"},"A.37.3. Multi traits pada parameter generic"),(0,i.kt)("p",null,"Bagaimana jika ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," perlu untuk diasosiasikan dengan banyak traits (lebih dari satu), apakah bisa? Bisa. Cara penulisannya kurang lebih seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// fn nama_fungsi(arg1 ...)\n\nfn print_largest_number(list: &[T]) {\n let largest = find_largest_number::(list);\n println!("largest number: {:?}", largest);\n}\n')),(0,i.kt)("p",null,"Fungsi di atas adalah fungsi baru, namanya ",(0,i.kt)("inlineCode",{parentName:"p"},"print_largest_number"),", tugasnya adalah mencari nilai maksimum kemudian menampilkannya."),(0,i.kt)("p",null,"Proses pencarian nilai maksimum dilakukan dengan memanfaatkan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"find_largest_number")," yang sebelumnya sudah dibuat. Fungsi tersebut memerlukan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::cmp::PartialOrd")," untuk diasosiasikan dengan tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("p",null,"Setelah nilai maksimum diketemukan, nilainya di-print ke stdout menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Nilai maksimum ditampung pada variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"largest"),", tipe data-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),". Agar bisa di-print maka perlu diasosiasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug"),"."),(0,i.kt)("p",null,"Bisa dilihat cara penulisan asosiasi multi trait ke parameter generic seperti apa. Cukup tulis saja dengan separator tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"+"),"."),(0,i.kt)("h2",{id:"a374-keyword-where"},"A.37.4. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"where")),(0,i.kt)("p",null,"Selain notasi penulisan yang sudah dipelajari di atas ada lagi alternatif lainnya, yaitu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"where"),". Kurang lebih seperti berikut penerapannya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn print_largest_number(list: &[T]) {\n let largest = find_largest_number::(list);\n println!("largest number: {:?}", largest);\n}\n\n// ... adalah ekuivalen dengan ...\n\nfn print_largest_number(list: &[T]) \nwhere\n T: std::cmp::PartialOrd + std::fmt::Debug,\n{\n let largest = find_largest_number::(list);\n println!("largest number: {:?}", largest);\n}\n')),(0,i.kt)("p",null,"Silakan gunakan sesuai preferensi dan kesepakatan tim. Kalau penulis lebih suka cara ke-2, karena terasa lebih rapi untuk penulisan fungsi yang ada banyak parameter generic beserta asosiasi traits-nya. Contohnya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something(arg1: T, arg2: U, arg3: V) \nwhere\n T: some::traits:TraitA,\n U: some::traits:TraitB + some::traits:TraitC + some::traits:TraitD,\n V: some::traits:TraitA + some::traits:TraitD,\n{\n // do something\n}\n")),(0,i.kt)("h2",{id:"a375-generics-struct"},"A.37.5. Generics struct"),(0,i.kt)("p",null,"Selain diterapkan di fungsi, generics bisa juga diterapkan di struct. Cara penulisannya, tambahkan notasi parameter generic di antara nama struct dan block struct."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct Point {\n x: T,\n y: T,\n z: U\n}\n\nfn main() {\n let num_one: Point = Point { x: 502, y: 120, z: 4.5 };\n let num_two: Point = Point { x: 1.2, y: 4.3, z: 534 };\n}\n")),(0,i.kt)("p",null,"Pada contoh di atas, struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," memiliki 2 parameter generic. Kemudian struct tersebut digunakan untuk membuat dua variabel berbeda:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"num_one"),", bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"Point"),". Tipe data property ",(0,i.kt)("inlineCode",{parentName:"li"},"x")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"y")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"i32")," sedangkan tipe data property ",(0,i.kt)("inlineCode",{parentName:"li"},"z")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"f64"),"."),(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"num_two"),", bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"Point"),". Tipe data property ",(0,i.kt)("inlineCode",{parentName:"li"},"x")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"y")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"f64")," sedangkan tipe data property ",(0,i.kt)("inlineCode",{parentName:"li"},"z")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"i32"),".")),(0,i.kt)("h2",{id:"a376-generics-method"},"A.37.6. Generics method"),(0,i.kt)("p",null,"Generic bisa diterapkan pada method. Notasi penulisannya kurang lebih sama seperti pada penulisan method, hanya saja pada syntax ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," perlu diikuti block parameter generics. Perbandingannya kurang lebih seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// method biasa\nimpl Square {\n fn x(&self) -> &i32 {\n &self.x\n }\n}\n\n// method dengan generic\nimpl Square {\n fn x(&self) -> &T {\n &self.x\n }\n}\n")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Di syntax bagian ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),", penulisan parameter generics di keduanya harus sama persis. Misal pada struct ada 3 buah parameter generic ",(0,i.kt)("inlineCode",{parentName:"p"},"Square")," maka pada syntax ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," juga harus sama, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"impl"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Jika tidak sama akan muncul error.")),(0,i.kt)("p",null,"Ok, sekarang mari kita praktikkan. Struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," yang sudah dibuat, kita siapkan method-nya. Ada 3 buah method yang akan dibuat dan kesemuanya adalah method ",(0,i.kt)("em",{parentName:"p"},"getter")," untuk masing-masing property struct (yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"x"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"y"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"z"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct Point {\n x: T,\n y: T,\n z: U\n}\n\nimpl Point {\n\n fn get_x(&self) -> &T {\n &self.x\n }\n\n fn get_y(&self) -> &T {\n &self.y\n }\n\n fn get_z(&self) -> &U {\n &self.z\n }\n}\n")),(0,i.kt)("p",null,"Bisa dilihat pada kode di atas, ada method ",(0,i.kt)("inlineCode",{parentName:"p"},"get_x")," untuk mengambil nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"x"),". Nilai baliknya bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dimana tipe tersebut juga dipakai sebagai tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"x"),"."),(0,i.kt)("p",null,"Kemudian coba gunakan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," untuk membuat satu atau dua variabel, lalu akses method-nya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let num_one: Point = Point { x: 502, y: 120, z: 4.5 };\n println!("{} {} {}", num_one.get_x(), num_one.get_y(), num_one.get_z());\n // 502 120 4.5\n\n let num_two: Point = Point { x: 1.2, y: 4.3, z: 534 };\n println!("{} {} {}", num_two.get_x(), num_two.get_y(), num_two.get_z());\n // 1.2 4.3 534\n}\n')),(0,i.kt)("p",null,"Hasilnya ketika di-run:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(5243).Z,width:"319",height:"110"})),(0,i.kt)("h2",{id:"a377-method-khusus-untuk-spesifik-tipe-parameter-generic-tertentu"},"A.37.7. Method khusus untuk spesifik tipe parameter generic tertentu"),(0,i.kt)("p",null,"Pada contoh di atas, struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," bisa digunakan dalam banyak kombinasi tipe data, misalnya: ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),", dan lainya."),(0,i.kt)("p",null,"Di luar itu, bisa lho mendefinisikan method hanya untuk tipe parameter generic. Misalnya, method hanya bisa diakses ketika ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"U")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),". Caranya kurang lebih seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl Point {\n\n fn get_x(&self) -> &i32 {\n &self.x\n }\n\n fn get_y(&self) -> &i32 {\n &self.y\n }\n\n fn get_z(&self) -> &f64 {\n &self.z\n }\n}\n")),(0,i.kt)("p",null,"Block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," tidak diterapkan dengan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"impl Point"),", melainkan ",(0,i.kt)("inlineCode",{parentName:"p"},"impl Point"),". Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," diganti dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"U")," diganti ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),". Dengan penulisan yang seperti ini, maka method dalam block kode hanya bisa diakses ketika data memiliki tipe data sesuai dengan yang dideklarasikan di block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," (yang pada contoh di atas adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),")."),(0,i.kt)("p",null,"Bisa dilihat pada gambar berikut, sekarang statement pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"num_two")," menjadi error, karena method-method tersebut hanya tersedia untuk tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Point")," sedangkan ",(0,i.kt)("inlineCode",{parentName:"p"},"num_two")," bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Point"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Generics",src:n(6714).Z,width:"851",height:"501"})),(0,i.kt)("p",null,"Jika ada keperluan untuk mendeklarasikan method lainnya khusus untuk tipe lainnya, cukup tulis lagi block ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," diikuti dengan tipe yang diinginkan. Misalnya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl Point {\n // method untuk tipe data Point\n}\n\nimpl Point {\n // method untuk tipe data Point\n}\n\nimpl Point {\n // method untuk tipe data Point\n}\n\n// ...\n")),(0,i.kt)("h2",{id:"a378-generics-enum"},"A.37.8. Generics enum"),(0,i.kt)("p",null,"Generic juga bisa diterapkan pada tipe enum. Caranya tulis saja deklarasi parameter generic setelah nama enum, lalu gunakan parameter generic-nya sesuai kebutuhan."),(0,i.kt)("p",null,"Contohnya pada kode berikut ini, enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Kendaraan")," memiliki parameter generic ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," yang tipe tersebut dipakai pada value enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Gledekan(T)"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'enum Kendaraan {\n Skateboard,\n SepedaPancal,\n Gledekan(T),\n}\n\nlet kendaraan1 = Kendaraan::<&str>::Skateboard;\nlet kendaraan2 = Kendaraan::<&str>::SepedaPancal;\nlet kendaraan3 = Kendaraan::<&str>::Gledekan("Artco");\n')),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/generics"},"github.com/novalagung/dasarpemrogramanrust-example/../generics")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch10-01-syntax.html"},"https://doc.rust-lang.org/book/ch10-01-syntax.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},"https://doc.rust-lang.org/std/fmt/trait.Debug.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.Add.html"},"https://doc.rust-lang.org/std/ops/trait.Add.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/cmp/trait.PartialOrd.html"},"https://doc.rust-lang.org/std/cmp/trait.PartialOrd.html"))))}u.isMDXComponent=!0},7757:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},4541:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},7811:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},5243:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},6714:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/generics-5-9f575c9b47e95f4301f0e9db22059d17.png"}}]); \ No newline at end of file diff --git a/assets/js/5f289fb7.c8228f07.js b/assets/js/5f289fb7.c8228f07.js new file mode 100644 index 00000000..3a097b10 --- /dev/null +++ b/assets/js/5f289fb7.c8228f07.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[2870],{3905:(a,e,t)=>{t.d(e,{Zo:()=>d,kt:()=>o});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var p=n.createContext({}),m=function(a){var e=n.useContext(p),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},d=function(a){var e=m(a.components);return n.createElement(p.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},s=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,p=a.parentName,d=u(a,["components","mdxType","originalType","parentName"]),s=m(t),o=i,b=s["".concat(p,".").concat(o)]||s[o]||k[o]||r;return t?n.createElement(b,l(l({ref:e},d),{},{components:t})):n.createElement(b,l({ref:e},d))}));function o(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=s;var u={};for(var p in e)hasOwnProperty.call(e,p)&&(u[p]=e[p]);u.originalType=a,u.mdxType="string"==typeof a?a:i,l[1]=u;for(var m=2;m{t.r(e),t.d(e,{assets:()=>p,contentTitle:()=>l,default:()=>k,frontMatter:()=>r,metadata:()=>u,toc:()=>m});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:49,title:"A.49. Attributes",sidebar_label:"A.49. Attributes"},l=void 0,u={unversionedId:"basic/attributes",id:"basic/attributes",title:"A.49. Attributes",description:"Kita telah beberapa kali menggunakan attributes pada chapter-chapter sebelumnya, contohnya seperti #[derive(Debug)]. Pada chapter ini kita akan bahas tentang apa sebenarnya attributes, macam-macam jenisnya, beserta kegunaannya.",source:"@site/docs/basic/attributes.md",sourceDirName:"basic",slug:"/basic/attributes",permalink:"/basic/attributes",draft:!1,tags:[],version:"current",sidebarPosition:49,frontMatter:{sidebar_position:49,title:"A.49. Attributes",sidebar_label:"A.49. Attributes"},sidebar:"tutorialSidebar",previous:{title:"A.48. Trait \u279c Iterator",permalink:"/basic/trait-iterator"},next:{title:"A.50. Date Time",permalink:"/basic/datetime"}},p={},m=[{value:"A.49.1. Konsep attributes",id:"a491-konsep-attributes",level:2},{value:"A.49.2. Attribute derive",id:"a492-attribute-derive",level:2},{value:"A.49.3. Attribute cfg / configuration",id:"a493-attribute-cfg--configuration",level:2},{value:"\u25c9 Configuration target_os",id:"-configuration-target_os",level:3},{value:"\u25c9 Configuration target_arch",id:"-configuration-target_arch",level:3},{value:"\u25c9 Other configuration",id:"-other-configuration",level:3},{value:"A.49.4. Attribute linting & diagnostic",id:"a494-attribute-linting--diagnostic",level:2},{value:"A.49.5. Attribute type system",id:"a495-attribute-type-system",level:2},{value:"A.49.6. Attribute modules",id:"a496-attribute-modules",level:2},{value:"A.49.7. Attribute testing",id:"a497-attribute-testing",level:2},{value:"A.49.8. Attribute macros",id:"a498-attribute-macros",level:2},{value:"A.49.9 Attribute lainnya",id:"a499-attribute-lainnya",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function k(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},d,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Kita telah beberapa kali menggunakan attributes pada chapter-chapter sebelumnya, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"#[derive(Debug)]"),". Pada chapter ini kita akan bahas tentang apa sebenarnya attributes, macam-macam jenisnya, beserta kegunaannya."),(0,i.kt)("h2",{id:"a491-konsep-attributes"},"A.49.1. Konsep attributes"),(0,i.kt)("p",null,"Attributes adalah metadata yang didefinisikan untuk suatu crate, module, atau module item. Kegunaan dari attributes berbeda satu sama lain, tergantung attribute apa yang dipakai (kita akan bahas satu per satu)."),(0,i.kt)("p",null,"Attribute dikategorikan menjadi 2:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Outer attributes"),(0,i.kt)("li",{parentName:"ul"},"Inner attributes")),(0,i.kt)("p",null,"Keduanya memiliki kegunaan yang sama, pembedanya adalah posisi di mana attribute harus dituliskan."),(0,i.kt)("p",null,"Outer attribute dituliskan tepat sebelum target (crate, module, module item, atau lainnya) dengan notasi penulisan seperti berikut:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},'#[attribute = "value"]')),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},'#[attribute(key = "value")]')),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[attribute(value)]"))),(0,i.kt)("p",null,"Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"#[derive(Debug)]\nstruct LegoSet {\n code: i32,\n name: String,\n category: String,\n age_minimum: i32,\n}\n")),(0,i.kt)("p",null,"Sedikit berbeda dengan inner attribute, penulisannya berada di dalam target (crate, module, module item, atau lainnya). Notasi penulisannya:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},'#![attribute = "value"]')),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},'#![attribute(key = "value")]')),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#![attribute(value)]"))),(0,i.kt)("p",null,"Rust mengenal beberapa jenis attributes, dan kita akan membahasnya satu per satu."),(0,i.kt)("h2",{id:"a492-attribute-derive"},"A.49.2. Attribute ",(0,i.kt)("inlineCode",{parentName:"h2"},"derive")),(0,i.kt)("p",null,"Attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," digunakan untuk mempermudah implementasi suatu trait ke tipe data."),(0,i.kt)("p",null,"Kita telah mempelajari cara implementasi trait pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits")," yaitu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"for"),", kemudian diikuti dengan method serta implementasinya."),(0,i.kt)("p",null,"Dengan memanfaatkan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," kita tidak perlu menggunakan cara tersebut. Cukup tulis saja attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," beserta trait yang ingin diimplementasikan."),(0,i.kt)("p",null,"Agar lebih jelas, silakan pelajari contoh berikut terlebih dahulu:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'const SuperheroSuperman: &str = "superman";\nconst SuperheroOmniMan: &str = "omniman";\nconst SuperheroHyperion: &str = "hyperion";\n\nenum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n\nfn main() {\n let value: Superhero = Superhero::Superman;\n\n if value == Superhero::Superman {\n println!("hello superman!");\n }\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(253).Z,width:"958",height:"719"})),(0,i.kt)("p",null,"Kode di atas menghasilkan error karena enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero")," tidak mengadopsi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"PartialEq")," yang mana trait ini diperlukan dalam seleksi kondisi menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," dan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"=="),"."),(0,i.kt)("p",null,"Cara mengatasi error tersebut adalah dengan mengimplementasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"PartialEq")," secara eksplisit. Sekarang coba tambahkan kode berikut pada deklarasi enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero"),", maka error akan hilang."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n\nimpl PartialEq for Superhero {\n\n fn eq(&self, other: &Self) -> bool {\n match (self, other) {\n (Superhero::Superman, Superhero::Superman) => true,\n (Superhero::OmniMan, Superhero::OmniMan) => true,\n (Superhero::Hyperion, Superhero::Hyperion) => true,\n _ => false,\n }\n }\n}\n")),(0,i.kt)("p",null,"Cara lain untuk mengatasi error di atas, selain menggunakan teknik implementasi trait secara eksplisit, adalah dengan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," disertai trait ",(0,i.kt)("inlineCode",{parentName:"p"},"PartialEq"),". Kurang lebih penulisan cara ini bisa dilihat di bawah ini. Cukup hapus block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," lalu tambahkan attribute pada deklarasi enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"#[derive(PartialEq)]\nenum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n")),(0,i.kt)("p",null,"Cukup mudah bukan?"),(0,i.kt)("p",null,"Contoh lainnya, misalnya variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"value")," ingin di-print seperti pada kode berikut ini, pasti hasilnya error."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let value: Superhero = Superhero::Superman;\n\n if value == Superhero::Superman {\n println!("hello superman!");\n }\n\n println!("{value} (via `Display` trait)");\n println!("{value:#?} (via `Debug` trait)");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(7095).Z,width:"900",height:"461"})),(0,i.kt)("p",null,"Cara ",(0,i.kt)("em",{parentName:"p"},"resolve")," error di atas adalah dengan mengimplementasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Display")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," secara eksplisit. Atau, bisa juga menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," yang pastinya lebih praktis."),(0,i.kt)("p",null,"Kabar buruknya, hanya trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," yang menyediakan fitur implementasi trait menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive"),". Untuk trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Display")," kita perlu melakukan implementasi secara eksplisit."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Untuk tau mana trait yang bisa di-",(0,i.kt)("em",{parentName:"p"},"derive")," atau tidak, cukup lihat saja ",(0,i.kt)("em",{parentName:"p"},"highlight")," error yang muncul saat penulisan kode atau kompilasi.")),(0,i.kt)("p",null,"Ok, kita implementasikan saja keduanya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(PartialEq, Debug)]\nenum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n\nimpl std::fmt::Display for Superhero {\n\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "{self:?}")\n }\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(635).Z,width:"537",height:"127"})),(0,i.kt)("p",null,"Bisa dilihat, hasilnya program tereksekusi tanpa error. Enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero")," kini mengadopsi 3 trait:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"PartialEq")," via attribute ",(0,i.kt)("inlineCode",{parentName:"li"},"derive")),(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Debug")," via attribute ",(0,i.kt)("inlineCode",{parentName:"li"},"derive")),(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Display")," via implementasi eksplisit")),(0,i.kt)("h2",{id:"a493-attribute-cfg--configuration"},"A.49.3. Attribute ",(0,i.kt)("inlineCode",{parentName:"h2"},"cfg")," / ",(0,i.kt)("em",{parentName:"h2"},"configuration")),(0,i.kt)("p",null,"Attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg")," digunakan untuk menentukan beberapa konfigurasi yang berhubungan dengan arsitekture hardware/prosesor."),(0,i.kt)("p",null,"Salah satu contoh penerapannya bisa dilihat pada kode berikut. Ada 2 buah module yang namanya sama persis, perbedaannya adalah satu didefinisikan khusus untuk platform ",(0,i.kt)("inlineCode",{parentName:"p"},"linux"),", dan satunya lagi untuk platform ",(0,i.kt)("inlineCode",{parentName:"p"},"windows"),". Hal seperti ini bisa dilakukan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg")," dengan key ",(0,i.kt)("inlineCode",{parentName:"p"},"target_os"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[cfg(target_os = "linux")]\nmod util {\n\n pub fn say_hello() {\n println!("hello (from linux)")\n }\n}\n\n#[cfg(target_os = "windows")]\nmod util {\n\n pub fn say_hello() {\n println!("hello (from windows)")\n }\n\n pub fn say_something() {\n println!("how are you")\n }\n}\n')),(0,i.kt)("p",null,"Bisa dilihat cara penulisannya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg()")," kemudian diisi key ",(0,i.kt)("inlineCode",{parentName:"p"},"target_os")," dengan value adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"windows")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"linux"),"."),(0,i.kt)("p",null,"Dengan kondisi kode seperti di atas, ketika berada di sistem operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"linux"),", item ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_hello")," bisa diakses. Sedangkan pada sistem operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"windows"),", item ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_hello")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_something")," bisa diakses."),(0,i.kt)("p",null,"Attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg(target_os)")," juga bisa diterapkan pada block kode. Contohnya seperti pada kode berikut. Item ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_hello")," dipanggil di fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),". Dan khusus untuk sistem operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"windows"),", block kode berisi pemanggilan ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_something")," juga dipanggil."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n util::say_hello();\n\n #[cfg(target_os = "windows")]\n {\n util::say_something();\n }\n}\n')),(0,i.kt)("p",null,"Ada beberapa key yang tersedia pada attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg"),", di antaranya:"),(0,i.kt)("h3",{id:"-configuration-target_os"},"\u25c9 Configuration ",(0,i.kt)("inlineCode",{parentName:"h3"},"target_os")),(0,i.kt)("p",null,"Digunakan untuk menandai bahwa item atau statement di bawah definisi attribute ini dikhususkan untuk sistem operasi tertentu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[cfg(target_os = "value")]\n')),(0,i.kt)("p",null,"Opsi value yang tersedia:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"windows")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"macos")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"ios")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"linux")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"android")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"freebsd")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"dragonfly")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"openbsd")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"netbsd"))),(0,i.kt)("h3",{id:"-configuration-target_arch"},"\u25c9 Configuration ",(0,i.kt)("inlineCode",{parentName:"h3"},"target_arch")),(0,i.kt)("p",null,"Digunakan untuk menandai bahwa item atau statement di bawah definisi attribute ini dikhususkan untuk arsitektur CPU tertentu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[cfg(target_arch = "value")]\n')),(0,i.kt)("p",null,"Opsi value yang tersedia:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"x86")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"x86_64")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"mips")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"powerpc")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"powerpc64")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"arm")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"aarch64"))),(0,i.kt)("h3",{id:"-other-configuration"},"\u25c9 Other configuration"),(0,i.kt)("p",null,"Ada beberapa key konfigurasi lainnya yang tersedia. Lebih detailnya silakan lihat di ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/reference/conditional-compilation.html#set-configuration-options"},"https://doc.rust-lang.org/reference/conditional-compilation.html#set-configuration-options"),"."),(0,i.kt)("h2",{id:"a494-attribute-linting--diagnostic"},"A.49.4. Attribute ",(0,i.kt)("em",{parentName:"h2"},"linting")," & ",(0,i.kt)("em",{parentName:"h2"},"diagnostic")),(0,i.kt)("p",null,"Ada beberapa attribute name yang bisa digunakan untuk meng-override ",(0,i.kt)("em",{parentName:"p"},"default linting")," milik Rust ataupun menandai indikator ",(0,i.kt)("em",{parentName:"p"},"diagnostic")," lainnya, seperti warning yang muncul karena ada kode yang tidak digunakan, dll; Warning sejenis ini bisa di-override menggunakan attribute."),(0,i.kt)("p",null,"Contoh kasus yang berhubungan dengan ",(0,i.kt)("em",{parentName:"p"},"linting")," bisa dilihat pada kode berikut."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(2329).Z,width:"351",height:"363"})),(0,i.kt)("p",null,"Kode di atas tidak menghasilkan error. Kode akan dieksekusi tanpa error. Namun ada 3 buah warning yang muncul karena beberapa baris kode tidak digunakan atau sia-sia."),(0,i.kt)("p",null,"Cara agar warning tidak muncul bisa dengan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"#[allow(value)]"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[allow(unused_imports)]\nuse std::fmt::Display;\n\nfn say_hello() {\n println!("hello")\n}\n\n#[allow(dead_code)]\nfn say_something() {\n println!("how are you")\n}\n\nfn main() {\n #[allow(unused_variables)]\n let name = "noval agung";\n\n say_hello();\n}\n')),(0,i.kt)("p",null,"Pada kode di atas, attribute name ",(0,i.kt)("inlineCode",{parentName:"p"},"allow")," digunakan pada 3 tempat:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[allow(unused_imports)]")," digunakan untuk antisipasi error yang muncul ketika module item di-import namun tidak digunakan."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[allow(dead_code)]")," digunakan untuk membolehkan kode yang tidak digunakan."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[allow(unused_variables)]")," digunakan untuk membolehkan variabel yang didefinisikan tapi tidak dimanfaatkan.")),(0,i.kt)("p",null,"Dengan penambahan 3 attribute di atas program akan tereksekusi tanpa warning."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(7964).Z,width:"463",height:"107"})),(0,i.kt)("p",null,"Ada beberapa attribute ",(0,i.kt)("em",{parentName:"p"},"key")," yang bisa digunakan untuk override ",(0,i.kt)("em",{parentName:"p"},"lint")," warning:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},(0,i.kt)("inlineCode",{parentName:"p"},"#[allow(lint_rule)]")," untuk membolehkan suatu ",(0,i.kt)("em",{parentName:"p"},"lint rule"),".",(0,i.kt)("br",null),"List ",(0,i.kt)("inlineCode",{parentName:"p"},"lint_rule")," bisa dilihat di ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/rustc/lints/listing/allowed-by-default.html"},"https://doc.rust-lang.org/rustc/lints/listing/allowed-by-default.html"),".")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},(0,i.kt)("inlineCode",{parentName:"p"},"#[warn(lint_rule)]")," untuk memunculkan warning untuk suatu ",(0,i.kt)("em",{parentName:"p"},"lint rule")," yang ",(0,i.kt)("em",{parentName:"p"},"default"),"-nya tidak memunculkan warning.",(0,i.kt)("br",null),"List ",(0,i.kt)("inlineCode",{parentName:"p"},"lint_rule")," bisa dilihat di ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/rustc/lints/listing/warn-by-default.html"},"https://doc.rust-lang.org/rustc/lints/listing/warn-by-default.html"),".")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},(0,i.kt)("inlineCode",{parentName:"p"},"#[deny(lint_rule)]")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"#[forbid(lint_rule)]")," untuk melarang suatu ",(0,i.kt)("em",{parentName:"p"},"lint rule")," yang ",(0,i.kt)("em",{parentName:"p"},"default"),"-nya adalah diperbolehkan.",(0,i.kt)("br",null),"List ",(0,i.kt)("inlineCode",{parentName:"p"},"lint_rule")," bisa dilihat di ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/rustc/lints/listing/deny-by-default.html"},"https://doc.rust-lang.org/rustc/lints/listing/deny-by-default.html"),"."))),(0,i.kt)("p",null,"Selain 3 attribute di atas, ada juga beberapa attribute lainnya untuk keperluan ",(0,i.kt)("em",{parentName:"p"},"diagnostic"),", di antaranya:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[deprecated]")," digunakan untuk menandai bahwa kode di bawahnya adalah ",(0,i.kt)("em",{parentName:"li"},"deprecated"),"."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[must_use]")," digunakan untuk mendandai bahwa kode di bawahnya harus digunakan, jika tidak maka akan muncul error.")),(0,i.kt)("h2",{id:"a495-attribute-type-system"},"A.49.5. Attribute ",(0,i.kt)("em",{parentName:"h2"},"type system")),(0,i.kt)("p",null,"Ada sebuah attribute bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"non_exhaustive")," gunanya untuk mem-",(0,i.kt)("em",{parentName:"p"},"bypass")," error yang muncul karena ada pattern matching yang tidak meng-cover semua kondisi, atau untuk mengantisipasi error yang muncul saat deklarasi variabel bertipe struct tapi value property-nya tidak diisi."),(0,i.kt)("p",null,"Salah satu contoh error yang dimaksud bisa dilihat pada kode berikut. Error ini muncul karena enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero::Superhero")," tidak ter-cover dalam pattern matching."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(7812).Z,width:"671",height:"669"})),(0,i.kt)("p",null,"Solusi untuk mengatasi error di atas bisa dengan cukup menambahkan case kondisi yang belum ter-cover:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'match value {\n Superhero::Superman => println!("stronk"),\n Superhero::OmniMan => println!("stronk"),\n Superhero::Hyperion => println!("stronk"),\n}\n')),(0,i.kt)("p",null,"Atau dengan memanfaatkan kondisi ",(0,i.kt)("em",{parentName:"p"},"other")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"_"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'match value {\n Superhero::Superman => println!("stronk"),\n Superhero::OmniMan => println!("stronk"),\n _ => println!("stronk"),\n}\n')),(0,i.kt)("p",null,"Atau, bisa dengan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"non_exhaustive"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[non_exhaustive]\npub enum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n\nfn main() {\n let value = Superhero::Superman;\n\n match value {\n Superhero::Superman => println!("stronk"),\n Superhero::OmniMan => println!("stronk"),\n }\n}\n')),(0,i.kt)("p",null,"Sayangnya dalam penggunaan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"non_exhaustive")," ini, efeknya ",(0,i.kt)("strong",{parentName:"p"},"hanya bisa dirasakan jika digunakan pada enum atau struct yang berbeda crate"),"."),(0,i.kt)("p",null,"Pada contoh di atas, tempat di mana enum dideklarasikan dan digunakan adalah masih dalam satu crate yang sama, jadi kode tetap menghasilkan error."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"non_exhaustive")," ini jika digunakan pada struct efeknya saat deklarasi variabel boleh tidak menuliskan value property.")),(0,i.kt)("h2",{id:"a496-attribute-modules"},"A.49.6. Attribute ",(0,i.kt)("em",{parentName:"h2"},"modules")),(0,i.kt)("p",null,"Aturan manajemen di Rust cukup ketat, dan sudah dibahas secara mendetal pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/module-basic"},"Module System \u279c Module"),", yang intinya adalah ada dua cara pembuatan module:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Cara ke-1: dengan mendefinisikan module pada file bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"nama_module.rs")),(0,i.kt)("li",{parentName:"ul"},"Cara ke-2: dengan mendefinisikan module pada file bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"nama_module/mod.rs"))),(0,i.kt)("p",null,"Rust memiliki sebuah attribute bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"path")," yang berguna untuk meng-override 2 aturan di atas secara paksa. Dengan memanfaatkan attribute ini kita bisa menulis module dengan nama sesuka hati."),(0,i.kt)("p",null,"Mari kita praktekan agar lebih jelas. Silakan buat package baru dengan struktur seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 util1.rs\n \u2502\u2500\u2500\u2500 util2\n \u2502 \u2514\u2500\u2500\u2500 mod.rs\n \u2514\u2500\u2500\u2500 util3_mymodule.rs\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="util1.rs"',title:'"util1.rs"'},'pub fn say_hello() {\n println!("hello (from util1)")\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="util2/mod.rs"',title:'"util2/mod.rs"'},'pub fn say_hello() {\n println!("hello (from util2)")\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="util3_mymodule.rs"',title:'"util3_mymodule.rs"'},'pub fn say_hello() {\n println!("hello (from util3_mymodule)")\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="main.rs"',title:'"main.rs"'},'mod util1;\nmod util2;\n\n#[path = "util3_mymodule.rs"]\nmod util3;\n\nfn main() {\n util1::say_hello();\n util2::say_hello();\n util3::say_hello();\n}\n')),(0,i.kt)("p",null,"Pada kode di atas bisa dilihat, module ",(0,i.kt)("inlineCode",{parentName:"p"},"util1")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"util2")," patuh mengikuti aturan deklarasi module system. Berbeda dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"util3")," yang sebenarnya dideklarasikan dengan nama ",(0,i.kt)("inlineCode",{parentName:"p"},"util3_mymodule")," (karena filename-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"util3_mymodule.rs"),"). Nama module satu ini diubah menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"util3")," lalu dengan memanfaatkan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"path")," kita arahkan isi module ",(0,i.kt)("inlineCode",{parentName:"p"},"util3")," adalah berada di ",(0,i.kt)("inlineCode",{parentName:"p"},"util3_mymodule.rs"),"."),(0,i.kt)("p",null,"Jialankan program, harusnya tidak ada error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(2259).Z,width:"456",height:"147"})),(0,i.kt)("h2",{id:"a497-attribute-testing"},"A.49.7. Attribute ",(0,i.kt)("em",{parentName:"h2"},"testing")),(0,i.kt)("p",null,"Lebih detailnya mengenai attribute testing dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/testing"},"Testing"),"."),(0,i.kt)("h2",{id:"a498-attribute-macros"},"A.49.8. Attribute ",(0,i.kt)("em",{parentName:"h2"},"macros")),(0,i.kt)("p",null,"Lebih detailnya mengenai attribute macros dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/macro"},"Macro"),"."),(0,i.kt)("h2",{id:"a499-attribute-lainnya"},"A.49.9 Attribute lainnya"),(0,i.kt)("p",null,"Rust memiliki cukup banyak attribute yang list-nya bisa dilihat pada link ini ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/reference/attributes.html"},"https://doc.rust-lang.org/reference/attributes.html"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/attributes"},"github.com/novalagung/dasarpemrogramanrust-example/../attributes")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/module-basic"},"Module System \u279c Module")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/enum"},"Enum")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/attributes.html"},"https://doc.rust-lang.org/reference/attributes.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/items/implementations.html"},"https://doc.rust-lang.org/reference/items/implementations.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rustc/lints/index.html"},"https://doc.rust-lang.org/rustc/lints/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/attributes/diagnostics.html"},"https://doc.rust-lang.org/reference/attributes/diagnostics.html"))))}k.isMDXComponent=!0},253:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/attribute-1-c3fd3abaf784b196c8b4da4512bb8abd.png"},7095:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/attribute-2-935bd54d9113cbcfb6a5c07a7ebd9f68.png"},635:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},2329:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/attribute-4-0ae9faf44e5f0113b2599d985a7f331d.png"},7964:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},7812:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/attribute-6-edad9c816a2c847561bcc988d0905b16.png"},2259:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/5f289fb7.e0fc9497.js b/assets/js/5f289fb7.e0fc9497.js deleted file mode 100644 index 6aafca39..00000000 --- a/assets/js/5f289fb7.e0fc9497.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[2870],{3905:(a,e,t)=>{t.d(e,{Zo:()=>d,kt:()=>o});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var p=n.createContext({}),m=function(a){var e=n.useContext(p),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},d=function(a){var e=m(a.components);return n.createElement(p.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},s=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,p=a.parentName,d=u(a,["components","mdxType","originalType","parentName"]),s=m(t),o=i,b=s["".concat(p,".").concat(o)]||s[o]||k[o]||r;return t?n.createElement(b,l(l({ref:e},d),{},{components:t})):n.createElement(b,l({ref:e},d))}));function o(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=s;var u={};for(var p in e)hasOwnProperty.call(e,p)&&(u[p]=e[p]);u.originalType=a,u.mdxType="string"==typeof a?a:i,l[1]=u;for(var m=2;m{t.r(e),t.d(e,{assets:()=>p,contentTitle:()=>l,default:()=>k,frontMatter:()=>r,metadata:()=>u,toc:()=>m});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:49,title:"A.49. Attributes",sidebar_label:"A.49. Attributes"},l=void 0,u={unversionedId:"basic/attributes",id:"basic/attributes",title:"A.49. Attributes",description:"Kita telah beberapa kali menggunakan attributes pada chapter-chapter sebelumnya, contohnya seperti #[derive(Debug)]. Pada chapter ini kita akan bahas tentang apa sebenarnya attributes, macam-macam jenisnya, beserta kegunaannya.",source:"@site/docs/basic/attributes.md",sourceDirName:"basic",slug:"/basic/attributes",permalink:"/basic/attributes",draft:!1,tags:[],version:"current",sidebarPosition:49,frontMatter:{sidebar_position:49,title:"A.49. Attributes",sidebar_label:"A.49. Attributes"},sidebar:"tutorialSidebar",previous:{title:"A.48. Trait \u279c Iterator",permalink:"/basic/trait-iterator"},next:{title:"A.50. Date Time",permalink:"/basic/datetime"}},p={},m=[{value:"A.49.1. Konsep attributes",id:"a491-konsep-attributes",level:2},{value:"A.49.2. Attribute derive",id:"a492-attribute-derive",level:2},{value:"A.49.3. Attribute cfg / configuration",id:"a493-attribute-cfg--configuration",level:2},{value:"\u25c9 Configuration target_os",id:"-configuration-target_os",level:3},{value:"\u25c9 Configuration target_arch",id:"-configuration-target_arch",level:3},{value:"\u25c9 Other configuration",id:"-other-configuration",level:3},{value:"A.49.4. Attribute linting & diagnostic",id:"a494-attribute-linting--diagnostic",level:2},{value:"A.49.5. Attribute type system",id:"a495-attribute-type-system",level:2},{value:"A.49.6. Attribute modules",id:"a496-attribute-modules",level:2},{value:"A.49.7. Attribute testing",id:"a497-attribute-testing",level:2},{value:"A.49.8. Attribute macros",id:"a498-attribute-macros",level:2},{value:"A.49.9 Attribute lainnya",id:"a499-attribute-lainnya",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function k(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},d,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Kita telah beberapa kali menggunakan attributes pada chapter-chapter sebelumnya, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"#[derive(Debug)]"),". Pada chapter ini kita akan bahas tentang apa sebenarnya attributes, macam-macam jenisnya, beserta kegunaannya."),(0,i.kt)("h2",{id:"a491-konsep-attributes"},"A.49.1. Konsep attributes"),(0,i.kt)("p",null,"Attributes adalah metadata yang didefinisikan untuk suatu crate, module, atau module item. Kegunaan dari attributes berbeda satu sama lain, tergantung attribute apa yang dipakai (kita akan bahas satu per satu)."),(0,i.kt)("p",null,"Attribute dikategorikan menjadi 2:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Outer attributes"),(0,i.kt)("li",{parentName:"ul"},"Inner attributes")),(0,i.kt)("p",null,"Keduanya memiliki kegunaan yang sama, pembedanya adalah posisi dimana attribute harus dituliskan."),(0,i.kt)("p",null,"Outer attribute dituliskan tepat sebelum target (crate, module, module item, atau lainnya) dengan notasi penulisan seperti berikut:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},'#[attribute = "value"]')),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},'#[attribute(key = "value")]')),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[attribute(value)]"))),(0,i.kt)("p",null,"Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"#[derive(Debug)]\nstruct LegoSet {\n code: i32,\n name: String,\n category: String,\n age_minimum: i32,\n}\n")),(0,i.kt)("p",null,"Sedikit berbeda dengan inner attribute, penulisannya berada didalam target (crate, module, module item, atau lainnya). Notasi penulisannya:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},'#![attribute = "value"]')),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},'#![attribute(key = "value")]')),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#![attribute(value)]"))),(0,i.kt)("p",null,"Rust mengenal beberapa jenis attributes, dan kita akan membahasnya satu per satu."),(0,i.kt)("h2",{id:"a492-attribute-derive"},"A.49.2. Attribute ",(0,i.kt)("inlineCode",{parentName:"h2"},"derive")),(0,i.kt)("p",null,"Attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," digunakan untuk mempermudah implementasi suatu trait ke tipe data."),(0,i.kt)("p",null,"Kita telah mempelajari cara implementasi trait pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits")," yaitu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"for"),", kemudian diikuti dengan method serta implementasinya."),(0,i.kt)("p",null,"Dengan memanfaatkan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," kita tidak perlu menggunakan cara tersebut. Cukup tulis saja attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," beserta trait yang ingin diimplementasikan."),(0,i.kt)("p",null,"Agar lebih jelas, silakan pelajari contoh berikut terlebih dahulu:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'const SuperheroSuperman: &str = "superman";\nconst SuperheroOmniMan: &str = "omniman";\nconst SuperheroHyperion: &str = "hyperion";\n\nenum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n\nfn main() {\n let value: Superhero = Superhero::Superman;\n\n if value == Superhero::Superman {\n println!("hello superman!");\n }\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(253).Z,width:"958",height:"719"})),(0,i.kt)("p",null,"Kode di atas menghasilkan error karena enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero")," tidak mengadopsi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"PartialEq")," yang dimana trait ini diperlukan dalam seleksi kondisi menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," dan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"=="),"."),(0,i.kt)("p",null,"Cara mengatasi error tersebut adalah dengan mengimplementasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"PartialEq")," secara eksplisit. Sekarang coba tambahkan kode berikut pada deklarasi enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero"),", maka error akan hilang."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"enum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n\nimpl PartialEq for Superhero {\n\n fn eq(&self, other: &Self) -> bool {\n match (self, other) {\n (Superhero::Superman, Superhero::Superman) => true,\n (Superhero::OmniMan, Superhero::OmniMan) => true,\n (Superhero::Hyperion, Superhero::Hyperion) => true,\n _ => false,\n }\n }\n}\n")),(0,i.kt)("p",null,"Cara lain untuk mengatasi error di atas, selain menggunakan teknik implementasi trait secara eksplisit, adalah dengan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," disertai trait ",(0,i.kt)("inlineCode",{parentName:"p"},"PartialEq"),". Kurang lebih penulisan cara ini bisa dilihat di bawah ini. Cukup hapus block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," lalu tambahkan attribute pada deklarasi enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"#[derive(PartialEq)]\nenum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n")),(0,i.kt)("p",null,"Cukup mudah bukan?"),(0,i.kt)("p",null,"Contoh lainnya, misalnya variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"value")," ingin di-print seperti pada kode berikut ini, pasti hasilnya error."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let value: Superhero = Superhero::Superman;\n\n if value == Superhero::Superman {\n println!("hello superman!");\n }\n\n println!("{value} (via `Display` trait)");\n println!("{value:#?} (via `Debug` trait)");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(7095).Z,width:"900",height:"461"})),(0,i.kt)("p",null,"Cara ",(0,i.kt)("em",{parentName:"p"},"resolve")," error di atas adalah dengan mengimplementasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Display")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," secara eksplisit. Atau, bisa juga menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive")," yang pastinya lebih praktis."),(0,i.kt)("p",null,"Kabar buruknya, hanya trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," yang menyediakan fitur implementasi trait menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive"),". Untuk trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Display")," kita perlu melakukan implementasi secara eksplisit."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Untuk tau mana trait yang bisa di-",(0,i.kt)("em",{parentName:"p"},"derive")," atau tidak, cukup lihat saja ",(0,i.kt)("em",{parentName:"p"},"highlight")," error yang muncul saat penulisan kode atau kompilasi.")),(0,i.kt)("p",null,"Ok, kita implementasikan saja keduanya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(PartialEq, Debug)]\nenum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n\nimpl std::fmt::Display for Superhero {\n\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "{self:?}")\n }\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(635).Z,width:"537",height:"127"})),(0,i.kt)("p",null,"Bisa dilihat, hasilnya program tereksekusi tanpa error. Enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero")," kini mengadopsi 3 trait:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"PartialEq")," via attribute ",(0,i.kt)("inlineCode",{parentName:"li"},"derive")),(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Debug")," via attribute ",(0,i.kt)("inlineCode",{parentName:"li"},"derive")),(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Display")," via implementasi eksplisit")),(0,i.kt)("h2",{id:"a493-attribute-cfg--configuration"},"A.49.3. Attribute ",(0,i.kt)("inlineCode",{parentName:"h2"},"cfg")," / ",(0,i.kt)("em",{parentName:"h2"},"configuration")),(0,i.kt)("p",null,"Attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg")," digunakan untuk menentukan beberapa konfigurasi yang berhubungan dengan arsitekture hardware/prosesor."),(0,i.kt)("p",null,"Salah satu contoh penerapannya bisa dilihat pada kode berikut. Ada 2 buah module yang namanya sama persis, perbedaannya adalah satu didefinisikan khusus untuk platform ",(0,i.kt)("inlineCode",{parentName:"p"},"linux"),", dan satunya lagi untuk platform ",(0,i.kt)("inlineCode",{parentName:"p"},"windows"),". Hal seperti ini bisa dilakukan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg")," dengan key ",(0,i.kt)("inlineCode",{parentName:"p"},"target_os"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[cfg(target_os = "linux")]\nmod util {\n\n pub fn say_hello() {\n println!("hello (from linux)")\n }\n}\n\n#[cfg(target_os = "windows")]\nmod util {\n\n pub fn say_hello() {\n println!("hello (from windows)")\n }\n\n pub fn say_something() {\n println!("how are you")\n }\n}\n')),(0,i.kt)("p",null,"Bisa dilihat cara penulisannya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg()")," kemudian diisi key ",(0,i.kt)("inlineCode",{parentName:"p"},"target_os")," dengan value adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"windows")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"linux"),"."),(0,i.kt)("p",null,"Dengan kondisi kode seperti di atas, ketika berada di sistem operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"linux"),", item ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_hello")," bisa diakses. Sedangkan pada sistem operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"windows"),", item ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_hello")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_something")," bisa diakses."),(0,i.kt)("p",null,"Attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg(target_os)")," juga bisa diterapkan pada block kode. Contohnya seperti pada kode berikut. Item ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_hello")," dipanggil di fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),". Dan khusus untuk sistem operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"windows"),", block kode berisi pemanggilan ",(0,i.kt)("inlineCode",{parentName:"p"},"util::say_something")," juga dipanggil."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n util::say_hello();\n\n #[cfg(target_os = "windows")]\n {\n util::say_something();\n }\n}\n')),(0,i.kt)("p",null,"Ada beberapa key yang tersedia pada attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"cfg"),", diantaranya:"),(0,i.kt)("h3",{id:"-configuration-target_os"},"\u25c9 Configuration ",(0,i.kt)("inlineCode",{parentName:"h3"},"target_os")),(0,i.kt)("p",null,"Digunakan untuk menandai bahwa item atau statement dibawah definisi attribute ini dikhususkan untuk sistem operasi tertentu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[cfg(target_os = "value")]\n')),(0,i.kt)("p",null,"Opsi value yang tersedia:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"windows")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"macos")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"ios")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"linux")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"android")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"freebsd")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"dragonfly")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"openbsd")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"netbsd"))),(0,i.kt)("h3",{id:"-configuration-target_arch"},"\u25c9 Configuration ",(0,i.kt)("inlineCode",{parentName:"h3"},"target_arch")),(0,i.kt)("p",null,"Digunakan untuk menandai bahwa item atau statement dibawah definisi attribute ini dikhususkan untuk arsitektur CPU tertentu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[cfg(target_arch = "value")]\n')),(0,i.kt)("p",null,"Opsi value yang tersedia:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"x86")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"x86_64")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"mips")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"powerpc")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"powerpc64")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"arm")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"aarch64"))),(0,i.kt)("h3",{id:"-other-configuration"},"\u25c9 Other configuration"),(0,i.kt)("p",null,"Ada beberapa key konfigurasi lainnya yang tersedia. Lebih detailnya silakan lihat di ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/reference/conditional-compilation.html#set-configuration-options"},"https://doc.rust-lang.org/reference/conditional-compilation.html#set-configuration-options"),"."),(0,i.kt)("h2",{id:"a494-attribute-linting--diagnostic"},"A.49.4. Attribute ",(0,i.kt)("em",{parentName:"h2"},"linting")," & ",(0,i.kt)("em",{parentName:"h2"},"diagnostic")),(0,i.kt)("p",null,"Ada beberapa attribute name yang bisa digunakan untuk meng-override ",(0,i.kt)("em",{parentName:"p"},"default linting")," milik Rust ataupun menandai indikator ",(0,i.kt)("em",{parentName:"p"},"diagnostic")," lainnya, seperti warning yang muncul karena ada kode yang tidak digunakan, dll; Warning sejenis ini bisa di-override menggunakan attribute."),(0,i.kt)("p",null,"Contoh kasus yang berhubungan dengan ",(0,i.kt)("em",{parentName:"p"},"linting")," bisa dilihat pada kode berikut."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(2329).Z,width:"351",height:"363"})),(0,i.kt)("p",null,"Kode di atas tidak menghasilkan error. Kode akan dieksekusi tanpa error. Namun ada 3 buah warning yang muncul karena beberapa baris kode tidak digunakan atau sia-sia."),(0,i.kt)("p",null,"Cara agar warning tidak muncul bisa dengan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"#[allow(value)]"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[allow(unused_imports)]\nuse std::fmt::Display;\n\nfn say_hello() {\n println!("hello")\n}\n\n#[allow(dead_code)]\nfn say_something() {\n println!("how are you")\n}\n\nfn main() {\n #[allow(unused_variables)]\n let name = "noval agung";\n\n say_hello();\n}\n')),(0,i.kt)("p",null,"Pada kode di atas, attribute name ",(0,i.kt)("inlineCode",{parentName:"p"},"allow")," digunakan pada 3 tempat:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[allow(unused_imports)]")," digunakan untuk antisipasi error yang muncul ketika module item di-import namun tidak digunakan."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[allow(dead_code)]")," digunakan untuk membolehkan kode yang tidak digunakan."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[allow(unused_variables)]")," digunakan untuk membolehkan variabel yang didefinisikan tapi tidak dimanfaatkan.")),(0,i.kt)("p",null,"Dengan penambahan 3 attribute di atas program akan tereksekusi tanpa warning."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(7964).Z,width:"463",height:"107"})),(0,i.kt)("p",null,"Ada beberapa attribute ",(0,i.kt)("em",{parentName:"p"},"key")," yang bisa digunakan untuk override ",(0,i.kt)("em",{parentName:"p"},"lint")," warning:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},(0,i.kt)("inlineCode",{parentName:"p"},"#[allow(lint_rule)]")," untuk membolehkan suatu ",(0,i.kt)("em",{parentName:"p"},"lint rule"),".",(0,i.kt)("br",null),"List ",(0,i.kt)("inlineCode",{parentName:"p"},"lint_rule")," bisa dilihat di ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/rustc/lints/listing/allowed-by-default.html"},"https://doc.rust-lang.org/rustc/lints/listing/allowed-by-default.html"),".")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},(0,i.kt)("inlineCode",{parentName:"p"},"#[warn(lint_rule)]")," untuk memunculkan warning untuk suatu ",(0,i.kt)("em",{parentName:"p"},"lint rule")," yang ",(0,i.kt)("em",{parentName:"p"},"default"),"-nya tidak memunculkan warning.",(0,i.kt)("br",null),"List ",(0,i.kt)("inlineCode",{parentName:"p"},"lint_rule")," bisa dilihat di ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/rustc/lints/listing/warn-by-default.html"},"https://doc.rust-lang.org/rustc/lints/listing/warn-by-default.html"),".")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},(0,i.kt)("inlineCode",{parentName:"p"},"#[deny(lint_rule)]")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"#[forbid(lint_rule)]")," untuk melarang suatu ",(0,i.kt)("em",{parentName:"p"},"lint rule")," yang ",(0,i.kt)("em",{parentName:"p"},"default"),"-nya adalah diperbolehkan.",(0,i.kt)("br",null),"List ",(0,i.kt)("inlineCode",{parentName:"p"},"lint_rule")," bisa dilihat di ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/rustc/lints/listing/deny-by-default.html"},"https://doc.rust-lang.org/rustc/lints/listing/deny-by-default.html"),"."))),(0,i.kt)("p",null,"Selain 3 attribute di atas, ada juga beberapa attribute lainnya untuk keperluan ",(0,i.kt)("em",{parentName:"p"},"diagnostic"),", diantaranya:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[deprecated]")," digunakan untuk menandai bahwa kode dibawahnya adalah ",(0,i.kt)("em",{parentName:"li"},"deprecated"),"."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"#[must_use]")," digunakan untuk mendandai bahwa kode dibawahnya harus digunakan, jika tidak maka akan muncul error.")),(0,i.kt)("h2",{id:"a495-attribute-type-system"},"A.49.5. Attribute ",(0,i.kt)("em",{parentName:"h2"},"type system")),(0,i.kt)("p",null,"Ada sebuah attribute bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"non_exhaustive")," gunanya untuk mem-",(0,i.kt)("em",{parentName:"p"},"bypass")," error yang muncul karena ada pattern matching yang tidak meng-cover semua kondisi, atau untuk mengantisipasi error yang muncul saat deklarasi variabel bertipe struct tapi value property-nya tidak diisi."),(0,i.kt)("p",null,"Salah satu contoh error yang dimaksud bisa dilihat pada kode berikut. Error ini muncul karena enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Superhero::Superhero")," tidak ter-cover dalam pattern matching."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(7812).Z,width:"671",height:"669"})),(0,i.kt)("p",null,"Solusi untuk mengatasi error di atas bisa dengan cukup menambahkan case kondisi yang belum ter-cover:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'match value {\n Superhero::Superman => println!("stronk"),\n Superhero::OmniMan => println!("stronk"),\n Superhero::Hyperion => println!("stronk"),\n}\n')),(0,i.kt)("p",null,"Atau dengan memanfaatkan kondisi ",(0,i.kt)("em",{parentName:"p"},"other")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"_"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'match value {\n Superhero::Superman => println!("stronk"),\n Superhero::OmniMan => println!("stronk"),\n _ => println!("stronk"),\n}\n')),(0,i.kt)("p",null,"Atau, bisa dengan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"non_exhaustive"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[non_exhaustive]\npub enum Superhero {\n Superman,\n OmniMan,\n Hyperion,\n}\n\nfn main() {\n let value = Superhero::Superman;\n\n match value {\n Superhero::Superman => println!("stronk"),\n Superhero::OmniMan => println!("stronk"),\n }\n}\n')),(0,i.kt)("p",null,"Sayangnya dalam penggunaan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"non_exhaustive")," ini, efeknya ",(0,i.kt)("strong",{parentName:"p"},"hanya bisa dirasakan jika digunakan pada enum atau struct yang berbeda crate"),"."),(0,i.kt)("p",null,"Pada contoh di atas, tempat dimana enum dideklarasikan dan digunakan adalah masih dalam satu crate yang sama, jadi kode tetap menghasilkan error."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"non_exhaustive")," ini jika digunakan pada struct efeknya saat deklarasi variabel boleh tidak menuliskan value property.")),(0,i.kt)("h2",{id:"a496-attribute-modules"},"A.49.6. Attribute ",(0,i.kt)("em",{parentName:"h2"},"modules")),(0,i.kt)("p",null,"Aturan manajemen di Rust cukup ketat, dan sudah dibahas secara mendetal pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/module-basic"},"Module System \u279c Module"),", yang intinya adalah ada dua cara pembuatan module:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Cara ke-1: dengan mendefinisikan module pada file bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"nama_module.rs")),(0,i.kt)("li",{parentName:"ul"},"Cara ke-2: dengan mendefinisikan module pada file bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"nama_module/mod.rs"))),(0,i.kt)("p",null,"Rust memiliki sebuah attribute bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"path")," yang berguna untuk meng-override 2 aturan di atas secara paksa. Dengan memanfaatkan attribute ini kita bisa menulis module dengan nama sesuka hati."),(0,i.kt)("p",null,"Mari kita praktekan agar lebih jelas. Silakan buat package baru dengan struktur seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2502\u2500\u2500\u2500 util1.rs\n \u2502\u2500\u2500\u2500 util2\n \u2502 \u2514\u2500\u2500\u2500 mod.rs\n \u2514\u2500\u2500\u2500 util3_mymodule.rs\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="util1.rs"',title:'"util1.rs"'},'pub fn say_hello() {\n println!("hello (from util1)")\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="util2/mod.rs"',title:'"util2/mod.rs"'},'pub fn say_hello() {\n println!("hello (from util2)")\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="util3_mymodule.rs"',title:'"util3_mymodule.rs"'},'pub fn say_hello() {\n println!("hello (from util3_mymodule)")\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="main.rs"',title:'"main.rs"'},'mod util1;\nmod util2;\n\n#[path = "util3_mymodule.rs"]\nmod util3;\n\nfn main() {\n util1::say_hello();\n util2::say_hello();\n util3::say_hello();\n}\n')),(0,i.kt)("p",null,"Pada kode di atas bisa dilihat, module ",(0,i.kt)("inlineCode",{parentName:"p"},"util1")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"util2")," patuh mengikuti aturan deklarasi module system. Berbeda dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"util3")," yang sebenarnya dideklarasikan dengan nama ",(0,i.kt)("inlineCode",{parentName:"p"},"util3_mymodule")," (karena filename-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"util3_mymodule.rs"),"). Nama module satu ini diubah menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"util3")," lalu dengan memanfaatkan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"path")," kita arahkan isi module ",(0,i.kt)("inlineCode",{parentName:"p"},"util3")," adalah berada di ",(0,i.kt)("inlineCode",{parentName:"p"},"util3_mymodule.rs"),"."),(0,i.kt)("p",null,"Jialankan program, harusnya tidak ada error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Attribute",src:t(2259).Z,width:"456",height:"147"})),(0,i.kt)("h2",{id:"a497-attribute-testing"},"A.49.7. Attribute ",(0,i.kt)("em",{parentName:"h2"},"testing")),(0,i.kt)("p",null,"Lebih detailnya mengenai attribute testing dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/testing"},"Testing"),"."),(0,i.kt)("h2",{id:"a498-attribute-macros"},"A.49.8. Attribute ",(0,i.kt)("em",{parentName:"h2"},"macros")),(0,i.kt)("p",null,"Lebih detailnya mengenai attribute macros dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/macro"},"Macro"),"."),(0,i.kt)("h2",{id:"a499-attribute-lainnya"},"A.49.9 Attribute lainnya"),(0,i.kt)("p",null,"Rust memiliki cukup banyak attribute yang list-nya bisa dilihat pada link ini ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/reference/attributes.html"},"https://doc.rust-lang.org/reference/attributes.html"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/attributes"},"github.com/novalagung/dasarpemrogramanrust-example/../attributes")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/module-basic"},"Module System \u279c Module")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/enum"},"Enum")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/attributes.html"},"https://doc.rust-lang.org/reference/attributes.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/items/implementations.html"},"https://doc.rust-lang.org/reference/items/implementations.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rustc/lints/index.html"},"https://doc.rust-lang.org/rustc/lints/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/attributes/diagnostics.html"},"https://doc.rust-lang.org/reference/attributes/diagnostics.html"))))}k.isMDXComponent=!0},253:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/attribute-1-c3fd3abaf784b196c8b4da4512bb8abd.png"},7095:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/attribute-2-935bd54d9113cbcfb6a5c07a7ebd9f68.png"},635:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},2329:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/attribute-4-0ae9faf44e5f0113b2599d985a7f331d.png"},7964:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},7812:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/attribute-6-edad9c816a2c847561bcc988d0905b16.png"},2259:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/62b99870.4801cdae.js b/assets/js/62b99870.4801cdae.js deleted file mode 100644 index 0d143b72..00000000 --- a/assets/js/62b99870.4801cdae.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[1422],{3905:(a,e,n)=>{n.d(e,{Zo:()=>m,kt:()=>c});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function r(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var s=t.createContext({}),d=function(a){var e=t.useContext(s),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},m=function(a){var e=d(a.components);return t.createElement(s.Provider,{value:e},a.children)},A={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},u=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,l=a.originalType,s=a.parentName,m=p(a,["components","mdxType","originalType","parentName"]),u=d(n),c=i,b=u["".concat(s,".").concat(c)]||u[c]||A[c]||l;return n?t.createElement(b,r(r({ref:e},m),{},{components:n})):t.createElement(b,r({ref:e},m))}));function c(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var l=n.length,r=new Array(l);r[0]=u;var p={};for(var s in e)hasOwnProperty.call(e,s)&&(p[s]=e[s]);p.originalType=a,p.mdxType="string"==typeof a?a:i,r[1]=p;for(var d=2;d{n.r(e),n.d(e,{assets:()=>s,contentTitle:()=>r,default:()=>A,frontMatter:()=>l,metadata:()=>p,toc:()=>d});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:14,title:"A.14. Slice (Basic)",sidebar_label:"A.14. Slice (Basic)"},r=void 0,p={unversionedId:"basic/slice",id:"basic/slice",title:"A.14. Slice (Basic)",description:"Pada bab ini kita belajar tentang apa itu slice, apa perbedaan slice dan array, slicing atau borrowing pada slice, dan juga slice mutability.",source:"@site/docs/basic/slice.md",sourceDirName:"basic",slug:"/basic/slice",permalink:"/basic/slice",draft:!1,tags:[],version:"current",sidebarPosition:14,frontMatter:{sidebar_position:14,title:"A.14. Slice (Basic)",sidebar_label:"A.14. Slice (Basic)"},sidebar:"tutorialSidebar",previous:{title:"A.13. Array",permalink:"/basic/array"},next:{title:"A.15. Tuple",permalink:"/basic/tuple"}},s={},d=[{value:"A.14.1. Tipe data slice",id:"a141-tipe-data-slice",level:2},{value:"A.14.2. Size slice",id:"a142-size-slice",level:2},{value:"A.14.3. Slice range syntax",id:"a143-slice-range-syntax",level:2},{value:"A.14.4. Mutability pada slice",id:"a144-mutability-pada-slice",level:2},{value:"A.14.5. Perulangan for in pada slice",id:"a145-perulangan-for-in-pada-slice",level:2},{value:"A.14.6. Perulangan for in pada mutable slice",id:"a146-perulangan-for-in-pada-mutable-slice",level:2},{value:"A.14.7. Append slice",id:"a147-append-slice",level:2},{value:"A.14.8. Memory management pada slice",id:"a148-memory-management-pada-slice",level:2},{value:"A.14.9. Summary",id:"a149-summary",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:d};function A(a){let{components:e,...l}=a;return(0,i.kt)("wrapper",(0,t.Z)({},m,l,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada bab ini kita belajar tentang apa itu slice, apa perbedaan slice dan array, ",(0,i.kt)("em",{parentName:"p"},"slicing")," atau ",(0,i.kt)("em",{parentName:"p"},"borrowing")," pada slice, dan juga slice mutability."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Pada chapter ini kita akan bahas slice secara garis besar saja, tanpa menyinggung masalah memory management."),(0,i.kt)("p",{parentName:"blockquote"},"Pembahasan lebih dalam perihal slice ada pada chapter terpisah (",(0,i.kt)("a",{parentName:"p",href:"/basic/slice-memory-management"},"Memory Management \u279c Slice"),"), setelah kita mempelajari dasar-dasar memory management nantinya.")),(0,i.kt)("h2",{id:"a141-tipe-data-slice"},"A.14.1. Tipe data slice"),(0,i.kt)("p",null,"Array adalah tipe data kolektif yang isinya bertipe sejenis. Contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},'["a", "b", "c"]')," adalah sebuah array dengan elemen bertipe string dan array tersebut memiliki size ",(0,i.kt)("inlineCode",{parentName:"p"},"3"),"."),(0,i.kt)("p",null,"Lalu apa itu slice? menurut laman dokumentasi Rust, slice adalah:"),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"A slice is a dynamically sized type representing a 'view' into a sequence of elements of type T. The slice type is written as ","[T]"),(0,i.kt)("p",{parentName:"blockquote"},"A dynamically-sized view into a contiguous sequence, ","[T]",". Contiguous here means that elements are laid out so that every element is the same distance from its neighbors."),(0,i.kt)("p",{parentName:"blockquote"},"Slices are a view into a block of memory represented as a pointer and a size.")),(0,i.kt)("p",null,"Slice adalah representasi ",(0,i.kt)("em",{parentName:"p"},"block of memory")," berbentuk pointer dan memiliki size yang dinamis (tidak fixed seperti array). Notasi tipe data slice adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"&[T]")," dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah tipe data element."),(0,i.kt)("p",null,"Slice bisa dibuat dari data array (atau dari tipe kolektif data lainnya) dengan menggunakan kombinasi operator ",(0,i.kt)("inlineCode",{parentName:"p"},"&")," dan ",(0,i.kt)("em",{parentName:"p"},"range syntax")," ",(0,i.kt)("inlineCode",{parentName:"p"},"..")," dengan notasi penulisan seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let sliced_value1 = &data[start_index..end_index]\nlet sliced_value2 = &data[start_index..=end_index]\nlet sliced_value3 = &data[start_index..end_index]\n...\n")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Slice juga bisa dibuat dari tipe data vector. Lebih jelasnya dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector"),".")),(0,i.kt)("p",null,"Silakan perhatikan contoh berikut dan praktikkan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let numbers = [12, 16, 8, 3];\nprintln!("numbers : {:?}, len: {}", numbers, numbers.len());\nprintln!("numbers[0]: {:?}", numbers[0]);\nprintln!("numbers[1]: {:?}", numbers[1]);\n\nlet slice_a = &numbers[0..3];\nprintln!("slice_a : {:?}, len: {}", slice_a, slice_a.len());\nprintln!("slice_a[0]: {:?}", slice_a[0]);\nprintln!("slice_a[1]: {:?}", slice_a[1]);\n\nlet slice_b = &slice_a[1..=2];\nprintln!("slice_b : {:?}, len: {}", slice_b, slice_b.len());\nprintln!("slice_b[0]: {:?}", slice_b[0]);\nprintln!("slice_b[1]: {:?}", slice_b[1]);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(8493).Z,width:"463",height:"256"})),(0,i.kt)("p",null,"Pada contoh di atas, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," didefinisikan bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"[i32; 4]"),". Data pada variabel tersebut kemudian dipinjam ditampung pada variabel baru beranama ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," yang merupakan slice bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&[i32]"),", dengan isi adalah element array ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," indeks ke ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," hingga sebelum ",(0,i.kt)("inlineCode",{parentName:"p"},"3")," (yang berarti index ke-",(0,i.kt)("inlineCode",{parentName:"p"},"2"),"). Dengan ini maka ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"[12, 16, 8]")," dengan size ",(0,i.kt)("inlineCode",{parentName:"p"},"3"),"."),(0,i.kt)("p",null,"Bisa dilihat pada statement print ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a[0]")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a[1]"),", nilai elemennya sesuai dengan dengan hasil peminjaman data array ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Data yang tipenya ",(0,i.kt)("inlineCode",{parentName:"p"},"&[T]")," biasa disebut sebagai ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/reference/types/slice.html"},(0,i.kt)("em",{parentName:"a"},"shared slice"))," atau cukup ",(0,i.kt)("em",{parentName:"p"},"slice"),". Contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," dengan tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"&[i32]"),".")),(0,i.kt)("p",null,"Slice ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," bukan merupakan pemilik sebenarnya data ",(0,i.kt)("inlineCode",{parentName:"p"},"[12, 16, 8]"),", slice tersebut hanya meminjam datanya dari ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," yang notabene dalah owner data ",(0,i.kt)("inlineCode",{parentName:"p"},"[12, 16, 8]"),"."),(0,i.kt)("p",null,"Meminjam di sini artinya variabel baru ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," memiliki data yang sama, dan alamat memori (atau pointer) data tersebut juga sama yaitu mengarah ke owner sebenarnya, yang pada contoh ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Di Rust, proses meminjam data secara umum disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"borrowing"),". Variabel atau data hasil dari borrowing biasa disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"reference"),". Lebih jelasnya akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References")," dan ",(0,i.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Variabel pemilik data yang sebenarnya disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"owner"),". Lebih jelasnya mengenai ownership dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership")),(0,i.kt)("p",{parentName:"blockquote"},"Dalam konteks slice, proses meminjam data (yg menggunakan teknik borrowing dan range syntax) disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"slicing"),".")),(0,i.kt)("p",null,"Dengan menggunakan VSCode ",(0,i.kt)("inlineCode",{parentName:"p"},"rust-analyzer")," kita bisa melihat dengan mudah tipe sebuah slice."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(2462).Z,width:"540",height:"346"})),(0,i.kt)("p",null,"Kembali ke pembahasan pada contoh di atas. Selain ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a"),", ada juga slice ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_b")," yang isinya merupakan data pinjaman dari ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a"),". Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"&slice_a[1..=2]")," artinya adalah ",(0,i.kt)("em",{parentName:"p"},"borrowing")," slice ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," mulai dari indeks ke-",(0,i.kt)("inlineCode",{parentName:"p"},"1")," hingga ",(0,i.kt)("inlineCode",{parentName:"p"},"2"),". Dengan ini indeks ke-1 milik ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," menjadi indeks ke-0 milik ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_b"),", ... dan seterusnya. Hasilnya, nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_b")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"[16, 8]")," dengan size ",(0,i.kt)("inlineCode",{parentName:"p"},"2"),"."),(0,i.kt)("p",null,"Silakan juga cek penjelasan tambahan berikut, agar semakin paham tentang slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let numbers = [12, 16, 8, 3];\n// variabel numbers isinya array [12, 16, 8, 3]\n\nlet slice_a = &numbers[0..3];\n// meminjam data milik numbers elemen ke-0 hingga sebelum 3 (yaitu 2)\n// hasilnya adalah [12, 16, 8]\n\nlet slice_b = &slice_a[1..=2];\n// meminjam data milik slice_a elemen ke-1 hingga 2\n// hasilnya adalah [16, 8]\n")),(0,i.kt)("h2",{id:"a142-size-slice"},"A.14.2. Size slice"),(0,i.kt)("p",null,"Gunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"len")," untuk mencari tau size dari slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let numbers = [12, 16, 8, 3];\nprintln!("{}", numbers.len());\n// output: 4\n\nlet slice_a = &numbers[0..3];\nprintln!("{}", slice_a.len());\n// output: 3\n\nlet slice_b = &slice_a[1..=2];\nprintln!("{}", slice_b.len());\n// output: 2\n')),(0,i.kt)("h2",{id:"a143-slice-range-syntax"},"A.14.3. Slice ",(0,i.kt)("em",{parentName:"h2"},"range syntax")),(0,i.kt)("p",null,"Di bawah ini adalah range syntax yang bisa digunakan untuk slicing. Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data")," pada contoh berikut dijadikan sebagai bahan slicing."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data = ["a", "b", "c", "d"];\n')),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[start_index..end_index]")," untuk slicing data dari ",(0,i.kt)("inlineCode",{parentName:"p"},"start_index")," hingga sebelum ",(0,i.kt)("inlineCode",{parentName:"p"},"end_index")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[1..3];\nprintln!("{:?}", sliced_data);\n// output => ["b", "c"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[start_index..=end_index]")," untuk slicing data dari ",(0,i.kt)("inlineCode",{parentName:"p"},"start_index")," hingga ",(0,i.kt)("inlineCode",{parentName:"p"},"end_index")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[1..=3];\nprintln!("{:?}", sliced_data);\n// output => ["b", "c", "d"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[..end_index]")," untuk slicing data dari 0 hingga sebelum ",(0,i.kt)("inlineCode",{parentName:"p"},"end_index")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[..3];\nprintln!("{:?}", sliced_data);\n// output => ["a", "b", "c"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[..=end_index]")," untuk slicing data dari 0 hingga ",(0,i.kt)("inlineCode",{parentName:"p"},"end_index")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[..=2];\nprintln!("{:?}", sliced_data);\n// output => ["a", "b", "c"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[start_index..]")," untuk slicing data dari ",(0,i.kt)("inlineCode",{parentName:"p"},"start_index")," hingga indeks terakhir"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[1..];\nprintln!("{:?}", sliced_data);\n// output => ["b", "c", "d"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[..]")," untuk slicing semua elemen yang ada"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[..];\nprintln!("{:?}", sliced_data);\n// output => ["A", "b", "c", "d"]\n')))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Perihal apa itu borrowing dan kegunaan dari operator ",(0,i.kt)("inlineCode",{parentName:"p"},"&")," yang ditulis di awal variabel dibahas lebih detail pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),".")),(0,i.kt)("h2",{id:"a144-mutability-pada-slice"},"A.14.4. Mutability pada slice"),(0,i.kt)("p",null,"Ada dua jenis data hasil operasi ",(0,i.kt)("em",{parentName:"p"},"borrowing")," (atau biasa disebut dengan data ",(0,i.kt)("em",{parentName:"p"},"reference"),")."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Read only atau shared reference, operator yang digunakan adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"&"),"."),(0,i.kt)("li",{parentName:"ul"},"Mutable reference, operator yang digunakan adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"&mut"),".")),(0,i.kt)("p",null,"Kita akan bahas garis besarnya saja pada chapter ini. Intinya, shared reference adalah data hasil peminjaman/borrowing yang hanya bisa dibaca. Sedangkan mutable reference adalah data hasil borrowing yang bisa diubah nilainya, yang jika ini dilakukan maka akan mengubah juga data pemilik sebenarnya."),(0,i.kt)("p",null,"Contoh berikut adalah salah satu penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut"),". Variabel slice ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers2")," dipinjam beberapa elemennya dengan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")," ke variabel baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_e"),". Dengan operator tersebut maka borrowing menghasilkan data ",(0,i.kt)("em",{parentName:"p"},"mutable reference"),", data yang nilainya diperbolehkan untuk diubah meskipun data pinjaman."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut numbers2 = [12, 16, 8, 3];\nprintln!("===== before =====");\nprintln!("numbers2 : {:?}", numbers2);\n\nlet slice_e = &mut numbers2[..=2];\nslice_e[1] = 99;\n\nprintln!("===== after =====");\nprintln!("slice_e : {:?}", slice_e);\nprintln!("numbers2 : {:?}", numbers2);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(7858).Z,width:"457",height:"189"})),(0,i.kt)("p",null,"Bisa dilihat, indeks ke-1 ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_e")," diubah nilainya yang sebelumnya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"16")," menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"99"),". Setelah itu di-print, di output nilai variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers2")," juga ikut berubah. Hal ini karena perubahan data pada variabel mutable reference juga punya pengaruh ke variabel pemilik data sebenarnya (yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers2"),")."),(0,i.kt)("p",null,"O iya, penggunaan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")," mengharuskan kita untuk tidak menuliskan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," pada variabel yang menampung nilai borrowing."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// statement yang direkomendasikan\nlet slice_e = &mut numbers2[..=2];\n\n// statement yang TIDAK DIREKOMENDASIKAN dan akan memunculkan warning.\nlet mut slice_e = &mut numbers2[..=2];\n")),(0,i.kt)("h2",{id:"a145-perulangan-for-in-pada-slice"},"A.14.5. Perulangan ",(0,i.kt)("inlineCode",{parentName:"h2"},"for in")," pada slice"),(0,i.kt)("p",null,"Slice merupakan tipe data yang implement trait Iterator (seperti array), dan semua data yang memiliki trait tersebut bisa digunakan pada perulangan. Slice ",(0,i.kt)("inlineCode",{parentName:"p"},"&[T]")," jika dipergunakan dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", tipe data penampung iterasi perulangan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"&T"),", bukan ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," yaa jadi jangan sampai keliru."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let scores1 = [7, 8, 9];\n\nfor score in &scores1[..] {\n print!("{:?} ", score);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(6760).Z,width:"462",height:"272"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Sangat dianjurkan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"rust-analyzer")," jika menggunakan VSCode, agar lebih mudah melihat tipe data variabel dan informasi penting lainnya yang berguna dalam proses coding atau development."),(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Iterator")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/trait-iterator"},"Trait \u279c Iterator")))),(0,i.kt)("h2",{id:"a146-perulangan-for-in-pada-mutable-slice"},"A.14.6. Perulangan ",(0,i.kt)("inlineCode",{parentName:"h2"},"for in")," pada mutable slice"),(0,i.kt)("p",null,"Sama seperti penggunaan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," pada shared slice, pada contoh kasus mutable slice perbedaannya hanyalah pada tipe data penampung iterasi yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut T"),", bukan ",(0,i.kt)("inlineCode",{parentName:"p"},"&T"),"."),(0,i.kt)("p",null,"Pada contoh berikut kita coba eksperimen mengubah nilai elemen sebuah mutable slice yang data aslinya juga hasil dari peminjaman mutable slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut scores2 = [7, 8, 9];\nprintln!("(before) scores2 : {:?}", scores2);\n\nlet slice_f = &mut scores2[..];\n\nfor score in &mut slice_f[..] {\n *score += 1;\n}\n\nprintln!("(after) scores2 : {:?}", scores2);\n')),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"scores")," dipinjam menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")," ke variabel baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_f"),". Kemudian ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_f")," dipinjam juga sebagai mutable slice pada perulangan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),"."),(0,i.kt)("p",null,"Setelah di-increment nilainya menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"*score += 1")," bisa dilihat hasil akhirnya juga mengubah nilai variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"scores2")," yang merupakan ",(0,i.kt)("em",{parentName:"p"},"owner")," atau pemilik data sebenarnya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(4820).Z,width:"467",height:"376"})),(0,i.kt)("p",null,"Operasi increment pada ",(0,i.kt)("inlineCode",{parentName:"p"},"score")," tidak bisa dituliskan dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"score += 1")," karena tipe data score adalah pointer mutable reference (ditandai dengan adanya ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut"),"), untuk increment nilainya perlu di-",(0,i.kt)("em",{parentName:"p"},"dereference")," terlebih dahulu menggunakan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"*"),". Lebih jelasnya kita bahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),"."),(0,i.kt)("h2",{id:"a147-append-slice"},"A.14.7. Append slice"),(0,i.kt)("p",null,"Slice by default tidak mendukung operasi append. Sebenarnya bisa saja dilakukan tapi agak panjang caranya (silakan cari referensinya di Google untuk ini)."),(0,i.kt)("p",null,"Perlu diketahui, jika kebutuhannya adalah untuk menampung jenis data yang size-nya bisa bertamah, penulis anjurkan untuk menggunakan ",(0,i.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector")," yang beberapa bab lagi akan kita bahas."),(0,i.kt)("h2",{id:"a148-memory-management-pada-slice"},"A.14.8. Memory management pada slice"),(0,i.kt)("p",null,"Nantinya setelah selesai dengan pembahasan dasar memory management di Rust, kita akan bahas lagi topik slice tapi dari sudut pandang memory management pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/slice-memory-management"},"Memory Management \u279c Slice"),"."),(0,i.kt)("h2",{id:"a149-summary"},"A.14.9. Summary"),(0,i.kt)("p",null,"Catatan ringkas perihal slice:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Slice memiliki notasi ",(0,i.kt)("inlineCode",{parentName:"li"},"&[T]"),(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"&")," disitu artinya adalah operasi borrowing/peminjaman"),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"T")," adalah tipe data tiap elemen"))),(0,i.kt)("li",{parentName:"ul"},"Slice bisa terbentuk dari hasil meminjam data array, vector, atau tipe data kolektif lainnya"),(0,i.kt)("li",{parentName:"ul"},"Data slice adalah selalu data pinjaman"),(0,i.kt)("li",{parentName:"ul"},"Slice memiliki lebar/size"),(0,i.kt)("li",{parentName:"ul"},"Slicing adalah cara pengaksesan slice menggunakan ",(0,i.kt)("em",{parentName:"li"},"range syntax")),(0,i.kt)("li",{parentName:"ul"},"Slice bisa immutable, bisa juga mutable (menggunakan ",(0,i.kt)("inlineCode",{parentName:"li"},"&mut"),")")),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/slice"},"github.com/novalagung/dasarpemrogramanrust-example/../slice")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-03-slices.html#the-slice-type"},"https://doc.rust-lang.org/book/ch04-03-slices.html#the-slice-type")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.slice.html"},"https://doc.rust-lang.org/std/primitive.slice.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/trait.Iterator.html"},"https://doc.rust-lang.org/std/iter/trait.Iterator.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/types/slice.html"},"https://doc.rust-lang.org/reference/types/slice.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/nomicon/references.html"},"https://doc.rust-lang.org/nomicon/references.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/primitives/array.html"},"https://doc.rust-lang.org/rust-by-example/primitives/array.html"))))}A.isMDXComponent=!0},8493:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},2462:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/slice-2-f573e34bdab45ca8821163ae546a42ba.png"},7858:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},6760:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/slice-5-541f1e4111281ebbac474d8d9828c0e8.png"},4820:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/slice-6-393c62bd8cdd9da05fcae56436a3cacd.png"}}]); \ No newline at end of file diff --git a/assets/js/62b99870.eac41936.js b/assets/js/62b99870.eac41936.js new file mode 100644 index 00000000..f2ce815f --- /dev/null +++ b/assets/js/62b99870.eac41936.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[1422],{3905:(a,e,n)=>{n.d(e,{Zo:()=>m,kt:()=>c});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function r(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var s=t.createContext({}),d=function(a){var e=t.useContext(s),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},m=function(a){var e=d(a.components);return t.createElement(s.Provider,{value:e},a.children)},A={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},u=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,l=a.originalType,s=a.parentName,m=p(a,["components","mdxType","originalType","parentName"]),u=d(n),c=i,b=u["".concat(s,".").concat(c)]||u[c]||A[c]||l;return n?t.createElement(b,r(r({ref:e},m),{},{components:n})):t.createElement(b,r({ref:e},m))}));function c(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var l=n.length,r=new Array(l);r[0]=u;var p={};for(var s in e)hasOwnProperty.call(e,s)&&(p[s]=e[s]);p.originalType=a,p.mdxType="string"==typeof a?a:i,r[1]=p;for(var d=2;d{n.r(e),n.d(e,{assets:()=>s,contentTitle:()=>r,default:()=>A,frontMatter:()=>l,metadata:()=>p,toc:()=>d});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:14,title:"A.14. Slice (Basic)",sidebar_label:"A.14. Slice (Basic)"},r=void 0,p={unversionedId:"basic/slice",id:"basic/slice",title:"A.14. Slice (Basic)",description:"Pada bab ini kita belajar tentang apa itu slice, apa perbedaan slice dan array, slicing atau borrowing pada slice, dan juga slice mutability.",source:"@site/docs/basic/slice.md",sourceDirName:"basic",slug:"/basic/slice",permalink:"/basic/slice",draft:!1,tags:[],version:"current",sidebarPosition:14,frontMatter:{sidebar_position:14,title:"A.14. Slice (Basic)",sidebar_label:"A.14. Slice (Basic)"},sidebar:"tutorialSidebar",previous:{title:"A.13. Array",permalink:"/basic/array"},next:{title:"A.15. Tuple",permalink:"/basic/tuple"}},s={},d=[{value:"A.14.1. Tipe data slice",id:"a141-tipe-data-slice",level:2},{value:"A.14.2. Size slice",id:"a142-size-slice",level:2},{value:"A.14.3. Slice range syntax",id:"a143-slice-range-syntax",level:2},{value:"A.14.4. Mutability pada slice",id:"a144-mutability-pada-slice",level:2},{value:"A.14.5. Perulangan for in pada slice",id:"a145-perulangan-for-in-pada-slice",level:2},{value:"A.14.6. Perulangan for in pada mutable slice",id:"a146-perulangan-for-in-pada-mutable-slice",level:2},{value:"A.14.7. Append slice",id:"a147-append-slice",level:2},{value:"A.14.8. Memory management pada slice",id:"a148-memory-management-pada-slice",level:2},{value:"A.14.9. Summary",id:"a149-summary",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:d};function A(a){let{components:e,...l}=a;return(0,i.kt)("wrapper",(0,t.Z)({},m,l,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada bab ini kita belajar tentang apa itu slice, apa perbedaan slice dan array, ",(0,i.kt)("em",{parentName:"p"},"slicing")," atau ",(0,i.kt)("em",{parentName:"p"},"borrowing")," pada slice, dan juga slice mutability."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Pada chapter ini kita akan bahas slice secara garis besar saja, tanpa menyinggung masalah memory management."),(0,i.kt)("p",{parentName:"blockquote"},"Pembahasan lebih dalam perihal slice ada pada chapter terpisah (",(0,i.kt)("a",{parentName:"p",href:"/basic/slice-memory-management"},"Memory Management \u279c Slice"),"), setelah kita mempelajari dasar-dasar memory management nantinya.")),(0,i.kt)("h2",{id:"a141-tipe-data-slice"},"A.14.1. Tipe data slice"),(0,i.kt)("p",null,"Array adalah tipe data kolektif yang isinya bertipe sejenis. Contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},'["a", "b", "c"]')," adalah sebuah array dengan elemen bertipe string dan array tersebut memiliki size ",(0,i.kt)("inlineCode",{parentName:"p"},"3"),"."),(0,i.kt)("p",null,"Lalu apa itu slice? menurut laman dokumentasi Rust, slice adalah:"),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"A slice is a dynamically sized type representing a 'view' into a sequence of elements of type T. The slice type is written as ","[T]"),(0,i.kt)("p",{parentName:"blockquote"},"A dynamically-sized view into a contiguous sequence, ","[T]",". Contiguous here means that elements are laid out so that every element is the same distance from its neighbors."),(0,i.kt)("p",{parentName:"blockquote"},"Slices are a view into a block of memory represented as a pointer and a size.")),(0,i.kt)("p",null,"Slice adalah representasi ",(0,i.kt)("em",{parentName:"p"},"block of memory")," berbentuk pointer dan memiliki size yang dinamis (tidak fixed seperti array). Notasi tipe data slice adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"&[T]")," yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah tipe data element."),(0,i.kt)("p",null,"Slice bisa dibuat dari data array (atau dari tipe kolektif data lainnya) dengan menggunakan kombinasi operator ",(0,i.kt)("inlineCode",{parentName:"p"},"&")," dan ",(0,i.kt)("em",{parentName:"p"},"range syntax")," ",(0,i.kt)("inlineCode",{parentName:"p"},"..")," dengan notasi penulisan seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let sliced_value1 = &data[start_index..end_index]\nlet sliced_value2 = &data[start_index..=end_index]\nlet sliced_value3 = &data[start_index..end_index]\n...\n")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Slice juga bisa dibuat dari tipe data vector. Lebih jelasnya dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector"),".")),(0,i.kt)("p",null,"Silakan perhatikan contoh berikut dan praktikkan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let numbers = [12, 16, 8, 3];\nprintln!("numbers : {:?}, len: {}", numbers, numbers.len());\nprintln!("numbers[0]: {:?}", numbers[0]);\nprintln!("numbers[1]: {:?}", numbers[1]);\n\nlet slice_a = &numbers[0..3];\nprintln!("slice_a : {:?}, len: {}", slice_a, slice_a.len());\nprintln!("slice_a[0]: {:?}", slice_a[0]);\nprintln!("slice_a[1]: {:?}", slice_a[1]);\n\nlet slice_b = &slice_a[1..=2];\nprintln!("slice_b : {:?}, len: {}", slice_b, slice_b.len());\nprintln!("slice_b[0]: {:?}", slice_b[0]);\nprintln!("slice_b[1]: {:?}", slice_b[1]);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(8493).Z,width:"463",height:"256"})),(0,i.kt)("p",null,"Pada contoh di atas, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," didefinisikan bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"[i32; 4]"),". Data pada variabel tersebut kemudian dipinjam ditampung pada variabel baru beranama ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," yang merupakan slice bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&[i32]"),", dengan isi adalah element array ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," indeks ke ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," hingga sebelum ",(0,i.kt)("inlineCode",{parentName:"p"},"3")," (yang berarti index ke-",(0,i.kt)("inlineCode",{parentName:"p"},"2"),"). Dengan ini maka ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"[12, 16, 8]")," dengan size ",(0,i.kt)("inlineCode",{parentName:"p"},"3"),"."),(0,i.kt)("p",null,"Bisa dilihat pada statement print ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a[0]")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a[1]"),", nilai elemennya sesuai dengan dengan hasil peminjaman data array ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Data yang tipenya ",(0,i.kt)("inlineCode",{parentName:"p"},"&[T]")," biasa disebut sebagai ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/reference/types/slice.html"},(0,i.kt)("em",{parentName:"a"},"shared slice"))," atau cukup ",(0,i.kt)("em",{parentName:"p"},"slice"),". Contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," dengan tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"&[i32]"),".")),(0,i.kt)("p",null,"Slice ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," bukan merupakan pemilik sebenarnya data ",(0,i.kt)("inlineCode",{parentName:"p"},"[12, 16, 8]"),", slice tersebut hanya meminjam datanya dari ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," yang notabene dalah owner data ",(0,i.kt)("inlineCode",{parentName:"p"},"[12, 16, 8]"),"."),(0,i.kt)("p",null,"Meminjam di sini artinya variabel baru ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," memiliki data yang sama, dan alamat memori (atau pointer) data tersebut juga sama yaitu mengarah ke owner sebenarnya, yang pada contoh ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Di Rust, proses meminjam data secara umum disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"borrowing"),". Variabel atau data hasil dari borrowing biasa disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"reference"),". Lebih jelasnya akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References")," dan ",(0,i.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Variabel pemilik data yang sebenarnya disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"owner"),". Lebih jelasnya mengenai ownership dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership")),(0,i.kt)("p",{parentName:"blockquote"},"Dalam konteks slice, proses meminjam data (yg menggunakan teknik borrowing dan range syntax) disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"slicing"),".")),(0,i.kt)("p",null,"Dengan menggunakan VSCode ",(0,i.kt)("inlineCode",{parentName:"p"},"rust-analyzer")," kita bisa melihat dengan mudah tipe sebuah slice."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(2462).Z,width:"540",height:"346"})),(0,i.kt)("p",null,"Kembali ke pembahasan pada contoh di atas. Selain ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a"),", ada juga slice ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_b")," yang isinya merupakan data pinjaman dari ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a"),". Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"&slice_a[1..=2]")," artinya adalah ",(0,i.kt)("em",{parentName:"p"},"borrowing")," slice ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," mulai dari indeks ke-",(0,i.kt)("inlineCode",{parentName:"p"},"1")," hingga ",(0,i.kt)("inlineCode",{parentName:"p"},"2"),". Dengan ini indeks ke-1 milik ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_a")," menjadi indeks ke-0 milik ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_b"),", ... dan seterusnya. Hasilnya, nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_b")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"[16, 8]")," dengan size ",(0,i.kt)("inlineCode",{parentName:"p"},"2"),"."),(0,i.kt)("p",null,"Silakan juga cek penjelasan tambahan berikut, agar semakin paham tentang slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let numbers = [12, 16, 8, 3];\n// variabel numbers isinya array [12, 16, 8, 3]\n\nlet slice_a = &numbers[0..3];\n// meminjam data milik numbers elemen ke-0 hingga sebelum 3 (yaitu 2)\n// hasilnya adalah [12, 16, 8]\n\nlet slice_b = &slice_a[1..=2];\n// meminjam data milik slice_a elemen ke-1 hingga 2\n// hasilnya adalah [16, 8]\n")),(0,i.kt)("h2",{id:"a142-size-slice"},"A.14.2. Size slice"),(0,i.kt)("p",null,"Gunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"len")," untuk mencari tau size dari slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let numbers = [12, 16, 8, 3];\nprintln!("{}", numbers.len());\n// output: 4\n\nlet slice_a = &numbers[0..3];\nprintln!("{}", slice_a.len());\n// output: 3\n\nlet slice_b = &slice_a[1..=2];\nprintln!("{}", slice_b.len());\n// output: 2\n')),(0,i.kt)("h2",{id:"a143-slice-range-syntax"},"A.14.3. Slice ",(0,i.kt)("em",{parentName:"h2"},"range syntax")),(0,i.kt)("p",null,"Di bawah ini adalah range syntax yang bisa digunakan untuk slicing. Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data")," pada contoh berikut dijadikan sebagai bahan slicing."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data = ["a", "b", "c", "d"];\n')),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[start_index..end_index]")," untuk slicing data dari ",(0,i.kt)("inlineCode",{parentName:"p"},"start_index")," hingga sebelum ",(0,i.kt)("inlineCode",{parentName:"p"},"end_index")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[1..3];\nprintln!("{:?}", sliced_data);\n// output => ["b", "c"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[start_index..=end_index]")," untuk slicing data dari ",(0,i.kt)("inlineCode",{parentName:"p"},"start_index")," hingga ",(0,i.kt)("inlineCode",{parentName:"p"},"end_index")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[1..=3];\nprintln!("{:?}", sliced_data);\n// output => ["b", "c", "d"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[..end_index]")," untuk slicing data dari 0 hingga sebelum ",(0,i.kt)("inlineCode",{parentName:"p"},"end_index")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[..3];\nprintln!("{:?}", sliced_data);\n// output => ["a", "b", "c"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[..=end_index]")," untuk slicing data dari 0 hingga ",(0,i.kt)("inlineCode",{parentName:"p"},"end_index")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[..=2];\nprintln!("{:?}", sliced_data);\n// output => ["a", "b", "c"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[start_index..]")," untuk slicing data dari ",(0,i.kt)("inlineCode",{parentName:"p"},"start_index")," hingga indeks terakhir"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[1..];\nprintln!("{:?}", sliced_data);\n// output => ["b", "c", "d"]\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"&data[..]")," untuk slicing semua elemen yang ada"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sliced_data = &data[..];\nprintln!("{:?}", sliced_data);\n// output => ["A", "b", "c", "d"]\n')))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Perihal apa itu borrowing dan kegunaan dari operator ",(0,i.kt)("inlineCode",{parentName:"p"},"&")," yang ditulis di awal variabel dibahas lebih detail pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),".")),(0,i.kt)("h2",{id:"a144-mutability-pada-slice"},"A.14.4. Mutability pada slice"),(0,i.kt)("p",null,"Ada dua jenis data hasil operasi ",(0,i.kt)("em",{parentName:"p"},"borrowing")," (atau biasa disebut dengan data ",(0,i.kt)("em",{parentName:"p"},"reference"),")."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Read only atau shared reference, operator yang digunakan adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"&"),"."),(0,i.kt)("li",{parentName:"ul"},"Mutable reference, operator yang digunakan adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"&mut"),".")),(0,i.kt)("p",null,"Kita akan bahas garis besarnya saja pada chapter ini. Intinya, shared reference adalah data hasil peminjaman/borrowing yang hanya bisa dibaca. Sedangkan mutable reference adalah data hasil borrowing yang bisa diubah nilainya, yang jika ini dilakukan maka akan mengubah juga data pemilik sebenarnya."),(0,i.kt)("p",null,"Contoh berikut adalah salah satu penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut"),". Variabel slice ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers2")," dipinjam beberapa elemennya dengan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")," ke variabel baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_e"),". Dengan operator tersebut maka borrowing menghasilkan data ",(0,i.kt)("em",{parentName:"p"},"mutable reference"),", data yang nilainya diperbolehkan untuk diubah meskipun data pinjaman."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut numbers2 = [12, 16, 8, 3];\nprintln!("===== before =====");\nprintln!("numbers2 : {:?}", numbers2);\n\nlet slice_e = &mut numbers2[..=2];\nslice_e[1] = 99;\n\nprintln!("===== after =====");\nprintln!("slice_e : {:?}", slice_e);\nprintln!("numbers2 : {:?}", numbers2);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(7858).Z,width:"457",height:"189"})),(0,i.kt)("p",null,"Bisa dilihat, indeks ke-1 ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_e")," diubah nilainya yang sebelumnya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"16")," menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"99"),". Setelah itu di-print, di output nilai variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers2")," juga ikut berubah. Hal ini karena perubahan data pada variabel mutable reference juga punya pengaruh ke variabel pemilik data sebenarnya (yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers2"),")."),(0,i.kt)("p",null,"O iya, penggunaan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")," mengharuskan kita untuk tidak menuliskan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," pada variabel yang menampung nilai borrowing."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// statement yang direkomendasikan\nlet slice_e = &mut numbers2[..=2];\n\n// statement yang TIDAK DIREKOMENDASIKAN dan akan memunculkan warning.\nlet mut slice_e = &mut numbers2[..=2];\n")),(0,i.kt)("h2",{id:"a145-perulangan-for-in-pada-slice"},"A.14.5. Perulangan ",(0,i.kt)("inlineCode",{parentName:"h2"},"for in")," pada slice"),(0,i.kt)("p",null,"Slice merupakan tipe data yang implement trait Iterator (seperti array), dan semua data yang memiliki trait tersebut bisa digunakan pada perulangan. Slice ",(0,i.kt)("inlineCode",{parentName:"p"},"&[T]")," jika dipergunakan dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", tipe data penampung iterasi perulangan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"&T"),", bukan ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," yaa jadi jangan sampai keliru."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let scores1 = [7, 8, 9];\n\nfor score in &scores1[..] {\n print!("{:?} ", score);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(6760).Z,width:"462",height:"272"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Sangat dianjurkan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"rust-analyzer")," jika menggunakan VSCode, agar lebih mudah melihat tipe data variabel dan informasi penting lainnya yang berguna dalam proses coding atau development."),(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Iterator")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/trait-iterator"},"Trait \u279c Iterator")))),(0,i.kt)("h2",{id:"a146-perulangan-for-in-pada-mutable-slice"},"A.14.6. Perulangan ",(0,i.kt)("inlineCode",{parentName:"h2"},"for in")," pada mutable slice"),(0,i.kt)("p",null,"Sama seperti penggunaan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," pada shared slice, pada contoh kasus mutable slice perbedaannya hanyalah pada tipe data penampung iterasi yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut T"),", bukan ",(0,i.kt)("inlineCode",{parentName:"p"},"&T"),"."),(0,i.kt)("p",null,"Pada contoh berikut kita coba eksperimen mengubah nilai elemen sebuah mutable slice yang data aslinya juga hasil dari peminjaman mutable slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut scores2 = [7, 8, 9];\nprintln!("(before) scores2 : {:?}", scores2);\n\nlet slice_f = &mut scores2[..];\n\nfor score in &mut slice_f[..] {\n *score += 1;\n}\n\nprintln!("(after) scores2 : {:?}", scores2);\n')),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"scores")," dipinjam menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")," ke variabel baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_f"),". Kemudian ",(0,i.kt)("inlineCode",{parentName:"p"},"slice_f")," dipinjam juga sebagai mutable slice pada perulangan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),"."),(0,i.kt)("p",null,"Setelah di-increment nilainya menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"*score += 1")," bisa dilihat hasil akhirnya juga mengubah nilai variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"scores2")," yang merupakan ",(0,i.kt)("em",{parentName:"p"},"owner")," atau pemilik data sebenarnya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"slice",src:n(4820).Z,width:"467",height:"376"})),(0,i.kt)("p",null,"Operasi increment pada ",(0,i.kt)("inlineCode",{parentName:"p"},"score")," tidak bisa dituliskan dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"score += 1")," karena tipe data score adalah pointer mutable reference (ditandai dengan adanya ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut"),"), untuk increment nilainya perlu di-",(0,i.kt)("em",{parentName:"p"},"dereference")," terlebih dahulu menggunakan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"*"),". Lebih jelasnya kita bahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),"."),(0,i.kt)("h2",{id:"a147-append-slice"},"A.14.7. Append slice"),(0,i.kt)("p",null,"Slice by default tidak mendukung operasi append. Sebenarnya bisa saja dilakukan tapi agak panjang caranya (silakan cari referensinya di Google untuk ini)."),(0,i.kt)("p",null,"Perlu diketahui, jika kebutuhannya adalah untuk menampung jenis data yang size-nya bisa bertamah, penulis anjurkan untuk menggunakan ",(0,i.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector")," yang beberapa bab lagi akan kita bahas."),(0,i.kt)("h2",{id:"a148-memory-management-pada-slice"},"A.14.8. Memory management pada slice"),(0,i.kt)("p",null,"Nantinya setelah selesai dengan pembahasan dasar memory management di Rust, kita akan bahas lagi topik slice tapi dari sudut pandang memory management pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/slice-memory-management"},"Memory Management \u279c Slice"),"."),(0,i.kt)("h2",{id:"a149-summary"},"A.14.9. Summary"),(0,i.kt)("p",null,"Catatan ringkas perihal slice:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Slice memiliki notasi ",(0,i.kt)("inlineCode",{parentName:"li"},"&[T]"),(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"&")," di situ artinya adalah operasi borrowing/peminjaman"),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"T")," adalah tipe data tiap elemen"))),(0,i.kt)("li",{parentName:"ul"},"Slice bisa terbentuk dari hasil meminjam data array, vector, atau tipe data kolektif lainnya"),(0,i.kt)("li",{parentName:"ul"},"Data slice adalah selalu data pinjaman"),(0,i.kt)("li",{parentName:"ul"},"Slice memiliki lebar/size"),(0,i.kt)("li",{parentName:"ul"},"Slicing adalah cara pengaksesan slice menggunakan ",(0,i.kt)("em",{parentName:"li"},"range syntax")),(0,i.kt)("li",{parentName:"ul"},"Slice bisa immutable, bisa juga mutable (menggunakan ",(0,i.kt)("inlineCode",{parentName:"li"},"&mut"),")")),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/slice"},"github.com/novalagung/dasarpemrogramanrust-example/../slice")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-03-slices.html#the-slice-type"},"https://doc.rust-lang.org/book/ch04-03-slices.html#the-slice-type")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.slice.html"},"https://doc.rust-lang.org/std/primitive.slice.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/trait.Iterator.html"},"https://doc.rust-lang.org/std/iter/trait.Iterator.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/types/slice.html"},"https://doc.rust-lang.org/reference/types/slice.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/nomicon/references.html"},"https://doc.rust-lang.org/nomicon/references.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/primitives/array.html"},"https://doc.rust-lang.org/rust-by-example/primitives/array.html"))))}A.isMDXComponent=!0},8493:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},2462:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/slice-2-f573e34bdab45ca8821163ae546a42ba.png"},7858:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},6760:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/slice-5-541f1e4111281ebbac474d8d9828c0e8.png"},4820:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/slice-6-393c62bd8cdd9da05fcae56436a3cacd.png"}}]); \ No newline at end of file diff --git a/assets/js/68e1906b.57a9a7b8.js b/assets/js/68e1906b.57a9a7b8.js new file mode 100644 index 00000000..ff557937 --- /dev/null +++ b/assets/js/68e1906b.57a9a7b8.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4758],{3905:(e,a,t)=>{t.d(a,{Zo:()=>o,kt:()=>c});var n=t(7294);function r(e,a,t){return a in e?Object.defineProperty(e,a,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[a]=t,e}function i(e,a){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);a&&(n=n.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),t.push.apply(t,n)}return t}function l(e){for(var a=1;a=0||(r[t]=e[t]);return r}(e,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(r[t]=e[t])}return r}var p=n.createContext({}),m=function(e){var a=n.useContext(p),t=a;return e&&(t="function"==typeof e?e(a):l(l({},a),e)),t},o=function(e){var a=m(e.components);return n.createElement(p.Provider,{value:a},e.children)},u={inlineCode:"code",wrapper:function(e){var a=e.children;return n.createElement(n.Fragment,{},a)}},s=n.forwardRef((function(e,a){var t=e.components,r=e.mdxType,i=e.originalType,p=e.parentName,o=d(e,["components","mdxType","originalType","parentName"]),s=m(t),c=r,A=s["".concat(p,".").concat(c)]||s[c]||u[c]||i;return t?n.createElement(A,l(l({ref:a},o),{},{components:t})):n.createElement(A,l({ref:a},o))}));function c(e,a){var t=arguments,r=a&&a.mdxType;if("string"==typeof e||r){var i=t.length,l=new Array(i);l[0]=s;var d={};for(var p in a)hasOwnProperty.call(a,p)&&(d[p]=a[p]);d.originalType=e,d.mdxType="string"==typeof e?e:r,l[1]=d;for(var m=2;m{t.r(a),t.d(a,{assets:()=>p,contentTitle:()=>l,default:()=>u,frontMatter:()=>i,metadata:()=>d,toc:()=>m});var n=t(7462),r=(t(7294),t(3905));const i={sidebar_position:25,title:"A.25. Method",sidebar_label:"A.25. Method"},l=void 0,d={unversionedId:"basic/method",id:"basic/method",title:"A.25. Method",description:"Pada chapter ini kita akan belajar tentang method beserta perbedaannya dengan associated function.",source:"@site/docs/basic/method.md",sourceDirName:"basic",slug:"/basic/method",permalink:"/basic/method",draft:!1,tags:[],version:"current",sidebarPosition:25,frontMatter:{sidebar_position:25,title:"A.25. Method",sidebar_label:"A.25. Method"},sidebar:"tutorialSidebar",previous:{title:"A.24. Associated Function",permalink:"/basic/associated-function"},next:{title:"A.26. Enum",permalink:"/basic/enum"}},p={},m=[{value:"A.25.1. Method vs associated function",id:"a251-method-vs-associated-function",level:2},{value:"A.25.2. Deklarasi method",id:"a252-deklarasi-method",level:2},{value:"\u25c9 Naming convention method",id:"-naming-convention-method",level:3},{value:"A.25.3. Method parameter",id:"a253-method-parameter",level:2},{value:"A.25.4. Struct property mutability via method",id:"a254-struct-property-mutability-via-method",level:2},{value:"A.25.5. Generic pada method",id:"a255-generic-pada-method",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:m};function u(e){let{components:a,...i}=e;return(0,r.kt)("wrapper",(0,n.Z)({},o,i,{components:a,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Pada chapter ini kita akan belajar tentang method beserta perbedaannya dengan ",(0,r.kt)("em",{parentName:"p"},"associated function"),"."),(0,r.kt)("h2",{id:"a251-method-vs-associated-function"},"A.25.1. Method vs ",(0,r.kt)("em",{parentName:"h2"},"associated function")),(0,r.kt)("p",null,"Method adalah ",(0,r.kt)("em",{parentName:"p"},"associated item")," yang hanya bisa diakses lewat instance/object, berbeda dengan ",(0,r.kt)("em",{parentName:"p"},"associated function")," yang pengaksesan fungsinya via tipe data struct."),(0,r.kt)("p",null,"Silakan lihat ilustrasi kode berikut, lalu pelajari penjelasan di bawahnya untuk mencari tau perbedaan ",(0,r.kt)("em",{parentName:"p"},"associated function")," vs method."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"struct Car {\n // ...\n}\n\nimpl Car {\n // ...\n}\n\n// associated function\nlet my_car: Car = Car::new();\n\n// method\nlet info: String = my_car.info();\n")),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Fungsi ",(0,r.kt)("inlineCode",{parentName:"li"},"new")," di situ adalah ",(0,r.kt)("em",{parentName:"li"},"associated function")," milik struct ",(0,r.kt)("inlineCode",{parentName:"li"},"Car"),". Dengannya object baru bernama ",(0,r.kt)("inlineCode",{parentName:"li"},"my_car")," bertipe ",(0,r.kt)("inlineCode",{parentName:"li"},"Car")," dibuat."),(0,r.kt)("li",{parentName:"ul"},"Object ",(0,r.kt)("inlineCode",{parentName:"li"},"my_car")," adalah variabel bertipe ",(0,r.kt)("inlineCode",{parentName:"li"},"Car"),". Via object tersebut method bernama ",(0,r.kt)("inlineCode",{parentName:"li"},"info")," diakses."),(0,r.kt)("li",{parentName:"ul"},"Method ",(0,r.kt)("inlineCode",{parentName:"li"},"info")," tidak bisa diakses via struct ",(0,r.kt)("inlineCode",{parentName:"li"},"Car"),". Dan ",(0,r.kt)("em",{parentName:"li"},"associated function")," ",(0,r.kt)("inlineCode",{parentName:"li"},"new")," juga tidak bisa diakses dari instance/object ",(0,r.kt)("inlineCode",{parentName:"li"},"my_car"),".")),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Di bahasa pemrograman lain, ",(0,r.kt)("em",{parentName:"p"},"associated function")," disebut dengan ",(0,r.kt)("strong",{parentName:"p"},"class method"),", sedangkan method disebut dengan ",(0,r.kt)("strong",{parentName:"p"},"instance method"))),(0,r.kt)("p",null,"Pengaksesan ",(0,r.kt)("em",{parentName:"p"},"associated function")," dilakukan menggunakan notasi path dengan separator ",(0,r.kt)("inlineCode",{parentName:"p"},"::"),", contohnya seperti ",(0,r.kt)("inlineCode",{parentName:"p"},"Car::new()"),". Sedangkan pengaksesan method menggunakan separator ",(0,r.kt)("inlineCode",{parentName:"p"},"."),", contoh: ",(0,r.kt)("inlineCode",{parentName:"p"},"my_car.info()"),"."),(0,r.kt)("p",null,"Agar lebih jelas, mari lanjut ke bagian praktik. Siapkan package dengan struktur seperti berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2514\u2500\u2500\u2500 models.rs\n")),(0,r.kt)("p",null,"Buka file ",(0,r.kt)("inlineCode",{parentName:"p"},"models.rs"),", isi dengan deklarasi struct ",(0,r.kt)("inlineCode",{parentName:"p"},"Car")," berikut diikuti dengan blok kode ",(0,r.kt)("inlineCode",{parentName:"p"},"impl")," untuk associated items-nya."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"#[derive(Debug)]\npub struct Car {\n brand: String,\n model: String,\n manufacture_year: i32\n}\n\nimpl Car {\n // ...\n}\n")),(0,r.kt)("h2",{id:"a252-deklarasi-method"},"A.25.2. Deklarasi method"),(0,r.kt)("p",null,"Cara deklarasi method mirip dengan ",(0,r.kt)("em",{parentName:"p"},"associated function"),", perbedaannya adalah parameter pertama harus diisi dengan ",(0,r.kt)("inlineCode",{parentName:"p"},"&self")," pada deklarasi method. Parameter tersebut menjadi identifier apakah fungsi merupakan ",(0,r.kt)("em",{parentName:"p"},"associated function")," atau ",(0,r.kt)("em",{parentName:"p"},"method"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Object ",(0,r.kt)("inlineCode",{parentName:"p"},"self")," merupakan representasi dari ",(0,r.kt)("em",{parentName:"p"},"current instance")," atau ",(0,r.kt)("em",{parentName:"p"},"current object"),"."),(0,r.kt)("p",{parentName:"blockquote"},"Statement ",(0,r.kt)("inlineCode",{parentName:"p"},"&self")," artinya kita melakukan operasi ",(0,r.kt)("em",{parentName:"p"},"borrowing")," terhadap object ",(0,r.kt)("inlineCode",{parentName:"p"},"self"),"."),(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai borrowing dibahas pada chapter terpisah.")),(0,r.kt)("p",null,"Silakan tambahkan ",(0,r.kt)("em",{parentName:"p"},"associated function")," bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"new")," dan method bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"info")," berikut. Tulis keduanya dalam blok kode ",(0,r.kt)("inlineCode",{parentName:"p"},"impl"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},'// ...\n\nimpl Car {\n \n pub fn new(brand: String, model: String) -> Self {\n Self { brand, model, manufacture_year: 0 }\n }\n\n pub fn info(&self) -> String {\n if self.manufacture_year == 0 {\n format!("{} model {}", self.brand, self.model)\n } else {\n format!(\n "{} model {}, manufactured at {}",\n self.brand,\n self.model,\n self.manufacture_year\n )\n }\n }\n}\n')),(0,r.kt)("p",null,"Bisa dilihat, deklarasi method adalah mirip dengan fungsi biasa, perbedaannya ada pada deklarasi parameter pertama fungsi yang diisi dengan ",(0,r.kt)("inlineCode",{parentName:"p"},"&self"),"."),(0,r.kt)("p",null,"Cara mengakses property milik ",(0,r.kt)("em",{parentName:"p"},"current object")," dari dalam method adalah menggunakan keyword ",(0,r.kt)("inlineCode",{parentName:"p"},"self"),", contohnya seperti ",(0,r.kt)("inlineCode",{parentName:"p"},"self.brand"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"self.model"),", dan ",(0,r.kt)("inlineCode",{parentName:"p"},"self.manufacture_year"),", pemanggilan ketiganya adalah mengarah ke value property milik ",(0,r.kt)("em",{parentName:"p"},"current object")," (",(0,r.kt)("inlineCode",{parentName:"p"},"self"),")."),(0,r.kt)("p",null,"Selanjutnya panggil keduanya di fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod models;\n\nfn main() {\n let car = models::Car::new(\n String::from("Mercedes-Benz"),\n String::from("Vision Gran Turismo")\n );\n println!("car: {:?}", car);\n\n let info = car.info();\n println!("info: {:?}", info);\n}\n')),(0,r.kt)("p",null,"Bisa dilihat, variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"car")," adalah instance dari struct ",(0,r.kt)("inlineCode",{parentName:"p"},"models::Car"),", dibuat menggunakan ",(0,r.kt)("em",{parentName:"p"},"associated function")," ",(0,r.kt)("inlineCode",{parentName:"p"},"models::Car::new"),". Dari variabel tersebut kemudian diakses method ",(0,r.kt)("inlineCode",{parentName:"p"},"info"),"."),(0,r.kt)("p",null,"Jalankan program untuk melihat hasilnya."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Method",src:t(2693).Z,width:"626",height:"110"})),(0,r.kt)("h3",{id:"-naming-convention-method"},"\u25c9 ",(0,r.kt)("em",{parentName:"h3"},"Naming convention")," method"),(0,r.kt)("p",null,"Sesuai anjuran di ",(0,r.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", snake case digunakan dalam penamaan method. Contoh:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"impl NamaStruct {\n\n fn nama_method(&self) {\n // ...\n }\n}\n")),(0,r.kt)("h2",{id:"a253-method-parameter"},"A.25.3. Method parameter"),(0,r.kt)("p",null,"Ok, penulis rasa sudah cukup jelas perihal perbedaan ",(0,r.kt)("em",{parentName:"p"},"associated function")," dengan method. Sekarang, bagaimana dengan method yang ada parameternya? Cara deklarasinya adalah cukup dengan menuliskan parameter yang diinginkan setelah ",(0,r.kt)("inlineCode",{parentName:"p"},"&self"),"."),(0,r.kt)("p",null,"Lanjut ke praktik berikutnya. Silakan buat method baru bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"congratulate")," yang memiliki 1 buah parameter bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"String"),", dengan tugas adalah menampilkan pesan selamat."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},'// ...\n\nimpl Car {\n \n // ...\n \n pub fn congratulate(&self, name: String) {\n println!("hello {}", name);\n println!("congrats with your new car {}", self.info());\n println!("vroooom vroooooooommmmm!");\n }\n}\n')),(0,r.kt)("p",null,"Dalam method ",(0,r.kt)("inlineCode",{parentName:"p"},"congratulate")," ada statement pemanggilan method ",(0,r.kt)("inlineCode",{parentName:"p"},"info"),", yang return value-nya ikut di-print. Notasi ",(0,r.kt)("inlineCode",{parentName:"p"},"self.nama_method()")," digunakan untuk pemanggilan method dari dalam method. Mirip seperti pemanggilan property yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"self.nama_property"),"."),(0,r.kt)("p",null,"Selanjutnya, panggil method ",(0,r.kt)("inlineCode",{parentName:"p"},"congratulate")," di fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),". Isi argumen pemanggilan method dengan sebuah ",(0,r.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'// ...\n\nfn main() {\n let car = models::Car::new(\n String::from("Mercedes-Benz"),\n String::from("Vision Gran Turismo")\n );\n\n car.congratulate(String::from("Sylvanas Windrunner"));\n}\n')),(0,r.kt)("p",null,"Meskipun pada definisi method parameter pertama adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"&self"),", pada saat pemanggilan method yang menjadi parameter pertama adalah parameter setelah ",(0,r.kt)("inlineCode",{parentName:"p"},"&self")," yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"name"),"."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Method",src:t(1460).Z,width:"487",height:"128"})),(0,r.kt)("h2",{id:"a254-struct-property-mutability-via-method"},"A.25.4. Struct property mutability via method"),(0,r.kt)("p",null,"Nilai property struct bisa diubah dari dalam method. Pada bagian ini kita akan buat method baru untuk struct ",(0,r.kt)("inlineCode",{parentName:"p"},"Car")," dengan nama ",(0,r.kt)("inlineCode",{parentName:"p"},"set_manufacture_year"),". Method ini bertugas untuk melakukan perubahan nilai property ",(0,r.kt)("inlineCode",{parentName:"p"},"manufacture_year"),"."),(0,r.kt)("p",null,"Silakan tulis kode berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"// ...\n\nimpl Car {\n \n // ...\n \n pub fn set_manufacture_year(&self, year: i32) {\n self.manufacture_year = year\n }\n}\n")),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Method",src:t(2653).Z,width:"678",height:"195"})),(0,r.kt)("p",null,"Hmm, tapi kenapa terdeteksi error? Penyebabnya error tersebut adalah karena ",(0,r.kt)("strong",{parentName:"p"},"mutable reference")," tidak digunakan dalam pengaksesan current object yang padahal ada operasi ",(0,r.kt)("em",{parentName:"p"},"mutable")," atau perubahan nilai terhadap property di situ. Syntax ",(0,r.kt)("inlineCode",{parentName:"p"},"&self")," artinya operasi peminjaman object ",(0,r.kt)("inlineCode",{parentName:"p"},"self")," adalah ",(0,r.kt)("em",{parentName:"p"},"read only"),"."),(0,r.kt)("p",null,"Cara mengambil mutable reference dari object ",(0,r.kt)("inlineCode",{parentName:"p"},"self")," adalah dengan menggunakan ",(0,r.kt)("inlineCode",{parentName:"p"},"&mut self"),". Cara tersebut kurang lebih sama seperti pengambilan mutable reference dari variabel biasa."),(0,r.kt)("p",null,"Ok, sekarang kita coba modifikasi deklarasi method ",(0,r.kt)("inlineCode",{parentName:"p"},"set_manufacture_year")," menjadi seperti berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"// ...\n\nimpl Car {\n \n // ...\n \n pub fn set_manufacture_year(&mut self, year: i32) {\n self.manufacture_year = year\n }\n}\n")),(0,r.kt)("p",null,"Lalu panggil method ",(0,r.kt)("inlineCode",{parentName:"p"},"set_manufacture_year")," di fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'// ...\n\nfn main() {\n let mut car = models::Car::new(\n String::from("Mercedes-Benz"),\n String::from("Vision Gran Turismo")\n );\n\n let info = car.info();\n println!("info: {:?}", info);\n\n car.set_manufacture_year(2013);\n let detailed_info = car.info();\n println!("detailed info: {:?}", detailed_info);\n}\n')),(0,r.kt)("p",null,"Jalankan program, lihat hasilnya."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Method",src:t(7797).Z,width:"570",height:"109"})),(0,r.kt)("p",null,"Setelah ",(0,r.kt)("inlineCode",{parentName:"p"},"manufacture_year")," di-set, method ",(0,r.kt)("inlineCode",{parentName:"p"},"info")," mengembalikan pesan yang berbeda. Dari sini bisa disimpulkan bahwa method ",(0,r.kt)("inlineCode",{parentName:"p"},"set_manufacture_year")," sukses menjalankan tugasnya untuk mengubah property ",(0,r.kt)("inlineCode",{parentName:"p"},"manufacture_year"),"."),(0,r.kt)("h2",{id:"a255-generic-pada-method"},"A.25.5. Generic pada method"),(0,r.kt)("p",null,"Pembahasan mengenai generic pada method ada pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/generics#a366-generics-method"},"Generics"),"."),(0,r.kt)("hr",null),(0,r.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,r.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,r.kt)("pre",null,(0,r.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/method"},"github.com/novalagung/dasarpemrogramanrust-example/../method")),(0,r.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch05-03-method-syntax.html"},"https://doc.rust-lang.org/book/ch05-03-method-syntax.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.impl.html"},"https://doc.rust-lang.org/std/keyword.impl.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.SelfTy.html"},"https://doc.rust-lang.org/std/keyword.SelfTy.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.self.html"},"https://doc.rust-lang.org/std/keyword.self.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/fn/methods.html"},"https://doc.rust-lang.org/rust-by-example/fn/methods.html"))))}u.isMDXComponent=!0},2693:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},1460:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},2653:(e,a,t)=>{t.d(a,{Z:()=>n});const n=t.p+"assets/images/method-3-2b2f604aaaa2cd04220b106aec22800d.png"},7797:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/68e1906b.f87d9410.js b/assets/js/68e1906b.f87d9410.js deleted file mode 100644 index ea4711ed..00000000 --- a/assets/js/68e1906b.f87d9410.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4758],{3905:(e,a,t)=>{t.d(a,{Zo:()=>o,kt:()=>c});var n=t(7294);function r(e,a,t){return a in e?Object.defineProperty(e,a,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[a]=t,e}function i(e,a){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);a&&(n=n.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),t.push.apply(t,n)}return t}function l(e){for(var a=1;a=0||(r[t]=e[t]);return r}(e,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(r[t]=e[t])}return r}var p=n.createContext({}),m=function(e){var a=n.useContext(p),t=a;return e&&(t="function"==typeof e?e(a):l(l({},a),e)),t},o=function(e){var a=m(e.components);return n.createElement(p.Provider,{value:a},e.children)},u={inlineCode:"code",wrapper:function(e){var a=e.children;return n.createElement(n.Fragment,{},a)}},s=n.forwardRef((function(e,a){var t=e.components,r=e.mdxType,i=e.originalType,p=e.parentName,o=d(e,["components","mdxType","originalType","parentName"]),s=m(t),c=r,A=s["".concat(p,".").concat(c)]||s[c]||u[c]||i;return t?n.createElement(A,l(l({ref:a},o),{},{components:t})):n.createElement(A,l({ref:a},o))}));function c(e,a){var t=arguments,r=a&&a.mdxType;if("string"==typeof e||r){var i=t.length,l=new Array(i);l[0]=s;var d={};for(var p in a)hasOwnProperty.call(a,p)&&(d[p]=a[p]);d.originalType=e,d.mdxType="string"==typeof e?e:r,l[1]=d;for(var m=2;m{t.r(a),t.d(a,{assets:()=>p,contentTitle:()=>l,default:()=>u,frontMatter:()=>i,metadata:()=>d,toc:()=>m});var n=t(7462),r=(t(7294),t(3905));const i={sidebar_position:25,title:"A.25. Method",sidebar_label:"A.25. Method"},l=void 0,d={unversionedId:"basic/method",id:"basic/method",title:"A.25. Method",description:"Pada chapter ini kita akan belajar tentang method beserta perbedaannya dengan associated function.",source:"@site/docs/basic/method.md",sourceDirName:"basic",slug:"/basic/method",permalink:"/basic/method",draft:!1,tags:[],version:"current",sidebarPosition:25,frontMatter:{sidebar_position:25,title:"A.25. Method",sidebar_label:"A.25. Method"},sidebar:"tutorialSidebar",previous:{title:"A.24. Associated Function",permalink:"/basic/associated-function"},next:{title:"A.26. Enum",permalink:"/basic/enum"}},p={},m=[{value:"A.25.1. Method vs associated function",id:"a251-method-vs-associated-function",level:2},{value:"A.25.2. Deklarasi method",id:"a252-deklarasi-method",level:2},{value:"\u25c9 Naming convention method",id:"-naming-convention-method",level:3},{value:"A.25.3. Method parameter",id:"a253-method-parameter",level:2},{value:"A.25.4. Struct property mutability via method",id:"a254-struct-property-mutability-via-method",level:2},{value:"A.25.5. Generic pada method",id:"a255-generic-pada-method",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:m};function u(e){let{components:a,...i}=e;return(0,r.kt)("wrapper",(0,n.Z)({},o,i,{components:a,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Pada chapter ini kita akan belajar tentang method beserta perbedaannya dengan ",(0,r.kt)("em",{parentName:"p"},"associated function"),"."),(0,r.kt)("h2",{id:"a251-method-vs-associated-function"},"A.25.1. Method vs ",(0,r.kt)("em",{parentName:"h2"},"associated function")),(0,r.kt)("p",null,"Method adalah ",(0,r.kt)("em",{parentName:"p"},"associated item")," yang hanya bisa diakses lewat instance/object, berbeda dengan ",(0,r.kt)("em",{parentName:"p"},"associated function")," yang pengaksesan fungsinya via tipe data struct."),(0,r.kt)("p",null,"Silakan lihat ilustrasi kode berikut, lalu pelajari penjelasan dibawahnya untuk mencari tau perbedaan ",(0,r.kt)("em",{parentName:"p"},"associated function")," vs method."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"struct Car {\n // ...\n}\n\nimpl Car {\n // ...\n}\n\n// associated function\nlet my_car: Car = Car::new();\n\n// method\nlet info: String = my_car.info();\n")),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Fungsi ",(0,r.kt)("inlineCode",{parentName:"li"},"new")," disitu adalah ",(0,r.kt)("em",{parentName:"li"},"associated function")," milik struct ",(0,r.kt)("inlineCode",{parentName:"li"},"Car"),". Dengannya object baru bernama ",(0,r.kt)("inlineCode",{parentName:"li"},"my_car")," bertipe ",(0,r.kt)("inlineCode",{parentName:"li"},"Car")," dibuat."),(0,r.kt)("li",{parentName:"ul"},"Object ",(0,r.kt)("inlineCode",{parentName:"li"},"my_car")," adalah variabel bertipe ",(0,r.kt)("inlineCode",{parentName:"li"},"Car"),". Via object tersebut method bernama ",(0,r.kt)("inlineCode",{parentName:"li"},"info")," diakses."),(0,r.kt)("li",{parentName:"ul"},"Method ",(0,r.kt)("inlineCode",{parentName:"li"},"info")," tidak bisa diakses via struct ",(0,r.kt)("inlineCode",{parentName:"li"},"Car"),". Dan ",(0,r.kt)("em",{parentName:"li"},"associated function")," ",(0,r.kt)("inlineCode",{parentName:"li"},"new")," juga tidak bisa diakses dari instance/object ",(0,r.kt)("inlineCode",{parentName:"li"},"my_car"),".")),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Di bahasa pemrograman lain, ",(0,r.kt)("em",{parentName:"p"},"associated function")," disebut dengan ",(0,r.kt)("strong",{parentName:"p"},"class method"),", sedangkan method disebut dengan ",(0,r.kt)("strong",{parentName:"p"},"instance method"))),(0,r.kt)("p",null,"Pengaksesan ",(0,r.kt)("em",{parentName:"p"},"associated function")," dilakukan menggunakan notasi path dengan separator ",(0,r.kt)("inlineCode",{parentName:"p"},"::"),", contohnya seperti ",(0,r.kt)("inlineCode",{parentName:"p"},"Car::new()"),". Sedangkan pengaksesan method menggunakan separator ",(0,r.kt)("inlineCode",{parentName:"p"},"."),", contoh: ",(0,r.kt)("inlineCode",{parentName:"p"},"my_car.info()"),"."),(0,r.kt)("p",null,"Agar lebih jelas, mari lanjut ke bagian praktik. Siapkan package dengan struktur seperti berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u2514\u2500\u2500\u2500 models.rs\n")),(0,r.kt)("p",null,"Buka file ",(0,r.kt)("inlineCode",{parentName:"p"},"models.rs"),", isi dengan deklarasi struct ",(0,r.kt)("inlineCode",{parentName:"p"},"Car")," berikut diikuti dengan blok kode ",(0,r.kt)("inlineCode",{parentName:"p"},"impl")," untuk associated items-nya."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"#[derive(Debug)]\npub struct Car {\n brand: String,\n model: String,\n manufacture_year: i32\n}\n\nimpl Car {\n // ...\n}\n")),(0,r.kt)("h2",{id:"a252-deklarasi-method"},"A.25.2. Deklarasi method"),(0,r.kt)("p",null,"Cara deklarasi method mirip dengan ",(0,r.kt)("em",{parentName:"p"},"associated function"),", perbedaannya adalah parameter pertama harus diisi dengan ",(0,r.kt)("inlineCode",{parentName:"p"},"&self")," pada deklarasi method. Parameter tersebut menjadi identifier apakah fungsi merupakan ",(0,r.kt)("em",{parentName:"p"},"associated function")," atau ",(0,r.kt)("em",{parentName:"p"},"method"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Object ",(0,r.kt)("inlineCode",{parentName:"p"},"self")," merupakan representasi dari ",(0,r.kt)("em",{parentName:"p"},"current instance")," atau ",(0,r.kt)("em",{parentName:"p"},"current object"),"."),(0,r.kt)("p",{parentName:"blockquote"},"Statement ",(0,r.kt)("inlineCode",{parentName:"p"},"&self")," artinya kita melakukan operasi ",(0,r.kt)("em",{parentName:"p"},"borrowing")," terhadap object ",(0,r.kt)("inlineCode",{parentName:"p"},"self"),"."),(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai borrowing dibahas pada chapter terpisah.")),(0,r.kt)("p",null,"Silakan tambahkan ",(0,r.kt)("em",{parentName:"p"},"associated function")," bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"new")," dan method bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"info")," berikut. Tulis keduanya dalam blok kode ",(0,r.kt)("inlineCode",{parentName:"p"},"impl"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},'// ...\n\nimpl Car {\n \n pub fn new(brand: String, model: String) -> Self {\n Self { brand, model, manufacture_year: 0 }\n }\n\n pub fn info(&self) -> String {\n if self.manufacture_year == 0 {\n format!("{} model {}", self.brand, self.model)\n } else {\n format!(\n "{} model {}, manufactured at {}",\n self.brand,\n self.model,\n self.manufacture_year\n )\n }\n }\n}\n')),(0,r.kt)("p",null,"Bisa dilihat, deklarasi method adalah mirip dengan fungsi biasa, perbedaannya ada pada deklarasi parameter pertama fungsi yang diisi dengan ",(0,r.kt)("inlineCode",{parentName:"p"},"&self"),"."),(0,r.kt)("p",null,"Cara mengakses property milik ",(0,r.kt)("em",{parentName:"p"},"current object")," dari dalam method adalah menggunakan keyword ",(0,r.kt)("inlineCode",{parentName:"p"},"self"),", contohnya seperti ",(0,r.kt)("inlineCode",{parentName:"p"},"self.brand"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"self.model"),", dan ",(0,r.kt)("inlineCode",{parentName:"p"},"self.manufacture_year"),", pemanggilan ketiganya adalah mengarah ke value property milik ",(0,r.kt)("em",{parentName:"p"},"current object")," (",(0,r.kt)("inlineCode",{parentName:"p"},"self"),")."),(0,r.kt)("p",null,"Selanjutnya panggil keduanya di fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod models;\n\nfn main() {\n let car = models::Car::new(\n String::from("Mercedes-Benz"),\n String::from("Vision Gran Turismo")\n );\n println!("car: {:?}", car);\n\n let info = car.info();\n println!("info: {:?}", info);\n}\n')),(0,r.kt)("p",null,"Bisa dilihat, variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"car")," adalah instance dari struct ",(0,r.kt)("inlineCode",{parentName:"p"},"models::Car"),", dibuat menggunakan ",(0,r.kt)("em",{parentName:"p"},"associated function")," ",(0,r.kt)("inlineCode",{parentName:"p"},"models::Car::new"),". Dari variabel tersebut kemudian diakses method ",(0,r.kt)("inlineCode",{parentName:"p"},"info"),"."),(0,r.kt)("p",null,"Jalankan program untuk melihat hasilnya."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Method",src:t(2693).Z,width:"626",height:"110"})),(0,r.kt)("h3",{id:"-naming-convention-method"},"\u25c9 ",(0,r.kt)("em",{parentName:"h3"},"Naming convention")," method"),(0,r.kt)("p",null,"Sesuai anjuran di ",(0,r.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", snake case digunakan dalam penamaan method. Contoh:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},"impl NamaStruct {\n\n fn nama_method(&self) {\n // ...\n }\n}\n")),(0,r.kt)("h2",{id:"a253-method-parameter"},"A.25.3. Method parameter"),(0,r.kt)("p",null,"Ok, penulis rasa sudah cukup jelas perihal perbedaan ",(0,r.kt)("em",{parentName:"p"},"associated function")," dengan method. Sekarang, bagaimana dengan method yang ada parameternya? Cara deklarasinya adalah cukup dengan menuliskan parameter yang diinginkan setelah ",(0,r.kt)("inlineCode",{parentName:"p"},"&self"),"."),(0,r.kt)("p",null,"Lanjut ke praktik berikutnya. Silakan buat method baru bernama ",(0,r.kt)("inlineCode",{parentName:"p"},"congratulate")," yang memiliki 1 buah parameter bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"String"),", dengan tugas adalah menampilkan pesan selamat."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},'// ...\n\nimpl Car {\n \n // ...\n \n pub fn congratulate(&self, name: String) {\n println!("hello {}", name);\n println!("congrats with your new car {}", self.info());\n println!("vroooom vroooooooommmmm!");\n }\n}\n')),(0,r.kt)("p",null,"Dalam method ",(0,r.kt)("inlineCode",{parentName:"p"},"congratulate")," ada statement pemanggilan method ",(0,r.kt)("inlineCode",{parentName:"p"},"info"),", yang return value-nya ikut di-print. Notasi ",(0,r.kt)("inlineCode",{parentName:"p"},"self.nama_method()")," digunakan untuk pemanggilan method dari dalam method. Mirip seperti pemanggilan property yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"self.nama_property"),"."),(0,r.kt)("p",null,"Selanjutnya, panggil method ",(0,r.kt)("inlineCode",{parentName:"p"},"congratulate")," di fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),". Isi argumen pemanggilan method dengan sebuah ",(0,r.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'// ...\n\nfn main() {\n let car = models::Car::new(\n String::from("Mercedes-Benz"),\n String::from("Vision Gran Turismo")\n );\n\n car.congratulate(String::from("Sylvanas Windrunner"));\n}\n')),(0,r.kt)("p",null,"Meskipun pada definisi method parameter pertama adalah ",(0,r.kt)("inlineCode",{parentName:"p"},"&self"),", pada saat pemanggilan method yang menjadi parameter pertama adalah parameter setelah ",(0,r.kt)("inlineCode",{parentName:"p"},"&self")," yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"name"),"."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Method",src:t(1460).Z,width:"487",height:"128"})),(0,r.kt)("h2",{id:"a254-struct-property-mutability-via-method"},"A.25.4. Struct property mutability via method"),(0,r.kt)("p",null,"Nilai property struct bisa diubah dari dalam method. Pada bagian ini kita akan buat method baru untuk struct ",(0,r.kt)("inlineCode",{parentName:"p"},"Car")," dengan nama ",(0,r.kt)("inlineCode",{parentName:"p"},"set_manufacture_year"),". Method ini bertugas untuk melakukan perubahan nilai property ",(0,r.kt)("inlineCode",{parentName:"p"},"manufacture_year"),"."),(0,r.kt)("p",null,"Silakan tulis kode berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"// ...\n\nimpl Car {\n \n // ...\n \n pub fn set_manufacture_year(&self, year: i32) {\n self.manufacture_year = year\n }\n}\n")),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Method",src:t(2653).Z,width:"678",height:"195"})),(0,r.kt)("p",null,"Hmm, tapi kenapa terdeteksi error? Penyebabnya error tersebut adalah karena ",(0,r.kt)("strong",{parentName:"p"},"mutable reference")," tidak digunakan dalam pengaksesan current object yang padahal ada operasi ",(0,r.kt)("em",{parentName:"p"},"mutable")," atau perubahan nilai terhadap property disitu. Syntax ",(0,r.kt)("inlineCode",{parentName:"p"},"&self")," artinya operasi peminjaman object ",(0,r.kt)("inlineCode",{parentName:"p"},"self")," adalah ",(0,r.kt)("em",{parentName:"p"},"read only"),"."),(0,r.kt)("p",null,"Cara mengambil mutable reference dari object ",(0,r.kt)("inlineCode",{parentName:"p"},"self")," adalah dengan menggunakan ",(0,r.kt)("inlineCode",{parentName:"p"},"&mut self"),". Cara tersebut kurang lebih sama seperti pengambilan mutable reference dari variabel biasa."),(0,r.kt)("p",null,"Ok, sekarang kita coba modifikasi deklarasi method ",(0,r.kt)("inlineCode",{parentName:"p"},"set_manufacture_year")," menjadi seperti berikut:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"// ...\n\nimpl Car {\n \n // ...\n \n pub fn set_manufacture_year(&mut self, year: i32) {\n self.manufacture_year = year\n }\n}\n")),(0,r.kt)("p",null,"Lalu panggil method ",(0,r.kt)("inlineCode",{parentName:"p"},"set_manufacture_year")," di fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'// ...\n\nfn main() {\n let mut car = models::Car::new(\n String::from("Mercedes-Benz"),\n String::from("Vision Gran Turismo")\n );\n\n let info = car.info();\n println!("info: {:?}", info);\n\n car.set_manufacture_year(2013);\n let detailed_info = car.info();\n println!("detailed info: {:?}", detailed_info);\n}\n')),(0,r.kt)("p",null,"Jalankan program, lihat hasilnya."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Method",src:t(7797).Z,width:"570",height:"109"})),(0,r.kt)("p",null,"Setelah ",(0,r.kt)("inlineCode",{parentName:"p"},"manufacture_year")," di-set, method ",(0,r.kt)("inlineCode",{parentName:"p"},"info")," mengembalikan pesan yang berbeda. Dari sini bisa disimpulkan bahwa method ",(0,r.kt)("inlineCode",{parentName:"p"},"set_manufacture_year")," sukses menjalankan tugasnya untuk mengubah property ",(0,r.kt)("inlineCode",{parentName:"p"},"manufacture_year"),"."),(0,r.kt)("h2",{id:"a255-generic-pada-method"},"A.25.5. Generic pada method"),(0,r.kt)("p",null,"Pembahasan mengenai generic pada method ada pada chapter ",(0,r.kt)("a",{parentName:"p",href:"/basic/generics#a366-generics-method"},"Generics"),"."),(0,r.kt)("hr",null),(0,r.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,r.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,r.kt)("pre",null,(0,r.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/method"},"github.com/novalagung/dasarpemrogramanrust-example/../method")),(0,r.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch05-03-method-syntax.html"},"https://doc.rust-lang.org/book/ch05-03-method-syntax.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.impl.html"},"https://doc.rust-lang.org/std/keyword.impl.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.SelfTy.html"},"https://doc.rust-lang.org/std/keyword.SelfTy.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.self.html"},"https://doc.rust-lang.org/std/keyword.self.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/fn/methods.html"},"https://doc.rust-lang.org/rust-by-example/fn/methods.html"))))}u.isMDXComponent=!0},2693:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},1460:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},2653:(e,a,t)=>{t.d(a,{Z:()=>n});const n=t.p+"assets/images/method-3-2b2f604aaaa2cd04220b106aec22800d.png"},7797:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/71880105.761a2054.js b/assets/js/71880105.761a2054.js new file mode 100644 index 00000000..de4b45e3 --- /dev/null +++ b/assets/js/71880105.761a2054.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[5758],{3905:(A,a,n)=>{n.d(a,{Zo:()=>p,kt:()=>m});var e=n(7294);function t(A,a,n){return a in A?Object.defineProperty(A,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):A[a]=n,A}function i(A,a){var n=Object.keys(A);if(Object.getOwnPropertySymbols){var e=Object.getOwnPropertySymbols(A);a&&(e=e.filter((function(a){return Object.getOwnPropertyDescriptor(A,a).enumerable}))),n.push.apply(n,e)}return n}function l(A){for(var a=1;a=0||(t[n]=A[n]);return t}(A,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(A);for(e=0;e=0||Object.prototype.propertyIsEnumerable.call(A,n)&&(t[n]=A[n])}return t}var d=e.createContext({}),b=function(A){var a=e.useContext(d),n=a;return A&&(n="function"==typeof A?A(a):l(l({},a),A)),n},p=function(A){var a=b(A.components);return e.createElement(d.Provider,{value:a},A.children)},c={inlineCode:"code",wrapper:function(A){var a=A.children;return e.createElement(e.Fragment,{},a)}},u=e.forwardRef((function(A,a){var n=A.components,t=A.mdxType,i=A.originalType,d=A.parentName,p=r(A,["components","mdxType","originalType","parentName"]),u=b(n),m=t,o=u["".concat(d,".").concat(m)]||u[m]||c[m]||i;return n?e.createElement(o,l(l({ref:a},p),{},{components:n})):e.createElement(o,l({ref:a},p))}));function m(A,a){var n=arguments,t=a&&a.mdxType;if("string"==typeof A||t){var i=n.length,l=new Array(i);l[0]=u;var r={};for(var d in a)hasOwnProperty.call(a,d)&&(r[d]=a[d]);r.originalType=A,r.mdxType="string"==typeof A?A:t,l[1]=r;for(var b=2;b{n.r(a),n.d(a,{assets:()=>d,contentTitle:()=>l,default:()=>c,frontMatter:()=>i,metadata:()=>r,toc:()=>b});var e=n(7462),t=(n(7294),n(3905));const i={sidebar_position:12,title:"A.12. Perulangan \u279c for in",sidebar_label:"A.12. Perulangan \u279c for in"},l=void 0,r={unversionedId:"basic/perulangan-for-in",id:"basic/perulangan-for-in",title:"A.12. Perulangan \u279c for in",description:"for in adalah salah satu keyword untuk operasi perulangan yang ada di Rust selain loop dan while. Pada chapter ini kita akan mempelajarinya.",source:"@site/docs/basic/perulangan-for-in.md",sourceDirName:"basic",slug:"/basic/perulangan-for-in",permalink:"/basic/perulangan-for-in",draft:!1,tags:[],version:"current",sidebarPosition:12,frontMatter:{sidebar_position:12,title:"A.12. Perulangan \u279c for in",sidebar_label:"A.12. Perulangan \u279c for in"},sidebar:"tutorialSidebar",previous:{title:"A.11. Perulangan \u279c loop, break, continue, label",permalink:"/basic/perulangan-loop-break-continue-label"},next:{title:"A.13. Array",permalink:"/basic/array"}},d={},b=[{value:"A.12.1. Keyword for in",id:"a121-keyword-for-in",level:2},{value:"A.12.2. Label perulangan",id:"a122-label-perulangan",level:2},{value:"A.12.3. Perulangan for in pada array",id:"a123-perulangan-for-in-pada-array",level:2},{value:"A.12.4. Perulangan for in pada tipe iterator lainnya",id:"a124-perulangan-for-in-pada-tipe-iterator-lainnya",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],p={toc:b};function c(A){let{components:a,...i}=A;return(0,t.kt)("wrapper",(0,e.Z)({},p,i,{components:a,mdxType:"MDXLayout"}),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"for in")," adalah salah satu keyword untuk operasi perulangan yang ada di Rust selain ",(0,t.kt)("inlineCode",{parentName:"p"},"loop")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"while"),". Pada chapter ini kita akan mempelajarinya."),(0,t.kt)("h2",{id:"a121-keyword-for-in"},"A.12.1. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"for in")),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"for in")," adalah keyword perulangan yang paling sering dipakai untuk meng-iterasi data yang tipe data-nya mengimplementasikan trait ",(0,t.kt)("inlineCode",{parentName:"p"},"Iterator"),". Ada banyak jenis tipe data dengan trait ini, salah satunya adalah tipe data ",(0,t.kt)("strong",{parentName:"p"},"range")," yang akan kita bahas pada chapter ini."),(0,t.kt)("blockquote",null,(0,t.kt)("ul",{parentName:"blockquote"},(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai trait ",(0,t.kt)("inlineCode",{parentName:"li"},"Iterator")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/trait-iterator"},"Trait \u279c Iterator")))),(0,t.kt)("p",null,"Tipe data range dibuat dengan notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"a..b"),". Mari lanjut dengan praktik agar lebih jelas. Silakan coba kode sederhana berikut lalu jalankan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'for i in 0..5 {\n println!("{i}");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"perulangan for in",src:n(2472).Z,width:"446",height:"169"})),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"for in")," jika digunakan pada notasi iterator ",(0,t.kt)("inlineCode",{parentName:"p"},"a..b")," maka akan menghasilkan sebuah perulangan dari angka ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," hingga angka di bawah ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),"."),(0,t.kt)("p",null,"Pada contoh di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"0..5")," artinya adalah objek iterator yang dimulai dari angka ",(0,t.kt)("inlineCode",{parentName:"p"},"0")," hingga di bawah ",(0,t.kt)("inlineCode",{parentName:"p"},"5")," (yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"4"),"). Object iterator tersebut kemudian diiterasi, dan ditiap perulangan di-print menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},'println!("{i}")'),". Dengan ini, nilai ",(0,t.kt)("inlineCode",{parentName:"p"},"i")," muncul di layar console dimulai angka ",(0,t.kt)("inlineCode",{parentName:"p"},"0")," hingga ",(0,t.kt)("inlineCode",{parentName:"p"},"4"),"."),(0,t.kt)("p",null,"Jika ingin melakukan perulangan dari ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"b")," (bukan dari ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," ke angka di bawah ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),") gunakan notasi iterator ",(0,t.kt)("inlineCode",{parentName:"p"},"a..=b"),". Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'for i in 0..=5 {\n println!("{i}");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"perulangan for in",src:n(1154).Z,width:"445",height:"192"})),(0,t.kt)("h2",{id:"a122-label-perulangan"},"A.12.2. Label perulangan"),(0,t.kt)("p",null,"Penambahan label pada perulangan ",(0,t.kt)("inlineCode",{parentName:"p"},"for in")," juga bisa dilakukan. Caranya dengan cukup menambahkan sintaks ",(0,t.kt)("inlineCode",{parentName:"p"},"'namaLabel: ")," sebelum statement ",(0,t.kt)("inlineCode",{parentName:"p"},"for in"),". Contohnya seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'\'perulangan: for i in 0..=5 {\n if i > 3 {\n println!("perulangan dihentikan paksa pada iterasi {i}");\n break \'perulangan;\n }\n\n println!("{i}");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"label perulangan for in",src:n(8267).Z,width:"448",height:"180"})),(0,t.kt)("p",null,"Selain ",(0,t.kt)("inlineCode",{parentName:"p"},"break"),", keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"continue")," juga bisa digunakan pada perulangan ",(0,t.kt)("inlineCode",{parentName:"p"},"for in"),"."),(0,t.kt)("h2",{id:"a123-perulangan-for-in-pada-array"},"A.12.3. Perulangan ",(0,t.kt)("inlineCode",{parentName:"h2"},"for in")," pada array"),(0,t.kt)("p",null,"Perulangan menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"for in")," adalah yang paling praktis digunakan untuk mengiterasi elemen array. Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let array = ["jason", "grayon", "drake", "damian"];\nfor name in array {\n println!("{name}");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"label perulangan for in",src:n(2977).Z,width:"445",height:"149"})),(0,t.kt)("p",null,"Perihal apa itu ",(0,t.kt)("inlineCode",{parentName:"p"},"array")," akan dibahas pada chapter selanjutnya, yaitu ",(0,t.kt)("a",{parentName:"p",href:"/basic/array"},"Array"),"."),(0,t.kt)("h2",{id:"a124-perulangan-for-in-pada-tipe-iterator-lainnya"},"A.12.4. Perulangan ",(0,t.kt)("inlineCode",{parentName:"h2"},"for in")," pada tipe iterator lainnya"),(0,t.kt)("p",null,"Di Rust ada banyak cara membuat object iterator. Mengenai topik satu ini kita akan bahas seiring berjalannya proses pembelajaran. Akan berpotesi makin membingungkan jika dibahas terlalu detail pada chapter ini."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/perulangan_for_in"},"github.com/novalagung/dasarpemrogramanrust-example/../perulangan_for_in")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/expressions/loop-expr.html"},"https://doc.rust-lang.org/reference/expressions/loop-expr.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.in.html"},"https://doc.rust-lang.org/std/keyword.in.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.for.html"},"https://doc.rust-lang.org/std/keyword.for.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/for.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/for.html"))))}c.isMDXComponent=!0},2472:(A,a,n)=>{n.d(a,{Z:()=>e});const e=""},1154:(A,a,n)=>{n.d(a,{Z:()=>e});const e=""},8267:(A,a,n)=>{n.d(a,{Z:()=>e});const e=""},2977:(A,a,n)=>{n.d(a,{Z:()=>e});const e=""}}]); \ No newline at end of file diff --git a/assets/js/71880105.a92b59bc.js b/assets/js/71880105.a92b59bc.js deleted file mode 100644 index 80a2571e..00000000 --- a/assets/js/71880105.a92b59bc.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[5758],{3905:(A,a,n)=>{n.d(a,{Zo:()=>p,kt:()=>m});var e=n(7294);function t(A,a,n){return a in A?Object.defineProperty(A,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):A[a]=n,A}function i(A,a){var n=Object.keys(A);if(Object.getOwnPropertySymbols){var e=Object.getOwnPropertySymbols(A);a&&(e=e.filter((function(a){return Object.getOwnPropertyDescriptor(A,a).enumerable}))),n.push.apply(n,e)}return n}function l(A){for(var a=1;a=0||(t[n]=A[n]);return t}(A,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(A);for(e=0;e=0||Object.prototype.propertyIsEnumerable.call(A,n)&&(t[n]=A[n])}return t}var d=e.createContext({}),b=function(A){var a=e.useContext(d),n=a;return A&&(n="function"==typeof A?A(a):l(l({},a),A)),n},p=function(A){var a=b(A.components);return e.createElement(d.Provider,{value:a},A.children)},c={inlineCode:"code",wrapper:function(A){var a=A.children;return e.createElement(e.Fragment,{},a)}},u=e.forwardRef((function(A,a){var n=A.components,t=A.mdxType,i=A.originalType,d=A.parentName,p=r(A,["components","mdxType","originalType","parentName"]),u=b(n),m=t,o=u["".concat(d,".").concat(m)]||u[m]||c[m]||i;return n?e.createElement(o,l(l({ref:a},p),{},{components:n})):e.createElement(o,l({ref:a},p))}));function m(A,a){var n=arguments,t=a&&a.mdxType;if("string"==typeof A||t){var i=n.length,l=new Array(i);l[0]=u;var r={};for(var d in a)hasOwnProperty.call(a,d)&&(r[d]=a[d]);r.originalType=A,r.mdxType="string"==typeof A?A:t,l[1]=r;for(var b=2;b{n.r(a),n.d(a,{assets:()=>d,contentTitle:()=>l,default:()=>c,frontMatter:()=>i,metadata:()=>r,toc:()=>b});var e=n(7462),t=(n(7294),n(3905));const i={sidebar_position:12,title:"A.12. Perulangan \u279c for in",sidebar_label:"A.12. Perulangan \u279c for in"},l=void 0,r={unversionedId:"basic/perulangan-for-in",id:"basic/perulangan-for-in",title:"A.12. Perulangan \u279c for in",description:"for in adalah salah satu keyword untuk operasi perulangan yang ada di Rust selain loop dan while. Pada chapter ini kita akan mempelajarinya.",source:"@site/docs/basic/perulangan-for-in.md",sourceDirName:"basic",slug:"/basic/perulangan-for-in",permalink:"/basic/perulangan-for-in",draft:!1,tags:[],version:"current",sidebarPosition:12,frontMatter:{sidebar_position:12,title:"A.12. Perulangan \u279c for in",sidebar_label:"A.12. Perulangan \u279c for in"},sidebar:"tutorialSidebar",previous:{title:"A.11. Perulangan \u279c loop, break, continue, label",permalink:"/basic/perulangan-loop-break-continue-label"},next:{title:"A.13. Array",permalink:"/basic/array"}},d={},b=[{value:"A.12.1. Keyword for in",id:"a121-keyword-for-in",level:2},{value:"A.12.2. Label perulangan",id:"a122-label-perulangan",level:2},{value:"A.12.3. Perulangan for in pada array",id:"a123-perulangan-for-in-pada-array",level:2},{value:"A.12.4. Perulangan for in pada tipe iterator lainnya",id:"a124-perulangan-for-in-pada-tipe-iterator-lainnya",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],p={toc:b};function c(A){let{components:a,...i}=A;return(0,t.kt)("wrapper",(0,e.Z)({},p,i,{components:a,mdxType:"MDXLayout"}),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"for in")," adalah salah satu keyword untuk operasi perulangan yang ada di Rust selain ",(0,t.kt)("inlineCode",{parentName:"p"},"loop")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"while"),". Pada chapter ini kita akan mempelajarinya."),(0,t.kt)("h2",{id:"a121-keyword-for-in"},"A.12.1. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"for in")),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"for in")," adalah keyword perulangan yang paling sering dipakai untuk meng-iterasi data yang tipe data-nya mengimplementasikan trait ",(0,t.kt)("inlineCode",{parentName:"p"},"Iterator"),". Ada banyak jenis tipe data dengan trait ini, salah satunya adalah tipe data ",(0,t.kt)("strong",{parentName:"p"},"range")," yang akan kita bahas pada chapter ini."),(0,t.kt)("blockquote",null,(0,t.kt)("ul",{parentName:"blockquote"},(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,t.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai trait ",(0,t.kt)("inlineCode",{parentName:"li"},"Iterator")," dibahas pada chapter ",(0,t.kt)("a",{parentName:"li",href:"/basic/trait-iterator"},"Trait \u279c Iterator")))),(0,t.kt)("p",null,"Tipe data range dibuat dengan notasi penulisan ",(0,t.kt)("inlineCode",{parentName:"p"},"a..b"),". Mari lanjut dengan praktik agar lebih jelas. Silakan coba kode sederhana berikut lalu jalankan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'for i in 0..5 {\n println!("{i}");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"perulangan for in",src:n(2472).Z,width:"446",height:"169"})),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"for in")," jika digunakan pada notasi iterator ",(0,t.kt)("inlineCode",{parentName:"p"},"a..b")," maka akan menghasilkan sebuah perulangan dari angka ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," hingga angka dibawah ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),"."),(0,t.kt)("p",null,"Pada contoh di atas, ",(0,t.kt)("inlineCode",{parentName:"p"},"0..5")," artinya adalah objek iterator yang dimulai dari angka ",(0,t.kt)("inlineCode",{parentName:"p"},"0")," hingga dibawah ",(0,t.kt)("inlineCode",{parentName:"p"},"5")," (yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"4"),"). Object iterator tersebut kemudian diiterasi, dan ditiap perulangan di-print menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},'println!("{i}")'),". Dengan ini, nilai ",(0,t.kt)("inlineCode",{parentName:"p"},"i")," muncul di layar console dimulai angka ",(0,t.kt)("inlineCode",{parentName:"p"},"0")," hingga ",(0,t.kt)("inlineCode",{parentName:"p"},"4"),"."),(0,t.kt)("p",null,"Jika ingin melakukan perulangan dari ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"b")," (bukan dari ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," ke angka dibawah ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),") gunakan notasi iterator ",(0,t.kt)("inlineCode",{parentName:"p"},"a..=b"),". Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'for i in 0..=5 {\n println!("{i}");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"perulangan for in",src:n(1154).Z,width:"445",height:"192"})),(0,t.kt)("h2",{id:"a122-label-perulangan"},"A.12.2. Label perulangan"),(0,t.kt)("p",null,"Penambahan label pada perulangan ",(0,t.kt)("inlineCode",{parentName:"p"},"for in")," juga bisa dilakukan. Caranya dengan cukup menambahkan sintaks ",(0,t.kt)("inlineCode",{parentName:"p"},"'namaLabel: ")," sebelum statement ",(0,t.kt)("inlineCode",{parentName:"p"},"for in"),". Contohnya seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'\'perulangan: for i in 0..=5 {\n if i > 3 {\n println!("perulangan dihentikan paksa pada iterasi {i}");\n break \'perulangan;\n }\n\n println!("{i}");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"label perulangan for in",src:n(8267).Z,width:"448",height:"180"})),(0,t.kt)("p",null,"Selain ",(0,t.kt)("inlineCode",{parentName:"p"},"break"),", keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"continue")," juga bisa digunakan pada perulangan ",(0,t.kt)("inlineCode",{parentName:"p"},"for in"),"."),(0,t.kt)("h2",{id:"a123-perulangan-for-in-pada-array"},"A.12.3. Perulangan ",(0,t.kt)("inlineCode",{parentName:"h2"},"for in")," pada array"),(0,t.kt)("p",null,"Perulangan menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"for in")," adalah yang paling praktis digunakan untuk mengiterasi elemen array. Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let array = ["jason", "grayon", "drake", "damian"];\nfor name in array {\n println!("{name}");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"label perulangan for in",src:n(2977).Z,width:"445",height:"149"})),(0,t.kt)("p",null,"Perihal apa itu ",(0,t.kt)("inlineCode",{parentName:"p"},"array")," akan dibahas pada chapter selanjutnya, yaitu ",(0,t.kt)("a",{parentName:"p",href:"/basic/array"},"Array"),"."),(0,t.kt)("h2",{id:"a124-perulangan-for-in-pada-tipe-iterator-lainnya"},"A.12.4. Perulangan ",(0,t.kt)("inlineCode",{parentName:"h2"},"for in")," pada tipe iterator lainnya"),(0,t.kt)("p",null,"Di Rust ada banyak cara membuat object iterator. Mengenai topik satu ini kita akan bahas seiring berjalannya proses pembelajaran. Akan berpotesi makin membingungkan jika dibahas terlalu detail pada chapter ini."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/perulangan_for_in"},"github.com/novalagung/dasarpemrogramanrust-example/../perulangan_for_in")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/expressions/loop-expr.html"},"https://doc.rust-lang.org/reference/expressions/loop-expr.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.in.html"},"https://doc.rust-lang.org/std/keyword.in.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.for.html"},"https://doc.rust-lang.org/std/keyword.for.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/for.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/for.html"))))}c.isMDXComponent=!0},2472:(A,a,n)=>{n.d(a,{Z:()=>e});const e=""},1154:(A,a,n)=>{n.d(a,{Z:()=>e});const e=""},8267:(A,a,n)=>{n.d(a,{Z:()=>e});const e=""},2977:(A,a,n)=>{n.d(a,{Z:()=>e});const e=""}}]); \ No newline at end of file diff --git a/assets/js/7390b81e.18c2ce22.js b/assets/js/7390b81e.18c2ce22.js new file mode 100644 index 00000000..f122c95e --- /dev/null +++ b/assets/js/7390b81e.18c2ce22.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4274],{3905:(a,e,n)=>{n.d(e,{Zo:()=>u,kt:()=>g});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function l(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var d=t.createContext({}),p=function(a){var e=t.useContext(d),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},u=function(a){var e=p(a.components);return t.createElement(d.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},o=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,r=a.originalType,d=a.parentName,u=m(a,["components","mdxType","originalType","parentName"]),o=p(n),g=i,c=o["".concat(d,".").concat(g)]||o[g]||k[g]||r;return n?t.createElement(c,l(l({ref:e},u),{},{components:n})):t.createElement(c,l({ref:e},u))}));function g(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=n.length,l=new Array(r);l[0]=o;var m={};for(var d in e)hasOwnProperty.call(e,d)&&(m[d]=e[d]);m.originalType=a,m.mdxType="string"==typeof a?a:i,l[1]=m;for(var p=2;p{n.r(e),n.d(e,{assets:()=>d,contentTitle:()=>l,default:()=>k,frontMatter:()=>r,metadata:()=>m,toc:()=>p});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:19,title:"A.19. Module System \u279c Package & Crate",sidebar_label:"A.19. Module System \u279c Package & Crate"},l=void 0,m={unversionedId:"basic/package-crate",id:"basic/package-crate",title:"A.19. Module System \u279c Package & Crate",description:"Chapter ini membahas mengenai konsep package dan path dalam Rust programming. Pembelajaran dimulai dari pembahasan mengenai konsep crate terlebih dahulu, kemudian masuk ke package.",source:"@site/docs/basic/package-crate.md",sourceDirName:"basic",slug:"/basic/package-crate",permalink:"/basic/package-crate",draft:!1,tags:[],version:"current",sidebarPosition:19,frontMatter:{sidebar_position:19,title:"A.19. Module System \u279c Package & Crate",sidebar_label:"A.19. Module System \u279c Package & Crate"},sidebar:"tutorialSidebar",previous:{title:"A.18. Module System \u279c Path & Item",permalink:"/basic/path-item"},next:{title:"A.20. Module System \u279c Module",permalink:"/basic/module-basic"}},d={},p=[{value:"A.19.1. Rust Crate",id:"a191-rust-crate",level:2},{value:"\u25c9 Binary crate",id:"-binary-crate",level:3},{value:"\u25c9 Library crate",id:"-library-crate",level:3},{value:"A.19.2. Rust Package",id:"a192-rust-package",level:2},{value:"\u25c9 Naming convention package",id:"-naming-convention-package",level:3},{value:"\u25c9 File Cargo.toml",id:"-file-cargotoml",level:3},{value:"\u25c9 Website crates.io",id:"-website-cratesio",level:3},{value:"\u25c9 Menambahkan dependency atau external crate",id:"-menambahkan-dependency-atau-external-crate",level:3},{value:"\u25c9 Command cargo add",id:"-command-cargo-add",level:3},{value:"A.19.3. Praktik membuat program menampilkan angka random",id:"a193-praktik-membuat-program-menampilkan-angka-random",level:2},{value:"\u25c9 Fungsi generate_random_number",id:"-fungsi-generate_random_number",level:3},{value:"\u25c9 Statement rand::thread_rng().gen_range(0..100)",id:"-statement-randthread_rnggen_range0100",level:3},{value:"\u25c9 Fungsi main",id:"-fungsi-main",level:3},{value:"A.19.4. Pembahasan lanjutan",id:"a194-pembahasan-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:p};function k(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,t.Z)({},u,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas mengenai konsep package dan path dalam Rust programming. Pembelajaran dimulai dari pembahasan mengenai konsep crate terlebih dahulu, kemudian masuk ke package."),(0,i.kt)("h2",{id:"a191-rust-crate"},"A.19.1. Rust Crate"),(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"Crate")," adalah satu unit kompilasi di Rust. Eksekusi command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo build"),", atau ",(0,i.kt)("inlineCode",{parentName:"p"},"rustc")," men-",(0,i.kt)("em",{parentName:"p"},"trigger")," proses kompilasi, dan unit (yang di sini disebut dengan crate) akan di-compile."),(0,i.kt)("p",null,"Crate bisa berisi banyak ",(0,i.kt)("em",{parentName:"p"},"module"),". Sebuah module definisinya bisa berada di banyak file. Agar lebih jelas silakan perhatikan contoh berikut:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"XYZ")," adalah sebuah crate, isinya ada dua module, yaitu module ",(0,i.kt)("inlineCode",{parentName:"li"},"Mod_ABC")," dan module ",(0,i.kt)("inlineCode",{parentName:"li"},"Mod_DEF"),"."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Mod_ABC")," adalah module yang didefinisikan dalam crate ",(0,i.kt)("inlineCode",{parentName:"li"},"XYZ"),", source code-nya berada di file bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"modul_a.rs"),"."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Mod_DEF")," adalah module yang didefinisikan dalam crate ",(0,i.kt)("inlineCode",{parentName:"li"},"XYZ"),", source code-nya berada di beberapa file ",(0,i.kt)("inlineCode",{parentName:"li"},"module_b_one.rs")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"module_b_two.rs"),".")),(0,i.kt)("p",null,"Dari contoh di atas, crate ",(0,i.kt)("inlineCode",{parentName:"p"},"XYZ")," adalah 1 unit kompilasi, yang mana di dalam crate tersebut ada dua modules yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"Mod_ABC")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Mod_DEF")),(0,i.kt)("p",null,"Rust mengkategorikan crate menjadi 2 jenis, ",(0,i.kt)("em",{parentName:"p"},"binary crate")," dan ",(0,i.kt)("em",{parentName:"p"},"library crate")),(0,i.kt)("h3",{id:"-binary-crate"},"\u25c9 Binary crate"),(0,i.kt)("p",null,"Binary crate adalah program yang dikompilasi ke bentuk ",(0,i.kt)("em",{parentName:"p"},"executable"),", untuk kemudian dijalankan, seperti program-program yang sudah kita buat menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo create")," dan run menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run")," itu adalah contoh dari binary crate."),(0,i.kt)("p",null,"Binary crate berada dalam sebuah package yang dibuat menggunakan command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo create ")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo create --bin "),", kedua command ini menjalankan perintah yang sama."),(0,i.kt)("p",null,"Ciri khas dari binary crate adalah memiliki fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", sebuah fungsi yang merupakan ",(0,i.kt)("em",{parentName:"p"},"entrypoint")," program."),(0,i.kt)("h3",{id:"-library-crate"},"\u25c9 Library crate"),(0,i.kt)("p",null,"Library crate berbeda dengan binary crate. Library crate tidak di-compile ke bentuk ",(0,i.kt)("em",{parentName:"p"},"executable")," dan tidak memiliki fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),". Library crate digunakan untuk mendefinisikan set ",(0,i.kt)("em",{parentName:"p"},"functionality")," yang ",(0,i.kt)("em",{parentName:"p"},"reusable")," atau bisa digunakan di banyak project/package."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Library crate di-import/digunakan dalam binary crate. Dalam proses kompilasinya, yang di-compile adalah binary crate. Library crate juga akan ikut dalam kompilasi tersebut.")),(0,i.kt)("p",null,"Sebagai contoh item ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration")," (yang sudah dipraktikkan pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/perulangan-while"},"Perulangan \u279c while"),") dan ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin")," (pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/path-item"},"Module System \u279c Path & Item"),") adalah dua buah item milik crate ",(0,i.kt)("em",{parentName:"p"},"Rust Standard Library")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"std"),". Crate ",(0,i.kt)("inlineCode",{parentName:"p"},"std")," ini akan sangat sering kita gunakan dalam package/project, isinya banyak sekali functionality untuk keperluan standar dalam Rust programming."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Di komunitas Rust, ketika ada kata ",(0,i.kt)("em",{parentName:"p"},"library")," atau ",(0,i.kt)("em",{parentName:"p"},"crate")," maka yang dimaksud biasanya adalah ",(0,i.kt)("em",{parentName:"p"},"library crate"))),(0,i.kt)("p",null,"Kita bisa membuat library crate kemudian di-publish ke ",(0,i.kt)("a",{parentName:"p",href:"https://crates.io/"},"crates.io")," agar bisa digunakan banyak orang. Command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo new --lib ")," digunakan untuk membuat library crate."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai library crate dibahas terpisah pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/library-crate"},"Library Crate"))),(0,i.kt)("h2",{id:"a192-rust-package"},"A.19.2. Rust Package"),(0,i.kt)("p",null,"Istilah package dalam Rust programming masih sama dengan package dalam pemrograman lain. Package adalah sebuah set yang berisi banyak ",(0,i.kt)("em",{parentName:"p"},"functionality"),". Satu buah package bisa berisi satu atau banyak crates."),(0,i.kt)("p",null,"Package di-manage oleh Cargo, yang merupakan package manager Rust. Command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo new ")," digunakan untuk membuat package. Command tersebut menghasilkan beberapa file yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"src/main.rs")," yang isinya adalah kode program, dan juga file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml")," yang isinya adalah informasi mengenai package tersebut."),(0,i.kt)("p",null,"Ok, sekalian praktik mungkin lebih pas. Silakan jalankan command berikut, kemudian ikuti penjelasan selanjutnya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash"},"cargo new belajar_package_crate\n")),(0,i.kt)("h3",{id:"-naming-convention-package"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Naming convention")," package"),(0,i.kt)("p",null,"Khusus untuk penamaan package dan juga crate, tidak ada rekomendasi dari dokumentasi official Rust. Pada ebook ini penulis menggunakan ",(0,i.kt)("em",{parentName:"p"},"snake case")," sebagai naming convention dalam pembuatan package."),(0,i.kt)("p",null,"Referensi: ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"https://rust-lang.github.io/api-guidelines/naming.html")),(0,i.kt)("h3",{id:"-file-cargotoml"},"\u25c9 File ",(0,i.kt)("inlineCode",{parentName:"h3"},"Cargo.toml")),(0,i.kt)("p",null,"File ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml")," menampung beberapa informasi penting milik package, di antaranya adalah nama package, versi package dan juga versi rust, serta ",(0,i.kt)("em",{parentName:"p"},"dependencies")," atau ",(0,i.kt)("em",{parentName:"p"},"3rd-party")," yang digunakan dalam package (dalam konteks Rust adalah ",(0,i.kt)("em",{parentName:"p"},"crate"),")."),(0,i.kt)("p",null,"Command yang sebelumnya di-run menghasilkan file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml")," berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-toml",metastring:'title="Cargo.toml"',title:'"Cargo.toml"'},'[package]\nname = "belajar_package_crate"\nversion = "0.1.0"\nedition = "2021"\n\n# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html\n\n[dependencies]\n')),(0,i.kt)("p",null,"Blok ",(0,i.kt)("inlineCode",{parentName:"p"},"package")," berisi berisi 3 buah field:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"name")," isinya adalah nama package, sesuai dengan argument command ",(0,i.kt)("inlineCode",{parentName:"li"},"cargo new "),"."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"version")," default-nya selalu ",(0,i.kt)("inlineCode",{parentName:"li"},"0.1.0"),", namun kita bisa ubah nilainya seiring berjalannya proses pengembangan aplikasi."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"edition")," di sini me-refer ke edisi rust yang dipakai. Di ebook ini, rust versi ",(0,i.kt)("strong",{parentName:"li"},"1.65.0")," digunakan, dan edisi untuk versi tersebut adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"2021"),".")),(0,i.kt)("p",null,"Blok ",(0,i.kt)("inlineCode",{parentName:"p"},"dependencies")," default-nya berisi kosong. Jika kita menambahkan external dependencies atau crates, maka detailnya tercatat pada blok ",(0,i.kt)("inlineCode",{parentName:"p"},"dependencies")," ini."),(0,i.kt)("h3",{id:"-website-cratesio"},"\u25c9 Website ",(0,i.kt)("a",{parentName:"h3",href:"https://crates.io/"},"crates.io")),(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"crates.io")," merupakan official website dari Rust untuk pencarian dan juga ",(0,i.kt)("em",{parentName:"p"},"managing")," dependency. Silakan manfaatkan website tersebut untuk mencari crates sesuai kebutuhan."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},(0,i.kt)("a",{parentName:"p",href:"https://crates.io/"},"https://crates.io/"))),(0,i.kt)("h3",{id:"-menambahkan-dependency-atau-external-crate"},"\u25c9 Menambahkan dependency atau external crate"),(0,i.kt)("p",null,"proses pembelajaran akan dilanjutkan sambil praktik. Pada bagian ini kita akan buat program sederhana yang di dalamnya memanfaatkan sebuah dependency atau external crate."),(0,i.kt)("p",null,"Pertama-tama buka ",(0,i.kt)("a",{parentName:"p",href:"https://crates.io/"},"crates.io"),", lalu gunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," dalam pencarian, hasilnya adalah dependency bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"rand"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"package crate",src:n(6637).Z,width:"814",height:"429"})),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"rand")," merupakan crate yang isinya banyak fungsi untuk keperluan ",(0,i.kt)("em",{parentName:"p"},"generate")," data random."),(0,i.kt)("p",null,"Sekarang buka file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml"),", dan tambahkan dependency ",(0,i.kt)("inlineCode",{parentName:"p"},"rand"),". Sesuaikan dengan versi yang muncul di layar masing-masing."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-toml",metastring:'title="Cargo.toml"',title:'"Cargo.toml"'},'[package]\nname = "package_crate"\nversion = "0.1.0"\nedition = "2021"\n\n# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html\n\n[dependencies]\nrand = "0.8.5"\n')),(0,i.kt)("p",null,"Jalankan command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo build")," untuk memaksa Cargo agar mendownload depdency yang sudah ditambahkan ke file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"cargo download dependency",src:n(2002).Z,width:"440",height:"172"})),(0,i.kt)("p",null,"Sukses! Sekarang dependency ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," sudah bisa digunakan dalam package yang sudah dibuat."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Jika pembaca menemui error ",(0,i.kt)("inlineCode",{parentName:"p"},"failed to authenticate when downloading repository"),", jalankan beberapa command berikut secara berurutan:"),(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"ssh-agent -s")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"ssh-add")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"cargo build")))),(0,i.kt)("p",null,"Jadi kurang lebih seperti itu cara menambahkan dependency di Rust. Seiring berjalannya proses pembelajaran penulis yakin pembaca akan terbiasa dengan flow dan juga command di atas."),(0,i.kt)("h3",{id:"-command-cargo-add"},"\u25c9 Command ",(0,i.kt)("inlineCode",{parentName:"h3"},"cargo add")),(0,i.kt)("p",null,"Selain cara di atas, ada cara lain yang lebih ringkas untuk penambahan package, yaitu menggunakan command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo add "),". Command ini secara otomatis menambahkan package yang diinginkan ke file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml")," kemudian mengunduhnya."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Menambahkan package ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," versi terbaru"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"cargo add rand\n"))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Menambahkan package ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," versi ",(0,i.kt)("inlineCode",{parentName:"p"},"0.8.5")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"cargo add rand@0.8.5\n")))),(0,i.kt)("h2",{id:"a193-praktik-membuat-program-menampilkan-angka-random"},"A.19.3. Praktik membuat program menampilkan angka random"),(0,i.kt)("p",null,"Ok, sekarang kita lanjutkan praktik pembuatan program sederhana untuk menampilkan angka random."),(0,i.kt)("p",null,"Angka random di sini akan di-generate menggunakan fungsi yang ada dalam dependency ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," yang sudah ditambahkan ke package."),(0,i.kt)("p",null,"Silakan tulis kode berikut ke file ",(0,i.kt)("inlineCode",{parentName:"p"},"src/main.rs"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use rand::Rng;\n\nfn generate_random_number() -> i32 {\n rand::thread_rng().gen_range(0..100)\n}\n\nfn main() {\n for i in 0..5 {\n println!("random number {}: {}", i, generate_random_number());\n }\n}\n')),(0,i.kt)("p",null,"Jalankan program, lalu lihat hasilnya. Sejumlah data angka random muncul di layar output."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rand crate",src:n(8240).Z,width:"455",height:"158"})),(0,i.kt)("p",null,"Silakan coba untuk run beberapa kali, angka random berbeda setiap kali run."),(0,i.kt)("h3",{id:"-fungsi-generate_random_number"},"\u25c9 Fungsi ",(0,i.kt)("inlineCode",{parentName:"h3"},"generate_random_number")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"generate_random_number")," adalah fungsi yang kita buat, yang tugasnya mengembalikan sebuah nilai numerik bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),". Angka tersebut berasal dari proses ",(0,i.kt)("em",{parentName:"p"},"generate random")," hasil eksekusi statement ",(0,i.kt)("inlineCode",{parentName:"p"},"rand::thread_rng().gen_range(0..100)"),"."),(0,i.kt)("h3",{id:"-statement-randthread_rnggen_range0100"},"\u25c9 Statement ",(0,i.kt)("inlineCode",{parentName:"h3"},"rand::thread_rng().gen_range(0..100)")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"rand::thread_rng().gen_range")," digunakan untuk generate data random. Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"gen_range")," menerima argument bertipe range, dan angka random akan di-generate sesuai range tersebut. Sebagai contoh, range ",(0,i.kt)("inlineCode",{parentName:"p"},"0..100")," menghasilkan angka random antara ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," hingga ",(0,i.kt)("inlineCode",{parentName:"p"},"99"),"."),(0,i.kt)("p",null,"O iya, penggunaan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"gen_range")," mewajibkan kita untuk import path ",(0,i.kt)("inlineCode",{parentName:"p"},"rand::Rng"),". Itulah kenapa ada statement ",(0,i.kt)("inlineCode",{parentName:"p"},"use rand::Rng"),"."),(0,i.kt)("h3",{id:"-fungsi-main"},"\u25c9 Fungsi ",(0,i.kt)("inlineCode",{parentName:"h3"},"main")),(0,i.kt)("p",null,"Dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", isinya sebuah perulangan sederhana yang menampilkan angka random hasil eksekusi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"generate_random_number")," di setiap iterasinya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'for i in 0..5 {\n println!("random number {}: {}", i, generate_random_number());\n}\n')),(0,i.kt)("h2",{id:"a194-pembahasan-lanjutan"},"A.19.4. Pembahasan lanjutan"),(0,i.kt)("p",null,"Sekian pembahasan mengenai package dan crate. Semoga membantu, silakan diulang-ulang jika perlu agar tidak bingung, ",(0,i.kt)("strong",{parentName:"p"},"module system")," merupakan salah satu hal yang sangat penting di Rust."),(0,i.kt)("p",null,"Masih dalam topik ",(0,i.kt)("em",{parentName:"p"},"module system"),", penulis anjurkan untuk lanjut ke chapter berikutnya karena berkaitan."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/package_crate"},"github.com/novalagung/dasarpemrogramanrust-example/../package_crate")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-01-packages-and-crates.html"},"https://doc.rust-lang.org/book/ch07-01-packages-and-crates.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/crates.html"},"https://doc.rust-lang.org/rust-by-example/crates.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"https://rust-lang.github.io/api-guidelines/naming.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://github.com/rust-lang/cargo/issues/3381"},"https://github.com/rust-lang/cargo/issues/3381"))))}k.isMDXComponent=!0},6637:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/package-crate-1-bc29d1ca7bd36fcd767c54466152bf67.png"},2002:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},8240:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/7390b81e.686f428e.js b/assets/js/7390b81e.686f428e.js deleted file mode 100644 index 40f9865d..00000000 --- a/assets/js/7390b81e.686f428e.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4274],{3905:(a,e,n)=>{n.d(e,{Zo:()=>u,kt:()=>g});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function l(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var d=t.createContext({}),p=function(a){var e=t.useContext(d),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},u=function(a){var e=p(a.components);return t.createElement(d.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},o=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,r=a.originalType,d=a.parentName,u=m(a,["components","mdxType","originalType","parentName"]),o=p(n),g=i,c=o["".concat(d,".").concat(g)]||o[g]||k[g]||r;return n?t.createElement(c,l(l({ref:e},u),{},{components:n})):t.createElement(c,l({ref:e},u))}));function g(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=n.length,l=new Array(r);l[0]=o;var m={};for(var d in e)hasOwnProperty.call(e,d)&&(m[d]=e[d]);m.originalType=a,m.mdxType="string"==typeof a?a:i,l[1]=m;for(var p=2;p{n.r(e),n.d(e,{assets:()=>d,contentTitle:()=>l,default:()=>k,frontMatter:()=>r,metadata:()=>m,toc:()=>p});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:19,title:"A.19. Module System \u279c Package & Crate",sidebar_label:"A.19. Module System \u279c Package & Crate"},l=void 0,m={unversionedId:"basic/package-crate",id:"basic/package-crate",title:"A.19. Module System \u279c Package & Crate",description:"Chapter ini membahas mengenai konsep package dan path dalam Rust programming. Pembelajaran dimulai dari pembahasan mengenai konsep crate terlebih dahulu, kemudian masuk ke package.",source:"@site/docs/basic/package-crate.md",sourceDirName:"basic",slug:"/basic/package-crate",permalink:"/basic/package-crate",draft:!1,tags:[],version:"current",sidebarPosition:19,frontMatter:{sidebar_position:19,title:"A.19. Module System \u279c Package & Crate",sidebar_label:"A.19. Module System \u279c Package & Crate"},sidebar:"tutorialSidebar",previous:{title:"A.18. Module System \u279c Path & Item",permalink:"/basic/path-item"},next:{title:"A.20. Module System \u279c Module",permalink:"/basic/module-basic"}},d={},p=[{value:"A.19.1. Rust Crate",id:"a191-rust-crate",level:2},{value:"\u25c9 Binary crate",id:"-binary-crate",level:3},{value:"\u25c9 Library crate",id:"-library-crate",level:3},{value:"A.19.2. Rust Package",id:"a192-rust-package",level:2},{value:"\u25c9 Naming convention package",id:"-naming-convention-package",level:3},{value:"\u25c9 File Cargo.toml",id:"-file-cargotoml",level:3},{value:"\u25c9 Website crates.io",id:"-website-cratesio",level:3},{value:"\u25c9 Menambahkan dependency atau external crate",id:"-menambahkan-dependency-atau-external-crate",level:3},{value:"\u25c9 Command cargo add",id:"-command-cargo-add",level:3},{value:"A.19.3. Praktik membuat program menampilkan angka random",id:"a193-praktik-membuat-program-menampilkan-angka-random",level:2},{value:"\u25c9 Fungsi generate_random_number",id:"-fungsi-generate_random_number",level:3},{value:"\u25c9 Statement rand::thread_rng().gen_range(0..100)",id:"-statement-randthread_rnggen_range0100",level:3},{value:"\u25c9 Fungsi main",id:"-fungsi-main",level:3},{value:"A.19.4. Pembahasan lanjutan",id:"a194-pembahasan-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:p};function k(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,t.Z)({},u,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas mengenai konsep package dan path dalam Rust programming. Pembelajaran dimulai dari pembahasan mengenai konsep crate terlebih dahulu, kemudian masuk ke package."),(0,i.kt)("h2",{id:"a191-rust-crate"},"A.19.1. Rust Crate"),(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"Crate")," adalah satu unit kompilasi di Rust. Eksekusi command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo build"),", atau ",(0,i.kt)("inlineCode",{parentName:"p"},"rustc")," men-",(0,i.kt)("em",{parentName:"p"},"trigger")," proses kompilasi, dan unit (yang di sini disebut dengan crate) akan di-compile."),(0,i.kt)("p",null,"Crate bisa berisi banyak ",(0,i.kt)("em",{parentName:"p"},"module"),". Sebuah module definisinya bisa berada di banyak file. Agar lebih jelas silakan perhatikan contoh berikut:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"XYZ")," adalah sebuah crate, isinya ada dua module, yaitu module ",(0,i.kt)("inlineCode",{parentName:"li"},"Mod_ABC")," dan module ",(0,i.kt)("inlineCode",{parentName:"li"},"Mod_DEF"),"."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Mod_ABC")," adalah module yang didefinisikan dalam crate ",(0,i.kt)("inlineCode",{parentName:"li"},"XYZ"),", source code-nya berada di file bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"modul_a.rs"),"."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Mod_DEF")," adalah module yang didefinisikan dalam crate ",(0,i.kt)("inlineCode",{parentName:"li"},"XYZ"),", source code-nya berada di beberapa file ",(0,i.kt)("inlineCode",{parentName:"li"},"module_b_one.rs")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"module_b_two.rs"),".")),(0,i.kt)("p",null,"Dari contoh di atas, crate ",(0,i.kt)("inlineCode",{parentName:"p"},"XYZ")," adalah 1 unit kompilasi, yang dimana di dalam crate tersebut ada dua modules yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"Mod_ABC")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Mod_DEF")),(0,i.kt)("p",null,"Rust mengkategorikan crate menjadi 2 jenis, ",(0,i.kt)("em",{parentName:"p"},"binary crate")," dan ",(0,i.kt)("em",{parentName:"p"},"library crate")),(0,i.kt)("h3",{id:"-binary-crate"},"\u25c9 Binary crate"),(0,i.kt)("p",null,"Binary crate adalah program yang dikompilasi ke bentuk ",(0,i.kt)("em",{parentName:"p"},"executable"),", untuk kemudian dijalankan, seperti program-program yang sudah kita buat menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo create")," dan run menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run")," itu adalah contoh dari binary crate."),(0,i.kt)("p",null,"Binary crate berada dalam sebuah package yang dibuat menggunakan command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo create ")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo create --bin "),", kedua command ini menjalankan perintah yang sama."),(0,i.kt)("p",null,"Ciri khas dari binary crate adalah memiliki fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", sebuah fungsi yang merupakan ",(0,i.kt)("em",{parentName:"p"},"entrypoint")," program."),(0,i.kt)("h3",{id:"-library-crate"},"\u25c9 Library crate"),(0,i.kt)("p",null,"Library crate berbeda dengan binary crate. Library crate tidak di-compile ke bentuk ",(0,i.kt)("em",{parentName:"p"},"executable")," dan tidak memiliki fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),". Library crate digunakan untuk mendefinisikan set ",(0,i.kt)("em",{parentName:"p"},"functionality")," yang ",(0,i.kt)("em",{parentName:"p"},"reusable")," atau bisa digunakan di banyak project/package."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Library crate di-import/digunakan dalam binary crate. Dalam proses kompilasinya, yang di-compile adalah binary crate. Library crate juga akan ikut dalam kompilasi tersebut.")),(0,i.kt)("p",null,"Sebagai contoh item ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration")," (yang sudah dipraktikkan pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/perulangan-while"},"Perulangan \u279c while"),") dan ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin")," (pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/path-item"},"Module System \u279c Path & Item"),") adalah dua buah item milik crate ",(0,i.kt)("em",{parentName:"p"},"Rust Standard Library")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"std"),". Crate ",(0,i.kt)("inlineCode",{parentName:"p"},"std")," ini akan sangat sering kita gunakan dalam package/project, isinya banyak sekali functionality untuk keperluan standar dalam Rust programming."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Di komunitas Rust, ketika ada kata ",(0,i.kt)("em",{parentName:"p"},"library")," atau ",(0,i.kt)("em",{parentName:"p"},"crate")," maka yang dimaksud biasanya adalah ",(0,i.kt)("em",{parentName:"p"},"library crate"))),(0,i.kt)("p",null,"Kita bisa membuat library crate kemudian di-publish ke ",(0,i.kt)("a",{parentName:"p",href:"https://crates.io/"},"crates.io")," agar bisa digunakan banyak orang. Command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo new --lib ")," digunakan untuk membuat library crate."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai library crate dibahas terpisah pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/library-crate"},"Library Crate"))),(0,i.kt)("h2",{id:"a192-rust-package"},"A.19.2. Rust Package"),(0,i.kt)("p",null,"Istilah package dalam Rust programming masih sama dengan package dalam pemrograman lain. Package adalah sebuah set yang berisi banyak ",(0,i.kt)("em",{parentName:"p"},"functionality"),". Satu buah package bisa berisi satu atau banyak crates."),(0,i.kt)("p",null,"Package di-manage oleh Cargo, yang merupakan package manager Rust. Command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo new ")," digunakan untuk membuat package. Command tersebut menghasilkan beberapa file yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"src/main.rs")," yang isinya adalah kode program, dan juga file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml")," yang isinya adalah informasi mengenai package tersebut."),(0,i.kt)("p",null,"Ok, sekalian praktik mungkin lebih pas. Silakan jalankan command berikut, kemudian ikuti penjelasan selanjutnya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash"},"cargo new belajar_package_crate\n")),(0,i.kt)("h3",{id:"-naming-convention-package"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Naming convention")," package"),(0,i.kt)("p",null,"Khusus untuk penamaan package dan juga crate, tidak ada rekomendasi dari dokumentasi official Rust. Pada ebook ini penulis menggunakan ",(0,i.kt)("em",{parentName:"p"},"snake case")," sebagai naming convention dalam pembuatan package."),(0,i.kt)("p",null,"Referensi: ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"https://rust-lang.github.io/api-guidelines/naming.html")),(0,i.kt)("h3",{id:"-file-cargotoml"},"\u25c9 File ",(0,i.kt)("inlineCode",{parentName:"h3"},"Cargo.toml")),(0,i.kt)("p",null,"File ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml")," menampung beberapa informasi penting milik package, di antaranya adalah nama package, versi package dan juga versi rust, serta ",(0,i.kt)("em",{parentName:"p"},"dependencies")," atau ",(0,i.kt)("em",{parentName:"p"},"3rd-party")," yang digunakan dalam package (dalam konteks Rust adalah ",(0,i.kt)("em",{parentName:"p"},"crate"),")."),(0,i.kt)("p",null,"Command yang sebelumnya di-run menghasilkan file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml")," berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-toml",metastring:'title="Cargo.toml"',title:'"Cargo.toml"'},'[package]\nname = "belajar_package_crate"\nversion = "0.1.0"\nedition = "2021"\n\n# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html\n\n[dependencies]\n')),(0,i.kt)("p",null,"Blok ",(0,i.kt)("inlineCode",{parentName:"p"},"package")," berisi berisi 3 buah field:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"name")," isinya adalah nama package, sesuai dengan argument command ",(0,i.kt)("inlineCode",{parentName:"li"},"cargo new "),"."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"version")," default-nya selalu ",(0,i.kt)("inlineCode",{parentName:"li"},"0.1.0"),", namun kita bisa ubah nilainya seiring berjalannya proses pengembangan aplikasi."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"edition")," di sini me-refer ke edisi rust yang dipakai. Di ebook ini, rust versi ",(0,i.kt)("strong",{parentName:"li"},"1.65.0")," digunakan, dan edisi untuk versi tersebut adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"2021"),".")),(0,i.kt)("p",null,"Blok ",(0,i.kt)("inlineCode",{parentName:"p"},"dependencies")," default-nya berisi kosong. Jika kita menambahkan external dependencies atau crates, maka detailnya tercatat pada blok ",(0,i.kt)("inlineCode",{parentName:"p"},"dependencies")," ini."),(0,i.kt)("h3",{id:"-website-cratesio"},"\u25c9 Website ",(0,i.kt)("a",{parentName:"h3",href:"https://crates.io/"},"crates.io")),(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"crates.io")," merupakan official website dari Rust untuk pencarian dan juga ",(0,i.kt)("em",{parentName:"p"},"managing")," dependency. Silakan manfaatkan website tersebut untuk mencari crates sesuai kebutuhan."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},(0,i.kt)("a",{parentName:"p",href:"https://crates.io/"},"https://crates.io/"))),(0,i.kt)("h3",{id:"-menambahkan-dependency-atau-external-crate"},"\u25c9 Menambahkan dependency atau external crate"),(0,i.kt)("p",null,"proses pembelajaran akan dilanjutkan sambil praktik. Pada bagian ini kita akan buat program sederhana yang di dalamnya memanfaatkan sebuah dependency atau external crate."),(0,i.kt)("p",null,"Pertama-tama buka ",(0,i.kt)("a",{parentName:"p",href:"https://crates.io/"},"crates.io"),", lalu gunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," dalam pencarian, hasilnya adalah dependency bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"rand"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"package crate",src:n(6637).Z,width:"814",height:"429"})),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"rand")," merupakan crate yang isinya banyak fungsi untuk keperluan ",(0,i.kt)("em",{parentName:"p"},"generate")," data random."),(0,i.kt)("p",null,"Sekarang buka file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml"),", dan tambahkan dependency ",(0,i.kt)("inlineCode",{parentName:"p"},"rand"),". Sesuaikan dengan versi yang muncul di layar masing-masing."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-toml",metastring:'title="Cargo.toml"',title:'"Cargo.toml"'},'[package]\nname = "package_crate"\nversion = "0.1.0"\nedition = "2021"\n\n# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html\n\n[dependencies]\nrand = "0.8.5"\n')),(0,i.kt)("p",null,"Jalankan command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo build")," untuk memaksa Cargo agar mendownload depdency yang sudah ditambahkan ke file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"cargo download dependency",src:n(2002).Z,width:"440",height:"172"})),(0,i.kt)("p",null,"Sukses! Sekarang dependency ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," sudah bisa digunakan dalam package yang sudah dibuat."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Jika pembaca menemui error ",(0,i.kt)("inlineCode",{parentName:"p"},"failed to authenticate when downloading repository"),", jalankan beberapa command berikut secara berurutan:"),(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"ssh-agent -s")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"ssh-add")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"cargo build")))),(0,i.kt)("p",null,"Jadi kurang lebih seperti itu cara menambahkan dependency di Rust. Seiring berjalannya proses pembelajaran penulis yakin pembaca akan terbiasa dengan flow dan juga command di atas."),(0,i.kt)("h3",{id:"-command-cargo-add"},"\u25c9 Command ",(0,i.kt)("inlineCode",{parentName:"h3"},"cargo add")),(0,i.kt)("p",null,"Selain cara di atas, ada cara lain yang lebih ringkas untuk penambahan package, yaitu menggunakan command ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo add "),". Command ini secara otomatis menambahkan package yang diinginkan ke file ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml")," kemudian mengunduhnya."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Menambahkan package ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," versi terbaru"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"cargo add rand\n"))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Menambahkan package ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," versi ",(0,i.kt)("inlineCode",{parentName:"p"},"0.8.5")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"cargo add rand@0.8.5\n")))),(0,i.kt)("h2",{id:"a193-praktik-membuat-program-menampilkan-angka-random"},"A.19.3. Praktik membuat program menampilkan angka random"),(0,i.kt)("p",null,"Ok, sekarang kita lanjutkan praktik pembuatan program sederhana untuk menampilkan angka random."),(0,i.kt)("p",null,"Angka random di sini akan di-generate menggunakan fungsi yang ada dalam dependency ",(0,i.kt)("inlineCode",{parentName:"p"},"rand")," yang sudah ditambahkan ke package."),(0,i.kt)("p",null,"Silakan tulis kode berikut ke file ",(0,i.kt)("inlineCode",{parentName:"p"},"src/main.rs"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use rand::Rng;\n\nfn generate_random_number() -> i32 {\n rand::thread_rng().gen_range(0..100)\n}\n\nfn main() {\n for i in 0..5 {\n println!("random number {}: {}", i, generate_random_number());\n }\n}\n')),(0,i.kt)("p",null,"Jalankan program, lalu lihat hasilnya. Sejumlah data angka random muncul di layar output."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rand crate",src:n(8240).Z,width:"455",height:"158"})),(0,i.kt)("p",null,"Silakan coba untuk run beberapa kali, angka random berbeda setiap kali run."),(0,i.kt)("h3",{id:"-fungsi-generate_random_number"},"\u25c9 Fungsi ",(0,i.kt)("inlineCode",{parentName:"h3"},"generate_random_number")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"generate_random_number")," adalah fungsi yang kita buat, yang tugasnya mengembalikan sebuah nilai numerik bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),". Angka tersebut berasal dari proses ",(0,i.kt)("em",{parentName:"p"},"generate random")," hasil eksekusi statement ",(0,i.kt)("inlineCode",{parentName:"p"},"rand::thread_rng().gen_range(0..100)"),"."),(0,i.kt)("h3",{id:"-statement-randthread_rnggen_range0100"},"\u25c9 Statement ",(0,i.kt)("inlineCode",{parentName:"h3"},"rand::thread_rng().gen_range(0..100)")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"rand::thread_rng().gen_range")," digunakan untuk generate data random. Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"gen_range")," menerima argument bertipe range, dan angka random akan di-generate sesuai range tersebut. Sebagai contoh, range ",(0,i.kt)("inlineCode",{parentName:"p"},"0..100")," menghasilkan angka random antara ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," hingga ",(0,i.kt)("inlineCode",{parentName:"p"},"99"),"."),(0,i.kt)("p",null,"O iya, penggunaan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"gen_range")," mewajibkan kita untuk import path ",(0,i.kt)("inlineCode",{parentName:"p"},"rand::Rng"),". Itulah kenapa ada statement ",(0,i.kt)("inlineCode",{parentName:"p"},"use rand::Rng"),"."),(0,i.kt)("h3",{id:"-fungsi-main"},"\u25c9 Fungsi ",(0,i.kt)("inlineCode",{parentName:"h3"},"main")),(0,i.kt)("p",null,"Dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", isinya sebuah perulangan sederhana yang menampilkan angka random hasil eksekusi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"generate_random_number")," di setiap iterasinya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'for i in 0..5 {\n println!("random number {}: {}", i, generate_random_number());\n}\n')),(0,i.kt)("h2",{id:"a194-pembahasan-lanjutan"},"A.19.4. Pembahasan lanjutan"),(0,i.kt)("p",null,"Sekian pembahasan mengenai package dan crate. Semoga membantu, silakan diulang-ulang jika perlu agar tidak bingung, ",(0,i.kt)("strong",{parentName:"p"},"module system")," merupakan salah satu hal yang sangat penting di Rust."),(0,i.kt)("p",null,"Masih dalam topik ",(0,i.kt)("em",{parentName:"p"},"module system"),", penulis anjurkan untuk lanjut ke chapter berikutnya karena berkaitan."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/package_crate"},"github.com/novalagung/dasarpemrogramanrust-example/../package_crate")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-01-packages-and-crates.html"},"https://doc.rust-lang.org/book/ch07-01-packages-and-crates.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/crates.html"},"https://doc.rust-lang.org/rust-by-example/crates.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"https://rust-lang.github.io/api-guidelines/naming.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://github.com/rust-lang/cargo/issues/3381"},"https://github.com/rust-lang/cargo/issues/3381"))))}k.isMDXComponent=!0},6637:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/package-crate-1-bc29d1ca7bd36fcd767c54466152bf67.png"},2002:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},8240:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/7794a932.0807e2e7.js b/assets/js/7794a932.0807e2e7.js deleted file mode 100644 index aa62b125..00000000 --- a/assets/js/7794a932.0807e2e7.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8358],{3905:(e,a,n)=>{n.d(a,{Zo:()=>u,kt:()=>s});var t=n(7294);function i(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function r(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function l(e){for(var a=1;a=0||(i[n]=e[n]);return i}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var d=t.createContext({}),o=function(e){var a=t.useContext(d),n=a;return e&&(n="function"==typeof e?e(a):l(l({},a),e)),n},u=function(e){var a=o(e.components);return t.createElement(d.Provider,{value:a},e.children)},k={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},m=t.forwardRef((function(e,a){var n=e.components,i=e.mdxType,r=e.originalType,d=e.parentName,u=p(e,["components","mdxType","originalType","parentName"]),m=o(n),s=i,c=m["".concat(d,".").concat(s)]||m[s]||k[s]||r;return n?t.createElement(c,l(l({ref:a},u),{},{components:n})):t.createElement(c,l({ref:a},u))}));function s(e,a){var n=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var r=n.length,l=new Array(r);l[0]=m;var p={};for(var d in a)hasOwnProperty.call(a,d)&&(p[d]=a[d]);p.originalType=e,p.mdxType="string"==typeof e?e:i,l[1]=p;for(var o=2;o{n.r(a),n.d(a,{assets:()=>d,contentTitle:()=>l,default:()=>k,frontMatter:()=>r,metadata:()=>p,toc:()=>o});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:39,title:"A.39. Tipe Data \u279c Result",sidebar_label:"A.39. Tipe Data \u279c Result"},l=void 0,p={unversionedId:"basic/result-type",id:"basic/result-type",title:"A.39. Tipe Data \u279c Result",description:"Chapter ini membahas tentang tipe data Result. Tipe data ini digunakan untuk menampung nilai hasil suatu proses yang isinya adalah bisa sukses (Ok) atau error (Err).",source:"@site/docs/basic/result-type.md",sourceDirName:"basic",slug:"/basic/result-type",permalink:"/basic/result-type",draft:!1,tags:[],version:"current",sidebarPosition:39,frontMatter:{sidebar_position:39,title:"A.39. Tipe Data \u279c Result",sidebar_label:"A.39. Tipe Data \u279c Result"},sidebar:"tutorialSidebar",previous:{title:"A.38. Tipe Data \u279c Option",permalink:"/basic/option-type"},next:{title:"A.40. Pattern Matching",permalink:"/basic/pattern-matching"}},d={},o=[{value:"A.39.1. Konsep Result",id:"a391-konsep-result",level:2},{value:"A.39.2. Pattern matching pada tipe Result",id:"a392-pattern-matching-pada-tipe-result",level:2},{value:"\u25c9 Tips pattern matching",id:"-tips-pattern-matching",level:3},{value:"A.39.3. Method tipe data Result",id:"a393-method-tipe-data-result",level:2},{value:"\u25c9 Method is_ok & unwrap",id:"-method-is_ok--unwrap",level:3},{value:"\u25c9 Method as_ref",id:"-method-as_ref",level:3},{value:"\u25c9 Method is_err & err",id:"-method-is_err--err",level:3},{value:"\u25c9 Method ok",id:"-method-ok",level:3},{value:"\u25c9 Method unwrap_or_default",id:"-method-unwrap_or_default",level:3},{value:"\u25c9 Method unwrap_or",id:"-method-unwrap_or",level:3},{value:"\u25c9 Method unwrap_or_else",id:"-method-unwrap_or_else",level:3},{value:"A.39.4. Error handling tipe Result",id:"a394-error-handling-tipe-result",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:o};function k(e){let{components:a,...r}=e;return(0,i.kt)("wrapper",(0,t.Z)({},u,r,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas tentang tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Tipe data ini digunakan untuk menampung nilai hasil suatu proses yang isinya adalah bisa sukses (",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),") atau error (",(0,i.kt)("inlineCode",{parentName:"p"},"Err"),")."),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," biasa digunakan untuk menampung hasil eksekusi proses dan error handling."),(0,i.kt)("h2",{id:"a391-konsep-result"},"A.39.1. Konsep ",(0,i.kt)("inlineCode",{parentName:"h2"},"Result")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," adalah enum dengan isi 2 buah enum value:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Result::Ok")," (atau ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok"),"), digunakan untuk menandai bahwa data isinya adalah kabar baik (oke / mantab / jos / sukses)."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Result::Err")," (atau ",(0,i.kt)("inlineCode",{parentName:"li"},"Err"),"), digunakan untuk menandai bawah data berisi kabar buruk.")),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"T")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"E")," merupakan parameter generic. Lebih jelasnya mengenai generic dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics"),"."))),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," memiliki notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," digunakan pada enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"E")," digunakan enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Err"),"."),(0,i.kt)("p",null,"Cara penerapan tipe data ini bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nenum MathError {\n DivisionByZero,\n InfinityNumber,\n OtherError,\n}\n\nfn main() {\n let result1 = divider(10.0, 5.0);\n println!("result: {:?}", result1);\n\n let result2: Result = divider(10.0, 0.0);\n println!("result: {:?}", result2);\n}\n\nfn divider(a: f64, b: f64) -> Result {\n if b == 0.0 {\n return Err(MathError::DivisionByZero);\n }\n\n let result = a / b;\n return Ok(result);\n}\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider")," di atas tugasnya adalah melakukan operasi aritmatika pembagian angka numerik ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),", parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"a")," dibagi ",(0,i.kt)("inlineCode",{parentName:"p"},"b"),"."),(0,i.kt)("p",null,"Pada fungsi tersebut terdapat pengecekan apabila nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"b")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),", maka yang dikembalikan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"E")," berisi pesan error, selainnya maka hasil operasi pembagian dikembalikan dibungkus dalam enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),"."),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider")," nilai baliknya bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Dari tipe data yang digunakan nantinya bisa diprediksi pasti akan ada 2 potensi value:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Return value adalah enum value ",(0,i.kt)("inlineCode",{parentName:"li"},"Err"),", muncul ketika nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"b")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"0")),(0,i.kt)("li",{parentName:"ul"},"Return value adalah nilai hasil numerik yang dibungkus oleh enum value ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok"))),(0,i.kt)("p",null,"Output program di atas saat di-run:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Result type",src:n(1463).Z,width:"454",height:"112"})),(0,i.kt)("h2",{id:"a392-pattern-matching-pada-tipe-result"},"A.39.2. Pattern matching pada tipe ",(0,i.kt)("inlineCode",{parentName:"h2"},"Result")),(0,i.kt)("p",null,"Dalam penerapannya, ketika ada data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," artinya data tersebut berpotensi untuk berisi nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),", pasti antara 2 nilai tersebut."),(0,i.kt)("p",null,"Umumnya penggunaan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," selalu diikuti dengan pattern matching menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),". Selain itu keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," sebenarnya juga bisa diterapkan pada pattern matching tipe data ini, namun kurang dianjurkan."),(0,i.kt)("p",null,"Mari kita praktikkan. Ubah isi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," dengan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nmatch result {\n Err(m) => println!("ERROR! {:?}", m),\n Ok(r) => println!("result: {r:.2}"),\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Option type",src:n(2565).Z,width:"450",height:"89"})),(0,i.kt)("p",null,"Bisa dilihat pada kode di atas mudahnya pengambilan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"m")," dari ",(0,i.kt)("inlineCode",{parentName:"p"},"Err(m)")," dan juga ",(0,i.kt)("inlineCode",{parentName:"p"},"r")," dari ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok(r)"),". Penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk seleksi kondisi biasa disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"pattern matching")," dan teknik ini sangat fleksibel dan advance."),(0,i.kt)("p",null,"Sebagai contoh, dengan penerapan match yang seperti ini kita bisa meng-handle 5 skenario seleksi kondisi:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nmatch result {\n Err(MathError::DivisionByZero) => println!("ERROR! unable to divide number by 0"),\n Err(MathError::InfinityNumber) => println!("ERROR! result is infinity number (\u221e)"),\n Err(_) => println!("ERROR! unknown error"),\n Ok(2.0) => println!("the result is 2"),\n Ok(x) => println!("result: {x:.2}"),\n}\n')),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-1: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(MathError::DivisionByZero)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"ERROR! unable to divide number by 0"),"."),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-2: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(MathError::InfinityNumber)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"ERROR! result is infinity number (\u221e)"),"."),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-3: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Err")," selain dari ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(MathError::DivisionByZero)")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(MathError::InfinityNumber)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"ERROR! unknown error"),"."),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-4: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok(2.0)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"the result is 2"),"."),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-5: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok")," selain dari ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok(2.0)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"result: {x:.2}"),".")),(0,i.kt)("h3",{id:"-tips-pattern-matching"},"\u25c9 Tips pattern matching"),(0,i.kt)("p",null,"Silakan perhatikan kode yang sudah kita praktikkan berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nmatch result {\n Err(m) => println!("ERROR! {:?}", m),\n Ok(r) => println!("result: {r:.2}"),\n}\n')),(0,i.kt)("p",null,"Penerapan pattern matching seperti contoh di atas memiliki konsekuensi, yaitu variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"r")," hanya bisa diakses pada block ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok(r)")," saja."),(0,i.kt)("p",null,"Adakalanya kita butuh untuk mengeluarkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"r")," ke luar block. Hal seperti ini mudah untuk dilakukan, dan ada beberapa cara yang bisa dipilih, namun menurut penulis yang paling elegan adalah cara berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let result: f64 = match divider(10.0, 5.0) {\n Err(m) => {\n println!("ERROR! {:?}", m);\n 0.0\n },\n Ok(r) => r,\n };\n\n println!("result: {:?}", result);\n}\n')),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10.0, 5.0)")," mengembalikan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Data tersebut digunakan pada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," seperti biasa. Namun pada contoh di atas ada yang berbeda, yaitu return value dari statement ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," ditampung ke variabel (",(0,i.kt)("inlineCode",{parentName:"p"},"result"),")."),(0,i.kt)("p",null,"Isi dari pattern matching ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," sendiri ada dua:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Ketika block ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(m)")," match, error di-print kemudian nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"0.0")," dijadikan return statement ",(0,i.kt)("inlineCode",{parentName:"li"},"match"),"."),(0,i.kt)("li",{parentName:"ul"},"Ketika block ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok")," match, data ",(0,i.kt)("inlineCode",{parentName:"li"},"r")," dijadikan return value statement ",(0,i.kt)("inlineCode",{parentName:"li"},"match"),".")),(0,i.kt)("p",null,"Dengan penerapan pattern matching seperti di atas, maka variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"result")," akan selalu berisi data hasil operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10.0, 5.0)"),". Dengan pengecualian ketika ada error, pesan errornya dimunculkan kemudian hasil operasi pembagian di-set sebagai ",(0,i.kt)("inlineCode",{parentName:"p"},"0.0"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai pattern matching dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"))),(0,i.kt)("h2",{id:"a393-method-tipe-data-result"},"A.39.3. Method tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Result")),(0,i.kt)("h3",{id:"-method-is_ok--unwrap"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_ok")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap")),(0,i.kt)("p",null,"Isi dari enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok")," bisa diakses tanpa menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," dengan cara memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Sebelum mengakses method tersebut sangat dianjurkan untuk mengecek apakah data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok atau tidak"),", karena jika data adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," menghasilkan error."),(0,i.kt)("p",null,"Pengecekan nilai ok atau tidak bisa dilakukan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_ok"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nif result.is_ok() {\n let number = result.unwrap();\n println!("result: {}", number);\n // result: 2\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Result type",src:n(5086).Z,width:"447",height:"88"})),(0,i.kt)("h3",{id:"-method-as_ref"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"as_ref")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_ref")," digunakan untuk mengakses reference ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"E")," pada ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Method ini sering kali dibutuhkan untuk menghindari terjadinya ",(0,i.kt)("em",{parentName:"p"},"move semantics")," pada owner data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),"."),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_ref")," mengembalikan data dalam tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result<&T, &E>"),". Jadi reference yang dipinjam bukan milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),"-nya melainkan milik ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"E"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let result: Result = divider(10.0, 0.0);\nlet result_borrow: Result<&f64, &MathError> = result.as_ref();\n")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai ",(0,i.kt)("em",{parentName:"p"},"move semantics")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership#a334-copy-semantics-vs-move-semantics"},"Ownership"))),(0,i.kt)("h3",{id:"-method-is_err--err"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_err")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"err")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"err")," mengembalikan data dalam tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Err"),". Pada pengaksesan method ini, pastikan untuk mengecek apakah ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," berisi data error atau ok dengan via method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_err"),". Selain itu, wajib untuk menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_ref")," sebelum method ",(0,i.kt)("inlineCode",{parentName:"p"},"err")," agar ownership data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," tidak berpindah (",(0,i.kt)("em",{parentName:"p"},"move semantics"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 0.0);\nif result.is_err() {\n let err = result.as_ref().err();\n let message = err.unwrap();\n println!("error: {:?}", message);\n // error: DivisionByZero\n}\n')),(0,i.kt)("h3",{id:"-method-ok"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"ok")),(0,i.kt)("p",null,"Aturan yang sama juga berlaku pada pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"ok")," yang mengembalikan data ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),". Method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_ref")," harus diakses terlebih dahulu sebelum memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},"ok")," agar tidak terjadi ",(0,i.kt)("em",{parentName:"p"},"move semantics"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nif result.is_ok() {\n let data = result.as_ref().ok();\n let number = data.unwrap();\n println!("result: {:?}", number);\n // result: 2\n}\n')),(0,i.kt)("h3",{id:"-method-unwrap_or_default"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or_default")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_default")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),", namun jika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," maka yang dikembalikan adalah ",(0,i.kt)("em",{parentName:"p"},"default value")," dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 0.0);\nlet number = result.unwrap_or_default();\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("h3",{id:"-method-unwrap_or"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),", namun jika data ternyata isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," maka yang dikembalikan adalah argument pemanggilan method tersebut."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 0.0);\nlet number = result.unwrap_or(0.0);\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("h3",{id:"-method-unwrap_or_else"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or_else")),(0,i.kt)("p",null,"Method ini mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),", namun jika data isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," maka yang dikembalikan adalah hasil eksekusi closure yang disisipkan saat memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_else"),". Contoh pengaplikasiannya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 0.0);\nlet number = result.unwrap_or_else(|_| 0.0);\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("p",null,"Closure harus dalam notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"FnOnce(E) -> T")," dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," pada konteks ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai closure dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/closures"},"Closures"),".")),(0,i.kt)("h2",{id:"a394-error-handling-tipe-result"},"A.39.4. Error handling tipe ",(0,i.kt)("inlineCode",{parentName:"h2"},"Result")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," banyak digunakan pada fungsi milik Rust standard library, dan kita selaku programmer pastinya juga akan menggunakannya dalam ",(0,i.kt)("em",{parentName:"p"},"real life")," project."),(0,i.kt)("p",null,"Tipe ini dipakai salah satunya untuk manajemen error. Lebih jelasnya mengenai topik tersebut dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/error-handling-panic"},"Error Handling & Panic")),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/result_type"},"github.com/novalagung/dasarpemrogramanrust-example/../result_type")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/closures"},"Closures"))),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Operator ",(0,i.kt)("inlineCode",{parentName:"li"},"?"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch18-03-pattern-syntax.html"},"https://doc.rust-lang.org/book/ch18-03-pattern-syntax.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/result/index.html"},"https://doc.rust-lang.org/std/result/index.html"))))}k.isMDXComponent=!0},1463:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2565:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},5086:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/7794a932.f8711f2e.js b/assets/js/7794a932.f8711f2e.js new file mode 100644 index 00000000..eb2dd46a --- /dev/null +++ b/assets/js/7794a932.f8711f2e.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8358],{3905:(e,a,n)=>{n.d(a,{Zo:()=>u,kt:()=>s});var t=n(7294);function i(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function r(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function l(e){for(var a=1;a=0||(i[n]=e[n]);return i}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var d=t.createContext({}),o=function(e){var a=t.useContext(d),n=a;return e&&(n="function"==typeof e?e(a):l(l({},a),e)),n},u=function(e){var a=o(e.components);return t.createElement(d.Provider,{value:a},e.children)},k={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},m=t.forwardRef((function(e,a){var n=e.components,i=e.mdxType,r=e.originalType,d=e.parentName,u=p(e,["components","mdxType","originalType","parentName"]),m=o(n),s=i,c=m["".concat(d,".").concat(s)]||m[s]||k[s]||r;return n?t.createElement(c,l(l({ref:a},u),{},{components:n})):t.createElement(c,l({ref:a},u))}));function s(e,a){var n=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var r=n.length,l=new Array(r);l[0]=m;var p={};for(var d in a)hasOwnProperty.call(a,d)&&(p[d]=a[d]);p.originalType=e,p.mdxType="string"==typeof e?e:i,l[1]=p;for(var o=2;o{n.r(a),n.d(a,{assets:()=>d,contentTitle:()=>l,default:()=>k,frontMatter:()=>r,metadata:()=>p,toc:()=>o});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:39,title:"A.39. Tipe Data \u279c Result",sidebar_label:"A.39. Tipe Data \u279c Result"},l=void 0,p={unversionedId:"basic/result-type",id:"basic/result-type",title:"A.39. Tipe Data \u279c Result",description:"Chapter ini membahas tentang tipe data Result. Tipe data ini digunakan untuk menampung nilai hasil suatu proses yang isinya adalah bisa sukses (Ok) atau error (Err).",source:"@site/docs/basic/result-type.md",sourceDirName:"basic",slug:"/basic/result-type",permalink:"/basic/result-type",draft:!1,tags:[],version:"current",sidebarPosition:39,frontMatter:{sidebar_position:39,title:"A.39. Tipe Data \u279c Result",sidebar_label:"A.39. Tipe Data \u279c Result"},sidebar:"tutorialSidebar",previous:{title:"A.38. Tipe Data \u279c Option",permalink:"/basic/option-type"},next:{title:"A.40. Pattern Matching",permalink:"/basic/pattern-matching"}},d={},o=[{value:"A.39.1. Konsep Result",id:"a391-konsep-result",level:2},{value:"A.39.2. Pattern matching pada tipe Result",id:"a392-pattern-matching-pada-tipe-result",level:2},{value:"\u25c9 Tips pattern matching",id:"-tips-pattern-matching",level:3},{value:"A.39.3. Method tipe data Result",id:"a393-method-tipe-data-result",level:2},{value:"\u25c9 Method is_ok & unwrap",id:"-method-is_ok--unwrap",level:3},{value:"\u25c9 Method as_ref",id:"-method-as_ref",level:3},{value:"\u25c9 Method is_err & err",id:"-method-is_err--err",level:3},{value:"\u25c9 Method ok",id:"-method-ok",level:3},{value:"\u25c9 Method unwrap_or_default",id:"-method-unwrap_or_default",level:3},{value:"\u25c9 Method unwrap_or",id:"-method-unwrap_or",level:3},{value:"\u25c9 Method unwrap_or_else",id:"-method-unwrap_or_else",level:3},{value:"A.39.4. Error handling tipe Result",id:"a394-error-handling-tipe-result",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:o};function k(e){let{components:a,...r}=e;return(0,i.kt)("wrapper",(0,t.Z)({},u,r,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas tentang tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Tipe data ini digunakan untuk menampung nilai hasil suatu proses yang isinya adalah bisa sukses (",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),") atau error (",(0,i.kt)("inlineCode",{parentName:"p"},"Err"),")."),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," biasa digunakan untuk menampung hasil eksekusi proses dan error handling."),(0,i.kt)("h2",{id:"a391-konsep-result"},"A.39.1. Konsep ",(0,i.kt)("inlineCode",{parentName:"h2"},"Result")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," adalah enum dengan isi 2 buah enum value:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Result::Ok")," (atau ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok"),"), digunakan untuk menandai bahwa data isinya adalah kabar baik (oke / mantab / jos / sukses)."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Result::Err")," (atau ",(0,i.kt)("inlineCode",{parentName:"li"},"Err"),"), digunakan untuk menandai bawah data berisi kabar buruk.")),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"T")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"E")," merupakan parameter generic. Lebih jelasnya mengenai generic dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics"),"."))),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," memiliki notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," digunakan pada enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"E")," digunakan enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Err"),"."),(0,i.kt)("p",null,"Cara penerapan tipe data ini bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nenum MathError {\n DivisionByZero,\n InfinityNumber,\n OtherError,\n}\n\nfn main() {\n let result1 = divider(10.0, 5.0);\n println!("result: {:?}", result1);\n\n let result2: Result = divider(10.0, 0.0);\n println!("result: {:?}", result2);\n}\n\nfn divider(a: f64, b: f64) -> Result {\n if b == 0.0 {\n return Err(MathError::DivisionByZero);\n }\n\n let result = a / b;\n return Ok(result);\n}\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider")," di atas tugasnya adalah melakukan operasi aritmatika pembagian angka numerik ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),", parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"a")," dibagi ",(0,i.kt)("inlineCode",{parentName:"p"},"b"),"."),(0,i.kt)("p",null,"Pada fungsi tersebut terdapat pengecekan apabila nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"b")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),", maka yang dikembalikan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"E")," berisi pesan error, selainnya maka hasil operasi pembagian dikembalikan dibungkus dalam enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),"."),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider")," nilai baliknya bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Dari tipe data yang digunakan nantinya bisa diprediksi pasti akan ada 2 potensi value:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Return value adalah enum value ",(0,i.kt)("inlineCode",{parentName:"li"},"Err"),", muncul ketika nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"b")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"0")),(0,i.kt)("li",{parentName:"ul"},"Return value adalah nilai hasil numerik yang dibungkus oleh enum value ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok"))),(0,i.kt)("p",null,"Output program di atas saat di-run:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Result type",src:n(1463).Z,width:"454",height:"112"})),(0,i.kt)("h2",{id:"a392-pattern-matching-pada-tipe-result"},"A.39.2. Pattern matching pada tipe ",(0,i.kt)("inlineCode",{parentName:"h2"},"Result")),(0,i.kt)("p",null,"Dalam penerapannya, ketika ada data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," artinya data tersebut berpotensi untuk berisi nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),", pasti antara 2 nilai tersebut."),(0,i.kt)("p",null,"Umumnya penggunaan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," selalu diikuti dengan pattern matching menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),". Selain itu keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," sebenarnya juga bisa diterapkan pada pattern matching tipe data ini, namun kurang dianjurkan."),(0,i.kt)("p",null,"Mari kita praktikkan. Ubah isi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," dengan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nmatch result {\n Err(m) => println!("ERROR! {:?}", m),\n Ok(r) => println!("result: {r:.2}"),\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Option type",src:n(2565).Z,width:"450",height:"89"})),(0,i.kt)("p",null,"Bisa dilihat pada kode di atas mudahnya pengambilan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"m")," dari ",(0,i.kt)("inlineCode",{parentName:"p"},"Err(m)")," dan juga ",(0,i.kt)("inlineCode",{parentName:"p"},"r")," dari ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok(r)"),". Penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk seleksi kondisi biasa disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"pattern matching")," dan teknik ini sangat fleksibel dan advance."),(0,i.kt)("p",null,"Sebagai contoh, dengan penerapan match yang seperti ini kita bisa meng-handle 5 skenario seleksi kondisi:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nmatch result {\n Err(MathError::DivisionByZero) => println!("ERROR! unable to divide number by 0"),\n Err(MathError::InfinityNumber) => println!("ERROR! result is infinity number (\u221e)"),\n Err(_) => println!("ERROR! unknown error"),\n Ok(2.0) => println!("the result is 2"),\n Ok(x) => println!("result: {x:.2}"),\n}\n')),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-1: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(MathError::DivisionByZero)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"ERROR! unable to divide number by 0"),"."),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-2: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(MathError::InfinityNumber)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"ERROR! result is infinity number (\u221e)"),"."),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-3: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Err")," selain dari ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(MathError::DivisionByZero)")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(MathError::InfinityNumber)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"ERROR! unknown error"),"."),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-4: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok(2.0)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"the result is 2"),"."),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-5: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok")," selain dari ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok(2.0)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"result: {x:.2}"),".")),(0,i.kt)("h3",{id:"-tips-pattern-matching"},"\u25c9 Tips pattern matching"),(0,i.kt)("p",null,"Silakan perhatikan kode yang sudah kita praktikkan berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nmatch result {\n Err(m) => println!("ERROR! {:?}", m),\n Ok(r) => println!("result: {r:.2}"),\n}\n')),(0,i.kt)("p",null,"Penerapan pattern matching seperti contoh di atas memiliki konsekuensi, yaitu variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"r")," hanya bisa diakses pada block ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok(r)")," saja."),(0,i.kt)("p",null,"Adakalanya kita butuh untuk mengeluarkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"r")," ke luar block. Hal seperti ini mudah untuk dilakukan, dan ada beberapa cara yang bisa dipilih, namun menurut penulis yang paling elegan adalah cara berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let result: f64 = match divider(10.0, 5.0) {\n Err(m) => {\n println!("ERROR! {:?}", m);\n 0.0\n },\n Ok(r) => r,\n };\n\n println!("result: {:?}", result);\n}\n')),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10.0, 5.0)")," mengembalikan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Data tersebut digunakan pada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," seperti biasa. Namun pada contoh di atas ada yang berbeda, yaitu return value dari statement ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," ditampung ke variabel (",(0,i.kt)("inlineCode",{parentName:"p"},"result"),")."),(0,i.kt)("p",null,"Isi dari pattern matching ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," sendiri ada dua:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Ketika block ",(0,i.kt)("inlineCode",{parentName:"li"},"Err(m)")," match, error di-print kemudian nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"0.0")," dijadikan return statement ",(0,i.kt)("inlineCode",{parentName:"li"},"match"),"."),(0,i.kt)("li",{parentName:"ul"},"Ketika block ",(0,i.kt)("inlineCode",{parentName:"li"},"Ok")," match, data ",(0,i.kt)("inlineCode",{parentName:"li"},"r")," dijadikan return value statement ",(0,i.kt)("inlineCode",{parentName:"li"},"match"),".")),(0,i.kt)("p",null,"Dengan penerapan pattern matching seperti di atas, maka variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"result")," akan selalu berisi data hasil operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10.0, 5.0)"),". Dengan pengecualian ketika ada error, pesan errornya dimunculkan kemudian hasil operasi pembagian di-set sebagai ",(0,i.kt)("inlineCode",{parentName:"p"},"0.0"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai pattern matching dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"))),(0,i.kt)("h2",{id:"a393-method-tipe-data-result"},"A.39.3. Method tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Result")),(0,i.kt)("h3",{id:"-method-is_ok--unwrap"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_ok")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap")),(0,i.kt)("p",null,"Isi dari enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok")," bisa diakses tanpa menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," dengan cara memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Sebelum mengakses method tersebut sangat dianjurkan untuk mengecek apakah data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok atau tidak"),", karena jika data adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," menghasilkan error."),(0,i.kt)("p",null,"Pengecekan nilai ok atau tidak bisa dilakukan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_ok"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nif result.is_ok() {\n let number = result.unwrap();\n println!("result: {}", number);\n // result: 2\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Result type",src:n(5086).Z,width:"447",height:"88"})),(0,i.kt)("h3",{id:"-method-as_ref"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"as_ref")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_ref")," digunakan untuk mengakses reference ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"E")," pada ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Method ini sering kali dibutuhkan untuk menghindari terjadinya ",(0,i.kt)("em",{parentName:"p"},"move semantics")," pada owner data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),"."),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_ref")," mengembalikan data dalam tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result<&T, &E>"),". Jadi reference yang dipinjam bukan milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),"-nya melainkan milik ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"E"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let result: Result = divider(10.0, 0.0);\nlet result_borrow: Result<&f64, &MathError> = result.as_ref();\n")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai ",(0,i.kt)("em",{parentName:"p"},"move semantics")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership#a334-copy-semantics-vs-move-semantics"},"Ownership"))),(0,i.kt)("h3",{id:"-method-is_err--err"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_err")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"err")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"err")," mengembalikan data dalam tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Err"),". Pada pengaksesan method ini, pastikan untuk mengecek apakah ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," berisi data error atau ok dengan via method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_err"),". Selain itu, wajib untuk menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_ref")," sebelum method ",(0,i.kt)("inlineCode",{parentName:"p"},"err")," agar ownership data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," tidak berpindah (",(0,i.kt)("em",{parentName:"p"},"move semantics"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 0.0);\nif result.is_err() {\n let err = result.as_ref().err();\n let message = err.unwrap();\n println!("error: {:?}", message);\n // error: DivisionByZero\n}\n')),(0,i.kt)("h3",{id:"-method-ok"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"ok")),(0,i.kt)("p",null,"Aturan yang sama juga berlaku pada pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"ok")," yang mengembalikan data ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),". Method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_ref")," harus diakses terlebih dahulu sebelum memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},"ok")," agar tidak terjadi ",(0,i.kt)("em",{parentName:"p"},"move semantics"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 5.0);\nif result.is_ok() {\n let data = result.as_ref().ok();\n let number = data.unwrap();\n println!("result: {:?}", number);\n // result: 2\n}\n')),(0,i.kt)("h3",{id:"-method-unwrap_or_default"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or_default")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_default")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),", namun jika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," maka yang dikembalikan adalah ",(0,i.kt)("em",{parentName:"p"},"default value")," dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 0.0);\nlet number = result.unwrap_or_default();\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("h3",{id:"-method-unwrap_or"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),", namun jika data ternyata isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," maka yang dikembalikan adalah argument pemanggilan method tersebut."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 0.0);\nlet number = result.unwrap_or(0.0);\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("h3",{id:"-method-unwrap_or_else"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or_else")),(0,i.kt)("p",null,"Method ini mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Ok"),", namun jika data isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Err")," maka yang dikembalikan adalah hasil eksekusi closure yang disisipkan saat memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_else"),". Contoh pengaplikasiannya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10.0, 0.0);\nlet number = result.unwrap_or_else(|_| 0.0);\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("p",null,"Closure harus dalam notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"FnOnce(E) -> T")," yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," pada konteks ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai closure dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/closures"},"Closures"),".")),(0,i.kt)("h2",{id:"a394-error-handling-tipe-result"},"A.39.4. Error handling tipe ",(0,i.kt)("inlineCode",{parentName:"h2"},"Result")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," banyak digunakan pada fungsi milik Rust standard library, dan kita selaku programmer pastinya juga akan menggunakannya dalam ",(0,i.kt)("em",{parentName:"p"},"real life")," project."),(0,i.kt)("p",null,"Tipe ini dipakai salah satunya untuk manajemen error. Lebih jelasnya mengenai topik tersebut dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/error-handling-panic"},"Error Handling & Panic")),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/result_type"},"github.com/novalagung/dasarpemrogramanrust-example/../result_type")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/closures"},"Closures"))),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Operator ",(0,i.kt)("inlineCode",{parentName:"li"},"?"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch18-03-pattern-syntax.html"},"https://doc.rust-lang.org/book/ch18-03-pattern-syntax.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/result/index.html"},"https://doc.rust-lang.org/std/result/index.html"))))}k.isMDXComponent=!0},1463:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2565:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},5086:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/7d03ccaf.46ca6fcc.js b/assets/js/7d03ccaf.a18e440e.js similarity index 78% rename from assets/js/7d03ccaf.46ca6fcc.js rename to assets/js/7d03ccaf.a18e440e.js index 5fb518cf..eb823d10 100644 --- a/assets/js/7d03ccaf.46ca6fcc.js +++ b/assets/js/7d03ccaf.a18e440e.js @@ -1 +1 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[1059],{3905:(a,e,n)=>{n.d(e,{Zo:()=>s,kt:()=>c});var r=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function i(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);e&&(r=r.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,r)}return n}function o(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var p=r.createContext({}),u=function(a){var e=r.useContext(p),n=e;return a&&(n="function"==typeof a?a(e):o(o({},e),a)),n},s=function(a){var e=u(a.components);return r.createElement(p.Provider,{value:e},a.children)},m={inlineCode:"code",wrapper:function(a){var e=a.children;return r.createElement(r.Fragment,{},e)}},g=r.forwardRef((function(a,e){var n=a.components,t=a.mdxType,i=a.originalType,p=a.parentName,s=l(a,["components","mdxType","originalType","parentName"]),g=u(n),c=t,d=g["".concat(p,".").concat(c)]||g[c]||m[c]||i;return n?r.createElement(d,o(o({ref:e},s),{},{components:n})):r.createElement(d,o({ref:e},s))}));function c(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var i=n.length,o=new Array(i);o[0]=g;var l={};for(var p in e)hasOwnProperty.call(e,p)&&(l[p]=e[p]);l.originalType=a,l.mdxType="string"==typeof a?a:t,o[1]=l;for(var u=2;u{n.r(e),n.d(e,{assets:()=>p,contentTitle:()=>o,default:()=>m,frontMatter:()=>i,metadata:()=>l,toc:()=>u});var r=n(7462),t=(n(7294),n(3905));const i={sidebar_position:2,title:"Pengenalan Rust Programming",sidebar_label:"Pengenalan Rust Programming"},o=void 0,l={unversionedId:"pengelanan-rust-programming",id:"pengelanan-rust-programming",title:"Pengenalan Rust Programming",description:"Rust adalah sebuah bahasa pemrograman general purpose yang fokus pada performance, type safety, dan concurrency. Bahasa ini diciptakan sebagai alternatif bahasa pemrograman yang berfokus pada hal-hal yang cukup low-level tapi tetap men-support fitur yang sifatnya high-level.",source:"@site/docs/pengelanan-rust-programming.md",sourceDirName:".",slug:"/pengelanan-rust-programming",permalink:"/pengelanan-rust-programming",draft:!1,tags:[],version:"current",sidebarPosition:2,frontMatter:{sidebar_position:2,title:"Pengenalan Rust Programming",sidebar_label:"Pengenalan Rust Programming"},sidebar:"tutorialSidebar",previous:{title:"Dasar Pemrograman Rust",permalink:"/"},next:{title:"Contribution & Tech Stack",permalink:"/CONTRIBUTING"}},p={},u=[],s={toc:u};function m(a){let{components:e,...n}=a;return(0,t.kt)("wrapper",(0,r.Z)({},s,n,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Rust adalah sebuah bahasa pemrograman general purpose yang fokus pada performance, type safety, dan concurrency. Bahasa ini diciptakan sebagai alternatif bahasa pemrograman yang berfokus pada hal-hal yang cukup low-level tapi tetap men-support fitur yang sifatnya high-level."),(0,t.kt)("p",null,"Rust dikembangkan oleh ",(0,t.kt)("a",{parentName:"p",href:"https://twitter.com/graydon_pub"},"Graydon Hoare")," sewaktu ia masih bekerja di Mozilla pada tahun 2009, kemudian di-maintain dibawah naungan Rust Foundation hingga sekarang."),(0,t.kt)("p",null,"Rust memiliki beberapa kelebihan dibanding bahasa system-programming lainnya, yang di antaranya adalah:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Keamanan memory yang sangat bagus, dengan pengecekan ada di waktu kompilasi"),(0,t.kt)("li",{parentName:"ul"},"Ownership memory management"),(0,t.kt)("li",{parentName:"ul"},"Type system (traits, generic, struct, dan lainnya)"),(0,t.kt)("li",{parentName:"ul"},"Macros untuk metaprogramming"),(0,t.kt)("li",{parentName:"ul"},"Tooling dan ",(0,t.kt)("inlineCode",{parentName:"li"},"std")," library yang lengkap"),(0,t.kt)("li",{parentName:"ul"},"Built-in package management tool (cargo & crates.io)"),(0,t.kt)("li",{parentName:"ul"},"Efisien dan memiliki performa tinggi"),(0,t.kt)("li",{parentName:"ul"},"Dukungan komunitas yang bagus. Banyak library open source yang tersedia")),(0,t.kt)("p",null,"Pada buku ini (terutama semua serial chapter A) kita akan belajar tentang dasar pemrograman Rust, mulai dari 0."))}m.isMDXComponent=!0}}]); \ No newline at end of file +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[1059],{3905:(a,e,n)=>{n.d(e,{Zo:()=>s,kt:()=>c});var r=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function i(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);e&&(r=r.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,r)}return n}function o(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var p=r.createContext({}),u=function(a){var e=r.useContext(p),n=e;return a&&(n="function"==typeof a?a(e):o(o({},e),a)),n},s=function(a){var e=u(a.components);return r.createElement(p.Provider,{value:e},a.children)},m={inlineCode:"code",wrapper:function(a){var e=a.children;return r.createElement(r.Fragment,{},e)}},g=r.forwardRef((function(a,e){var n=a.components,t=a.mdxType,i=a.originalType,p=a.parentName,s=l(a,["components","mdxType","originalType","parentName"]),g=u(n),c=t,d=g["".concat(p,".").concat(c)]||g[c]||m[c]||i;return n?r.createElement(d,o(o({ref:e},s),{},{components:n})):r.createElement(d,o({ref:e},s))}));function c(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var i=n.length,o=new Array(i);o[0]=g;var l={};for(var p in e)hasOwnProperty.call(e,p)&&(l[p]=e[p]);l.originalType=a,l.mdxType="string"==typeof a?a:t,o[1]=l;for(var u=2;u{n.r(e),n.d(e,{assets:()=>p,contentTitle:()=>o,default:()=>m,frontMatter:()=>i,metadata:()=>l,toc:()=>u});var r=n(7462),t=(n(7294),n(3905));const i={sidebar_position:2,title:"Pengenalan Rust Programming",sidebar_label:"Pengenalan Rust Programming"},o=void 0,l={unversionedId:"pengelanan-rust-programming",id:"pengelanan-rust-programming",title:"Pengenalan Rust Programming",description:"Rust adalah sebuah bahasa pemrograman general purpose yang fokus pada performance, type safety, dan concurrency. Bahasa ini diciptakan sebagai alternatif bahasa pemrograman yang berfokus pada hal-hal yang cukup low-level tapi tetap men-support fitur yang sifatnya high-level.",source:"@site/docs/pengelanan-rust-programming.md",sourceDirName:".",slug:"/pengelanan-rust-programming",permalink:"/pengelanan-rust-programming",draft:!1,tags:[],version:"current",sidebarPosition:2,frontMatter:{sidebar_position:2,title:"Pengenalan Rust Programming",sidebar_label:"Pengenalan Rust Programming"},sidebar:"tutorialSidebar",previous:{title:"Dasar Pemrograman Rust",permalink:"/"},next:{title:"Contribution & Tech Stack",permalink:"/CONTRIBUTING"}},p={},u=[],s={toc:u};function m(a){let{components:e,...n}=a;return(0,t.kt)("wrapper",(0,r.Z)({},s,n,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Rust adalah sebuah bahasa pemrograman general purpose yang fokus pada performance, type safety, dan concurrency. Bahasa ini diciptakan sebagai alternatif bahasa pemrograman yang berfokus pada hal-hal yang cukup low-level tapi tetap men-support fitur yang sifatnya high-level."),(0,t.kt)("p",null,"Rust dikembangkan oleh ",(0,t.kt)("a",{parentName:"p",href:"https://twitter.com/graydon_pub"},"Graydon Hoare")," sewaktu ia masih bekerja di Mozilla pada tahun 2009, kemudian di-maintain di bawah naungan Rust Foundation hingga sekarang."),(0,t.kt)("p",null,"Rust memiliki beberapa kelebihan dibanding bahasa system-programming lainnya, yang di antaranya adalah:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Keamanan memory yang sangat bagus, dengan pengecekan ada di waktu kompilasi"),(0,t.kt)("li",{parentName:"ul"},"Ownership memory management"),(0,t.kt)("li",{parentName:"ul"},"Type system (traits, generic, struct, dan lainnya)"),(0,t.kt)("li",{parentName:"ul"},"Macros untuk metaprogramming"),(0,t.kt)("li",{parentName:"ul"},"Tooling dan ",(0,t.kt)("inlineCode",{parentName:"li"},"std")," library yang lengkap"),(0,t.kt)("li",{parentName:"ul"},"Built-in package management tool (cargo & crates.io)"),(0,t.kt)("li",{parentName:"ul"},"Efisien dan memiliki performa tinggi"),(0,t.kt)("li",{parentName:"ul"},"Dukungan komunitas yang bagus. Banyak library open source yang tersedia")),(0,t.kt)("p",null,"Pada buku ini (terutama semua serial chapter A) kita akan belajar tentang dasar pemrograman Rust, mulai dari 0."))}m.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/82a9b669.402fdaf8.js b/assets/js/82a9b669.402fdaf8.js new file mode 100644 index 00000000..d1f80607 --- /dev/null +++ b/assets/js/82a9b669.402fdaf8.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[3257],{3905:(e,a,t)=>{t.d(a,{Zo:()=>m,kt:()=>c});var n=t(7294);function i(e,a,t){return a in e?Object.defineProperty(e,a,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[a]=t,e}function r(e,a){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);a&&(n=n.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),t.push.apply(t,n)}return t}function l(e){for(var a=1;a=0||(i[t]=e[t]);return i}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(i[t]=e[t])}return i}var p=n.createContext({}),o=function(e){var a=n.useContext(p),t=a;return e&&(t="function"==typeof e?e(a):l(l({},a),e)),t},m=function(e){var a=o(e.components);return n.createElement(p.Provider,{value:a},e.children)},A={inlineCode:"code",wrapper:function(e){var a=e.children;return n.createElement(n.Fragment,{},a)}},k=n.forwardRef((function(e,a){var t=e.components,i=e.mdxType,r=e.originalType,p=e.parentName,m=d(e,["components","mdxType","originalType","parentName"]),k=o(t),c=i,u=k["".concat(p,".").concat(c)]||k[c]||A[c]||r;return t?n.createElement(u,l(l({ref:a},m),{},{components:t})):n.createElement(u,l({ref:a},m))}));function c(e,a){var t=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var r=t.length,l=new Array(r);l[0]=k;var d={};for(var p in a)hasOwnProperty.call(a,p)&&(d[p]=a[p]);d.originalType=e,d.mdxType="string"==typeof e?e:i,l[1]=d;for(var o=2;o{t.r(a),t.d(a,{assets:()=>p,contentTitle:()=>l,default:()=>A,frontMatter:()=>r,metadata:()=>d,toc:()=>o});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:48,title:"A.48. Trait \u279c Iterator",sidebar_label:"A.48. Trait \u279c Iterator"},l=void 0,d={unversionedId:"basic/trait-iterator",id:"basic/trait-iterator",title:"A.48. Trait \u279c Iterator",description:"Iterator adalah salah satu trait dan tipe data custom penting pada Rust programming, gunanya untuk iterasi data. Di chapter ini kita akan mempelajarinya beserta beberapa module item lainnya yang masih relevan dengan topik Iterator.",source:"@site/docs/basic/trait-iterator.md",sourceDirName:"basic",slug:"/basic/trait-iterator",permalink:"/basic/trait-iterator",draft:!1,tags:[],version:"current",sidebarPosition:48,frontMatter:{sidebar_position:48,title:"A.48. Trait \u279c Iterator",sidebar_label:"A.48. Trait \u279c Iterator"},sidebar:"tutorialSidebar",previous:{title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",permalink:"/basic/trait-function"},next:{title:"A.49. Attributes",permalink:"/basic/attributes"}},p={},o=[{value:"A.48.1. Iterator & IntoIterator",id:"a481-iterator--intoiterator",level:2},{value:"\u25c9 Trait Iterator & IntoIterator",id:"-trait-iterator--intoiterator",level:3},{value:"\u25c9 Struct Iterator",id:"-struct-iterator",level:3},{value:"A.48.2. Pemanfaatan tipe data Iterator",id:"a482-pemanfaatan-tipe-data-iterator",level:2},{value:"\u25c9 Praktik ke-1",id:"-praktik-ke-1",level:3},{value:"\u25c9 Praktik ke-2",id:"-praktik-ke-2",level:3},{value:"A.48.3. Method tipe data Iterator",id:"a483-method-tipe-data-iterator",level:2},{value:"\u25c9 Method iter, iter_mut, into_iter",id:"-method-iter-iter_mut-into_iter",level:3},{value:"\u25c9 Method collect",id:"-method-collect",level:3},{value:"\u25c9 Method map",id:"-method-map",level:3},{value:"\u25c9 Method rev",id:"-method-rev",level:3},{value:"\u25c9 Method filter",id:"-method-filter",level:3},{value:"\u25c9 Method cloned",id:"-method-cloned",level:3},{value:"\u25c9 Method copied",id:"-method-copied",level:3},{value:"\u25c9 Method cmp",id:"-method-cmp",level:3},{value:"\u25c9 Method count",id:"-method-count",level:3},{value:"\u25c9 Method eq, ne, gt, ge, lt, le",id:"-method-eq-ne-gt-ge-lt-le",level:3},{value:"\u25c9 Method find",id:"-method-find",level:3},{value:"\u25c9 Method last",id:"-method-last",level:3},{value:"\u25c9 Method fold",id:"-method-fold",level:3},{value:"\u25c9 Method inspect",id:"-method-inspect",level:3},{value:"\u25c9 Method sum",id:"-method-sum",level:3},{value:"\u25c9 Method reduce",id:"-method-reduce",level:3},{value:"\u25c9 Method min, max",id:"-method-min-max",level:3},{value:"A.48.4. Method lainnya",id:"a484-method-lainnya",level:2},{value:"\u25c9 Method sort milik Vec<T>",id:"-method-sort-milik-vect",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:o};function A(e){let{components:a,...r}=e;return(0,i.kt)("wrapper",(0,n.Z)({},m,r,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Iterator adalah salah satu trait dan tipe data custom penting pada Rust programming, gunanya untuk iterasi data. Di chapter ini kita akan mempelajarinya beserta beberapa module item lainnya yang masih relevan dengan topik Iterator."),(0,i.kt)("h2",{id:"a481-iterator--intoiterator"},"A.48.1. ",(0,i.kt)("inlineCode",{parentName:"h2"},"Iterator")," & ",(0,i.kt)("inlineCode",{parentName:"h2"},"IntoIterator")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," adalah sebuah nama yang dipakai sebagai nama module item dalam ",(0,i.kt)("strong",{parentName:"p"},"Rust Standard Library")," maupun ",(0,i.kt)("strong",{parentName:"p"},"Rust Core Library"),", digunakan untuk iterasi data dan operasi lain yang berhubungan dengannya."),(0,i.kt)("p",null,"Iterator sendiri merupakan istilah untuk object atau trait yang bisa diiterasi, baik menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," ataupun menggunakan method iterator seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each")," (yang juga akan kita bahas di sini)."),(0,i.kt)("h3",{id:"-trait-iterator--intoiterator"},"\u25c9 Trait ",(0,i.kt)("inlineCode",{parentName:"h3"},"Iterator")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"IntoIterator")),(0,i.kt)("p",null,"Dalam perulangan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", tipe data variabel yang digunakan harus memiliki trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),". Jika tidak, pasti error muncul."),(0,i.kt)("p",null,"Tipe data slice di Rust by default mengadopsi trait bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"IntoIterator"),", yang trait ini digunakan untuk konversi data bertipe slice ke bentuk iterator."),(0,i.kt)("p",null,"Dalam praktiknya, tidak perlu mengakses method tertentu untuk mendapatkan object iterator suatu data. Cukup gunakan variabel tipe data slice pada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", maka proses konversi ke bentuk iterator akan dilakukan oleh Rust secara otomatis dibalik layar."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// array\nlet data_arr = ["a", "b", "c", "d"];\nfor e in &data_arr {\n print!("{e:?} ")\n}\n\nprintln!();\n\n// vector\nlet data_vec = vec!["a", "b", "c", "d"];\nfor e in &data_vec {\n print!("{e:?} ")\n}\n\nprintln!();\n\n// slice\nlet data_vec = vec!["a", "b", "c", "d"];\nlet data_slice_vec = &data_vec[..];\nfor e in data_slice_vec {\n print!("{e:?} ")\n}\n')),(0,i.kt)("p",null,"Dianjurkan untuk selalu menggunakan teknik ",(0,i.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing")," dalam penggunaan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," pada tipe data yang karakteristiknya adalah ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership#a344-copy-semantics-vs-move-semantics"},"Move Semantics"),", hal ini karena dalam penerapan keyword tersebut, terjadi proses konversi tipe data dengan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"IntoIterator")," ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),"."),(0,i.kt)("p",null,"Jika data yang digunakan bukan data borrow, maka pasti owner berpindah."),(0,i.kt)("h3",{id:"-struct-iterator"},"\u25c9 Struct ",(0,i.kt)("inlineCode",{parentName:"h3"},"Iterator")),(0,i.kt)("p",null,"Selain trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),", ada juga tipe data struct bernama sama, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),". Perbedaan antara trait vs struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," adalah tipe struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," memiliki beberapa method untuk keperluan iterasi object. Jadi dengan memanfaatkan method-method tersebut, kita bisa melakukan iterasi data dan operasi sejenisnya tanpa menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),"."),(0,i.kt)("p",null,"Semua tipe data slice bisa dikonversi ke tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),", caranya dengan mengakses method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," (atau method ",(0,i.kt)("inlineCode",{parentName:"p"},"chars")," khusus untuk tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),")."),(0,i.kt)("p",null,"Dari object iterator tersebut, perulangan bisa dilakukan via keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", atau dengan memanfaatkan method bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each"),". Perbedaannya pada iterasi menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each"),", block perulangan dituliskan dalam bentuk closure."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// array\nlet data_arr = ["a", "b", "c", "d"];\nlet iterator_arr = data_arr.iter();\niterator_arr.for_each(|e| {\n print!("{e:?} ")\n});\n\nprintln!();\n\n// vector\nlet data_vec = vec!["a", "b", "c", "d"];\ndata_vec.iter().for_each(|e| {\n print!("{e:?} ")\n});\n\nprintln!();\n\n// slice from vector\nlet data_vec = vec!["a", "b", "c", "d"];\nlet data_slice_vec = &data_vec[..];\ndata_slice_vec.iter().for_each(|e| {\n print!("{e:?} ")\n});\n\nprintln!();\n\n// slice from String\nlet data_str = "abcd".to_string();\nlet data_borrow_str = &data_str;\ndata_borrow_str.chars().for_each(|e| {\n print!("{e:?} ")\n});\n\nprintln!();\n')),(0,i.kt)("p",null,"Bisa dilihat pada contoh di atas, data slice diambil objek Iterator-nya menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," (atau ",(0,i.kt)("inlineCode",{parentName:"p"},"chars")," khusus untuk tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"), kemudian di-iterasi menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(3753).Z,width:"456",height:"149"})),(0,i.kt)("h2",{id:"a482-pemanfaatan-tipe-data-iterator"},"A.48.2. Pemanfaatan tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Iterator")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," memiliki cukup banyak method untuk keperluan operasi data iterator, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"map"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"filter"),", dan lainnya."),(0,i.kt)("h3",{id:"-praktik-ke-1"},"\u25c9 Praktik ke-1"),(0,i.kt)("p",null,"Sebagai contoh, kita akan buat sebuah program sederhana yang melakukan operasi kuadrat pada elemen tiap-tiap array."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_arr = [1, 3, 5];\n\nlet doubles: Vec = data_arr\n .iter()\n .map(|e| e * e)\n .collect();\n\nprintln!("{doubles:?}");\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(9339).Z,width:"447",height:"121"})),(0,i.kt)("p",null,"Penjelasan:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"data_arr")," yang bertipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"[i32; 3]")," dikonversi ke tipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"Iterator")," menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"iter"),"."),(0,i.kt)("li",{parentName:"ol"},"Object ",(0,i.kt)("inlineCode",{parentName:"li"},"Iterator")," kemudian di-iterasi menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"map")," dan nilai baliknya dijadikan ",(0,i.kt)("em",{parentName:"li"},"replacement")," data elemen tersebut."),(0,i.kt)("li",{parentName:"ol"},"Object ",(0,i.kt)("inlineCode",{parentName:"li"},"Iterator")," kemudian di-collect data-nya ke bentuk ",(0,i.kt)("inlineCode",{parentName:"li"},"Vec")," menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"collect"),".")),(0,i.kt)("h3",{id:"-praktik-ke-2"},"\u25c9 Praktik ke-2"),(0,i.kt)("p",null,"Contoh lain penerapan method iterator bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec!["1", "2", "3", "4", "a"];\n\nlet numbers: Vec = data_vec\n .iter()\n .map(|e| -> i32 {\n match e.parse::() {\n Ok(n) => n,\n Err(_) => 0,\n }\n })\n .filter(|e| *e > 0 && *e % 2 == 0)\n .rev()\n .collect::>();\n\nprintln!("{numbers:?}");\n')),(0,i.kt)("p",null,"Program di atas melakukan beberapa hal:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Data ",(0,i.kt)("inlineCode",{parentName:"p"},"data_vec")," yang merupakan koleksi string di-konversi menjadi object ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter"),".")),(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Kemudian method ",(0,i.kt)("inlineCode",{parentName:"p"},"map")," diakses. Setiap elemen ",(0,i.kt)("inlineCode",{parentName:"p"},"data_vec")," di-iterasi, kemudian dikonversi dari ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", lalu dijadikan ",(0,i.kt)("em",{parentName:"p"},"replacement")," data elemen tersebut. Jika proses konversi gagal, maka angka ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," digunakan sebagai data element tersebut."),(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},"Sampai sini, data yang sebelumnya ",(0,i.kt)("inlineCode",{parentName:"li"},'["1", "2", "3", "4", "a"]')," sekarang menjadi ",(0,i.kt)("inlineCode",{parentName:"li"},"[1, 2, 3, 4, 0]"),"."))),(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Selanjutnya, dilakukan proses filtering menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"filter")," dengan kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"*e > 0 && *e % 2 == 0")," yang kurang lebih artinya, jika nilai ",(0,i.kt)("em",{parentName:"p"},"dereference")," ",(0,i.kt)("inlineCode",{parentName:"p"},"e")," lebih besar dari ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," dan nilai tersebut adalah genap, maka filter bernilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),"."),(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},"Sampai sini, data yang sebelumnya ",(0,i.kt)("inlineCode",{parentName:"li"},"[1, 2, 3, 4, 0]")," sekarang menjadi ",(0,i.kt)("inlineCode",{parentName:"li"},"[2, 4]"),"."))),(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Kemudian, data kolektif tersebut dibalik urutan elemennya menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"rev"),"."),(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},"Sampai sini, data yang sebelumnya ",(0,i.kt)("inlineCode",{parentName:"li"},"[2, 4]")," sekarang menjadi ",(0,i.kt)("inlineCode",{parentName:"li"},"[4, 2]"),"."))),(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Terakhir data di-collect menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"collect")," dengan ditentukan tipe data masing-masing elemen adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),"."))),(0,i.kt)("p",null,"Hasilnya ketika di run:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(6644).Z,width:"447",height:"118"})),(0,i.kt)("h2",{id:"a483-method-tipe-data-iterator"},"A.48.3. Method tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Iterator")),(0,i.kt)("h3",{id:"-method-iter-iter_mut-into_iter"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"iter"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"iter_mut"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"into_iter")),(0,i.kt)("p",null,"Ketiga method ini berguna untuk konversi data slice ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),", perbedannya:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," mengembalikan data ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," yang isinya adalah reference (",(0,i.kt)("inlineCode",{parentName:"p"},"&T"),") setiap element. Contoh penerapan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\ndata_vec.iter().for_each(|d| {\n println!("{}", *d * 2)\n});\n\nfor d in &data_vec {\n println!("{}", *d * 2)\n}\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter_mut")," mengembalikan data ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," yang isinya adalah mutable reference (",(0,i.kt)("inlineCode",{parentName:"p"},"&mut T"),") setiap element. Contoh penerapan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut data_vec = vec![1, 2, 3, 4];\n\ndata_vec.iter_mut().for_each(|d| {\n *d = *d * 2;\n});\nprintln!("{:?}", data_vec);\n\nfor d in &mut data_vec {\n *d = *d * 2;\n}\nprintln!("{:?}", data_vec);\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"into_iter")," mengkonversi data slice ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," (",(0,i.kt)("em",{parentName:"p"},"move semantics"),")."),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\ndata_vec.into_iter().for_each(|d| {\n println!("{}", d * 2)\n});\n\n// statement di bawah ini menghasilkan error,\n// karena ownership `data_vec` telah berpindah setelah method `into_iter` dipanggil\n\n// println!("{}", data_vec)\n')))),(0,i.kt)("h3",{id:"-method-collect"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"collect")),(0,i.kt)("p",null,"Method ini digunakan untuk ",(0,i.kt)("em",{parentName:"p"},"collecting")," data Iterator ke bentuk tertentu. Contoh penerapannya bisa dilihat pada kode berikut ini."),(0,i.kt)("p",null,"Data ",(0,i.kt)("inlineCode",{parentName:"p"},"data_vec")," yang merupakan vektor bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," dikonversi ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," kemudian di-collect datanya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec: Vec = vec![1, 2, 3, 4];\n\nlet result: Vec<&i32> = data_vec.iter().collect();\nprintln!("{:?}", result);\n// [1, 2, 3, 4]\n')),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," menghasilkan object ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," yang menampung reference ",(0,i.kt)("inlineCode",{parentName:"p"},"&T")," setiap element slice. Hal ini membuat penerapan method ",(0,i.kt)("inlineCode",{parentName:"p"},"collect")," menghasilkan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec<&i32>")," (bukan ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),")."),(0,i.kt)("h3",{id:"-method-map"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"map")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"map")," digunakan untuk mapping element setiap data iterator ke nilai baru."),(0,i.kt)("p",null,"Sebelumnya kita telah mempelajari method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"collect")," yang keduanya jika dikombinasikan dan digunakan pada tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," hasilnya adalah data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec<&i32>"),". Dengan memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"map"),", tipe data setiap element bisa di-mapping ke nilai baru dengan tipe data berbeda, misalnya menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec: Vec = vec![1, 2, 3, 4];\n\nlet result1: Vec<&i32> = data_vec.iter().collect();\nprintln!("{:?}", result1);\n// [1, 2, 3, 4]\n\nlet result2: Vec = data_vec\n .iter()\n .map(|d: &i32| -> i32 { *d })\n .collect();\nprintln!("{:?}", result2);\n// [1, 2, 3, 4]\n')),(0,i.kt)("p",null,"Contoh lainnya bisa dilihat pada ",(0,i.kt)("em",{parentName:"p"},"section")," ",(0,i.kt)("a",{parentName:"p",href:"/basic/trait-iterator#a482-pemanfaatan-tipe-data-iterator"},"A.48.2. Pemanfaatan tipe data ",(0,i.kt)("inlineCode",{parentName:"a"},"Iterator")),", di situ terdapat operasi mapping data slice numerik ke bentuk yang sama tapi nilai setiap element adalah kuadrat, dan ke bentuk lain dengan tipe data berbeda."),(0,i.kt)("h3",{id:"-method-rev"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"rev")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"rev")," kependekan dari ",(0,i.kt)("em",{parentName:"p"},"reverse"),", digunakan untuk membalikan urutan data slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nprintln!("{:?}", data_vec);\n// [1, 2, 3, 4]\n\nlet result: Vec<&i32> = data_vec.iter().rev().collect();\nprintln!("{:?}", result);\n// [4, 3, 2, 1]\n')),(0,i.kt)("h3",{id:"-method-filter"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"filter")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"filter")," digunakan untuk memfilter element data slice. Data kolektif diiterasi kemudian dicek menggunakan closure, jika nilai balik bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," maka elemen tersebut masuk dalam dalam hasil filter, selebihnya maka dianggap tidak memenuhi kondisi filter dan elemen di-",(0,i.kt)("em",{parentName:"p"},"exclude"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\nlet odd: Vec<&i32> = data_vec.iter().filter(|d| *d % 2 != 0).collect();\nprintln!("odd numbers: {:?}", odd);\n// odd numbers: [1, 3]\n\nlet even: Vec<&i32> = data_vec.iter().filter(|d| *d % 2 == 0).collect();\nprintln!("even numbers: {:?}", even);\n// odd numbers: [2, 4]\n')),(0,i.kt)("h3",{id:"-method-cloned"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"cloned")),(0,i.kt)("p",null,"Digunakan untuk ",(0,i.kt)("em",{parentName:"p"},"cloning")," data slice secara keseluruhan tanpa mengubah tipe data. Hasilnya adalah data dengan skema sama persis (dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),") tapi berbeda owner."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet result: Vec = data_vec.iter().cloned().collect();\nprintln!("{:?}", result);\n// [1, 2, 3, 4]\n')),(0,i.kt)("h3",{id:"-method-copied"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"copied")),(0,i.kt)("p",null,"Secara ",(0,i.kt)("em",{parentName:"p"},"high-level")," ",(0,i.kt)("inlineCode",{parentName:"p"},"copied")," menghasilkan output yang sama dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"cloned"),". Namun jika dibahas dari sisi management memory-nya, ada perbedaan yang cukup besar."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet result: Vec = data_vec.iter().copied().collect();\nprintln!("{:?}", result);\n// [1, 2, 3, 4]\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih detailnya mengenai clone vs copy akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/copy-clone-move-drop"},"Copy, Clone, Move, Drop"))),(0,i.kt)("h3",{id:"-method-cmp"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"cmp")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"cmp")," kependekan dari ",(0,i.kt)("em",{parentName:"p"},"compare"),", digunakan untuk membandingkan 2 buah data iterasi dengan nilai balik bertipe enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Ordering"),". Dari tipe data enum tersebut nantinya bisa dicek apakah 2 buah data slice tersebut sama (secara ",(0,i.kt)("em",{parentName:"p"},"Lexicographical"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec1 = vec![1, 2, 3, 4];\nlet data_vec2 = vec![1, 2, 3, 4];\nlet result = data_vec1.iter().cmp(data_vec2.iter());\nprintln!("{:?}", result.is_eq());\n// true\n')),(0,i.kt)("p",null,"Enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Ordering")," memiliki beberapa method, salah satunya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"is_eq")," yang mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," jika dua buah data slice tersebut adalah sama."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih detailnya mengenai ",(0,i.kt)("inlineCode",{parentName:"p"},"Ordering")," akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/enum-ordering"},"Enum Ordering"))),(0,i.kt)("h3",{id:"-method-count"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"count")),(0,i.kt)("p",null,"Digunakan untuk melihat size dari elemen Iterator."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet length = data_vec.iter().count();\nprintln!("{:?}", length);\n// 4\n')),(0,i.kt)("h3",{id:"-method-eq-ne-gt-ge-lt-le"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"eq"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"ne"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"gt"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"ge"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"lt"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"le")),(0,i.kt)("p",null,"6 Method ini digunakan untuk komparasi dua buah slice. Penggunakan 6 method ini merupakan alternatif selain menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"cmp"),"."),(0,i.kt)("p",null,"Sebagai contoh, penerapan method ",(0,i.kt)("inlineCode",{parentName:"p"},"eq")," berikut untuk untuk mengecek apakah 2 buah data slice adalah sama (secara ",(0,i.kt)("em",{parentName:"p"},"Lexicographical"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec1 = vec![1, 2, 3, 4];\nlet data_vec2 = vec![1, 2, 3, 4];\nlet result = data_vec1.iter().eq(data_vec2.iter());\nprintln!("{:?}", result);\n// true\n')),(0,i.kt)("p",null,"Kode di atas adalah ekuivalen dengan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec1 = vec![1, 2, 3, 4];\nlet data_vec2 = vec![1, 2, 3, 4];\nlet result = data_vec1.iter().cmp(data_vec2.iter());\nprintln!("{:?}", result.is_eq());\n// true\n')),(0,i.kt)("p",null,"Berikut merupakan kegunaan tiap-tiap method di atas:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"eq")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"equal lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"ne")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"not equal lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"gt")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"greater than lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"ge")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"greater than or equal lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"lt")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"lower than lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"le")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"lower than or equal lexicographical comparison"))),(0,i.kt)("h3",{id:"-method-find"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"find")),(0,i.kt)("p",null,"Digunakan untuk mencari apakah suatu elemen ada atau tidak dengan kondisi pencarian dituliskan dalam bentuk closure. Nilai balik method ini bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("p",null,"Sebagai contoh, perhatikan kode berikut. Method ",(0,i.kt)("inlineCode",{parentName:"p"},"find")," digunakan untuk mencari elemen yang nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"4"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet result = match data_vec.iter().find(|d: &&i32| **d == 4) {\n Some(d) => *d,\n None => 0\n};\nprintln!("{:?}", result);\n// 4\n')),(0,i.kt)("h3",{id:"-method-last"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"last")),(0,i.kt)("p",null,"Digunakan untuk mengambil elemen terakhir data Iterator. Method ",(0,i.kt)("inlineCode",{parentName:"p"},"last")," ini mengembalikan nilai balik bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet result: &i32 = data_vec.iter().last().unwrap();\nprintln!("{:?}", result);\n// 4\n')),(0,i.kt)("p",null,"Satu hal yang unik perihal notasi closure method ",(0,i.kt)("inlineCode",{parentName:"p"},"find"),", parameter closure adalah bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&&T"),". Dari tipe tersebut, untuk mengambil ",(0,i.kt)("em",{parentName:"p"},"underlying value")," gunakan operator ",(0,i.kt)("em",{parentName:"p"},"dereference")," dua kali. Contohnya pada kode di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"d")," bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"&&i32"),", untuk mengambil nilai sebenarnya digunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"**d"),"."),(0,i.kt)("h3",{id:"-method-fold"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"fold")),(0,i.kt)("p",null,"Method ini digunakan untuk mengkonversi setiap element ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," menjadi sebuah akumulator yang direpresentasikan oleh 1 buah variabel."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"konsep ",(0,i.kt)("inlineCode",{parentName:"p"},"fold")," milik Rust mirip seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"reduce")," pada bahasa pemrograman lain")),(0,i.kt)("p",null,"Pada contoh berikut, kita coba terapkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"fold")," untuk grouping data vector ",(0,i.kt)("inlineCode",{parentName:"p"},"data_vec")," menjadi 1 buah data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"NumberCategory"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Data numerik genap elemen ",(0,i.kt)("inlineCode",{parentName:"li"},"data_vec")," masuk ke property ",(0,i.kt)("inlineCode",{parentName:"li"},"even")," milik ",(0,i.kt)("inlineCode",{parentName:"li"},"NumberCategory"),"."),(0,i.kt)("li",{parentName:"ul"},"Data numerik ganjil elemen ",(0,i.kt)("inlineCode",{parentName:"li"},"data_vec")," masuk ke property ",(0,i.kt)("inlineCode",{parentName:"li"},"odd")," milik ",(0,i.kt)("inlineCode",{parentName:"li"},"NumberCategory"),".")),(0,i.kt)("p",null,"Dalam penerapannnya, parameter pertama method ",(0,i.kt)("inlineCode",{parentName:"p"},"fold")," diisi dengan ",(0,i.kt)("em",{parentName:"p"},"initial value"),", dan parameter ke-2 isinya closure untuk akumulasi data."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nstruct NumberCategory {\n even: Vec,\n odd: Vec,\n}\n\nlet data_vec = vec![1, 2, 3, 4];\n\nlet data_grouped = data_vec.iter().fold(\n NumberCategory{ even: Vec::new(), odd: Vec::new() }, \n |mut group, each| {\n if *each % 2 == 0 {\n group.even.push(*each)\n } else {\n group.odd.push(*each)\n }\n group\n }\n);\n\nprintln!("{:?}", data_grouped);\n')),(0,i.kt)("p",null,"Pada kode di atas, data ",(0,i.kt)("inlineCode",{parentName:"p"},"each")," di-cek nilainya, jika genap maka dimasukan dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"group.even"),", dan sisanya masuk ",(0,i.kt)("inlineCode",{parentName:"p"},"group.odd"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(2986).Z,width:"454",height:"122"})),(0,i.kt)("h3",{id:"-method-inspect"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"inspect")),(0,i.kt)("p",null,"Biasanya digunakan sewaktu debugging, untuk inspeksi flow pemanggilan method-method milik Iterator. Contoh penerapannya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\nlet result = data_vec.iter()\n .cloned()\n .inspect(|x| println!("about to filter: {x}"))\n .filter(|x| x % 2 == 0)\n .inspect(|x| println!("made it through filter: {x}"))\n .fold(0, |sum, i| sum + i);\n\nprintln!("sum: {:?}", result);\n// sum: 10\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(4695).Z,width:"449",height:"222"})),(0,i.kt)("h3",{id:"-method-sum"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"sum")),(0,i.kt)("p",null,"Digunakan untuk mencari total/summary data slice numerik."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1.1, 2.2, 3.3, 4.5];\nlet result: f64 = data_vec.iter().sum();\nprintln!("sum: {:?}", result);\n// sum: 11.1\n')),(0,i.kt)("h3",{id:"-method-reduce"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"reduce")),(0,i.kt)("p",null,"Method ini digunakan untuk mengiterasi setiap element array dengan setiap iterasi menampilkan data element ke-",(0,i.kt)("inlineCode",{parentName:"p"},"n")," dan element ke-",(0,i.kt)("inlineCode",{parentName:"p"},"n+1"),"."),(0,i.kt)("p",null,"Contoh penerapan sederhananya pada pencarian angka numerik terbesar, yang kode-nya bisa dilihat berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\nlet max_number = data_vec.iter().reduce(|left, right| {\n print!("left ({left}) vs right ({right})");\n\n if *left >= *right { \n println!(" -> left ({left}) is greater");\n left\n } else {\n println!(" -> right ({right}) is greater");\n right\n }\n});\n\nmatch max_number {\n Some(n) => println!("max_number: {:?}", n),\n None => println!("no data found"),\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(1611).Z,width:"449",height:"170"})),(0,i.kt)("h3",{id:"-method-min-max"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"min"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"max")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"min")," digunakan untuk mencari elemen terkecil pada Iterator, dan ",(0,i.kt)("inlineCode",{parentName:"p"},"max")," untuk mencari elemen terbesar. Keduanya mengembalikan nilai balik bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\nlet min = data_vec.iter().min().unwrap();\nprintln!("min: {min}");\n// min: 1\n\nlet max = data_vec.iter().max().unwrap();\nprintln!("max: {max}");\n// max: 4\n')),(0,i.kt)("h2",{id:"a484-method-lainnya"},"A.48.4. Method lainnya"),(0,i.kt)("h3",{id:"-method-sort-milik-vect"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"sort")," milik ",(0,i.kt)("inlineCode",{parentName:"h3"},"Vec")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"sort")," bukanlah property Iterator, melainkan milik tipe data vector. Kegunaan method ini adalah untuk sorting urutan elemen vector."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut data_vec = vec![2, 3, 1, 4];\nprintln!("before: {data_vec:?}");\n// before: [2, 3, 1, 4]\n\ndata_vec.sort();\nprintln!("after: {data_vec:?}");\n// after: [1, 2, 3, 4]\n')),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/trait_iterator"},"github.com/novalagung/dasarpemrogramanrust-example/../trait_iterator")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/vector"},"Vector"))),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Desugar iterator")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/index.html"},"https://doc.rust-lang.org/std/iter/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/trait.Iterator.html"},"https://doc.rust-lang.org/std/iter/trait.Iterator.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://becomebetterprogrammer.com/rust-iter-vs-iter_mut-vs-into_iter/"},"https://becomebetterprogrammer.com/rust-iter-vs-iter_mut-vs-into_iter/")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://blog.thoughtram.io/iterators-in-rust/"},"https://blog.thoughtram.io/iterators-in-rust/")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://blog.logrocket.com/rust-iterators-closures-deep-dive/"},"https://blog.logrocket.com/rust-iterators-closures-deep-dive/"))))}A.isMDXComponent=!0},3753:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},9339:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},6644:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},2986:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},4695:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},1611:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/82a9b669.bb49d726.js b/assets/js/82a9b669.bb49d726.js deleted file mode 100644 index c1af1c12..00000000 --- a/assets/js/82a9b669.bb49d726.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[3257],{3905:(e,a,t)=>{t.d(a,{Zo:()=>m,kt:()=>c});var n=t(7294);function i(e,a,t){return a in e?Object.defineProperty(e,a,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[a]=t,e}function r(e,a){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);a&&(n=n.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),t.push.apply(t,n)}return t}function l(e){for(var a=1;a=0||(i[t]=e[t]);return i}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(i[t]=e[t])}return i}var p=n.createContext({}),o=function(e){var a=n.useContext(p),t=a;return e&&(t="function"==typeof e?e(a):l(l({},a),e)),t},m=function(e){var a=o(e.components);return n.createElement(p.Provider,{value:a},e.children)},A={inlineCode:"code",wrapper:function(e){var a=e.children;return n.createElement(n.Fragment,{},a)}},k=n.forwardRef((function(e,a){var t=e.components,i=e.mdxType,r=e.originalType,p=e.parentName,m=d(e,["components","mdxType","originalType","parentName"]),k=o(t),c=i,u=k["".concat(p,".").concat(c)]||k[c]||A[c]||r;return t?n.createElement(u,l(l({ref:a},m),{},{components:t})):n.createElement(u,l({ref:a},m))}));function c(e,a){var t=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var r=t.length,l=new Array(r);l[0]=k;var d={};for(var p in a)hasOwnProperty.call(a,p)&&(d[p]=a[p]);d.originalType=e,d.mdxType="string"==typeof e?e:i,l[1]=d;for(var o=2;o{t.r(a),t.d(a,{assets:()=>p,contentTitle:()=>l,default:()=>A,frontMatter:()=>r,metadata:()=>d,toc:()=>o});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:48,title:"A.48. Trait \u279c Iterator",sidebar_label:"A.48. Trait \u279c Iterator"},l=void 0,d={unversionedId:"basic/trait-iterator",id:"basic/trait-iterator",title:"A.48. Trait \u279c Iterator",description:"Iterator adalah salah satu trait dan tipe data custom penting pada Rust programming, gunanya untuk iterasi data. Di chapter ini kita akan mempelajarinya beserta beberapa module item lainnya yang masih relevan dengan topik Iterator.",source:"@site/docs/basic/trait-iterator.md",sourceDirName:"basic",slug:"/basic/trait-iterator",permalink:"/basic/trait-iterator",draft:!1,tags:[],version:"current",sidebarPosition:48,frontMatter:{sidebar_position:48,title:"A.48. Trait \u279c Iterator",sidebar_label:"A.48. Trait \u279c Iterator"},sidebar:"tutorialSidebar",previous:{title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",permalink:"/basic/trait-function"},next:{title:"A.49. Attributes",permalink:"/basic/attributes"}},p={},o=[{value:"A.48.1. Iterator & IntoIterator",id:"a481-iterator--intoiterator",level:2},{value:"\u25c9 Trait Iterator & IntoIterator",id:"-trait-iterator--intoiterator",level:3},{value:"\u25c9 Struct Iterator",id:"-struct-iterator",level:3},{value:"A.48.2. Pemanfaatan tipe data Iterator",id:"a482-pemanfaatan-tipe-data-iterator",level:2},{value:"\u25c9 Praktik ke-1",id:"-praktik-ke-1",level:3},{value:"\u25c9 Praktik ke-2",id:"-praktik-ke-2",level:3},{value:"A.48.3. Method tipe data Iterator",id:"a483-method-tipe-data-iterator",level:2},{value:"\u25c9 Method iter, iter_mut, into_iter",id:"-method-iter-iter_mut-into_iter",level:3},{value:"\u25c9 Method collect",id:"-method-collect",level:3},{value:"\u25c9 Method map",id:"-method-map",level:3},{value:"\u25c9 Method rev",id:"-method-rev",level:3},{value:"\u25c9 Method filter",id:"-method-filter",level:3},{value:"\u25c9 Method cloned",id:"-method-cloned",level:3},{value:"\u25c9 Method copied",id:"-method-copied",level:3},{value:"\u25c9 Method cmp",id:"-method-cmp",level:3},{value:"\u25c9 Method count",id:"-method-count",level:3},{value:"\u25c9 Method eq, ne, gt, ge, lt, le",id:"-method-eq-ne-gt-ge-lt-le",level:3},{value:"\u25c9 Method find",id:"-method-find",level:3},{value:"\u25c9 Method last",id:"-method-last",level:3},{value:"\u25c9 Method fold",id:"-method-fold",level:3},{value:"\u25c9 Method inspect",id:"-method-inspect",level:3},{value:"\u25c9 Method sum",id:"-method-sum",level:3},{value:"\u25c9 Method reduce",id:"-method-reduce",level:3},{value:"\u25c9 Method min, max",id:"-method-min-max",level:3},{value:"A.48.4. Method lainnya",id:"a484-method-lainnya",level:2},{value:"\u25c9 Method sort milik Vec<T>",id:"-method-sort-milik-vect",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:o};function A(e){let{components:a,...r}=e;return(0,i.kt)("wrapper",(0,n.Z)({},m,r,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Iterator adalah salah satu trait dan tipe data custom penting pada Rust programming, gunanya untuk iterasi data. Di chapter ini kita akan mempelajarinya beserta beberapa module item lainnya yang masih relevan dengan topik Iterator."),(0,i.kt)("h2",{id:"a481-iterator--intoiterator"},"A.48.1. ",(0,i.kt)("inlineCode",{parentName:"h2"},"Iterator")," & ",(0,i.kt)("inlineCode",{parentName:"h2"},"IntoIterator")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," adalah sebuah nama yang dipakai sebagai nama module item dalam ",(0,i.kt)("strong",{parentName:"p"},"Rust Standard Library")," maupun ",(0,i.kt)("strong",{parentName:"p"},"Rust Core Library"),", digunakan untuk iterasi data dan operasi lain yang berhubungan dengannya."),(0,i.kt)("p",null,"Iterator sendiri merupakan istilah untuk object atau trait yang bisa diiterasi, baik menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," ataupun menggunakan method iterator seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each")," (yang juga akan kita bahas disini)."),(0,i.kt)("h3",{id:"-trait-iterator--intoiterator"},"\u25c9 Trait ",(0,i.kt)("inlineCode",{parentName:"h3"},"Iterator")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"IntoIterator")),(0,i.kt)("p",null,"Dalam perulangan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", tipe data variabel yang digunakan harus memiliki trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),". Jika tidak, pasti error muncul."),(0,i.kt)("p",null,"Tipe data slice di Rust by default mengadopsi trait bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"IntoIterator"),", yang trait ini digunakan untuk konversi data bertipe slice ke bentuk iterator."),(0,i.kt)("p",null,"Dalam praktiknya, tidak perlu mengakses method tertentu untuk mendapatkan object iterator suatu data. Cukup gunakan variabel tipe data slice pada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", maka proses konversi ke bentuk iterator akan dilakukan oleh Rust secara otomatis dibalik layar."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// array\nlet data_arr = ["a", "b", "c", "d"];\nfor e in &data_arr {\n print!("{e:?} ")\n}\n\nprintln!();\n\n// vector\nlet data_vec = vec!["a", "b", "c", "d"];\nfor e in &data_vec {\n print!("{e:?} ")\n}\n\nprintln!();\n\n// slice\nlet data_vec = vec!["a", "b", "c", "d"];\nlet data_slice_vec = &data_vec[..];\nfor e in data_slice_vec {\n print!("{e:?} ")\n}\n')),(0,i.kt)("p",null,"Dianjurkan untuk selalu menggunakan teknik ",(0,i.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing")," dalam penggunaan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," pada tipe data yang karakteristiknya adalah ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership#a344-copy-semantics-vs-move-semantics"},"Move Semantics"),", hal ini karena dalam penerapan keyword tersebut, terjadi proses konversi tipe data dengan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"IntoIterator")," ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),"."),(0,i.kt)("p",null,"Jika data yang digunakan bukan data borrow, maka pasti owner berpindah."),(0,i.kt)("h3",{id:"-struct-iterator"},"\u25c9 Struct ",(0,i.kt)("inlineCode",{parentName:"h3"},"Iterator")),(0,i.kt)("p",null,"Selain trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),", ada juga tipe data struct bernama sama, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),". Perbedaan antara trait vs struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," adalah tipe struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," memiliki beberapa method untuk keperluan iterasi object. Jadi dengan memanfaatkan method-method tersebut, kita bisa melakukan iterasi data dan operasi sejenisnya tanpa menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),"."),(0,i.kt)("p",null,"Semua tipe data slice bisa dikonversi ke tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),", caranya dengan mengakses method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," (atau method ",(0,i.kt)("inlineCode",{parentName:"p"},"chars")," khusus untuk tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),")."),(0,i.kt)("p",null,"Dari object iterator tersebut, perulangan bisa dilakukan via keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", atau dengan memanfaatkan method bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each"),". Perbedaannya pada iterasi menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each"),", block perulangan dituliskan dalam bentuk closure."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// array\nlet data_arr = ["a", "b", "c", "d"];\nlet iterator_arr = data_arr.iter();\niterator_arr.for_each(|e| {\n print!("{e:?} ")\n});\n\nprintln!();\n\n// vector\nlet data_vec = vec!["a", "b", "c", "d"];\ndata_vec.iter().for_each(|e| {\n print!("{e:?} ")\n});\n\nprintln!();\n\n// slice from vector\nlet data_vec = vec!["a", "b", "c", "d"];\nlet data_slice_vec = &data_vec[..];\ndata_slice_vec.iter().for_each(|e| {\n print!("{e:?} ")\n});\n\nprintln!();\n\n// slice from String\nlet data_str = "abcd".to_string();\nlet data_borrow_str = &data_str;\ndata_borrow_str.chars().for_each(|e| {\n print!("{e:?} ")\n});\n\nprintln!();\n')),(0,i.kt)("p",null,"Bisa dilihat pada contoh di atas, data slice diambil objek Iterator-nya menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," (atau ",(0,i.kt)("inlineCode",{parentName:"p"},"chars")," khusus untuk tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"), kemudian di-iterasi menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(3753).Z,width:"456",height:"149"})),(0,i.kt)("h2",{id:"a482-pemanfaatan-tipe-data-iterator"},"A.48.2. Pemanfaatan tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Iterator")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," memiliki cukup banyak method untuk keperluan operasi data iterator, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"for_each"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"map"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"filter"),", dan lainnya."),(0,i.kt)("h3",{id:"-praktik-ke-1"},"\u25c9 Praktik ke-1"),(0,i.kt)("p",null,"Sebagai contoh, kita akan buat sebuah program sederhana yang melakukan operasi kuadrat pada elemen tiap-tiap array."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_arr = [1, 3, 5];\n\nlet doubles: Vec = data_arr\n .iter()\n .map(|e| e * e)\n .collect();\n\nprintln!("{doubles:?}");\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(9339).Z,width:"447",height:"121"})),(0,i.kt)("p",null,"Penjelasan:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"data_arr")," yang bertipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"[i32; 3]")," dikonversi ke tipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"Iterator")," menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"iter"),"."),(0,i.kt)("li",{parentName:"ol"},"Object ",(0,i.kt)("inlineCode",{parentName:"li"},"Iterator")," kemudian di-iterasi menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"map")," dan nilai baliknya dijadikan ",(0,i.kt)("em",{parentName:"li"},"replacement")," data elemen tersebut."),(0,i.kt)("li",{parentName:"ol"},"Object ",(0,i.kt)("inlineCode",{parentName:"li"},"Iterator")," kemudian di-collect data-nya ke bentuk ",(0,i.kt)("inlineCode",{parentName:"li"},"Vec")," menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"collect"),".")),(0,i.kt)("h3",{id:"-praktik-ke-2"},"\u25c9 Praktik ke-2"),(0,i.kt)("p",null,"Contoh lain penerapan method iterator bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec!["1", "2", "3", "4", "a"];\n\nlet numbers: Vec = data_vec\n .iter()\n .map(|e| -> i32 {\n match e.parse::() {\n Ok(n) => n,\n Err(_) => 0,\n }\n })\n .filter(|e| *e > 0 && *e % 2 == 0)\n .rev()\n .collect::>();\n\nprintln!("{numbers:?}");\n')),(0,i.kt)("p",null,"Program di atas melakukan beberapa hal:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Data ",(0,i.kt)("inlineCode",{parentName:"p"},"data_vec")," yang merupakan koleksi string di-konversi menjadi object ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter"),".")),(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Kemudian method ",(0,i.kt)("inlineCode",{parentName:"p"},"map")," diakses. Setiap elemen ",(0,i.kt)("inlineCode",{parentName:"p"},"data_vec")," di-iterasi, kemudian dikonversi dari ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", lalu dijadikan ",(0,i.kt)("em",{parentName:"p"},"replacement")," data elemen tersebut. Jika proses konversi gagal, maka angka ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," digunakan sebagai data element tersebut."),(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},"Sampai sini, data yang sebelumnya ",(0,i.kt)("inlineCode",{parentName:"li"},'["1", "2", "3", "4", "a"]')," sekarang menjadi ",(0,i.kt)("inlineCode",{parentName:"li"},"[1, 2, 3, 4, 0]"),"."))),(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Selanjutnya, dilakukan proses filtering menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"filter")," dengan kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"*e > 0 && *e % 2 == 0")," yang kurang lebih artinya, jika nilai ",(0,i.kt)("em",{parentName:"p"},"dereference")," ",(0,i.kt)("inlineCode",{parentName:"p"},"e")," lebih besar dari ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," dan nilai tersebut adalah genap, maka filter bernilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),"."),(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},"Sampai sini, data yang sebelumnya ",(0,i.kt)("inlineCode",{parentName:"li"},"[1, 2, 3, 4, 0]")," sekarang menjadi ",(0,i.kt)("inlineCode",{parentName:"li"},"[2, 4]"),"."))),(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Kemudian, data kolektif tersebut dibalik urutan elemennya menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"rev"),"."),(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},"Sampai sini, data yang sebelumnya ",(0,i.kt)("inlineCode",{parentName:"li"},"[2, 4]")," sekarang menjadi ",(0,i.kt)("inlineCode",{parentName:"li"},"[4, 2]"),"."))),(0,i.kt)("li",{parentName:"ol"},(0,i.kt)("p",{parentName:"li"},"Terakhir data di-collect menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"collect")," dengan ditentukan tipe data masing-masing elemen adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),"."))),(0,i.kt)("p",null,"Hasilnya ketika di run:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(6644).Z,width:"447",height:"118"})),(0,i.kt)("h2",{id:"a483-method-tipe-data-iterator"},"A.48.3. Method tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Iterator")),(0,i.kt)("h3",{id:"-method-iter-iter_mut-into_iter"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"iter"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"iter_mut"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"into_iter")),(0,i.kt)("p",null,"Ketiga method ini berguna untuk konversi data slice ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),", perbedannya:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," mengembalikan data ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," yang isinya adalah reference (",(0,i.kt)("inlineCode",{parentName:"p"},"&T"),") setiap element. Contoh penerapan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\ndata_vec.iter().for_each(|d| {\n println!("{}", *d * 2)\n});\n\nfor d in &data_vec {\n println!("{}", *d * 2)\n}\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter_mut")," mengembalikan data ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," yang isinya adalah mutable reference (",(0,i.kt)("inlineCode",{parentName:"p"},"&mut T"),") setiap element. Contoh penerapan:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut data_vec = vec![1, 2, 3, 4];\n\ndata_vec.iter_mut().for_each(|d| {\n *d = *d * 2;\n});\nprintln!("{:?}", data_vec);\n\nfor d in &mut data_vec {\n *d = *d * 2;\n}\nprintln!("{:?}", data_vec);\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"into_iter")," mengkonversi data slice ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," (",(0,i.kt)("em",{parentName:"p"},"move semantics"),")."),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\ndata_vec.into_iter().for_each(|d| {\n println!("{}", d * 2)\n});\n\n// statement di bawah ini menghasilkan error,\n// karena ownership `data_vec` telah berpindah setelah method `into_iter` dipanggil\n\n// println!("{}", data_vec)\n')))),(0,i.kt)("h3",{id:"-method-collect"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"collect")),(0,i.kt)("p",null,"Method ini digunakan untuk ",(0,i.kt)("em",{parentName:"p"},"collecting")," data Iterator ke bentuk tertentu. Contoh penerapannya bisa dilihat pada kode berikut ini."),(0,i.kt)("p",null,"Data ",(0,i.kt)("inlineCode",{parentName:"p"},"data_vec")," yang merupakan vektor bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," dikonversi ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," kemudian di-collect datanya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec: Vec = vec![1, 2, 3, 4];\n\nlet result: Vec<&i32> = data_vec.iter().collect();\nprintln!("{:?}", result);\n// [1, 2, 3, 4]\n')),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," menghasilkan object ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," yang menampung reference ",(0,i.kt)("inlineCode",{parentName:"p"},"&T")," setiap element slice. Hal ini membuat penerapan method ",(0,i.kt)("inlineCode",{parentName:"p"},"collect")," menghasilkan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec<&i32>")," (bukan ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),")."),(0,i.kt)("h3",{id:"-method-map"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"map")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"map")," digunakan untuk mapping element setiap data iterator ke nilai baru."),(0,i.kt)("p",null,"Sebelumnya kita telah mempelajari method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"collect")," yang keduanya jika dikombinasikan dan digunakan pada tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," hasilnya adalah data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec<&i32>"),". Dengan memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"map"),", tipe data setiap element bisa di-mapping ke nilai baru dengan tipe data berbeda, misalnya menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec: Vec = vec![1, 2, 3, 4];\n\nlet result1: Vec<&i32> = data_vec.iter().collect();\nprintln!("{:?}", result1);\n// [1, 2, 3, 4]\n\nlet result2: Vec = data_vec\n .iter()\n .map(|d: &i32| -> i32 { *d })\n .collect();\nprintln!("{:?}", result2);\n// [1, 2, 3, 4]\n')),(0,i.kt)("p",null,"Contoh lainnya bisa dilihat pada ",(0,i.kt)("em",{parentName:"p"},"section")," ",(0,i.kt)("a",{parentName:"p",href:"/basic/trait-iterator#a482-pemanfaatan-tipe-data-iterator"},"A.48.2. Pemanfaatan tipe data ",(0,i.kt)("inlineCode",{parentName:"a"},"Iterator"))," dimana dilakukan mapping data slice numerik ke bentuk yang sama tapi nilai setiap element adalah kuadrat, dan juga ke bentuk lain dengan tipe data berbeda."),(0,i.kt)("h3",{id:"-method-rev"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"rev")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"rev")," kependekan dari ",(0,i.kt)("em",{parentName:"p"},"reverse"),", digunakan untuk membalikan urutan data slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nprintln!("{:?}", data_vec);\n// [1, 2, 3, 4]\n\nlet result: Vec<&i32> = data_vec.iter().rev().collect();\nprintln!("{:?}", result);\n// [4, 3, 2, 1]\n')),(0,i.kt)("h3",{id:"-method-filter"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"filter")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"filter")," digunakan untuk memfilter element data slice. Data kolektif diiterasi kemudian dicek menggunakan closure, jika nilai balik bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," maka elemen tersebut masuk dalam dalam hasil filter, selebihnya maka dianggap tidak memenuhi kondisi filter dan elemen di-",(0,i.kt)("em",{parentName:"p"},"exclude"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\nlet odd: Vec<&i32> = data_vec.iter().filter(|d| *d % 2 != 0).collect();\nprintln!("odd numbers: {:?}", odd);\n// odd numbers: [1, 3]\n\nlet even: Vec<&i32> = data_vec.iter().filter(|d| *d % 2 == 0).collect();\nprintln!("even numbers: {:?}", even);\n// odd numbers: [2, 4]\n')),(0,i.kt)("h3",{id:"-method-cloned"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"cloned")),(0,i.kt)("p",null,"Digunakan untuk ",(0,i.kt)("em",{parentName:"p"},"cloning")," data slice secara keseluruhan tanpa mengubah tipe data. Hasilnya adalah data dengan skema sama persis (dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator"),") tapi berbeda owner."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet result: Vec = data_vec.iter().cloned().collect();\nprintln!("{:?}", result);\n// [1, 2, 3, 4]\n')),(0,i.kt)("h3",{id:"-method-copied"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"copied")),(0,i.kt)("p",null,"Secara ",(0,i.kt)("em",{parentName:"p"},"high-level")," ",(0,i.kt)("inlineCode",{parentName:"p"},"copied")," menghasilkan output yang sama dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"cloned"),". Namun jika dibahas dari sisi management memory-nya, ada perbedaan yang cukup besar."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet result: Vec = data_vec.iter().copied().collect();\nprintln!("{:?}", result);\n// [1, 2, 3, 4]\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih detailnya mengenai clone vs copy akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/copy-clone-move-drop"},"Copy, Clone, Move, Drop"))),(0,i.kt)("h3",{id:"-method-cmp"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"cmp")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"cmp")," kependekan dari ",(0,i.kt)("em",{parentName:"p"},"compare"),", digunakan untuk membandingkan 2 buah data iterasi dengan nilai balik bertipe enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Ordering"),". Dari tipe data enum tersebut nantinya bisa dicek apakah 2 buah data slice tersebut sama (secara ",(0,i.kt)("em",{parentName:"p"},"Lexicographical"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec1 = vec![1, 2, 3, 4];\nlet data_vec2 = vec![1, 2, 3, 4];\nlet result = data_vec1.iter().cmp(data_vec2.iter());\nprintln!("{:?}", result.is_eq());\n// true\n')),(0,i.kt)("p",null,"Enum ",(0,i.kt)("inlineCode",{parentName:"p"},"Ordering")," memiliki beberapa method, salah satunya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"is_eq")," yang mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," jika dua buah data slice tersebut adalah sama."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih detailnya mengenai ",(0,i.kt)("inlineCode",{parentName:"p"},"Ordering")," akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/enum-ordering"},"Enum Ordering"))),(0,i.kt)("h3",{id:"-method-count"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"count")),(0,i.kt)("p",null,"Digunakan untuk melihat size dari elemen Iterator."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet length = data_vec.iter().count();\nprintln!("{:?}", length);\n// 4\n')),(0,i.kt)("h3",{id:"-method-eq-ne-gt-ge-lt-le"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"eq"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"ne"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"gt"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"ge"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"lt"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"le")),(0,i.kt)("p",null,"6 Method ini digunakan untuk komparasi dua buah slice. Penggunakan 6 method ini merupakan alternatif selain menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"cmp"),"."),(0,i.kt)("p",null,"Sebagai contoh, penerapan method ",(0,i.kt)("inlineCode",{parentName:"p"},"eq")," berikut untuk untuk mengecek apakah 2 buah data slice adalah sama (secara ",(0,i.kt)("em",{parentName:"p"},"Lexicographical"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec1 = vec![1, 2, 3, 4];\nlet data_vec2 = vec![1, 2, 3, 4];\nlet result = data_vec1.iter().eq(data_vec2.iter());\nprintln!("{:?}", result);\n// true\n')),(0,i.kt)("p",null,"Kode di atas adalah ekuivalen dengan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec1 = vec![1, 2, 3, 4];\nlet data_vec2 = vec![1, 2, 3, 4];\nlet result = data_vec1.iter().cmp(data_vec2.iter());\nprintln!("{:?}", result.is_eq());\n// true\n')),(0,i.kt)("p",null,"Berikut merupakan kegunaan tiap-tiap method di atas:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"eq")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"equal lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"ne")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"not equal lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"gt")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"greater than lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"ge")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"greater than or equal lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"lt")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"lower than lexicographical comparison")),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"le")," digunakan untuk ",(0,i.kt)("em",{parentName:"li"},"lower than or equal lexicographical comparison"))),(0,i.kt)("h3",{id:"-method-find"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"find")),(0,i.kt)("p",null,"Digunakan untuk mencari apakah suatu elemen ada atau tidak dengan kondisi pencarian dituliskan dalam bentuk closure. Nilai balik method ini bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("p",null,"Sebagai contoh, perhatikan kode berikut. Method ",(0,i.kt)("inlineCode",{parentName:"p"},"find")," digunakan untuk mencari elemen yang nilainya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"4"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet result = match data_vec.iter().find(|d: &&i32| **d == 4) {\n Some(d) => *d,\n None => 0\n};\nprintln!("{:?}", result);\n// 4\n')),(0,i.kt)("h3",{id:"-method-last"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"last")),(0,i.kt)("p",null,"Digunakan untuk mengambil elemen terakhir data Iterator. Method ",(0,i.kt)("inlineCode",{parentName:"p"},"last")," ini mengembalikan nilai balik bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\nlet result: &i32 = data_vec.iter().last().unwrap();\nprintln!("{:?}", result);\n// 4\n')),(0,i.kt)("p",null,"Satu hal yang unik perihal notasi closure method ",(0,i.kt)("inlineCode",{parentName:"p"},"find"),", parameter closure adalah bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&&T"),". Dari tipe tersebut, untuk mengambil ",(0,i.kt)("em",{parentName:"p"},"underlying value")," gunakan operator ",(0,i.kt)("em",{parentName:"p"},"dereference")," dua kali. Contohnya pada kode di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"d")," bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"&&i32"),", untuk mengambil nilai sebenarnya digunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"**d"),"."),(0,i.kt)("h3",{id:"-method-fold"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"fold")),(0,i.kt)("p",null,"Method ini digunakan untuk mengkonversi setiap element ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," menjadi sebuah akumulator yang direpresentasikan oleh 1 buah variabel."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"konsep ",(0,i.kt)("inlineCode",{parentName:"p"},"fold")," milik Rust mirip seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"reduce")," pada bahasa pemrograman lain")),(0,i.kt)("p",null,"Pada contoh berikut, kita coba terapkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"fold")," untuk grouping data vector ",(0,i.kt)("inlineCode",{parentName:"p"},"data_vec")," menjadi 1 buah data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"NumberCategory"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Data numerik genap elemen ",(0,i.kt)("inlineCode",{parentName:"li"},"data_vec")," masuk ke property ",(0,i.kt)("inlineCode",{parentName:"li"},"even")," milik ",(0,i.kt)("inlineCode",{parentName:"li"},"NumberCategory"),"."),(0,i.kt)("li",{parentName:"ul"},"Data numerik ganjil elemen ",(0,i.kt)("inlineCode",{parentName:"li"},"data_vec")," masuk ke property ",(0,i.kt)("inlineCode",{parentName:"li"},"odd")," milik ",(0,i.kt)("inlineCode",{parentName:"li"},"NumberCategory"),".")),(0,i.kt)("p",null,"Dalam penerapannnya, parameter pertama method ",(0,i.kt)("inlineCode",{parentName:"p"},"fold")," diisi dengan ",(0,i.kt)("em",{parentName:"p"},"initial value"),", dan parameter ke-2 isinya closure untuk akumulasi data."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nstruct NumberCategory {\n even: Vec,\n odd: Vec,\n}\n\nlet data_vec = vec![1, 2, 3, 4];\n\nlet data_grouped = data_vec.iter().fold(\n NumberCategory{ even: Vec::new(), odd: Vec::new() }, \n |mut group, each| {\n if *each % 2 == 0 {\n group.even.push(*each)\n } else {\n group.odd.push(*each)\n }\n group\n }\n);\n\nprintln!("{:?}", data_grouped);\n')),(0,i.kt)("p",null,"Pada kode di atas, data ",(0,i.kt)("inlineCode",{parentName:"p"},"each")," di-cek nilainya, jika genap maka dimasukan dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"group.even"),", dan sisanya masuk ",(0,i.kt)("inlineCode",{parentName:"p"},"group.odd"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(2986).Z,width:"454",height:"122"})),(0,i.kt)("h3",{id:"-method-inspect"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"inspect")),(0,i.kt)("p",null,"Biasanya digunakan sewaktu debugging, untuk inspeksi flow pemanggilan method-method milik Iterator. Contoh penerapannya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\nlet result = data_vec.iter()\n .cloned()\n .inspect(|x| println!("about to filter: {x}"))\n .filter(|x| x % 2 == 0)\n .inspect(|x| println!("made it through filter: {x}"))\n .fold(0, |sum, i| sum + i);\n\nprintln!("sum: {:?}", result);\n// sum: 10\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(4695).Z,width:"449",height:"222"})),(0,i.kt)("h3",{id:"-method-sum"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"sum")),(0,i.kt)("p",null,"Digunakan untuk mencari total/summary data slice numerik."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1.1, 2.2, 3.3, 4.5];\nlet result: f64 = data_vec.iter().sum();\nprintln!("sum: {:?}", result);\n// sum: 11.1\n')),(0,i.kt)("h3",{id:"-method-reduce"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"reduce")),(0,i.kt)("p",null,"Method ini digunakan untuk mengiterasi setiap element array dengan setiap iterasi menampilkan data element ke-",(0,i.kt)("inlineCode",{parentName:"p"},"n")," dan element ke-",(0,i.kt)("inlineCode",{parentName:"p"},"n+1"),"."),(0,i.kt)("p",null,"Contoh penerapan sederhananya pada pencarian angka numerik terbesar, yang kode-nya bisa dilihat berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\nlet max_number = data_vec.iter().reduce(|left, right| {\n print!("left ({left}) vs right ({right})");\n\n if *left >= *right { \n println!(" -> left ({left}) is greater");\n left\n } else {\n println!(" -> right ({right}) is greater");\n right\n }\n});\n\nmatch max_number {\n Some(n) => println!("max_number: {:?}", n),\n None => println!("no data found"),\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait iterator",src:t(1611).Z,width:"449",height:"170"})),(0,i.kt)("h3",{id:"-method-min-max"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"min"),", ",(0,i.kt)("inlineCode",{parentName:"h3"},"max")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"min")," digunakan untuk mencari elemen terkecil pada Iterator, dan ",(0,i.kt)("inlineCode",{parentName:"p"},"max")," untuk mencari elemen terbesar. Keduanya mengembalikan nilai balik bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_vec = vec![1, 2, 3, 4];\n\nlet min = data_vec.iter().min().unwrap();\nprintln!("min: {min}");\n// min: 1\n\nlet max = data_vec.iter().max().unwrap();\nprintln!("max: {max}");\n// max: 4\n')),(0,i.kt)("h2",{id:"a484-method-lainnya"},"A.48.4. Method lainnya"),(0,i.kt)("h3",{id:"-method-sort-milik-vect"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"sort")," milik ",(0,i.kt)("inlineCode",{parentName:"h3"},"Vec")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"sort")," bukanlah property Iterator, melainkan milik tipe data vector. Kegunaan method ini adalah untuk sorting urutan elemen vector."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut data_vec = vec![2, 3, 1, 4];\nprintln!("before: {data_vec:?}");\n// before: [2, 3, 1, 4]\n\ndata_vec.sort();\nprintln!("after: {data_vec:?}");\n// after: [1, 2, 3, 4]\n')),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/trait_iterator"},"github.com/novalagung/dasarpemrogramanrust-example/../trait_iterator")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/vector"},"Vector"))),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Desugar iterator")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/index.html"},"https://doc.rust-lang.org/std/iter/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/trait.Iterator.html"},"https://doc.rust-lang.org/std/iter/trait.Iterator.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://becomebetterprogrammer.com/rust-iter-vs-iter_mut-vs-into_iter/"},"https://becomebetterprogrammer.com/rust-iter-vs-iter_mut-vs-into_iter/")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://blog.thoughtram.io/iterators-in-rust/"},"https://blog.thoughtram.io/iterators-in-rust/")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://blog.logrocket.com/rust-iterators-closures-deep-dive/"},"https://blog.logrocket.com/rust-iterators-closures-deep-dive/"))))}A.isMDXComponent=!0},3753:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},9339:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},6644:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},2986:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},4695:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},1611:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/8333c691.54569977.js b/assets/js/8333c691.d59a9952.js similarity index 61% rename from assets/js/8333c691.54569977.js rename to assets/js/8333c691.d59a9952.js index 6020153b..b608163d 100644 --- a/assets/js/8333c691.54569977.js +++ b/assets/js/8333c691.d59a9952.js @@ -1 +1 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4248],{3905:(a,e,n)=>{n.d(e,{Zo:()=>d,kt:()=>u});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function l(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var s=i.createContext({}),m=function(a){var e=i.useContext(s),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},d=function(a){var e=m(a.components);return i.createElement(s.Provider,{value:e},a.children)},o={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},k=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,r=a.originalType,s=a.parentName,d=p(a,["components","mdxType","originalType","parentName"]),k=m(n),u=t,g=k["".concat(s,".").concat(u)]||k[u]||o[u]||r;return n?i.createElement(g,l(l({ref:e},d),{},{components:n})):i.createElement(g,l({ref:e},d))}));function u(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var r=n.length,l=new Array(r);l[0]=k;var p={};for(var s in e)hasOwnProperty.call(e,s)&&(p[s]=e[s]);p.originalType=a,p.mdxType="string"==typeof a?a:t,l[1]=p;for(var m=2;m{n.r(e),n.d(e,{assets:()=>s,contentTitle:()=>l,default:()=>o,frontMatter:()=>r,metadata:()=>p,toc:()=>m});var i=n(7462),t=(n(7294),n(3905));const r={sidebar_position:34,title:"A.34. Ownership",sidebar_label:"A.34. Ownership"},l=void 0,p={unversionedId:"basic/ownership",id:"basic/ownership",title:"A.34. Ownership",description:"Chapter ini berisi pembahasan tentang ownership, bagaimana Rust melakukan manajemen memory dengan menerapkan approach ownership ini.",source:"@site/docs/basic/ownership.md",sourceDirName:"basic",slug:"/basic/ownership",permalink:"/basic/ownership",draft:!1,tags:[],version:"current",sidebarPosition:34,frontMatter:{sidebar_position:34,title:"A.34. Ownership",sidebar_label:"A.34. Ownership"},sidebar:"tutorialSidebar",previous:{title:"A.33. Pointer & References",permalink:"/basic/pointer-references"},next:{title:"A.35. Borrowing",permalink:"/basic/borrowing"}},s={},m=[{value:"A.34.1. Konsep ownership",id:"a341-konsep-ownership",level:2},{value:"A.34.2. Aturan ownership",id:"a342-aturan-ownership",level:2},{value:"A.34.3. Variable scope",id:"a343-variable-scope",level:2},{value:"A.34.4. Copy semantics vs. move semantics",id:"a344-copy-semantics-vs-move-semantics",level:2},{value:"A.34.5. Alokasi & dealokasi",id:"a345-alokasi--dealokasi",level:2},{value:"A.34.6. Transfer ownership",id:"a346-transfer-ownership",level:2},{value:"\u25c9 Transfer ownership via return value",id:"-transfer-ownership-via-return-value",level:3},{value:"\u25c9 Transfer ownership via parameter/argument",id:"-transfer-ownership-via-parameterargument",level:3},{value:"A.34.7. Clone data",id:"a347-clone-data",level:2},{value:"A.34.8. Ownership pada data literal",id:"a348-ownership-pada-data-literal",level:2},{value:"A.34.9. Move semantics pada macro println",id:"a349-move-semantics-pada-macro-println",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function o(a){let{components:e,...r}=a;return(0,t.kt)("wrapper",(0,i.Z)({},d,r,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Chapter ini berisi pembahasan tentang ownership, bagaimana Rust melakukan manajemen memory dengan menerapkan approach ownership ini."),(0,t.kt)("p",null,"O iya, diwajibkan untuk mempelajari terlebih dahulu tentang dasar ",(0,t.kt)("a",{parentName:"p",href:"/basic/basic-memory-management"},"Memory Management")," dan juga ",(0,t.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),", yang keduanya adalah dibahas pada chapter sebelumnya. Jika pembaca mempelajari ebook ini secara urut maka tidak usah khawatir."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Pembahasan pada chapter ini adalah salah satu hal yang paling penting untuk dipahami dalam Rust programming, karena topik yang berhubungan dengan memory management adalah hal yang krusial pada system programming."),(0,t.kt)("p",{parentName:"blockquote"},"Silakan ulang-ulang chapter ini jika diperlukan.")),(0,t.kt)("h2",{id:"a341-konsep-ownership"},"A.34.1. Konsep ownership"),(0,t.kt)("p",null,"Ownership merupakan kumpulan aturan yang ada di Rust yang dijadikan acuan oleh compiler dalam pengelolahan memory."),(0,t.kt)("p",null,"Sudah disinggung pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/basic-memory-management"},"Memory Management")," bahwa Rust tidak menerapkan GC ataupun ARC dalam manajemen memory-nya. Rust membebankan manajemen memory pada penulis kode program, yaitu kita/programmer."),(0,t.kt)("p",null,"Aturan ownership ada banyak, dan programmer harus mengikutinya, karena jika tidak, maka proses kompilasi program dan eksekusi program akan gagal dan hasilnya error."),(0,t.kt)("h2",{id:"a342-aturan-ownership"},"A.34.2. Aturan ownership"),(0,t.kt)("p",null,"Ada 3 aturan penting yang wajib diketahui:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},(0,t.kt)("strong",{parentName:"p"},"Semua nilai/data/value di Rust memiliki owner"),". Misal kita berbicara tentang deklarasi variabel dengan predefined value, maka value variabel adalah yang dimaksud dengan nilai/data/value, dan variabel itu sendiri adalah owner dari nilai/data/value tersebut.")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Pada waktu yang sama, ",(0,t.kt)("strong",{parentName:"p"},"hanya boleh ada 1 owner"),". Satu data, ownernya hanya satu.")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},(0,t.kt)("strong",{parentName:"p"},"Ketika eksekusi sebuah block scope selesai, maka owner dari data-data yang ada dalam scope tersebut akan di-drop atau di-dealokasi")," (dengan pengecualian yaitu owner berpindah ke luar scope)."))),(0,t.kt)("h2",{id:"a343-variable-scope"},"A.34.3. Variable scope"),(0,t.kt)("p",null,"Sebelum kita masuk ke pembahasan yang lebih detail mengenai ownership, mari pelajari terlebih dahulu tentang apa itu variable scope."),(0,t.kt)("p",null,"Variable scope maksudnya adalah di block scope mana suatu variabel dideklarasikan, dan dalam block scope tersebut variabel menjadi valid (bisa digunakan). Di luar scope-nya variabel menjadi tidak valid, tidak bisa digunakan."),(0,t.kt)("p",null,"Agar lebih jelas, silakan perhatikan kode berikut, tidak perlu dipraktikkan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn do_something() {\n let data_one = "one";\n // ...\n}\n\nfn main() {\n let data_two = "two";\n // ...\n\n {\n let data_three = "three";\n // ...\n }\n\n do_something();\n\n if true {\n let data_four = "four";\n // ...\n }\n}\n')),(0,t.kt)("p",null,"Program di atas memiliki 4 buah block kode:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", yang isinya adalah variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_two")," dan beberapa sub-block dan 1 buah pemanggilan fungsi. Di dalam block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_two")," adalah valid, dan bisa digunakan dalam fungsi scope tersebut beserta sub-block lainnya (block expression dan block kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),").")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Block expression yang berisi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_three"),". Pada block scope ini, variabel tersebut dan juga variable ",(0,t.kt)("inlineCode",{parentName:"p"},"data_two")," adalah valid.")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),", yang isinya variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_one"),". Variabel tersebut hanya akan valid dalam block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),".")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Block seleksi kondisi ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_four")," berada dalam block ini, maka variabel tersebut adalah valid dalam block tersebut. Selain itu ",(0,t.kt)("inlineCode",{parentName:"p"},"data_two")," juga valid dalam block seleksi kondisi ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," ini."))),(0,t.kt)("p",null,"Intinya, variabel adalah valid ketika berada di dalam scope-nya (istilahnya ",(0,t.kt)("em",{parentName:"p"},"into scope"),"), dan variabel menjadi tidak valid atau invalid ketika keluar dari scope (istilahnya ",(0,t.kt)("em",{parentName:"p"},"out of scope"),")."),(0,t.kt)("h2",{id:"a344-copy-semantics-vs-move-semantics"},"A.34.4. Copy semantics vs. move semantics"),(0,t.kt)("p",null,"Di atas sudah dijelaskan mengenai aturan ownership, yang salah satunya adalah setiap data yang ada di Rust memiliki owner, dan 1 data owner pasti 1 (tidak lebih)."),(0,t.kt)("p",null,"Agar lebih jelas mari kita perhatikan contoh berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let x = 24;\nlet y = x;\nprintln!("x: {:?}, y: {:?}", x, y);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(1146).Z,width:"333",height:"110"})),(0,t.kt)("p",null,"Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," dideklarasikan dengan nilai ",(0,t.kt)("inlineCode",{parentName:"p"},"24"),", artinya variabel tersebut adalah owner dari data ",(0,t.kt)("inlineCode",{parentName:"p"},"24"),". Kemudian variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," dijadikan sebagai nilai variabel baru bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),", dari sini apakah berarti owner data ",(0,t.kt)("inlineCode",{parentName:"p"},"24")," adalah berpindah dari variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),"? jawabannya adalah ",(0,t.kt)("strong",{parentName:"p"},"tidak"),"."),(0,t.kt)("p",null,"Yang terjadi adalah data ",(0,t.kt)("inlineCode",{parentName:"p"},"24")," milik owner ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," di-copy atau diduplikasi sebagai data baru yang owner-nya juga baru, yang pada contoh ini adalah variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),"."),(0,t.kt)("p",null,"Hasilnya, kedua variabel tersebut masing-masing adalah owner dari data yang berbeda, meskipun sumbernya adalah dari data yang sama. Perilaku ini disebut dengan ",(0,t.kt)("strong",{parentName:"p"},"copy semantics"),"."),(0,t.kt)("p",null,"Semua tipe data primitif di Rust ",(0,t.kt)("em",{parentName:"p"},"by default")," mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),". Ketika terjadi operasi assignment seperti pada contoh di atas, maka yang terjadi adalah data di-copy sebagai data baru, dengan owner baru, dan di sisi memory juga terjadi alokasi alamat baru untuk menampung data hasil copy tersebut."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Copy semantics ",(0,t.kt)("em",{parentName:"p"},"by default")," berlaku pada tipe primitif."),(0,t.kt)("p",{parentName:"blockquote"},"Sudah dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/basic-memory-management#a323-stack-memory"},"Memory Management \u279c Stack")," bahwa data primitif disimpan di stack dan pengaksesannya sangat cepat."),(0,t.kt)("p",{parentName:"blockquote"},"Dengan ini, copy data pada tipe primitif meskipun menghasilkan alokasi memory baru, konsekuensinya tidak terlalu besar karena pengaksesannya sangat cepat.")),(0,t.kt)("p",null,"Ok, lalu bagaimana dengan contoh ke-2 berikut? Apakah yang terjadi juga sama?"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let a = String::from("hello rust");\nlet b = a;\nprintln!("a: {:?}, b: {:?}", a, b);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(5572).Z,width:"804",height:"394"})),(0,t.kt)("p",null,"Wow, malah muncul error. Kok bisa, padahal kode program sangat straightforward."),(0,t.kt)("p",null,"Jadi begini, sebelumnya sudah dibahas bahwa semua tipe data primitif mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),". Namun untuk tipe data non-primitif (yang salah satunya adalah custom type ",(0,t.kt)("inlineCode",{parentName:"p"},"String"),") yang diadopsi adalah ",(0,t.kt)("strong",{parentName:"p"},"move semantics"),"."),(0,t.kt)("p",null,"Pada ",(0,t.kt)("em",{parentName:"p"},"move semantics"),", ketika ada operasi assignment seperti ",(0,t.kt)("inlineCode",{parentName:"p"},"let a = b;"),", maka yang terjadi adalah owner dari data string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello rust")," berpindah dari variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),". Jadinya, mulai dari statement ",(0,t.kt)("inlineCode",{parentName:"p"},"let a = b;")," dan statement seterusnya, owner dari data string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello rust")," bukan lagi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a"),", tetapi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),"."),(0,t.kt)("p",null,"Ok, sampai sini cukup jelas. Tapi kenapa bisa error? Error muncul karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," sudah tidak bisa digunakan lagi, data-nya (string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello rust"),") sudah berpindah ke variabel lain. Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," menjadi tidak valid setelah owner-nya berpindah, dan ini adalah penyebab kenapa pemanggilan variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," via macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," menghasilkan error."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Ingat, setiap data pasti punya owner, dan satu data ownernya hanya satu."),(0,t.kt)("p",{parentName:"blockquote"},"Pada contoh di atas, owner data ",(0,t.kt)("inlineCode",{parentName:"p"},"hello rust")," sudah berpindah dari variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),".")),(0,t.kt)("p",null,"Intinya, variabel yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics"),", setiap kali ada operasi assignment maka owner akan berpindah ke variabel baru."),(0,t.kt)("p",null,"Jika perlu, silakan coba praktikkan dengan tipe data non-primitif lainnya, seperti struct atau lainnya. Dengan pseudocode yang sama seperti contoh di atas, hasilnya adalah sama, yaitu error."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nstruct MyStruct;\n\nlet g = MyStruct{};\nlet h = g;\n\nprintln!("g: {:?}, h: {:?}", g, h);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(5914).Z,width:"821",height:"449"})),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Copy semantics di Rust merupakan sifat yang dimiliki oleh trait ",(0,t.kt)("inlineCode",{parentName:"p"},"std::marker::Copy"),"."),(0,t.kt)("ul",{parentName:"blockquote"},(0,t.kt)("li",{parentName:"ul"},"Semua data primitif meng-implement trait ",(0,t.kt)("inlineCode",{parentName:"li"},"std::marker::Copy")," yang berarti mengadopsi ",(0,t.kt)("em",{parentName:"li"},"copy semantics"),"."),(0,t.kt)("li",{parentName:"ul"},"Data non-primitif mengadopsi ",(0,t.kt)("em",{parentName:"li"},"move semantics"),".")),(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),".")),(0,t.kt)("h2",{id:"a345-alokasi--dealokasi"},"A.34.5. Alokasi & dealokasi"),(0,t.kt)("p",null,"Ok, masuk ke pembahasan selanjutnya, yaitu tentang alokasi dan dealokasi sebuah variabel dalam scope-nya."),(0,t.kt)("p",null,"Di sini kita akan gunakan custom type ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," sebagai contoh untuk mempelajari ownership. Untuk tipe non-primitif lainnya yang juga mengadopsi ",(0,t.kt)("strong",{parentName:"p"},"move semantics")," sebenarnya bisa dijadikan contoh. Tapi penulis memilih tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," karena sering digunakan."),(0,t.kt)("p",null,"Sekarang perhatikan kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n do_something();\n}\n\nfn do_something() {\n let mut k = String::from("hello");\n\n {\n let m = String::from("hello world");\n let n = String::from("from rust");\n k = n;\n \n println!("{:?}", m);\n }\n\n println!("{:?}", k);\n}\n')),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main")," dideklarasikan, isinya adalah pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),"."),(0,t.kt)("p",null,"Di dalam fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),", variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"k")," dideklarasikan. Statement deklarasi tersebut adalah kapan proses alokasi memory berlangsung untuk data variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"k"),"."),(0,t.kt)("p",null,"Kemudian ada block expression, dan di dalamnya terjadi lagi proses alokasi memory untuk data variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," dengan nilai adalah string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello world"),", dan variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"n")," berisi string ",(0,t.kt)("inlineCode",{parentName:"p"},"from rust"),"."),(0,t.kt)("p",null,"Masih di dalam block expression, terjadi assignment operation, data variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"n")," berpindah ke variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"k"),". Lalu bagaimana nasib data string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello")," yang sebelumnya dimiliki oleh ",(0,t.kt)("inlineCode",{parentName:"p"},"n"),"? Yang terjadi adalah data tersebut tidak memiliki reference, dan nantinya di akhir fungsi akan di-dealokasi. Proses dealokasi terjadi setelah fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," selesai dieksekusi karena variable scope data tersebut adalah di block kode fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),"."),(0,t.kt)("p",null,"Ok, masih dalam block expression. Ada statement print untuk data variable ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," yang merupakan statement terakhir di block expression tersebut. Setelah eksekusi block expression selesai, yang terjadi kemudian adalah:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"data variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"m")," di-dealokasi, karena sudah ",(0,t.kt)("em",{parentName:"li"},"out of scope"),"."),(0,t.kt)("li",{parentName:"ul"},"data variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"n")," tidak di-dealokasi, karena sudah berpindah scope-nya ke fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"do_something"),". Data tersebut sekarang ownernya adalah variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"k")," yang scope-nya ada di block fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"do_something"),".")),(0,t.kt)("p",null,"Kemudian variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"k")," di-print, dan fungsi selesai dieksekusi. Pada moment inilah semua data dalam scope block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," di-dealokasi, karena kesemua data tersebut adalah ",(0,t.kt)("em",{parentName:"p"},"out of scope")," dan tidak ada yang berpindah ke block fungsi lainnya."),(0,t.kt)("p",null,"Secara garis besar seperti itu proses manajemen memory pada Rust yang menerapkan konsep ",(0,t.kt)("em",{parentName:"p"},"ownership"),"."),(0,t.kt)("h2",{id:"a346-transfer-ownership"},"A.34.6. Transfer ownership"),(0,t.kt)("p",null,"Di atas sudah dibahas bagaimana cara untuk transfer ownership data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics"),", yaitu cukup dengan statement assignment."),(0,t.kt)("p",null,"Berikut adalah contoh lain perihal transfer ownership. Data string yang owner awalnya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"msg1"),", berpindah ke ",(0,t.kt)("inlineCode",{parentName:"p"},"msg2"),", kemudian berpindah lagi ke ",(0,t.kt)("inlineCode",{parentName:"p"},"msg3"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let msg1 = String::from("hello");\nlet msg2 = msg1;\nlet msg3 = msg2;\nprintln!("{:?}", msg3);\n')),(0,t.kt)("p",null,"Coba modifikasi sedikit kode tersebut, dengan menambahkan deklarasi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"msg4")," yang nilai-nya didapat dari ",(0,t.kt)("inlineCode",{parentName:"p"},"msg2"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let msg1 = String::from("hello");\nlet msg2 = msg1;\nlet msg3 = msg2;\n\nlet msg4 = msg2;\nprintln!("{:?}", msg4);\n')),(0,t.kt)("p",null,"Hasilnya adalah error, karena ",(0,t.kt)("inlineCode",{parentName:"p"},"msg2")," sudah invalid."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(3760).Z,width:"792",height:"394"})),(0,t.kt)("p",null,"Jika ingin memindah datanya ke ",(0,t.kt)("inlineCode",{parentName:"p"},"msg4"),", maka gunakan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"let msg4 = msg3"),"."),(0,t.kt)("h3",{id:"-transfer-ownership-via-return-value"},"\u25c9 Transfer ownership via return value"),(0,t.kt)("p",null,"Transfer ownership data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics")," juga bisa dilakukan antar fungsi via return value. Sebagai contoh pada kode berikut, variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," yang berada di dalam block expression berpindah ke luar scope yaitu ke block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," via operasi assignment ",(0,t.kt)("inlineCode",{parentName:"p"},"k = m"),". Kemudian berpindah lagi ke fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main")," via return value pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let msg = do_something();\n println!("{:?}", msg);\n}\n \nfn do_something() -> String {\n let mut k = String::from("hello");\n\n {\n let m = String::from("hello world");\n k = m;\n }\n\n return k;\n}\n')),(0,t.kt)("h3",{id:"-transfer-ownership-via-parameterargument"},"\u25c9 Transfer ownership via parameter/argument"),(0,t.kt)("p",null,"Pemanggilan fungsi dengan menyisipkan argument juga menghasilkan proses transfer ownership untuk data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics"),". Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let msg = String::from("hello rust");\n say_hello(msg);\n}\n \nfn say_hello(param: String) {\n println!("{:?}", param);\n}\n')),(0,t.kt)("p",null,"Pada kode di atas, data variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"msg")," owner-nya berpindah ke parameter bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"param")," milik fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello"),"."),(0,t.kt)("p",null,"Ok, sampai di sini semoga cukup jelas ya tentang bagaimana proses transfer ownership terjadi pada data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics"),"."),(0,t.kt)("p",null,"Untuk data bertipe primitif (yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),") kita tidak perlu repot memikirkan dimana owner datanya, karena setiap operasi assignment, data akan di-copy dan hasilnya ada data baru dengan owner baru."),(0,t.kt)("p",null,"Tapi kalau dipikir-pikir justru lebih repot mengurus data yang ownernya berpindah saat assignment. Sebagai contoh, misal variabel digunakan di fungsi lain, kemudian digunakan lagi di scope asalnya. Repot juga kalau setiap saat harus dikembalikan lagi via return value."),(0,t.kt)("p",null,"Misalnya pada kasus berikut ini. Hasilnya pasti error, karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"msg")," di print setelah owner-nya berpindah ke fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let msg = String::from("hello rust");\n say_hello(msg);\n println!("{:?}", msg);\n}\n \nfn say_hello(param: String) {\n println!("{:?}", param);\n}\n')),(0,t.kt)("p",null,"Lalu apa solusinya? apakah harus mengembalikannya via return value? Contohnya seperti kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut msg = String::from("hello rust"); // 1. awalnya data di sini\n msg = say_hello(msg); // 3. lalu dikembalikan lagi ke sini\n println!("{:?}", msg);\n}\n \nfn say_hello(param: String) -> String { // 2. kemudian berpindah ke sini\n println!("{:?}", param);\n param\n}\n')),(0,t.kt)("p",null,"Boleh-boleh saja sebenarnya pakai approach tersebut, tapi malah makin repot bukan?"),(0,t.kt)("p",null,"Ada lagi solusi lainnya yang bisa digunakan, yaitu dengan memanfaatkan method ",(0,t.kt)("inlineCode",{parentName:"p"},"clone")," untuk cloning data."),(0,t.kt)("h2",{id:"a347-clone-data"},"A.34.7. Clone data"),(0,t.kt)("p",null,"Semua tipe data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics")," meng-implement trait ",(0,t.kt)("inlineCode",{parentName:"p"},"std::clone::Clone"),". Trait ini memiliki method bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"clone")," yang gunanya adalah menduplikasi atau clonging data. Cukup panggil saja method tersebut, maka data akan ter-cloning."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),".")),(0,t.kt)("p",null,"Kita akan terapkan pada kode sebelumnya, hasilnya kurang lebih seperti ini. Pada argument pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," disisipkan data cloning via statement ",(0,t.kt)("inlineCode",{parentName:"p"},"msg.clone()"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let msg = String::from("hello rust");\n say_hello(msg.clone());\n println!("{:?}", msg);\n}\n \nfn say_hello(param: String) {\n println!("{:?}", param);\n}\n')),(0,t.kt)("p",null,"Ok, dengan ini masalah untuk me-reuse data yang bisa berpindah ownernya dianggap beres. Memang beres, tapi apakah cara ini baik kalau dilihat dari sudut pandang memory management?"),(0,t.kt)("p",null,"Konsekuensi dari cloning data adalah terjadi proses alokasi lagi di memory. Data akan di-duplikasi dan dialokasikan ke alamat memory baru, jadinya kurang efisien. Ditambah lagi, jika kita mengacu ke penjelasan pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/basic-memory-management#a324-heap-memory"},"Memory Management \u279c Heap Memory"),", data ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," isinya disimpan di heap memory yang pengaksesannya lebih lambat dibanding pengaksesan data stack. Dari sini bisa disimpulkan bahwa cloning bukan solusi yang paling baik (kecuali terpaksa)."),(0,t.kt)("p",null,(0,t.kt)("strong",{parentName:"p"},"Solusi yang lebih baik adalah dengan melakukan operasi pinjam data dari owner aslinya tanpa perlu melakukan operasi perpindahan owner"),", yang pada Rust programming disebut dengan ",(0,t.kt)("strong",{parentName:"p"},"borrowing"),"."),(0,t.kt)("p",null,"Kita sebenarnya sudah menerapkannya beberapa kali pada chapter yang lalu, tapi kita akan bahas lagi lebih detail (dari sudut pandang ownership) pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),"."),(0,t.kt)("h2",{id:"a348-ownership-pada-data-literal"},"A.34.8. Ownership pada data literal"),(0,t.kt)("p",null,"Kita akan bahas topik ini pada chapter berikutnya, yaitu chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/borrowing#a347-owner-dan-borrower-pada-literal"},"Borrowing"),"."),(0,t.kt)("h2",{id:"a349-move-semantics-pada-macro-println"},"A.34.9. ",(0,t.kt)("em",{parentName:"h2"},"Move semantics")," pada macro ",(0,t.kt)("inlineCode",{parentName:"h2"},"println")),(0,t.kt)("p",null,"Ada yang unik dengan macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),". Silakan coba kode berikut agar terlihat keunikannya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let str1 = String::from("luwe");\n println!("{str1}");\n do_something(str1);\n \n let str2 = String::from("ngelak");\n do_something(str2);\n println!("{str2}");\n}\n\nfn do_something(str: String) {\n println!("{}", str);\n}\n')),(0,t.kt)("p",null,"Jika dijalankan ada error di statement terakhir fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(8255).Z,width:"883",height:"682"})),(0,t.kt)("p",null,"Seperti yang sudah dibahas, bahwa tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," mengadopsi move semantics. Ketika data bertipe ini digunakan pada operasi assignment seperti contohnya sebagai argument pemanggilan fungsi, maka owner berpindah."),(0,t.kt)("p",null,"Tapi entah kenapa, khusus dalam pemanggilan macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),", owner-nya tidak berpindah. Ajaib."),(0,t.kt)("p",null,"Silakan lihat sendiri di gambar di atas, ketika ",(0,t.kt)("inlineCode",{parentName:"p"},"str1")," digunakan pada macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),", kemudian digunakan lagi pada argument pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),", hasilnya tidak error."),(0,t.kt)("p",null,"Akan tetapi ketika digunakan pada pemanggilan fungsi terlebih dahulu, jika digunakan lagi di statement di bawahnya hasilnya error."),(0,t.kt)("p",null,"Ini adalah keistimewaan dari macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," dan beberapa macro untuk keperluan printing lainnya."),(0,t.kt)("p",null,"Lebih jelasnya mengenai macro dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"#/wip/macro"},"Macro"),"."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/ownership"},"github.com/novalagung/dasarpemrogramanrust-example/../ownership")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-01-what-is-ownership.html"},"https://doc.rust-lang.org/book/ch04-01-what-is-ownership.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/marker/trait.Copy.html"},"https://doc.rust-lang.org/std/marker/trait.Copy.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/nomicon/ownership.html"},"https://doc.rust-lang.org/nomicon/ownership.html"))))}o.isMDXComponent=!0},1146:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},5572:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/ownership-2-b8db9994157f3e86c2a74d048ca45363.png"},5914:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/ownership-3-f9834609466195e2d46433bbc089153c.png"},3760:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/ownership-4-90c3dccbd5f0429a8760c9a445f446af.png"},8255:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/ownership-5-42080e6ebdac72f8d756dcf0dacb2061.png"}}]); \ No newline at end of file +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4248],{3905:(a,e,n)=>{n.d(e,{Zo:()=>d,kt:()=>u});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function l(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var s=i.createContext({}),m=function(a){var e=i.useContext(s),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},d=function(a){var e=m(a.components);return i.createElement(s.Provider,{value:e},a.children)},o={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},k=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,r=a.originalType,s=a.parentName,d=p(a,["components","mdxType","originalType","parentName"]),k=m(n),u=t,g=k["".concat(s,".").concat(u)]||k[u]||o[u]||r;return n?i.createElement(g,l(l({ref:e},d),{},{components:n})):i.createElement(g,l({ref:e},d))}));function u(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var r=n.length,l=new Array(r);l[0]=k;var p={};for(var s in e)hasOwnProperty.call(e,s)&&(p[s]=e[s]);p.originalType=a,p.mdxType="string"==typeof a?a:t,l[1]=p;for(var m=2;m{n.r(e),n.d(e,{assets:()=>s,contentTitle:()=>l,default:()=>o,frontMatter:()=>r,metadata:()=>p,toc:()=>m});var i=n(7462),t=(n(7294),n(3905));const r={sidebar_position:34,title:"A.34. Ownership",sidebar_label:"A.34. Ownership"},l=void 0,p={unversionedId:"basic/ownership",id:"basic/ownership",title:"A.34. Ownership",description:"Chapter ini berisi pembahasan tentang ownership, bagaimana Rust melakukan manajemen memory dengan menerapkan approach ownership ini.",source:"@site/docs/basic/ownership.md",sourceDirName:"basic",slug:"/basic/ownership",permalink:"/basic/ownership",draft:!1,tags:[],version:"current",sidebarPosition:34,frontMatter:{sidebar_position:34,title:"A.34. Ownership",sidebar_label:"A.34. Ownership"},sidebar:"tutorialSidebar",previous:{title:"A.33. Pointer & References",permalink:"/basic/pointer-references"},next:{title:"A.35. Borrowing",permalink:"/basic/borrowing"}},s={},m=[{value:"A.34.1. Konsep ownership",id:"a341-konsep-ownership",level:2},{value:"A.34.2. Aturan ownership",id:"a342-aturan-ownership",level:2},{value:"A.34.3. Variable scope",id:"a343-variable-scope",level:2},{value:"A.34.4. Copy semantics vs. move semantics",id:"a344-copy-semantics-vs-move-semantics",level:2},{value:"A.34.5. Alokasi & dealokasi",id:"a345-alokasi--dealokasi",level:2},{value:"A.34.6. Transfer ownership",id:"a346-transfer-ownership",level:2},{value:"\u25c9 Transfer ownership via return value",id:"-transfer-ownership-via-return-value",level:3},{value:"\u25c9 Transfer ownership via parameter/argument",id:"-transfer-ownership-via-parameterargument",level:3},{value:"A.34.7. Clone data",id:"a347-clone-data",level:2},{value:"A.34.8. Ownership pada data literal",id:"a348-ownership-pada-data-literal",level:2},{value:"A.34.9. Move semantics pada macro println",id:"a349-move-semantics-pada-macro-println",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function o(a){let{components:e,...r}=a;return(0,t.kt)("wrapper",(0,i.Z)({},d,r,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Chapter ini berisi pembahasan tentang ownership, bagaimana Rust melakukan manajemen memory dengan menerapkan approach ownership ini."),(0,t.kt)("p",null,"O iya, diwajibkan untuk mempelajari terlebih dahulu tentang dasar ",(0,t.kt)("a",{parentName:"p",href:"/basic/basic-memory-management"},"Memory Management")," dan juga ",(0,t.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),", yang keduanya adalah dibahas pada chapter sebelumnya. Jika pembaca mempelajari ebook ini secara urut maka tidak usah khawatir."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Pembahasan pada chapter ini adalah salah satu hal yang paling penting untuk dipahami dalam Rust programming, karena topik yang berhubungan dengan memory management adalah hal yang krusial pada system programming."),(0,t.kt)("p",{parentName:"blockquote"},"Silakan ulang-ulang chapter ini jika diperlukan.")),(0,t.kt)("h2",{id:"a341-konsep-ownership"},"A.34.1. Konsep ownership"),(0,t.kt)("p",null,"Ownership merupakan kumpulan aturan yang ada di Rust yang dijadikan acuan oleh compiler dalam pengelolahan memory."),(0,t.kt)("p",null,"Sudah disinggung pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/basic-memory-management"},"Memory Management")," bahwa Rust tidak menerapkan GC ataupun ARC dalam manajemen memory-nya. Rust membebankan manajemen memory pada penulis kode program, yaitu kita/programmer."),(0,t.kt)("p",null,"Aturan ownership ada banyak, dan programmer harus mengikutinya, karena jika tidak, maka proses kompilasi program dan eksekusi program akan gagal dan hasilnya error."),(0,t.kt)("h2",{id:"a342-aturan-ownership"},"A.34.2. Aturan ownership"),(0,t.kt)("p",null,"Ada 3 aturan penting yang wajib diketahui:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},(0,t.kt)("strong",{parentName:"p"},"Semua nilai/data/value di Rust memiliki owner"),". Misal kita berbicara tentang deklarasi variabel dengan predefined value, maka value variabel adalah yang dimaksud dengan nilai/data/value, dan variabel itu sendiri adalah owner dari nilai/data/value tersebut.")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Pada waktu yang sama, ",(0,t.kt)("strong",{parentName:"p"},"hanya boleh ada 1 owner"),". Satu data, ownernya hanya satu.")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},(0,t.kt)("strong",{parentName:"p"},"Ketika eksekusi sebuah block scope selesai, maka owner dari data-data yang ada dalam scope tersebut akan di-drop atau di-dealokasi")," (dengan pengecualian yaitu owner berpindah ke luar scope)."))),(0,t.kt)("h2",{id:"a343-variable-scope"},"A.34.3. Variable scope"),(0,t.kt)("p",null,"Sebelum kita masuk ke pembahasan yang lebih detail mengenai ownership, mari pelajari terlebih dahulu tentang apa itu variable scope."),(0,t.kt)("p",null,"Variable scope maksudnya adalah di block scope mana suatu variabel dideklarasikan, dan dalam block scope tersebut variabel menjadi valid (bisa digunakan). Di luar scope-nya variabel menjadi tidak valid, tidak bisa digunakan."),(0,t.kt)("p",null,"Agar lebih jelas, silakan perhatikan kode berikut, tidak perlu dipraktikkan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn do_something() {\n let data_one = "one";\n // ...\n}\n\nfn main() {\n let data_two = "two";\n // ...\n\n {\n let data_three = "three";\n // ...\n }\n\n do_something();\n\n if true {\n let data_four = "four";\n // ...\n }\n}\n')),(0,t.kt)("p",null,"Program di atas memiliki 4 buah block kode:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", yang isinya adalah variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_two")," dan beberapa sub-block dan 1 buah pemanggilan fungsi. Di dalam block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_two")," adalah valid, dan bisa digunakan dalam fungsi scope tersebut beserta sub-block lainnya (block expression dan block kode ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),").")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Block expression yang berisi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_three"),". Pada block scope ini, variabel tersebut dan juga variable ",(0,t.kt)("inlineCode",{parentName:"p"},"data_two")," adalah valid.")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),", yang isinya variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_one"),". Variabel tersebut hanya akan valid dalam block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),".")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Block seleksi kondisi ",(0,t.kt)("inlineCode",{parentName:"p"},"if"),". Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data_four")," berada dalam block ini, maka variabel tersebut adalah valid dalam block tersebut. Selain itu ",(0,t.kt)("inlineCode",{parentName:"p"},"data_two")," juga valid dalam block seleksi kondisi ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," ini."))),(0,t.kt)("p",null,"Intinya, variabel adalah valid ketika berada di dalam scope-nya (istilahnya ",(0,t.kt)("em",{parentName:"p"},"into scope"),"), dan variabel menjadi tidak valid atau invalid ketika keluar dari scope (istilahnya ",(0,t.kt)("em",{parentName:"p"},"out of scope"),")."),(0,t.kt)("h2",{id:"a344-copy-semantics-vs-move-semantics"},"A.34.4. Copy semantics vs. move semantics"),(0,t.kt)("p",null,"Di atas sudah dijelaskan mengenai aturan ownership, yang salah satunya adalah setiap data yang ada di Rust memiliki owner, dan 1 data owner pasti 1 (tidak lebih)."),(0,t.kt)("p",null,"Agar lebih jelas mari kita perhatikan contoh berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let x = 24;\nlet y = x;\nprintln!("x: {:?}, y: {:?}", x, y);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(1146).Z,width:"333",height:"110"})),(0,t.kt)("p",null,"Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," dideklarasikan dengan nilai ",(0,t.kt)("inlineCode",{parentName:"p"},"24"),", artinya variabel tersebut adalah owner dari data ",(0,t.kt)("inlineCode",{parentName:"p"},"24"),". Kemudian variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," dijadikan sebagai nilai variabel baru bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),", dari sini apakah berarti owner data ",(0,t.kt)("inlineCode",{parentName:"p"},"24")," adalah berpindah dari variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),"? jawabannya adalah ",(0,t.kt)("strong",{parentName:"p"},"tidak"),"."),(0,t.kt)("p",null,"Yang terjadi adalah data ",(0,t.kt)("inlineCode",{parentName:"p"},"24")," milik owner ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," di-copy atau diduplikasi sebagai data baru yang owner-nya juga baru, yang pada contoh ini adalah variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),"."),(0,t.kt)("p",null,"Hasilnya, kedua variabel tersebut masing-masing adalah owner dari data yang berbeda, meskipun sumbernya adalah dari data yang sama. Perilaku ini disebut dengan ",(0,t.kt)("strong",{parentName:"p"},"copy semantics"),"."),(0,t.kt)("p",null,"Semua tipe data primitif di Rust ",(0,t.kt)("em",{parentName:"p"},"by default")," mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),". Ketika terjadi operasi assignment seperti pada contoh di atas, maka yang terjadi adalah data di-copy sebagai data baru, dengan owner baru, dan di sisi memory juga terjadi alokasi alamat baru untuk menampung data hasil copy tersebut."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Copy semantics ",(0,t.kt)("em",{parentName:"p"},"by default")," berlaku pada tipe primitif."),(0,t.kt)("p",{parentName:"blockquote"},"Sudah dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/basic-memory-management#a323-stack-memory"},"Memory Management \u279c Stack")," bahwa data primitif disimpan di stack dan pengaksesannya sangat cepat."),(0,t.kt)("p",{parentName:"blockquote"},"Dengan ini, copy data pada tipe primitif meskipun menghasilkan alokasi memory baru, konsekuensinya tidak terlalu besar karena pengaksesannya sangat cepat.")),(0,t.kt)("p",null,"Ok, lalu bagaimana dengan contoh ke-2 berikut? Apakah yang terjadi juga sama?"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let a = String::from("hello rust");\nlet b = a;\nprintln!("a: {:?}, b: {:?}", a, b);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(5572).Z,width:"804",height:"394"})),(0,t.kt)("p",null,"Wow, malah muncul error. Kok bisa, padahal kode program sangat straightforward."),(0,t.kt)("p",null,"Jadi begini, sebelumnya sudah dibahas bahwa semua tipe data primitif mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),". Namun untuk tipe data non-primitif (yang salah satunya adalah custom type ",(0,t.kt)("inlineCode",{parentName:"p"},"String"),") yang diadopsi adalah ",(0,t.kt)("strong",{parentName:"p"},"move semantics"),"."),(0,t.kt)("p",null,"Pada ",(0,t.kt)("em",{parentName:"p"},"move semantics"),", ketika ada operasi assignment seperti ",(0,t.kt)("inlineCode",{parentName:"p"},"let a = b;"),", maka yang terjadi adalah owner dari data string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello rust")," berpindah dari variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),". Jadinya, mulai dari statement ",(0,t.kt)("inlineCode",{parentName:"p"},"let a = b;")," dan statement seterusnya, owner dari data string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello rust")," bukan lagi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a"),", tetapi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),"."),(0,t.kt)("p",null,"Ok, sampai sini cukup jelas. Tapi kenapa bisa error? Error muncul karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," sudah tidak bisa digunakan lagi, data-nya (string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello rust"),") sudah berpindah ke variabel lain. Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," menjadi tidak valid setelah owner-nya berpindah, dan ini adalah penyebab kenapa pemanggilan variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," via macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," menghasilkan error."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Ingat, setiap data pasti punya owner, dan satu data ownernya hanya satu."),(0,t.kt)("p",{parentName:"blockquote"},"Pada contoh di atas, owner data ",(0,t.kt)("inlineCode",{parentName:"p"},"hello rust")," sudah berpindah dari variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"a")," ke ",(0,t.kt)("inlineCode",{parentName:"p"},"b"),".")),(0,t.kt)("p",null,"Intinya, variabel yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics"),", setiap kali ada operasi assignment maka owner akan berpindah ke variabel baru."),(0,t.kt)("p",null,"Jika perlu, silakan coba praktikkan dengan tipe data non-primitif lainnya, seperti struct atau lainnya. Dengan pseudocode yang sama seperti contoh di atas, hasilnya adalah sama, yaitu error."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nstruct MyStruct;\n\nlet g = MyStruct{};\nlet h = g;\n\nprintln!("g: {:?}, h: {:?}", g, h);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(5914).Z,width:"821",height:"449"})),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Copy semantics di Rust merupakan sifat yang dimiliki oleh trait ",(0,t.kt)("inlineCode",{parentName:"p"},"std::marker::Copy"),"."),(0,t.kt)("ul",{parentName:"blockquote"},(0,t.kt)("li",{parentName:"ul"},"Semua data primitif meng-implement trait ",(0,t.kt)("inlineCode",{parentName:"li"},"std::marker::Copy")," yang berarti mengadopsi ",(0,t.kt)("em",{parentName:"li"},"copy semantics"),"."),(0,t.kt)("li",{parentName:"ul"},"Data non-primitif mengadopsi ",(0,t.kt)("em",{parentName:"li"},"move semantics"),".")),(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),".")),(0,t.kt)("h2",{id:"a345-alokasi--dealokasi"},"A.34.5. Alokasi & dealokasi"),(0,t.kt)("p",null,"Ok, masuk ke pembahasan selanjutnya, yaitu tentang alokasi dan dealokasi sebuah variabel dalam scope-nya."),(0,t.kt)("p",null,"Di sini kita akan gunakan custom type ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," sebagai contoh untuk mempelajari ownership. Untuk tipe non-primitif lainnya yang juga mengadopsi ",(0,t.kt)("strong",{parentName:"p"},"move semantics")," sebenarnya bisa dijadikan contoh. Tapi penulis memilih tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," karena sering digunakan."),(0,t.kt)("p",null,"Sekarang perhatikan kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n do_something();\n}\n\nfn do_something() {\n let mut k = String::from("hello");\n\n {\n let m = String::from("hello world");\n let n = String::from("from rust");\n k = n;\n \n println!("{:?}", m);\n }\n\n println!("{:?}", k);\n}\n')),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main")," dideklarasikan, isinya adalah pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),"."),(0,t.kt)("p",null,"Di dalam fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),", variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"k")," dideklarasikan. Statement deklarasi tersebut adalah kapan proses alokasi memory berlangsung untuk data variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"k"),"."),(0,t.kt)("p",null,"Kemudian ada block expression, dan di dalamnya terjadi lagi proses alokasi memory untuk data variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," dengan nilai adalah string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello world"),", dan variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"n")," berisi string ",(0,t.kt)("inlineCode",{parentName:"p"},"from rust"),"."),(0,t.kt)("p",null,"Masih di dalam block expression, terjadi assignment operation, data variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"n")," berpindah ke variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"k"),". Lalu bagaimana nasib data string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello")," yang sebelumnya dimiliki oleh ",(0,t.kt)("inlineCode",{parentName:"p"},"n"),"? Yang terjadi adalah data tersebut tidak memiliki reference, dan nantinya di akhir fungsi akan di-dealokasi. Proses dealokasi terjadi setelah fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," selesai dieksekusi karena variable scope data tersebut adalah di block kode fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),"."),(0,t.kt)("p",null,"Ok, masih dalam block expression. Ada statement print untuk data variable ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," yang merupakan statement terakhir di block expression tersebut. Setelah eksekusi block expression selesai, yang terjadi kemudian adalah:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"data variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"m")," di-dealokasi, karena sudah ",(0,t.kt)("em",{parentName:"li"},"out of scope"),"."),(0,t.kt)("li",{parentName:"ul"},"data variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"n")," tidak di-dealokasi, karena sudah berpindah scope-nya ke fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"do_something"),". Data tersebut sekarang ownernya adalah variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"k")," yang scope-nya ada di block fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"do_something"),".")),(0,t.kt)("p",null,"Kemudian variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"k")," di-print, dan fungsi selesai dieksekusi. Pada moment inilah semua data dalam scope block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," di-dealokasi, karena kesemua data tersebut adalah ",(0,t.kt)("em",{parentName:"p"},"out of scope")," dan tidak ada yang berpindah ke block fungsi lainnya."),(0,t.kt)("p",null,"Secara garis besar seperti itu proses manajemen memory pada Rust yang menerapkan konsep ",(0,t.kt)("em",{parentName:"p"},"ownership"),"."),(0,t.kt)("h2",{id:"a346-transfer-ownership"},"A.34.6. Transfer ownership"),(0,t.kt)("p",null,"Di atas sudah dibahas bagaimana cara untuk transfer ownership data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics"),", yaitu cukup dengan statement assignment."),(0,t.kt)("p",null,"Berikut adalah contoh lain perihal transfer ownership. Data string yang owner awalnya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"msg1"),", berpindah ke ",(0,t.kt)("inlineCode",{parentName:"p"},"msg2"),", kemudian berpindah lagi ke ",(0,t.kt)("inlineCode",{parentName:"p"},"msg3"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let msg1 = String::from("hello");\nlet msg2 = msg1;\nlet msg3 = msg2;\nprintln!("{:?}", msg3);\n')),(0,t.kt)("p",null,"Coba modifikasi sedikit kode tersebut, dengan menambahkan deklarasi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"msg4")," yang nilai-nya didapat dari ",(0,t.kt)("inlineCode",{parentName:"p"},"msg2"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let msg1 = String::from("hello");\nlet msg2 = msg1;\nlet msg3 = msg2;\n\nlet msg4 = msg2;\nprintln!("{:?}", msg4);\n')),(0,t.kt)("p",null,"Hasilnya adalah error, karena ",(0,t.kt)("inlineCode",{parentName:"p"},"msg2")," sudah invalid."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(3760).Z,width:"792",height:"394"})),(0,t.kt)("p",null,"Jika ingin memindah datanya ke ",(0,t.kt)("inlineCode",{parentName:"p"},"msg4"),", maka gunakan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"let msg4 = msg3"),"."),(0,t.kt)("h3",{id:"-transfer-ownership-via-return-value"},"\u25c9 Transfer ownership via return value"),(0,t.kt)("p",null,"Transfer ownership data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics")," juga bisa dilakukan antar fungsi via return value. Sebagai contoh pada kode berikut, variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," yang berada di dalam block expression berpindah ke luar scope yaitu ke block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," via operasi assignment ",(0,t.kt)("inlineCode",{parentName:"p"},"k = m"),". Kemudian berpindah lagi ke fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main")," via return value pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let msg = do_something();\n println!("{:?}", msg);\n}\n \nfn do_something() -> String {\n let mut k = String::from("hello");\n\n {\n let m = String::from("hello world");\n k = m;\n }\n\n return k;\n}\n')),(0,t.kt)("h3",{id:"-transfer-ownership-via-parameterargument"},"\u25c9 Transfer ownership via parameter/argument"),(0,t.kt)("p",null,"Pemanggilan fungsi dengan menyisipkan argument juga menghasilkan proses transfer ownership untuk data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics"),". Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let msg = String::from("hello rust");\n say_hello(msg);\n}\n \nfn say_hello(param: String) {\n println!("{:?}", param);\n}\n')),(0,t.kt)("p",null,"Pada kode di atas, data variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"msg")," owner-nya berpindah ke parameter bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"param")," milik fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello"),"."),(0,t.kt)("p",null,"Ok, sampai di sini semoga cukup jelas ya tentang bagaimana proses transfer ownership terjadi pada data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics"),"."),(0,t.kt)("p",null,"Untuk data bertipe primitif (yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),") kita tidak perlu repot memikirkan di mana letak owner datanya, karena setiap operasi assignment, data akan di-copy dan hasilnya ada data baru dengan owner baru."),(0,t.kt)("p",null,"Tapi kalau dipikir-pikir justru lebih repot mengurus data yang ownernya berpindah saat assignment. Sebagai contoh, misal variabel digunakan di fungsi lain, kemudian digunakan lagi di scope asalnya. Repot juga kalau setiap saat harus dikembalikan lagi via return value."),(0,t.kt)("p",null,"Misalnya pada kasus berikut ini. Hasilnya pasti error, karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"msg")," di print setelah owner-nya berpindah ke fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let msg = String::from("hello rust");\n say_hello(msg);\n println!("{:?}", msg);\n}\n \nfn say_hello(param: String) {\n println!("{:?}", param);\n}\n')),(0,t.kt)("p",null,"Lalu apa solusinya? apakah harus mengembalikannya via return value? Contohnya seperti kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut msg = String::from("hello rust"); // 1. awalnya data di sini\n msg = say_hello(msg); // 3. lalu dikembalikan lagi ke sini\n println!("{:?}", msg);\n}\n \nfn say_hello(param: String) -> String { // 2. kemudian berpindah ke sini\n println!("{:?}", param);\n param\n}\n')),(0,t.kt)("p",null,"Boleh-boleh saja sebenarnya pakai approach tersebut, tapi malah makin repot bukan?"),(0,t.kt)("p",null,"Ada lagi solusi lainnya yang bisa digunakan, yaitu dengan memanfaatkan method ",(0,t.kt)("inlineCode",{parentName:"p"},"clone")," untuk cloning data."),(0,t.kt)("h2",{id:"a347-clone-data"},"A.34.7. Clone data"),(0,t.kt)("p",null,"Semua tipe data yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics")," meng-implement trait ",(0,t.kt)("inlineCode",{parentName:"p"},"std::clone::Clone"),". Trait ini memiliki method bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"clone")," yang gunanya adalah menduplikasi atau clonging data. Cukup panggil saja method tersebut, maka data akan ter-cloning."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),".")),(0,t.kt)("p",null,"Kita akan terapkan pada kode sebelumnya, hasilnya kurang lebih seperti ini. Pada argument pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"say_hello")," disisipkan data cloning via statement ",(0,t.kt)("inlineCode",{parentName:"p"},"msg.clone()"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let msg = String::from("hello rust");\n say_hello(msg.clone());\n println!("{:?}", msg);\n}\n \nfn say_hello(param: String) {\n println!("{:?}", param);\n}\n')),(0,t.kt)("p",null,"Ok, dengan ini masalah untuk me-reuse data yang bisa berpindah ownernya dianggap beres. Memang beres, tapi apakah cara ini baik kalau dilihat dari sudut pandang memory management?"),(0,t.kt)("p",null,"Konsekuensi dari cloning data adalah terjadi proses alokasi lagi di memory. Data akan di-duplikasi dan dialokasikan ke alamat memory baru, jadinya kurang efisien. Ditambah lagi, jika kita mengacu ke penjelasan pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/basic-memory-management#a324-heap-memory"},"Memory Management \u279c Heap Memory"),", data ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," isinya disimpan di heap memory yang pengaksesannya lebih lambat dibanding pengaksesan data stack. Dari sini bisa disimpulkan bahwa cloning bukan solusi yang paling baik (kecuali terpaksa)."),(0,t.kt)("p",null,(0,t.kt)("strong",{parentName:"p"},"Solusi yang lebih baik adalah dengan melakukan operasi pinjam data dari owner aslinya tanpa perlu melakukan operasi perpindahan owner"),", yang pada Rust programming disebut dengan ",(0,t.kt)("strong",{parentName:"p"},"borrowing"),"."),(0,t.kt)("p",null,"Kita sebenarnya sudah menerapkannya beberapa kali pada chapter yang lalu, tapi kita akan bahas lagi lebih detail (dari sudut pandang ownership) pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing"),"."),(0,t.kt)("h2",{id:"a348-ownership-pada-data-literal"},"A.34.8. Ownership pada data literal"),(0,t.kt)("p",null,"Kita akan bahas topik ini pada chapter berikutnya, yaitu chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/borrowing#a347-owner-dan-borrower-pada-literal"},"Borrowing"),"."),(0,t.kt)("h2",{id:"a349-move-semantics-pada-macro-println"},"A.34.9. ",(0,t.kt)("em",{parentName:"h2"},"Move semantics")," pada macro ",(0,t.kt)("inlineCode",{parentName:"h2"},"println")),(0,t.kt)("p",null,"Ada yang unik dengan macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),". Silakan coba kode berikut agar terlihat keunikannya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let str1 = String::from("luwe");\n println!("{str1}");\n do_something(str1);\n \n let str2 = String::from("ngelak");\n do_something(str2);\n println!("{str2}");\n}\n\nfn do_something(str: String) {\n println!("{}", str);\n}\n')),(0,t.kt)("p",null,"Jika dijalankan ada error di statement terakhir fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),"."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Ownership",src:n(8255).Z,width:"883",height:"682"})),(0,t.kt)("p",null,"Seperti yang sudah dibahas, bahwa tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," mengadopsi move semantics. Ketika data bertipe ini digunakan pada operasi assignment seperti contohnya sebagai argument pemanggilan fungsi, maka owner berpindah."),(0,t.kt)("p",null,"Tapi entah kenapa, khusus dalam pemanggilan macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),", owner-nya tidak berpindah. Ajaib."),(0,t.kt)("p",null,"Silakan lihat sendiri di gambar di atas, ketika ",(0,t.kt)("inlineCode",{parentName:"p"},"str1")," digunakan pada macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),", kemudian digunakan lagi pada argument pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),", hasilnya tidak error."),(0,t.kt)("p",null,"Akan tetapi ketika digunakan pada pemanggilan fungsi terlebih dahulu, jika digunakan lagi di statement di bawahnya hasilnya error."),(0,t.kt)("p",null,"Ini adalah keistimewaan dari macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," dan beberapa macro untuk keperluan printing lainnya."),(0,t.kt)("p",null,"Lebih jelasnya mengenai macro dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"#/wip/macro"},"Macro"),"."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/ownership"},"github.com/novalagung/dasarpemrogramanrust-example/../ownership")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-01-what-is-ownership.html"},"https://doc.rust-lang.org/book/ch04-01-what-is-ownership.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/marker/trait.Copy.html"},"https://doc.rust-lang.org/std/marker/trait.Copy.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/nomicon/ownership.html"},"https://doc.rust-lang.org/nomicon/ownership.html"))))}o.isMDXComponent=!0},1146:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},5572:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/ownership-2-b8db9994157f3e86c2a74d048ca45363.png"},5914:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/ownership-3-f9834609466195e2d46433bbc089153c.png"},3760:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/ownership-4-90c3dccbd5f0429a8760c9a445f446af.png"},8255:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/ownership-5-42080e6ebdac72f8d756dcf0dacb2061.png"}}]); \ No newline at end of file diff --git a/assets/js/8333ea4d.79e6ada8.js b/assets/js/8333ea4d.79e6ada8.js new file mode 100644 index 00000000..3f9e8750 --- /dev/null +++ b/assets/js/8333ea4d.79e6ada8.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[9323],{3905:(e,t,a)=>{a.d(t,{Zo:()=>d,kt:()=>s});var n=a(7294);function i(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function l(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,n)}return a}function r(e){for(var t=1;t=0||(i[a]=e[a]);return i}(e,t);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(i[a]=e[a])}return i}var u=n.createContext({}),A=function(e){var t=n.useContext(u),a=t;return e&&(a="function"==typeof e?e(t):r(r({},t),e)),a},d=function(e){var t=A(e.components);return n.createElement(u.Provider,{value:t},e.children)},k={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},b=n.forwardRef((function(e,t){var a=e.components,i=e.mdxType,l=e.originalType,u=e.parentName,d=p(e,["components","mdxType","originalType","parentName"]),b=A(a),s=i,c=b["".concat(u,".").concat(s)]||b[s]||k[s]||l;return a?n.createElement(c,r(r({ref:t},d),{},{components:a})):n.createElement(c,r({ref:t},d))}));function s(e,t){var a=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var l=a.length,r=new Array(l);r[0]=b;var p={};for(var u in t)hasOwnProperty.call(t,u)&&(p[u]=t[u]);p.originalType=e,p.mdxType="string"==typeof e?e:i,r[1]=p;for(var A=2;A{a.r(t),a.d(t,{assets:()=>u,contentTitle:()=>r,default:()=>k,frontMatter:()=>l,metadata:()=>p,toc:()=>A});var n=a(7462),i=(a(7294),a(3905));const l={sidebar_position:15,title:"A.15. Tipe Data \u279c Tuple",sidebar_label:"A.15. Tuple"},r=void 0,p={unversionedId:"basic/tuple",id:"basic/tuple",title:"A.15. Tipe Data \u279c Tuple",description:"Pada chapter ini kita akan membahas tentang tipe data kolektif bernama Tuple.",source:"@site/docs/basic/tuple.md",sourceDirName:"basic",slug:"/basic/tuple",permalink:"/basic/tuple",draft:!1,tags:[],version:"current",sidebarPosition:15,frontMatter:{sidebar_position:15,title:"A.15. Tipe Data \u279c Tuple",sidebar_label:"A.15. Tuple"},sidebar:"tutorialSidebar",previous:{title:"A.14. Slice (Basic)",permalink:"/basic/slice"},next:{title:"A.16. Vector",permalink:"/basic/vector"}},u={},A=[{value:"A.15.1. Tipe data tuple",id:"a151-tipe-data-tuple",level:2},{value:"A.15.2. Mutable tuple",id:"a152-mutable-tuple",level:2},{value:"A.15.3. Notasi deklarasi tuple",id:"a153-notasi-deklarasi-tuple",level:2},{value:"\u25c9 Type inference",id:"-type-inference",level:3},{value:"\u25c9 Manifest typing",id:"-manifest-typing",level:3},{value:"\u25c9 Packing tuple",id:"-packing-tuple",level:3},{value:"\u25c9 Unpacking tuple",id:"-unpacking-tuple",level:3},{value:"A.15.4. Tuple ()",id:"a154-tuple-",level:2},{value:"A.15.5. Tuple Structs",id:"a155-tuple-structs",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:A};function k(e){let{components:t,...l}=e;return(0,i.kt)("wrapper",(0,n.Z)({},d,l,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan membahas tentang tipe data kolektif bernama Tuple."),(0,i.kt)("h2",{id:"a151-tipe-data-tuple"},"A.15.1. Tipe data tuple"),(0,i.kt)("p",null,"Tuple merupakan tipe data yang isinya koleksi dari banyak data atau value, yang bisa jadi tiap value tersebut tipe datanya berbeda satu sama lain. Tuple biasa digunakan untuk menampung data yang ",(0,i.kt)("em",{parentName:"p"},"heterogeneous")," atau campuran."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tipe ini tidak diciptakan sebagai pengganti array."),(0,i.kt)("p",{parentName:"blockquote"},"Cukup mudah untuk mengidentifikasi kapan harus menggunakan tuple, contohnya: jika ada kebutuhan data harus bisa di-iterate, atau data memiliki pattern yg pasti seperti size-nya fixed, atau tipe datanya ",(0,i.kt)("em",{parentName:"p"},"homogeneous")," atau sejenis, ... maka baiknya gunakan tipe data array atau slice.")),(0,i.kt)("p",null,"Tuple dibuat dengan notasi penulisan diapit tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"(")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},")"),". Contoh deklarasi tuple:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_a = ("jason", 27, ["racing", "working out"], true);\nprintln!("tuple_a: {:?}", tuple_a);\n')),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"tuple_a")," di atas bertipe data tuple, dengan tipe data spesifik per-elemennya bervariasi, ada string, numerik, array ",(0,i.kt)("inlineCode",{parentName:"p"},"[&str; 2]"),", dan boolean."),(0,i.kt)("p",null,"Untuk menampilkan nilai per-elemen, gunakan notasi ",(0,i.kt)("inlineCode",{parentName:"p"},".N")," yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"N")," merupakan indeks elemen. Contohnya seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'println!("index 0: {:?}", tuple_a.0);\nprintln!("index 1: {:?}", tuple_a.1);\nprintln!("index 2: {:?} {:?}", tuple_a.2[0], tuple_a.2[1]);\nprintln!("index 3: {:?}", tuple_a.3);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"tuple",src:a(5607).Z,width:"811",height:"433"})),(0,i.kt)("p",null,"Data pada ",(0,i.kt)("inlineCode",{parentName:"p"},"tuple_a")," indeks ke-2 bertipe slice, untuk mengakses tiap elemennya bisa menggunakan notasi pengaksesan indeks array/slice seperti biasa."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Tuple ",(0,i.kt)("inlineCode",{parentName:"li"},"tuple_a.2")," nilainya ",(0,i.kt)("inlineCode",{parentName:"li"},'["racing", "working out"]')),(0,i.kt)("li",{parentName:"ul"},"Tuple ",(0,i.kt)("inlineCode",{parentName:"li"},"tuple_a.2[0]")," nilainya ",(0,i.kt)("inlineCode",{parentName:"li"},'"racing"')),(0,i.kt)("li",{parentName:"ul"},"Tuple ",(0,i.kt)("inlineCode",{parentName:"li"},"tuple_a.2[1]")," nilainya ",(0,i.kt)("inlineCode",{parentName:"li"},'"working out"'))),(0,i.kt)("h2",{id:"a152-mutable-tuple"},"A.15.2. Mutable tuple"),(0,i.kt)("p",null,"Cara untuk membuat tuple menjadi data yang mutable adalah dengan menambahkan ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," pada saat deklarasi. Pada contoh berikut variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"tuple_b")," dideklarasikan sebagai mutable dengan teknik ",(0,i.kt)("em",{parentName:"p"},"manifest typing"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut tuple_b: (&str, i32, [&str; 2], bool) = ("default", 0, [""; 2], false);\ntuple_b.0 = "damian";\ntuple_b.1 = 18;\ntuple_b.2 = ["gaming", "adventuring"];\ntuple_b.3 = true;\n\nprintln!("tuple_b: {:?}", tuple_b);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"tuple",src:a(8923).Z,width:"460",height:"123"})),(0,i.kt)("h2",{id:"a153-notasi-deklarasi-tuple"},"A.15.3. Notasi deklarasi tuple"),(0,i.kt)("p",null,"Ada beberapa cara yang bisa digunakan dalam deklarasi tuple."),(0,i.kt)("h3",{id:"-type-inference"},"\u25c9 Type inference"),(0,i.kt)("p",null,"Cara pertama menggunakan teknik ",(0,i.kt)("em",{parentName:"p"},"type inference")," seperti yang sudah dipraktikkan pada contoh pertama."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_a = ("jason", 27, ["racing", "working out"], true);\n')),(0,i.kt)("h3",{id:"-manifest-typing"},"\u25c9 Manifest typing"),(0,i.kt)("p",null,"Bisa juga menggunakan teknik deklarasi ",(0,i.kt)("em",{parentName:"p"},"manifest typing"),", biasanya diterapkan pada pembuatan mutable tuple dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value")," adalah nilai kosong seperti empty string, 0, dan lainnya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_b: (&str, i32, [&str; 2], bool) = ("damian", 18, ["gaming", "adventuring"], true);\n')),(0,i.kt)("p",null,"Contoh lain:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut tuple_b: (&str, i32, [&str; 2], bool) = ("default", 0, [""; 2], false);\ntuple_b.0 = "damian";\ntuple_b.1 = 18;\ntuple_b.2 = ["gaming", "adventuring"];\ntuple_b.3 = true;\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"tuple",src:a(6333).Z,width:"464",height:"166"})),(0,i.kt)("h3",{id:"-packing-tuple"},"\u25c9 Packing tuple"),(0,i.kt)("p",null,"Adalah cara pembuatan tuple yang mana nilai elemenya bersumber dari variabel lain."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let name = "grayson";\nlet age = 29;\nlet hobbies = ["sleeping", "parkour"];\n\nlet tuple_c = (name, age, hobbies);\n\nprintln!("name : {:?}", tuple_c.0);\nprintln!("age : {:?}", tuple_c.1);\nprintln!("hobbies : {:?}", tuple_c.2);\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Istilah packing tuple bukan resmi dari official Rust. Istilah ini penulis buat sendiri.")),(0,i.kt)("h3",{id:"-unpacking-tuple"},"\u25c9 Unpacking tuple"),(0,i.kt)("p",null,"Unpacking tuple adalah kebalikan dari packing tuple. Data tuple didistribusikan ke banyak variabel dalam 1 baris deklarasi."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_d = ("stephanie", 28, ["software engineering"], false);\nlet (name, age, hobbies, is_male) = tuple_d;\n\nprintln!("name : {:?}", name);\nprintln!("age : {:?}", age);\nprintln!("hobbies : {:?}", hobbies);\nprintln!("is_male : {:?}", is_male);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"tuple",src:a(3437).Z,width:"463",height:"166"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Pada bahasa lain teknik ini biasa disebut dengan destructuring assignment")),(0,i.kt)("h2",{id:"a154-tuple-"},"A.15.4. Tuple ",(0,i.kt)("inlineCode",{parentName:"h2"},"()")),(0,i.kt)("p",null,"Tuple bisa didefinisikan dengan tanpa isi dengan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"()"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_d = ();\nprintln!("{:?}", tuple_d);\n')),(0,i.kt)("h2",{id:"a155-tuple-structs"},"A.15.5. Tuple Structs"),(0,i.kt)("p",null,"Pembahasan mengenai tuple structs dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/struct#a247-tuple-struct"},"Struct"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/tuple"},"github.com/novalagung/dasarpemrogramanrust-example/../tuple")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.tuple.html"},"https://doc.rust-lang.org/std/primitive.tuple.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/primitives/tuples.html"},"https://doc.rust-lang.org/rust-by-example/primitives/tuples.html"))))}k.isMDXComponent=!0},5607:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/tuple-1-3234f50043ed15a37da7ad04884b7f8d.png"},8923:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},6333:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/tuple-3-3378d68b3ca9526a7b5cb148b01c8450.png"},3437:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/8333ea4d.d3688653.js b/assets/js/8333ea4d.d3688653.js deleted file mode 100644 index 4280226b..00000000 --- a/assets/js/8333ea4d.d3688653.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[9323],{3905:(e,t,a)=>{a.d(t,{Zo:()=>d,kt:()=>s});var n=a(7294);function i(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function l(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,n)}return a}function r(e){for(var t=1;t=0||(i[a]=e[a]);return i}(e,t);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(i[a]=e[a])}return i}var u=n.createContext({}),A=function(e){var t=n.useContext(u),a=t;return e&&(a="function"==typeof e?e(t):r(r({},t),e)),a},d=function(e){var t=A(e.components);return n.createElement(u.Provider,{value:t},e.children)},k={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},b=n.forwardRef((function(e,t){var a=e.components,i=e.mdxType,l=e.originalType,u=e.parentName,d=p(e,["components","mdxType","originalType","parentName"]),b=A(a),s=i,c=b["".concat(u,".").concat(s)]||b[s]||k[s]||l;return a?n.createElement(c,r(r({ref:t},d),{},{components:a})):n.createElement(c,r({ref:t},d))}));function s(e,t){var a=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var l=a.length,r=new Array(l);r[0]=b;var p={};for(var u in t)hasOwnProperty.call(t,u)&&(p[u]=t[u]);p.originalType=e,p.mdxType="string"==typeof e?e:i,r[1]=p;for(var A=2;A{a.r(t),a.d(t,{assets:()=>u,contentTitle:()=>r,default:()=>k,frontMatter:()=>l,metadata:()=>p,toc:()=>A});var n=a(7462),i=(a(7294),a(3905));const l={sidebar_position:15,title:"A.15. Tipe Data \u279c Tuple",sidebar_label:"A.15. Tuple"},r=void 0,p={unversionedId:"basic/tuple",id:"basic/tuple",title:"A.15. Tipe Data \u279c Tuple",description:"Pada chapter ini kita akan membahas tentang tipe data kolektif bernama Tuple.",source:"@site/docs/basic/tuple.md",sourceDirName:"basic",slug:"/basic/tuple",permalink:"/basic/tuple",draft:!1,tags:[],version:"current",sidebarPosition:15,frontMatter:{sidebar_position:15,title:"A.15. Tipe Data \u279c Tuple",sidebar_label:"A.15. Tuple"},sidebar:"tutorialSidebar",previous:{title:"A.14. Slice (Basic)",permalink:"/basic/slice"},next:{title:"A.16. Vector",permalink:"/basic/vector"}},u={},A=[{value:"A.15.1. Tipe data tuple",id:"a151-tipe-data-tuple",level:2},{value:"A.15.2. Mutable tuple",id:"a152-mutable-tuple",level:2},{value:"A.15.3. Notasi deklarasi tuple",id:"a153-notasi-deklarasi-tuple",level:2},{value:"\u25c9 Type inference",id:"-type-inference",level:3},{value:"\u25c9 Manifest typing",id:"-manifest-typing",level:3},{value:"\u25c9 Packing tuple",id:"-packing-tuple",level:3},{value:"\u25c9 Unpacking tuple",id:"-unpacking-tuple",level:3},{value:"A.15.4. Tuple ()",id:"a154-tuple-",level:2},{value:"A.15.5. Tuple Structs",id:"a155-tuple-structs",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:A};function k(e){let{components:t,...l}=e;return(0,i.kt)("wrapper",(0,n.Z)({},d,l,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan membahas tentang tipe data kolektif bernama Tuple."),(0,i.kt)("h2",{id:"a151-tipe-data-tuple"},"A.15.1. Tipe data tuple"),(0,i.kt)("p",null,"Tuple merupakan tipe data yang isinya koleksi dari banyak data atau value, yang bisa jadi tiap value tersebut tipe datanya berbeda satu sama lain. Tuple biasa digunakan untuk menampung data yang ",(0,i.kt)("em",{parentName:"p"},"heterogeneous")," atau campuran."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tipe ini tidak diciptakan sebagai pengganti array."),(0,i.kt)("p",{parentName:"blockquote"},"Cukup mudah untuk mengidentifikasi kapan harus menggunakan tuple, contohnya: jika ada kebutuhan data harus bisa di-iterate, atau data memiliki pattern yg pasti seperti size-nya fixed, atau tipe datanya ",(0,i.kt)("em",{parentName:"p"},"homogeneous")," atau sejenis, ... maka baiknya gunakan tipe data array atau slice.")),(0,i.kt)("p",null,"Tuple dibuat dengan notasi penulisan diapit tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"(")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},")"),". Contoh deklarasi tuple:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_a = ("jason", 27, ["racing", "working out"], true);\nprintln!("tuple_a: {:?}", tuple_a);\n')),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"tuple_a")," di atas bertipe data tuple, dengan tipe data spesifik per-elemennya bervariasi, ada string, numerik, array ",(0,i.kt)("inlineCode",{parentName:"p"},"[&str; 2]"),", dan boolean."),(0,i.kt)("p",null,"Untuk menampilkan nilai per-elemen, gunakan notasi ",(0,i.kt)("inlineCode",{parentName:"p"},".N")," dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"N")," merupakan indeks elemen. Contohnya seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'println!("index 0: {:?}", tuple_a.0);\nprintln!("index 1: {:?}", tuple_a.1);\nprintln!("index 2: {:?} {:?}", tuple_a.2[0], tuple_a.2[1]);\nprintln!("index 3: {:?}", tuple_a.3);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"tuple",src:a(5607).Z,width:"811",height:"433"})),(0,i.kt)("p",null,"Data pada ",(0,i.kt)("inlineCode",{parentName:"p"},"tuple_a")," indeks ke-2 bertipe slice, untuk mengakses tiap elemennya bisa menggunakan notasi pengaksesan indeks array/slice seperti biasa."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Tuple ",(0,i.kt)("inlineCode",{parentName:"li"},"tuple_a.2")," nilainya ",(0,i.kt)("inlineCode",{parentName:"li"},'["racing", "working out"]')),(0,i.kt)("li",{parentName:"ul"},"Tuple ",(0,i.kt)("inlineCode",{parentName:"li"},"tuple_a.2[0]")," nilainya ",(0,i.kt)("inlineCode",{parentName:"li"},'"racing"')),(0,i.kt)("li",{parentName:"ul"},"Tuple ",(0,i.kt)("inlineCode",{parentName:"li"},"tuple_a.2[1]")," nilainya ",(0,i.kt)("inlineCode",{parentName:"li"},'"working out"'))),(0,i.kt)("h2",{id:"a152-mutable-tuple"},"A.15.2. Mutable tuple"),(0,i.kt)("p",null,"Cara untuk membuat tuple menjadi data yang mutable adalah dengan menambahkan ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," pada saat deklarasi. Pada contoh berikut variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"tuple_b")," dideklarasikan sebagai mutable dengan teknik ",(0,i.kt)("em",{parentName:"p"},"manifest typing"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut tuple_b: (&str, i32, [&str; 2], bool) = ("default", 0, [""; 2], false);\ntuple_b.0 = "damian";\ntuple_b.1 = 18;\ntuple_b.2 = ["gaming", "adventuring"];\ntuple_b.3 = true;\n\nprintln!("tuple_b: {:?}", tuple_b);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"tuple",src:a(8923).Z,width:"460",height:"123"})),(0,i.kt)("h2",{id:"a153-notasi-deklarasi-tuple"},"A.15.3. Notasi deklarasi tuple"),(0,i.kt)("p",null,"Ada beberapa cara yang bisa digunakan dalam deklarasi tuple."),(0,i.kt)("h3",{id:"-type-inference"},"\u25c9 Type inference"),(0,i.kt)("p",null,"Cara pertama menggunakan teknik ",(0,i.kt)("em",{parentName:"p"},"type inference")," seperti yang sudah dipraktikkan pada contoh pertama."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_a = ("jason", 27, ["racing", "working out"], true);\n')),(0,i.kt)("h3",{id:"-manifest-typing"},"\u25c9 Manifest typing"),(0,i.kt)("p",null,"Bisa juga menggunakan teknik deklarasi ",(0,i.kt)("em",{parentName:"p"},"manifest typing"),", biasanya diterapkan pada pembuatan mutable tuple dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value")," adalah nilai kosong seperti empty string, 0, dan lainnya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_b: (&str, i32, [&str; 2], bool) = ("damian", 18, ["gaming", "adventuring"], true);\n')),(0,i.kt)("p",null,"Contoh lain:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut tuple_b: (&str, i32, [&str; 2], bool) = ("default", 0, [""; 2], false);\ntuple_b.0 = "damian";\ntuple_b.1 = 18;\ntuple_b.2 = ["gaming", "adventuring"];\ntuple_b.3 = true;\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"tuple",src:a(6333).Z,width:"464",height:"166"})),(0,i.kt)("h3",{id:"-packing-tuple"},"\u25c9 Packing tuple"),(0,i.kt)("p",null,"Adalah cara pembuatan tuple yang dimana nilai elemenya bersumber dari variabel lain."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let name = "grayson";\nlet age = 29;\nlet hobbies = ["sleeping", "parkour"];\n\nlet tuple_c = (name, age, hobbies);\n\nprintln!("name : {:?}", tuple_c.0);\nprintln!("age : {:?}", tuple_c.1);\nprintln!("hobbies : {:?}", tuple_c.2);\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Istilah packing tuple bukan resmi dari official Rust. Istilah ini penulis buat sendiri.")),(0,i.kt)("h3",{id:"-unpacking-tuple"},"\u25c9 Unpacking tuple"),(0,i.kt)("p",null,"Unpacking tuple adalah kebalikan dari packing tuple. Data tuple didistribusikan ke banyak variabel dalam 1 baris deklarasi."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_d = ("stephanie", 28, ["software engineering"], false);\nlet (name, age, hobbies, is_male) = tuple_d;\n\nprintln!("name : {:?}", name);\nprintln!("age : {:?}", age);\nprintln!("hobbies : {:?}", hobbies);\nprintln!("is_male : {:?}", is_male);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"tuple",src:a(3437).Z,width:"463",height:"166"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Pada bahasa lain teknik ini biasa disebut dengan destructuring assignment")),(0,i.kt)("h2",{id:"a154-tuple-"},"A.15.4. Tuple ",(0,i.kt)("inlineCode",{parentName:"h2"},"()")),(0,i.kt)("p",null,"Tuple bisa didefinisikan dengan tanpa isi dengan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"()"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let tuple_d = ();\nprintln!("{:?}", tuple_d);\n')),(0,i.kt)("h2",{id:"a155-tuple-structs"},"A.15.5. Tuple Structs"),(0,i.kt)("p",null,"Pembahasan mengenai tuple structs dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/struct#a247-tuple-struct"},"Struct"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/tuple"},"github.com/novalagung/dasarpemrogramanrust-example/../tuple")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.tuple.html"},"https://doc.rust-lang.org/std/primitive.tuple.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/primitives/tuples.html"},"https://doc.rust-lang.org/rust-by-example/primitives/tuples.html"))))}k.isMDXComponent=!0},5607:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/tuple-1-3234f50043ed15a37da7ad04884b7f8d.png"},8923:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},6333:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/tuple-3-3378d68b3ca9526a7b5cb148b01c8450.png"},3437:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/87499f15.3255091a.js b/assets/js/87499f15.3255091a.js new file mode 100644 index 00000000..e84c3400 --- /dev/null +++ b/assets/js/87499f15.3255091a.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7742],{3905:(a,e,n)=>{n.d(e,{Zo:()=>s,kt:()=>k});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function r(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var p=i.createContext({}),d=function(a){var e=i.useContext(p),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},s=function(a){var e=d(a.components);return i.createElement(p.Provider,{value:e},a.children)},u={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},A=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,l=a.originalType,p=a.parentName,s=m(a,["components","mdxType","originalType","parentName"]),A=d(n),k=t,g=A["".concat(p,".").concat(k)]||A[k]||u[k]||l;return n?i.createElement(g,r(r({ref:e},s),{},{components:n})):i.createElement(g,r({ref:e},s))}));function k(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var l=n.length,r=new Array(l);r[0]=A;var m={};for(var p in e)hasOwnProperty.call(e,p)&&(m[p]=e[p]);m.originalType=a,m.mdxType="string"==typeof a?a:t,r[1]=m;for(var d=2;d{n.r(e),n.d(e,{assets:()=>p,contentTitle:()=>r,default:()=>u,frontMatter:()=>l,metadata:()=>m,toc:()=>d});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:4,title:"A.4. Variabel",sidebar_label:"A.4. Variabel"},r=void 0,m={unversionedId:"basic/variabel",id:"basic/variabel",title:"A.4. Variabel",description:"Rust merupakan bahasa yang kalau dikategorikan berdasarkan type system-nya adalah statically typed, yang artinya semua tipe data harus diketahui saat kompilasi (compile time).",source:"@site/docs/basic/variabel.md",sourceDirName:"basic",slug:"/basic/variabel",permalink:"/basic/variabel",draft:!1,tags:[],version:"current",sidebarPosition:4,frontMatter:{sidebar_position:4,title:"A.4. Variabel",sidebar_label:"A.4. Variabel"},sidebar:"tutorialSidebar",previous:{title:"A.3. Komentar",permalink:"/basic/komentar"},next:{title:"A.5. Tipe Data \u279c Primitive Scalar",permalink:"/basic/tipe-data-primitive-scalar"}},p={},d=[{value:"A.4.1. Deklarasi variabel menggunakan let",id:"a41-deklarasi-variabel-menggunakan-let",level:2},{value:"\u25c9 Aturan penamaan variabel (naming convention)",id:"-aturan-penamaan-variabel-naming-convention",level:3},{value:"\u25c9 Macro println untuk print string",id:"-macro-println-untuk-print-string",level:3},{value:"A.4.2. Immutability pada variabel",id:"a42-immutability-pada-variabel",level:2},{value:"A.4.3. Keyword mut",id:"a43-keyword-mut",level:2},{value:"\u25c9 Argument parameter macro println",id:"-argument-parameter-macro-println",level:3},{value:"A.4.4. Type Inference vs Manifest Typing",id:"a44-type-inference-vs-manifest-typing",level:2},{value:"\u25c9 Metode Type Inference",id:"-metode-type-inference",level:3},{value:"\u25c9 Metode Manifest Typing",id:"-metode-manifest-typing",level:3},{value:"A.4.5. Deklarasi variabel tanpa predefined value",id:"a45-deklarasi-variabel-tanpa-predefined-value",level:2},{value:"A.4.6. Deklarasi banyak variabel dalam satu statement",id:"a46-deklarasi-banyak-variabel-dalam-satu-statement",level:2},{value:"A.4.7. Deklarasi variabel dengan tipe data ditentukan dari value",id:"a47-deklarasi-variabel-dengan-tipe-data-ditentukan-dari-value",level:2},{value:"A.4.8. Variable Shadowing",id:"a48-variable-shadowing",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],s={toc:d};function u(a){let{components:e,...l}=a;return(0,t.kt)("wrapper",(0,i.Z)({},s,l,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Rust merupakan bahasa yang kalau dikategorikan berdasarkan ",(0,t.kt)("em",{parentName:"p"},"type system"),"-nya adalah ",(0,t.kt)("em",{parentName:"p"},"statically typed"),", yang artinya semua tipe data harus diketahui saat kompilasi (compile time)."),(0,t.kt)("p",null,"Pada chapter ini kita akan belajar mengenai variabel pada Rust, cara deklarasinya, dan juga mengenai ",(0,t.kt)("em",{parentName:"p"},"immutability")," variabel."),(0,t.kt)("h2",{id:"a41-deklarasi-variabel-menggunakan-let"},"A.4.1. Deklarasi variabel menggunakan ",(0,t.kt)("inlineCode",{parentName:"h2"},"let")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"let")," digunakan untuk deklarasi variabel. Notasinya kurang lebih seperti berikut. Di contoh ini sebuah variabel bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_variabel")," didefinisikan dan diisi nilainya dengan sebuah string ",(0,t.kt)("inlineCode",{parentName:"p"},'"predefined value"'),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let nama_variabel = "predefined value";\n println!(nama_variabel);\n}\n')),(0,t.kt)("p",null,"Ok, sekarang coba jalankan, dan lanjut ke pembahasan di bawah."),(0,t.kt)("h3",{id:"-aturan-penamaan-variabel-naming-convention"},"\u25c9 Aturan penamaan variabel (naming convention)"),(0,t.kt)("p",null,(0,t.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},(0,t.kt)("em",{parentName:"a"},"Naming convention"))," variabel di Rust adalah ",(0,t.kt)("strong",{parentName:"p"},"snake case"),". Nama variabel dituliskan dalam huruf kecil dengan separator tanda garis bawah atau underscore (",(0,t.kt)("inlineCode",{parentName:"p"},"_"),"). Contohnya seperti: ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_variabel"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"hasil_operasi_pembagian"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"kota_kelahiran"),", dll."),(0,t.kt)("h3",{id:"-macro-println-untuk-print-string"},"\u25c9 Macro ",(0,t.kt)("inlineCode",{parentName:"h3"},"println")," untuk print string"),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"variabel rust",src:n(7281).Z,width:"455",height:"228"})),(0,t.kt)("p",null,"Error? kok bisa?"),(0,t.kt)("p",null,"Perlu diketahui bahwa pada penggunaan macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),", parameter pertama wajib diisi dengan sebuah ",(0,t.kt)("strong",{parentName:"p"},"string literal"),". Contohnya seperti ",(0,t.kt)("inlineCode",{parentName:"p"},'"Hello, world!"'),". String literal yang ditampung terlebih dahulu dalam variabel, kemudian variabel-nya di print, akan menghasilkan error, contohnya seperti di atas."),(0,t.kt)("p",null,"Solusinya bagaimana? ya tetap menggunakan string literal, tapi dengan menerapkan teknik ",(0,t.kt)("strong",{parentName:"p"},"formatted print"),"."),(0,t.kt)("p",null,"Sekarang ubah kode menjadi seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let nama_variabel = "predefined value";\nprintln!("{}", nama_variabel);\n')),(0,t.kt)("p",null,"Bisa dilihat argument pertama statement macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," adalah string literal ",(0,t.kt)("inlineCode",{parentName:"p"},'"{}"'),". String ",(0,t.kt)("inlineCode",{parentName:"p"},"{}")," pada macro tersebut akan di-replace dengan isi argument setelahnya, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_variabel"),"."),(0,t.kt)("p",null,"Jika dijalankan, maka hasilnya tidak ada error dan menampilkan output sesuai harapan."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"string literal println",src:n(3283).Z,width:"473",height:"137"})),(0,t.kt)("p",null,"Ok sekarang kembali ke topik, yaitu perihal variabel."),(0,t.kt)("h2",{id:"a42-immutability-pada-variabel"},"A.4.2. ",(0,t.kt)("em",{parentName:"h2"},"Immutability")," pada variabel"),(0,t.kt)("p",null,"Tulis kode berikut. Ada 2 variabel yang dideklarasikan, ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number")," yang isinya numerik, dan ",(0,t.kt)("inlineCode",{parentName:"p"},"message1")," yang isinya string."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let message_number = 1;\nlet message1 = "hello";\nprintln!("message number {}: {}", message_number, message1);\n')),(0,t.kt)("p",null,"Jalankan, hasilnya tidak ada error, aman."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"var immutability",src:n(5394).Z,width:"467",height:"119"})),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"O iya, jika ingin menampilkan banyak variabel via macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),", manfaatkan string ",(0,t.kt)("inlineCode",{parentName:"p"},"{}"),"."),(0,t.kt)("p",{parentName:"blockquote"},"Tulis ",(0,t.kt)("inlineCode",{parentName:"p"},"{}")," sejumlah variabel yang ingin ditampilkan, pada output string ",(0,t.kt)("inlineCode",{parentName:"p"},"{}")," akan di-replace oleh variabel yang disisipkan pada argument saat pemanggilan macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," secara berurutan."),(0,t.kt)("p",{parentName:"blockquote"},"Pada contoh di atas, string ",(0,t.kt)("inlineCode",{parentName:"p"},'"message number {}: {}"'),(0,t.kt)("br",null)," ... akan menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},'"message number {message_number}: {message1}"'),(0,t.kt)("br",null)," ... kemudian menghasilkan ",(0,t.kt)("inlineCode",{parentName:"p"},'"message number 1: hello"'))),(0,t.kt)("p",null,"Sekarang ubah isi ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number")," menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"2")," lalu siapkan juga variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message2"),", kurang lebih kodenya menjadi seperti berikut, kemudian jalankan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let message_number = 1;\nlet message1 = "hello";\nprintln!("message number {}: {}", message_number, message1);\n\nmessage_number = 2;\nlet message2 = "world";\nprintln!("message number {}: {}", message_number, message2);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"var immutability",src:n(6204).Z,width:"585",height:"308"})),(0,t.kt)("p",null,"Error? kok bisa?"),(0,t.kt)("p",null,"Perlu diketahui, bahwa ",(0,t.kt)("em",{parentName:"p"},"by default")," semua variabel adalah ",(0,t.kt)("em",{parentName:"p"},"immutable"),". Immutable itu artinya gak bisa diubah nilai/value-nya. Jadi mirip seperti konstanta."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Variabel immutable mirip seperti konstanta, tapi sebenarnya berbeda. Akan kita bahas lebih detail di beberapa chapter berbeda perihal perbedaannya.")),(0,t.kt)("p",null,"Lalu bagaimana cara agar nilai variabel bisa diubah? Caranya dengan menambahkan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut")," (yang merupakan kependekan dari ",(0,t.kt)("em",{parentName:"p"},"mutable"),") saat pendefinisian variabel tersebut."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Catatan tambahan:"),(0,t.kt)("ul",{parentName:"blockquote"},(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Operasi assignment pada variabel immutable hanya boleh dilakukan sekali. Dan pada kode di atas telah dicontohkan caranya bagaimana. Variabel dideklarasikan sekaligus diisi nilainya.")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Perlu diketahui bahwa sebenarnya diperbolehkan untuk memecah statement deklarasi variable dan assignment nilai sebagai 2 statement terpisah. Lebih jelasnya dibahas setelah ini pada section ",(0,t.kt)("a",{parentName:"p",href:"/basic/variabel#a45-deklarasi-variabel-tanpa-predefined-value"},"section A.4.5"),".")))),(0,t.kt)("h2",{id:"a43-keyword-mut"},"A.4.3. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"mut")),(0,t.kt)("p",null,"Ok, mari kita coba, ubah statement deklarasi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number"),", tambahkan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut"),". Lalu jalankan ulang program."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut message_number = 1;\nlet message1 = "hello";\nprintln!("message number {}: {}", message_number, message1);\n\nmessage_number = 2;\nlet message2 = "world";\nprintln!("message number {}: {}", message_number, message2);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"mut keyword",src:n(3264).Z,width:"470",height:"132"})),(0,t.kt)("p",null,"Ok, sekarang tidak muncul error."),(0,t.kt)("p",null,"Dengan menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"let mut")," pada pendefinisian ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number"),", membuat variabel tersebut menjadi mutable atau bisa diubah nilainya."),(0,t.kt)("h3",{id:"-argument-parameter-macro-println"},"\u25c9 Argument parameter macro ",(0,t.kt)("inlineCode",{parentName:"h3"},"println")),(0,t.kt)("p",null,"Selanjutnya, coba tambahkan lagi 1 message baru pada program yang sudah dibuat, lalu run."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut message_number = 1;\nlet message1 = "hello";\nprintln!("message number {}: {}", message_number, message1);\n\nmessage_number = 2;\nlet message2 = "world";\nprintln!("message number {}: {}", message_number, message2);\n\nmessage_number = 3;\nlet message3: i8 = 24;\nprintln!("message number {1}: {0}", message3, message_number);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"formatted print macro `println`",src:n(4247).Z,width:"470",height:"154"})),(0,t.kt)("p",null,"Jika dilihat ada yg berbeda pada cara deklarasi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message3")," dan juga pada statement ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," untuk ",(0,t.kt)("inlineCode",{parentName:"p"},"message3")," yang di situ digunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"{1}")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"{0}"),", tidak seperti sebelumnya yg menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"{}"),". Kita akan bahas yg ke-2 terlebih dahulu."),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Jika menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"{}"),", maka string akan di-replace sesuai urutan argument pada pemanggilan ",(0,t.kt)("inlineCode",{parentName:"li"},"println"),"."),(0,t.kt)("li",{parentName:"ul"},"Jika menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"{0}"),", maka string akan di-replace dengan data pada argument ke ",(0,t.kt)("inlineCode",{parentName:"li"},"1")," pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"println"),", yang pada contoh di atas adalah ",(0,t.kt)("inlineCode",{parentName:"li"},"message3"),"."),(0,t.kt)("li",{parentName:"ul"},"Jika menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"{1}"),", maka string akan di-replace dengan data pada argument ke ",(0,t.kt)("inlineCode",{parentName:"li"},"2")," pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"println"),", yang pada contoh di atas adalah ",(0,t.kt)("inlineCode",{parentName:"li"},"message_number"),"."),(0,t.kt)("li",{parentName:"ul"},"Jika menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"{n}"),", maka string akan di-replace dengan data pada argument ke ",(0,t.kt)("inlineCode",{parentName:"li"},"n+1")," pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"println"),".")),(0,t.kt)("p",null,"Dengan ini maka 3 statement berbeda berikut akan menghasilkan output yang sama:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'println!("message number {}: {}", message_number, message3);\nprintln!("message number {0}: {1}", message_number, message3);\nprintln!("message number {1}: {0}", message3, message_number);\n')),(0,t.kt)("p",null,"Sekarang perihal perbedaan cara deklarasi ",(0,t.kt)("inlineCode",{parentName:"p"},"message3")," akan kita bahas di bawah ini."),(0,t.kt)("h2",{id:"a44-type-inference-vs-manifest-typing"},"A.4.4. ",(0,t.kt)("em",{parentName:"h2"},"Type Inference")," vs ",(0,t.kt)("em",{parentName:"h2"},"Manifest Typing")),(0,t.kt)("p",null,"Rust mendukung dua metode deklarasi variabel, yaitu ",(0,t.kt)("em",{parentName:"p"},"type inference")," dan ",(0,t.kt)("em",{parentName:"p"},"manifest typing"),"."),(0,t.kt)("h3",{id:"-metode-type-inference"},"\u25c9 Metode ",(0,t.kt)("em",{parentName:"h3"},"Type Inference")),(0,t.kt)("p",null,"Penulisan variabel dengan metode ini ditandai dengan tidak menuliskan tipe data secara jelas/eksplisit. Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let var1 = "hello" // compiler akan secara cerdas mendeteksi var1 tipe data nya string\nlet var2 = 12 // compiler akan secara cerdas mendeteksi var2 tipe data nya numerik\n')),(0,t.kt)("p",null,"Metode deklarasi yang selama ini sudah kita terapkan adalah ",(0,t.kt)("em",{parentName:"p"},"type inference"),"."),(0,t.kt)("h3",{id:"-metode-manifest-typing"},"\u25c9 Metode ",(0,t.kt)("em",{parentName:"h3"},"Manifest Typing")),(0,t.kt)("p",null,"Metode ini mewajibkan programmer untuk menuliskan secara jelas/eksplisit tipe data variabel. Contoh penerapannya bisa dilihat di praktik sebelumnya, pada definisi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message3"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"let message3: i8 = 24;\n")),(0,t.kt)("p",null,"Notasi penulisan tipe data adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"namavariabel: tipedata"),". Contoh seperti di atas, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"let message3: i8 = 24")," artinya variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message3")," didefinisikan memiliki tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"i8"),"."),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"i8")," merupakan salah satu tipe data dari sekian banyak tipe data yang ada di Rust. Lebih jelasnya kita akan bahas pada chapter selanjutnya."),(0,t.kt)("h2",{id:"a45-deklarasi-variabel-tanpa-predefined-value"},"A.4.5. Deklarasi variabel tanpa ",(0,t.kt)("em",{parentName:"h2"},"predefined value")),(0,t.kt)("p",null,"Sesuai dengan penjelasan di ",(0,t.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/error-index.html#E0381"},"dokumentasi spesifikasi Rust"),", variable yang tidak ada value-nya akan memunculkan error saat program di-run."),(0,t.kt)("p",null,"Meski demikian, diperbolehkan untuk memisahkan variable statement untuk deklarasi variable dan pengisian value. Sebagai contoh, statement ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number")," pada kode berikut ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let message_number = 1;\nprintln!("message number {}", message_number);\n')),(0,t.kt)("p",null,"... boleh dituliskan dalam 2 statement terpisah seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let message_number: i32;\nmessage_number = 1;\nprintln!("message number {}", message_number);\n')),(0,t.kt)("p",null,"Intinya, operasi assignment hanya diperbolehkan 1x saja pada variabel ",(0,t.kt)("em",{parentName:"p"},"immutable"),", baik itu saat deklarasi variable atau setelahnya."),(0,t.kt)("h2",{id:"a46-deklarasi-banyak-variabel-dalam-satu-statement"},"A.4.6. Deklarasi banyak variabel dalam satu statement"),(0,t.kt)("p",null,"Di Rust memungkinkan untuk mendefinisikan banyak variabel dalam 1 baris statement. Notasi penulisannya seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let (var1, var2) = (24, "hello");\nprintln!("var1: {0}", var1); // hasilnya => var1: 24\nprintln!("var2: {0}", var2); // hasilnya => var2: hello\n')),(0,t.kt)("p",null,"Pendefinisian banyak variabel dalam 1 statement dilakukan dengan menuliskan semua variabelnya dengan separator tanda ",(0,t.kt)("inlineCode",{parentName:"p"},",")," dan diapit tanda kurung ",(0,t.kt)("inlineCode",{parentName:"p"},"()"),"."),(0,t.kt)("p",null,"Sebagai contoh di atas 2 variabel didefinisikan, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"var1")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"var2"),", dan keduanya memiliki value yang berbeda tipe-datanya."),(0,t.kt)("p",null,"Bisa juga saat definisi variabel sekaligus ditentukan tipe data variabel, notasi penulisan bisa dilihat pada contoh kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let (var3, var4): (i8, i8) = (32, 12);\nprintln!("var3: {0}", var3); // hasilnya => var3: 32\nprintln!("var4: {0}", var4); // hasilnya => var4: 12\n')),(0,t.kt)("p",null,"Dengan notasi ini bisa juga untuk definisi variabel ",(0,t.kt)("em",{parentName:"p"},"mutable"),". Cukup tambahkan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut")," pada variabel yang ingin bisa diubah nilainya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let (var5, mut var6, var7): (i8, i8, i8) = (64, 12, 4);\nprintln!("var5: {0}", var5); // hasilnya => var5: 64\nprintln!("var6: {0}", var6); // hasilnya => var6: 12\nvar6 = 24;\nprintln!("var6: {0}", var6); // hasilnya => var6: 24\nprintln!("var7: {0}", var7); // hasilnya => var7: 4\n')),(0,t.kt)("h2",{id:"a47-deklarasi-variabel-dengan-tipe-data-ditentukan-dari-value"},"A.4.7. Deklarasi variabel dengan tipe data ditentukan dari value"),(0,t.kt)("p",null,"Ini merupakan salah satu alternatif cara penulisan untuk men-specify tipe data variabel, caranya dengan menuliskan tipe data tepat setelah value, contohnya seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let data1 = 24i8;\nprintln!("data1: {0}", data1); // hasilnya => data1: 24\n')),(0,t.kt)("p",null,"Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data")," didefinisikan dengan predefined value adalah numerik ",(0,t.kt)("inlineCode",{parentName:"p"},"24")," dengan tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"i8"),", penulisan value nya menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"24i8"),"."),(0,t.kt)("p",null,"Boleh juga menggunakan separator ",(0,t.kt)("inlineCode",{parentName:"p"},"_")," dalam penulisan value-nya sebagai pembatas antara nilai dan tipe data, contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let data1 = 24_i8;\nprintln!("data1: {0}", data1); // hasilnya => data1: 24\n')),(0,t.kt)("h2",{id:"a48-variable-shadowing"},"A.4.8. Variable ",(0,t.kt)("em",{parentName:"h2"},"Shadowing")),(0,t.kt)("p",null,"Di Rust ada konsep bernama ",(0,t.kt)("strong",{parentName:"p"},"variable shadowing"),". Shadowing sendiri adalah pendefinisan ulang variabel yang sebelumnya sudah didefinisikan. Biasanya teknik ini dipakai untuk isolasi variabel dalam sebuah blok kode."),(0,t.kt)("p",null,"Contoh shadowing bisa dilihat pada kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let x = 5;\nprintln!("x: {}", x); // hasilnya => x: 5\n\nlet x = x + 1;\nprintln!("x: {}", x); // hasilnya => x: 6\n')),(0,t.kt)("p",null,"Lebih jelasnya akan kita bahas pada chapter terpisah, yaitu ",(0,t.kt)("a",{parentName:"p",href:"/basic/shadowing"},"Shadowing"),"."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"_"))),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/variabel"},"github.com/novalagung/dasarpemrogramanrust-example/../variabel")),(0,t.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/shadowing"},"Shadowing")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"#/wip/macro"},"Macro")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"#/wip/formatted-print"},"Formatted Print"))),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-01-variables-and-mutability.html"},"https://doc.rust-lang.org/book/ch03-01-variables-and-mutability.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/hello/print.html"},"https://doc.rust-lang.org/rust-by-example/hello/print.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/variable_bindings/scope.html"},"https://doc.rust-lang.org/rust-by-example/variable_bindings/scope.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/variables.html"},"https://doc.rust-lang.org/reference/variables.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"https://rust-lang.github.io/api-guidelines/naming.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/38045700/how-do-i-declare-multiple-mutable-variables-at-the-same-time"},"https://stackoverflow.com/questions/38045700/how-do-i-declare-multiple-mutable-variables-at-the-same-time"))))}u.isMDXComponent=!0},7281:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/variabel-1-c9d8201d5914e2a0da75b23e92fa5bb9.png"},3283:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},5394:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},6204:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/variabel-4-11c9b8e77e4ddf88a314c0b480337a42.png"},3264:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},4247:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/87499f15.679ef493.js b/assets/js/87499f15.679ef493.js deleted file mode 100644 index 8bb3f26c..00000000 --- a/assets/js/87499f15.679ef493.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7742],{3905:(a,e,n)=>{n.d(e,{Zo:()=>s,kt:()=>k});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function r(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var p=i.createContext({}),d=function(a){var e=i.useContext(p),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},s=function(a){var e=d(a.components);return i.createElement(p.Provider,{value:e},a.children)},u={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},A=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,l=a.originalType,p=a.parentName,s=m(a,["components","mdxType","originalType","parentName"]),A=d(n),k=t,g=A["".concat(p,".").concat(k)]||A[k]||u[k]||l;return n?i.createElement(g,r(r({ref:e},s),{},{components:n})):i.createElement(g,r({ref:e},s))}));function k(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var l=n.length,r=new Array(l);r[0]=A;var m={};for(var p in e)hasOwnProperty.call(e,p)&&(m[p]=e[p]);m.originalType=a,m.mdxType="string"==typeof a?a:t,r[1]=m;for(var d=2;d{n.r(e),n.d(e,{assets:()=>p,contentTitle:()=>r,default:()=>u,frontMatter:()=>l,metadata:()=>m,toc:()=>d});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:4,title:"A.4. Variabel",sidebar_label:"A.4. Variabel"},r=void 0,m={unversionedId:"basic/variabel",id:"basic/variabel",title:"A.4. Variabel",description:"Rust merupakan bahasa yang kalau dikategorikan berdasarkan type system-nya adalah statically typed, yang artinya semua tipe data harus diketahui saat kompilasi (compile time).",source:"@site/docs/basic/variabel.md",sourceDirName:"basic",slug:"/basic/variabel",permalink:"/basic/variabel",draft:!1,tags:[],version:"current",sidebarPosition:4,frontMatter:{sidebar_position:4,title:"A.4. Variabel",sidebar_label:"A.4. Variabel"},sidebar:"tutorialSidebar",previous:{title:"A.3. Komentar",permalink:"/basic/komentar"},next:{title:"A.5. Tipe Data \u279c Primitive Scalar",permalink:"/basic/tipe-data-primitive-scalar"}},p={},d=[{value:"A.4.1. Deklarasi variabel menggunakan let",id:"a41-deklarasi-variabel-menggunakan-let",level:2},{value:"\u25c9 Aturan penamaan variabel (naming convention)",id:"-aturan-penamaan-variabel-naming-convention",level:3},{value:"\u25c9 Macro println untuk print string",id:"-macro-println-untuk-print-string",level:3},{value:"A.4.2. Immutability pada variabel",id:"a42-immutability-pada-variabel",level:2},{value:"A.4.3. Keyword mut",id:"a43-keyword-mut",level:2},{value:"\u25c9 Argument parameter macro println",id:"-argument-parameter-macro-println",level:3},{value:"A.4.4. Type Inference vs Manifest Typing",id:"a44-type-inference-vs-manifest-typing",level:2},{value:"\u25c9 Metode Type Inference",id:"-metode-type-inference",level:3},{value:"\u25c9 Metode Manifest Typing",id:"-metode-manifest-typing",level:3},{value:"A.4.5. Deklarasi variabel tanpa predefined value",id:"a45-deklarasi-variabel-tanpa-predefined-value",level:2},{value:"A.4.6. Deklarasi banyak variabel dalam satu statement",id:"a46-deklarasi-banyak-variabel-dalam-satu-statement",level:2},{value:"A.4.7. Deklarasi variabel dengan tipe data ditentukan dari value",id:"a47-deklarasi-variabel-dengan-tipe-data-ditentukan-dari-value",level:2},{value:"A.4.8. Variable Shadowing",id:"a48-variable-shadowing",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],s={toc:d};function u(a){let{components:e,...l}=a;return(0,t.kt)("wrapper",(0,i.Z)({},s,l,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Rust merupakan bahasa yang kalau dikategorikan berdasarkan ",(0,t.kt)("em",{parentName:"p"},"type system"),"-nya adalah ",(0,t.kt)("em",{parentName:"p"},"statically typed"),", yang artinya semua tipe data harus diketahui saat kompilasi (compile time)."),(0,t.kt)("p",null,"Pada chapter ini kita akan belajar mengenai variabel pada Rust, cara deklarasinya, dan juga mengenai ",(0,t.kt)("em",{parentName:"p"},"immutability")," variabel."),(0,t.kt)("h2",{id:"a41-deklarasi-variabel-menggunakan-let"},"A.4.1. Deklarasi variabel menggunakan ",(0,t.kt)("inlineCode",{parentName:"h2"},"let")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"let")," digunakan untuk deklarasi variabel. Notasinya kurang lebih seperti berikut. Di contoh ini sebuah variabel bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_variabel")," didefinisikan dan diisi nilainya dengan sebuah string ",(0,t.kt)("inlineCode",{parentName:"p"},'"predefined value"'),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let nama_variabel = "predefined value";\n println!(nama_variabel);\n}\n')),(0,t.kt)("p",null,"Ok, sekarang coba jalankan, dan lanjut ke pembahasan dibawah."),(0,t.kt)("h3",{id:"-aturan-penamaan-variabel-naming-convention"},"\u25c9 Aturan penamaan variabel (naming convention)"),(0,t.kt)("p",null,(0,t.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},(0,t.kt)("em",{parentName:"a"},"Naming convention"))," variabel di Rust adalah ",(0,t.kt)("strong",{parentName:"p"},"snake case"),". Nama variabel dituliskan dalam huruf kecil dengan separator tanda garis bawah atau underscore (",(0,t.kt)("inlineCode",{parentName:"p"},"_"),"). Contohnya seperti: ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_variabel"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"hasil_operasi_pembagian"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"kota_kelahiran"),", dll."),(0,t.kt)("h3",{id:"-macro-println-untuk-print-string"},"\u25c9 Macro ",(0,t.kt)("inlineCode",{parentName:"h3"},"println")," untuk print string"),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"variabel rust",src:n(7281).Z,width:"455",height:"228"})),(0,t.kt)("p",null,"Error? kok bisa?"),(0,t.kt)("p",null,"Perlu diketahui bahwa pada penggunaan macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),", parameter pertama wajib diisi dengan sebuah ",(0,t.kt)("strong",{parentName:"p"},"string literal"),". Contohnya seperti ",(0,t.kt)("inlineCode",{parentName:"p"},'"Hello, world!"'),". String literal yang ditampung terlebih dahulu dalam variabel, kemudian variabel-nya di print, akan menghasilkan error, contohnya seperti di atas."),(0,t.kt)("p",null,"Solusinya bagaimana? ya tetap menggunakan string literal, tapi dengan menerapkan teknik ",(0,t.kt)("strong",{parentName:"p"},"formatted print"),"."),(0,t.kt)("p",null,"Sekarang ubah kode menjadi seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let nama_variabel = "predefined value";\nprintln!("{}", nama_variabel);\n')),(0,t.kt)("p",null,"Bisa dilihat argument pertama statement macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," adalah string literal ",(0,t.kt)("inlineCode",{parentName:"p"},'"{}"'),". String ",(0,t.kt)("inlineCode",{parentName:"p"},"{}")," pada macro tersebut akan di-replace dengan isi argument setelahnya, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"nama_variabel"),"."),(0,t.kt)("p",null,"Jika dijalankan, maka hasilnya tidak ada error dan menampilkan output sesuai harapan."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"string literal println",src:n(3283).Z,width:"473",height:"137"})),(0,t.kt)("p",null,"Ok sekarang kembali ke topik, yaitu perihal variabel."),(0,t.kt)("h2",{id:"a42-immutability-pada-variabel"},"A.4.2. ",(0,t.kt)("em",{parentName:"h2"},"Immutability")," pada variabel"),(0,t.kt)("p",null,"Tulis kode berikut. Ada 2 variabel yang dideklarasikan, ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number")," yang isinya numerik, dan ",(0,t.kt)("inlineCode",{parentName:"p"},"message1")," yang isinya string."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let message_number = 1;\nlet message1 = "hello";\nprintln!("message number {}: {}", message_number, message1);\n')),(0,t.kt)("p",null,"Jalankan, hasilnya tidak ada error, aman."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"var immutability",src:n(5394).Z,width:"467",height:"119"})),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"O iya, jika ingin menampilkan banyak variabel via macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println"),", manfaatkan string ",(0,t.kt)("inlineCode",{parentName:"p"},"{}"),"."),(0,t.kt)("p",{parentName:"blockquote"},"Tulis ",(0,t.kt)("inlineCode",{parentName:"p"},"{}")," sejumlah variabel yang ingin ditampilkan, pada output string ",(0,t.kt)("inlineCode",{parentName:"p"},"{}")," akan di-replace oleh variabel yang disisipkan pada argument saat pemanggilan macro ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," secara berurutan."),(0,t.kt)("p",{parentName:"blockquote"},"Pada contoh di atas, string ",(0,t.kt)("inlineCode",{parentName:"p"},'"message number {}: {}"'),(0,t.kt)("br",null)," ... akan menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},'"message number {message_number}: {message1}"'),(0,t.kt)("br",null)," ... kemudian menghasilkan ",(0,t.kt)("inlineCode",{parentName:"p"},'"message number 1: hello"'))),(0,t.kt)("p",null,"Sekarang ubah isi ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number")," menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"2")," lalu siapkan juga variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message2"),", kurang lebih kodenya menjadi seperti berikut, kemudian jalankan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let message_number = 1;\nlet message1 = "hello";\nprintln!("message number {}: {}", message_number, message1);\n\nmessage_number = 2;\nlet message2 = "world";\nprintln!("message number {}: {}", message_number, message2);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"var immutability",src:n(6204).Z,width:"585",height:"308"})),(0,t.kt)("p",null,"Error? kok bisa?"),(0,t.kt)("p",null,"Perlu diketahui, bahwa ",(0,t.kt)("em",{parentName:"p"},"by default")," semua variabel adalah ",(0,t.kt)("em",{parentName:"p"},"immutable"),". Immutable itu artinya gak bisa diubah nilai/value-nya. Jadi mirip seperti konstanta."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Variabel immutable mirip seperti konstanta, tapi sebenarnya berbeda. Akan kita bahas lebih detail di beberapa chapter berbeda perihal perbedaannya.")),(0,t.kt)("p",null,"Lalu bagaimana cara agar nilai variabel bisa diubah? Caranya dengan menambahkan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut")," (yang merupakan kependekan dari ",(0,t.kt)("em",{parentName:"p"},"mutable"),") saat pendefinisian variabel tersebut."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Catatan tambahan:"),(0,t.kt)("ul",{parentName:"blockquote"},(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Operasi assignment pada variabel immutable hanya boleh dilakukan sekali. Dan pada kode di atas telah dicontohkan caranya bagaimana. Variabel dideklarasikan sekaligus diisi nilainya.")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("p",{parentName:"li"},"Perlu diketahui bahwa sebenarnya diperbolehkan untuk memecah statement deklarasi variable dan assignment nilai sebagai 2 statement terpisah. Lebih jelasnya dibahas setelah ini pada section ",(0,t.kt)("a",{parentName:"p",href:"/basic/variabel#a45-deklarasi-variabel-tanpa-predefined-value"},"section A.4.5"),".")))),(0,t.kt)("h2",{id:"a43-keyword-mut"},"A.4.3. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"mut")),(0,t.kt)("p",null,"Ok, mari kita coba, ubah statement deklarasi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number"),", tambahkan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut"),". Lalu jalankan ulang program."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut message_number = 1;\nlet message1 = "hello";\nprintln!("message number {}: {}", message_number, message1);\n\nmessage_number = 2;\nlet message2 = "world";\nprintln!("message number {}: {}", message_number, message2);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"mut keyword",src:n(3264).Z,width:"470",height:"132"})),(0,t.kt)("p",null,"Ok, sekarang tidak muncul error."),(0,t.kt)("p",null,"Dengan menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"let mut")," pada pendefinisian ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number"),", membuat variabel tersebut menjadi mutable atau bisa diubah nilainya."),(0,t.kt)("h3",{id:"-argument-parameter-macro-println"},"\u25c9 Argument parameter macro ",(0,t.kt)("inlineCode",{parentName:"h3"},"println")),(0,t.kt)("p",null,"Selanjutnya, coba tambahkan lagi 1 message baru pada program yang sudah dibuat, lalu run."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut message_number = 1;\nlet message1 = "hello";\nprintln!("message number {}: {}", message_number, message1);\n\nmessage_number = 2;\nlet message2 = "world";\nprintln!("message number {}: {}", message_number, message2);\n\nmessage_number = 3;\nlet message3: i8 = 24;\nprintln!("message number {1}: {0}", message3, message_number);\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"formatted print macro `println`",src:n(4247).Z,width:"470",height:"154"})),(0,t.kt)("p",null,"Jika dilihat ada yg berbeda pada cara deklarasi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message3")," dan juga pada statement ",(0,t.kt)("inlineCode",{parentName:"p"},"println")," untuk ",(0,t.kt)("inlineCode",{parentName:"p"},"message3")," yang disitu digunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"{1}")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"{0}"),", tidak seperti sebelumnya yg menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"{}"),". Kita akan bahas yg ke-2 terlebih dahulu."),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Jika menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"{}"),", maka string akan di-replace sesuai urutan argument pada pemanggilan ",(0,t.kt)("inlineCode",{parentName:"li"},"println"),"."),(0,t.kt)("li",{parentName:"ul"},"Jika menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"{0}"),", maka string akan di-replace dengan data pada argument ke ",(0,t.kt)("inlineCode",{parentName:"li"},"1")," pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"println"),", yang pada contoh di atas adalah ",(0,t.kt)("inlineCode",{parentName:"li"},"message3"),"."),(0,t.kt)("li",{parentName:"ul"},"Jika menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"{1}"),", maka string akan di-replace dengan data pada argument ke ",(0,t.kt)("inlineCode",{parentName:"li"},"2")," pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"println"),", yang pada contoh di atas adalah ",(0,t.kt)("inlineCode",{parentName:"li"},"message_number"),"."),(0,t.kt)("li",{parentName:"ul"},"Jika menggunakan ",(0,t.kt)("inlineCode",{parentName:"li"},"{n}"),", maka string akan di-replace dengan data pada argument ke ",(0,t.kt)("inlineCode",{parentName:"li"},"n+1")," pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"println"),".")),(0,t.kt)("p",null,"Dengan ini maka 3 statement berbeda berikut akan menghasilkan output yang sama:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'println!("message number {}: {}", message_number, message3);\nprintln!("message number {0}: {1}", message_number, message3);\nprintln!("message number {1}: {0}", message3, message_number);\n')),(0,t.kt)("p",null,"Sekarang perihal perbedaan cara deklarasi ",(0,t.kt)("inlineCode",{parentName:"p"},"message3")," akan kita bahas dibawah ini."),(0,t.kt)("h2",{id:"a44-type-inference-vs-manifest-typing"},"A.4.4. ",(0,t.kt)("em",{parentName:"h2"},"Type Inference")," vs ",(0,t.kt)("em",{parentName:"h2"},"Manifest Typing")),(0,t.kt)("p",null,"Rust mendukung dua metode deklarasi variabel, yaitu ",(0,t.kt)("em",{parentName:"p"},"type inference")," dan ",(0,t.kt)("em",{parentName:"p"},"manifest typing"),"."),(0,t.kt)("h3",{id:"-metode-type-inference"},"\u25c9 Metode ",(0,t.kt)("em",{parentName:"h3"},"Type Inference")),(0,t.kt)("p",null,"Penulisan variabel dengan metode ini ditandai dengan tidak menuliskan tipe data secara jelas/eksplisit. Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let var1 = "hello" // compiler akan secara cerdas mendeteksi var1 tipe data nya string\nlet var2 = 12 // compiler akan secara cerdas mendeteksi var2 tipe data nya numerik\n')),(0,t.kt)("p",null,"Metode deklarasi yang selama ini sudah kita terapkan adalah ",(0,t.kt)("em",{parentName:"p"},"type inference"),"."),(0,t.kt)("h3",{id:"-metode-manifest-typing"},"\u25c9 Metode ",(0,t.kt)("em",{parentName:"h3"},"Manifest Typing")),(0,t.kt)("p",null,"Metode ini mewajibkan programmer untuk menuliskan secara jelas/eksplisit tipe data variabel. Contoh penerapannya bisa dilihat di praktik sebelumnya, pada definisi variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message3"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"let message3: i8 = 24;\n")),(0,t.kt)("p",null,"Notasi penulisan tipe data adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"namavariabel: tipedata"),". Contoh seperti di atas, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"let message3: i8 = 24")," artinya variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message3")," didefinisikan memiliki tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"i8"),"."),(0,t.kt)("p",null,(0,t.kt)("inlineCode",{parentName:"p"},"i8")," merupakan salah satu tipe data dari sekian banyak tipe data yang ada di Rust. Lebih jelasnya kita akan bahas pada chapter selanjutnya."),(0,t.kt)("h2",{id:"a45-deklarasi-variabel-tanpa-predefined-value"},"A.4.5. Deklarasi variabel tanpa ",(0,t.kt)("em",{parentName:"h2"},"predefined value")),(0,t.kt)("p",null,"Sesuai dengan penjelasan di ",(0,t.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/error-index.html#E0381"},"dokumentasi spesifikasi Rust"),", variable yang tidak ada value-nya akan memunculkan error saat program di-run."),(0,t.kt)("p",null,"Meski demikian, diperbolehkan untuk memisahkan variable statement untuk deklarasi variable dan pengisian value. Sebagai contoh, statement ",(0,t.kt)("inlineCode",{parentName:"p"},"message_number")," pada kode berikut ..."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let message_number = 1;\nprintln!("message number {}", message_number);\n')),(0,t.kt)("p",null,"... boleh dituliskan dalam 2 statement terpisah seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let message_number: i32;\nmessage_number = 1;\nprintln!("message number {}", message_number);\n')),(0,t.kt)("p",null,"Intinya, operasi assignment hanya diperbolehkan 1x saja pada variabel ",(0,t.kt)("em",{parentName:"p"},"immutable"),", baik itu saat deklarasi variable atau setelahnya."),(0,t.kt)("h2",{id:"a46-deklarasi-banyak-variabel-dalam-satu-statement"},"A.4.6. Deklarasi banyak variabel dalam satu statement"),(0,t.kt)("p",null,"Di Rust memungkinkan untuk mendefinisikan banyak variabel dalam 1 baris statement. Notasi penulisannya seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let (var1, var2) = (24, "hello");\nprintln!("var1: {0}", var1); // hasilnya => var1: 24\nprintln!("var2: {0}", var2); // hasilnya => var2: hello\n')),(0,t.kt)("p",null,"Pendefinisian banyak variabel dalam 1 statement dilakukan dengan menuliskan semua variabelnya dengan separator tanda ",(0,t.kt)("inlineCode",{parentName:"p"},",")," dan diapit tanda kurung ",(0,t.kt)("inlineCode",{parentName:"p"},"()"),"."),(0,t.kt)("p",null,"Sebagai contoh di atas 2 variabel didefinisikan, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"var1")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"var2"),", dan keduanya memiliki value yang berbeda tipe-datanya."),(0,t.kt)("p",null,"Bisa juga saat definisi variabel sekaligus ditentukan tipe data variabel, notasi penulisan bisa dilihat pada contoh kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let (var3, var4): (i8, i8) = (32, 12);\nprintln!("var3: {0}", var3); // hasilnya => var3: 32\nprintln!("var4: {0}", var4); // hasilnya => var4: 12\n')),(0,t.kt)("p",null,"Dengan notasi ini bisa juga untuk definisi variabel ",(0,t.kt)("em",{parentName:"p"},"mutable"),". Cukup tambahkan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut")," pada variabel yang ingin bisa diubah nilainya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let (var5, mut var6, var7): (i8, i8, i8) = (64, 12, 4);\nprintln!("var5: {0}", var5); // hasilnya => var5: 64\nprintln!("var6: {0}", var6); // hasilnya => var6: 12\nvar6 = 24;\nprintln!("var6: {0}", var6); // hasilnya => var6: 24\nprintln!("var7: {0}", var7); // hasilnya => var7: 4\n')),(0,t.kt)("h2",{id:"a47-deklarasi-variabel-dengan-tipe-data-ditentukan-dari-value"},"A.4.7. Deklarasi variabel dengan tipe data ditentukan dari value"),(0,t.kt)("p",null,"Ini merupakan salah satu alternatif cara penulisan untuk men-specify tipe data variabel, caranya dengan menuliskan tipe data tepat setelah value, contohnya seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let data1 = 24i8;\nprintln!("data1: {0}", data1); // hasilnya => data1: 24\n')),(0,t.kt)("p",null,"Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"data")," didefinisikan dengan predefined value adalah numerik ",(0,t.kt)("inlineCode",{parentName:"p"},"24")," dengan tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"i8"),", penulisan value nya menjadi ",(0,t.kt)("inlineCode",{parentName:"p"},"24i8"),"."),(0,t.kt)("p",null,"Boleh juga menggunakan separator ",(0,t.kt)("inlineCode",{parentName:"p"},"_")," dalam penulisan value-nya sebagai pembatas antara nilai dan tipe data, contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let data1 = 24_i8;\nprintln!("data1: {0}", data1); // hasilnya => data1: 24\n')),(0,t.kt)("h2",{id:"a48-variable-shadowing"},"A.4.8. Variable ",(0,t.kt)("em",{parentName:"h2"},"Shadowing")),(0,t.kt)("p",null,"Di Rust ada konsep bernama ",(0,t.kt)("strong",{parentName:"p"},"variable shadowing"),". Shadowing sendiri adalah pendefinisan ulang variabel yang sebelumnya sudah didefinisikan. Biasanya teknik ini dipakai untuk isolasi variabel dalam sebuah blok kode."),(0,t.kt)("p",null,"Contoh shadowing bisa dilihat pada kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let x = 5;\nprintln!("x: {}", x); // hasilnya => x: 5\n\nlet x = x + 1;\nprintln!("x: {}", x); // hasilnya => x: 6\n')),(0,t.kt)("p",null,"Lebih jelasnya akan kita bahas pada chapter terpisah, yaitu ",(0,t.kt)("a",{parentName:"p",href:"/basic/shadowing"},"Shadowing"),"."),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"_"))),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/variabel"},"github.com/novalagung/dasarpemrogramanrust-example/../variabel")),(0,t.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/shadowing"},"Shadowing")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"#/wip/macro"},"Macro")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"#/wip/formatted-print"},"Formatted Print"))),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-01-variables-and-mutability.html"},"https://doc.rust-lang.org/book/ch03-01-variables-and-mutability.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/hello/print.html"},"https://doc.rust-lang.org/rust-by-example/hello/print.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/variable_bindings/scope.html"},"https://doc.rust-lang.org/rust-by-example/variable_bindings/scope.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/variables.html"},"https://doc.rust-lang.org/reference/variables.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"https://rust-lang.github.io/api-guidelines/naming.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/38045700/how-do-i-declare-multiple-mutable-variables-at-the-same-time"},"https://stackoverflow.com/questions/38045700/how-do-i-declare-multiple-mutable-variables-at-the-same-time"))))}u.isMDXComponent=!0},7281:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/variabel-1-c9d8201d5914e2a0da75b23e92fa5bb9.png"},3283:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},5394:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},6204:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/variabel-4-11c9b8e77e4ddf88a314c0b480337a42.png"},3264:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},4247:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/8b55bf27.a01d27dd.js b/assets/js/8b55bf27.c5401f96.js similarity index 65% rename from assets/js/8b55bf27.a01d27dd.js rename to assets/js/8b55bf27.c5401f96.js index c92d97ec..a3e5107b 100644 --- a/assets/js/8b55bf27.a01d27dd.js +++ b/assets/js/8b55bf27.c5401f96.js @@ -1 +1 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[9886],{3905:(e,t,n)=>{n.d(t,{Zo:()=>d,kt:()=>k});var a=n(7294);function i(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function l(e){for(var t=1;t=0||(i[n]=e[n]);return i}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var p=a.createContext({}),o=function(e){var t=a.useContext(p),n=t;return e&&(n="function"==typeof e?e(t):l(l({},t),e)),n},d=function(e){var t=o(e.components);return a.createElement(p.Provider,{value:t},e.children)},m={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},u=a.forwardRef((function(e,t){var n=e.components,i=e.mdxType,r=e.originalType,p=e.parentName,d=s(e,["components","mdxType","originalType","parentName"]),u=o(n),k=i,c=u["".concat(p,".").concat(k)]||u[k]||m[k]||r;return n?a.createElement(c,l(l({ref:t},d),{},{components:n})):a.createElement(c,l({ref:t},d))}));function k(e,t){var n=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var r=n.length,l=new Array(r);l[0]=u;var s={};for(var p in t)hasOwnProperty.call(t,p)&&(s[p]=t[p]);s.originalType=e,s.mdxType="string"==typeof e?e:i,l[1]=s;for(var o=2;o{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>r,metadata:()=>s,toc:()=>o});var a=n(7462),i=(n(7294),n(3905));const r={sidebar_position:45,title:"A.45. Tipe Data \u279c String Custom Type",sidebar_label:"A.45. Tipe Data \u279c String Custom Type"},l=void 0,s={unversionedId:"basic/tipe-data-custom-type-string-slice",id:"basic/tipe-data-custom-type-string-slice",title:"A.45. Tipe Data \u279c String Custom Type",description:"Pembahasan kali ini lingkupnya masih dalam topik custom type String.",source:"@site/docs/basic/tipe-data-custom-type-string-slice.md",sourceDirName:"basic",slug:"/basic/tipe-data-custom-type-string-slice",permalink:"/basic/tipe-data-custom-type-string-slice",draft:!1,tags:[],version:"current",sidebarPosition:45,frontMatter:{sidebar_position:45,title:"A.45. Tipe Data \u279c String Custom Type",sidebar_label:"A.45. Tipe Data \u279c String Custom Type"},sidebar:"tutorialSidebar",previous:{title:"A.44. String Literal (&str) vs. String Custom Type",permalink:"/basic/string-slice-vs-string-literal"},next:{title:"A.46. Closures",permalink:"/basic/closures"}},p={},o=[{value:"A.45.1. Pembuatan string slice",id:"a451-pembuatan-string-slice",level:2},{value:"\u25c9 Method to_string milik &str",id:"-method-to_string-milik-str",level:3},{value:"\u25c9 String::from",id:"-stringfrom",level:3},{value:"\u25c9 String::new",id:"-stringnew",level:3},{value:"\u25c9 String::from_utf8",id:"-stringfrom_utf8",level:3},{value:"A.45.2. String mutability",id:"a452-string-mutability",level:2},{value:"\u25c9 Keyword mut",id:"-keyword-mut",level:3},{value:"\u25c9 Replace string / method replace",id:"-replace-string--method-replace",level:3},{value:"\u25c9 Prepend string / method insert_str",id:"-prepend-string--method-insert_str",level:3},{value:"\u25c9 Prepend char / method insert",id:"-prepend-char--method-insert",level:3},{value:"\u25c9 Append string / method push_str",id:"-append-string--method-push_str",level:3},{value:"\u25c9 Append char / method push",id:"-append-char--method-push",level:3},{value:"\u25c9 Clear string / method clear",id:"-clear-string--method-clear",level:3},{value:"A.45.3. Operasi string lainnya",id:"a453-operasi-string-lainnya",level:2},{value:"\u25c9 Cek substring / method contains",id:"-cek-substring--method-contains",level:3},{value:"\u25c9 Concat strings / slice join",id:"-concat-strings--slice-join",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:o};function m(e){let{components:t,...n}=e;return(0,i.kt)("wrapper",(0,a.Z)({},d,n,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pembahasan kali ini lingkupnya masih dalam topik custom type ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("p",null,"Penulis rasa 2 chapter terakhir sudah cukup banyak menjelaskan perihal apa itu tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", apa perbedaannya dibanding ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),", dan kapan harus menggunakannya."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Pembahasan mengenai perbedaan ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," vs ",(0,i.kt)("inlineCode",{parentName:"li"},"&str")," bisa dilihat pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type")),(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang bagaimana data pada string slice di-manage di memory bisa dilihat pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/slice-memory-management"},"Slice Memory Management")))),(0,i.kt)("p",null,"Chapter ini akan lebih fokus ke fitur yang tersedia pada tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("h2",{id:"a451-pembuatan-string-slice"},"A.45.1. Pembuatan string slice"),(0,i.kt)("p",null,"Ada banyak cara yang bisa digunakan dalam membuat data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", di antaranya:"),(0,i.kt)("h3",{id:"-method-to_string-milik-str"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"to_string")," milik ",(0,i.kt)("inlineCode",{parentName:"h3"},"&str")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"to_string")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," mengkonversi string literal menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str2 = "iPhone 8".to_string();\nprintln!("{str2}"); // iPhone 8\n')),(0,i.kt)("h3",{id:"-stringfrom"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"String::from")),(0,i.kt)("p",null,"Associated function ",(0,i.kt)("inlineCode",{parentName:"p"},"String::from")," digunakan untuk mengkonversi ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),". Kita sudah cukup sering menggunakan fungsi ini."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str1 = String::from("Nokia 3310");\nprintln!("{str1}"); // Nokia 3310\n')),(0,i.kt)("h3",{id:"-stringnew"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"String::new")),(0,i.kt)("p",null,"Associated function ",(0,i.kt)("inlineCode",{parentName:"p"},"String::new")," menghasilkan data string kosong."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str3 = String::new();\nprintln!("{str3}"); // ""\n')),(0,i.kt)("h3",{id:"-stringfrom_utf8"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"String::from_utf8")),(0,i.kt)("p",null,"Berguna untuk mengkonversi data bytes ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),". Fungsi ini nilai baliknya bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),", maka dalam penerapannya harus di-chain dengan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," untuk mendapatkan data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"-nya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str4 = String::from_utf8(vec![78, 55, 51]).unwrap();\nprintln!("{str4}"); // N73\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,i.kt)("h2",{id:"a452-string-mutability"},"A.45.2. String mutability"),(0,i.kt)("h3",{id:"-keyword-mut"},"\u25c9 Keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"mut")),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," bisa digunakan untuk mengganti/replace data string dengan data baru. Sebagai contoh, ",(0,i.kt)("inlineCode",{parentName:"p"},"str5")," berikut yang awalnya adalah string kosong di-replace dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"Pixel 5"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str5 = String::new();\nprintln!("{str5}");\n// ""\n\nstr5 = String::from("Pixel 5");\nprintln!("{str5}");\n// Pixel 5\n')),(0,i.kt)("h3",{id:"-replace-string--method-replace"},"\u25c9 Replace string / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"replace")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"replace")," digunakan untuk mengganti suatu substring dengan string lain. Method ini menghasilkan object ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," baru dan tidak mengubah data string aslinya. Karena alasan itu juga kenapa tidak perlu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," dalam penerapannya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str9 = String::from("my phone is Pixel 6");\nlet str10 = str9.replace("Pixel 6", "Nokia 3310");\n\nprintln!("str9: {str9}"); // my phone is Pixel 6\nprintln!("str10: {str10}"); // my phone is Nokia 3310\n')),(0,i.kt)("h3",{id:"-prepend-string--method-insert_str"},"\u25c9 Prepend string / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"insert_str")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"insert_str")," digunakan untuk menyisipkan substring pada posisi tertentu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str6 = String::from("Pixel 6");\nprintln!("{str6}");\n// Pixel 6\n\nstr6.insert_str(0, "my phone");\nprintln!("{str6}");\n// my phonePixel 6\n\nstr6.insert_str(8, " is ");\nprintln!("{str6}");\n// my phone is Pixel 6\n')),(0,i.kt)("p",null,"Parameter pertama perlu diisi dengan indeks dimana string akan disisipkan."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"String ",(0,i.kt)("inlineCode",{parentName:"li"},"Pixel 6")," pada indeks 0 disisipi string ",(0,i.kt)("inlineCode",{parentName:"li"},"my phone"),", hasilnya ",(0,i.kt)("inlineCode",{parentName:"li"},"my phonePixel 6")),(0,i.kt)("li",{parentName:"ul"},"String ",(0,i.kt)("inlineCode",{parentName:"li"},"my phonePixel 6")," pada indeks 8 disisipi string ",(0,i.kt)("inlineCode",{parentName:"li"},"is"),", hasilnya ",(0,i.kt)("inlineCode",{parentName:"li"},"my phone is Pixel 6"))),(0,i.kt)("h3",{id:"-prepend-char--method-insert"},"\u25c9 Prepend char / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"insert")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"insert")," kegunaannya sama seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"insert_str")," yaitu untuk menyisipkan string pada posisi tertentu. Perbedaannya, pada method ",(0,i.kt)("inlineCode",{parentName:"p"},"insert")," parameter kedua diisi dengan char."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut str7 = String::from(\"3310\");\n\nstr7.insert(0, 'N'); // N3310\nstr7.insert(1, 'o'); // No3310\nstr7.insert(2, 'k'); // Nok3310\nstr7.insert(3, 'i'); // Noki3310\nstr7.insert(4, 'a'); // Nokia3310\nstr7.insert(5, ' '); // Nokia 3310\n")),(0,i.kt)("h3",{id:"-append-string--method-push_str"},"\u25c9 Append string / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"push_str")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"push_str")," digunakan untuk menambahkan string di akhir."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str8 = String::from("Pixel 6"); // Pixel 6\nstr8.push_str(" is a good phone"); // Pixel 6 is a good phone\n')),(0,i.kt)("h3",{id:"-append-char--method-push"},"\u25c9 Append char / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"push")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"push")," sama kegunaannya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"push_str"),", namun untuk penambahan data ",(0,i.kt)("inlineCode",{parentName:"p"},"char"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str8 = String::from("Pixel");\n\nstr8.push(\' \'); // "Pixel "\nstr8.push(\'7\'); // "Pixel 7"\n')),(0,i.kt)("h3",{id:"-clear-string--method-clear"},"\u25c9 Clear string / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"clear")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"clear")," digunakan untuk mengosongkan data string."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str11 = String::from("Nokia 3310");\nstr11.clear();\nprintln!("{str11}"); // ""\n')),(0,i.kt)("h2",{id:"a453-operasi-string-lainnya"},"A.45.3. Operasi string lainnya"),(0,i.kt)("h3",{id:"-cek-substring--method-contains"},"\u25c9 Cek substring / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"contains")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"contains")," digunakan untuk mengecek apakah suatu substring yang dicari ada atau tidak. Method ini mengembalikan nilai bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"bool"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str11 = String::from("Nokia 3310");\n\nlet is_exists = str11.contains("3310");\nprintln!("{is_exists}"); // true\n\nlet is_exists = str11.contains("3315");\nprintln!("{is_exists}"); // false\n')),(0,i.kt)("h3",{id:"-concat-strings--slice-join"},"\u25c9 Concat strings / slice ",(0,i.kt)("inlineCode",{parentName:"h3"},"join")),(0,i.kt)("p",null,"Operasi concat string bisa dilakukan memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"insert_str")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"push_str"),". Selain itu juga bisa dengan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},".join")," milik slice. Caranya, jadikan string yang ingin di-gabung sebagai element array, kemudian akses method ",(0,i.kt)("inlineCode",{parentName:"p"},".join"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str12 = String::from("iPhone");\nlet str13 = String::from("12");\nlet str14 = String::from("Pro");\n\nlet str: String = [str12, str13, str14].join(" ");\n\nprintln!("{str}"); // iPhone 12 Pro\n')),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/tipe_data_custom_type_string_slice"},"github.com/novalagung/dasarpemrogramanrust-example/../tipe_data_custom_type_string_slice")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-string-literal"},"Tipe Data \u279c String Literal (&str)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/static"},"Static Item")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/lifetime"},"Lifetime")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice-memory-management"},"Slice Memory Management")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/string/struct.String.html"},"https://doc.rust-lang.org/std/string/struct.String.html"))))}m.isMDXComponent=!0}}]); \ No newline at end of file +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[9886],{3905:(e,t,n)=>{n.d(t,{Zo:()=>d,kt:()=>k});var a=n(7294);function i(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function l(e){for(var t=1;t=0||(i[n]=e[n]);return i}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var p=a.createContext({}),o=function(e){var t=a.useContext(p),n=t;return e&&(n="function"==typeof e?e(t):l(l({},t),e)),n},d=function(e){var t=o(e.components);return a.createElement(p.Provider,{value:t},e.children)},m={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},u=a.forwardRef((function(e,t){var n=e.components,i=e.mdxType,r=e.originalType,p=e.parentName,d=s(e,["components","mdxType","originalType","parentName"]),u=o(n),k=i,c=u["".concat(p,".").concat(k)]||u[k]||m[k]||r;return n?a.createElement(c,l(l({ref:t},d),{},{components:n})):a.createElement(c,l({ref:t},d))}));function k(e,t){var n=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var r=n.length,l=new Array(r);l[0]=u;var s={};for(var p in t)hasOwnProperty.call(t,p)&&(s[p]=t[p]);s.originalType=e,s.mdxType="string"==typeof e?e:i,l[1]=s;for(var o=2;o{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>r,metadata:()=>s,toc:()=>o});var a=n(7462),i=(n(7294),n(3905));const r={sidebar_position:45,title:"A.45. Tipe Data \u279c String Custom Type",sidebar_label:"A.45. Tipe Data \u279c String Custom Type"},l=void 0,s={unversionedId:"basic/tipe-data-custom-type-string-slice",id:"basic/tipe-data-custom-type-string-slice",title:"A.45. Tipe Data \u279c String Custom Type",description:"Pembahasan kali ini lingkupnya masih dalam topik custom type String.",source:"@site/docs/basic/tipe-data-custom-type-string-slice.md",sourceDirName:"basic",slug:"/basic/tipe-data-custom-type-string-slice",permalink:"/basic/tipe-data-custom-type-string-slice",draft:!1,tags:[],version:"current",sidebarPosition:45,frontMatter:{sidebar_position:45,title:"A.45. Tipe Data \u279c String Custom Type",sidebar_label:"A.45. Tipe Data \u279c String Custom Type"},sidebar:"tutorialSidebar",previous:{title:"A.44. String Literal (&str) vs. String Custom Type",permalink:"/basic/string-slice-vs-string-literal"},next:{title:"A.46. Closures",permalink:"/basic/closures"}},p={},o=[{value:"A.45.1. Pembuatan string slice",id:"a451-pembuatan-string-slice",level:2},{value:"\u25c9 Method to_string milik &str",id:"-method-to_string-milik-str",level:3},{value:"\u25c9 String::from",id:"-stringfrom",level:3},{value:"\u25c9 String::new",id:"-stringnew",level:3},{value:"\u25c9 String::from_utf8",id:"-stringfrom_utf8",level:3},{value:"A.45.2. String mutability",id:"a452-string-mutability",level:2},{value:"\u25c9 Keyword mut",id:"-keyword-mut",level:3},{value:"\u25c9 Replace string / method replace",id:"-replace-string--method-replace",level:3},{value:"\u25c9 Prepend string / method insert_str",id:"-prepend-string--method-insert_str",level:3},{value:"\u25c9 Prepend char / method insert",id:"-prepend-char--method-insert",level:3},{value:"\u25c9 Append string / method push_str",id:"-append-string--method-push_str",level:3},{value:"\u25c9 Append char / method push",id:"-append-char--method-push",level:3},{value:"\u25c9 Clear string / method clear",id:"-clear-string--method-clear",level:3},{value:"A.45.3. Operasi string lainnya",id:"a453-operasi-string-lainnya",level:2},{value:"\u25c9 Cek substring / method contains",id:"-cek-substring--method-contains",level:3},{value:"\u25c9 Concat strings / slice join",id:"-concat-strings--slice-join",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:o};function m(e){let{components:t,...n}=e;return(0,i.kt)("wrapper",(0,a.Z)({},d,n,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pembahasan kali ini lingkupnya masih dalam topik custom type ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("p",null,"Penulis rasa 2 chapter terakhir sudah cukup banyak menjelaskan perihal apa itu tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", apa perbedaannya dibanding ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),", dan kapan harus menggunakannya."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Pembahasan mengenai perbedaan ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," vs ",(0,i.kt)("inlineCode",{parentName:"li"},"&str")," bisa dilihat pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type")),(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang bagaimana data pada string slice di-manage di memory bisa dilihat pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/slice-memory-management"},"Slice Memory Management")))),(0,i.kt)("p",null,"Chapter ini akan lebih fokus ke fitur yang tersedia pada tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("h2",{id:"a451-pembuatan-string-slice"},"A.45.1. Pembuatan string slice"),(0,i.kt)("p",null,"Ada banyak cara yang bisa digunakan dalam membuat data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", di antaranya:"),(0,i.kt)("h3",{id:"-method-to_string-milik-str"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"to_string")," milik ",(0,i.kt)("inlineCode",{parentName:"h3"},"&str")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"to_string")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," mengkonversi string literal menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str2 = "iPhone 8".to_string();\nprintln!("{str2}"); // iPhone 8\n')),(0,i.kt)("h3",{id:"-stringfrom"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"String::from")),(0,i.kt)("p",null,"Associated function ",(0,i.kt)("inlineCode",{parentName:"p"},"String::from")," digunakan untuk mengkonversi ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),". Kita sudah cukup sering menggunakan fungsi ini."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str1 = String::from("Nokia 3310");\nprintln!("{str1}"); // Nokia 3310\n')),(0,i.kt)("h3",{id:"-stringnew"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"String::new")),(0,i.kt)("p",null,"Associated function ",(0,i.kt)("inlineCode",{parentName:"p"},"String::new")," menghasilkan data string kosong."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str3 = String::new();\nprintln!("{str3}"); // ""\n')),(0,i.kt)("h3",{id:"-stringfrom_utf8"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"String::from_utf8")),(0,i.kt)("p",null,"Berguna untuk mengkonversi data bytes ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),". Fungsi ini nilai baliknya bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),", maka dalam penerapannya harus di-chain dengan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," untuk mendapatkan data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"-nya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str4 = String::from_utf8(vec![78, 55, 51]).unwrap();\nprintln!("{str4}"); // N73\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,i.kt)("h2",{id:"a452-string-mutability"},"A.45.2. String mutability"),(0,i.kt)("h3",{id:"-keyword-mut"},"\u25c9 Keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"mut")),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," bisa digunakan untuk mengganti/replace data string dengan data baru. Sebagai contoh, ",(0,i.kt)("inlineCode",{parentName:"p"},"str5")," berikut yang awalnya adalah string kosong di-replace dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"Pixel 5"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str5 = String::new();\nprintln!("{str5}");\n// ""\n\nstr5 = String::from("Pixel 5");\nprintln!("{str5}");\n// Pixel 5\n')),(0,i.kt)("h3",{id:"-replace-string--method-replace"},"\u25c9 Replace string / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"replace")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"replace")," digunakan untuk mengganti suatu substring dengan string lain. Method ini menghasilkan object ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," baru dan tidak mengubah data string aslinya. Karena alasan itu juga kenapa tidak perlu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," dalam penerapannya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str9 = String::from("my phone is Pixel 6");\nlet str10 = str9.replace("Pixel 6", "Nokia 3310");\n\nprintln!("str9: {str9}"); // my phone is Pixel 6\nprintln!("str10: {str10}"); // my phone is Nokia 3310\n')),(0,i.kt)("h3",{id:"-prepend-string--method-insert_str"},"\u25c9 Prepend string / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"insert_str")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"insert_str")," digunakan untuk menyisipkan substring pada posisi tertentu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str6 = String::from("Pixel 6");\nprintln!("{str6}");\n// Pixel 6\n\nstr6.insert_str(0, "my phone");\nprintln!("{str6}");\n// my phonePixel 6\n\nstr6.insert_str(8, " is ");\nprintln!("{str6}");\n// my phone is Pixel 6\n')),(0,i.kt)("p",null,"Parameter pertama menentukan posisi indeks string yang disisipkan."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"String ",(0,i.kt)("inlineCode",{parentName:"li"},"Pixel 6")," pada indeks 0 disisipi string ",(0,i.kt)("inlineCode",{parentName:"li"},"my phone"),", hasilnya ",(0,i.kt)("inlineCode",{parentName:"li"},"my phonePixel 6")),(0,i.kt)("li",{parentName:"ul"},"String ",(0,i.kt)("inlineCode",{parentName:"li"},"my phonePixel 6")," pada indeks 8 disisipi string ",(0,i.kt)("inlineCode",{parentName:"li"},"is"),", hasilnya ",(0,i.kt)("inlineCode",{parentName:"li"},"my phone is Pixel 6"))),(0,i.kt)("h3",{id:"-prepend-char--method-insert"},"\u25c9 Prepend char / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"insert")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"insert")," kegunaannya sama seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"insert_str")," yaitu untuk menyisipkan string pada posisi tertentu. Perbedaannya, pada method ",(0,i.kt)("inlineCode",{parentName:"p"},"insert")," parameter kedua diisi dengan char."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut str7 = String::from(\"3310\");\n\nstr7.insert(0, 'N'); // N3310\nstr7.insert(1, 'o'); // No3310\nstr7.insert(2, 'k'); // Nok3310\nstr7.insert(3, 'i'); // Noki3310\nstr7.insert(4, 'a'); // Nokia3310\nstr7.insert(5, ' '); // Nokia 3310\n")),(0,i.kt)("h3",{id:"-append-string--method-push_str"},"\u25c9 Append string / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"push_str")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"push_str")," digunakan untuk menambahkan string di akhir."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str8 = String::from("Pixel 6"); // Pixel 6\nstr8.push_str(" is a good phone"); // Pixel 6 is a good phone\n')),(0,i.kt)("h3",{id:"-append-char--method-push"},"\u25c9 Append char / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"push")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"push")," sama kegunaannya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"push_str"),", namun untuk penambahan data ",(0,i.kt)("inlineCode",{parentName:"p"},"char"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str8 = String::from("Pixel");\n\nstr8.push(\' \'); // "Pixel "\nstr8.push(\'7\'); // "Pixel 7"\n')),(0,i.kt)("h3",{id:"-clear-string--method-clear"},"\u25c9 Clear string / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"clear")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"clear")," digunakan untuk mengosongkan data string."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str11 = String::from("Nokia 3310");\nstr11.clear();\nprintln!("{str11}"); // ""\n')),(0,i.kt)("h2",{id:"a453-operasi-string-lainnya"},"A.45.3. Operasi string lainnya"),(0,i.kt)("h3",{id:"-cek-substring--method-contains"},"\u25c9 Cek substring / method ",(0,i.kt)("inlineCode",{parentName:"h3"},"contains")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"contains")," digunakan untuk mengecek apakah suatu substring yang dicari ada atau tidak. Method ini mengembalikan nilai bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"bool"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str11 = String::from("Nokia 3310");\n\nlet is_exists = str11.contains("3310");\nprintln!("{is_exists}"); // true\n\nlet is_exists = str11.contains("3315");\nprintln!("{is_exists}"); // false\n')),(0,i.kt)("h3",{id:"-concat-strings--slice-join"},"\u25c9 Concat strings / slice ",(0,i.kt)("inlineCode",{parentName:"h3"},"join")),(0,i.kt)("p",null,"Operasi concat string bisa dilakukan memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"insert_str")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"push_str"),". Selain itu juga bisa dengan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},".join")," milik slice. Caranya, jadikan string yang ingin di-gabung sebagai element array, kemudian akses method ",(0,i.kt)("inlineCode",{parentName:"p"},".join"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str12 = String::from("iPhone");\nlet str13 = String::from("12");\nlet str14 = String::from("Pro");\n\nlet str: String = [str12, str13, str14].join(" ");\n\nprintln!("{str}"); // iPhone 12 Pro\n')),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/tipe_data_custom_type_string_slice"},"github.com/novalagung/dasarpemrogramanrust-example/../tipe_data_custom_type_string_slice")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-string-literal"},"Tipe Data \u279c String Literal (&str)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/static"},"Static Item")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/lifetime"},"Lifetime")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice-memory-management"},"Slice Memory Management")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/string/struct.String.html"},"https://doc.rust-lang.org/std/string/struct.String.html"))))}m.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/8d193b98.c7232d69.js b/assets/js/8d193b98.9b1c2896.js similarity index 61% rename from assets/js/8d193b98.c7232d69.js rename to assets/js/8d193b98.9b1c2896.js index 711b0a06..1f430ee6 100644 --- a/assets/js/8d193b98.c7232d69.js +++ b/assets/js/8d193b98.9b1c2896.js @@ -1 +1 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7505],{3905:(e,t,a)=>{a.d(t,{Zo:()=>s,kt:()=>k});var r=a(7294);function n(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function i(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,r)}return a}function o(e){for(var t=1;t=0||(n[a]=e[a]);return n}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(n[a]=e[a])}return n}var l=r.createContext({}),p=function(e){var t=r.useContext(l),a=t;return e&&(a="function"==typeof e?e(t):o(o({},t),e)),a},s=function(e){var t=p(e.components);return r.createElement(l.Provider,{value:t},e.children)},c={inlineCode:"code",wrapper:function(e){var t=e.children;return r.createElement(r.Fragment,{},t)}},m=r.forwardRef((function(e,t){var a=e.components,n=e.mdxType,i=e.originalType,l=e.parentName,s=u(e,["components","mdxType","originalType","parentName"]),m=p(a),k=n,b=m["".concat(l,".").concat(k)]||m[k]||c[k]||i;return a?r.createElement(b,o(o({ref:t},s),{},{components:a})):r.createElement(b,o({ref:t},s))}));function k(e,t){var a=arguments,n=t&&t.mdxType;if("string"==typeof e||n){var i=a.length,o=new Array(i);o[0]=m;var u={};for(var l in t)hasOwnProperty.call(t,l)&&(u[l]=t[l]);u.originalType=e,u.mdxType="string"==typeof e?e:n,o[1]=u;for(var p=2;p{a.r(t),a.d(t,{assets:()=>l,contentTitle:()=>o,default:()=>c,frontMatter:()=>i,metadata:()=>u,toc:()=>p});var r=a(7462),n=(a(7294),a(3905));const i={sidebar_position:3,title:"Contribution & Tech Stack",sidebar_label:"Contribution & Tech Stack"},o=void 0,u={unversionedId:"CONTRIBUTING",id:"CONTRIBUTING",title:"Contribution & Tech Stack",description:"Ebook Dasar Pemrograman Rust adalah project open source. Siapapun bebas untuk berkontribusi di sini, bisa dalam bentuk perbaikan typo, update kalimat, maupun submit tulisan baru.",source:"@site/docs/CONTRIBUTING.md",sourceDirName:".",slug:"/CONTRIBUTING",permalink:"/CONTRIBUTING",draft:!1,tags:[],version:"current",sidebarPosition:3,frontMatter:{sidebar_position:3,title:"Contribution & Tech Stack",sidebar_label:"Contribution & Tech Stack"},sidebar:"tutorialSidebar",previous:{title:"Pengenalan Rust Programming",permalink:"/pengelanan-rust-programming"},next:{title:"Download versi PDF",permalink:"/download-pdf"}},l={},p=[{value:"Checkout project",id:"checkout-project",level:2},{value:"Maintainer",id:"maintainer",level:2},{value:"Contributors",id:"contributors",level:2},{value:"Ebook/webhook techstack & architecture",id:"ebookwebhook-techstack--architecture",level:2}],s={toc:p};function c(e){let{components:t,...a}=e;return(0,n.kt)("wrapper",(0,r.Z)({},s,a,{components:t,mdxType:"MDXLayout"}),(0,n.kt)("p",null,"Ebook Dasar Pemrograman Rust adalah project open source. Siapapun bebas untuk berkontribusi di sini, bisa dalam bentuk perbaikan typo, update kalimat, maupun submit tulisan baru."),(0,n.kt)("p",null,"Bagi kawan-kawan yang berminat untuk berkontribusi, silakan fork ",(0,n.kt)("a",{parentName:"p",href:"https://github.com/novalagung/dasarpemrogramanrust"},"github.com/novalagung/dasarpemrogramanrust"),", kemudian langsung saja cek/buat issue kemudian submit relevan pull request untuk issue tersebut \ud83d\ude0a"),(0,n.kt)("h2",{id:"checkout-project"},"Checkout project"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre"},"git clone https://github.com/novalagung/dasarpemrogramanrust.git\ngit submodule update --init --recursive --remote\n")),(0,n.kt)("h2",{id:"maintainer"},"Maintainer"),(0,n.kt)("p",null,"E-book ini di-inisialisasi dan di-maintain oleh Noval Agung Prayogo."),(0,n.kt)("h2",{id:"contributors"},"Contributors"),(0,n.kt)("p",null,"Berikut merupakan hall of fame kontributor yang sudah berbaik hati menyisihkan waktunya untuk membantu pengembangan e-book ini."),(0,n.kt)("ol",null,(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/acepsaepudin"},"Acep Saepudin")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/afifurrohman-id"},"Afifurrohman")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/d0tf"},"Fal")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/nferdazel"},"Fredianto")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/iyansr"},"I Putu Saputrayana")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/khisby"},"Khisby Al Ghofari")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/chud-lori"},"Lori")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/zakiego"},"M. Zakiyuddin Munziri")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/opakholis"},"Opa Kholis Majid")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/rian256"},"Rian")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/shandysiswandi"},"Shandy Siswandi")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/antheiz"},"Theis Andatu")),(0,n.kt)("li",{parentName:"ol"},"... anda :-)")),(0,n.kt)("h2",{id:"ebookwebhook-techstack--architecture"},"Ebook/webhook techstack & architecture"),(0,n.kt)("p",null,"Jika pembaca tertarik untuk membuat konten ebook berbasis web sekaligus versi PDF file-nya, bisa menggunakan link berikut sebagai referensi:"),(0,n.kt)("blockquote",null,(0,n.kt)("p",{parentName:"blockquote"},(0,n.kt)("a",{parentName:"p",href:"https://medium.com/geekculture/serverless-e-book-web-book-using-docusaurus-v2-github-pages-actions-and-pdf-tools-4fef54847b85"},"https://medium.com/geekculture/serverless-e-book-web-book-using-docusaurus-v2-github-pages-actions-and-pdf-tools-4fef54847b85"))),(0,n.kt)("p",null,"Tools yang digunakan:"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},"Web-book engine \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://docusaurus.io/"},"Docusaurus v2")),(0,n.kt)("li",{parentName:"ul"},"PDF generator \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://www.princexml.com/"},"Prince XML")," + ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/signcl/docusaurus-prince-pdf"},"docusaurus-prince-pdf")," + ",(0,n.kt)("a",{parentName:"li",href:"https://www.pdflabs.com/tools/pdftk-the-pdf-toolkit/"},"PDFtk")),(0,n.kt)("li",{parentName:"ul"},"Source code repository \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/"},"GitHub")),(0,n.kt)("li",{parentName:"ul"},"Hosting \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://pages.github.com/"},"GitHub Pages")),(0,n.kt)("li",{parentName:"ul"},"CI/CD pipeline \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/features/actions"},"GitHub Actions")),(0,n.kt)("li",{parentName:"ul"},"Content Search \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://docsearch.algolia.com/"},"Algolia DocSearch"))),(0,n.kt)("p",null,"Desain arsitektur:"),(0,n.kt)("p",null,(0,n.kt)("img",{parentName:"p",src:"https://raw.githubusercontent.com/novalagung/dasarpemrogramanrust/master/etc/base%20architecture.jpg",alt:"Dasar pemrograman python"})))}c.isMDXComponent=!0}}]); \ No newline at end of file +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7505],{3905:(e,t,a)=>{a.d(t,{Zo:()=>s,kt:()=>k});var r=a(7294);function n(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function i(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);t&&(r=r.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,r)}return a}function o(e){for(var t=1;t=0||(n[a]=e[a]);return n}(e,t);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(n[a]=e[a])}return n}var l=r.createContext({}),p=function(e){var t=r.useContext(l),a=t;return e&&(a="function"==typeof e?e(t):o(o({},t),e)),a},s=function(e){var t=p(e.components);return r.createElement(l.Provider,{value:t},e.children)},c={inlineCode:"code",wrapper:function(e){var t=e.children;return r.createElement(r.Fragment,{},t)}},m=r.forwardRef((function(e,t){var a=e.components,n=e.mdxType,i=e.originalType,l=e.parentName,s=u(e,["components","mdxType","originalType","parentName"]),m=p(a),k=n,b=m["".concat(l,".").concat(k)]||m[k]||c[k]||i;return a?r.createElement(b,o(o({ref:t},s),{},{components:a})):r.createElement(b,o({ref:t},s))}));function k(e,t){var a=arguments,n=t&&t.mdxType;if("string"==typeof e||n){var i=a.length,o=new Array(i);o[0]=m;var u={};for(var l in t)hasOwnProperty.call(t,l)&&(u[l]=t[l]);u.originalType=e,u.mdxType="string"==typeof e?e:n,o[1]=u;for(var p=2;p{a.r(t),a.d(t,{assets:()=>l,contentTitle:()=>o,default:()=>c,frontMatter:()=>i,metadata:()=>u,toc:()=>p});var r=a(7462),n=(a(7294),a(3905));const i={sidebar_position:3,title:"Contribution & Tech Stack",sidebar_label:"Contribution & Tech Stack"},o=void 0,u={unversionedId:"CONTRIBUTING",id:"CONTRIBUTING",title:"Contribution & Tech Stack",description:"Ebook Dasar Pemrograman Rust adalah project open source. Siapapun bebas untuk berkontribusi di sini, bisa dalam bentuk perbaikan typo, update kalimat, maupun submit tulisan baru.",source:"@site/docs/CONTRIBUTING.md",sourceDirName:".",slug:"/CONTRIBUTING",permalink:"/CONTRIBUTING",draft:!1,tags:[],version:"current",sidebarPosition:3,frontMatter:{sidebar_position:3,title:"Contribution & Tech Stack",sidebar_label:"Contribution & Tech Stack"},sidebar:"tutorialSidebar",previous:{title:"Pengenalan Rust Programming",permalink:"/pengelanan-rust-programming"},next:{title:"Download versi PDF",permalink:"/download-pdf"}},l={},p=[{value:"Checkout project",id:"checkout-project",level:2},{value:"Maintainer",id:"maintainer",level:2},{value:"Contributors",id:"contributors",level:2},{value:"Ebook/webhook techstack & architecture",id:"ebookwebhook-techstack--architecture",level:2}],s={toc:p};function c(e){let{components:t,...a}=e;return(0,n.kt)("wrapper",(0,r.Z)({},s,a,{components:t,mdxType:"MDXLayout"}),(0,n.kt)("p",null,"Ebook Dasar Pemrograman Rust adalah project open source. Siapapun bebas untuk berkontribusi di sini, bisa dalam bentuk perbaikan typo, update kalimat, maupun submit tulisan baru."),(0,n.kt)("p",null,"Bagi kawan-kawan yang berminat untuk berkontribusi, silakan fork ",(0,n.kt)("a",{parentName:"p",href:"https://github.com/novalagung/dasarpemrogramanrust"},"github.com/novalagung/dasarpemrogramanrust"),", kemudian langsung saja cek/buat issue kemudian submit relevan pull request untuk issue tersebut \ud83d\ude0a"),(0,n.kt)("h2",{id:"checkout-project"},"Checkout project"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre"},"git clone https://github.com/novalagung/dasarpemrogramanrust.git\ngit submodule update --init --recursive --remote\n")),(0,n.kt)("h2",{id:"maintainer"},"Maintainer"),(0,n.kt)("p",null,"E-book ini di-inisialisasi dan di-maintain oleh Noval Agung Prayogo."),(0,n.kt)("h2",{id:"contributors"},"Contributors"),(0,n.kt)("p",null,"Berikut merupakan hall of fame kontributor yang sudah berbaik hati menyisihkan waktunya untuk membantu pengembangan e-book ini."),(0,n.kt)("ol",null,(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/acepsaepudin"},"Acep Saepudin")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/afifurrohman-id"},"Afifurrohman")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/ardiantovn"},"ardiantovn")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/d0tf"},"Fal")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/nferdazel"},"Fredianto")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/iyansr"},"I Putu Saputrayana")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/khisby"},"Khisby Al Ghofari")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/chud-lori"},"Lori")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/zakiego"},"M. Zakiyuddin Munziri")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/opakholis"},"Opa Kholis Majid")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/rian256"},"Rian")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/shandysiswandi"},"Shandy Siswandi")),(0,n.kt)("li",{parentName:"ol"},(0,n.kt)("a",{parentName:"li",href:"https://github.com/antheiz"},"Theis Andatu")),(0,n.kt)("li",{parentName:"ol"},"... anda :-)")),(0,n.kt)("h2",{id:"ebookwebhook-techstack--architecture"},"Ebook/webhook techstack & architecture"),(0,n.kt)("p",null,"Jika pembaca tertarik untuk membuat konten ebook berbasis web sekaligus versi PDF file-nya, bisa menggunakan link berikut sebagai referensi:"),(0,n.kt)("blockquote",null,(0,n.kt)("p",{parentName:"blockquote"},(0,n.kt)("a",{parentName:"p",href:"https://medium.com/geekculture/serverless-e-book-web-book-using-docusaurus-v2-github-pages-actions-and-pdf-tools-4fef54847b85"},"https://medium.com/geekculture/serverless-e-book-web-book-using-docusaurus-v2-github-pages-actions-and-pdf-tools-4fef54847b85"))),(0,n.kt)("p",null,"Tools yang digunakan:"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},"Web-book engine \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://docusaurus.io/"},"Docusaurus v2")),(0,n.kt)("li",{parentName:"ul"},"PDF generator \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://www.princexml.com/"},"Prince XML")," + ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/signcl/docusaurus-prince-pdf"},"docusaurus-prince-pdf")," + ",(0,n.kt)("a",{parentName:"li",href:"https://www.pdflabs.com/tools/pdftk-the-pdf-toolkit/"},"PDFtk")),(0,n.kt)("li",{parentName:"ul"},"Source code repository \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/"},"GitHub")),(0,n.kt)("li",{parentName:"ul"},"Hosting \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://pages.github.com/"},"GitHub Pages")),(0,n.kt)("li",{parentName:"ul"},"CI/CD pipeline \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://github.com/features/actions"},"GitHub Actions")),(0,n.kt)("li",{parentName:"ul"},"Content Search \u279c ",(0,n.kt)("a",{parentName:"li",href:"https://docsearch.algolia.com/"},"Algolia DocSearch"))),(0,n.kt)("p",null,"Desain arsitektur:"),(0,n.kt)("p",null,(0,n.kt)("img",{parentName:"p",src:"https://raw.githubusercontent.com/novalagung/dasarpemrogramanrust/master/etc/base%20architecture.jpg",alt:"Dasar pemrograman python"})))}c.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/93422b31.2fa67f2b.js b/assets/js/93422b31.2fa67f2b.js new file mode 100644 index 00000000..071632ee --- /dev/null +++ b/assets/js/93422b31.2fa67f2b.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8469],{3905:(a,e,t)=>{t.d(e,{Zo:()=>d,kt:()=>c});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var p=n.createContext({}),s=function(a){var e=n.useContext(p),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},d=function(a){var e=s(a.components);return n.createElement(p.Provider,{value:e},a.children)},o={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},m=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,p=a.parentName,d=u(a,["components","mdxType","originalType","parentName"]),m=s(t),c=i,k=m["".concat(p,".").concat(c)]||m[c]||o[c]||r;return t?n.createElement(k,l(l({ref:e},d),{},{components:t})):n.createElement(k,l({ref:e},d))}));function c(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=m;var u={};for(var p in e)hasOwnProperty.call(e,p)&&(u[p]=e[p]);u.originalType=a,u.mdxType="string"==typeof a?a:i,l[1]=u;for(var s=2;s{t.r(e),t.d(e,{assets:()=>p,contentTitle:()=>l,default:()=>o,frontMatter:()=>r,metadata:()=>u,toc:()=>s});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:23,title:"A.23. Struct",sidebar_label:"A.23. Struct"},l=void 0,u={unversionedId:"basic/struct",id:"basic/struct",title:"A.23. Struct",description:"Pada chapter ini kita akan belajar tentang struct.",source:"@site/docs/basic/struct.md",sourceDirName:"basic",slug:"/basic/struct",permalink:"/basic/struct",draft:!1,tags:[],version:"current",sidebarPosition:23,frontMatter:{sidebar_position:23,title:"A.23. Struct",sidebar_label:"A.23. Struct"},sidebar:"tutorialSidebar",previous:{title:"A.22. Module System \u279c Scope & Akses Item",permalink:"/basic/module-scope-item-access"},next:{title:"A.24. Associated Function",permalink:"/basic/associated-function"}},p={},s=[{value:"A.23.1. Konsep Struct",id:"a231-konsep-struct",level:2},{value:"A.23.2. Pembuatan struct",id:"a232-pembuatan-struct",level:2},{value:"\u25c9 Fungsi String::from()",id:"-fungsi-stringfrom",level:3},{value:"\u25c9 Naming convention struct",id:"-naming-convention-struct",level:3},{value:"A.23.3. Mutable struct",id:"a233-mutable-struct",level:2},{value:"A.23.4. Macam-macam notasi deklarasi variabel struct",id:"a234-macam-macam-notasi-deklarasi-variabel-struct",level:2},{value:"\u25c9 type inference / manifest typing",id:"-type-inference--manifest-typing",level:3},{value:"\u25c9 Variabel struct tanpa predefined value",id:"-variabel-struct-tanpa-predefined-value",level:3},{value:"\u25c9 Variabel struct dengan nilai berasal dari struct lain",id:"-variabel-struct-dengan-nilai-berasal-dari-struct-lain",level:3},{value:"\u25c9 Field init shorthand",id:"-field-init-shorthand",level:3},{value:"\u25c9 Deklarasi nilai struct secara horizontal",id:"-deklarasi-nilai-struct-secara-horizontal",level:3},{value:"\u25c9 Destructuring assignment",id:"-destructuring-assignment",level:3},{value:"A.23.5. Unit-like structs",id:"a235-unit-like-structs",level:2},{value:"A.23.6. Debugging value struct menggunakan #[derive(Debug)]",id:"a236-debugging-value-struct-menggunakan-derivedebug",level:2},{value:"A.23.7. Tuple struct",id:"a237-tuple-struct",level:2},{value:"A.23.8. Struct property visibility",id:"a238-struct-property-visibility",level:2},{value:"A.23.9. Tuple struct property visibility",id:"a239-tuple-struct-property-visibility",level:2},{value:"A.23.10. Generic pada struct",id:"a2310-generic-pada-struct",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:s};function o(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},d,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tentang struct."),(0,i.kt)("h2",{id:"a231-konsep-struct"},"A.23.1. Konsep Struct"),(0,i.kt)("p",null,"Struct (kependekan dari ",(0,i.kt)("em",{parentName:"p"},"structure"),") adalah tipe data custom yang dengannya kita bisa mengumpulkan beberapa definisi tipe data lalu menjadikannya sebagai satu buah tipe data dalam struktur tertentu."),(0,i.kt)("p",null,"Contoh analogi seperti sebuah mobil. Mobil memiliki roda, mesin, tempat kemudi, dan banyak lainnya. Mobil pada contoh ini adalah struct, sedang isi dari mobil tersebut biasa disebut dengan ",(0,i.kt)("em",{parentName:"p"},"attribute")," atau ",(0,i.kt)("em",{parentName:"p"},"property")," atau ",(0,i.kt)("em",{parentName:"p"},"field"),"."),(0,i.kt)("p",null,"Selain property, mobil juga bisa melakukan aksi, contohnya seperti jalan, belok kanan, berhenti. Aksi tersebut pada contoh ini adalah yang disebut dengan ",(0,i.kt)("em",{parentName:"p"},"method"),". Method sendiri adalah fungsi yang merupakan property sebuah struct."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Chapter ini fokus pada pembahasan struct beserta property-nya. Topik tentang method dibahas pada chapter selanjutnya, yaitu ",(0,i.kt)("a",{parentName:"p",href:"/basic/method"},"Method"),".")),(0,i.kt)("h2",{id:"a232-pembuatan-struct"},"A.23.2. Pembuatan struct"),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"struct")," digunakan untuk membuat custom type struct. Notasi penulisannya seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct NamaStruct {\n property_satu: String,\n property_dua: u64,\n // ...\n}\n")),(0,i.kt)("p",null,"Contoh, sebuah struct bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," yang memiliki 4 buah property:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"name")," bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"String")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"sign_in_count")," bertipe numerik ",(0,i.kt)("inlineCode",{parentName:"li"},"u64")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"affliation")," bertipe vektor ",(0,i.kt)("inlineCode",{parentName:"li"},"Vec")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"active")," bertipe boolean")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct User {\n name: String,\n sign_in_count: u64,\n affliation: Vec,\n active: bool,\n}\n")),(0,i.kt)("p",null,"Struct adalah ",(0,i.kt)("em",{parentName:"p"},"hanya definisi structure"),"-nya saja, struct tidak menampung nilai atau value. Contoh pada struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," di atas bisa dilihat bahwa yang didefinisikan hanya nama struct dan property (beserta tipe datanya). Tidak ada pengisian nilai sama sekali."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," merupakan salah satu contoh penerapan generics. Topik ini nantinya dibahas lebih detail pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics"},"Generics"),".")),(0,i.kt)("p",null,"Struct merupakan tipe data data custom, yang berarti tipe data tersebut bisa digunakan dalam pembuatan variabel. Sebagai contoh di bawah ini, sebuah variabel bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"user_one")," didefinisikan dengan tipe adalah struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," yang telah dibuat."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct User {\n name: String,\n sign_in_count: u64,\n affliation: Vec,\n active: bool,\n}\n\nfn main() {\n let user_one = User{\n name: String::from("Orgrim Doomhammer"),\n sign_in_count: 12,\n affliation: vec![\n String::from("Warchief of the Horde"),\n String::from("Blackrock Chieftain"),\n String::from("The Doomhammer"),\n ],\n active: false,\n };\n\n println!("name: {}", user_one.name);\n println!("sign-in count: {}", user_one.sign_in_count);\n println!("affliation: {:?}", user_one.affliation);\n println!("is active? {}", user_one.active);\n}\n')),(0,i.kt)("p",null,"Jalankan program di atas, lihat hasilnya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(1226).Z,width:"572",height:"123"})),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"user_one")," didefinisikan tipe datanya adalah struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User"),". Notasinya penulisannya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let user_one = User{\n // property_satu: value,\n // property_dua: value,\n // ...\n};\n")),(0,i.kt)("p",null,"Deklarasi variabel bertipe data struct mewajibkan isi masing-masing property harus dituliskan beserta nilainya. Sebagai contoh variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"user_one")," yang bertipe struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," di atas, dalam statement bisa dilihat ke-empat property milik struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," dituliskan beserta nilainya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let user_one = User{\n name: String::from("Orgrim Doomhammer"),\n sign_in_count: 12,\n affliation: vec![\n String::from("Warchief of the Horde"),\n String::from("Blackrock Chieftain"),\n String::from("The Doomhammer"),\n ],\n active: false,\n};\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Variabel yang tipe data-nya adalah struct biasa disebut dengan ",(0,i.kt)("em",{parentName:"p"},"object")," atau ",(0,i.kt)("em",{parentName:"p"},"instance"),".")),(0,i.kt)("h3",{id:"-fungsi-stringfrom"},"\u25c9 Fungsi ",(0,i.kt)("inlineCode",{parentName:"h3"},"String::from()")),(0,i.kt)("p",null,"Pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/path-item#-stdstringstringnew"},"Module System \u279c Path & Item")," kita telah sedikit mengenal fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"String::new()")," yang kegunaannya adalah untuk membuat data string kosong bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("p",null,"Selain fungsi tersebut, ada juga fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"String::from()")," yang kegunaannya juga untuk pembuatan data string bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", tapi dengan isi ditentukan lewat argumen pemanggilan fungsi."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let a = "";\n// variabel `a` bertipe data `&str`, isi value-nya ""\n\nlet b = String::new();\n// variabel `b` bertipe data `String`, isi value-nya ""\n\nlet c = "hello";\n// variabel `c` bertipe data `&str`, isi value-nya "hello"\n\nlet d = String::from("world");\n// variabel `d` bertipe data `String`, isi value-nya "world"\n')),(0,i.kt)("p",null,"Untuk sekarang, boleh menggunakan kesimpulan berikut:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Hasil dari ",(0,i.kt)("inlineCode",{parentName:"li"},"String::new()")," adalah sama dengan literal string ",(0,i.kt)("inlineCode",{parentName:"li"},'""')," tapi bertipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," (bukan ",(0,i.kt)("inlineCode",{parentName:"li"},"&str"),")"),(0,i.kt)("li",{parentName:"ul"},"Hasil dari ",(0,i.kt)("inlineCode",{parentName:"li"},'String::from("hello")')," adalah sama dengan literal string ",(0,i.kt)("inlineCode",{parentName:"li"},'"hello"')," tapi bertipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," (bukan ",(0,i.kt)("inlineCode",{parentName:"li"},"&str"),")")),(0,i.kt)("p",null,"Sebenarnya perbedaannya tidak hanya itu saja, nantinya kita pelajari lebih detail pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type"),"."),(0,i.kt)("h3",{id:"-naming-convention-struct"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Naming convention")," struct"),(0,i.kt)("p",null,"Sesuai anjuran di ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", upper camel case digunakan dalam penamaan struct dan camel case untuk penamaan property-nya. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct User {\n name: String,\n sign_in_count: u64,\n affliation: Vec,\n active: bool,\n}\n")),(0,i.kt)("h2",{id:"a233-mutable-struct"},"A.23.3. Mutable struct"),(0,i.kt)("p",null,"Penentuan ",(0,i.kt)("em",{parentName:"p"},"mutability")," sebuah struct dilakukan dengan cara menambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," seperti pada umumnya variabel. Contohnya bisa dilihat pada variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"user_two")," berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut user_two: User = User{\n name: String::from("Varian Wrynn"),\n sign_in_count: 12,\n affliation: vec![\n String::from("High King of the Alliance"),\n String::from("King of Stormwind"),\n String::from("Champion of the Crimson Ring"),\n ],\n active: false,\n};\n\nuser_two.name = String::from("Anduin Wrynn");\nuser_two.affliation.pop();\nuser_two.active = true;\n\nprintln!("name: {}", user_two.name);\nprintln!("sign-in count: {}", user_two.sign_in_count);\nprintln!("affliation: {:?}", user_two.affliation);\nprintln!("is active? {}", user_two.active);\n')),(0,i.kt)("p",null,"Pada contoh di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"user_two")," adalah object struct yang bisa diubah nilainya (mutable). Perubahan nilai struct bisa dilakukan pada property, contohnya seperti property ",(0,i.kt)("inlineCode",{parentName:"p"},"name"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"affliation"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"active")," yang dicontohkan diubah nilainya."),(0,i.kt)("p",null,"Atau bisa juga perubahannya pada value struct itu sendiri, diganti dengan value baru."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut user_two = User{\n name: String::from("Varian Wrynn"),\n sign_in_count: 12,\n affliation: vec![\n String::from("High King of the Alliance"),\n String::from("King of Stormwind"),\n String::from("Champion of the Crimson Ring"),\n ],\n active: false,\n};\n\nuser_two = User{\n name: String::from("Anduin Wrynn"),\n sign_in_count: 12,\n affliation: vec![\n String::from("High King of the Alliance"),\n String::from("King of Stormwind"),\n ],\n active: true,\n};\n')),(0,i.kt)("h2",{id:"a234-macam-macam-notasi-deklarasi-variabel-struct"},"A.23.4. Macam-macam notasi deklarasi variabel struct"),(0,i.kt)("h3",{id:"-type-inference--manifest-typing"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"type inference")," / ",(0,i.kt)("em",{parentName:"h3"},"manifest typing")),(0,i.kt)("p",null,"Metode deklarasi ",(0,i.kt)("em",{parentName:"p"},"type inference")," ataupun ",(0,i.kt)("em",{parentName:"p"},"manifest typing")," bisa digunakan dalam variabel struct."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct Car {\n brand: String,\n model: String,\n}\n\nlet car_one = Car{\n brand: String::from("Toyota"),\n model: String::from("Sprinter Trueno AE86"),\n};\n\nlet car_two: Car = Car{\n brand: String::from("BMW"),\n model: String::from("M3 GTR"),\n};\n')),(0,i.kt)("h3",{id:"-variabel-struct-tanpa-predefined-value"},"\u25c9 Variabel struct tanpa ",(0,i.kt)("em",{parentName:"h3"},"predefined value")),(0,i.kt)("p",null,"Variabel struct boleh didefinisikan tanpa ",(0,i.kt)("em",{parentName:"p"},"predefined value"),", jadi cukup tipe datanya saja yang ditentukan saat deklarasi variabel. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut car_three: Car;\ncar_three = Car{\n brand: String::from("Audi"),\n model: String::from("Le Mans Quattro"),\n};\n')),(0,i.kt)("h3",{id:"-variabel-struct-dengan-nilai-berasal-dari-struct-lain"},"\u25c9 Variabel struct dengan nilai berasal dari struct lain"),(0,i.kt)("p",null,"Jika ada kebutuhan untuk membuat variabel object struct yang nilai property-nya sebagian adalah berasal dari variabel struct lain, bisa manfaatkan syntax ",(0,i.kt)("inlineCode",{parentName:"p"},".."),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut car_three: Car;\ncar_three = Car{\n brand: String::from("Audi"),\n model: String::from("Le Mans Quattro"),\n};\nprintln!("{} {}", car_three.brand, car_three.model);\n\nlet mut car_four: Car;\ncar_four = Car{\n brand: String::from("Audi Brand"),\n ..car_three\n};\nprintln!("{} {}", car_four.brand, car_four.model);\n')),(0,i.kt)("p",null,"Pada contoh di atas, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"car_four")," property ",(0,i.kt)("inlineCode",{parentName:"p"},"brand"),"-nya diisi dengan ",(0,i.kt)("inlineCode",{parentName:"p"},'"Audi Brand"'),". Sedangkan property-property lainnya nilai didapat dari nilai propery milik ",(0,i.kt)("inlineCode",{parentName:"p"},"car_three"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(341).Z,width:"511",height:"108"})),(0,i.kt)("p",null,"Coba jalankan program, akan terlihat nilai dari variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"car_four")," untuk property selain ",(0,i.kt)("inlineCode",{parentName:"p"},"brand")," adalah sama dengan nilai property-property ",(0,i.kt)("inlineCode",{parentName:"p"},"car_three"),"."),(0,i.kt)("h3",{id:"-field-init-shorthand"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Field init shorthand")),(0,i.kt)("p",null,"Jika ada variabel yang namanya sama persis dengan nama property sebuah struct, maka dalam deklarasi variabel struct bisa menggunakan teknik penulisan seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let model = String::from("Corvette C1");\n\nlet car_five = Car{\n brand: String::from("Chevrolet"),\n model,\n};\n')),(0,i.kt)("p",null,"Cukup tulis nama variabelnya saja tanpa value."),(0,i.kt)("p",null,"Teknik ",(0,i.kt)("em",{parentName:"p"},"shorthand")," ini bisa juga digunakan dalam fungsi. Contoh di bawah ini ada fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"new_car")," yang memiliki nama parameter adalah sama persis dengan nama property struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Car"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn new_car(brand: String, model: String) -> Car {\n Car{\n brand,\n model,\n }\n}\n\nfn main() {\n let car_six = new_car(\n String::from("Chevrolet"),\n String::from("Corvette C6")\n );\n // ...\n}\n')),(0,i.kt)("h3",{id:"-deklarasi-nilai-struct-secara-horizontal"},"\u25c9 Deklarasi nilai struct secara horizontal"),(0,i.kt)("p",null,"Umumnya, operasi ",(0,i.kt)("em",{parentName:"p"},"assignment")," nilai struct dituliskan secara vertikal. Untuk struct yang property-nya sedikit biasanya dituliskan secara horizontal contohnya seperti di bawah ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct Point {\n x: f32,\n y: f32,\n}\n\nlet point_one = Point { x: 3.14, y: 8.0 };\n")),(0,i.kt)("h3",{id:"-destructuring-assignment"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Destructuring assignment")),(0,i.kt)("p",null,"Teknik penulisan ini bisa dipakai dalam case di mana nilai property struct perlu ditampung ke variabel baru. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let point_one = Point { x: 3.14, y: 8.0 };\n\nlet Point { x: x_one, y: y_one } = point_one;\nprintln!("x_one: {}", x_one);\nprintln!("y_one: {}", y_one);\n')),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"x_one")," di atas akan menampung nilai dari ",(0,i.kt)("inlineCode",{parentName:"li"},"point_one.x")),(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"y_one")," di atas akan menampung nilai dari ",(0,i.kt)("inlineCode",{parentName:"li"},"point_one.y"))),(0,i.kt)("p",null,"Jika tidak semua property struct perlu untuk ditampung ke variabel baru, maka gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"_")," untuk property yang nilainya tidak ditampung. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let point_one = Point { x: 3.14, y: 8.0 };\n\nlet Point { x: _, y: y_one } = point_one;\nprintln!("y_one: {}", y_one);\n')),(0,i.kt)("h2",{id:"a235-unit-like-structs"},"A.23.5. ",(0,i.kt)("em",{parentName:"h2"},"Unit-like structs")),(0,i.kt)("p",null,(0,i.kt)("em",{parentName:"p"},"Unit-like structs")," adalah struct yang didefinisikan tanpa property. Cara deklrasinya bisa dilihat pada contoh berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct StructOne;\n\nlet data_one = StructOne;\n")),(0,i.kt)("p",null,"Teknik pembuatan struct ini berguna ketika ada case di mana ada kebutuhan untuk mengimplementasikan sebuah trait ke suatu tipe data. Lebih jelasnya akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),"."),(0,i.kt)("h2",{id:"a236-debugging-value-struct-menggunakan-derivedebug"},"A.23.6. Debugging value struct menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"#[derive(Debug)]")),(0,i.kt)("p",null,(0,i.kt)("em",{parentName:"p"},"By default"),", error akan muncul saat berusaha menampilkan nilai variabel struct (bukan nilai property-nya, tapi nilai variabel struct-nya) menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Ini disebabkan karena data yang bisa ditampilkan menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println")," harus memiliki trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," (atau ",(0,i.kt)("inlineCode",{parentName:"p"},"Display"),")."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(6091).Z,width:"743",height:"308"})),(0,i.kt)("p",null,"Lalu jika ingin melihat nilai property struct bagaimana solusinya? Bisa dengan menampilkan nilai property satu per satu, tapi pastinya butuh effort."),(0,i.kt)("p",null,"Cara yang lebih elegan adalah dengan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"#[derive(Debug)]"),". Penerapannya cukup dengan menuliskan atribut tersebut tepat di atas deklarasi struct. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nstruct GamingConsole {\n name: String\n}\n\nlet console_one = GamingConsole{\n name: String::from("PlayStation 5"),\n};\n\nprintln!("data_struct_one: {:#?}", console_one);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(7542).Z,width:"510",height:"330"})),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih detailnya mengenai traits dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},"Lebih detailnya mengenai attribute dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/attributes"},"Attributes")))),(0,i.kt)("h2",{id:"a237-tuple-struct"},"A.23.7. Tuple struct"),(0,i.kt)("p",null,"Tuple struct adalah struct yang didefinisikan dengan gaya tuple. Property pada tuple struct diakses menggunakan notasi pengaksesan tuple item."),(0,i.kt)("p",null,"Pada contoh berikut tuple struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Color")," didefinisikan dengan isi 3 buah item yang kesemuanya bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),". Lalu tipe data tuple struct tersebut digunakan untuk membuat variabel baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"red"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct Color(i32, i32, i32);\n\nlet red = Color(255, 0, 0);\n\nprintln!("{:?} {:?} {:?}", red.0, red.1, red.2);\n')),(0,i.kt)("p",null,"Bisa dilihat di contoh, property diakses menggunakan nomor indeks dengan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"variable.index"),"."),(0,i.kt)("p",null,"Contoh lainnya, tuple struct ",(0,i.kt)("inlineCode",{parentName:"p"},"SomeTupleStruct")," berikut dideklarasikan memiliki item ke-1 bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," dan item ke-2 bertipe boolean."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct SomeTupleStruct(i32, bool);\n\nlet some_data = SomeTupleStruct(0, false);\n\nprintln!("{:?} {:?}", some_data.0, some_data.1);\n')),(0,i.kt)("h2",{id:"a238-struct-property-visibility"},"A.23.8. Struct property visibility"),(0,i.kt)("p",null,"Sekarang mari kita coba praktikkan satu program lagi, namun kali ini sedikit berbeda, struct kita definisikan sebagai module item."),(0,i.kt)("p",null,"Silakan siapkan program dengan struktur seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u251c\u2500\u2500\u2500 models.rs\n \u2514\u2500\u2500\u2500 models\n \u2514\u2500\u2500\u2500 game.rs\n")),(0,i.kt)("p",null,"Lalu isi file program dengan kode di bawah ini. Sebuah struct dengan path ",(0,i.kt)("inlineCode",{parentName:"p"},"models::game::GamingConsole"),", dan satu buah object di fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," yang dibuat dari struct tersebut. Pastikan definisi semua module item adalah publik (ada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod models;\n\nfn main() {\n let ps5 = models::game::GamingConsole{\n name: String::from("PS 5")\n };\n println!("{:#?}", ps5);\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"pub mod game;\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models/game.rs"',title:'"src/models/game.rs"'},"#[derive(Debug)]\npub struct GamingConsole {\n name: String\n}\n")),(0,i.kt)("p",null,"Ada yang aneh, baris kode pembuatan variabel object ",(0,i.kt)("inlineCode",{parentName:"p"},"ps5")," terdeteksi sebagai error. Padahal definisi submodule dan item-nya sudah publik semua."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(7728).Z,width:"419",height:"185"})),(0,i.kt)("p",null,"Jika dilihat dari keterangan error, sebenarnya cukup jelas bagian mana yang menjadi sumber masalah, yaitu field ",(0,i.kt)("inlineCode",{parentName:"p"},"name")," yang terdeteksi sebagai private property."),(0,i.kt)("p",null,"Struct jika didefinisikan di file yang sama dengan statement pemanggilan struct tersebut tidak akan menghasilkan error. Tetapi jika definisi struct-nya terpisah dari statement pemanggilan struct (seperti contoh di atas), maka field dari struct tersebut harus publik."),(0,i.kt)("p",null,"Caranya membuat field sebagai publik adalah dengan menambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," pada property struct. Silakan ubah definisi struct ",(0,i.kt)("inlineCode",{parentName:"p"},"GamingConsole")," menjadi seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// before\npub struct GamingConsole {\n name: String\n}\n\n// after\npub struct GamingConsole {\n pub name: String\n}\n")),(0,i.kt)("p",null,"Bisa dilihat keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," ditambahkan pada deklarasi property struct. yang sebelumnya ",(0,i.kt)("inlineCode",{parentName:"p"},"name: String")," sekarang ada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," didepannya."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai visibility property dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,i.kt)("h2",{id:"a239-tuple-struct-property-visibility"},"A.23.9. Tuple struct property visibility"),(0,i.kt)("p",null,"Lalu bagaimana dengan tuple struct? apakah property-nya juga harus didefinisikan publik agar bisa diakses dari tempat lain? Jawabannya ada di praktik berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u251c\u2500\u2500\u2500 models.rs\n \u2514\u2500\u2500\u2500 models\n \u2514\u2500\u2500\u2500 color.rs\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod models;\n\nfn main() {\n let red = models::color::Color(255, 255, 0);\n println!("{:#?}", red);\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"pub mod color;\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models/color.rs"',title:'"src/models/color.rs"'},"#[derive(Debug)]\npub struct Color(i32, i32, i32);\n")),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(6610).Z,width:"626",height:"255"})),(0,i.kt)("p",null,"Yap, error yang mirip juga muncul. Jadi jawaban dari pertanyaan sebelumnya adalah iya, property tuple struct juga harus publik agar bisa diakses dari tempat lain."),(0,i.kt)("p",null,"Cara deklarasi tuple struct dengan property publik adalah dengan menambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," di masing-masing deklarasi parameter tuple struct. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// before\npub struct Color(i32, i32, i32);\n\n// after\npub struct Color(pub i32, pub i32, pub i32);\n")),(0,i.kt)("p",null,"Jalankan program untuk melihat hasilnya, error tidak muncul."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai visibility property dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,i.kt)("h2",{id:"a2310-generic-pada-struct"},"A.23.10. Generic pada struct"),(0,i.kt)("p",null,"Pembahasan mengenai generic pada struct ada pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics#a365-generics-struct"},"Generics"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/struct"},"github.com/novalagung/dasarpemrogramanrust-example/../struct")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch05-00-structs.html"},"https://doc.rust-lang.org/book/ch05-00-structs.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch05-01-defining-structs.html"},"https://doc.rust-lang.org/book/ch05-01-defining-structs.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.struct.html"},"https://doc.rust-lang.org/std/keyword.struct.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/custom_types/structs.html"},"https://doc.rust-lang.org/rust-by-example/custom_types/structs.html"))))}o.isMDXComponent=!0},1226:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},341:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},6091:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/struct-4-86fb5f564d330bc2d1cc10e13ea2d649.png"},7542:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/struct-5-5e013d5895bef6a938e2c1333ff52205.png"},7728:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},6610:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/struct-7-517c0402e96dfd5898464887fa79ec98.png"}}]); \ No newline at end of file diff --git a/assets/js/93422b31.7d2d5eae.js b/assets/js/93422b31.7d2d5eae.js deleted file mode 100644 index b6e5e2df..00000000 --- a/assets/js/93422b31.7d2d5eae.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8469],{3905:(a,e,t)=>{t.d(e,{Zo:()=>d,kt:()=>c});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var p=n.createContext({}),s=function(a){var e=n.useContext(p),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},d=function(a){var e=s(a.components);return n.createElement(p.Provider,{value:e},a.children)},o={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},m=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,p=a.parentName,d=u(a,["components","mdxType","originalType","parentName"]),m=s(t),c=i,k=m["".concat(p,".").concat(c)]||m[c]||o[c]||r;return t?n.createElement(k,l(l({ref:e},d),{},{components:t})):n.createElement(k,l({ref:e},d))}));function c(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=m;var u={};for(var p in e)hasOwnProperty.call(e,p)&&(u[p]=e[p]);u.originalType=a,u.mdxType="string"==typeof a?a:i,l[1]=u;for(var s=2;s{t.r(e),t.d(e,{assets:()=>p,contentTitle:()=>l,default:()=>o,frontMatter:()=>r,metadata:()=>u,toc:()=>s});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:23,title:"A.23. Struct",sidebar_label:"A.23. Struct"},l=void 0,u={unversionedId:"basic/struct",id:"basic/struct",title:"A.23. Struct",description:"Pada chapter ini kita akan belajar tentang struct.",source:"@site/docs/basic/struct.md",sourceDirName:"basic",slug:"/basic/struct",permalink:"/basic/struct",draft:!1,tags:[],version:"current",sidebarPosition:23,frontMatter:{sidebar_position:23,title:"A.23. Struct",sidebar_label:"A.23. Struct"},sidebar:"tutorialSidebar",previous:{title:"A.22. Module System \u279c Scope & Akses Item",permalink:"/basic/module-scope-item-access"},next:{title:"A.24. Associated Function",permalink:"/basic/associated-function"}},p={},s=[{value:"A.23.1. Konsep Struct",id:"a231-konsep-struct",level:2},{value:"A.23.2. Pembuatan struct",id:"a232-pembuatan-struct",level:2},{value:"\u25c9 Fungsi String::from()",id:"-fungsi-stringfrom",level:3},{value:"\u25c9 Naming convention struct",id:"-naming-convention-struct",level:3},{value:"A.23.3. Mutable struct",id:"a233-mutable-struct",level:2},{value:"A.23.4. Macam-macam notasi deklarasi variabel struct",id:"a234-macam-macam-notasi-deklarasi-variabel-struct",level:2},{value:"\u25c9 type inference / manifest typing",id:"-type-inference--manifest-typing",level:3},{value:"\u25c9 Variabel struct tanpa predefined value",id:"-variabel-struct-tanpa-predefined-value",level:3},{value:"\u25c9 Variabel struct dengan nilai berasal dari struct lain",id:"-variabel-struct-dengan-nilai-berasal-dari-struct-lain",level:3},{value:"\u25c9 Field init shorthand",id:"-field-init-shorthand",level:3},{value:"\u25c9 Deklarasi nilai struct secara horizontal",id:"-deklarasi-nilai-struct-secara-horizontal",level:3},{value:"\u25c9 Destructuring assignment",id:"-destructuring-assignment",level:3},{value:"A.23.5. Unit-like structs",id:"a235-unit-like-structs",level:2},{value:"A.23.6. Debugging value struct menggunakan #[derive(Debug)]",id:"a236-debugging-value-struct-menggunakan-derivedebug",level:2},{value:"A.23.7. Tuple struct",id:"a237-tuple-struct",level:2},{value:"A.23.8. Struct property visibility",id:"a238-struct-property-visibility",level:2},{value:"A.23.9. Tuple struct property visibility",id:"a239-tuple-struct-property-visibility",level:2},{value:"A.23.10. Generic pada struct",id:"a2310-generic-pada-struct",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:s};function o(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},d,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tentang struct."),(0,i.kt)("h2",{id:"a231-konsep-struct"},"A.23.1. Konsep Struct"),(0,i.kt)("p",null,"Struct (kependekan dari ",(0,i.kt)("em",{parentName:"p"},"structure"),") adalah tipe data custom yang dengannya kita bisa mengumpulkan beberapa definisi tipe data lalu menjadikannya sebagai satu buah tipe data dalam struktur tertentu."),(0,i.kt)("p",null,"Contoh analogi seperti sebuah mobil. Mobil memiliki roda, mesin, tempat kemudi, dan banyak lainnya. Mobil pada contoh ini adalah struct, sedang isi dari mobil tersebut biasa disebut dengan ",(0,i.kt)("em",{parentName:"p"},"attribute")," atau ",(0,i.kt)("em",{parentName:"p"},"property")," atau ",(0,i.kt)("em",{parentName:"p"},"field"),"."),(0,i.kt)("p",null,"Selain property, mobil juga bisa melakukan aksi, contohnya seperti jalan, belok kanan, berhenti. Aksi tersebut pada contoh ini adalah yang disebut dengan ",(0,i.kt)("em",{parentName:"p"},"method"),". Method sendiri adalah fungsi yang merupakan property sebuah struct."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Chapter ini fokus pada pembahasan struct beserta property-nya. Topik tentang method dibahas pada chapter selanjutnya, yaitu ",(0,i.kt)("a",{parentName:"p",href:"/basic/method"},"Method"),".")),(0,i.kt)("h2",{id:"a232-pembuatan-struct"},"A.23.2. Pembuatan struct"),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"struct")," digunakan untuk membuat custom type struct. Notasi penulisannya seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct NamaStruct {\n property_satu: String,\n property_dua: u64,\n // ...\n}\n")),(0,i.kt)("p",null,"Contoh, sebuah struct bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," yang memiliki 4 buah property:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"name")," bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"String")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"sign_in_count")," bertipe numerik ",(0,i.kt)("inlineCode",{parentName:"li"},"u64")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"affliation")," bertipe vektor ",(0,i.kt)("inlineCode",{parentName:"li"},"Vec")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"active")," bertipe boolean")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct User {\n name: String,\n sign_in_count: u64,\n affliation: Vec,\n active: bool,\n}\n")),(0,i.kt)("p",null,"Struct adalah ",(0,i.kt)("em",{parentName:"p"},"hanya definisi structure"),"-nya saja, struct tidak menampung nilai atau value. Contoh pada struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," di atas bisa dilihat bahwa yang didefinisikan hanya nama struct dan property (beserta tipe datanya). Tidak ada pengisian nilai sama sekali."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," merupakan salah satu contoh penerapan generics. Topik ini nantinya dibahas lebih detail pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics"},"Generics"),".")),(0,i.kt)("p",null,"Struct merupakan tipe data data custom, yang berarti tipe data tersebut bisa digunakan dalam pembuatan variabel. Sebagai contoh dibawah ini, sebuah variabel bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"user_one")," didefinisikan dengan tipe adalah struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," yang telah dibuat."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct User {\n name: String,\n sign_in_count: u64,\n affliation: Vec,\n active: bool,\n}\n\nfn main() {\n let user_one = User{\n name: String::from("Orgrim Doomhammer"),\n sign_in_count: 12,\n affliation: vec![\n String::from("Warchief of the Horde"),\n String::from("Blackrock Chieftain"),\n String::from("The Doomhammer"),\n ],\n active: false,\n };\n\n println!("name: {}", user_one.name);\n println!("sign-in count: {}", user_one.sign_in_count);\n println!("affliation: {:?}", user_one.affliation);\n println!("is active? {}", user_one.active);\n}\n')),(0,i.kt)("p",null,"Jalankan program di atas, lihat hasilnya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(1226).Z,width:"572",height:"123"})),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"user_one")," didefinisikan tipe datanya adalah struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User"),". Notasinya penulisannya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let user_one = User{\n // property_satu: value,\n // property_dua: value,\n // ...\n};\n")),(0,i.kt)("p",null,"Deklarasi variabel bertipe data struct mewajibkan isi masing-masing property harus dituliskan beserta nilainya. Sebagai contoh variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"user_one")," yang bertipe struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," di atas, dalam statement bisa dilihat ke-empat property milik struct ",(0,i.kt)("inlineCode",{parentName:"p"},"User")," dituliskan beserta nilainya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let user_one = User{\n name: String::from("Orgrim Doomhammer"),\n sign_in_count: 12,\n affliation: vec![\n String::from("Warchief of the Horde"),\n String::from("Blackrock Chieftain"),\n String::from("The Doomhammer"),\n ],\n active: false,\n};\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Variabel yang tipe data-nya adalah struct biasa disebut dengan ",(0,i.kt)("em",{parentName:"p"},"object")," atau ",(0,i.kt)("em",{parentName:"p"},"instance"),".")),(0,i.kt)("h3",{id:"-fungsi-stringfrom"},"\u25c9 Fungsi ",(0,i.kt)("inlineCode",{parentName:"h3"},"String::from()")),(0,i.kt)("p",null,"Pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/path-item#-stdstringstringnew"},"Module System \u279c Path & Item")," kita telah sedikit mengenal fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"String::new()")," yang kegunaannya adalah untuk membuat data string kosong bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("p",null,"Selain fungsi tersebut, ada juga fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"String::from()")," yang kegunaannya juga untuk pembuatan data string bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", tapi dengan isi ditentukan lewat argumen pemanggilan fungsi."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let a = "";\n// variabel `a` bertipe data `&str`, isi value-nya ""\n\nlet b = String::new();\n// variabel `b` bertipe data `String`, isi value-nya ""\n\nlet c = "hello";\n// variabel `c` bertipe data `&str`, isi value-nya "hello"\n\nlet d = String::from("world");\n// variabel `d` bertipe data `String`, isi value-nya "world"\n')),(0,i.kt)("p",null,"Untuk sekarang, boleh menggunakan kesimpulan berikut:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Hasil dari ",(0,i.kt)("inlineCode",{parentName:"li"},"String::new()")," adalah sama dengan literal string ",(0,i.kt)("inlineCode",{parentName:"li"},'""')," tapi bertipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," (bukan ",(0,i.kt)("inlineCode",{parentName:"li"},"&str"),")"),(0,i.kt)("li",{parentName:"ul"},"Hasil dari ",(0,i.kt)("inlineCode",{parentName:"li"},'String::from("hello")')," adalah sama dengan literal string ",(0,i.kt)("inlineCode",{parentName:"li"},'"hello"')," tapi bertipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," (bukan ",(0,i.kt)("inlineCode",{parentName:"li"},"&str"),")")),(0,i.kt)("p",null,"Sebenarnya perbedaannya tidak hanya itu saja, nantinya kita pelajari lebih detail pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type"),"."),(0,i.kt)("h3",{id:"-naming-convention-struct"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Naming convention")," struct"),(0,i.kt)("p",null,"Sesuai anjuran di ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", upper camel case digunakan dalam penamaan struct dan camel case untuk penamaan property-nya. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct User {\n name: String,\n sign_in_count: u64,\n affliation: Vec,\n active: bool,\n}\n")),(0,i.kt)("h2",{id:"a233-mutable-struct"},"A.23.3. Mutable struct"),(0,i.kt)("p",null,"Penentuan ",(0,i.kt)("em",{parentName:"p"},"mutability")," sebuah struct dilakukan dengan cara menambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," seperti pada umumnya variabel. Contohnya bisa dilihat pada variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"user_two")," berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut user_two: User = User{\n name: String::from("Varian Wrynn"),\n sign_in_count: 12,\n affliation: vec![\n String::from("High King of the Alliance"),\n String::from("King of Stormwind"),\n String::from("Champion of the Crimson Ring"),\n ],\n active: false,\n};\n\nuser_two.name = String::from("Anduin Wrynn");\nuser_two.affliation.pop();\nuser_two.active = true;\n\nprintln!("name: {}", user_two.name);\nprintln!("sign-in count: {}", user_two.sign_in_count);\nprintln!("affliation: {:?}", user_two.affliation);\nprintln!("is active? {}", user_two.active);\n')),(0,i.kt)("p",null,"Pada contoh di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"user_two")," adalah object struct yang bisa diubah nilainya (mutable). Perubahan nilai struct bisa dilakukan pada property, contohnya seperti property ",(0,i.kt)("inlineCode",{parentName:"p"},"name"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"affliation"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"active")," yang dicontohkan diubah nilainya."),(0,i.kt)("p",null,"Atau bisa juga perubahannya pada value struct itu sendiri, diganti dengan value baru."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut user_two = User{\n name: String::from("Varian Wrynn"),\n sign_in_count: 12,\n affliation: vec![\n String::from("High King of the Alliance"),\n String::from("King of Stormwind"),\n String::from("Champion of the Crimson Ring"),\n ],\n active: false,\n};\n\nuser_two = User{\n name: String::from("Anduin Wrynn"),\n sign_in_count: 12,\n affliation: vec![\n String::from("High King of the Alliance"),\n String::from("King of Stormwind"),\n ],\n active: true,\n};\n')),(0,i.kt)("h2",{id:"a234-macam-macam-notasi-deklarasi-variabel-struct"},"A.23.4. Macam-macam notasi deklarasi variabel struct"),(0,i.kt)("h3",{id:"-type-inference--manifest-typing"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"type inference")," / ",(0,i.kt)("em",{parentName:"h3"},"manifest typing")),(0,i.kt)("p",null,"Metode deklarasi ",(0,i.kt)("em",{parentName:"p"},"type inference")," ataupun ",(0,i.kt)("em",{parentName:"p"},"manifest typing")," bisa digunakan dalam variabel struct."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct Car {\n brand: String,\n model: String,\n}\n\nlet car_one = Car{\n brand: String::from("Toyota"),\n model: String::from("Sprinter Trueno AE86"),\n};\n\nlet car_two: Car = Car{\n brand: String::from("BMW"),\n model: String::from("M3 GTR"),\n};\n')),(0,i.kt)("h3",{id:"-variabel-struct-tanpa-predefined-value"},"\u25c9 Variabel struct tanpa ",(0,i.kt)("em",{parentName:"h3"},"predefined value")),(0,i.kt)("p",null,"Variabel struct boleh didefinisikan tanpa ",(0,i.kt)("em",{parentName:"p"},"predefined value"),", jadi cukup tipe datanya saja yang ditentukan saat deklarasi variabel. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut car_three: Car;\ncar_three = Car{\n brand: String::from("Audi"),\n model: String::from("Le Mans Quattro"),\n};\n')),(0,i.kt)("h3",{id:"-variabel-struct-dengan-nilai-berasal-dari-struct-lain"},"\u25c9 Variabel struct dengan nilai berasal dari struct lain"),(0,i.kt)("p",null,"Jika ada kebutuhan untuk membuat variabel object struct yang nilai property-nya sebagian adalah berasal dari variabel struct lain, bisa manfaatkan syntax ",(0,i.kt)("inlineCode",{parentName:"p"},".."),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut car_three: Car;\ncar_three = Car{\n brand: String::from("Audi"),\n model: String::from("Le Mans Quattro"),\n};\nprintln!("{} {}", car_three.brand, car_three.model);\n\nlet mut car_four: Car;\ncar_four = Car{\n brand: String::from("Audi Brand"),\n ..car_three\n};\nprintln!("{} {}", car_four.brand, car_four.model);\n')),(0,i.kt)("p",null,"Pada contoh di atas, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"car_four")," property ",(0,i.kt)("inlineCode",{parentName:"p"},"brand"),"-nya diisi dengan ",(0,i.kt)("inlineCode",{parentName:"p"},'"Audi Brand"'),". Sedangkan property-property lainnya nilai didapat dari nilai propery milik ",(0,i.kt)("inlineCode",{parentName:"p"},"car_three"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(341).Z,width:"511",height:"108"})),(0,i.kt)("p",null,"Coba jalankan program, akan terlihat nilai dari variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"car_four")," untuk property selain ",(0,i.kt)("inlineCode",{parentName:"p"},"brand")," adalah sama dengan nilai property-property ",(0,i.kt)("inlineCode",{parentName:"p"},"car_three"),"."),(0,i.kt)("h3",{id:"-field-init-shorthand"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Field init shorthand")),(0,i.kt)("p",null,"Jika ada variabel yang namanya sama persis dengan nama property sebuah struct, maka dalam deklarasi variabel struct bisa menggunakan teknik penulisan seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let model = String::from("Corvette C1");\n\nlet car_five = Car{\n brand: String::from("Chevrolet"),\n model,\n};\n')),(0,i.kt)("p",null,"Cukup tulis nama variabelnya saja tanpa value."),(0,i.kt)("p",null,"Teknik ",(0,i.kt)("em",{parentName:"p"},"shorthand")," ini bisa juga digunakan dalam fungsi. Contoh dibawah ini ada fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"new_car")," yang memiliki nama parameter adalah sama persis dengan nama property struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Car"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn new_car(brand: String, model: String) -> Car {\n Car{\n brand,\n model,\n }\n}\n\nfn main() {\n let car_six = new_car(\n String::from("Chevrolet"),\n String::from("Corvette C6")\n );\n // ...\n}\n')),(0,i.kt)("h3",{id:"-deklarasi-nilai-struct-secara-horizontal"},"\u25c9 Deklarasi nilai struct secara horizontal"),(0,i.kt)("p",null,"Umumnya, operasi ",(0,i.kt)("em",{parentName:"p"},"assignment")," nilai struct dituliskan secara vertikal. Untuk struct yang property-nya sedikit biasanya dituliskan secara horizontal contohnya seperti di bawah ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct Point {\n x: f32,\n y: f32,\n}\n\nlet point_one = Point { x: 3.14, y: 8.0 };\n")),(0,i.kt)("h3",{id:"-destructuring-assignment"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Destructuring assignment")),(0,i.kt)("p",null,"Teknik penulisan ini bisa dipakai dalam case dimana nilai property struct perlu ditampung ke variabel baru. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let point_one = Point { x: 3.14, y: 8.0 };\n\nlet Point { x: x_one, y: y_one } = point_one;\nprintln!("x_one: {}", x_one);\nprintln!("y_one: {}", y_one);\n')),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"x_one")," di atas akan menampung nilai dari ",(0,i.kt)("inlineCode",{parentName:"li"},"point_one.x")),(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"y_one")," di atas akan menampung nilai dari ",(0,i.kt)("inlineCode",{parentName:"li"},"point_one.y"))),(0,i.kt)("p",null,"Jika tidak semua property struct perlu untuk ditampung ke variabel baru, maka gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"_")," untuk property yang nilainya tidak ditampung. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let point_one = Point { x: 3.14, y: 8.0 };\n\nlet Point { x: _, y: y_one } = point_one;\nprintln!("y_one: {}", y_one);\n')),(0,i.kt)("h2",{id:"a235-unit-like-structs"},"A.23.5. ",(0,i.kt)("em",{parentName:"h2"},"Unit-like structs")),(0,i.kt)("p",null,(0,i.kt)("em",{parentName:"p"},"Unit-like structs")," adalah struct yang didefinisikan tanpa property. Cara deklrasinya bisa dilihat pada contoh berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"struct StructOne;\n\nlet data_one = StructOne;\n")),(0,i.kt)("p",null,"Teknik pembuatan struct ini berguna ketika ada case dimana ada kebutuhan untuk mengimplementasikan sebuah trait ke suatu tipe data. Lebih jelasnya akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits"},"Traits"),"."),(0,i.kt)("h2",{id:"a236-debugging-value-struct-menggunakan-derivedebug"},"A.23.6. Debugging value struct menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"#[derive(Debug)]")),(0,i.kt)("p",null,(0,i.kt)("em",{parentName:"p"},"By default"),", error akan muncul saat berusaha menampilkan nilai variabel struct (bukan nilai property-nya, tapi nilai variabel struct-nya) menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Ini disebabkan karena data yang bisa ditampilkan menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println")," harus memiliki trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," (atau ",(0,i.kt)("inlineCode",{parentName:"p"},"Display"),")."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(6091).Z,width:"743",height:"308"})),(0,i.kt)("p",null,"Lalu jika ingin melihat nilai property struct bagaimana solusinya? Bisa dengan menampilkan nilai property satu per satu, tapi pastinya butuh effort."),(0,i.kt)("p",null,"Cara yang lebih elegan adalah dengan menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"#[derive(Debug)]"),". Penerapannya cukup dengan menuliskan atribut tersebut tepat di atas deklarasi struct. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nstruct GamingConsole {\n name: String\n}\n\nlet console_one = GamingConsole{\n name: String::from("PlayStation 5"),\n};\n\nprintln!("data_struct_one: {:#?}", console_one);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(7542).Z,width:"510",height:"330"})),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih detailnya mengenai traits dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},"Lebih detailnya mengenai attribute dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/attributes"},"Attributes")))),(0,i.kt)("h2",{id:"a237-tuple-struct"},"A.23.7. Tuple struct"),(0,i.kt)("p",null,"Tuple struct adalah struct yang didefinisikan dengan gaya tuple. Property pada tuple struct diakses menggunakan notasi pengaksesan tuple item."),(0,i.kt)("p",null,"Pada contoh berikut tuple struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Color")," didefinisikan dengan isi 3 buah item yang kesemuanya bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),". Lalu tipe data tuple struct tersebut digunakan untuk membuat variabel baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"red"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct Color(i32, i32, i32);\n\nlet red = Color(255, 0, 0);\n\nprintln!("{:?} {:?} {:?}", red.0, red.1, red.2);\n')),(0,i.kt)("p",null,"Bisa dilihat di contoh, property diakses menggunakan nomor indeks dengan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"variable.index"),"."),(0,i.kt)("p",null,"Contoh lainnya, tuple struct ",(0,i.kt)("inlineCode",{parentName:"p"},"SomeTupleStruct")," berikut dideklarasikan memiliki item ke-1 bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," dan item ke-2 bertipe boolean."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct SomeTupleStruct(i32, bool);\n\nlet some_data = SomeTupleStruct(0, false);\n\nprintln!("{:?} {:?}", some_data.0, some_data.1);\n')),(0,i.kt)("h2",{id:"a238-struct-property-visibility"},"A.23.8. Struct property visibility"),(0,i.kt)("p",null,"Sekarang mari kita coba praktikkan satu program lagi, namun kali ini sedikit berbeda, struct kita definisikan sebagai module item."),(0,i.kt)("p",null,"Silakan siapkan program dengan struktur seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u251c\u2500\u2500\u2500 models.rs\n \u2514\u2500\u2500\u2500 models\n \u2514\u2500\u2500\u2500 game.rs\n")),(0,i.kt)("p",null,"Lalu isi file program dengan kode di bawah ini. Sebuah struct dengan path ",(0,i.kt)("inlineCode",{parentName:"p"},"models::game::GamingConsole"),", dan satu buah object di fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," yang dibuat dari struct tersebut. Pastikan definisi semua module item adalah publik (ada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub"),")."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod models;\n\nfn main() {\n let ps5 = models::game::GamingConsole{\n name: String::from("PS 5")\n };\n println!("{:#?}", ps5);\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"pub mod game;\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models/game.rs"',title:'"src/models/game.rs"'},"#[derive(Debug)]\npub struct GamingConsole {\n name: String\n}\n")),(0,i.kt)("p",null,"Ada yang aneh, baris kode pembuatan variabel object ",(0,i.kt)("inlineCode",{parentName:"p"},"ps5")," terdeteksi sebagai error. Padahal definisi submodule dan item-nya sudah publik semua."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(7728).Z,width:"419",height:"185"})),(0,i.kt)("p",null,"Jika dilihat dari keterangan error, sebenarnya cukup jelas bagian mana yang menjadi sumber masalah, yaitu field ",(0,i.kt)("inlineCode",{parentName:"p"},"name")," yang terdeteksi sebagai private property."),(0,i.kt)("p",null,"Struct jika didefinisikan di file yang sama dengan statement pemanggilan struct tersebut tidak akan menghasilkan error. Tetapi jika definisi struct-nya terpisah dari statement pemanggilan struct (seperti contoh di atas), maka field dari struct tersebut harus publik."),(0,i.kt)("p",null,"Caranya membuat field sebagai publik adalah dengan menambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," pada property struct. Silakan ubah definisi struct ",(0,i.kt)("inlineCode",{parentName:"p"},"GamingConsole")," menjadi seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// before\npub struct GamingConsole {\n name: String\n}\n\n// after\npub struct GamingConsole {\n pub name: String\n}\n")),(0,i.kt)("p",null,"Bisa dilihat keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," ditambahkan pada deklarasi property struct. yang sebelumnya ",(0,i.kt)("inlineCode",{parentName:"p"},"name: String")," sekarang ada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," didepannya."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai visibility property dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,i.kt)("h2",{id:"a239-tuple-struct-property-visibility"},"A.23.9. Tuple struct property visibility"),(0,i.kt)("p",null,"Lalu bagaimana dengan tuple struct? apakah property-nya juga harus didefinisikan publik agar bisa diakses dari tempat lain? Jawabannya ada di praktik berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 main.rs\n \u251c\u2500\u2500\u2500 models.rs\n \u2514\u2500\u2500\u2500 models\n \u2514\u2500\u2500\u2500 color.rs\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod models;\n\nfn main() {\n let red = models::color::Color(255, 255, 0);\n println!("{:#?}", red);\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models.rs"',title:'"src/models.rs"'},"pub mod color;\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/models/color.rs"',title:'"src/models/color.rs"'},"#[derive(Debug)]\npub struct Color(i32, i32, i32);\n")),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Struct",src:t(6610).Z,width:"626",height:"255"})),(0,i.kt)("p",null,"Yap, error yang mirip juga muncul. Jadi jawaban dari pertanyaan sebelumnya adalah iya, property tuple struct juga harus publik agar bisa diakses dari tempat lain."),(0,i.kt)("p",null,"Cara deklarasi tuple struct dengan property publik adalah dengan menambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," di masing-masing deklarasi parameter tuple struct. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// before\npub struct Color(i32, i32, i32);\n\n// after\npub struct Color(pub i32, pub i32, pub i32);\n")),(0,i.kt)("p",null,"Jalankan program untuk melihat hasilnya, error tidak muncul."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai visibility property dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,i.kt)("h2",{id:"a2310-generic-pada-struct"},"A.23.10. Generic pada struct"),(0,i.kt)("p",null,"Pembahasan mengenai generic pada struct ada pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics#a365-generics-struct"},"Generics"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/struct"},"github.com/novalagung/dasarpemrogramanrust-example/../struct")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch05-00-structs.html"},"https://doc.rust-lang.org/book/ch05-00-structs.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch05-01-defining-structs.html"},"https://doc.rust-lang.org/book/ch05-01-defining-structs.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.struct.html"},"https://doc.rust-lang.org/std/keyword.struct.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/custom_types/structs.html"},"https://doc.rust-lang.org/rust-by-example/custom_types/structs.html"))))}o.isMDXComponent=!0},1226:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},341:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},6091:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/struct-4-86fb5f564d330bc2d1cc10e13ea2d649.png"},7542:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/struct-5-5e013d5895bef6a938e2c1333ff52205.png"},7728:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},6610:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/struct-7-517c0402e96dfd5898464887fa79ec98.png"}}]); \ No newline at end of file diff --git a/assets/js/9a3629a6.035a9d13.js b/assets/js/9a3629a6.035a9d13.js new file mode 100644 index 00000000..f74e1486 --- /dev/null +++ b/assets/js/9a3629a6.035a9d13.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7865],{3905:(e,t,a)=>{a.d(t,{Zo:()=>o,kt:()=>h});var n=a(7294);function i(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function r(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,n)}return a}function l(e){for(var t=1;t=0||(i[a]=e[a]);return i}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(i[a]=e[a])}return i}var s=n.createContext({}),u=function(e){var t=n.useContext(s),a=t;return e&&(a="function"==typeof e?e(t):l(l({},t),e)),a},o=function(e){var t=u(e.components);return n.createElement(s.Provider,{value:t},e.children)},d={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},m=n.forwardRef((function(e,t){var a=e.components,i=e.mdxType,r=e.originalType,s=e.parentName,o=p(e,["components","mdxType","originalType","parentName"]),m=u(a),h=i,k=m["".concat(s,".").concat(h)]||m[h]||d[h]||r;return a?n.createElement(k,l(l({ref:t},o),{},{components:a})):n.createElement(k,l({ref:t},o))}));function h(e,t){var a=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var r=a.length,l=new Array(r);l[0]=m;var p={};for(var s in t)hasOwnProperty.call(t,s)&&(p[s]=t[s]);p.originalType=e,p.mdxType="string"==typeof e?e:i,l[1]=p;for(var u=2;u{a.r(t),a.d(t,{assets:()=>s,contentTitle:()=>l,default:()=>d,frontMatter:()=>r,metadata:()=>p,toc:()=>u});var n=a(7462),i=(a(7294),a(3905));const r={sidebar_position:51,title:"A.51. File, Path, Directory",sidebar_label:"A.51. File, Path, Directory"},l=void 0,p={unversionedId:"basic/file-path-directory",id:"basic/file-path-directory",title:"A.51. File, Path, Directory",description:"Pada chapter ini kita akan belajar tentang beberapa API milik Rust yang disediakan untuk manajemen path dan file system.",source:"@site/docs/basic/file-path-directory.md",sourceDirName:"basic",slug:"/basic/file-path-directory",permalink:"/basic/file-path-directory",draft:!1,tags:[],version:"current",sidebarPosition:51,frontMatter:{sidebar_position:51,title:"A.51. File, Path, Directory",sidebar_label:"A.51. File, Path, Directory"},sidebar:"tutorialSidebar",previous:{title:"A.50. Date Time",permalink:"/basic/datetime"}},s={},u=[{value:"A.51.1. File path (std::path::Path)",id:"a511-file-path-stdpathpath",level:2},{value:"A.52.2. Method filepath",id:"a522-method-filepath",level:2},{value:"\u25c9 Associated function Path::new()",id:"-associated-function-pathnew",level:3},{value:"\u25c9 Method join()",id:"-method-join",level:3},{value:"\u25c9 Method exists()",id:"-method-exists",level:3},{value:"\u25c9 Method is_file() & is_dir()",id:"-method-is_file--is_dir",level:3},{value:"\u25c9 Method is_absolute() & is_relative()",id:"-method-is_absolute--is_relative",level:3},{value:"A.52.3. Module std::fs (file system)",id:"a523-module-stdfs-file-system",level:2},{value:"A.52.4. Manajemen file system",id:"a524-manajemen-file-system",level:2},{value:"\u25c9 Pembuatan folder (fs::create_dir)",id:"-pembuatan-folder-fscreate_dir",level:3},{value:"\u25c9 Menulis konten file (fs::write)",id:"-menulis-konten-file-fswrite",level:3},{value:"\u25c9 Menghapus file (fs::remove_file)",id:"-menghapus-file-fsremove_file",level:3},{value:"\u25c9 Menghapus folder (fs::remove_dir)",id:"-menghapus-folder-fsremove_dir",level:3},{value:"\u25c9 List items dalam folder",id:"-list-items-dalam-folder",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:u};function d(e){let{components:t,...r}=e;return(0,i.kt)("wrapper",(0,n.Z)({},o,r,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tentang beberapa API milik Rust yang disediakan untuk manajemen path dan file system."),(0,i.kt)("h2",{id:"a511-file-path-stdpathpath"},"A.51.1. File path (",(0,i.kt)("inlineCode",{parentName:"h2"},"std::path::Path"),")"),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},'"/home/novalagung/Desktop/my text.txt"')," adalah salah satu contoh file path. Di Rust, file path bisa direpresentasikan sebagai string (seperti yang sudah dicontohkan), atau menggunakan tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Path"),"."),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Path")," tersedia dalam module ",(0,i.kt)("inlineCode",{parentName:"p"},"std::path"),". Cara penggunaannya cukup mudah, sebagai contoh, beberapa filepath berikut menghasilkan path yang sama:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let filepath_1 = "/home/novalagung/Desktop/my text.txt";\nlet filepath_2 = Path::new("/home/novalagung/Desktop/my text.txt");\nlet filepath_3 = Path::new("/home/novalagung/Desktop").join("my text.txt");\nlet filepath_4 = Path::new("/home/novalagung").join("Desktop").join("my text.txt");\nlet filepath_5 = Path::new("/home").join("novalagung/Desktop/my text.txt");\n')),(0,i.kt)("h2",{id:"a522-method-filepath"},"A.52.2. Method filepath"),(0,i.kt)("h3",{id:"-associated-function-pathnew"},"\u25c9 Associated function ",(0,i.kt)("inlineCode",{parentName:"h3"},"Path::new()")),(0,i.kt)("p",null,"Digunakan untuk membuat object ",(0,i.kt)("inlineCode",{parentName:"p"},"Path")," dari suatu string. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'Path::new("/home/novalagung/Desktop/my text.txt");\n')),(0,i.kt)("h3",{id:"-method-join"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"join()")),(0,i.kt)("p",null,"Digunakan untuk menggabungkan segment filepath. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'Path::new("/home/novalagung").join("Desktop").join("my text.txt");\n')),(0,i.kt)("p",null,"Hasil dari operasi di atas adalah path:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre"},"/home/novalagung/Desktop/my text.txt\n")),(0,i.kt)("h3",{id:"-method-exists"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"exists()")),(0,i.kt)("p",null,"Digunakan untuk mengecek apakah suatu filepath ada atau tidak."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).exists() {\n // path exists\n}\n")),(0,i.kt)("h3",{id:"-method-is_file--is_dir"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_file()")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_dir()")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_file()")," digunakan untuk mengecek apakah suatu filepath berisi file."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).is_file() {\n // path is a file\n}\n")),(0,i.kt)("p",null,"Sedangkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_dir()")," digunakan untuk mengecek apakah suatu filepath adalah folder/directory."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).is_dir() {\n // path contains directory\n}\n")),(0,i.kt)("h3",{id:"-method-is_absolute--is_relative"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_absolute()")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_relative()")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_absolute()")," digunakan untuk mengecek apakah suatu filepath adalah ",(0,i.kt)("em",{parentName:"p"},"absolute path"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).is_absolute() {\n // path is an absolute path\n}\n")),(0,i.kt)("p",null,"Sedangkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_relative()")," digunakan untuk mengecek apakah suatu filepath adalah ",(0,i.kt)("em",{parentName:"p"},"relative path"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).is_relative() {\n // path is a relative path\n}\n")),(0,i.kt)("h2",{id:"a523-module-stdfs-file-system"},"A.52.3. Module ",(0,i.kt)("inlineCode",{parentName:"h2"},"std::fs")," (file system)"),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"std::fs")," merupakan module yang disediakan Rust untuk pengolahan file system. Di dalamnya berisinya banyak sekali fungsi untuk keperluan seperti pembuatan file, modifikasi konten file, dan lainnya."),(0,i.kt)("p",null,"Sebagai contoh, untuk membuat suatu directory bisa menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::create_dir"),". Isi argument pemanggilan fungsi dengan path dalam bentuk string (atau ",(0,i.kt)("inlineCode",{parentName:"p"},"std::path::Path")," juga boleh)."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = "./files";\nlet res = fs::create_dir(path);\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::create_dir")," mengembalikan value bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result<(), Error>"),". Gunakan pattern matching ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk mengecek hasil operasi apakah sukses atau tidak."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'match fs::create_dir("./files") {\n Err(err) => {\n println!("error on creating directory! {}", err);\n return\n },\n _ => {},\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih detailnya mengenai tipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"Result")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/result-type"},"Tipe Data \u279c Result"),"."),(0,i.kt)("li",{parentName:"ul"},"Lebih detailnya mengenai pattern matching dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching"),"."))),(0,i.kt)("h2",{id:"a524-manajemen-file-system"},"A.52.4. Manajemen file system"),(0,i.kt)("h3",{id:"-pembuatan-folder-fscreate_dir"},"\u25c9 Pembuatan folder (",(0,i.kt)("inlineCode",{parentName:"h3"},"fs::create_dir"),")"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::create_dir")," digunakan untuk membuat folder. Contoh penerapannya sudah dibahas di sesi ",(0,i.kt)("a",{parentName:"p",href:"/basic/file-path-directory#a523-module-stdfs-file-system"},"A.52.3. Module std::fs (file system)")," di atas."),(0,i.kt)("h3",{id:"-menulis-konten-file-fswrite"},"\u25c9 Menulis konten file (",(0,i.kt)("inlineCode",{parentName:"h3"},"fs::write"),")"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::write")," digunakan untuk membuat folder. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = Path::new("./files").join("target.txt");\nlet content = "hello rust!";\nlet res = fs::write(path, content);\n\nmatch res {\n Err(err) => {\n println!("error on writing file {}! {}", path.to_str().unwrap_or_default(), err);\n return\n },\n _ => {},\n}\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::write")," melakukan penulisan konten pada variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"content")," ke path ",(0,i.kt)("inlineCode",{parentName:"p"},"path"),". Mode penulisannya adalah ",(0,i.kt)("em",{parentName:"p"},"overwrite")," (bukan ",(0,i.kt)("em",{parentName:"p"},"append"),"), yang artinya konten lama pada filepath akan di-",(0,i.kt)("em",{parentName:"p"},"replace")," total dengan konten baru."),(0,i.kt)("p",null,"Jika file tidak ada pada ",(0,i.kt)("inlineCode",{parentName:"p"},"path")," tujuan, maka otomatis dibuatkan file baru. Namun jika folder/directory di mana file akan dibuat/ditulis tidak ada, maka muncul error."),(0,i.kt)("h3",{id:"-menghapus-file-fsremove_file"},"\u25c9 Menghapus file (",(0,i.kt)("inlineCode",{parentName:"h3"},"fs::remove_file"),")"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::remove_file")," digunakan untuk menghapus file pada suatu path. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = Path::new("./files").join("target.txt");\nlet res = fs::remove_file(path);\n\nmatch res {\n Err(err) => {\n println!("error on deleting file {}! {}", path.to_str().unwrap_or_default(), err);\n return\n },\n _ => {},\n}\n')),(0,i.kt)("h3",{id:"-menghapus-folder-fsremove_dir"},"\u25c9 Menghapus folder (",(0,i.kt)("inlineCode",{parentName:"h3"},"fs::remove_dir"),")"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::remove_dir")," digunakan untuk menghapus folder/directory. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = Path::new("./files");\nlet res = fs::remove_dir(path);\n\nmatch res {\n Err(err) => {\n println!("error on deleting directory {}! {}", path.to_str().unwrap_or_default(), err);\n return\n },\n _ => {},\n}\n')),(0,i.kt)("h3",{id:"-list-items-dalam-folder"},"\u25c9 List items dalam folder"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::read_dir")," digunakan untuk menampilkan list items suatu folder (baik subfolder ataupun file). Fungsi ini mengmbalikan nilai bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),", untuk mengakses list items bisa menggunakan teknik pattern matching, atau boleh juga langsung di ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap()")," (dengan resiko program bisa panic ketika ada error)."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = Path::new("D:\\\\Labs\\\\Adam Studio\\\\Ebook\\\\dasarpemrogramanrust\\\\file_path_directory_1");\nlet paths = fs::read_dir(path).unwrap();\n\nfor path in paths {\n let item = path.unwrap();\n println!("file name: {:?}, file path: {:?}", item.file_name(), item.path().display())\n}\n')),(0,i.kt)("p",null,"Pada contoh di atas, penulis menggunakan testing path yang isinya 4 buah item:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"File, Path, Directory",src:a(2944).Z,width:"595",height:"176"})),(0,i.kt)("p",null,"Output program bisa dilihat pada gambar berikut:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"File, Path, Directory",src:a(214).Z,width:"887",height:"149"})),(0,i.kt)("p",null,"Tipe data variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"paths")," di atas adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"ReadDir")," yang merupakan tipe data iterator. Jika digunakan pada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for"),", maka variabel element perulangan ",(0,i.kt)("inlineCode",{parentName:"p"},"path")," akan bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Dari tipe ini kita bisa mengakses file name maupun file path."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/file_path_directory"},"github.com/novalagung/dasarpemrogramanrust-example/../file_path_directory")),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang append content file (",(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/30684624/what-is-the-best-variant-for-appending-a-new-line-in-a-text-file"},"https://stackoverflow.com/questions/30684624/what-is-the-best-variant-for-appending-a-new-line-in-a-text-file"),")")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/result-type"},"Tipe Data \u279c Result")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/path/struct.Path.html"},"https://doc.rust-lang.org/std/path/struct.Path.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/std_misc/path.html"},"https://doc.rust-lang.org/rust-by-example/std_misc/path.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/std_misc/fs.html"},"https://doc.rust-lang.org/rust-by-example/std_misc/fs.html"))))}d.isMDXComponent=!0},2944:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},214:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/file-path-directory-2-bbe0bb5499225e418b251930c44fb8f6.png"}}]); \ No newline at end of file diff --git a/assets/js/9a3629a6.fcabd2cf.js b/assets/js/9a3629a6.fcabd2cf.js deleted file mode 100644 index 7b7954f5..00000000 --- a/assets/js/9a3629a6.fcabd2cf.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7865],{3905:(e,t,a)=>{a.d(t,{Zo:()=>o,kt:()=>h});var n=a(7294);function i(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function r(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,n)}return a}function l(e){for(var t=1;t=0||(i[a]=e[a]);return i}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(i[a]=e[a])}return i}var s=n.createContext({}),u=function(e){var t=n.useContext(s),a=t;return e&&(a="function"==typeof e?e(t):l(l({},t),e)),a},o=function(e){var t=u(e.components);return n.createElement(s.Provider,{value:t},e.children)},d={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},m=n.forwardRef((function(e,t){var a=e.components,i=e.mdxType,r=e.originalType,s=e.parentName,o=p(e,["components","mdxType","originalType","parentName"]),m=u(a),h=i,k=m["".concat(s,".").concat(h)]||m[h]||d[h]||r;return a?n.createElement(k,l(l({ref:t},o),{},{components:a})):n.createElement(k,l({ref:t},o))}));function h(e,t){var a=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var r=a.length,l=new Array(r);l[0]=m;var p={};for(var s in t)hasOwnProperty.call(t,s)&&(p[s]=t[s]);p.originalType=e,p.mdxType="string"==typeof e?e:i,l[1]=p;for(var u=2;u{a.r(t),a.d(t,{assets:()=>s,contentTitle:()=>l,default:()=>d,frontMatter:()=>r,metadata:()=>p,toc:()=>u});var n=a(7462),i=(a(7294),a(3905));const r={sidebar_position:51,title:"A.51. File, Path, Directory",sidebar_label:"A.51. File, Path, Directory"},l=void 0,p={unversionedId:"basic/file-path-directory",id:"basic/file-path-directory",title:"A.51. File, Path, Directory",description:"Pada chapter ini kita akan belajar tentang beberapa API milik Rust yang disediakan untuk manajemen path dan file system.",source:"@site/docs/basic/file-path-directory.md",sourceDirName:"basic",slug:"/basic/file-path-directory",permalink:"/basic/file-path-directory",draft:!1,tags:[],version:"current",sidebarPosition:51,frontMatter:{sidebar_position:51,title:"A.51. File, Path, Directory",sidebar_label:"A.51. File, Path, Directory"},sidebar:"tutorialSidebar",previous:{title:"A.50. Date Time",permalink:"/basic/datetime"}},s={},u=[{value:"A.51.1. File path (std::path::Path)",id:"a511-file-path-stdpathpath",level:2},{value:"A.52.2. Method filepath",id:"a522-method-filepath",level:2},{value:"\u25c9 Associated function Path::new()",id:"-associated-function-pathnew",level:3},{value:"\u25c9 Method join()",id:"-method-join",level:3},{value:"\u25c9 Method exists()",id:"-method-exists",level:3},{value:"\u25c9 Method is_file() & is_dir()",id:"-method-is_file--is_dir",level:3},{value:"\u25c9 Method is_absolute() & is_relative()",id:"-method-is_absolute--is_relative",level:3},{value:"A.52.3. Module std::fs (file system)",id:"a523-module-stdfs-file-system",level:2},{value:"A.52.4. Manajemen file system",id:"a524-manajemen-file-system",level:2},{value:"\u25c9 Pembuatan folder (fs::create_dir)",id:"-pembuatan-folder-fscreate_dir",level:3},{value:"\u25c9 Menulis konten file (fs::write)",id:"-menulis-konten-file-fswrite",level:3},{value:"\u25c9 Menghapus file (fs::remove_file)",id:"-menghapus-file-fsremove_file",level:3},{value:"\u25c9 Menghapus folder (fs::remove_dir)",id:"-menghapus-folder-fsremove_dir",level:3},{value:"\u25c9 List items dalam folder",id:"-list-items-dalam-folder",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:u};function d(e){let{components:t,...r}=e;return(0,i.kt)("wrapper",(0,n.Z)({},o,r,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tentang beberapa API milik Rust yang disediakan untuk manajemen path dan file system."),(0,i.kt)("h2",{id:"a511-file-path-stdpathpath"},"A.51.1. File path (",(0,i.kt)("inlineCode",{parentName:"h2"},"std::path::Path"),")"),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},'"/home/novalagung/Desktop/my text.txt"')," adalah salah satu contoh file path. Di Rust, file path bisa direpresentasikan sebagai string (seperti yang sudah dicontohkan), atau menggunakan tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Path"),"."),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Path")," tersedia dalam module ",(0,i.kt)("inlineCode",{parentName:"p"},"std::path"),". Cara penggunaannya cukup mudah, sebagai contoh, beberapa filepath berikut menghasilkan path yang sama:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let filepath_1 = "/home/novalagung/Desktop/my text.txt";\nlet filepath_2 = Path::new("/home/novalagung/Desktop/my text.txt");\nlet filepath_3 = Path::new("/home/novalagung/Desktop").join("my text.txt");\nlet filepath_4 = Path::new("/home/novalagung").join("Desktop").join("my text.txt");\nlet filepath_5 = Path::new("/home").join("novalagung/Desktop/my text.txt");\n')),(0,i.kt)("h2",{id:"a522-method-filepath"},"A.52.2. Method filepath"),(0,i.kt)("h3",{id:"-associated-function-pathnew"},"\u25c9 Associated function ",(0,i.kt)("inlineCode",{parentName:"h3"},"Path::new()")),(0,i.kt)("p",null,"Digunakan untuk membuat object ",(0,i.kt)("inlineCode",{parentName:"p"},"Path")," dari suatu string. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'Path::new("/home/novalagung/Desktop/my text.txt");\n')),(0,i.kt)("h3",{id:"-method-join"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"join()")),(0,i.kt)("p",null,"Digunakan untuk menggabungkan segment filepath. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'Path::new("/home/novalagung").join("Desktop").join("my text.txt");\n')),(0,i.kt)("p",null,"Hasil dari operasi di atas adalah path:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre"},"/home/novalagung/Desktop/my text.txt\n")),(0,i.kt)("h3",{id:"-method-exists"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"exists()")),(0,i.kt)("p",null,"Digunakan untuk mengecek apakah suatu filepath ada atau tidak."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).exists() {\n // path exists\n}\n")),(0,i.kt)("h3",{id:"-method-is_file--is_dir"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_file()")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_dir()")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_file()")," digunakan untuk mengecek apakah suatu filepath berisi file."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).is_file() {\n // path is a file\n}\n")),(0,i.kt)("p",null,"Sedangkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_dir()")," digunakan untuk mengecek apakah suatu filepath adalah folder/directory."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).is_dir() {\n // path contains directory\n}\n")),(0,i.kt)("h3",{id:"-method-is_absolute--is_relative"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_absolute()")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_relative()")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_absolute()")," digunakan untuk mengecek apakah suatu filepath adalah ",(0,i.kt)("em",{parentName:"p"},"absolute path"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).is_absolute() {\n // path is an absolute path\n}\n")),(0,i.kt)("p",null,"Sedangkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_relative()")," digunakan untuk mengecek apakah suatu filepath adalah ",(0,i.kt)("em",{parentName:"p"},"relative path"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"if Path::new(path).is_relative() {\n // path is a relative path\n}\n")),(0,i.kt)("h2",{id:"a523-module-stdfs-file-system"},"A.52.3. Module ",(0,i.kt)("inlineCode",{parentName:"h2"},"std::fs")," (file system)"),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"std::fs")," merupakan module yang disediakan Rust untuk pengolahan file system. Didalamnya berisinya banyak sekali fungsi untuk keperluan seperti pembuatan file, modifikasi konten file, dan lainnya."),(0,i.kt)("p",null,"Sebagai contoh, untuk membuat suatu directory bisa menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::create_dir"),". Isi argument pemanggilan fungsi dengan path dalam bentuk string (atau ",(0,i.kt)("inlineCode",{parentName:"p"},"std::path::Path")," juga boleh)."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = "./files";\nlet res = fs::create_dir(path);\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::create_dir")," mengembalikan value bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result<(), Error>"),". Gunakan pattern matching ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk mengecek hasil operasi apakah sukses atau tidak."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'match fs::create_dir("./files") {\n Err(err) => {\n println!("error on creating directory! {}", err);\n return\n },\n _ => {},\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih detailnya mengenai tipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"Result")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/result-type"},"Tipe Data \u279c Result"),"."),(0,i.kt)("li",{parentName:"ul"},"Lebih detailnya mengenai pattern matching dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching"),"."))),(0,i.kt)("h2",{id:"a524-manajemen-file-system"},"A.52.4. Manajemen file system"),(0,i.kt)("h3",{id:"-pembuatan-folder-fscreate_dir"},"\u25c9 Pembuatan folder (",(0,i.kt)("inlineCode",{parentName:"h3"},"fs::create_dir"),")"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::create_dir")," digunakan untuk membuat folder. Contoh penerapannya sudah dibahas di sesi ",(0,i.kt)("a",{parentName:"p",href:"/basic/file-path-directory#a523-module-stdfs-file-system"},"A.52.3. Module std::fs (file system)")," di atas."),(0,i.kt)("h3",{id:"-menulis-konten-file-fswrite"},"\u25c9 Menulis konten file (",(0,i.kt)("inlineCode",{parentName:"h3"},"fs::write"),")"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::write")," digunakan untuk membuat folder. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = Path::new("./files").join("target.txt");\nlet content = "hello rust!";\nlet res = fs::write(path, content);\n\nmatch res {\n Err(err) => {\n println!("error on writing file {}! {}", path.to_str().unwrap_or_default(), err);\n return\n },\n _ => {},\n}\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::write")," melakukan penulisan konten pada variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"content")," ke path ",(0,i.kt)("inlineCode",{parentName:"p"},"path"),". Mode penulisannya adalah ",(0,i.kt)("em",{parentName:"p"},"overwrite")," (bukan ",(0,i.kt)("em",{parentName:"p"},"append"),"), yang artinya konten lama pada filepath akan di-",(0,i.kt)("em",{parentName:"p"},"replace")," total dengan konten baru."),(0,i.kt)("p",null,"Jika file tidak ada pada ",(0,i.kt)("inlineCode",{parentName:"p"},"path")," tujuan, maka otomatis dibuatkan file baru. Namun jika folder/directory dimana file akan dibuat/ditulis tidak ada, maka muncul error."),(0,i.kt)("h3",{id:"-menghapus-file-fsremove_file"},"\u25c9 Menghapus file (",(0,i.kt)("inlineCode",{parentName:"h3"},"fs::remove_file"),")"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::remove_file")," digunakan untuk menghapus file pada suatu path. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = Path::new("./files").join("target.txt");\nlet res = fs::remove_file(path);\n\nmatch res {\n Err(err) => {\n println!("error on deleting file {}! {}", path.to_str().unwrap_or_default(), err);\n return\n },\n _ => {},\n}\n')),(0,i.kt)("h3",{id:"-menghapus-folder-fsremove_dir"},"\u25c9 Menghapus folder (",(0,i.kt)("inlineCode",{parentName:"h3"},"fs::remove_dir"),")"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::remove_dir")," digunakan untuk menghapus folder/directory. Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = Path::new("./files");\nlet res = fs::remove_dir(path);\n\nmatch res {\n Err(err) => {\n println!("error on deleting directory {}! {}", path.to_str().unwrap_or_default(), err);\n return\n },\n _ => {},\n}\n')),(0,i.kt)("h3",{id:"-list-items-dalam-folder"},"\u25c9 List items dalam folder"),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"fs::read_dir")," digunakan untuk menampilkan list items suatu folder (baik subfolder ataupun file). Fungsi ini mengmbalikan nilai bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),", untuk mengakses list items bisa menggunakan teknik pattern matching, atau boleh juga langsung di ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap()")," (dengan resiko program bisa panic ketika ada error)."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let path = Path::new("D:\\\\Labs\\\\Adam Studio\\\\Ebook\\\\dasarpemrogramanrust\\\\file_path_directory_1");\nlet paths = fs::read_dir(path).unwrap();\n\nfor path in paths {\n let item = path.unwrap();\n println!("file name: {:?}, file path: {:?}", item.file_name(), item.path().display())\n}\n')),(0,i.kt)("p",null,"Pada contoh di atas, penulis menggunakan testing path yang isinya 4 buah item:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"File, Path, Directory",src:a(2944).Z,width:"595",height:"176"})),(0,i.kt)("p",null,"Output program bisa dilihat pada gambar berikut:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"File, Path, Directory",src:a(214).Z,width:"887",height:"149"})),(0,i.kt)("p",null,"Tipe data variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"paths")," di atas adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"ReadDir")," yang merupakan tipe data iterator. Jika digunakan pada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for"),", maka variabel element perulangan ",(0,i.kt)("inlineCode",{parentName:"p"},"path")," akan bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Dari tipe ini kita bisa mengakses file name maupun file path."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/file_path_directory"},"github.com/novalagung/dasarpemrogramanrust-example/../file_path_directory")),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang append content file (",(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/30684624/what-is-the-best-variant-for-appending-a-new-line-in-a-text-file"},"https://stackoverflow.com/questions/30684624/what-is-the-best-variant-for-appending-a-new-line-in-a-text-file"),")")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/result-type"},"Tipe Data \u279c Result")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/path/struct.Path.html"},"https://doc.rust-lang.org/std/path/struct.Path.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/std_misc/path.html"},"https://doc.rust-lang.org/rust-by-example/std_misc/path.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/std_misc/fs.html"},"https://doc.rust-lang.org/rust-by-example/std_misc/fs.html"))))}d.isMDXComponent=!0},2944:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},214:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/file-path-directory-2-bbe0bb5499225e418b251930c44fb8f6.png"}}]); \ No newline at end of file diff --git a/assets/js/a2bea3ed.53e03a1b.js b/assets/js/a2bea3ed.53e03a1b.js new file mode 100644 index 00000000..aea9a296 --- /dev/null +++ b/assets/js/a2bea3ed.53e03a1b.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6345],{3905:(a,e,t)=>{t.d(e,{Zo:()=>o,kt:()=>u});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var p=n.createContext({}),d=function(a){var e=n.useContext(p),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},o=function(a){var e=d(a.components);return n.createElement(p.Provider,{value:e},a.children)},m={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},k=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,p=a.parentName,o=s(a,["components","mdxType","originalType","parentName"]),k=d(t),u=i,g=k["".concat(p,".").concat(u)]||k[u]||m[u]||r;return t?n.createElement(g,l(l({ref:e},o),{},{components:t})):n.createElement(g,l({ref:e},o))}));function u(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=k;var s={};for(var p in e)hasOwnProperty.call(e,p)&&(s[p]=e[p]);s.originalType=a,s.mdxType="string"==typeof a?a:i,l[1]=s;for(var d=2;d{t.r(e),t.d(e,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>r,metadata:()=>s,toc:()=>d});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:44,title:"A.44. String Literal (&str) vs. String Custom Type",sidebar_label:"A.44. String Literal (&str) vs. String Custom Type"},l=void 0,s={unversionedId:"basic/string-slice-vs-string-literal",id:"basic/string-slice-vs-string-literal",title:"A.44. String Literal (&str) vs. String Custom Type",description:"Pada chapter sebelumnya kita telah membahas tentang bagaimana data slice di-manage di memory. Ada beberapa tipe data yang masuk dalam kategori slice, yang salah satunya adalah string slice atau String.",source:"@site/docs/basic/string-slice-vs-string-literal.md",sourceDirName:"basic",slug:"/basic/string-slice-vs-string-literal",permalink:"/basic/string-slice-vs-string-literal",draft:!1,tags:[],version:"current",sidebarPosition:44,frontMatter:{sidebar_position:44,title:"A.44. String Literal (&str) vs. String Custom Type",sidebar_label:"A.44. String Literal (&str) vs. String Custom Type"},sidebar:"tutorialSidebar",previous:{title:"A.43. Slice Memory Management",permalink:"/basic/slice-memory-management"},next:{title:"A.45. Tipe Data \u279c String Custom Type",permalink:"/basic/tipe-data-custom-type-string-slice"}},p={},d=[{value:"A.44.1. String slice (String)",id:"a441-string-slice-string",level:2},{value:"A.44.2. String literal (&str)",id:"a442-string-literal-str",level:2},{value:"A.44.3. Konversi data string",id:"a443-konversi-data-string",level:2},{value:"\u25c9 Konversi String ke &str",id:"-konversi-string-ke-str",level:3},{value:"\u25c9 Konversi &str ke String",id:"-konversi-str-ke-string",level:3},{value:"A.44.4. String literal & string slice",id:"a444-string-literal--string-slice",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:d};function m(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},o,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter sebelumnya kita telah membahas tentang bagaimana data slice di-manage di memory. Ada beberapa tipe data yang masuk dalam kategori slice, yang salah satunya adalah string slice atau ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("p",null,"Di chapter ini kita akan bahas apa perbedaan antara tipe data string slice (",(0,i.kt)("inlineCode",{parentName:"p"},"String"),") dan string literal ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("p",null,"Pembahasan mengenai topik ini sengaja dilakukan tidak di awal-awal ebook, karena ada banyak hal yang perlu dipahami sebelum mempelajarinya, contohnya seperti aspek management memory dan ownership. Dan karena topik tersebut sudah selesai dibahas, berarti ini adalah waktu yang tepat untuk membahas string slice."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Silakan pelajari kembali pembahasan detail tentang tipe slice pada chapter sebelumnya jika diperlukan. Chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/slice-memory-management"},"Slice Memory Management"))),(0,i.kt)("h2",{id:"a441-string-slice-string"},"A.44.1. String slice (",(0,i.kt)("inlineCode",{parentName:"h2"},"String"),")"),(0,i.kt)("p",null,"String slice atau custom type ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," merupakan tipe data bawaan Rust, dibuat via ",(0,i.kt)("inlineCode",{parentName:"p"},"struct"),", kegunaannya untuk menampung data UTF-8 bytes yang dinamis (bisa berkembang isinya)."),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"String")," masuk dalam kategori tipe data slice, isinya adalah data kolektif bertipe bytes, datanya disimpan di heap memory, dan metadata-nya di stack memory. Tipe data ini dikategorikan sebagai tipe data ",(0,i.kt)("strong",{parentName:"p"},"owned"),", yang artinya owner data bisa direpresentasikan oleh variabel. Sebagai contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str1 = String::from("Lisa Blackpink");\nprintln!("str1: {str1}");\n')),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"str1")," di atas merupakan owner dari string ",(0,i.kt)("inlineCode",{parentName:"p"},"Lisa Blackpink"),". Dari string tersebut operasi borrow bisa dilakukan. Contohnya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str1 = String::from("Lisa Blackpink");\nlet slice1 = &str1[..]; // "Lisa Blackpink"\nlet slice2 = &str1[4..7]; // " Bl"\n')),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"slice1")," adalah variabel baru yang datanya didapat dari borrowing seluruh elemen string milik ",(0,i.kt)("inlineCode",{parentName:"p"},"str1"),". Sedangkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"slice2")," hanya meminjam elemen indeks ke-4 hingga elemen sebelum 7 (yaitu 6)."),(0,i.kt)("p",null,"Karena ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," sebenarnya adalah UTF-8 bytes, maka kita bisa juga membuatnya menggunakan data Bytes. Tipe bytes (atau kadang disebut ",(0,i.kt)("em",{parentName:"p"},"chars"),") di Rust direpresentasikan oleh tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"[u8]"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let bytes = vec![69, 108, 117, 118, 101, 105, 116, 105, 101, 32, 243, 159, 164, 152];\nlet str2 = String::from_utf8(bytes).unwrap();\nprintln!("str2: {}", str2);\n')),(0,i.kt)("p",null,"Pada contoh di atas, data bytes dipersiapkan dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),". Data tersebut kemudian digunakan untuk membuat string menggunakan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"String::from_utf8()"),". Nilai balik fungsi tersebut adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Pemanggilan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," di situ agar data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"-nya di-return."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"String slcie vs string literal",src:t(108).Z,width:"458",height:"108"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Kita akan bahas tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,i.kt)("h2",{id:"a442-string-literal-str"},"A.44.2. String literal (",(0,i.kt)("inlineCode",{parentName:"h2"},"&str"),")"),(0,i.kt)("p",null,"Tipe data string literal atau ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," adalah tipe yang menampung data kolektif UTF-8 bytes (seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),") tetapi ",(0,i.kt)("strong",{parentName:"p"},"immutable")," dan disimpannya tidak di heap dan tidak juga di stack, melainkan di static storage."),(0,i.kt)("p",null,"String literal hanya bisa direpresentasikan dalam bentuk reference ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," (pointer yang mengarah ke suatu bytes)."),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," termasuk kategori tipe data yang ",(0,i.kt)("strong",{parentName:"p"},"unowned")," atau reference tanpa owner (atau boleh juga diartikan sebagai tipe data yang owner-nya adalah program)."),(0,i.kt)("p",null,"Rust menjamin data string literal selalu valid. Kita juga bisa menentukan lifetime-nya secara eksplist jike diperlukan, contohnya pada tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&'static str"),"."),(0,i.kt)("p",null,"Cara termudah membuat ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," adalah menggunakan string literal."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str3 = "Helena Iren Michaelsen Epica";\nprintln!("str3: {str3}");\n')),(0,i.kt)("h2",{id:"a443-konversi-data-string"},"A.44.3. Konversi data string"),(0,i.kt)("h3",{id:"-konversi-string-ke-str"},"\u25c9 Konversi ",(0,i.kt)("inlineCode",{parentName:"h3"},"String")," ke ",(0,i.kt)("inlineCode",{parentName:"h3"},"&str")),(0,i.kt)("p",null,"Data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," juga didapat melalui operasi borrow dari data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", caranya dengan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_str"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str4: String = String::from("Hiroyuki Sawano");\nlet str4_slice1: &str = str4.as_str();\nprintln!("str4: {str4}"); // str4: Hiroyuki Sawano\nprintln!("str4_slice1: {str4_slice1}"); // str4_slice1: Hiroyuki Sawano\n')),(0,i.kt)("p",null,"Bisa juga menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_mut_str")," untuk mutable borrow. Namun dalam penggunaannya, owner data diwajibkan mutable. Contohnya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str5: String = String::from("Hans Zimmer");\nlet str5_slice1: &mut str = str5.as_mut_str();\nprintln!("str5: {str5}");\nprintln!("str5_slice1: {str5_slice1}");\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"String slcie vs string literal",src:t(6484).Z,width:"650",height:"237"})),(0,i.kt)("p",null,"Hmm, error. Perlu diingat kembali aturan ",(0,i.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing")," bahwa tidak boleh ada operasi borrow dan juga mutable borrow dalam satu waktu. Pada contoh di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"str5")," dipinjam oleh ",(0,i.kt)("inlineCode",{parentName:"p"},"str5_slice1")," menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_mut_str")," (yang di dalamnya menjalankan operasi mutable borrow). Selain itu, operasi mutable borrow maksimal hanya boleh dilakukan 1x dalam satu waktu, inilah kenapa ketika kita berusaha menampilkan data ",(0,i.kt)("inlineCode",{parentName:"p"},"str5")," hasilnya error, karena statement print data ",(0,i.kt)("inlineCode",{parentName:"p"},"str5")," adalah operasi borrow."),(0,i.kt)("p",null,"Solusinya masalah di atas bisa menggunakan block expression:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str5: String = String::from("Hans Zimmer");\n{\n let str5_slice1: &mut str = str5.as_mut_str();\n println!("str5_slice1: {str5_slice1}"); // str5_slice1: Hans Zimmer\n}\nprintln!("str5: {}", str5); // str5_slice1: Hans Zimmer\n')),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," bisa diakses mutable reference-nya karena memang tipe tersebut size-nya adalah dinamis."),(0,i.kt)("h3",{id:"-konversi-str-ke-string"},"\u25c9 Konversi ",(0,i.kt)("inlineCode",{parentName:"h3"},"&str")," ke ",(0,i.kt)("inlineCode",{parentName:"h3"},"String")),(0,i.kt)("p",null,"Masih dalam topik konversi tipe data string. Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," memiliki method bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"to_string")," yang gunanya adalah untuk konversi data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str6: &str = "John Towner Williams";\nlet str6_slice1: String = str6.to_string();\nprintln!("str6: {str6}"); // str6: John Towner Williams\nprintln!("str6_slice1: {str6_slice1}"); // str6_slice1: John Towner Williams\n')),(0,i.kt)("p",null,"Konversi pada tipe data ini sedikit berbeda dibandingkan konversi ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),". Pada contoh di atas, yang terjadi adalah data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," di-copy sebagai data baru bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," yang kemudian ditampung variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"str6_slice1")," (yang juga berperan sebagai owner untuk data baru tersebut)."),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"to_string")," melakukan operasi copy, bukan borrow. Artinya setelah dipanggil akan ada 2 data yang reference-nya sudah berbeda."),(0,i.kt)("h2",{id:"a444-string-literal--string-slice"},"A.44.4. String literal & string slice"),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," memiliki hubungan dekat dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),". Data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," reference-nya bisa diakses dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"&String"),", maupun dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," (menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_str")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"as_mut_str"),"). Data text pada string tersebut bisa dimodifikasi, ditambahi, dan juga dikurangi."),(0,i.kt)("p",null,"Berbeda dengan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," ",(0,i.kt)("strong",{parentName:"p"},"(di paragraph ini dan setelahnya yang kita bahas adalah data yang dari awal tipe-nya sudah ",(0,i.kt)("inlineCode",{parentName:"strong"},"&str"),", bukan data hasil operasi pinjam dari ",(0,i.kt)("inlineCode",{parentName:"strong"},"String"),")"),", data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," adalah fixed dan immutable. Konversi data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," menghasilkan data baru dengan owner baru. Bisa dibilang sangat terbatas apa yang bisa kita lakukan pada tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("p",null,"Meski demikian, tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," lebih cepat performa-nya dibanding ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," karena disimpan di static storage. Selain itu dijamin valid oleh Rust. Kekurangannya hanya pada ownership-nya. Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," adalah ",(0,i.kt)("strong",{parentName:"p"},"unowned"),", operasi mutability tidak bisa dilakukan pada tipe ini."),(0,i.kt)("p",null,"Dalam case normal, sangat dianjurkan untuk menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),", kecuali memang yang dibutuhkan adalah ",(0,i.kt)("strong",{parentName:"p"},"owned")," string."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/string_slice_vs_string_literal"},"github.com/novalagung/dasarpemrogramanrust-example/../string_slice_vs_string_literal")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-string-literal"},"Tipe Data \u279c String Literal (&str)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice-memory-management"},"Slice Memory Management")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-custom-type-string-slice"},"Tipe Data \u279c String Custom Type"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch08-02-strings.html"},"https://doc.rust-lang.org/book/ch08-02-strings.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://users.rust-lang.org/t/str-string-literals/29635"},"https://users.rust-lang.org/t/str-string-literals/29635")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://www.quora.com/Why-does-Rust-have-two-different-string-types-static-str-and-String"},"https://www.quora.com/Why-does-Rust-have-two-different-string-types-static-str-and-String")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/24158114/what-are-the-differences-between-rusts-string-and-str"},"https://stackoverflow.com/questions/24158114/what-are-the-differences-between-rusts-string-and-str")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/30154541/how-do-i-concatenate-strings"},"https://stackoverflow.com/questions/30154541/how-do-i-concatenate-strings"))))}m.isMDXComponent=!0},108:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},6484:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/string-slice-vs-string-literal-2-1b0caccba69a37db29f101f921a76df1.png"}}]); \ No newline at end of file diff --git a/assets/js/a2bea3ed.c14ec614.js b/assets/js/a2bea3ed.c14ec614.js deleted file mode 100644 index 2c91d368..00000000 --- a/assets/js/a2bea3ed.c14ec614.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6345],{3905:(a,e,t)=>{t.d(e,{Zo:()=>o,kt:()=>u});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var p=n.createContext({}),d=function(a){var e=n.useContext(p),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},o=function(a){var e=d(a.components);return n.createElement(p.Provider,{value:e},a.children)},m={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},k=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,p=a.parentName,o=s(a,["components","mdxType","originalType","parentName"]),k=d(t),u=i,g=k["".concat(p,".").concat(u)]||k[u]||m[u]||r;return t?n.createElement(g,l(l({ref:e},o),{},{components:t})):n.createElement(g,l({ref:e},o))}));function u(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=k;var s={};for(var p in e)hasOwnProperty.call(e,p)&&(s[p]=e[p]);s.originalType=a,s.mdxType="string"==typeof a?a:i,l[1]=s;for(var d=2;d{t.r(e),t.d(e,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>r,metadata:()=>s,toc:()=>d});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:44,title:"A.44. String Literal (&str) vs. String Custom Type",sidebar_label:"A.44. String Literal (&str) vs. String Custom Type"},l=void 0,s={unversionedId:"basic/string-slice-vs-string-literal",id:"basic/string-slice-vs-string-literal",title:"A.44. String Literal (&str) vs. String Custom Type",description:"Pada chapter sebelumnya kita telah membahas tentang bagaimana data slice di-manage di memory. Ada beberapa tipe data yang masuk dalam kategori slice, yang salah satunya adalah string slice atau String.",source:"@site/docs/basic/string-slice-vs-string-literal.md",sourceDirName:"basic",slug:"/basic/string-slice-vs-string-literal",permalink:"/basic/string-slice-vs-string-literal",draft:!1,tags:[],version:"current",sidebarPosition:44,frontMatter:{sidebar_position:44,title:"A.44. String Literal (&str) vs. String Custom Type",sidebar_label:"A.44. String Literal (&str) vs. String Custom Type"},sidebar:"tutorialSidebar",previous:{title:"A.43. Slice Memory Management",permalink:"/basic/slice-memory-management"},next:{title:"A.45. Tipe Data \u279c String Custom Type",permalink:"/basic/tipe-data-custom-type-string-slice"}},p={},d=[{value:"A.44.1. String slice (String)",id:"a441-string-slice-string",level:2},{value:"A.44.2. String literal (&str)",id:"a442-string-literal-str",level:2},{value:"A.44.3. Konversi data string",id:"a443-konversi-data-string",level:2},{value:"\u25c9 Konversi String ke &str",id:"-konversi-string-ke-str",level:3},{value:"\u25c9 Konversi &str ke String",id:"-konversi-str-ke-string",level:3},{value:"A.44.4. String literal & string slice",id:"a444-string-literal--string-slice",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:d};function m(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},o,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter sebelumnya kita telah membahas tentang bagaimana data slice di-manage di memory. Ada beberapa tipe data yang masuk dalam kategori slice, yang salah satunya adalah string slice atau ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("p",null,"Di chapter ini kita akan bahas apa perbedaan antara tipe data string slice (",(0,i.kt)("inlineCode",{parentName:"p"},"String"),") dan string literal ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("p",null,"Pembahasan mengenai topik ini sengaja dilakukan tidak di awal-awal ebook, karena ada banyak hal yang perlu dipahami sebelum mempelajarinya, contohnya seperti aspek management memory dan ownership. Dan karena topik tersebut sudah selesai dibahas, berarti ini adalah waktu yang tepat untuk membahas string slice."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Silakan pelajari kembali pembahasan detail tentang tipe slice pada chapter sebelumnya jika diperlukan. Chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/slice-memory-management"},"Slice Memory Management"))),(0,i.kt)("h2",{id:"a441-string-slice-string"},"A.44.1. String slice (",(0,i.kt)("inlineCode",{parentName:"h2"},"String"),")"),(0,i.kt)("p",null,"String slice atau custom type ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," merupakan tipe data bawaan Rust, dibuat via ",(0,i.kt)("inlineCode",{parentName:"p"},"struct"),", kegunaannya untuk menampung data UTF-8 bytes yang dinamis (bisa berkembang isinya)."),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"String")," masuk dalam kategori tipe data slice, isinya adalah data kolektif bertipe bytes, datanya disimpan di heap memory, dan metadata-nya di stack memory. Tipe data ini dikategorikan sebagai tipe data ",(0,i.kt)("strong",{parentName:"p"},"owned"),", yang artinya owner data bisa direpresentasikan oleh variabel. Sebagai contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str1 = String::from("Lisa Blackpink");\nprintln!("str1: {str1}");\n')),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"str1")," di atas merupakan owner dari string ",(0,i.kt)("inlineCode",{parentName:"p"},"Lisa Blackpink"),". Dari string tersebut operasi borrow bisa dilakukan. Contohnya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str1 = String::from("Lisa Blackpink");\nlet slice1 = &str1[..]; // "Lisa Blackpink"\nlet slice2 = &str1[4..7]; // " Bl"\n')),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"slice1")," adalah variabel baru yang datanya didapat dari borrowing seluruh elemen string milik ",(0,i.kt)("inlineCode",{parentName:"p"},"str1"),". Sedangkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"slice2")," hanya meminjam elemen indeks ke-4 hingga elemen sebelum 7 (yaitu 6)."),(0,i.kt)("p",null,"Karena ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," sebenarnya adalah UTF-8 bytes, maka kita bisa juga membuatnya menggunakan data Bytes. Tipe bytes (atau kadang disebut ",(0,i.kt)("em",{parentName:"p"},"chars"),") di Rust direpresentasikan oleh tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"[u8]"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let bytes = vec![69, 108, 117, 118, 101, 105, 116, 105, 101, 32, 243, 159, 164, 152];\nlet str2 = String::from_utf8(bytes).unwrap();\nprintln!("str2: {}", str2);\n')),(0,i.kt)("p",null,"Pada contoh di atas, data bytes dipersiapkan dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),". Data tersebut kemudian digunakan untuk membuat string menggunakan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"String::from_utf8()"),". Nilai balik fungsi tersebut adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Result"),". Pemanggilan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," disitu agar data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"-nya di-return."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"String slcie vs string literal",src:t(108).Z,width:"458",height:"108"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Kita akan bahas tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,i.kt)("h2",{id:"a442-string-literal-str"},"A.44.2. String literal (",(0,i.kt)("inlineCode",{parentName:"h2"},"&str"),")"),(0,i.kt)("p",null,"Tipe data string literal atau ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," adalah tipe yang menampung data kolektif UTF-8 bytes (seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),") tetapi ",(0,i.kt)("strong",{parentName:"p"},"immutable")," dan disimpannya tidak di heap dan tidak juga di stack, melainkan di static storage."),(0,i.kt)("p",null,"String literal hanya bisa direpresentasikan dalam bentuk reference ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," (pointer yang mengarah ke suatu bytes)."),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," termasuk kategori tipe data yang ",(0,i.kt)("strong",{parentName:"p"},"unowned")," atau reference tanpa owner (atau boleh juga diartikan sebagai tipe data yang owner-nya adalah program)."),(0,i.kt)("p",null,"Rust menjamin data string literal selalu valid. Kita juga bisa menentukan lifetime-nya secara eksplist jike diperlukan, contohnya pada tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&'static str"),"."),(0,i.kt)("p",null,"Cara termudah membuat ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," adalah menggunakan string literal."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str3 = "Helena Iren Michaelsen Epica";\nprintln!("str3: {str3}");\n')),(0,i.kt)("h2",{id:"a443-konversi-data-string"},"A.44.3. Konversi data string"),(0,i.kt)("h3",{id:"-konversi-string-ke-str"},"\u25c9 Konversi ",(0,i.kt)("inlineCode",{parentName:"h3"},"String")," ke ",(0,i.kt)("inlineCode",{parentName:"h3"},"&str")),(0,i.kt)("p",null,"Data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," juga didapat melalui operasi borrow dari data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", caranya dengan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_str"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str4: String = String::from("Hiroyuki Sawano");\nlet str4_slice1: &str = str4.as_str();\nprintln!("str4: {str4}"); // str4: Hiroyuki Sawano\nprintln!("str4_slice1: {str4_slice1}"); // str4_slice1: Hiroyuki Sawano\n')),(0,i.kt)("p",null,"Bisa juga menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_mut_str")," untuk mutable borrow. Namun dalam penggunaannya, owner data diwajibkan mutable. Contohnya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str5: String = String::from("Hans Zimmer");\nlet str5_slice1: &mut str = str5.as_mut_str();\nprintln!("str5: {str5}");\nprintln!("str5_slice1: {str5_slice1}");\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"String slcie vs string literal",src:t(6484).Z,width:"650",height:"237"})),(0,i.kt)("p",null,"Hmm, error. Perlu diingat kembali aturan ",(0,i.kt)("a",{parentName:"p",href:"/basic/borrowing"},"Borrowing")," bahwa tidak boleh ada operasi borrow dan juga mutable borrow dalam satu waktu. Pada contoh di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"str5")," dipinjam oleh ",(0,i.kt)("inlineCode",{parentName:"p"},"str5_slice1")," menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_mut_str")," (yang di dalamnya menjalankan operasi mutable borrow). Selain itu, operasi mutable borrow maksimal hanya boleh dilakukan 1x dalam satu waktu, inilah kenapa ketika kita berusaha menampilkan data ",(0,i.kt)("inlineCode",{parentName:"p"},"str5")," hasilnya error, karena statement print data ",(0,i.kt)("inlineCode",{parentName:"p"},"str5")," adalah operasi borrow."),(0,i.kt)("p",null,"Solusinya masalah di atas bisa menggunakan block expression:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut str5: String = String::from("Hans Zimmer");\n{\n let str5_slice1: &mut str = str5.as_mut_str();\n println!("str5_slice1: {str5_slice1}"); // str5_slice1: Hans Zimmer\n}\nprintln!("str5: {}", str5); // str5_slice1: Hans Zimmer\n')),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," bisa diakses mutable reference-nya karena memang tipe tersebut size-nya adalah dinamis."),(0,i.kt)("h3",{id:"-konversi-str-ke-string"},"\u25c9 Konversi ",(0,i.kt)("inlineCode",{parentName:"h3"},"&str")," ke ",(0,i.kt)("inlineCode",{parentName:"h3"},"String")),(0,i.kt)("p",null,"Masih dalam topik konversi tipe data string. Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," memiliki method bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"to_string")," yang gunanya adalah untuk konversi data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str6: &str = "John Towner Williams";\nlet str6_slice1: String = str6.to_string();\nprintln!("str6: {str6}"); // str6: John Towner Williams\nprintln!("str6_slice1: {str6_slice1}"); // str6_slice1: John Towner Williams\n')),(0,i.kt)("p",null,"Konversi pada tipe data ini sedikit berbeda dibandingkan konversi ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),". Pada contoh di atas, yang terjadi adalah data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," di-copy sebagai data baru bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," yang kemudian ditampung variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"str6_slice1")," (yang juga berperan sebagai owner untuk data baru tersebut)."),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"to_string")," melakukan operasi copy, bukan borrow. Artinya setelah dipanggil akan ada 2 data yang reference-nya sudah berbeda."),(0,i.kt)("h2",{id:"a444-string-literal--string-slice"},"A.44.4. String literal & string slice"),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," memiliki hubungan dekat dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),". Data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," reference-nya bisa diakses dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"&String"),", maupun dalam bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," (menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_str")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"as_mut_str"),"). Data text pada string tersebut bisa dimodifikasi, ditambahi, dan juga dikurangi."),(0,i.kt)("p",null,"Berbeda dengan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," ",(0,i.kt)("strong",{parentName:"p"},"(di paragraph ini dan setelahnya yang kita bahas adalah data yang dari awal tipe-nya sudah ",(0,i.kt)("inlineCode",{parentName:"strong"},"&str"),", bukan data hasil operasi pinjam dari ",(0,i.kt)("inlineCode",{parentName:"strong"},"String"),")"),", data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," adalah fixed dan immutable. Konversi data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," menghasilkan data baru dengan owner baru. Bisa dibilang sangat terbatas apa yang bisa kita lakukan pada tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("p",null,"Meski demikian, tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," lebih cepat performa-nya dibanding ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," karena disimpan di static storage. Selain itu dijamin valid oleh Rust. Kekurangannya hanya pada ownership-nya. Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," adalah ",(0,i.kt)("strong",{parentName:"p"},"unowned"),", operasi mutability tidak bisa dilakukan pada tipe ini."),(0,i.kt)("p",null,"Dalam case normal, sangat dianjurkan untuk menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),", kecuali memang yang dibutuhkan adalah ",(0,i.kt)("strong",{parentName:"p"},"owned")," string."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/string_slice_vs_string_literal"},"github.com/novalagung/dasarpemrogramanrust-example/../string_slice_vs_string_literal")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-string-literal"},"Tipe Data \u279c String Literal (&str)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice-memory-management"},"Slice Memory Management")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-custom-type-string-slice"},"Tipe Data \u279c String Custom Type"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch08-02-strings.html"},"https://doc.rust-lang.org/book/ch08-02-strings.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://users.rust-lang.org/t/str-string-literals/29635"},"https://users.rust-lang.org/t/str-string-literals/29635")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://www.quora.com/Why-does-Rust-have-two-different-string-types-static-str-and-String"},"https://www.quora.com/Why-does-Rust-have-two-different-string-types-static-str-and-String")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/24158114/what-are-the-differences-between-rusts-string-and-str"},"https://stackoverflow.com/questions/24158114/what-are-the-differences-between-rusts-string-and-str")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/30154541/how-do-i-concatenate-strings"},"https://stackoverflow.com/questions/30154541/how-do-i-concatenate-strings"))))}m.isMDXComponent=!0},108:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},6484:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/string-slice-vs-string-literal-2-1b0caccba69a37db29f101f921a76df1.png"}}]); \ No newline at end of file diff --git a/assets/js/af6aab29.62e5b74a.js b/assets/js/af6aab29.62e5b74a.js new file mode 100644 index 00000000..e3d71402 --- /dev/null +++ b/assets/js/af6aab29.62e5b74a.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8794],{3905:(e,t,a)=>{a.d(t,{Zo:()=>o,kt:()=>c});var n=a(7294);function i(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function r(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,n)}return a}function l(e){for(var t=1;t=0||(i[a]=e[a]);return i}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(i[a]=e[a])}return i}var d=n.createContext({}),p=function(e){var t=n.useContext(d),a=t;return e&&(a="function"==typeof e?e(t):l(l({},t),e)),a},o=function(e){var t=p(e.components);return n.createElement(d.Provider,{value:t},e.children)},k={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},s=n.forwardRef((function(e,t){var a=e.components,i=e.mdxType,r=e.originalType,d=e.parentName,o=m(e,["components","mdxType","originalType","parentName"]),s=p(a),c=i,u=s["".concat(d,".").concat(c)]||s[c]||k[c]||r;return a?n.createElement(u,l(l({ref:t},o),{},{components:a})):n.createElement(u,l({ref:t},o))}));function c(e,t){var a=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var r=a.length,l=new Array(r);l[0]=s;var m={};for(var d in t)hasOwnProperty.call(t,d)&&(m[d]=t[d]);m.originalType=e,m.mdxType="string"==typeof e?e:i,l[1]=m;for(var p=2;p{a.r(t),a.d(t,{assets:()=>d,contentTitle:()=>l,default:()=>k,frontMatter:()=>r,metadata:()=>m,toc:()=>p});var n=a(7462),i=(a(7294),a(3905));const r={sidebar_position:50,title:"A.50. Date Time",sidebar_label:"A.50. Date Time"},l=void 0,m={unversionedId:"basic/datetime",id:"basic/datetime",title:"A.50. Date Time",description:"Pada chapter ini kita akan mempelajari tentang tipe data datetime yang ada di Rust programming.",source:"@site/docs/basic/datetime.md",sourceDirName:"basic",slug:"/basic/datetime",permalink:"/basic/datetime",draft:!1,tags:[],version:"current",sidebarPosition:50,frontMatter:{sidebar_position:50,title:"A.50. Date Time",sidebar_label:"A.50. Date Time"},sidebar:"tutorialSidebar",previous:{title:"A.49. Attributes",permalink:"/basic/attributes"},next:{title:"A.51. File, Path, Directory",permalink:"/basic/file-path-directory"}},d={},p=[{value:"A.50.1. Persiapan",id:"a501-persiapan",level:2},{value:"A.50.2. Tipe data DateTime",id:"a502-tipe-data-datetime",level:2},{value:"\u25c9 Tipe DateTime<Utc>",id:"-tipe-datetimeutc",level:3},{value:"\u25c9 Tipe DateTime<Local>",id:"-tipe-datetimelocal",level:3},{value:"\u25c9 Tipe NaiveDateTime",id:"-tipe-naivedatetime",level:3},{value:"A.50.3. Pembuatan objek DateTime",id:"a503-pembuatan-objek-datetime",level:2},{value:"\u25c9 Via Utc::now()",id:"-via-utcnow",level:3},{value:"\u25c9 Via Utc.with_ymd_and_hms()",id:"-via-utcwith_ymd_and_hms",level:3},{value:"\u25c9 Via DateTime::<Utc>::from()",id:"-via-datetimeutcfrom",level:3},{value:"\u25c9 Via String method parse()",id:"-via-string-method-parse",level:3},{value:"\u25c9 Cara lainnya",id:"-cara-lainnya",level:3},{value:"A.50.4. Utc vs Local",id:"a504-utc-vs-local",level:2},{value:"A.50.5. Tipe NaiveDateTime",id:"a505-tipe-naivedatetime",level:2},{value:"A.50.6. Konversi datetime UTC ke local timezone, dan sebaliknya",id:"a506-konversi-datetime-utc-ke-local-timezone-dan-sebaliknya",level:2},{value:"A.50.7. UNIX time",id:"a507-unix-time",level:2},{value:"A.50.8. Date formatting & parsing",id:"a508-date-formatting--parsing",level:2},{value:"A.50.9. DateTime formatting syntax",id:"a509-datetime-formatting-syntax",level:2},{value:"\u25c9 Date specifiers",id:"-date-specifiers",level:3},{value:"\u25c9 Time specifiers",id:"-time-specifiers",level:3},{value:"\u25c9 Timezone specifiers",id:"-timezone-specifiers",level:3},{value:"\u25c9 Date & time specifiers",id:"-date--time-specifiers",level:3},{value:"\u25c9 Special specifiers",id:"-special-specifiers",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:p};function k(e){let{components:t,...r}=e;return(0,i.kt)("wrapper",(0,n.Z)({},o,r,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan mempelajari tentang tipe data datetime yang ada di Rust programming."),(0,i.kt)("p",null,"Rust menyediakan cukup banyak API untuk keperluan pengolahan datetime, tersedia dalam Standard Library crate. Namun pada chapter ini kita tidak akan menggunakannya, karena yang akan dipakai adalah ",(0,i.kt)("a",{parentName:"p",href:"https://crates.io/crates/chrono"},(0,i.kt)("inlineCode",{parentName:"a"},"chrono")," crate")," (dianjurkan untuk menggunakan crate ini karena fiturnya lebih banyak)."),(0,i.kt)("h2",{id:"a501-persiapan"},"A.50.1. Persiapan"),(0,i.kt)("p",null,"Sebelum memulai pembelajaran, siapkan satu project baru, kemudian tambahkan care ",(0,i.kt)("inlineCode",{parentName:"p"},"crono")," dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-toml"},'[package]\nname = "time_chrono"\nversion = "0.1.0"\nedition = "2021"\n\n[dependencies]\nchrono = "0.4.23"\n')),(0,i.kt)("h2",{id:"a502-tipe-data-datetime"},"A.50.2. Tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"DateTime")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"crono")," menyediakan cukup banyak tipe data untuk merepresentasikan informasi tanggal dan waktu, tetapi yang paling penting untuk diketahui dan dipahami ada dua, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"DateTime")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"DateTime")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"NaiveDateTime"))),(0,i.kt)("h3",{id:"-tipe-datetimeutc"},"\u25c9 Tipe ",(0,i.kt)("inlineCode",{parentName:"h3"},"DateTime")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," (gabungan dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::datetime::DateTime")," dan generic ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::offset::utc::Utc"),") adalah representasi untuk tipe data datetime dengan timezone offset ",(0,i.kt)("strong",{parentName:"p"},"UTC")," (atau ",(0,i.kt)("strong",{parentName:"p"},"GMT+0"),")."),(0,i.kt)("h3",{id:"-tipe-datetimelocal"},"\u25c9 Tipe ",(0,i.kt)("inlineCode",{parentName:"h3"},"DateTime")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," (gabungan dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::datetime::DateTime")," dan generic ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::offset::local::Local"),") adalah representasi untuk tipe data datetime dengan ",(0,i.kt)("strong",{parentName:"p"},"timezone offset sesuai dengan di mana program dijalankan"),"."),(0,i.kt)("p",null,"Sebagai contoh, penulis berlokasi di Jawa Timur, maka ketika ada suatu data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," artinya timezone offset-nya adalah WIB (atau GMT+7)."),(0,i.kt)("h3",{id:"-tipe-naivedatetime"},"\u25c9 Tipe ",(0,i.kt)("inlineCode",{parentName:"h3"},"NaiveDateTime")),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::naive::datetime::NaiveDateTime")," merupakan tipe data datetime yang tidak menyimpan informasi timezone. Nanti akan kita bahas apa kegunaan dari tipe ini."),(0,i.kt)("h2",{id:"a503-pembuatan-objek-datetime"},"A.50.3. Pembuatan objek ",(0,i.kt)("inlineCode",{parentName:"h2"},"DateTime")),(0,i.kt)("h3",{id:"-via-utcnow"},"\u25c9 Via ",(0,i.kt)("inlineCode",{parentName:"h3"},"Utc::now()")),(0,i.kt)("p",null,"Ada banyak cara untuk membuat date time, dan kita akan bahas satu-per-satu."),(0,i.kt)("p",null,"Cara yang pertama, mari kita pelajari sambil praktek. Isi file program ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs")," dengan kode berikut, lalu jalankan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use chrono::prelude::*;\nuse std::time::{UNIX_EPOCH, Duration};\n\nfn main() {\n let sample_date1_in_utc: DateTime = Utc::now();\n println!("sample date 1 (in utc): {sample_date1_in_utc}");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(6635).Z,width:"571",height:"95"})),(0,i.kt)("p",null,"Contoh sederhana di atas adalah salah satu cara membuat object datetime dengan isi adalah informasi waktu sekarang (waktu ketika baris kode tersebut dieksekusi)."),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc::now()")," mengembalikan informasi waktu sekarang dalam timezone UTC (representasi tipe data-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime"),")."),(0,i.kt)("p",null,"Module item ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc")," sendiri harus diimport terlebih dahulu agar bisa digunakan. Itulah kenapa ada statement ",(0,i.kt)("inlineCode",{parentName:"p"},"use chrono::prelude::*"),"."),(0,i.kt)("h3",{id:"-via-utcwith_ymd_and_hms"},"\u25c9 Via ",(0,i.kt)("inlineCode",{parentName:"h3"},"Utc.with_ymd_and_hms()")),(0,i.kt)("p",null,"Pembuatan object datetime dengan isi informasi waktu spesifik bisa dilakukan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc.with_ymd_and_hms()"),". Fungsi ini mewajibkan developer untuk mengisi 6 parameternya dengan segment waktu (tahun, bulan, dan lainnya)."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sample_date2_in_utc = Utc.with_ymd_and_hms(2023, 3, 1, 1, 2, 3).unwrap();\nprintln!("sample date 2 (in utc): {sample_date2_in_utc}");\n// output: sample date 2 (in utc): 2023-03-01 01:02:03 UTC\n')),(0,i.kt)("p",null,"Statement di atas menghasilkan data datetime dengan isi ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-01 01:02:03 UTC"),"."),(0,i.kt)("h3",{id:"-via-datetimeutcfrom"},"\u25c9 Via ",(0,i.kt)("inlineCode",{parentName:"h3"},"DateTime::::from()")),(0,i.kt)("p",null,"Cara ini pas digunakan pada situasi di mana kita perlu membuat object datetime dari sebuah ",(0,i.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Unix_time"},"UNIX timestamp"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sample_date3_in_utc = DateTime::::from(UNIX_EPOCH + Duration::from_secs(1524885322));\nprintln!("sample date 3 (in utc): {sample_date3_in_utc}");\n// output: sample date 3 (in utc): 2018-04-28 03:15:22 UTC\n')),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime::::from()")," di atas menghasilkan data datetime ",(0,i.kt)("inlineCode",{parentName:"p"},"2018-04-28 03:15:22 UTC"),", karena nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"1524885322")," adalah representasi UNIX time untuk waktu tersebut."),(0,i.kt)("h3",{id:"-via-string-method-parse"},"\u25c9 Via String method ",(0,i.kt)("inlineCode",{parentName:"h3"},"parse()")),(0,i.kt)("p",null,"Tipe data string bisa langsung dikonversi ke bentuk datetime menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"parse()")," yang dikombinasikan dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," sebagai generic parameter."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sample_date4_in_utc = "2023-03-01 01:02:03 UTC".parse::>().unwrap();\nprintln!("sample date 4 (in utc): {sample_date4_in_utc}");\n// output: sample date 4 (in utc): 2023-03-01 01:02:03 UTC\n')),(0,i.kt)("p",null,"Statement parsing di atas menghasilkan data datetime ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-01 01:02:03 UTC"),", sesuai dengan isi data string."),(0,i.kt)("p",null,"O iya, perlu diketahui bahwa tidak semua string bisa di-parse ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime"),". Jika data string memiliki format yang tidak sesuai dengan aturan parsing di Rust, maka proses parsing menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"parse()")," pasti menghasilkan error."),(0,i.kt)("p",null,"Penerapan method ",(0,i.kt)("inlineCode",{parentName:"p"},"parse::>()")," sebenarnya mengembalikan tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result, ParseError>"),", itulah kenapa pada contoh perlu di chain lagi dengan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih detailnya mengenai tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,i.kt)("h3",{id:"-cara-lainnya"},"\u25c9 Cara lainnya"),(0,i.kt)("p",null,"Ada beberapa cara lain yang bisa digunakan untuk pembuatan object datetime, dan kita akan bahas juga sebentar lagi."),(0,i.kt)("h2",{id:"a504-utc-vs-local"},"A.50.4. ",(0,i.kt)("inlineCode",{parentName:"h2"},"Utc")," vs ",(0,i.kt)("inlineCode",{parentName:"h2"},"Local")),(0,i.kt)("p",null,"Kesemua cara yang sudah dipelajari di atas juga bisa diterapkan dalam pembuatan object date time dengan timezone offset ",(0,i.kt)("inlineCode",{parentName:"p"},"Local"),". Penerapannya kurang lebih sama, cukup ganti ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc")," menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"Local"),"."),(0,i.kt)("p",null,"Jika disandingkan, akan seperti ini kodenya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let date1_in_utc: DateTime = Utc::now();\nlet date2_in_utc = Utc.with_ymd_and_hms(2023, 3, 1, 1, 2, 3).unwrap();\nlet date3_in_utc = DateTime::::from(UNIX_EPOCH + Duration::from_secs(1524885322));\nlet date4_in_utc = "2023-03-01 01:02:03 UTC".parse::>().unwrap();\n\nprintln!("sample date 1 (in utc): {date1_in_utc}");\nprintln!("sample date 2 (in utc): {date2_in_utc}");\nprintln!("sample date 3 (in utc): {date3_in_utc}");\nprintln!("sample date 4 (in utc): {date4_in_utc}");\n\n// vs\n\nlet date1_in_local_tz: DateTime = Local::now();\nlet date2_in_local_tz = Local.with_ymd_and_hms(2023, 3, 1, 1, 2, 3).unwrap();\nlet date3_in_local_tz = DateTime::::from(UNIX_EPOCH + Duration::from_secs(1524885322));\nlet date4_in_local_tz = "2023-03-01 01:02:03 UTC".parse::>().unwrap();\n\nprintln!("sample date 1 (in local_tz): {date1_in_local_tz}");\nprintln!("sample date 2 (in local_tz): {date2_in_local_tz}");\nprintln!("sample date 3 (in local_tz): {date3_in_local_tz}");\nprintln!("sample date 4 (in local_tz): {date4_in_local_tz}");\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(239).Z,width:"571",height:"231"})),(0,i.kt)("p",null,"Semoga cukup jelas."),(0,i.kt)("p",null,"Dari contoh di atas, sebenarnya ada 2 variabel yang menarik untuk dibahas, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"sample_date1_in_utc")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"sample_date1_in_local_tz"),". Dua variabel tersebut berisi informasi waktu sekarang, perbedaannya hanya pada timezone offset saja. Namun, meski timezone offset-nya berbeda, kedua datetime tersebut sejatinya adalah merepresentasikan waktu yang sama (ekuivalen)."),(0,i.kt)("p",null,"Agar lebih jelas, silakan perhatikan contoh berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash"},"sample date 1 (in utc): 2023-03-02 09:53:44.561519500 UTC\nsample date 1 (in local_tz): 2023-03-02 16:53:44.562389600 +07:00\n")),(0,i.kt)("p",null,"Kedua datetime tersebut terpaku ",(0,i.kt)("strong",{parentName:"p"},"7 jam")," persis. Data ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-02 09:53:44")," UTC (atau GMT+0), jika dikonversi ke waktu lokal penulis (WIB atau GMT+7) maka akan menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-02 16:53:44"),". Selisihnya sesuai dengan timezone offset, yaitu 7 jam."),(0,i.kt)("h2",{id:"a505-tipe-naivedatetime"},"A.50.5. Tipe ",(0,i.kt)("inlineCode",{parentName:"h2"},"NaiveDateTime")),(0,i.kt)("p",null,"Rust memilik caranya sendiri dalam pengelolaan data datetime (yang menurut penulis cukup unik). Ada satu tipe data bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime"),", yang kegunaannya adalah untuk menampung tipe data datetime ",(0,i.kt)("strong",{parentName:"p"},"tanpa spesifik timezone offset"),"."),(0,i.kt)("p",null,"Tipe data ini sangat berguna dibeberapa case yang kebanyakan adalah perihal konversi data ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime"),"."),(0,i.kt)("p",null,"Mari kita praktekan dengan contoh agar lebih jelas. Dimisalkan ada keperluan di mana data UNIX time perlu di konversi ke 2 bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime"),", dengan timezone offset ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Local"),". Pada kasus ini, cara ke-3 dari praktek sebelumnya bisa dilakukan untuk penyelesaian case ini."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let timestamp: u64 = 1524885322;\nlet date2_in_utc = DateTime::::from(UNIX_EPOCH + Duration::from_secs(timestamp));\nlet date2_in_local_tz = DateTime::::from(UNIX_EPOCH + Duration::from_secs(timestamp));\n")),(0,i.kt)("p",null,"Cara lain adalah dengan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime"),", yang contoh penerapannnya kurang lebih seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let timestamp_in_ms: i64 = 1524885322000;\nlet naive_date_time: NaiveDateTime = NaiveDateTime::from_timestamp_millis(timestamp_in_ms).unwrap();\n\nlet sample_date_in_utc = Utc.from_utc_datetime(&naive_date_time);\nlet sample_date_in_local_tz = Local.from_local_datetime(&naive_date_time).unwrap();\n\nprintln!("sample date 1 (in utc): {sample_date_in_utc}");\nprintln!("sample date 2 (in local_tz): {sample_date_in_local_tz}");\n\n// output:\n// sample date 1 (in utc): 2018-04-28 03:15:22 UTC\n// sample date 2 (in local_tz): 2018-04-28 03:15:22 +07:00\n')),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime::from_timestamp_millis")," berguna untuk mengkonversi data UNIX time (milisecond) ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime"),". Kemudian, dari data tersebut dapat kita konversi lagi ke bentuk UTC dengan mudah menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc.from_utc_datetime()")," dan local timezone menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"Local.from_local_datetime()"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(2330).Z,width:"571",height:"115"})),(0,i.kt)("p",null,"Bisa dilihat pada output di atas, bahwa hasil data datetime berbeda untuk timezone UTC dan local."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Jangan sampai bingung, 2 data datetime di atas tidak sama dan tidak ekuivalen meskipun informasi jam menit detiknya sama persis. Hal ini dikarenakan timezone offset-nya berbeda.")),(0,i.kt)("h2",{id:"a506-konversi-datetime-utc-ke-local-timezone-dan-sebaliknya"},"A.50.6. Konversi datetime UTC ke local timezone, dan sebaliknya"),(0,i.kt)("p",null,"Dalam bekerja dengan tipe data datetime, selalu memperhatikan timezone adalah hal yang sangat penting. Bisa saja dua buah datetime memiliki informasi tanggal dan waktu sama persis, tapi jika timezone-nya berbeda maka keduanya adalah berbeda!"),(0,i.kt)("p",null,"Agar lebih jelas, silakan perhatikan kode berikut beserta outputnya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let date1_in_local_tz: DateTime = Local::now();\nprintln!("date (in local): {date1_in_local_tz}");\n\nlet date_in_utc = DateTime::::from(date1_in_local_tz);\nprintln!("date (in utc): {date_in_utc}");\n\nlet date2_in_local_tz = DateTime::::from(date_in_utc);\nprintln!("date (in local): {date2_in_local_tz}");\n\n// output:\n// date (in local): 2023-03-02 17:33:37.750279900 +07:00\n// date (in utc): 2023-03-02 10:33:37.750279900 UTC\n// date (in local): 2023-03-02 17:33:37.750279900 +07:00\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(7934).Z,width:"571",height:"129"})),(0,i.kt)("p",null,"Ketiga data datetime di atas adalah ekuivalen:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"date1_in_local_tz")," berisi informasi datetime waktu sekarang, dengan timezone GMT+7."),(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"date_in_utc")," adalah datetime yang ekuivalen dengan ",(0,i.kt)("inlineCode",{parentName:"li"},"date1_in_local_tz"),", hanya saja direpresentasikan dalam timezone UTC."),(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"date2_in_local_tz")," juga ekuivalen dengan ",(0,i.kt)("inlineCode",{parentName:"li"},"date_in_utc")," dan juga ",(0,i.kt)("inlineCode",{parentName:"li"},"date1_in_local_tz"),".")),(0,i.kt)("p",null,"Data ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-02 10 UTC")," jika ditambah 7 jam offset, maka menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-02 17 GMT+7"),"."),(0,i.kt)("h2",{id:"a507-unix-time"},"A.50.7. UNIX time"),(0,i.kt)("p",null,"Cara konversi data ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," ke bentuk UNIX time bisa dilakukan lewat 3 cara:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"timestamp"),", nilai baliknya dalam bentuk detik/second"),(0,i.kt)("li",{parentName:"ul"},"menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"timestamp_millis"),", nilai baliknya dalam bentuk miliseconds"),(0,i.kt)("li",{parentName:"ul"},"menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"timestamp_micros"),", nilai baliknya dalam bentuk microseconds")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let date_in_local_tz: DateTime = Local::now();\nprintln!("date: {date_in_local_tz}");\nprintln!("date (in second timestamp): {}", date_in_local_tz.timestamp());\nprintln!("date (in milisecond timestamp): {}", date_in_local_tz.timestamp_millis());\nprintln!("date (in microsecond timestamp): {}", date_in_local_tz.timestamp_micros());\n')),(0,i.kt)("p",null,"Sedangkan cara untuk mengkonversi data timestamp ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," bisa dilakukan via ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime")," ataupun ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime::from"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// via NaiveDateTime\nlet naive_date_time = NaiveDateTime::from_timestamp_millis(date_in_local_tz.timestamp_millis()).unwrap();\nlet new_datetime1 = Local.from_local_datetime(&naive_date_time).unwrap();\nprintln!("date1: {new_datetime1}");\n\n// via DateTime::from\nlet new_datetime2 = DateTime::::from(UNIX_EPOCH + Duration::from_millis(date_in_local_tz.timestamp_millis().unsigned_abs()));\nprintln!("date2: {new_datetime2}");\n')),(0,i.kt)("p",null,"Output program di atas:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(6451).Z,width:"571",height:"182"})),(0,i.kt)("h2",{id:"a508-date-formatting--parsing"},"A.50.8. Date formatting & parsing"),(0,i.kt)("p",null,"Konversi datetime ke string (atau biasa disebut dengan date formatting) dilakukan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"format")," yang dalam penerapannya membutuhkan argument ",(0,i.kt)("em",{parentName:"p"},"datetime formatting syntax"),"."),(0,i.kt)("p",null,"Sedangkan konversi dari string ke datetime dilakukan via ",(0,i.kt)("inlineCode",{parentName:"p"},"Local.datetime_from_str")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc.datetime_from_str"),"."),(0,i.kt)("p",null,"Contoh pengaplikasian keduanya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let date1: DateTime = Local::now();\nprintln!("date1 (in local): {}", date1);\n// date1 (in local): 2023-03-02 18:13:39.954831600 +07:00\n\nlet str_from_date1 = date1.format("%Y-%m-%d %H:%M:%S %z").to_string();\nprintln!("date1 (in string): {}", str_from_date1);\n// date1 (in string): 2023-03-02 18:13:39 +0700\n\nlet date1_from_str = Local.datetime_from_str(&str_from_date1, "%Y-%m-%d %H:%M:%S %z").unwrap();\nprintln!("date1 (in local): {}", date1_from_str);\n// date1 (in local): 2023-03-02 18:13:39 +07:00\n\nlet date2_from_str = Utc.datetime_from_str("03/01/2023 13:04 +0000", "%m/%d/%Y %H:%M %z").unwrap();\nprintln!("date2 (in utc): {}", date2_from_str);\n// date2 (in utc): 2023-03-01 13:04:00 UTC\n\nlet str_from_date2 = date2_from_str.format("%Y-%m-%d %H:%M:%S %z").to_string();\nprintln!("date2 (in string): {}", str_from_date2);\n// date2 (in string): 2023-03-01 13:04:00 +0000\n')),(0,i.kt)("h2",{id:"a509-datetime-formatting-syntax"},"A.50.9. DateTime formatting syntax"),(0,i.kt)("p",null,"Berikut adalah tabel formatting syntax untuk tipe data datetime."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Selengkapnya silakan cek ",(0,i.kt)("a",{parentName:"p",href:"https://docs.rs/chrono/latest/chrono/format/strftime/index.html"},"https://docs.rs/chrono/latest/chrono/format/strftime/index.html"))),(0,i.kt)("h3",{id:"-date-specifiers"},"\u25c9 Date specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"center"},"Example"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%Y"),(0,i.kt)("td",{parentName:"tr",align:"center"},"2001"),(0,i.kt)("td",{parentName:"tr",align:"left"},"The full proleptic Gregorian year, zero-padded to 4 digits. chrono supports years from -262144 to 262143.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%C"),(0,i.kt)("td",{parentName:"tr",align:"center"},"20"),(0,i.kt)("td",{parentName:"tr",align:"left"},"The proleptic Gregorian year divided by 100, zero-padded to 2 digits. 1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%y"),(0,i.kt)("td",{parentName:"tr",align:"center"},"01"),(0,i.kt)("td",{parentName:"tr",align:"left"},"The proleptic Gregorian year modulo 100, zero-padded to 2 digits. 1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%m"),(0,i.kt)("td",{parentName:"tr",align:"center"},"07"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Month number (01\u201312), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%b"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Jul"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Abbreviated month name. Always 3 letters.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%B"),(0,i.kt)("td",{parentName:"tr",align:"center"},"July"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Full month name. Also accepts corresponding abbreviation in parsing.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%h"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Jul"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %b.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%d"),(0,i.kt)("td",{parentName:"tr",align:"center"},"08"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Day number (01\u201331), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%e"),(0,i.kt)("td",{parentName:"tr",align:"center"},"8"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %d but space-padded. Same as %_d.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%a"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Sun"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Abbreviated weekday name. Always 3 letters.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%A"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Sunday"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Full weekday name. Also accepts corresponding abbreviation in parsing.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%w"),(0,i.kt)("td",{parentName:"tr",align:"center"},"0"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Sunday = 0, Monday = 1, \u2026, Saturday = 6.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%u"),(0,i.kt)("td",{parentName:"tr",align:"center"},"7"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Monday = 1, Tuesday = 2, \u2026, Sunday = 7. (ISO 8601)")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%U"),(0,i.kt)("td",{parentName:"tr",align:"center"},"28"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Week number starting with Sunday (00\u201353), zero-padded to 2 digits. 2")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%W"),(0,i.kt)("td",{parentName:"tr",align:"center"},"27"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %U, but week 1 starts with the first Monday in that year instead.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%G"),(0,i.kt)("td",{parentName:"tr",align:"center"},"2001"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %Y but uses the year number in ISO 8601 week date. 3")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%g"),(0,i.kt)("td",{parentName:"tr",align:"center"},"01"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %y but uses the year number in ISO 8601 week date. 3")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%V"),(0,i.kt)("td",{parentName:"tr",align:"center"},"27"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %U but uses the week number in ISO 8601 week date (01\u201353). 3")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%j"),(0,i.kt)("td",{parentName:"tr",align:"center"},"189"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Day of the year (001\u2013366), zero-padded to 3 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%D"),(0,i.kt)("td",{parentName:"tr",align:"center"},"07/08/01"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Month-day-year format. Same as %m/%d/%y.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%x"),(0,i.kt)("td",{parentName:"tr",align:"center"},"07/08/01"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Locale\u2019s date representation (e.g., 12/31/99).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%F"),(0,i.kt)("td",{parentName:"tr",align:"center"},"2001-07-08"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Year-month-day format (ISO 8601). Same as %Y-%m-%d.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%v"),(0,i.kt)("td",{parentName:"tr",align:"center"},"8-Jul-2001"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Day-month-year format. Same as %e-%b-%Y.")))),(0,i.kt)("h3",{id:"-time-specifiers"},"\u25c9 Time specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"center"},"Example"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%H"),(0,i.kt)("td",{parentName:"tr",align:"center"},"00"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour number (00\u201323), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%k"),(0,i.kt)("td",{parentName:"tr",align:"center"},"0"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %H but space-padded. Same as %_H.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%I"),(0,i.kt)("td",{parentName:"tr",align:"center"},"12"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour number in 12-hour clocks (01\u201312), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%l"),(0,i.kt)("td",{parentName:"tr",align:"center"},"12"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %I but space-padded. Same as %_I.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%P"),(0,i.kt)("td",{parentName:"tr",align:"center"},"am"),(0,i.kt)("td",{parentName:"tr",align:"left"},"am or pm in 12-hour clocks.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%p"),(0,i.kt)("td",{parentName:"tr",align:"center"},"AM"),(0,i.kt)("td",{parentName:"tr",align:"left"},"AM or PM in 12-hour clocks.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%M"),(0,i.kt)("td",{parentName:"tr",align:"center"},"34"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Minute number (00\u201359), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%S"),(0,i.kt)("td",{parentName:"tr",align:"center"},"60"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Second number (00\u201360), zero-padded to 2 digits. 4")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%f"),(0,i.kt)("td",{parentName:"tr",align:"center"},"026490000"),(0,i.kt)("td",{parentName:"tr",align:"left"},"The fractional seconds (in nanoseconds) since last whole second. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%.f"),(0,i.kt)("td",{parentName:"tr",align:"center"},".026490"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to .%f but left-aligned. These all consume the leading dot. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%.3f"),(0,i.kt)("td",{parentName:"tr",align:"center"},".026"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to .%f but left-aligned but fixed to a length of 3. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%.6f"),(0,i.kt)("td",{parentName:"tr",align:"center"},".026490"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to .%f but left-aligned but fixed to a length of 6. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%.9f"),(0,i.kt)("td",{parentName:"tr",align:"center"},".026490000"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to .%f but left-aligned but fixed to a length of 9. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%3f"),(0,i.kt)("td",{parentName:"tr",align:"center"},"026"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to %.3f but without the leading dot. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%6f"),(0,i.kt)("td",{parentName:"tr",align:"center"},"026490"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to %.6f but without the leading dot. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%9f"),(0,i.kt)("td",{parentName:"tr",align:"center"},"026490000"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to %.9f but without the leading dot. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%R"),(0,i.kt)("td",{parentName:"tr",align:"center"},"00:34"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour-minute format. Same as %H:%M.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%T"),(0,i.kt)("td",{parentName:"tr",align:"center"},"00:34:60"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour-minute-second format. Same as %H:%M:%S.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%X"),(0,i.kt)("td",{parentName:"tr",align:"center"},"00:34:60"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Locale\u2019s time representation (e.g., 23:13:48).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%r"),(0,i.kt)("td",{parentName:"tr",align:"center"},"12:34:60 AM"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour-minute-second format in 12-hour clocks. Same as %I:%M:%S %p.")))),(0,i.kt)("h3",{id:"-timezone-specifiers"},"\u25c9 Timezone specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"center"},"Example"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%Z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"ACST"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Local time zone name. Skips all non-whitespace characters during parsing. 6")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+0930"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Offset from the local time to UTC (with UTC being +0000).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%:z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+09:30"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %z but with a colon.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%::z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+09:30:00"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Offset from the local time to UTC with seconds.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%:::z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+09"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Offset from the local time to UTC without minutes.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%#z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+09"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Parsing only: Same as %z but allows minutes to be missing or present.")))),(0,i.kt)("h3",{id:"-date--time-specifiers"},"\u25c9 Date & time specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"center"},"Example"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%c"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Sun Jul 8 00:34:60 2001"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Locale\u2019s date and time (e.g., Thu Mar 3 23:05:25 2005).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%+"),(0,i.kt)("td",{parentName:"tr",align:"center"},"2001-07-08T00:34:60.026490+09:30"),(0,i.kt)("td",{parentName:"tr",align:"left"},"ISO 8601 / RFC 3339 date & time format. 7")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%s"),(0,i.kt)("td",{parentName:"tr",align:"center"},"994518299"),(0,i.kt)("td",{parentName:"tr",align:"left"},"UNIX timestamp, the number of seconds since 1970-01-01 00:00 UTC. 8")))),(0,i.kt)("h3",{id:"-special-specifiers"},"\u25c9 Special specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%t"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Literal tab (\\t).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%n"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Literal newline (\\n).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%%"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Literal percent sign.")))),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/datetime"},"github.com/novalagung/dasarpemrogramanrust-example/../datetime")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/result-type"},"Tipe Data \u279c Result")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generic"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/time/index.html"},"https://doc.rust-lang.org/std/time/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://docs.rs/time/latest/time/"},"https://docs.rs/time/latest/time/")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://github.com/rust-lang/rust-wiki-backup/blob/master/Lib-datetime.md"},"https://github.com/rust-lang/rust-wiki-backup/blob/master/Lib-datetime.md")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://docs.rs/chrono/latest/chrono/"},"https://docs.rs/chrono/latest/chrono/"))))}k.isMDXComponent=!0},6635:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-1-c0119211ce2f1f5f14eaf24316a52b54.png"},239:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-2-998514b9bc7ad92870392305578f2a14.png"},2330:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-3-52a5ccd1fb458c09958eafdba3f6c0d4.png"},7934:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-4-62b94d0c3e24bea70a3355c814ed8d79.png"},6451:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-5-eb678ecf97453409d153b4f9727e6347.png"}}]); \ No newline at end of file diff --git a/assets/js/af6aab29.fff2a138.js b/assets/js/af6aab29.fff2a138.js deleted file mode 100644 index 2aa4ed4d..00000000 --- a/assets/js/af6aab29.fff2a138.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8794],{3905:(e,t,a)=>{a.d(t,{Zo:()=>o,kt:()=>c});var n=a(7294);function i(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function r(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,n)}return a}function l(e){for(var t=1;t=0||(i[a]=e[a]);return i}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(i[a]=e[a])}return i}var d=n.createContext({}),p=function(e){var t=n.useContext(d),a=t;return e&&(a="function"==typeof e?e(t):l(l({},t),e)),a},o=function(e){var t=p(e.components);return n.createElement(d.Provider,{value:t},e.children)},k={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},s=n.forwardRef((function(e,t){var a=e.components,i=e.mdxType,r=e.originalType,d=e.parentName,o=m(e,["components","mdxType","originalType","parentName"]),s=p(a),c=i,u=s["".concat(d,".").concat(c)]||s[c]||k[c]||r;return a?n.createElement(u,l(l({ref:t},o),{},{components:a})):n.createElement(u,l({ref:t},o))}));function c(e,t){var a=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var r=a.length,l=new Array(r);l[0]=s;var m={};for(var d in t)hasOwnProperty.call(t,d)&&(m[d]=t[d]);m.originalType=e,m.mdxType="string"==typeof e?e:i,l[1]=m;for(var p=2;p{a.r(t),a.d(t,{assets:()=>d,contentTitle:()=>l,default:()=>k,frontMatter:()=>r,metadata:()=>m,toc:()=>p});var n=a(7462),i=(a(7294),a(3905));const r={sidebar_position:50,title:"A.50. Date Time",sidebar_label:"A.50. Date Time"},l=void 0,m={unversionedId:"basic/datetime",id:"basic/datetime",title:"A.50. Date Time",description:"Pada chapter ini kita akan mempelajari tentang tipe data datetime yang ada di Rust programming.",source:"@site/docs/basic/datetime.md",sourceDirName:"basic",slug:"/basic/datetime",permalink:"/basic/datetime",draft:!1,tags:[],version:"current",sidebarPosition:50,frontMatter:{sidebar_position:50,title:"A.50. Date Time",sidebar_label:"A.50. Date Time"},sidebar:"tutorialSidebar",previous:{title:"A.49. Attributes",permalink:"/basic/attributes"},next:{title:"A.51. File, Path, Directory",permalink:"/basic/file-path-directory"}},d={},p=[{value:"A.50.1. Persiapan",id:"a501-persiapan",level:2},{value:"A.50.2. Tipe data DateTime",id:"a502-tipe-data-datetime",level:2},{value:"\u25c9 Tipe DateTime<Utc>",id:"-tipe-datetimeutc",level:3},{value:"\u25c9 Tipe DateTime<Local>",id:"-tipe-datetimelocal",level:3},{value:"\u25c9 Tipe NaiveDateTime",id:"-tipe-naivedatetime",level:3},{value:"A.50.3. Pembuatan objek DateTime",id:"a503-pembuatan-objek-datetime",level:2},{value:"\u25c9 Via Utc::now()",id:"-via-utcnow",level:3},{value:"\u25c9 Via Utc.with_ymd_and_hms()",id:"-via-utcwith_ymd_and_hms",level:3},{value:"\u25c9 Via DateTime::<Utc>::from()",id:"-via-datetimeutcfrom",level:3},{value:"\u25c9 Via String method parse()",id:"-via-string-method-parse",level:3},{value:"\u25c9 Cara lainnya",id:"-cara-lainnya",level:3},{value:"A.50.4. Utc vs Local",id:"a504-utc-vs-local",level:2},{value:"A.50.5. Tipe NaiveDateTime",id:"a505-tipe-naivedatetime",level:2},{value:"A.50.6. Konversi datetime UTC ke local timezone, dan sebaliknya",id:"a506-konversi-datetime-utc-ke-local-timezone-dan-sebaliknya",level:2},{value:"A.50.7. UNIX time",id:"a507-unix-time",level:2},{value:"A.50.8. Date formatting & parsing",id:"a508-date-formatting--parsing",level:2},{value:"A.50.9. DateTime formatting syntax",id:"a509-datetime-formatting-syntax",level:2},{value:"\u25c9 Date specifiers",id:"-date-specifiers",level:3},{value:"\u25c9 Time specifiers",id:"-time-specifiers",level:3},{value:"\u25c9 Timezone specifiers",id:"-timezone-specifiers",level:3},{value:"\u25c9 Date & time specifiers",id:"-date--time-specifiers",level:3},{value:"\u25c9 Special specifiers",id:"-special-specifiers",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:p};function k(e){let{components:t,...r}=e;return(0,i.kt)("wrapper",(0,n.Z)({},o,r,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan mempelajari tentang tipe data datetime yang ada di Rust programming."),(0,i.kt)("p",null,"Rust menyediakan cukup banyak API untuk keperluan pengolahan datetime, tersedia dalam Standard Library crate. Namun pada chapter ini kita tidak akan menggunakannya, karena yang akan dipakai adalah ",(0,i.kt)("a",{parentName:"p",href:"https://crates.io/crates/chrono"},(0,i.kt)("inlineCode",{parentName:"a"},"chrono")," crate")," (dianjurkan untuk menggunakan crate ini karena fiturnya lebih banyak)."),(0,i.kt)("h2",{id:"a501-persiapan"},"A.50.1. Persiapan"),(0,i.kt)("p",null,"Sebelum memulai pembelajaran, siapkan satu project baru, kemudian tambahkan care ",(0,i.kt)("inlineCode",{parentName:"p"},"crono")," dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"Cargo.toml"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-toml"},'[package]\nname = "time_chrono"\nversion = "0.1.0"\nedition = "2021"\n\n[dependencies]\nchrono = "0.4.23"\n')),(0,i.kt)("h2",{id:"a502-tipe-data-datetime"},"A.50.2. Tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"DateTime")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"crono")," menyediakan cukup banyak tipe data untuk merepresentasikan informasi tanggal dan waktu, tetapi yang paling penting untuk diketahui dan dipahami ada dua, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"DateTime")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"DateTime")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"NaiveDateTime"))),(0,i.kt)("h3",{id:"-tipe-datetimeutc"},"\u25c9 Tipe ",(0,i.kt)("inlineCode",{parentName:"h3"},"DateTime")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," (gabungan dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::datetime::DateTime")," dan generic ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::offset::utc::Utc"),") adalah representasi untuk tipe data datetime dengan timezone offset ",(0,i.kt)("strong",{parentName:"p"},"UTC")," (atau ",(0,i.kt)("strong",{parentName:"p"},"GMT+0"),")."),(0,i.kt)("h3",{id:"-tipe-datetimelocal"},"\u25c9 Tipe ",(0,i.kt)("inlineCode",{parentName:"h3"},"DateTime")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," (gabungan dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::datetime::DateTime")," dan generic ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::offset::local::Local"),") adalah representasi untuk tipe data datetime dengan ",(0,i.kt)("strong",{parentName:"p"},"timezone offset sesuai dengan dimana program dijalankan"),"."),(0,i.kt)("p",null,"Sebagai contoh, penulis berlokasi di Jawa Timur, maka ketika ada suatu data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," artinya timezone offset-nya adalah WIB (atau GMT+7)."),(0,i.kt)("h3",{id:"-tipe-naivedatetime"},"\u25c9 Tipe ",(0,i.kt)("inlineCode",{parentName:"h3"},"NaiveDateTime")),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"chrono::naive::datetime::NaiveDateTime")," merupakan tipe data datetime yang tidak menyimpan informasi timezone. Nanti akan kita bahas apa kegunaan dari tipe ini."),(0,i.kt)("h2",{id:"a503-pembuatan-objek-datetime"},"A.50.3. Pembuatan objek ",(0,i.kt)("inlineCode",{parentName:"h2"},"DateTime")),(0,i.kt)("h3",{id:"-via-utcnow"},"\u25c9 Via ",(0,i.kt)("inlineCode",{parentName:"h3"},"Utc::now()")),(0,i.kt)("p",null,"Ada banyak cara untuk membuat date time, dan kita akan bahas satu-per-satu."),(0,i.kt)("p",null,"Cara yang pertama, mari kita pelajari sambil praktek. Isi file program ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs")," dengan kode berikut, lalu jalankan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use chrono::prelude::*;\nuse std::time::{UNIX_EPOCH, Duration};\n\nfn main() {\n let sample_date1_in_utc: DateTime = Utc::now();\n println!("sample date 1 (in utc): {sample_date1_in_utc}");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(6635).Z,width:"571",height:"95"})),(0,i.kt)("p",null,"Contoh sederhana di atas adalah salah satu cara membuat object datetime dengan isi adalah informasi waktu sekarang (waktu ketika baris kode tersebut dieksekusi)."),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc::now()")," mengembalikan informasi waktu sekarang dalam timezone UTC (representasi tipe data-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime"),")."),(0,i.kt)("p",null,"Module item ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc")," sendiri harus diimport terlebih dahulu agar bisa digunakan. Itulah kenapa ada statement ",(0,i.kt)("inlineCode",{parentName:"p"},"use chrono::prelude::*"),"."),(0,i.kt)("h3",{id:"-via-utcwith_ymd_and_hms"},"\u25c9 Via ",(0,i.kt)("inlineCode",{parentName:"h3"},"Utc.with_ymd_and_hms()")),(0,i.kt)("p",null,"Pembuatan object datetime dengan isi informasi waktu spesifik bisa dilakukan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc.with_ymd_and_hms()"),". Fungsi ini mewajibkan developer untuk mengisi 6 parameternya dengan segment waktu (tahun, bulan, dan lainnya)."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sample_date2_in_utc = Utc.with_ymd_and_hms(2023, 3, 1, 1, 2, 3).unwrap();\nprintln!("sample date 2 (in utc): {sample_date2_in_utc}");\n// output: sample date 2 (in utc): 2023-03-01 01:02:03 UTC\n')),(0,i.kt)("p",null,"Statement di atas menghasilkan data datetime dengan isi ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-01 01:02:03 UTC"),"."),(0,i.kt)("h3",{id:"-via-datetimeutcfrom"},"\u25c9 Via ",(0,i.kt)("inlineCode",{parentName:"h3"},"DateTime::::from()")),(0,i.kt)("p",null,"Cara ini pas digunakan pada situasi dimana kita perlu membuat object datetime dari sebuah ",(0,i.kt)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Unix_time"},"UNIX timestamp"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sample_date3_in_utc = DateTime::::from(UNIX_EPOCH + Duration::from_secs(1524885322));\nprintln!("sample date 3 (in utc): {sample_date3_in_utc}");\n// output: sample date 3 (in utc): 2018-04-28 03:15:22 UTC\n')),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime::::from()")," di atas menghasilkan data datetime ",(0,i.kt)("inlineCode",{parentName:"p"},"2018-04-28 03:15:22 UTC"),", karena nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"1524885322")," adalah representasi UNIX time untuk waktu tersebut."),(0,i.kt)("h3",{id:"-via-string-method-parse"},"\u25c9 Via String method ",(0,i.kt)("inlineCode",{parentName:"h3"},"parse()")),(0,i.kt)("p",null,"Tipe data string bisa langsung dikonversi ke bentuk datetime menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"parse()")," yang dikombinasikan dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," sebagai generic parameter."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let sample_date4_in_utc = "2023-03-01 01:02:03 UTC".parse::>().unwrap();\nprintln!("sample date 4 (in utc): {sample_date4_in_utc}");\n// output: sample date 4 (in utc): 2023-03-01 01:02:03 UTC\n')),(0,i.kt)("p",null,"Statement parsing di atas menghasilkan data datetime ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-01 01:02:03 UTC"),", sesuai dengan isi data string."),(0,i.kt)("p",null,"O iya, perlu diketahui bahwa tidak semua string bisa di-parse ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime"),". Jika data string memiliki format yang tidak sesuai dengan aturan parsing di Rust, maka proses parsing menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"parse()")," pasti menghasilkan error."),(0,i.kt)("p",null,"Penerapan method ",(0,i.kt)("inlineCode",{parentName:"p"},"parse::>()")," sebenarnya mengembalikan tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result, ParseError>"),", itulah kenapa pada contoh perlu di chain lagi dengan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih detailnya mengenai tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Result")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,i.kt)("h3",{id:"-cara-lainnya"},"\u25c9 Cara lainnya"),(0,i.kt)("p",null,"Ada beberapa cara lain yang bisa digunakan untuk pembuatan object datetime, dan kita akan bahas juga sebentar lagi."),(0,i.kt)("h2",{id:"a504-utc-vs-local"},"A.50.4. ",(0,i.kt)("inlineCode",{parentName:"h2"},"Utc")," vs ",(0,i.kt)("inlineCode",{parentName:"h2"},"Local")),(0,i.kt)("p",null,"Kesemua cara yang sudah dipelajari di atas juga bisa diterapkan dalam pembuatan object date time dengan timezone offset ",(0,i.kt)("inlineCode",{parentName:"p"},"Local"),". Penerapannya kurang lebih sama, cukup ganti ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc")," menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"Local"),"."),(0,i.kt)("p",null,"Jika disandingkan, akan seperti ini kodenya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let date1_in_utc: DateTime = Utc::now();\nlet date2_in_utc = Utc.with_ymd_and_hms(2023, 3, 1, 1, 2, 3).unwrap();\nlet date3_in_utc = DateTime::::from(UNIX_EPOCH + Duration::from_secs(1524885322));\nlet date4_in_utc = "2023-03-01 01:02:03 UTC".parse::>().unwrap();\n\nprintln!("sample date 1 (in utc): {date1_in_utc}");\nprintln!("sample date 2 (in utc): {date2_in_utc}");\nprintln!("sample date 3 (in utc): {date3_in_utc}");\nprintln!("sample date 4 (in utc): {date4_in_utc}");\n\n// vs\n\nlet date1_in_local_tz: DateTime = Local::now();\nlet date2_in_local_tz = Local.with_ymd_and_hms(2023, 3, 1, 1, 2, 3).unwrap();\nlet date3_in_local_tz = DateTime::::from(UNIX_EPOCH + Duration::from_secs(1524885322));\nlet date4_in_local_tz = "2023-03-01 01:02:03 UTC".parse::>().unwrap();\n\nprintln!("sample date 1 (in local_tz): {date1_in_local_tz}");\nprintln!("sample date 2 (in local_tz): {date2_in_local_tz}");\nprintln!("sample date 3 (in local_tz): {date3_in_local_tz}");\nprintln!("sample date 4 (in local_tz): {date4_in_local_tz}");\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(239).Z,width:"571",height:"231"})),(0,i.kt)("p",null,"Semoga cukup jelas."),(0,i.kt)("p",null,"Dari contoh di atas, sebenarnya ada 2 variabel yang menarik untuk dibahas, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"sample_date1_in_utc")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"sample_date1_in_local_tz"),". Dua variabel tersebut berisi informasi waktu sekarang, perbedaannya hanya pada timezone offset saja. Namun, meski timezone offset-nya berbeda, kedua datetime tersebut sejatinya adalah merepresentasikan waktu yang sama (ekuivalen)."),(0,i.kt)("p",null,"Agar lebih jelas, silakan perhatikan contoh berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash"},"sample date 1 (in utc): 2023-03-02 09:53:44.561519500 UTC\nsample date 1 (in local_tz): 2023-03-02 16:53:44.562389600 +07:00\n")),(0,i.kt)("p",null,"Kedua datetime tersebut terpaku ",(0,i.kt)("strong",{parentName:"p"},"7 jam")," persis. Data ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-02 09:53:44")," UTC (atau GMT+0), jika dikonversi ke waktu lokal penulis (WIB atau GMT+7) maka akan menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-02 16:53:44"),". Selisihnya sesuai dengan timezone offset, yaitu 7 jam."),(0,i.kt)("h2",{id:"a505-tipe-naivedatetime"},"A.50.5. Tipe ",(0,i.kt)("inlineCode",{parentName:"h2"},"NaiveDateTime")),(0,i.kt)("p",null,"Rust memilik caranya sendiri dalam pengelolaan data datetime (yang menurut penulis cukup unik). Ada satu tipe data bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime"),", yang kegunaannya adalah untuk menampung tipe data datetime ",(0,i.kt)("strong",{parentName:"p"},"tanpa spesifik timezone offset"),"."),(0,i.kt)("p",null,"Tipe data ini sangat berguna dibeberapa case yang kebanyakan adalah perihal konversi data ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime"),"."),(0,i.kt)("p",null,"Mari kita praktekan dengan contoh agar lebih jelas. Dimisalkan ada keperluan dimana data UNIX time perlu di konversi ke 2 bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime"),", dengan timezone offset ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Local"),". Pada kasus ini, cara ke-3 dari praktek sebelumnya bisa dilakukan untuk penyelesaian case ini."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let timestamp: u64 = 1524885322;\nlet date2_in_utc = DateTime::::from(UNIX_EPOCH + Duration::from_secs(timestamp));\nlet date2_in_local_tz = DateTime::::from(UNIX_EPOCH + Duration::from_secs(timestamp));\n")),(0,i.kt)("p",null,"Cara lain adalah dengan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime"),", yang contoh penerapannnya kurang lebih seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let timestamp_in_ms: i64 = 1524885322000;\nlet naive_date_time: NaiveDateTime = NaiveDateTime::from_timestamp_millis(timestamp_in_ms).unwrap();\n\nlet sample_date_in_utc = Utc.from_utc_datetime(&naive_date_time);\nlet sample_date_in_local_tz = Local.from_local_datetime(&naive_date_time).unwrap();\n\nprintln!("sample date 1 (in utc): {sample_date_in_utc}");\nprintln!("sample date 2 (in local_tz): {sample_date_in_local_tz}");\n\n// output:\n// sample date 1 (in utc): 2018-04-28 03:15:22 UTC\n// sample date 2 (in local_tz): 2018-04-28 03:15:22 +07:00\n')),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime::from_timestamp_millis")," berguna untuk mengkonversi data UNIX time (milisecond) ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime"),". Kemudian, dari data tersebut dapat kita konversi lagi ke bentuk UTC dengan mudah menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc.from_utc_datetime()")," dan local timezone menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"Local.from_local_datetime()"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(2330).Z,width:"571",height:"115"})),(0,i.kt)("p",null,"Bisa dilihat pada output di atas, bahwa hasil data datetime berbeda untuk timezone UTC dan local."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Jangan sampai bingung, 2 data datetime di atas tidak sama dan tidak ekuivalen meskipun informasi jam menit detiknya sama persis. Hal ini dikarenakan timezone offset-nya berbeda.")),(0,i.kt)("h2",{id:"a506-konversi-datetime-utc-ke-local-timezone-dan-sebaliknya"},"A.50.6. Konversi datetime UTC ke local timezone, dan sebaliknya"),(0,i.kt)("p",null,"Dalam bekerja dengan tipe data datetime, selalu memperhatikan timezone adalah hal yang sangat penting. Bisa saja dua buah datetime memiliki informasi tanggal dan waktu sama persis, tapi jika timezone-nya berbeda maka keduanya adalah berbeda!"),(0,i.kt)("p",null,"Agar lebih jelas, silakan perhatikan kode berikut beserta outputnya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let date1_in_local_tz: DateTime = Local::now();\nprintln!("date (in local): {date1_in_local_tz}");\n\nlet date_in_utc = DateTime::::from(date1_in_local_tz);\nprintln!("date (in utc): {date_in_utc}");\n\nlet date2_in_local_tz = DateTime::::from(date_in_utc);\nprintln!("date (in local): {date2_in_local_tz}");\n\n// output:\n// date (in local): 2023-03-02 17:33:37.750279900 +07:00\n// date (in utc): 2023-03-02 10:33:37.750279900 UTC\n// date (in local): 2023-03-02 17:33:37.750279900 +07:00\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(7934).Z,width:"571",height:"129"})),(0,i.kt)("p",null,"Ketiga data datetime di atas adalah ekuivalen:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"date1_in_local_tz")," berisi informasi datetime waktu sekarang, dengan timezone GMT+7."),(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"date_in_utc")," adalah datetime yang ekuivalen dengan ",(0,i.kt)("inlineCode",{parentName:"li"},"date1_in_local_tz"),", hanya saja direpresentasikan dalam timezone UTC."),(0,i.kt)("li",{parentName:"ul"},"Variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"date2_in_local_tz")," juga ekuivalen dengan ",(0,i.kt)("inlineCode",{parentName:"li"},"date_in_utc")," dan juga ",(0,i.kt)("inlineCode",{parentName:"li"},"date1_in_local_tz"),".")),(0,i.kt)("p",null,"Data ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-02 10 UTC")," jika ditambah 7 jam offset, maka menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"2023-03-02 17 GMT+7"),"."),(0,i.kt)("h2",{id:"a507-unix-time"},"A.50.7. UNIX time"),(0,i.kt)("p",null,"Cara konversi data ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," ke bentuk UNIX time bisa dilakukan lewat 3 cara:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"timestamp"),", nilai baliknya dalam bentuk detik/second"),(0,i.kt)("li",{parentName:"ul"},"menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"timestamp_millis"),", nilai baliknya dalam bentuk miliseconds"),(0,i.kt)("li",{parentName:"ul"},"menggunakan method ",(0,i.kt)("inlineCode",{parentName:"li"},"timestamp_micros"),", nilai baliknya dalam bentuk microseconds")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let date_in_local_tz: DateTime = Local::now();\nprintln!("date: {date_in_local_tz}");\nprintln!("date (in second timestamp): {}", date_in_local_tz.timestamp());\nprintln!("date (in milisecond timestamp): {}", date_in_local_tz.timestamp_millis());\nprintln!("date (in microsecond timestamp): {}", date_in_local_tz.timestamp_micros());\n')),(0,i.kt)("p",null,"Sedangkan cara untuk mengkonversi data timestamp ke bentuk ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime")," bisa dilakukan via ",(0,i.kt)("inlineCode",{parentName:"p"},"NaiveDateTime")," ataupun ",(0,i.kt)("inlineCode",{parentName:"p"},"DateTime::from"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// via NaiveDateTime\nlet naive_date_time = NaiveDateTime::from_timestamp_millis(date_in_local_tz.timestamp_millis()).unwrap();\nlet new_datetime1 = Local.from_local_datetime(&naive_date_time).unwrap();\nprintln!("date1: {new_datetime1}");\n\n// via DateTime::from\nlet new_datetime2 = DateTime::::from(UNIX_EPOCH + Duration::from_millis(date_in_local_tz.timestamp_millis().unsigned_abs()));\nprintln!("date2: {new_datetime2}");\n')),(0,i.kt)("p",null,"Output program di atas:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"DateTime",src:a(6451).Z,width:"571",height:"182"})),(0,i.kt)("h2",{id:"a508-date-formatting--parsing"},"A.50.8. Date formatting & parsing"),(0,i.kt)("p",null,"Konversi datetime ke string (atau biasa disebut dengan date formatting) dilakukan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"format")," yang dalam penerapannya membutuhkan argument ",(0,i.kt)("em",{parentName:"p"},"datetime formatting syntax"),"."),(0,i.kt)("p",null,"Sedangkan konversi dari string ke datetime dilakukan via ",(0,i.kt)("inlineCode",{parentName:"p"},"Local.datetime_from_str")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"Utc.datetime_from_str"),"."),(0,i.kt)("p",null,"Contoh pengaplikasian keduanya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let date1: DateTime = Local::now();\nprintln!("date1 (in local): {}", date1);\n// date1 (in local): 2023-03-02 18:13:39.954831600 +07:00\n\nlet str_from_date1 = date1.format("%Y-%m-%d %H:%M:%S %z").to_string();\nprintln!("date1 (in string): {}", str_from_date1);\n// date1 (in string): 2023-03-02 18:13:39 +0700\n\nlet date1_from_str = Local.datetime_from_str(&str_from_date1, "%Y-%m-%d %H:%M:%S %z").unwrap();\nprintln!("date1 (in local): {}", date1_from_str);\n// date1 (in local): 2023-03-02 18:13:39 +07:00\n\nlet date2_from_str = Utc.datetime_from_str("03/01/2023 13:04 +0000", "%m/%d/%Y %H:%M %z").unwrap();\nprintln!("date2 (in utc): {}", date2_from_str);\n// date2 (in utc): 2023-03-01 13:04:00 UTC\n\nlet str_from_date2 = date2_from_str.format("%Y-%m-%d %H:%M:%S %z").to_string();\nprintln!("date2 (in string): {}", str_from_date2);\n// date2 (in string): 2023-03-01 13:04:00 +0000\n')),(0,i.kt)("h2",{id:"a509-datetime-formatting-syntax"},"A.50.9. DateTime formatting syntax"),(0,i.kt)("p",null,"Berikut adalah tabel formatting syntax untuk tipe data datetime."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Selengkapnya silakan cek ",(0,i.kt)("a",{parentName:"p",href:"https://docs.rs/chrono/latest/chrono/format/strftime/index.html"},"https://docs.rs/chrono/latest/chrono/format/strftime/index.html"))),(0,i.kt)("h3",{id:"-date-specifiers"},"\u25c9 Date specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"center"},"Example"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%Y"),(0,i.kt)("td",{parentName:"tr",align:"center"},"2001"),(0,i.kt)("td",{parentName:"tr",align:"left"},"The full proleptic Gregorian year, zero-padded to 4 digits. chrono supports years from -262144 to 262143.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%C"),(0,i.kt)("td",{parentName:"tr",align:"center"},"20"),(0,i.kt)("td",{parentName:"tr",align:"left"},"The proleptic Gregorian year divided by 100, zero-padded to 2 digits. 1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%y"),(0,i.kt)("td",{parentName:"tr",align:"center"},"01"),(0,i.kt)("td",{parentName:"tr",align:"left"},"The proleptic Gregorian year modulo 100, zero-padded to 2 digits. 1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%m"),(0,i.kt)("td",{parentName:"tr",align:"center"},"07"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Month number (01\u201312), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%b"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Jul"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Abbreviated month name. Always 3 letters.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%B"),(0,i.kt)("td",{parentName:"tr",align:"center"},"July"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Full month name. Also accepts corresponding abbreviation in parsing.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%h"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Jul"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %b.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%d"),(0,i.kt)("td",{parentName:"tr",align:"center"},"08"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Day number (01\u201331), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%e"),(0,i.kt)("td",{parentName:"tr",align:"center"},"8"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %d but space-padded. Same as %_d.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%a"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Sun"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Abbreviated weekday name. Always 3 letters.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%A"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Sunday"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Full weekday name. Also accepts corresponding abbreviation in parsing.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%w"),(0,i.kt)("td",{parentName:"tr",align:"center"},"0"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Sunday = 0, Monday = 1, \u2026, Saturday = 6.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%u"),(0,i.kt)("td",{parentName:"tr",align:"center"},"7"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Monday = 1, Tuesday = 2, \u2026, Sunday = 7. (ISO 8601)")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%U"),(0,i.kt)("td",{parentName:"tr",align:"center"},"28"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Week number starting with Sunday (00\u201353), zero-padded to 2 digits. 2")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%W"),(0,i.kt)("td",{parentName:"tr",align:"center"},"27"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %U, but week 1 starts with the first Monday in that year instead.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%G"),(0,i.kt)("td",{parentName:"tr",align:"center"},"2001"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %Y but uses the year number in ISO 8601 week date. 3")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%g"),(0,i.kt)("td",{parentName:"tr",align:"center"},"01"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %y but uses the year number in ISO 8601 week date. 3")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%V"),(0,i.kt)("td",{parentName:"tr",align:"center"},"27"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %U but uses the week number in ISO 8601 week date (01\u201353). 3")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%j"),(0,i.kt)("td",{parentName:"tr",align:"center"},"189"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Day of the year (001\u2013366), zero-padded to 3 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%D"),(0,i.kt)("td",{parentName:"tr",align:"center"},"07/08/01"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Month-day-year format. Same as %m/%d/%y.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%x"),(0,i.kt)("td",{parentName:"tr",align:"center"},"07/08/01"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Locale\u2019s date representation (e.g., 12/31/99).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%F"),(0,i.kt)("td",{parentName:"tr",align:"center"},"2001-07-08"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Year-month-day format (ISO 8601). Same as %Y-%m-%d.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%v"),(0,i.kt)("td",{parentName:"tr",align:"center"},"8-Jul-2001"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Day-month-year format. Same as %e-%b-%Y.")))),(0,i.kt)("h3",{id:"-time-specifiers"},"\u25c9 Time specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"center"},"Example"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%H"),(0,i.kt)("td",{parentName:"tr",align:"center"},"00"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour number (00\u201323), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%k"),(0,i.kt)("td",{parentName:"tr",align:"center"},"0"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %H but space-padded. Same as %_H.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%I"),(0,i.kt)("td",{parentName:"tr",align:"center"},"12"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour number in 12-hour clocks (01\u201312), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%l"),(0,i.kt)("td",{parentName:"tr",align:"center"},"12"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %I but space-padded. Same as %_I.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%P"),(0,i.kt)("td",{parentName:"tr",align:"center"},"am"),(0,i.kt)("td",{parentName:"tr",align:"left"},"am or pm in 12-hour clocks.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%p"),(0,i.kt)("td",{parentName:"tr",align:"center"},"AM"),(0,i.kt)("td",{parentName:"tr",align:"left"},"AM or PM in 12-hour clocks.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%M"),(0,i.kt)("td",{parentName:"tr",align:"center"},"34"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Minute number (00\u201359), zero-padded to 2 digits.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%S"),(0,i.kt)("td",{parentName:"tr",align:"center"},"60"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Second number (00\u201360), zero-padded to 2 digits. 4")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%f"),(0,i.kt)("td",{parentName:"tr",align:"center"},"026490000"),(0,i.kt)("td",{parentName:"tr",align:"left"},"The fractional seconds (in nanoseconds) since last whole second. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%.f"),(0,i.kt)("td",{parentName:"tr",align:"center"},".026490"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to .%f but left-aligned. These all consume the leading dot. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%.3f"),(0,i.kt)("td",{parentName:"tr",align:"center"},".026"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to .%f but left-aligned but fixed to a length of 3. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%.6f"),(0,i.kt)("td",{parentName:"tr",align:"center"},".026490"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to .%f but left-aligned but fixed to a length of 6. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%.9f"),(0,i.kt)("td",{parentName:"tr",align:"center"},".026490000"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to .%f but left-aligned but fixed to a length of 9. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%3f"),(0,i.kt)("td",{parentName:"tr",align:"center"},"026"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to %.3f but without the leading dot. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%6f"),(0,i.kt)("td",{parentName:"tr",align:"center"},"026490"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to %.6f but without the leading dot. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%9f"),(0,i.kt)("td",{parentName:"tr",align:"center"},"026490000"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Similar to %.9f but without the leading dot. 5")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%R"),(0,i.kt)("td",{parentName:"tr",align:"center"},"00:34"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour-minute format. Same as %H:%M.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%T"),(0,i.kt)("td",{parentName:"tr",align:"center"},"00:34:60"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour-minute-second format. Same as %H:%M:%S.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%X"),(0,i.kt)("td",{parentName:"tr",align:"center"},"00:34:60"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Locale\u2019s time representation (e.g., 23:13:48).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%r"),(0,i.kt)("td",{parentName:"tr",align:"center"},"12:34:60 AM"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Hour-minute-second format in 12-hour clocks. Same as %I:%M:%S %p.")))),(0,i.kt)("h3",{id:"-timezone-specifiers"},"\u25c9 Timezone specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"center"},"Example"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%Z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"ACST"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Local time zone name. Skips all non-whitespace characters during parsing. 6")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+0930"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Offset from the local time to UTC (with UTC being +0000).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%:z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+09:30"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Same as %z but with a colon.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%::z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+09:30:00"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Offset from the local time to UTC with seconds.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%:::z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+09"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Offset from the local time to UTC without minutes.")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%#z"),(0,i.kt)("td",{parentName:"tr",align:"center"},"+09"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Parsing only: Same as %z but allows minutes to be missing or present.")))),(0,i.kt)("h3",{id:"-date--time-specifiers"},"\u25c9 Date & time specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"center"},"Example"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%c"),(0,i.kt)("td",{parentName:"tr",align:"center"},"Sun Jul 8 00:34:60 2001"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Locale\u2019s date and time (e.g., Thu Mar 3 23:05:25 2005).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%+"),(0,i.kt)("td",{parentName:"tr",align:"center"},"2001-07-08T00:34:60.026490+09:30"),(0,i.kt)("td",{parentName:"tr",align:"left"},"ISO 8601 / RFC 3339 date & time format. 7")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%s"),(0,i.kt)("td",{parentName:"tr",align:"center"},"994518299"),(0,i.kt)("td",{parentName:"tr",align:"left"},"UNIX timestamp, the number of seconds since 1970-01-01 00:00 UTC. 8")))),(0,i.kt)("h3",{id:"-special-specifiers"},"\u25c9 Special specifiers"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:"center"},"Spec"),(0,i.kt)("th",{parentName:"tr",align:"left"},"Description"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%t"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Literal tab (\\t).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%n"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Literal newline (\\n).")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:"center"},"%%"),(0,i.kt)("td",{parentName:"tr",align:"left"},"Literal percent sign.")))),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/datetime"},"github.com/novalagung/dasarpemrogramanrust-example/../datetime")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/result-type"},"Tipe Data \u279c Result")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generic"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/time/index.html"},"https://doc.rust-lang.org/std/time/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://docs.rs/time/latest/time/"},"https://docs.rs/time/latest/time/")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://github.com/rust-lang/rust-wiki-backup/blob/master/Lib-datetime.md"},"https://github.com/rust-lang/rust-wiki-backup/blob/master/Lib-datetime.md")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://docs.rs/chrono/latest/chrono/"},"https://docs.rs/chrono/latest/chrono/"))))}k.isMDXComponent=!0},6635:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-1-c0119211ce2f1f5f14eaf24316a52b54.png"},239:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-2-998514b9bc7ad92870392305578f2a14.png"},2330:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-3-52a5ccd1fb458c09958eafdba3f6c0d4.png"},7934:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-4-62b94d0c3e24bea70a3355c814ed8d79.png"},6451:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/datetime-5-eb678ecf97453409d153b4f9727e6347.png"}}]); \ No newline at end of file diff --git a/assets/js/bc4edc5c.30092a0e.js b/assets/js/bc4edc5c.30092a0e.js new file mode 100644 index 00000000..c11e3780 --- /dev/null +++ b/assets/js/bc4edc5c.30092a0e.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4452],{3905:(a,e,n)=>{n.d(e,{Zo:()=>d,kt:()=>o});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function r(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var m=i.createContext({}),s=function(a){var e=i.useContext(m),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},d=function(a){var e=s(a.components);return i.createElement(m.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},p=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,l=a.originalType,m=a.parentName,d=u(a,["components","mdxType","originalType","parentName"]),p=s(n),o=t,c=p["".concat(m,".").concat(o)]||p[o]||k[o]||l;return n?i.createElement(c,r(r({ref:e},d),{},{components:n})):i.createElement(c,r({ref:e},d))}));function o(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var l=n.length,r=new Array(l);r[0]=p;var u={};for(var m in e)hasOwnProperty.call(e,m)&&(u[m]=e[m]);u.originalType=a,u.mdxType="string"==typeof a?a:t,r[1]=u;for(var s=2;s{n.r(e),n.d(e,{assets:()=>m,contentTitle:()=>r,default:()=>k,frontMatter:()=>l,metadata:()=>u,toc:()=>s});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:46,title:"A.46. Closures",sidebar_label:"A.46. Closures"},r=void 0,u={unversionedId:"basic/closures",id:"basic/closures",title:"A.46. Closures",description:"Chapter ini membahas tentang closures. Closures sendiri merupakan block fungsi anonimus (anonymous function) yang memiliki kelebihan bisa mengakses item-tem yang posisinya berada di luar block closure tersebut tetapi masih dalam current block scope.",source:"@site/docs/basic/closures.md",sourceDirName:"basic",slug:"/basic/closures",permalink:"/basic/closures",draft:!1,tags:[],version:"current",sidebarPosition:46,frontMatter:{sidebar_position:46,title:"A.46. Closures",sidebar_label:"A.46. Closures"},sidebar:"tutorialSidebar",previous:{title:"A.45. Tipe Data \u279c String Custom Type",permalink:"/basic/tipe-data-custom-type-string-slice"},next:{title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",permalink:"/basic/trait-function"}},m={},s=[{value:"A.46.1. Konsep Closures",id:"a461-konsep-closures",level:2},{value:"\u25c9 formatted print {:.n}",id:"-formatted-print-n",level:3},{value:"A.46.2. Notasi penulisan closure",id:"a462-notasi-penulisan-closure",level:2},{value:"A.46.3. Mutable closure",id:"a463-mutable-closure",level:2},{value:"A.46.4. Borrowing pada closure",id:"a464-borrowing-pada-closure",level:2},{value:"A.46.5. Keyword move",id:"a465-keyword-move",level:2},{value:"A.46.6. Closure sebagai return type",id:"a466-closure-sebagai-return-type",level:2},{value:"\u25c9 Praktik ke-1",id:"-praktik-ke-1",level:3},{value:"\u25c9 Praktik ke-2",id:"-praktik-ke-2",level:3},{value:"A.46.7. Closure sebagai parameter fungsi",id:"a467-closure-sebagai-parameter-fungsi",level:2},{value:"\u25c9 Praktik ke-1",id:"-praktik-ke-1-1",level:3},{value:"\u25c9 Praktik ke-2",id:"-praktik-ke-2-1",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:s};function k(a){let{components:e,...l}=a;return(0,t.kt)("wrapper",(0,i.Z)({},d,l,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Chapter ini membahas tentang closures. Closures sendiri merupakan block fungsi anonimus (anonymous function) yang memiliki kelebihan bisa mengakses item-tem yang posisinya berada di luar block closure tersebut tetapi masih dalam current block scope."),(0,t.kt)("p",null,"Topik closures sangat erat hubungannya dengan ",(0,t.kt)("a",{parentName:"p",href:"/basic/trait-function"},"Trait \u279c Function"),", namun pada chapter ini pembahasan hanya difokuskan pada bagian penerapan closures saja. Penulis anjurkan untuk lanjut ke chapter berikutnya setelah selesai dengan chapter ini."),(0,t.kt)("h2",{id:"a461-konsep-closures"},"A.46.1. Konsep Closures"),(0,t.kt)("p",null,"Cara penerapan closure sangat mirip seperti fungsi, perbedaannya ada pada notasi penulisan-nya. Agar lebih jelas silakan perhatikan program sederhana di bawah ini."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let r = 10.0;\n let volume = calculate_circle_volume_v1(r);\n println!("{volume:.2}");\n}\n\nfn calculate_circle_volume_v1(e: f64) -> f64 {\n const PI: f64 = 3.14;\n let volume = 4.0 / 3.0 * PI * e.powi(3);\n volume\n}\n')),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"calculate_circle_volume_v1")," akan kita refactor ke bentuk closure, hasilnya adalah berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let calculate_circle_volume_v2 = |e: f64| -> f64 {\n const PI: f64 = 3.14;\n let volume = 4.0 / 3.0 * PI * e.powi(3);\n volume\n };\n \n let r = 10.0;\n let volume = calculate_circle_volume_v2(r);\n println!("{:.2}", volume);\n}\n')),(0,t.kt)("p",null,"Silakan pelajari perbedaan dan juga kemiripannya."),(0,t.kt)("p",null,"Fungsi dideklarasikan menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"fn")," dan memiliki nama. Closure tidak memiliki nama, namun bisa disimpan dalam variabel (yang disimpan adalah block closure-nya, bukan return value-nya). Contohnya variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"calculate_circle_volume_v2")," di atas."),(0,t.kt)("p",null,"Perbedaan minor lainnya ada pada notasi penulisan parameter. Pada fungsi tanda ",(0,t.kt)("inlineCode",{parentName:"p"},"()")," digunakan sebagai penanda parameter, sedangkan pada closure tanda ",(0,t.kt)("inlineCode",{parentName:"p"},"||")," digunakan."),(0,t.kt)("h3",{id:"-formatted-print-n"},"\u25c9 formatted print ",(0,t.kt)("inlineCode",{parentName:"h3"},"{:.n}")),(0,t.kt)("p",null,"Notasi penulisan formatted print ",(0,t.kt)("inlineCode",{parentName:"p"},"{:.n}")," digunakan untuk mem-format bilangan desimal di mana ",(0,t.kt)("inlineCode",{parentName:"p"},"n")," adalah jumlah digit setelah tanda ",(0,t.kt)("inlineCode",{parentName:"p"},"."),"."),(0,t.kt)("p",null,"Sebagai contoh, variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"pi")," berikut memiliki 0 digit angka dibelakang koma. Untuk menampilkan hanya 4 angka terdepan, bisa gunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"{:.4}"),". Perlu diketahui bahwa angka dibelakang koma yang muncul otomatis dibulatkan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let pi = 3.1415926535;\n\nprintln!("{:.4}", pi); // 3.1416\nprintln!("{pi:.4}"); // 3.1416\n')),(0,t.kt)("h2",{id:"a462-notasi-penulisan-closure"},"A.46.2. Notasi penulisan closure"),(0,t.kt)("p",null,"Closure ",(0,t.kt)("inlineCode",{parentName:"p"},"calculate_circle_volume_v2")," pada contoh di atas adalah salah satu contoh penulisan closure. Sebuah closure bisa memiliki parameter, bisa juga tidak, dan aturan tersebut juga berlaku pada return value."),(0,t.kt)("p",null,"Bebebrapa contoh lain penulisan closure bisa dilihat di bawah ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"// closure dengan 2 parameter tanpa return value\nlet do_something_v1 = | a: i32, b: String | {\n // ...\n};\n\n// closure dengan 2 parameter dan return value bertipe tuple\nlet do_something_v2 = | a: i32, b: String | -> (i32, bool) {\n // ...\n};\n\n// closure tanpa parameter dan return value bertipe Vec\nlet do_something_v3 = || -> Vec {\n // ...\n};\n\n// closure tanpa parameter dan tanpa return value\nlet do_something_v4 = || {\n // ...\n};\n")),(0,t.kt)("p",null,"Jika tipe return value tidak dideklarasikan secara eksplist, maka Rust menganggap tipe return value adalah sesuai dengan tipe data pada statement terakhir."),(0,t.kt)("p",null,"Untuk closure yang isinya hanya 1 baris statement, boleh tidak dituliskan block kurung kurawal-nya (",(0,t.kt)("inlineCode",{parentName:"p"},"{}"),")."),(0,t.kt)("p",null,"Tambahan contoh, satu fungsi dan tiga buah closures berikut adalah ekuivalen."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn pow_v1(x: i32) -> i32 {\n x.pow(2)\n}\n\nlet pow_v2 = |x: i32| -> i32 {\n x.pow(2)\n};\n\nlet pow_v3 = |x: i32| {\n x.pow(2)\n};\n\nlet pow_v4 = |x: i32| x.pow(2);\n")),(0,t.kt)("p",null,"Ok, sekarang bagaimana dengan closure yang tidak memiliki parameter dengan isi hanya return value saja? Penulisannya seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let get_pi = || 3.14;\n\nprintln!("{:?}", get_pi());\n')),(0,t.kt)("h2",{id:"a463-mutable-closure"},"A.46.3. Mutable closure"),(0,t.kt)("p",null,"Tidak ada yang spesial mengenai cara mengakses item yang berada di luar block closure. Caranya cukup dengan panggil saja item seperti biasanya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let num = 5;\nlet display = || println!("{num}");\n\nprintln!("{num}"); // 5\ndisplay(); // 5\n')),(0,t.kt)("p",null,"Beda lagi jika data di luar block closure adalah diubah (di-mutate) nilainya dari dalam closure, jika seperti itu maka ada beberapa hal yang perlu diperhatikan."),(0,t.kt)("p",null,"Sebagai contoh, pelajari kode berikut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\n\nlet increase_by = |x: i32| {\n num += x\n};\n\nincrease_by(10);\nprintln!("{num}"); // 15\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(6894).Z,width:"765",height:"176"})),(0,t.kt)("p",null,"Solusi untuk menghilangkan error di atas adalah dengan menambahkan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut")," pada variabel closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"// before\nlet increase_by = |x: i32| {\n num += x\n};\n\n// after\nlet mut increase_by = |x: i32| {\n num += x\n};\n")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut")," wajib ditambahkan ke variabel penampung closure ketika di dalamnya terdapat operasi perubahan data terhadap variabel yang posisinya di-luar closure. Contohnya bisa dilihat di atas, variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," nilainya di-mutate atau diubah dari dalam closure, karena inilah variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," harus didefinisikan mutable."),(0,t.kt)("h2",{id:"a464-borrowing-pada-closure"},"A.46.4. Borrowing pada closure"),(0,t.kt)("p",null,(0,t.kt)("strong",{parentName:"p"},"Semua variabel di luar block closure ketika digunakan di dalam closure maka terjadi operasi borrowing pada variabel tersebut"),". Variabel di luar block closure dipinjam agar bisa digunakan di dalam closure."),(0,t.kt)("p",null,"Sebagai contoh, kode sederhana berikut menghasilkan error, karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," adalah dipinjam oleh closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," untuk dipergunakan di dalam block-nya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\nlet mut increase_by = |x: i32| num += x;\n\nnum += 5;\nincrease_by(10);\n\nprintln!("{num}"); // 20\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(3022).Z,width:"525",height:"222"})),(0,t.kt)("p",null,"Error muncul di statement setelahnya, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"num += 5")," karena ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," statusnya masih dipinjam oleh closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by"),"."),(0,t.kt)("p",null,"Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," dipinjam dengan mode peminjaman adalah ",(0,t.kt)("em",{parentName:"p"},"mutable borrow")," karena closure didefinisikan mutable. Salah satu aturan pada borrowing: bahwa dalam waktu yang sama, ketika sudah terjadi mutable borrow, maka tidak boleh ada borrowing lainnya. Itulah alasan kenapa statement ",(0,t.kt)("inlineCode",{parentName:"p"},"num += 5")," menghasilkan error."),(0,t.kt)("p",null,"Solusi dari masalah di atas ada beberapa, yang pertama adalah menggunakan block expression untuk meng-isolasi closure, agar peminjaman pada closure tersebut dan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"num += 5")," tidak terjadi dalam waktu yang sama."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\nnum += 5;\n\n{\n let mut increase_by = |x: i32| num += x;\n increase_by(10);\n}\n\nprintln!("{num}"); // 20\n')),(0,t.kt)("p",null,"Solusi di atas efektif untuk menghilangkan error borrowing yang sebelumnya muncul, tapi setelah d-refactor kode menjadi tidak sesuai spesifikasi awal. Pada kode yang baru di atas, deklarasi closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," terjadi di dalam block expression, artinya closure ini hanya akan bisa digunakan pada block kode tersebut saja, tidak bisa di-reuse di luar block."),(0,t.kt)("p",null,"Solusi yang lebih baik pada kasus di atas adalah dengan tidak menggunakan default borrowing (yang terjadi di dalam block closure ketika mengakses variabel yang posisinya berada di luar closure), melainkan gunakan saja borrowing pada parameter closure. Kita ubah lagi kodenya menjadi seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\nlet increase_by = |num: &mut i32, x: i32| *num += x;\n\nnum += 5;\nincrease_by(&mut num, 10);\n\nprintln!("{num}"); // 20\n')),(0,t.kt)("p",null,"Pada contoh di atas, closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," ditambahi parameter baru. Sekarang ada 2 parameter, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," yang tipe-nya adalah pointer ",(0,t.kt)("inlineCode",{parentName:"p"},"&mut i32")," dan parameter ",(0,t.kt)("inlineCode",{parentName:"p"},"x"),". Di dalam block closure, ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," di-dereference (menggunakan operator ",(0,t.kt)("inlineCode",{parentName:"p"},"*"),") kemudian diubah nilainya. Setelah eksekusi statement selesai, data pinjaman tersebut langsung dikembalikan ke pemilik. Inilah kenapa kode di atas tidak menghasilkan error."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Pengaksesan variabel yang berada di luar scope closure tanpa via parameter berarti adalah borrowing, maka dalam penerapannya wajib untuk memperhatikan aturan yang berlaku pada ownership dan borrowing.")),(0,t.kt)("h2",{id:"a465-keyword-move"},"A.46.5. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"move")),(0,t.kt)("p",null,"Telah dijelaskan di atas bahwa variabel di luar closure, jika diakses dari dalam closure maka terjadi borrowing. Ada cara agar variabel tersebut ownership-nya berpindah ke dalam closure (",(0,t.kt)("em",{parentName:"p"},"move semantics"),"), yaitu menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"move"),"."),(0,t.kt)("p",null,"Contohnya bisa dilihat berikut ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\nlet mut increase_by = move |x: i32| {\n num += x;\n println!("{num} (from closure)"); // 15\n};\n\nincrease_by(10);\nprintln!("{num}"); // 5\n')),(0,t.kt)("p",null,"Closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," di atas memiliki keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"move")," dalam pendefinisiannya. Dengan ini maka semua variabel di luar scope closure jika diakses dari dalam closure, maka variabel tersebut berpindah owner-nya (",(0,t.kt)("em",{parentName:"p"},"move semantics"),")."),(0,t.kt)("p",null,"Lalu bagaimana dengan nasib variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," yang berada di luar closure setelah owner-nya berpindah? Pada kondisi normal jawaban pertanyaan ini adalah tergantung tipe data-nya, jika ",(0,t.kt)("em",{parentName:"p"},"by default")," variabel adalah mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics")," maka variabel tersebut menjadi invalid. Namun pada contoh di atas ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," bertipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"i32")," yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),", maka variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," tersebut masih bisa digunakan di luar closure."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(2682).Z,width:"324",height:"105"})),(0,t.kt)("p",null,"Bisa dilihat dari gambar di atas, statement ",(0,t.kt)("inlineCode",{parentName:"p"},"print")," dari dalam block closure hasilnya adalah benar, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"5 + 10 = 15"),". Dan variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," di luar block closure tidak berubah nilainya."),(0,t.kt)("h2",{id:"a466-closure-sebagai-return-type"},"A.46.6. Closure sebagai return type"),(0,t.kt)("h3",{id:"-praktik-ke-1"},"\u25c9 Praktik ke-1"),(0,t.kt)("p",null,"Fungsi bisa memiliki nilai balik bertipe closure. Caranya dengan menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"impl Fn()")," sebagai tipe data nilai balik. Contoh penerapannya:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn do_something() -> impl Fn() {\n println!("hello (from do_something)");\n\n return || {\n println!("hello (from closure)");\n };\n}\n')),(0,t.kt)("p",null,"Tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"impl Fn()")," adalah ekuvalen dengan closure ",(0,t.kt)("inlineCode",{parentName:"p"},"|| { }"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let my_closure = do_something();\n println!("hello (from main)");\n my_closure();\n}\n\nfn main() {\n let my_closure = do_something_v2();\n let message = my_closure(123, "hello rust".to_owned());\n println!("{message} (from main)");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(4600).Z,width:"321",height:"124"})),(0,t.kt)("p",null,"Bisa dilihat, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," di atas mengembalikan closure yang menampilkan string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello (from closure)"),". String tersebut hanya muncul ketika closure dipanggil."),(0,t.kt)("p",null,"Sedangkan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," sendiri juga menampilkan pesan string lainnya, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"hello (from do_something)"),", yang pesan ini adalah muncul langsung saat pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),". Berbeda dengan pesan ",(0,t.kt)("inlineCode",{parentName:"p"},"hello (from closure)")," yang hanya muncul ketika closure dieksekusi."),(0,t.kt)("h3",{id:"-praktik-ke-2"},"\u25c9 Praktik ke-2"),(0,t.kt)("p",null,"Pada contoh di atas, closure yang dikembalikan fungsi memiliki skema sangat sederhana, tanpa parameter dan argument. Mari coba praktik dengan contoh yang lebih kompleks."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn do_something_v2() -> impl Fn(i32, String) -> String {\n println!("hello (from do_something_v2)");\n\n return |a: i32, b: String| -> String {\n let message = format!("{b} {a}");\n message\n };\n}\n')),(0,t.kt)("p",null,"Pada kode di atas fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something_v2")," mengembalikan closure dengan skema ",(0,t.kt)("inlineCode",{parentName:"p"},"Fn(i32, String) -> String"),", yang artinya:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Parameter pertamanya bertipe ",(0,t.kt)("inlineCode",{parentName:"li"},"i32")),(0,t.kt)("li",{parentName:"ul"},"Parameter keduanya bertipe ",(0,t.kt)("inlineCode",{parentName:"li"},"String")),(0,t.kt)("li",{parentName:"ul"},"Mengembalikan nilai bertipe ",(0,t.kt)("inlineCode",{parentName:"li"},"String"))),(0,t.kt)("p",null,"Di dalam closure tersebut, data parameter digabung menjadi sebuah pesan string yang kemudian dijadikan nilai balik."),(0,t.kt)("p",null,"Sekarang jalankan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something_v2")," di atas, kemudian lihat hasilnya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let my_closure = do_something_v2();\nlet message = my_closure(123, "hello rust".to_owned());\nprintln!("{message} (from main)");\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(3546).Z,width:"323",height:"106"})),(0,t.kt)("h2",{id:"a467-closure-sebagai-parameter-fungsi"},"A.46.7. Closure sebagai parameter fungsi"),(0,t.kt)("h3",{id:"-praktik-ke-1-1"},"\u25c9 Praktik ke-1"),(0,t.kt)("p",null,"Pengaplikasian closure sebagai parameter fungsi banyak dilakukan di Rust, hal ini bisa dilakukan dengan memanfaatkan generic parameter. Silakan perhatikan kode berikut untuk contoh penerapannya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn run_x_times(x: i32, my_closure: F)\nwhere\n F: Fn(i32),\n{\n for i in 0..x {\n my_closure(i)\n }\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"run_x_times")," dibuat dengan tugas mengeksekusi closure ",(0,t.kt)("inlineCode",{parentName:"p"},"my_closure")," sebanyak ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," kali. Parameter closure harus selalu memiliki tipe data generic, contohnya ",(0,t.kt)("inlineCode",{parentName:"p"},"my_closure")," di atas yang tipe data-nya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"F"),"."),(0,t.kt)("p",null,"Fungsi yang memiliki parameter closure wajib menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"where")," yang keyword ini digunakan untuk mendaftarkan skema closure yang nantiya bisa diterima saat pemanggilan fungsi. Tipe generic ",(0,t.kt)("inlineCode",{parentName:"p"},"F")," di daftarkan dengan notasi ",(0,t.kt)("inlineCode",{parentName:"p"},"Fn(i32)"),", artinya dalam pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"run_x_times"),", parameter ",(0,t.kt)("inlineCode",{parentName:"p"},"my_closure")," harus diisi dengan closure yang skema-nya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"Fn(i32)")," yang jika di ilustrasikan dalam bentuk closure adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"|param1: i32| { }"),"."),(0,t.kt)("p",null,"Sekarang mari kita test fungsi di atas. Panggil fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"run_x_times")," kemudian pada bagian parameter closure isi dengan block closure yang memiliki skema sesuai requirement, contohnya ",(0,t.kt)("inlineCode",{parentName:"p"},'|i: i32| { println!("hello rust {i}") }'),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'run_x_times(4, |i: i32| {\n println!("hello rust {i}");\n});\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(2376).Z,width:"314",height:"147"})),(0,t.kt)("p",null,"O iya, closure di atas juga bisa dituliskan dalam bentuk seperti ini. Bebas, pilih sesuai preferensi dan/atau kesepakatan team perihal code convention."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'run_x_times(4, |i: i32| println!("hello rust {i}"));\n')),(0,t.kt)("h3",{id:"-praktik-ke-2-1"},"\u25c9 Praktik ke-2"),(0,t.kt)("p",null,"Ok, sekarang mari kita coba praktikkan contoh yang lebih kompleks."),(0,t.kt)("p",null,"Pada praktik ini kita akan buat sebuah fungsi yang tugasnya melakukan pencarian indeks slice dengan kondisi pencarian didapat dari eksekusi closure."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn find_index(data: &[T], cond_fn: F) -> i32\nwhere\n F: Fn(&T) -> bool,\n{\n for i in 0..data.len() {\n if cond_fn(&data[i]) {\n return i as i32\n }\n }\n\n return -1\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"find_index")," memiliki 2 parameter generic, yaitu:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("inlineCode",{parentName:"li"},"T")," yang digunakan sebagai tipe data element slice ",(0,t.kt)("inlineCode",{parentName:"li"},"data"),"."),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("inlineCode",{parentName:"li"},"F")," yang digunakan sebagai tipe data closure ",(0,t.kt)("inlineCode",{parentName:"li"},"cond_fn")," dengan skema ",(0,t.kt)("inlineCode",{parentName:"li"},"Fn(&T) -> bool"),", yang jika diilustrasikan dalam bentuk closure adalah ",(0,t.kt)("inlineCode",{parentName:"li"},"|param1: &T| -> bool { }"),".")),(0,t.kt)("p",null,"Di dalam fungsi tersebut, data slice di-loop, kemudian tiap elemen-nya digunakan sebagai parameter pemanggilan closure ",(0,t.kt)("inlineCode",{parentName:"p"},"cond_fn"),"."),(0,t.kt)("p",null,"Jika nilai balik pemanggilan closure adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"true")," maka ",(0,t.kt)("inlineCode",{parentName:"p"},"i")," dikembalikan dalam bentuk ",(0,t.kt)("inlineCode",{parentName:"p"},"i32")," (ada proses casting). Dan jika tidak diketemukan, maka indeks ",(0,t.kt)("inlineCode",{parentName:"p"},"-1")," dikembalikan."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"as")," digunakan untuk casting tipe data. Lebih jelasnya dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/type-alias-casting"},"Type Alias & Casting")," dan ",(0,t.kt)("a",{parentName:"p",href:"#/wip/trait-conversion-from-into"},"Trait \u279c Conversion (From & Into)"))),(0,t.kt)("p",null,"Jalankan program tersebut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let numbers = [24, 13, 2, 53, 3];\nlet number_to_find = 53;\nlet index = find_index(&numbers, |e: &i32| -> bool {\n if *e == number_to_find {\n true \n } else {\n false\n }\n});\n\nprintln!("number_to_find: {number_to_find}");\nprintln!("index: {index}");\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(3451).Z,width:"324",height:"108"})),(0,t.kt)("p",null,"Bisa dilihat program berjalan sesuai harapan."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"Fn")," merupakan salah satu trait function yang ada di Rust. Lebih jelasnya perihal keyword tersebut dibahas pada chapter selanjutnya, yaitu ",(0,t.kt)("a",{parentName:"p",href:"/basic/trait-function"},"Trait \u279c Function"))),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/closures"},"github.com/novalagung/dasarpemrogramanrust-example/../closures")),(0,t.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/function"},"Function")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/borrowing"},"Borrowing")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/trait-function"},"Trait \u279c Function (Fn, FnMut, FnOnce)"))),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch13-01-closures.html"},"https://doc.rust-lang.org/book/ch13-01-closures.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/beta/rust-by-example/fn/closures.html"},"https://doc.rust-lang.org/beta/rust-by-example/fn/closures.html"))))}k.isMDXComponent=!0},6894:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/closures-1-2c7599bfed946d5bd1fb34a57acd9de7.png"},3022:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/closures-2-35b45e21715884c98b155c79b2d8f1c4.png"},2682:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},4600:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},3546:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},2376:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},3451:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/bc4edc5c.f0b380ec.js b/assets/js/bc4edc5c.f0b380ec.js deleted file mode 100644 index 468f7a9e..00000000 --- a/assets/js/bc4edc5c.f0b380ec.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4452],{3905:(a,e,n)=>{n.d(e,{Zo:()=>d,kt:()=>o});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function r(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var m=i.createContext({}),s=function(a){var e=i.useContext(m),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},d=function(a){var e=s(a.components);return i.createElement(m.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},p=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,l=a.originalType,m=a.parentName,d=u(a,["components","mdxType","originalType","parentName"]),p=s(n),o=t,c=p["".concat(m,".").concat(o)]||p[o]||k[o]||l;return n?i.createElement(c,r(r({ref:e},d),{},{components:n})):i.createElement(c,r({ref:e},d))}));function o(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var l=n.length,r=new Array(l);r[0]=p;var u={};for(var m in e)hasOwnProperty.call(e,m)&&(u[m]=e[m]);u.originalType=a,u.mdxType="string"==typeof a?a:t,r[1]=u;for(var s=2;s{n.r(e),n.d(e,{assets:()=>m,contentTitle:()=>r,default:()=>k,frontMatter:()=>l,metadata:()=>u,toc:()=>s});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:46,title:"A.46. Closures",sidebar_label:"A.46. Closures"},r=void 0,u={unversionedId:"basic/closures",id:"basic/closures",title:"A.46. Closures",description:"Chapter ini membahas tentang closures. Closures sendiri merupakan block fungsi anonimus (anonymous function) yang memiliki kelebihan bisa mengakses item-tem yang posisinya berada di luar block closure tersebut tetapi masih dalam current block scope.",source:"@site/docs/basic/closures.md",sourceDirName:"basic",slug:"/basic/closures",permalink:"/basic/closures",draft:!1,tags:[],version:"current",sidebarPosition:46,frontMatter:{sidebar_position:46,title:"A.46. Closures",sidebar_label:"A.46. Closures"},sidebar:"tutorialSidebar",previous:{title:"A.45. Tipe Data \u279c String Custom Type",permalink:"/basic/tipe-data-custom-type-string-slice"},next:{title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",permalink:"/basic/trait-function"}},m={},s=[{value:"A.46.1. Konsep Closures",id:"a461-konsep-closures",level:2},{value:"\u25c9 formatted print {:.n}",id:"-formatted-print-n",level:3},{value:"A.46.2. Notasi penulisan closure",id:"a462-notasi-penulisan-closure",level:2},{value:"A.46.3. Mutable closure",id:"a463-mutable-closure",level:2},{value:"A.46.4. Borrowing pada closure",id:"a464-borrowing-pada-closure",level:2},{value:"A.46.5. Keyword move",id:"a465-keyword-move",level:2},{value:"A.46.6. Closure sebagai return type",id:"a466-closure-sebagai-return-type",level:2},{value:"\u25c9 Praktik ke-1",id:"-praktik-ke-1",level:3},{value:"\u25c9 Praktik ke-2",id:"-praktik-ke-2",level:3},{value:"A.46.7. Closure sebagai parameter fungsi",id:"a467-closure-sebagai-parameter-fungsi",level:2},{value:"\u25c9 Praktik ke-1",id:"-praktik-ke-1-1",level:3},{value:"\u25c9 Praktik ke-2",id:"-praktik-ke-2-1",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:s};function k(a){let{components:e,...l}=a;return(0,t.kt)("wrapper",(0,i.Z)({},d,l,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Chapter ini membahas tentang closures. Closures sendiri merupakan block fungsi anonimus (anonymous function) yang memiliki kelebihan bisa mengakses item-tem yang posisinya berada di luar block closure tersebut tetapi masih dalam current block scope."),(0,t.kt)("p",null,"Topik closures sangat erat hubungannya dengan ",(0,t.kt)("a",{parentName:"p",href:"/basic/trait-function"},"Trait \u279c Function"),", namun pada chapter ini pembahasan hanya difokuskan pada bagian penerapan closures saja. Penulis anjurkan untuk lanjut ke chapter berikutnya setelah selesai dengan chapter ini."),(0,t.kt)("h2",{id:"a461-konsep-closures"},"A.46.1. Konsep Closures"),(0,t.kt)("p",null,"Cara penerapan closure sangat mirip seperti fungsi, perbedaannya ada pada notasi penulisan-nya. Agar lebih jelas silakan perhatikan program sederhana di bawah ini."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let r = 10.0;\n let volume = calculate_circle_volume_v1(r);\n println!("{volume:.2}");\n}\n\nfn calculate_circle_volume_v1(e: f64) -> f64 {\n const PI: f64 = 3.14;\n let volume = 4.0 / 3.0 * PI * e.powi(3);\n volume\n}\n')),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"calculate_circle_volume_v1")," akan kita refactor ke bentuk closure, hasilnya adalah berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let calculate_circle_volume_v2 = |e: f64| -> f64 {\n const PI: f64 = 3.14;\n let volume = 4.0 / 3.0 * PI * e.powi(3);\n volume\n };\n \n let r = 10.0;\n let volume = calculate_circle_volume_v2(r);\n println!("{:.2}", volume);\n}\n')),(0,t.kt)("p",null,"Silakan pelajari perbedaan dan juga kemiripannya."),(0,t.kt)("p",null,"Fungsi dideklarasikan menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"fn")," dan memiliki nama. Closure tidak memiliki nama, namun bisa disimpan dalam variabel (yang disimpan adalah block closure-nya, bukan return value-nya). Contohnya variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"calculate_circle_volume_v2")," di atas."),(0,t.kt)("p",null,"Perbedaan minor lainnya ada pada notasi penulisan parameter. Pada fungsi tanda ",(0,t.kt)("inlineCode",{parentName:"p"},"()")," digunakan sebagai penanda parameter, sedangkan pada closure tanda ",(0,t.kt)("inlineCode",{parentName:"p"},"||")," digunakan."),(0,t.kt)("h3",{id:"-formatted-print-n"},"\u25c9 formatted print ",(0,t.kt)("inlineCode",{parentName:"h3"},"{:.n}")),(0,t.kt)("p",null,"Notasi penulisan formatted print ",(0,t.kt)("inlineCode",{parentName:"p"},"{:.n}")," digunakan untuk mem-format bilangan desimal dimana ",(0,t.kt)("inlineCode",{parentName:"p"},"n")," adalah jumlah digit setelah tanda ",(0,t.kt)("inlineCode",{parentName:"p"},"."),"."),(0,t.kt)("p",null,"Sebagai contoh, variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"pi")," berikut memiliki 0 digit angka dibelakang koma. Untuk menampilkan hanya 4 angka terdepan, bisa gunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"{:.4}"),". Perlu diketahui bahwa angka dibelakang koma yang muncul otomatis dibulatkan."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let pi = 3.1415926535;\n\nprintln!("{:.4}", pi); // 3.1416\nprintln!("{pi:.4}"); // 3.1416\n')),(0,t.kt)("h2",{id:"a462-notasi-penulisan-closure"},"A.46.2. Notasi penulisan closure"),(0,t.kt)("p",null,"Closure ",(0,t.kt)("inlineCode",{parentName:"p"},"calculate_circle_volume_v2")," pada contoh di atas adalah salah satu contoh penulisan closure. Sebuah closure bisa memiliki parameter, bisa juga tidak, dan aturan tersebut juga berlaku pada return value."),(0,t.kt)("p",null,"Bebebrapa contoh lain penulisan closure bisa dilihat di bawah ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"// closure dengan 2 parameter tanpa return value\nlet do_something_v1 = | a: i32, b: String | {\n // ...\n};\n\n// closure dengan 2 parameter dan return value bertipe tuple\nlet do_something_v2 = | a: i32, b: String | -> (i32, bool) {\n // ...\n};\n\n// closure tanpa parameter dan return value bertipe Vec\nlet do_something_v3 = || -> Vec {\n // ...\n};\n\n// closure tanpa parameter dan tanpa return value\nlet do_something_v4 = || {\n // ...\n};\n")),(0,t.kt)("p",null,"Jika tipe return value tidak dideklarasikan secara eksplist, maka Rust menganggap tipe return value adalah sesuai dengan tipe data pada statement terakhir."),(0,t.kt)("p",null,"Untuk closure yang isinya hanya 1 baris statement, boleh tidak dituliskan block kurung kurawal-nya (",(0,t.kt)("inlineCode",{parentName:"p"},"{}"),")."),(0,t.kt)("p",null,"Tambahan contoh, satu fungsi dan tiga buah closures berikut adalah ekuivalen."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn pow_v1(x: i32) -> i32 {\n x.pow(2)\n}\n\nlet pow_v2 = |x: i32| -> i32 {\n x.pow(2)\n};\n\nlet pow_v3 = |x: i32| {\n x.pow(2)\n};\n\nlet pow_v4 = |x: i32| x.pow(2);\n")),(0,t.kt)("p",null,"Ok, sekarang bagaimana dengan closure yang tidak memiliki parameter dengan isi hanya return value saja? Penulisannya seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let get_pi = || 3.14;\n\nprintln!("{:?}", get_pi());\n')),(0,t.kt)("h2",{id:"a463-mutable-closure"},"A.46.3. Mutable closure"),(0,t.kt)("p",null,"Tidak ada yang spesial mengenai cara mengakses item yang berada di luar block closure. Caranya cukup dengan panggil saja item seperti biasanya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let num = 5;\nlet display = || println!("{num}");\n\nprintln!("{num}"); // 5\ndisplay(); // 5\n')),(0,t.kt)("p",null,"Beda lagi jika data di luar block closure adalah diubah (di-mutate) nilainya dari dalam closure, jika seperti itu maka ada beberapa hal yang perlu diperhatikan."),(0,t.kt)("p",null,"Sebagai contoh, pelajari kode berikut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\n\nlet increase_by = |x: i32| {\n num += x\n};\n\nincrease_by(10);\nprintln!("{num}"); // 15\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(6894).Z,width:"765",height:"176"})),(0,t.kt)("p",null,"Solusi untuk menghilangkan error di atas adalah dengan menambahkan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut")," pada variabel closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"// before\nlet increase_by = |x: i32| {\n num += x\n};\n\n// after\nlet mut increase_by = |x: i32| {\n num += x\n};\n")),(0,t.kt)("p",null,"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"mut")," wajib ditambahkan ke variabel penampung closure ketika di dalamnya terdapat operasi perubahan data terhadap variabel yang posisinya di-luar closure. Contohnya bisa dilihat di atas, variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," nilainya di-mutate atau diubah dari dalam closure, karena inilah variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," harus didefinisikan mutable."),(0,t.kt)("h2",{id:"a464-borrowing-pada-closure"},"A.46.4. Borrowing pada closure"),(0,t.kt)("p",null,(0,t.kt)("strong",{parentName:"p"},"Semua variabel di luar block closure ketika digunakan di dalam closure maka terjadi operasi borrowing pada variabel tersebut"),". Variabel di luar block closure dipinjam agar bisa digunakan di dalam closure."),(0,t.kt)("p",null,"Sebagai contoh, kode sederhana berikut menghasilkan error, karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," adalah dipinjam oleh closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," untuk dipergunakan di dalam block-nya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\nlet mut increase_by = |x: i32| num += x;\n\nnum += 5;\nincrease_by(10);\n\nprintln!("{num}"); // 20\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(3022).Z,width:"525",height:"222"})),(0,t.kt)("p",null,"Error muncul di statement setelahnya, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"num += 5")," karena ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," statusnya masih dipinjam oleh closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by"),"."),(0,t.kt)("p",null,"Variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," dipinjam dengan mode peminjaman adalah ",(0,t.kt)("em",{parentName:"p"},"mutable borrow")," karena closure didefinisikan mutable. Salah satu aturan pada borrowing: bahwa dalam waktu yang sama, ketika sudah terjadi mutable borrow, maka tidak boleh ada borrowing lainnya. Itulah alasan kenapa statement ",(0,t.kt)("inlineCode",{parentName:"p"},"num += 5")," menghasilkan error."),(0,t.kt)("p",null,"Solusi dari masalah di atas ada beberapa, yang pertama adalah menggunakan block expression untuk meng-isolasi closure, agar peminjaman pada closure tersebut dan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"num += 5")," tidak terjadi dalam waktu yang sama."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\nnum += 5;\n\n{\n let mut increase_by = |x: i32| num += x;\n increase_by(10);\n}\n\nprintln!("{num}"); // 20\n')),(0,t.kt)("p",null,"Solusi di atas efektif untuk menghilangkan error borrowing yang sebelumnya muncul, tapi setelah d-refactor kode menjadi tidak sesuai spesifikasi awal. Pada kode yang baru di atas, deklarasi closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," terjadi di dalam block expression, artinya closure ini hanya akan bisa digunakan pada block kode tersebut saja, tidak bisa di-reuse di luar block."),(0,t.kt)("p",null,"Solusi yang lebih baik pada kasus di atas adalah dengan tidak menggunakan default borrowing (yang terjadi di dalam block closure ketika mengakses variabel yang posisinya berada di luar closure), melainkan gunakan saja borrowing pada parameter closure. Kita ubah lagi kodenya menjadi seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\nlet increase_by = |num: &mut i32, x: i32| *num += x;\n\nnum += 5;\nincrease_by(&mut num, 10);\n\nprintln!("{num}"); // 20\n')),(0,t.kt)("p",null,"Pada contoh di atas, closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," ditambahi parameter baru. Sekarang ada 2 parameter, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," yang tipe-nya adalah pointer ",(0,t.kt)("inlineCode",{parentName:"p"},"&mut i32")," dan parameter ",(0,t.kt)("inlineCode",{parentName:"p"},"x"),". Di dalam block closure, ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," di-dereference (menggunakan operator ",(0,t.kt)("inlineCode",{parentName:"p"},"*"),") kemudian diubah nilainya. Setelah eksekusi statement selesai, data pinjaman tersebut langsung dikembalikan ke pemilik. Inilah kenapa kode di atas tidak menghasilkan error."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Pengaksesan variabel yang berada di luar scope closure tanpa via parameter berarti adalah borrowing, maka dalam penerapannya wajib untuk memperhatikan aturan yang berlaku pada ownership dan borrowing.")),(0,t.kt)("h2",{id:"a465-keyword-move"},"A.46.5. Keyword ",(0,t.kt)("inlineCode",{parentName:"h2"},"move")),(0,t.kt)("p",null,"Telah dijelaskan di atas bahwa variabel di luar closure, jika diakses dari dalam closure maka terjadi borrowing. Ada cara agar variabel tersebut ownership-nya berpindah ke dalam closure (",(0,t.kt)("em",{parentName:"p"},"move semantics"),"), yaitu menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"move"),"."),(0,t.kt)("p",null,"Contohnya bisa dilihat berikut ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let mut num = 5;\nlet mut increase_by = move |x: i32| {\n num += x;\n println!("{num} (from closure)"); // 15\n};\n\nincrease_by(10);\nprintln!("{num}"); // 5\n')),(0,t.kt)("p",null,"Closure ",(0,t.kt)("inlineCode",{parentName:"p"},"increase_by")," di atas memiliki keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"move")," dalam pendefinisiannya. Dengan ini maka semua variabel di luar scope closure jika diakses dari dalam closure, maka variabel tersebut berpindah owner-nya (",(0,t.kt)("em",{parentName:"p"},"move semantics"),")."),(0,t.kt)("p",null,"Lalu bagaimana dengan nasib variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," yang berada di luar closure setelah owner-nya berpindah? Pada kondisi normal jawaban pertanyaan ini adalah tergantung tipe data-nya, jika ",(0,t.kt)("em",{parentName:"p"},"by default")," variabel adalah mengadopsi ",(0,t.kt)("em",{parentName:"p"},"move semantics")," maka variabel tersebut menjadi invalid. Namun pada contoh di atas ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," bertipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"i32")," yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),", maka variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," tersebut masih bisa digunakan di luar closure."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(2682).Z,width:"324",height:"105"})),(0,t.kt)("p",null,"Bisa dilihat dari gambar di atas, statement ",(0,t.kt)("inlineCode",{parentName:"p"},"print")," dari dalam block closure hasilnya adalah benar, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"5 + 10 = 15"),". Dan variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"num")," di luar block closure tidak berubah nilainya."),(0,t.kt)("h2",{id:"a466-closure-sebagai-return-type"},"A.46.6. Closure sebagai return type"),(0,t.kt)("h3",{id:"-praktik-ke-1"},"\u25c9 Praktik ke-1"),(0,t.kt)("p",null,"Fungsi bisa memiliki nilai balik bertipe closure. Caranya dengan menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"impl Fn()")," sebagai tipe data nilai balik. Contoh penerapannya:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn do_something() -> impl Fn() {\n println!("hello (from do_something)");\n\n return || {\n println!("hello (from closure)");\n };\n}\n')),(0,t.kt)("p",null,"Tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"impl Fn()")," adalah ekuvalen dengan closure ",(0,t.kt)("inlineCode",{parentName:"p"},"|| { }"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let my_closure = do_something();\n println!("hello (from main)");\n my_closure();\n}\n\nfn main() {\n let my_closure = do_something_v2();\n let message = my_closure(123, "hello rust".to_owned());\n println!("{message} (from main)");\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(4600).Z,width:"321",height:"124"})),(0,t.kt)("p",null,"Bisa dilihat, fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," di atas mengembalikan closure yang menampilkan string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello (from closure)"),". String tersebut hanya muncul ketika closure dipanggil."),(0,t.kt)("p",null,"Sedangkan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something")," sendiri juga menampilkan pesan string lainnya, yaitu ",(0,t.kt)("inlineCode",{parentName:"p"},"hello (from do_something)"),", yang pesan ini adalah muncul langsung saat pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something"),". Berbeda dengan pesan ",(0,t.kt)("inlineCode",{parentName:"p"},"hello (from closure)")," yang hanya muncul ketika closure dieksekusi."),(0,t.kt)("h3",{id:"-praktik-ke-2"},"\u25c9 Praktik ke-2"),(0,t.kt)("p",null,"Pada contoh di atas, closure yang dikembalikan fungsi memiliki skema sangat sederhana, tanpa parameter dan argument. Mari coba praktik dengan contoh yang lebih kompleks."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn do_something_v2() -> impl Fn(i32, String) -> String {\n println!("hello (from do_something_v2)");\n\n return |a: i32, b: String| -> String {\n let message = format!("{b} {a}");\n message\n };\n}\n')),(0,t.kt)("p",null,"Pada kode di atas fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something_v2")," mengembalikan closure dengan skema ",(0,t.kt)("inlineCode",{parentName:"p"},"Fn(i32, String) -> String"),", yang artinya:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Parameter pertamanya bertipe ",(0,t.kt)("inlineCode",{parentName:"li"},"i32")),(0,t.kt)("li",{parentName:"ul"},"Parameter keduanya bertipe ",(0,t.kt)("inlineCode",{parentName:"li"},"String")),(0,t.kt)("li",{parentName:"ul"},"Mengembalikan nilai bertipe ",(0,t.kt)("inlineCode",{parentName:"li"},"String"))),(0,t.kt)("p",null,"Di dalam closure tersebut, data parameter digabung menjadi sebuah pesan string yang kemudian dijadikan nilai balik."),(0,t.kt)("p",null,"Sekarang jalankan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something_v2")," di atas, kemudian lihat hasilnya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let my_closure = do_something_v2();\nlet message = my_closure(123, "hello rust".to_owned());\nprintln!("{message} (from main)");\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(3546).Z,width:"323",height:"106"})),(0,t.kt)("h2",{id:"a467-closure-sebagai-parameter-fungsi"},"A.46.7. Closure sebagai parameter fungsi"),(0,t.kt)("h3",{id:"-praktik-ke-1-1"},"\u25c9 Praktik ke-1"),(0,t.kt)("p",null,"Pengaplikasian closure sebagai parameter fungsi banyak dilakukan di Rust, hal ini bisa dilakukan dengan memanfaatkan generic parameter. Silakan perhatikan kode berikut untuk contoh penerapannya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn run_x_times(x: i32, my_closure: F)\nwhere\n F: Fn(i32),\n{\n for i in 0..x {\n my_closure(i)\n }\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"run_x_times")," dibuat dengan tugas mengeksekusi closure ",(0,t.kt)("inlineCode",{parentName:"p"},"my_closure")," sebanyak ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," kali. Parameter closure harus selalu memiliki tipe data generic, contohnya ",(0,t.kt)("inlineCode",{parentName:"p"},"my_closure")," di atas yang tipe data-nya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"F"),"."),(0,t.kt)("p",null,"Fungsi yang memiliki parameter closure wajib menggunakan keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"where")," yang keyword ini digunakan untuk mendaftarkan skema closure yang nantiya bisa diterima saat pemanggilan fungsi. Tipe generic ",(0,t.kt)("inlineCode",{parentName:"p"},"F")," di daftarkan dengan notasi ",(0,t.kt)("inlineCode",{parentName:"p"},"Fn(i32)"),", artinya dalam pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"run_x_times"),", parameter ",(0,t.kt)("inlineCode",{parentName:"p"},"my_closure")," harus diisi dengan closure yang skema-nya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"Fn(i32)")," yang jika di ilustrasikan dalam bentuk closure adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"|param1: i32| { }"),"."),(0,t.kt)("p",null,"Sekarang mari kita test fungsi di atas. Panggil fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"run_x_times")," kemudian pada bagian parameter closure isi dengan block closure yang memiliki skema sesuai requirement, contohnya ",(0,t.kt)("inlineCode",{parentName:"p"},'|i: i32| { println!("hello rust {i}") }'),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'run_x_times(4, |i: i32| {\n println!("hello rust {i}");\n});\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(2376).Z,width:"314",height:"147"})),(0,t.kt)("p",null,"O iya, closure di atas juga bisa dituliskan dalam bentuk seperti ini. Bebas, pilih sesuai preferensi dan/atau kesepakatan team perihal code convention."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'run_x_times(4, |i: i32| println!("hello rust {i}"));\n')),(0,t.kt)("h3",{id:"-praktik-ke-2-1"},"\u25c9 Praktik ke-2"),(0,t.kt)("p",null,"Ok, sekarang mari kita coba praktikkan contoh yang lebih kompleks."),(0,t.kt)("p",null,"Pada praktik ini kita akan buat sebuah fungsi yang tugasnya melakukan pencarian indeks slice dengan kondisi pencarian didapat dari eksekusi closure."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn find_index(data: &[T], cond_fn: F) -> i32\nwhere\n F: Fn(&T) -> bool,\n{\n for i in 0..data.len() {\n if cond_fn(&data[i]) {\n return i as i32\n }\n }\n\n return -1\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"find_index")," memiliki 2 parameter generic, yaitu:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("inlineCode",{parentName:"li"},"T")," yang digunakan sebagai tipe data element slice ",(0,t.kt)("inlineCode",{parentName:"li"},"data"),"."),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("inlineCode",{parentName:"li"},"F")," yang digunakan sebagai tipe data closure ",(0,t.kt)("inlineCode",{parentName:"li"},"cond_fn")," dengan skema ",(0,t.kt)("inlineCode",{parentName:"li"},"Fn(&T) -> bool"),", yang jika diilustrasikan dalam bentuk closure adalah ",(0,t.kt)("inlineCode",{parentName:"li"},"|param1: &T| -> bool { }"),".")),(0,t.kt)("p",null,"Di dalam fungsi tersebut, data slice di-loop, kemudian tiap elemen-nya digunakan sebagai parameter pemanggilan closure ",(0,t.kt)("inlineCode",{parentName:"p"},"cond_fn"),"."),(0,t.kt)("p",null,"Jika nilai balik pemanggilan closure adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"true")," maka ",(0,t.kt)("inlineCode",{parentName:"p"},"i")," dikembalikan dalam bentuk ",(0,t.kt)("inlineCode",{parentName:"p"},"i32")," (ada proses casting). Dan jika tidak diketemukan, maka indeks ",(0,t.kt)("inlineCode",{parentName:"p"},"-1")," dikembalikan."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"as")," digunakan untuk casting tipe data. Lebih jelasnya dibahas pada chapter ",(0,t.kt)("a",{parentName:"p",href:"/basic/type-alias-casting"},"Type Alias & Casting")," dan ",(0,t.kt)("a",{parentName:"p",href:"#/wip/trait-conversion-from-into"},"Trait \u279c Conversion (From & Into)"))),(0,t.kt)("p",null,"Jalankan program tersebut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'let numbers = [24, 13, 2, 53, 3];\nlet number_to_find = 53;\nlet index = find_index(&numbers, |e: &i32| -> bool {\n if *e == number_to_find {\n true \n } else {\n false\n }\n});\n\nprintln!("number_to_find: {number_to_find}");\nprintln!("index: {index}");\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Closure",src:n(3451).Z,width:"324",height:"108"})),(0,t.kt)("p",null,"Bisa dilihat program berjalan sesuai harapan."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Keyword ",(0,t.kt)("inlineCode",{parentName:"p"},"Fn")," merupakan salah satu trait function yang ada di Rust. Lebih jelasnya perihal keyword tersebut dibahas pada chapter selanjutnya, yaitu ",(0,t.kt)("a",{parentName:"p",href:"/basic/trait-function"},"Trait \u279c Function"))),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/closures"},"github.com/novalagung/dasarpemrogramanrust-example/../closures")),(0,t.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/function"},"Function")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/borrowing"},"Borrowing")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"/basic/trait-function"},"Trait \u279c Function (Fn, FnMut, FnOnce)"))),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch13-01-closures.html"},"https://doc.rust-lang.org/book/ch13-01-closures.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/beta/rust-by-example/fn/closures.html"},"https://doc.rust-lang.org/beta/rust-by-example/fn/closures.html"))))}k.isMDXComponent=!0},6894:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/closures-1-2c7599bfed946d5bd1fb34a57acd9de7.png"},3022:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/closures-2-35b45e21715884c98b155c79b2d8f1c4.png"},2682:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},4600:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},3546:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},2376:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},3451:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/bd9b9d52.8d109af6.js b/assets/js/bd9b9d52.8d109af6.js deleted file mode 100644 index 47976681..00000000 --- a/assets/js/bd9b9d52.8d109af6.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[909],{3905:(a,e,t)=>{t.d(e,{Zo:()=>k,kt:()=>o});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var u=n.createContext({}),d=function(a){var e=n.useContext(u),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},k=function(a){var e=d(a.components);return n.createElement(u.Provider,{value:e},a.children)},m={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},s=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,u=a.parentName,k=p(a,["components","mdxType","originalType","parentName"]),s=d(t),o=i,g=s["".concat(u,".").concat(o)]||s[o]||m[o]||r;return t?n.createElement(g,l(l({ref:e},k),{},{components:t})):n.createElement(g,l({ref:e},k))}));function o(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=s;var p={};for(var u in e)hasOwnProperty.call(e,u)&&(p[u]=e[u]);p.originalType=a,p.mdxType="string"==typeof a?a:i,l[1]=p;for(var d=2;d{t.r(e),t.d(e,{assets:()=>u,contentTitle:()=>l,default:()=>m,frontMatter:()=>r,metadata:()=>p,toc:()=>d});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:5,title:"A.5. Tipe Data \u279c Primitive Scalar",sidebar_label:"A.5. Tipe Data \u279c Primitive Scalar"},l=void 0,p={unversionedId:"basic/tipe-data-primitive-scalar",id:"basic/tipe-data-primitive-scalar",title:"A.5. Tipe Data \u279c Primitive Scalar",description:"Tipe data di Rust dikategorikan menjadi beberapa group, salah satunya adalah group tipe data primitif scalar atau biasa disebut scalar types. Pada chapter ini kita akan mempelajarinya.",source:"@site/docs/basic/tipe-data-primitive-scalar.md",sourceDirName:"basic",slug:"/basic/tipe-data-primitive-scalar",permalink:"/basic/tipe-data-primitive-scalar",draft:!1,tags:[],version:"current",sidebarPosition:5,frontMatter:{sidebar_position:5,title:"A.5. Tipe Data \u279c Primitive Scalar",sidebar_label:"A.5. Tipe Data \u279c Primitive Scalar"},sidebar:"tutorialSidebar",previous:{title:"A.4. Variabel",permalink:"/basic/variabel"},next:{title:"A.6. Tipe Data \u279c String Literal (&str)",permalink:"/basic/tipe-data-string-literal"}},u={},d=[{value:"A.5.1. Signed integers",id:"a51-signed-integers",level:2},{value:"A.5.2. Unsigned integers",id:"a52-unsigned-integers",level:2},{value:"A.5.3. Floating point",id:"a53-floating-point",level:2},{value:"A.5.4. Bool",id:"a54-bool",level:2},{value:"A.5.5. Char",id:"a55-char",level:2},{value:"A.5.6. Pointer scalar",id:"a56-pointer-scalar",level:2},{value:"A.5.7. Tipe data primitive compound",id:"a57-tipe-data-primitive-compound",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],k={toc:d};function m(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},k,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Tipe data di Rust dikategorikan menjadi beberapa group, salah satunya adalah group tipe data primitif scalar atau biasa disebut ",(0,i.kt)("strong",{parentName:"p"},"scalar types"),". Pada chapter ini kita akan mempelajarinya."),(0,i.kt)("p",null,"Tipe data scalar sendiri merupakan tipe data primitif yang isinya hanya 1 nilai. Rust memiliki 4 tipe scalar, yaitu integers, floating-point, boolean, dan character."),(0,i.kt)("h2",{id:"a51-signed-integers"},"A.5.1. Signed integers"),(0,i.kt)("p",null,"Signed integer merupakan tipe data numerik/integer yang bisa menampung nilai positif dan juga negatif. Ada beberapa tipe data singed integer tersedia di Rust yang dibedakan sesuai size-nya."),(0,i.kt)("p",null,"Tipe data ini keyword-nya ditandai dengan huruf awalan ",(0,i.kt)("inlineCode",{parentName:"p"},"i"),", contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},"i8"),", yang dimana tipe ini adalah tipe data numerik integer dengan range value yang bisa ditampung adalah mulai dari angka ",(0,i.kt)("strong",{parentName:"p"},"\u2212128")," (didapat dari \u2212(2",(0,i.kt)("sup",null,"7"),") hingga ",(0,i.kt)("strong",{parentName:"p"},"127")," (didapat dari 2",(0,i.kt)("sup",null,"7"),"-1)."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let numerik1 = 24;\nlet numerik2: i8 = 2;\nlet numerik3: i64 = 12;\n\nprintln!("{} | {} | {}", numerik1, numerik2, numerik3);\n// output ==> 24 | 2 | 12\n')),(0,i.kt)("p",null,"Dengan menggunakan teknik deklarasi ",(0,i.kt)("em",{parentName:"p"},"type inference"),", maka default angka numerik tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,i.kt)("p",null,"Range value pada tipe data itu cukup penting untuk diperhatikan, jika ada sebuah variabel yang tipe datanya pasti dan diisi dengan nilai di luar kapasitas yang bisa ditampung, maka akan muncul error. Contoh, tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i8"),", jika diisi nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"128")," maka error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"data type error",src:t(4362).Z,width:"642",height:"488"})),(0,i.kt)("p",null,"Umumnya, tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," cukup untuk kebutuhan menampung nilai, tapi ada banyak case dimana kita perlu tipe dengan size yang lebih besar seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"i64"),"."),(0,i.kt)("p",null,"Berikut merupakan list tipe data signed integers yang ada di Rust. Tidak perlu dihafal."),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:null},"Tipe data"),(0,i.kt)("th",{parentName:"tr",align:null},"Deskripsi"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i8")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"7"),") hingga (2",(0,i.kt)("sup",null,"7"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i16")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"15"),") hingga (2",(0,i.kt)("sup",null,"15"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i32")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"31"),") hingga (2",(0,i.kt)("sup",null,"31"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i64")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"63"),") hingga (2",(0,i.kt)("sup",null,"63"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i128")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"127"),") hingga (2",(0,i.kt)("sup",null,"127"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"isize")),(0,i.kt)("td",{parentName:"tr",align:null},"tipe pointer")))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang tipe data ",(0,i.kt)("strong",{parentName:"p"},"pointer")," dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"))),(0,i.kt)("p",null,"Bisa gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MIN")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MAX")," untuk mengetahui nilai minimal dan juga maksimal tipe data signed integers."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let min_i8 = i8::MIN;\nlet max_i8 = i8::MAX;\n\nprintln!("{} | {}", min_i8, max_i8);\n// output ==> -128 | 127\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"i8::MIN")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"i8::MAX")," memiliki tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"::")," yang di Rust merupakan path separator. Untuk sekarang penulis anjurkan tidak terlalu pusing memahami apa itu path separator, karena nantinya akan ada pembahasan yang cukup detail khusus untuk mengupas topik tersebut.")),(0,i.kt)("h2",{id:"a52-unsigned-integers"},"A.5.2. Unsigned integers"),(0,i.kt)("p",null,"Perbedaan antara signed dan unsigned adalah pada range value yang ditampung, size-nya sama, hanya angka minimal dan maksimalnya berbeda."),(0,i.kt)("p",null,"Unsigned di sini maksudnya adalah hanya bisa menampung angka mulai dari 0. Tipe data unsigned tidak bisa menampung angka negatif (jika dipaksa akan memunculkan error)."),(0,i.kt)("p",null,"Berikut list tipe data unsigned integer pada Rust:"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:null},"Tipe data"),(0,i.kt)("th",{parentName:"tr",align:null},"Deskripsi"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u8")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"8"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u16")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"16"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u32")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"32"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u64")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"64"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u128")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"128"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"usize")),(0,i.kt)("td",{parentName:"tr",align:null},"tipe pointer")))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang tipe data ",(0,i.kt)("strong",{parentName:"p"},"pointer")," dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"))),(0,i.kt)("p",null,"Contoh penerapan unsigned integer:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let numerik4: u32 = 28;\nlet numerik5: u8 = 16;\nlet numerik6: u64 = 42;\n\nprintln!("{} | {} | {}", numerik4, numerik5, numerik6);\n// output ==> 28 | 16 | 42\n')),(0,i.kt)("p",null,"Bisa gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MIN")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MAX")," untuk mengetahui nilai minimal dan juga maksimal tipe data unsigned integers."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let min_u8 = u8::MIN;\nlet max_u8 = u8::MAX;\n\nprintln!("{} | {}", min_u8, max_u8);\n// output ==> 0 | 255\n')),(0,i.kt)("p",null,"Catatan saja, variabel yang dideklarasikan dengan predefined value adalah numerik, by default tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," (signed)."),(0,i.kt)("h2",{id:"a53-floating-point"},"A.5.3. Floating point"),(0,i.kt)("p",null,"Floating point adalah tipe data yang mendukung nilai dibelakang koma, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"3.14"),". Di Rust ada dua tipe data floating point, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"f34")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),". Contoh penggunaan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let fp1: f32 = 3.14;\nlet fp2: f64 = 3.1415926535;\n\nprintln!("{} | {:.5}", fp1, fp2);\n// output ==> 3.14 | 3.14159\n')),(0,i.kt)("p",null,"O iya, jumlah digit di belakang koma bisa diatur saat diprint menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Caranya dengan menggunakan notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"{:.n}"),". Sebagai contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"{:.5}")," maka akan menampilkan hanya 5 digit dibelakang koma."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Dalam penerapan definisi variabel sekaligus nilai, jika nilai dituliskan dalam desimal seperti contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},"24"),", maka tipe data variabel penampung adalah numerik."),(0,i.kt)("p",{parentName:"blockquote"},"Jika dituliskan dalam notasi floating point, contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},"3.14"),", maka tipe data variabel penampung adalah float.")),(0,i.kt)("p",null,"Bisa gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MIN")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MAX")," untuk mengetahui nilai minimal dan juga maksimal tipe data floating point."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let min_f32 = f32::MIN;\nlet max_f32 = f32::MAX;\n\nprintln!("min_f32={} | max_f32={}", min_f32, max_f32);\n// output ==> min_f32=-340282350000000000000000000000000000000 | max_f32=340282350000000000000000000000000000000\n')),(0,i.kt)("p",null,"Catatan saja, variabel yang dideklarasikan dengan predefined value numerik yang ada titiknya (",(0,i.kt)("inlineCode",{parentName:"p"},"3.10"),") adalah floating point, by default tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),"."),(0,i.kt)("h2",{id:"a54-bool"},"A.5.4. Bool"),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"bool")," menerima dua pilihan nilai saja, ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"false"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let b1 = true;\nlet b2 = false;\n\nprintln!("{} | {}", b1, b2);\n// output ==> true | false\n')),(0,i.kt)("h2",{id:"a55-char"},"A.5.5. Char"),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"char")," menampung sebuah data (unicode), contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"'n'"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"'-'"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"'2'"),". Penulisan literal untuk tipe ini menggunakan notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"''"),", diapit tanda petik satu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let c1 = 'n';\nlet c2 = '-';\nlet c3 = '2';\n\nprintln!(\"{} | {} | {}\", c1, c2, c3);\n// output ==> n | - | 2\n")),(0,i.kt)("h2",{id:"a56-pointer-scalar"},"A.5.6. Pointer scalar"),(0,i.kt)("p",null,"Deklarasi tipe data pointer cukup mudah, yaitu dengan menuliskan deklarasinya seperti biasa tapi ditambahkan karakter ",(0,i.kt)("inlineCode",{parentName:"p"},"&"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let ptr1: &i32 = &24;\nprintln!("{}", ptr1);\n// output ==> 24\n')),(0,i.kt)("p",null,"Perihal apa itu tipe data pointer dan apa kegunaan prefix ",(0,i.kt)("inlineCode",{parentName:"p"},"&")," akan dibahas pada chapter terpisah, yaitu ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),"."),(0,i.kt)("h2",{id:"a57-tipe-data-primitive-compound"},"A.5.7. Tipe data primitive compound"),(0,i.kt)("p",null,"Selain beberapa tipe data yang sudah dibahas di atas, ada juga jenis tipe data primitif jenis lainnya, yaitu primitive compound yang di antaranya adalah ",(0,i.kt)("a",{parentName:"p",href:"/basic/array"},"Array")," dan ",(0,i.kt)("a",{parentName:"p",href:"/basic/tuple"},"Tuple"),". Lebih detailnya mengenai tipe tersebut dibahas pada chapter terpisah."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/tipe_data_primitive_scalar"},"github.com/novalagung/dasarpemrogramanrust-example/../tipe_data_primitive_scalar")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-string-literal"},"Tipe Data \u279c String Literal (&str)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/array"},"Array")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice"},"Slice (Basic)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tuple"},"Tuple")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-custom-type-string-slice"},"Tipe Data \u279c String Custom Type"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/primitives.html"},"https://doc.rust-lang.org/rust-by-example/primitives.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/index.html#syntax"},"https://doc.rust-lang.org/std/fmt/index.html#syntax")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Integer_(computer_science)"},"https://en.wikipedia.org/wiki/Integer_(computer_science)"))))}m.isMDXComponent=!0},4362:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/tipe-data-primitive-scalar-1-7710e1ad76b0cc40fbae8a8d42ead294.png"}}]); \ No newline at end of file diff --git a/assets/js/bd9b9d52.c2ec587a.js b/assets/js/bd9b9d52.c2ec587a.js new file mode 100644 index 00000000..74682a5d --- /dev/null +++ b/assets/js/bd9b9d52.c2ec587a.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[909],{3905:(a,e,t)=>{t.d(e,{Zo:()=>k,kt:()=>o});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var u=n.createContext({}),d=function(a){var e=n.useContext(u),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},k=function(a){var e=d(a.components);return n.createElement(u.Provider,{value:e},a.children)},m={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},s=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,u=a.parentName,k=p(a,["components","mdxType","originalType","parentName"]),s=d(t),o=i,g=s["".concat(u,".").concat(o)]||s[o]||m[o]||r;return t?n.createElement(g,l(l({ref:e},k),{},{components:t})):n.createElement(g,l({ref:e},k))}));function o(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=s;var p={};for(var u in e)hasOwnProperty.call(e,u)&&(p[u]=e[u]);p.originalType=a,p.mdxType="string"==typeof a?a:i,l[1]=p;for(var d=2;d{t.r(e),t.d(e,{assets:()=>u,contentTitle:()=>l,default:()=>m,frontMatter:()=>r,metadata:()=>p,toc:()=>d});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:5,title:"A.5. Tipe Data \u279c Primitive Scalar",sidebar_label:"A.5. Tipe Data \u279c Primitive Scalar"},l=void 0,p={unversionedId:"basic/tipe-data-primitive-scalar",id:"basic/tipe-data-primitive-scalar",title:"A.5. Tipe Data \u279c Primitive Scalar",description:"Tipe data di Rust dikategorikan menjadi beberapa group, salah satunya adalah group tipe data primitif scalar atau biasa disebut scalar types. Pada chapter ini kita akan mempelajarinya.",source:"@site/docs/basic/tipe-data-primitive-scalar.md",sourceDirName:"basic",slug:"/basic/tipe-data-primitive-scalar",permalink:"/basic/tipe-data-primitive-scalar",draft:!1,tags:[],version:"current",sidebarPosition:5,frontMatter:{sidebar_position:5,title:"A.5. Tipe Data \u279c Primitive Scalar",sidebar_label:"A.5. Tipe Data \u279c Primitive Scalar"},sidebar:"tutorialSidebar",previous:{title:"A.4. Variabel",permalink:"/basic/variabel"},next:{title:"A.6. Tipe Data \u279c String Literal (&str)",permalink:"/basic/tipe-data-string-literal"}},u={},d=[{value:"A.5.1. Signed integers",id:"a51-signed-integers",level:2},{value:"A.5.2. Unsigned integers",id:"a52-unsigned-integers",level:2},{value:"A.5.3. Floating point",id:"a53-floating-point",level:2},{value:"A.5.4. Bool",id:"a54-bool",level:2},{value:"A.5.5. Char",id:"a55-char",level:2},{value:"A.5.6. Pointer scalar",id:"a56-pointer-scalar",level:2},{value:"A.5.7. Tipe data primitive compound",id:"a57-tipe-data-primitive-compound",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],k={toc:d};function m(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},k,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Tipe data di Rust dikategorikan menjadi beberapa group, salah satunya adalah group tipe data primitif scalar atau biasa disebut ",(0,i.kt)("strong",{parentName:"p"},"scalar types"),". Pada chapter ini kita akan mempelajarinya."),(0,i.kt)("p",null,"Tipe data scalar sendiri merupakan tipe data primitif yang isinya hanya 1 nilai. Rust memiliki 4 tipe scalar, yaitu integers, floating-point, boolean, dan character."),(0,i.kt)("h2",{id:"a51-signed-integers"},"A.5.1. Signed integers"),(0,i.kt)("p",null,"Signed integer merupakan tipe data numerik/integer yang bisa menampung nilai positif dan juga negatif. Ada beberapa tipe data singed integer tersedia di Rust yang dibedakan sesuai size-nya."),(0,i.kt)("p",null,"Tipe data ini keyword-nya ditandai dengan huruf awalan ",(0,i.kt)("inlineCode",{parentName:"p"},"i"),", contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},"i8"),", yang mana tipe ini adalah tipe data numerik integer dengan range value yang bisa ditampung adalah mulai dari angka ",(0,i.kt)("strong",{parentName:"p"},"\u2212128")," (didapat dari \u2212(2",(0,i.kt)("sup",null,"7"),") hingga ",(0,i.kt)("strong",{parentName:"p"},"127")," (didapat dari 2",(0,i.kt)("sup",null,"7"),"-1)."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let numerik1 = 24;\nlet numerik2: i8 = 2;\nlet numerik3: i64 = 12;\n\nprintln!("{} | {} | {}", numerik1, numerik2, numerik3);\n// output ==> 24 | 2 | 12\n')),(0,i.kt)("p",null,"Dengan menggunakan teknik deklarasi ",(0,i.kt)("em",{parentName:"p"},"type inference"),", maka default angka numerik tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,i.kt)("p",null,"Range value pada tipe data itu cukup penting untuk diperhatikan, jika ada sebuah variabel yang tipe datanya pasti dan diisi dengan nilai di luar kapasitas yang bisa ditampung, maka akan muncul error. Contoh, tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i8"),", jika diisi nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"128")," maka error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"data type error",src:t(4362).Z,width:"642",height:"488"})),(0,i.kt)("p",null,"Umumnya, tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," cukup digunakan untuk kebutuhan menampung nilai, namun sebenarnya ada banyak case lainnya di mana kita perlu tipe dengan size yang lebih besar seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"i64"),"."),(0,i.kt)("p",null,"Berikut merupakan list tipe data signed integers yang ada di Rust. Tidak perlu dihafal."),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:null},"Tipe data"),(0,i.kt)("th",{parentName:"tr",align:null},"Deskripsi"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i8")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"7"),") hingga (2",(0,i.kt)("sup",null,"7"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i16")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"15"),") hingga (2",(0,i.kt)("sup",null,"15"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i32")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"31"),") hingga (2",(0,i.kt)("sup",null,"31"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i64")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"63"),") hingga (2",(0,i.kt)("sup",null,"63"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"i128")),(0,i.kt)("td",{parentName:"tr",align:null},"\u2212(2",(0,i.kt)("sup",null,"127"),") hingga (2",(0,i.kt)("sup",null,"127"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"isize")),(0,i.kt)("td",{parentName:"tr",align:null},"tipe pointer")))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang tipe data ",(0,i.kt)("strong",{parentName:"p"},"pointer")," dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"))),(0,i.kt)("p",null,"Bisa gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MIN")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MAX")," untuk mengetahui nilai minimal dan juga maksimal tipe data signed integers."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let min_i8 = i8::MIN;\nlet max_i8 = i8::MAX;\n\nprintln!("{} | {}", min_i8, max_i8);\n// output ==> -128 | 127\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"i8::MIN")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"i8::MAX")," memiliki tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"::")," yang di Rust merupakan path separator. Untuk sekarang penulis anjurkan tidak terlalu pusing memahami apa itu path separator, karena nantinya akan ada pembahasan yang cukup detail khusus untuk mengupas topik tersebut.")),(0,i.kt)("h2",{id:"a52-unsigned-integers"},"A.5.2. Unsigned integers"),(0,i.kt)("p",null,"Perbedaan antara signed dan unsigned adalah pada range value yang ditampung, size-nya sama, hanya angka minimal dan maksimalnya berbeda."),(0,i.kt)("p",null,"Unsigned di sini maksudnya adalah hanya bisa menampung angka mulai dari 0. Tipe data unsigned tidak bisa menampung angka negatif (jika dipaksa akan memunculkan error)."),(0,i.kt)("p",null,"Berikut list tipe data unsigned integer pada Rust:"),(0,i.kt)("table",null,(0,i.kt)("thead",{parentName:"table"},(0,i.kt)("tr",{parentName:"thead"},(0,i.kt)("th",{parentName:"tr",align:null},"Tipe data"),(0,i.kt)("th",{parentName:"tr",align:null},"Deskripsi"))),(0,i.kt)("tbody",{parentName:"table"},(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u8")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"8"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u16")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"16"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u32")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"32"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u64")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"64"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"u128")),(0,i.kt)("td",{parentName:"tr",align:null},"0 hingga (2",(0,i.kt)("sup",null,"128"),")-1")),(0,i.kt)("tr",{parentName:"tbody"},(0,i.kt)("td",{parentName:"tr",align:null},(0,i.kt)("inlineCode",{parentName:"td"},"usize")),(0,i.kt)("td",{parentName:"tr",align:null},"tipe pointer")))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya tentang tipe data ",(0,i.kt)("strong",{parentName:"p"},"pointer")," dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"))),(0,i.kt)("p",null,"Contoh penerapan unsigned integer:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let numerik4: u32 = 28;\nlet numerik5: u8 = 16;\nlet numerik6: u64 = 42;\n\nprintln!("{} | {} | {}", numerik4, numerik5, numerik6);\n// output ==> 28 | 16 | 42\n')),(0,i.kt)("p",null,"Bisa gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MIN")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MAX")," untuk mengetahui nilai minimal dan juga maksimal tipe data unsigned integers."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let min_u8 = u8::MIN;\nlet max_u8 = u8::MAX;\n\nprintln!("{} | {}", min_u8, max_u8);\n// output ==> 0 | 255\n')),(0,i.kt)("p",null,"Catatan saja, variabel yang dideklarasikan dengan predefined value adalah numerik, by default tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," (signed)."),(0,i.kt)("h2",{id:"a53-floating-point"},"A.5.3. Floating point"),(0,i.kt)("p",null,"Floating point adalah tipe data yang mendukung nilai dibelakang koma, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"3.14"),". Di Rust ada dua tipe data floating point, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"f32")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),". Contoh penggunaan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let fp1: f32 = 3.14;\nlet fp2: f64 = 3.1415926535;\n\nprintln!("{} | {:.5}", fp1, fp2);\n// output ==> 3.14 | 3.14159\n')),(0,i.kt)("p",null,"O iya, jumlah digit di belakang koma bisa diatur saat diprint menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Caranya dengan menggunakan notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"{:.n}"),". Sebagai contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"{:.5}")," maka akan menampilkan hanya 5 digit dibelakang koma."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Dalam penerapan definisi variabel sekaligus nilai, jika nilai dituliskan dalam desimal seperti contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},"24"),", maka tipe data variabel penampung adalah numerik."),(0,i.kt)("p",{parentName:"blockquote"},"Jika dituliskan dalam notasi floating point, contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},"3.14"),", maka tipe data variabel penampung adalah float.")),(0,i.kt)("p",null,"Bisa gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MIN")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"tipedata::MAX")," untuk mengetahui nilai minimal dan juga maksimal tipe data floating point."),(0,i.kt)("p",null,"Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let min_f32 = f32::MIN;\nlet max_f32 = f32::MAX;\n\nprintln!("min_f32={} | max_f32={}", min_f32, max_f32);\n// output ==> min_f32=-340282350000000000000000000000000000000 | max_f32=340282350000000000000000000000000000000\n')),(0,i.kt)("p",null,"Catatan saja, variabel yang dideklarasikan dengan predefined value numerik yang ada titiknya (",(0,i.kt)("inlineCode",{parentName:"p"},"3.10"),") adalah floating point, by default tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"f64"),"."),(0,i.kt)("h2",{id:"a54-bool"},"A.5.4. Bool"),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"bool")," menerima dua pilihan nilai saja, ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"false"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let b1 = true;\nlet b2 = false;\n\nprintln!("{} | {}", b1, b2);\n// output ==> true | false\n')),(0,i.kt)("h2",{id:"a55-char"},"A.5.5. Char"),(0,i.kt)("p",null,"Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"char")," menampung sebuah data (unicode), contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"'n'"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"'-'"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"'2'"),". Penulisan literal untuk tipe ini menggunakan notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"''"),", diapit tanda petik satu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let c1 = 'n';\nlet c2 = '-';\nlet c3 = '2';\n\nprintln!(\"{} | {} | {}\", c1, c2, c3);\n// output ==> n | - | 2\n")),(0,i.kt)("h2",{id:"a56-pointer-scalar"},"A.5.6. Pointer scalar"),(0,i.kt)("p",null,"Deklarasi tipe data pointer cukup mudah, yaitu dengan menuliskan deklarasinya seperti biasa tapi ditambahkan karakter ",(0,i.kt)("inlineCode",{parentName:"p"},"&"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let ptr1: &i32 = &24;\nprintln!("{}", ptr1);\n// output ==> 24\n')),(0,i.kt)("p",null,"Perihal apa itu tipe data pointer dan apa kegunaan prefix ",(0,i.kt)("inlineCode",{parentName:"p"},"&")," akan dibahas pada chapter terpisah, yaitu ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),"."),(0,i.kt)("h2",{id:"a57-tipe-data-primitive-compound"},"A.5.7. Tipe data primitive compound"),(0,i.kt)("p",null,"Selain beberapa tipe data yang sudah dibahas di atas, ada juga jenis tipe data primitif jenis lainnya, yaitu primitive compound yang di antaranya adalah ",(0,i.kt)("a",{parentName:"p",href:"/basic/array"},"Array")," dan ",(0,i.kt)("a",{parentName:"p",href:"/basic/tuple"},"Tuple"),". Lebih detailnya mengenai tipe tersebut dibahas pada chapter terpisah."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/tipe_data_primitive_scalar"},"github.com/novalagung/dasarpemrogramanrust-example/../tipe_data_primitive_scalar")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-string-literal"},"Tipe Data \u279c String Literal (&str)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/array"},"Array")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice"},"Slice (Basic)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tuple"},"Tuple")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-custom-type-string-slice"},"Tipe Data \u279c String Custom Type"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/primitives.html"},"https://doc.rust-lang.org/rust-by-example/primitives.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/index.html#syntax"},"https://doc.rust-lang.org/std/fmt/index.html#syntax")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://en.wikipedia.org/wiki/Integer_(computer_science)"},"https://en.wikipedia.org/wiki/Integer_(computer_science)"))))}m.isMDXComponent=!0},4362:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/tipe-data-primitive-scalar-1-7710e1ad76b0cc40fbae8a8d42ead294.png"}}]); \ No newline at end of file diff --git a/assets/js/bdfc7662.5ed547e8.js b/assets/js/bdfc7662.5ed547e8.js new file mode 100644 index 00000000..889c2c09 --- /dev/null +++ b/assets/js/bdfc7662.5ed547e8.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7720],{3905:(e,a,n)=>{n.d(a,{Zo:()=>d,kt:()=>c});var t=n(7294);function i(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function r(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function l(e){for(var a=1;a=0||(i[n]=e[n]);return i}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var u=t.createContext({}),m=function(e){var a=t.useContext(u),n=a;return e&&(n="function"==typeof e?e(a):l(l({},a),e)),n},d=function(e){var a=m(e.components);return t.createElement(u.Provider,{value:a},e.children)},s={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},p=t.forwardRef((function(e,a){var n=e.components,i=e.mdxType,r=e.originalType,u=e.parentName,d=o(e,["components","mdxType","originalType","parentName"]),p=m(n),c=i,g=p["".concat(u,".").concat(c)]||p[c]||s[c]||r;return n?t.createElement(g,l(l({ref:a},d),{},{components:n})):t.createElement(g,l({ref:a},d))}));function c(e,a){var n=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var r=n.length,l=new Array(r);l[0]=p;var o={};for(var u in a)hasOwnProperty.call(a,u)&&(o[u]=a[u]);o.originalType=e,o.mdxType="string"==typeof e?e:i,l[1]=o;for(var m=2;m{n.r(a),n.d(a,{assets:()=>u,contentTitle:()=>l,default:()=>s,frontMatter:()=>r,metadata:()=>o,toc:()=>m});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:24,title:"A.24. Associated Function",sidebar_label:"A.24. Associated Function"},l=void 0,o={unversionedId:"basic/associated-function",id:"basic/associated-function",title:"A.24. Associated Function",description:"Chapter ini membahas tentang associated function. Kita akan belajar apa itu associated function dan apa perbedannya dengan fungsi biasa.",source:"@site/docs/basic/associated-function.md",sourceDirName:"basic",slug:"/basic/associated-function",permalink:"/basic/associated-function",draft:!1,tags:[],version:"current",sidebarPosition:24,frontMatter:{sidebar_position:24,title:"A.24. Associated Function",sidebar_label:"A.24. Associated Function"},sidebar:"tutorialSidebar",previous:{title:"A.23. Struct",permalink:"/basic/struct"},next:{title:"A.25. Method",permalink:"/basic/method"}},u={},m=[{value:"A.24.1. Associated item & associated function",id:"a241-associated-item--associated-function",level:2},{value:"A.24.2. Keyword impl",id:"a242-keyword-impl",level:2},{value:"\u25c9 Fungsi LegoSet::new",id:"-fungsi-legosetnew",level:3},{value:"\u25c9 Naming convention associated function",id:"-naming-convention-associated-function",level:3},{value:"A.24.3. Tipe data Self",id:"a243-tipe-data-self",level:2},{value:"A.24.4. Associated function dalam module",id:"a244-associated-function-dalam-module",level:2},{value:"A.24.5. Tuple struct associated function",id:"a245-tuple-struct-associated-function",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function s(e){let{components:a,...r}=e;return(0,i.kt)("wrapper",(0,t.Z)({},d,r,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas tentang ",(0,i.kt)("em",{parentName:"p"},"associated function"),". Kita akan belajar apa itu associated function dan apa perbedannya dengan fungsi biasa."),(0,i.kt)("h2",{id:"a241-associated-item--associated-function"},"A.24.1. ",(0,i.kt)("em",{parentName:"h2"},"Associated item")," & ",(0,i.kt)("em",{parentName:"h2"},"associated function")),(0,i.kt)("p",null,(0,i.kt)("em",{parentName:"p"},"Associated item")," adalah item yang memiliki asosiasi/hubungan dengan struct atau trait. Item di sini bisa dalam banyak hal, bisa berupa fungsi atau lainnya."),(0,i.kt)("p",null,"Fungsi yang terhubung dengan suatu struct atau trait disebut dengan ",(0,i.kt)("em",{parentName:"p"},"associated function"),". Fungsi jenis ini ekuivalen seperti fungsi biasa, perbedaannya adalah pada deklarasinya yang harus berada di dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl"),", dan pemanggilannya harus menggunakan notasi path ",(0,i.kt)("inlineCode",{parentName:"p"},"NamaStruct::nama_fungsi"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai trait dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai macam-macam item dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/path-item#a183-rust-items"},"Path & Item")))),(0,i.kt)("p",null,"Ok, mari kita lanjut ke bagian praktik. Silakan tulis kode berikut terlebih dahulu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nstruct LegoSet {\n code: i32,\n name: String,\n category: String,\n age_minimum: i32,\n}\n\nfn main() {\n let rough_terrain_crane = LegoSet{\n code: 42082,\n name: String::from("Rough Terrain Crane"),\n category: String::from("Technic"),\n age_minimum: 11,\n };\n\n println!("{:#?}", rough_terrain_crane);\n}\n')),(0,i.kt)("p",null,"Sebuah struct bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," didefinisikan memiliki 4 buah property. Di blok kode fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", dibuat sebuah variabel bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"rough_terrain_crane")," yang merupakan object/instance dari struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),", kemudian object tersebut di-print."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Object struct bisa di-print menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println")," karena pada definisi struct ditambahkan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"#[derive(Debug)]"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Lebih detailnya mengenai attribute dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/attributes"},"Attributes"),".")),(0,i.kt)("p",null,"Selanjutnya kita siapkan dua buah fungsi yang berasosiasi dengan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),"."),(0,i.kt)("h2",{id:"a242-keyword-impl"},"A.24.2. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"impl")),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," digunakan untuk membuat ",(0,i.kt)("em",{parentName:"p"},"associated item"),"."),(0,i.kt)("p",null,"Pada konteks ini keyword tersebut digunakan untuk membuat ",(0,i.kt)("em",{parentName:"p"},"associated function")," untuk struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),", sebuah fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," dengan tugas adalah untuk membantu pembuatan object ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),"."),(0,i.kt)("p",null,"Silakan tambahkan blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," berikut setelah definisi struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"#[derive(Debug)]\nstruct LegoSet {\n code: i32,\n name: String,\n category: String,\n age_minimum: i32,\n}\n\nimpl LegoSet {\n\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("p",null,"Notasi penulisan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," bisa dilihat pada contoh di atas, cukup tulis saja keyword tersebut diikuti nama struct yang diinginkan, kemudian diikuti dengan blok kode berisi definisi fungsi."),(0,i.kt)("p",null,"Fungsi dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," adalah yang disebut dengan ",(0,i.kt)("em",{parentName:"p"},"associated function"),". Pada contoh di atas, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," memiliki asosiasi dengan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),"."),(0,i.kt)("p",null,"Lanjut, panggil fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," tersebut untuk membuat object baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"xtreme_offroader")," lalu print isinya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let rough_terrain_crane = LegoSet{\n code: 42082,\n name: String::from("Rough Terrain Crane"),\n category: String::from("Technic"),\n age_minimum: 11,\n };\n println!("{:#?}", rough_terrain_crane);\n \n let xtreme_offroader = LegoSet::new(\n 42099, \n String::from("4X4 X-treme Off-Roader"), \n String::from("Technic"), \n 11,\n );\n println!("{:#?}", xtreme_offroader);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(8317).Z,width:"447",height:"276"})),(0,i.kt)("p",null,"Bisa dilihat pada blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", cara pemanggilan associated function adalah dengan menuliskan nama struct diikuti nama fungsi dengan notasi penulisan path."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"NamaStruct::nama_fungsi();\nNamaStruct::nama_fungsi(arg1, arg2, arg3, arg4);\nLegoSet::new(arg1, arg2, arg3, arg4);\n")),(0,i.kt)("p",null,"Jadi seperti itu, semoga cukup jelas. Silakan berkreasi dengan menambahkan ",(0,i.kt)("em",{parentName:"p"},"associated function")," lainnya agar makin terbiasa. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'impl LegoSet {\n\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet { code, name, category, age_minimum }\n }\n\n fn what_is_lego() {\n println!("Lego is a line of plastic construction toys")\n }\n}\n\nfn main() {\n LegoSet::what_is_lego();\n\n let xtreme_offroader = LegoSet::new(\n 42099, \n String::from("4X4 X-treme Off-Roader"), \n String::from("Technic"), \n 11,\n );\n println!("{:#?}", xtreme_offroader);\n}\n')),(0,i.kt)("h3",{id:"-fungsi-legosetnew"},"\u25c9 Fungsi ",(0,i.kt)("inlineCode",{parentName:"h3"},"LegoSet::new")),(0,i.kt)("p",null,"O iya, sedikit tambahan info saja, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," di-atas didefinisikan menggunakan metode ",(0,i.kt)("a",{parentName:"p",href:"/basic/struct#-field-init-shorthand"},"field init shorthand"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet { code, name, category, age_minimum }\n}\n\n// ... adalah ekuivalen dengan ...\n\nfn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet {\n code: code,\n name: name,\n category: category,\n age_minimum: age_minimum\n }\n}\n")),(0,i.kt)("h3",{id:"-naming-convention-associated-function"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Naming convention")," associated function"),(0,i.kt)("p",null,"Sesuai anjuran di ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", snake case digunakan dalam penamaan associated function. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl NamaStruct {\n\n fn nama_associated_func() {\n // ...\n }\n}\n")),(0,i.kt)("h2",{id:"a243-tipe-data-self"},"A.24.3. Tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Self")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Self")," (perhatikan huruf ",(0,i.kt)("inlineCode",{parentName:"p"},"S"),"-nya adalah kapital) merupakan representasi untuk tipe data struct atau trait di mana blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," dideklarasikan."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Self")," hanya bisa digunakan dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl"))),(0,i.kt)("p",null,"Sebagai contoh, pada kode yang sudah dipraktikkan, keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," diterapkan dalam pembuatan ",(0,i.kt)("em",{parentName:"p"},"associated items")," untuk struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),". Dalam blok kode tersebut, tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," bisa diganti dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"Self"),"."),(0,i.kt)("p",null,"Silakan lihat contoh di bawah ini, ada 4 buah cara deklarasi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," yang kesemuanya adalah ekuivalen."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl LegoSet {\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl LegoSet {\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n Self { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl LegoSet {\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> Self {\n LegoSet { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl LegoSet {\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> Self {\n Self { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("h2",{id:"a244-associated-function-dalam-module"},"A.24.4. Associated function dalam module"),(0,i.kt)("p",null,"Struct adalah salah satu dari beberapa item yang ada di Rust. Struct bisa saja berada dalam sebuah module, baik inline maupun ",(0,i.kt)("em",{parentName:"p"},"normal")," module."),(0,i.kt)("p",null,"Sekarang, kode yang sudah dipraktikkan di atas akan kita refactor. Struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," beserta associated items-nya dipindah ke module file bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"lego"),", dengan itu maka pemanggilan struct tersebut harus menggunakan path ",(0,i.kt)("inlineCode",{parentName:"p"},"lego::LegoSet"),"."),(0,i.kt)("p",null,"Kurang lebih struktur package mendi seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 lego.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(6314).Z,width:"918",height:"814"})),(0,i.kt)("p",null,"Ada yang aneh, padahal pemanggilan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," sudah diganti menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"lego::LegoSet"),", tapi kenapa ada error di kode? Silakan ",(0,i.kt)("em",{parentName:"p"},"hover")," baris kode yang ada highlight merah, atau jalankan saja program untuk melihat detail errornya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(8820).Z,width:"481",height:"216"})),(0,i.kt)("p",null,"Error muncul karena struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," adalah private. Solusinya cukup tambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," dalam definisi struct beserta ",(0,i.kt)("em",{parentName:"p"},"associated function"),"-nya."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Untuk blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," tidak perlu ditambahi keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub"))),(0,i.kt)("p",null,"Ok, setelah update diaplikasikan, coba lihat lagi. Sekarang error-nya berubah, bagian baris pengisian property struct yang jadi error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(2039).Z,width:"741",height:"798"})),(0,i.kt)("p",null,"Error ini terjadi karena property dari struct terdeteksi sebagai ",(0,i.kt)("strong",{parentName:"p"},"private"),". Solusi yang bisa dipergunakan ada 2:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Tambahkan saja keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," pada definisi property struct."),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/lego.rs"',title:'"src/lego.rs"'},"pub struct LegoSet {\n pub code: i32,\n pub name: String,\n pub category: String,\n pub age_minimum: i32,\n}\n"))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Atau, tetap biarkan property struct sebagai ",(0,i.kt)("strong",{parentName:"p"},"private"),", namun pada semua statement pembuatan object menggunakan tipe tersebut harus selalu dilakukan via ",(0,i.kt)("inlineCode",{parentName:"p"},"lego::LegoSet::new()"),"."),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'let object = lego::LegoSet::new(\n 42099, \n String::from("4X4 X-treme Off-Roader"), \n String::from("Technic"), \n 11,\n);\n')))),(0,i.kt)("p",null,"Opsi mana yang paling pas? Pertanyaan ini jawabannya adalah tergantung kebutuhan dan ",(0,i.kt)("em",{parentName:"p"},"case"),". Jika memang property struct di-desain agar private (tidak perlu diakses secara publik), maka gunakan saja associated function dalam pembuatan object-nya."),(0,i.kt)("p",null,"Namun jika memang ada kebutuhan salah satu property atau kesemuanya harus bisa diakses secara publik, maka tambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," sesuai kebutuhan."),(0,i.kt)("h2",{id:"a245-tuple-struct-associated-function"},"A.24.5. Tuple struct associated function"),(0,i.kt)("p",null,"Sama seperti struct, tuple struct juga bisa memiliki ",(0,i.kt)("em",{parentName:"p"},"associated items"),". Cara deklarasi dan pemanggilan item-nya juga sama."),(0,i.kt)("p",null,"Pada contoh berikut, tuple struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Color")," memiliki 3 buah associated functions, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"red"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"green"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"blue"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 model.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/model.rs"',title:'"src/model.rs"'},"#[derive(Debug)]\npub struct Color(i32, i32, i32);\n\nimpl Color {\n\n pub fn red() -> Self {\n Self(255, 0, 0)\n }\n\n pub fn green() -> Self {\n Self(0, 255, 0)\n }\n\n pub fn blue() -> Self {\n Self(0, 0, 255)\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod model;\n\nfn main() {\n let red = model::Color::red();\n let green = model::Color::green();\n let blue = model::Color::blue();\n\n println!("{:#?} {:#?} {:#?}", red, green, blue);\n \n let random_color = model::Color(12, 25, 47);\n println!("{:#?} ", random_color);\n}\n')),(0,i.kt)("p",null,"Hmm, ada yang aneh, karena suatu alasan statement ",(0,i.kt)("inlineCode",{parentName:"p"},"model::Color")," dianggap error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(6605).Z,width:"662",height:"244"})),(0,i.kt)("p",null,"Error ini sebenarnya mirip dengan error praktik sebelumnya, ketika mencoba membuat object baru dari struct sedangkan property struct tersebut adalah private."),(0,i.kt)("p",null,"Pada kasus struct, solusinya cukup dengan tambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," atau siapkan ",(0,i.kt)("em",{parentName:"p"},"associated function")," untuk pembuatan object. Pada tuple struct, solusinya juga mirip, ada dua opsi yang bisa dipilih."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Tambahkan saja keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," pada definisi parameter tuple struct."),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/model.rs"',title:'"src/model.rs"'},"// before\npub struct Color(i32, i32, i32);\n\n// after\npub struct Color(pub i32, pub i32, pub i32);\n"))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Atau, tetap biarkan parameter tuple struct sebagai ",(0,i.kt)("strong",{parentName:"p"},"private"),", namun siapkan ",(0,i.kt)("em",{parentName:"p"},"associated function")," baru untuk pembuatan object struct tuple, seperti ini:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/model.rs"',title:'"src/model.rs"'},"// ...\n\nimpl Color {\n pub fn new(r: i32, g: i32, b: i32) -> Self {\n Self(r, g, b)\n }\n\n // ...\n}\n")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'fn main() {\n // ...\n\n let random_color = model::Color::new(12, 25, 47);\n println!("{:#?} ", random_color);\n}\n')))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai visibility property dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/associated_function"},"github.com/novalagung/dasarpemrogramanrust-example/../associated_function")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/function"},"Function")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/method"},"Method")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch05-03-method-syntax.html"},"https://doc.rust-lang.org/book/ch05-03-method-syntax.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.impl.html"},"https://doc.rust-lang.org/std/keyword.impl.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.SelfTy.html"},"https://doc.rust-lang.org/std/keyword.SelfTy.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/fn/methods.html"},"https://doc.rust-lang.org/rust-by-example/fn/methods.html"))))}s.isMDXComponent=!0},8317:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},6314:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/associated-function-2-a72ff54955e3f2afff46536bcf659087.png"},8820:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2039:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/associated-function-4-91b4741ed57bb3ce42997c90f9c0c58e.png"},6605:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/associated-function-5-cb72a7e4944bd7b12502c0dc6e7acd7a.png"}}]); \ No newline at end of file diff --git a/assets/js/bdfc7662.e5ac5751.js b/assets/js/bdfc7662.e5ac5751.js deleted file mode 100644 index 96112c15..00000000 --- a/assets/js/bdfc7662.e5ac5751.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7720],{3905:(e,a,n)=>{n.d(a,{Zo:()=>d,kt:()=>c});var t=n(7294);function i(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function r(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function l(e){for(var a=1;a=0||(i[n]=e[n]);return i}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var u=t.createContext({}),m=function(e){var a=t.useContext(u),n=a;return e&&(n="function"==typeof e?e(a):l(l({},a),e)),n},d=function(e){var a=m(e.components);return t.createElement(u.Provider,{value:a},e.children)},s={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},p=t.forwardRef((function(e,a){var n=e.components,i=e.mdxType,r=e.originalType,u=e.parentName,d=o(e,["components","mdxType","originalType","parentName"]),p=m(n),c=i,g=p["".concat(u,".").concat(c)]||p[c]||s[c]||r;return n?t.createElement(g,l(l({ref:a},d),{},{components:n})):t.createElement(g,l({ref:a},d))}));function c(e,a){var n=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var r=n.length,l=new Array(r);l[0]=p;var o={};for(var u in a)hasOwnProperty.call(a,u)&&(o[u]=a[u]);o.originalType=e,o.mdxType="string"==typeof e?e:i,l[1]=o;for(var m=2;m{n.r(a),n.d(a,{assets:()=>u,contentTitle:()=>l,default:()=>s,frontMatter:()=>r,metadata:()=>o,toc:()=>m});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:24,title:"A.24. Associated Function",sidebar_label:"A.24. Associated Function"},l=void 0,o={unversionedId:"basic/associated-function",id:"basic/associated-function",title:"A.24. Associated Function",description:"Chapter ini membahas tentang associated function. Kita akan belajar apa itu associated function dan apa perbedannya dengan fungsi biasa.",source:"@site/docs/basic/associated-function.md",sourceDirName:"basic",slug:"/basic/associated-function",permalink:"/basic/associated-function",draft:!1,tags:[],version:"current",sidebarPosition:24,frontMatter:{sidebar_position:24,title:"A.24. Associated Function",sidebar_label:"A.24. Associated Function"},sidebar:"tutorialSidebar",previous:{title:"A.23. Struct",permalink:"/basic/struct"},next:{title:"A.25. Method",permalink:"/basic/method"}},u={},m=[{value:"A.24.1. Associated item & associated function",id:"a241-associated-item--associated-function",level:2},{value:"A.24.2. Keyword impl",id:"a242-keyword-impl",level:2},{value:"\u25c9 Fungsi LegoSet::new",id:"-fungsi-legosetnew",level:3},{value:"\u25c9 Naming convention associated function",id:"-naming-convention-associated-function",level:3},{value:"A.24.3. Tipe data Self",id:"a243-tipe-data-self",level:2},{value:"A.24.4. Associated function dalam module",id:"a244-associated-function-dalam-module",level:2},{value:"A.24.5. Tuple struct associated function",id:"a245-tuple-struct-associated-function",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function s(e){let{components:a,...r}=e;return(0,i.kt)("wrapper",(0,t.Z)({},d,r,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas tentang ",(0,i.kt)("em",{parentName:"p"},"associated function"),". Kita akan belajar apa itu associated function dan apa perbedannya dengan fungsi biasa."),(0,i.kt)("h2",{id:"a241-associated-item--associated-function"},"A.24.1. ",(0,i.kt)("em",{parentName:"h2"},"Associated item")," & ",(0,i.kt)("em",{parentName:"h2"},"associated function")),(0,i.kt)("p",null,(0,i.kt)("em",{parentName:"p"},"Associated item")," adalah item yang memiliki asosiasi/hubungan dengan struct atau trait. Item di sini bisa dalam banyak hal, bisa berupa fungsi atau lainnya."),(0,i.kt)("p",null,"Fungsi yang terhubung dengan suatu struct atau trait disebut dengan ",(0,i.kt)("em",{parentName:"p"},"associated function"),". Fungsi jenis ini ekuivalen seperti fungsi biasa, perbedaannya adalah pada deklarasinya yang harus berada di dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl"),", dan pemanggilannya harus menggunakan notasi path ",(0,i.kt)("inlineCode",{parentName:"p"},"NamaStruct::nama_fungsi"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai trait dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai macam-macam item dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/path-item#a183-rust-items"},"Path & Item")))),(0,i.kt)("p",null,"Ok, mari kita lanjut ke bagian praktik. Silakan tulis kode berikut terlebih dahulu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'#[derive(Debug)]\nstruct LegoSet {\n code: i32,\n name: String,\n category: String,\n age_minimum: i32,\n}\n\nfn main() {\n let rough_terrain_crane = LegoSet{\n code: 42082,\n name: String::from("Rough Terrain Crane"),\n category: String::from("Technic"),\n age_minimum: 11,\n };\n\n println!("{:#?}", rough_terrain_crane);\n}\n')),(0,i.kt)("p",null,"Sebuah struct bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," didefinisikan memiliki 4 buah property. Di blok kode fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", dibuat sebuah variabel bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"rough_terrain_crane")," yang merupakan object/instance dari struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),", kemudian object tersebut di-print."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Object struct bisa di-print menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println")," karena pada definisi struct ditambahkan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"#[derive(Debug)]"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Lebih detailnya mengenai attribute dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/attributes"},"Attributes"),".")),(0,i.kt)("p",null,"Selanjutnya kita siapkan dua buah fungsi yang berasosiasi dengan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),"."),(0,i.kt)("h2",{id:"a242-keyword-impl"},"A.24.2. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"impl")),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," digunakan untuk membuat ",(0,i.kt)("em",{parentName:"p"},"associated item"),"."),(0,i.kt)("p",null,"Pada konteks ini keyword tersebut digunakan untuk membuat ",(0,i.kt)("em",{parentName:"p"},"associated function")," untuk struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),", sebuah fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," dengan tugas adalah untuk membantu pembuatan object ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),"."),(0,i.kt)("p",null,"Silakan tambahkan blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," berikut setelah definisi struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"#[derive(Debug)]\nstruct LegoSet {\n code: i32,\n name: String,\n category: String,\n age_minimum: i32,\n}\n\nimpl LegoSet {\n\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("p",null,"Notasi penulisan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," bisa dilihat pada contoh di atas, cukup tulis saja keyword tersebut diikuti nama struct yang diinginkan, kemudian diikuti dengan blok kode berisi definisi fungsi."),(0,i.kt)("p",null,"Fungsi dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," adalah yang disebut dengan ",(0,i.kt)("em",{parentName:"p"},"associated function"),". Pada contoh di atas, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," memiliki asosiasi dengan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),"."),(0,i.kt)("p",null,"Lanjut, panggil fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," tersebut untuk membuat object baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"xtreme_offroader")," lalu print isinya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let rough_terrain_crane = LegoSet{\n code: 42082,\n name: String::from("Rough Terrain Crane"),\n category: String::from("Technic"),\n age_minimum: 11,\n };\n println!("{:#?}", rough_terrain_crane);\n \n let xtreme_offroader = LegoSet::new(\n 42099, \n String::from("4X4 X-treme Off-Roader"), \n String::from("Technic"), \n 11,\n );\n println!("{:#?}", xtreme_offroader);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(8317).Z,width:"447",height:"276"})),(0,i.kt)("p",null,"Bisa dilihat pada blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", cara pemanggilan associated function adalah dengan menuliskan nama struct diikuti nama fungsi dengan notasi penulisan path."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"NamaStruct::nama_fungsi();\nNamaStruct::nama_fungsi(arg1, arg2, arg3, arg4);\nLegoSet::new(arg1, arg2, arg3, arg4);\n")),(0,i.kt)("p",null,"Jadi seperti itu, semoga cukup jelas. Silakan berkreasi dengan menambahkan ",(0,i.kt)("em",{parentName:"p"},"associated function")," lainnya agar makin terbiasa. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'impl LegoSet {\n\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet { code, name, category, age_minimum }\n }\n\n fn what_is_lego() {\n println!("Lego is a line of plastic construction toys")\n }\n}\n\nfn main() {\n LegoSet::what_is_lego();\n\n let xtreme_offroader = LegoSet::new(\n 42099, \n String::from("4X4 X-treme Off-Roader"), \n String::from("Technic"), \n 11,\n );\n println!("{:#?}", xtreme_offroader);\n}\n')),(0,i.kt)("h3",{id:"-fungsi-legosetnew"},"\u25c9 Fungsi ",(0,i.kt)("inlineCode",{parentName:"h3"},"LegoSet::new")),(0,i.kt)("p",null,"O iya, sedikit tambahan info saja, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," di-atas didefinisikan menggunakan metode ",(0,i.kt)("a",{parentName:"p",href:"/basic/struct#-field-init-shorthand"},"field init shorthand"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet { code, name, category, age_minimum }\n}\n\n// ... adalah ekuivalen dengan ...\n\nfn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet {\n code: code,\n name: name,\n category: category,\n age_minimum: age_minimum\n }\n}\n")),(0,i.kt)("h3",{id:"-naming-convention-associated-function"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Naming convention")," associated function"),(0,i.kt)("p",null,"Sesuai anjuran di ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", snake case digunakan dalam penamaan associated function. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl NamaStruct {\n\n fn nama_associated_func() {\n // ...\n }\n}\n")),(0,i.kt)("h2",{id:"a243-tipe-data-self"},"A.24.3. Tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Self")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Self")," (perhatikan huruf ",(0,i.kt)("inlineCode",{parentName:"p"},"S"),"-nya adalah kapital) merupakan representasi untuk tipe data struct atau trait dimana blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," dideklarasikan."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Self")," hanya bisa digunakan dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl"))),(0,i.kt)("p",null,"Sebagai contoh, pada kode yang sudah dipraktikkan, keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," diterapkan dalam pembuatan ",(0,i.kt)("em",{parentName:"p"},"associated items")," untuk struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet"),". Dalam blok kode tersebut, tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," bisa diganti dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"Self"),"."),(0,i.kt)("p",null,"Silakan lihat contoh dibawah ini, ada 4 buah cara deklarasi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"new")," yang kesemuanya adalah ekuivalen."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl LegoSet {\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n LegoSet { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl LegoSet {\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> LegoSet {\n Self { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl LegoSet {\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> Self {\n LegoSet { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl LegoSet {\n fn new(code: i32, name: String, category: String, age_minimum: i32) -> Self {\n Self { code, name, category, age_minimum }\n }\n}\n")),(0,i.kt)("h2",{id:"a244-associated-function-dalam-module"},"A.24.4. Associated function dalam module"),(0,i.kt)("p",null,"Struct adalah salah satu dari beberapa item yang ada di Rust. Struct bisa saja berada dalam sebuah module, baik inline maupun ",(0,i.kt)("em",{parentName:"p"},"normal")," module."),(0,i.kt)("p",null,"Sekarang, kode yang sudah dipraktikkan di atas akan kita refactor. Struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," beserta associated items-nya dipindah ke module file bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"lego"),", dengan itu maka pemanggilan struct tersebut harus menggunakan path ",(0,i.kt)("inlineCode",{parentName:"p"},"lego::LegoSet"),"."),(0,i.kt)("p",null,"Kurang lebih struktur package mendi seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 lego.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(6314).Z,width:"918",height:"814"})),(0,i.kt)("p",null,"Ada yang aneh, padahal pemanggilan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," sudah diganti menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"lego::LegoSet"),", tapi kenapa ada error di kode? Silakan ",(0,i.kt)("em",{parentName:"p"},"hover")," baris kode yang ada highlight merah, atau jalankan saja program untuk melihat detail errornya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(8820).Z,width:"481",height:"216"})),(0,i.kt)("p",null,"Error muncul karena struct ",(0,i.kt)("inlineCode",{parentName:"p"},"LegoSet")," adalah private. Solusinya cukup tambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," dalam definisi struct beserta ",(0,i.kt)("em",{parentName:"p"},"associated function"),"-nya."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Untuk blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," tidak perlu ditambahi keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub"))),(0,i.kt)("p",null,"Ok, setelah update diaplikasikan, coba lihat lagi. Sekarang error-nya berubah, bagian baris pengisian property struct yang jadi error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(2039).Z,width:"741",height:"798"})),(0,i.kt)("p",null,"Error ini terjadi karena property dari struct terdeteksi sebagai ",(0,i.kt)("strong",{parentName:"p"},"private"),". Solusi yang bisa dipergunakan ada 2:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Tambahkan saja keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," pada definisi property struct."),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/lego.rs"',title:'"src/lego.rs"'},"pub struct LegoSet {\n pub code: i32,\n pub name: String,\n pub category: String,\n pub age_minimum: i32,\n}\n"))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Atau, tetap biarkan property struct sebagai ",(0,i.kt)("strong",{parentName:"p"},"private"),", namun pada semua statement pembuatan object menggunakan tipe tersebut harus selalu dilakukan via ",(0,i.kt)("inlineCode",{parentName:"p"},"lego::LegoSet::new()"),"."),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'let object = lego::LegoSet::new(\n 42099, \n String::from("4X4 X-treme Off-Roader"), \n String::from("Technic"), \n 11,\n);\n')))),(0,i.kt)("p",null,"Opsi mana yang paling pas? Pertanyaan ini jawabannya adalah tergantung kebutuhan dan ",(0,i.kt)("em",{parentName:"p"},"case"),". Jika memang property struct di-desain agar private (tidak perlu diakses secara publik), maka gunakan saja associated function dalam pembuatan object-nya."),(0,i.kt)("p",null,"Namun jika memang ada kebutuhan salah satu property atau kesemuanya harus bisa diakses secara publik, maka tambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," sesuai kebutuhan."),(0,i.kt)("h2",{id:"a245-tuple-struct-associated-function"},"A.24.5. Tuple struct associated function"),(0,i.kt)("p",null,"Sama seperti struct, tuple struct juga bisa memiliki ",(0,i.kt)("em",{parentName:"p"},"associated items"),". Cara deklarasi dan pemanggilan item-nya juga sama."),(0,i.kt)("p",null,"Pada contoh berikut, tuple struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Color")," memiliki 3 buah associated functions, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"red"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"green"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"blue"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 model.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/model.rs"',title:'"src/model.rs"'},"#[derive(Debug)]\npub struct Color(i32, i32, i32);\n\nimpl Color {\n\n pub fn red() -> Self {\n Self(255, 0, 0)\n }\n\n pub fn green() -> Self {\n Self(0, 255, 0)\n }\n\n pub fn blue() -> Self {\n Self(0, 0, 255)\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod model;\n\nfn main() {\n let red = model::Color::red();\n let green = model::Color::green();\n let blue = model::Color::blue();\n\n println!("{:#?} {:#?} {:#?}", red, green, blue);\n \n let random_color = model::Color(12, 25, 47);\n println!("{:#?} ", random_color);\n}\n')),(0,i.kt)("p",null,"Hmm, ada yang aneh, karena suatu alasan statement ",(0,i.kt)("inlineCode",{parentName:"p"},"model::Color")," dianggap error."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Associated function",src:n(6605).Z,width:"662",height:"244"})),(0,i.kt)("p",null,"Error ini sebenarnya mirip dengan error praktik sebelumnya, ketika mencoba membuat object baru dari struct sedangkan property struct tersebut adalah private."),(0,i.kt)("p",null,"Pada kasus struct, solusinya cukup dengan tambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," atau siapkan ",(0,i.kt)("em",{parentName:"p"},"associated function")," untuk pembuatan object. Pada tuple struct, solusinya juga mirip, ada dua opsi yang bisa dipilih."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Tambahkan saja keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub")," pada definisi parameter tuple struct."),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/model.rs"',title:'"src/model.rs"'},"// before\npub struct Color(i32, i32, i32);\n\n// after\npub struct Color(pub i32, pub i32, pub i32);\n"))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Atau, tetap biarkan parameter tuple struct sebagai ",(0,i.kt)("strong",{parentName:"p"},"private"),", namun siapkan ",(0,i.kt)("em",{parentName:"p"},"associated function")," baru untuk pembuatan object struct tuple, seperti ini:"),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/model.rs"',title:'"src/model.rs"'},"// ...\n\nimpl Color {\n pub fn new(r: i32, g: i32, b: i32) -> Self {\n Self(r, g, b)\n }\n\n // ...\n}\n")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'fn main() {\n // ...\n\n let random_color = model::Color::new(12, 25, 47);\n println!("{:#?} ", random_color);\n}\n')))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai visibility property dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/associated_function"},"github.com/novalagung/dasarpemrogramanrust-example/../associated_function")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/function"},"Function")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/method"},"Method")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/visibility-privacy"},"Module System \u279c Visibility & Privacy"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch05-03-method-syntax.html"},"https://doc.rust-lang.org/book/ch05-03-method-syntax.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.impl.html"},"https://doc.rust-lang.org/std/keyword.impl.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.SelfTy.html"},"https://doc.rust-lang.org/std/keyword.SelfTy.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/fn/methods.html"},"https://doc.rust-lang.org/rust-by-example/fn/methods.html"))))}s.isMDXComponent=!0},8317:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},6314:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/associated-function-2-a72ff54955e3f2afff46536bcf659087.png"},8820:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2039:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/associated-function-4-91b4741ed57bb3ce42997c90f9c0c58e.png"},6605:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/associated-function-5-cb72a7e4944bd7b12502c0dc6e7acd7a.png"}}]); \ No newline at end of file diff --git a/assets/js/be3f5c40.3675591d.js b/assets/js/be3f5c40.3675591d.js new file mode 100644 index 00000000..bfb38427 --- /dev/null +++ b/assets/js/be3f5c40.3675591d.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[5572],{3905:(e,a,n)=>{n.d(a,{Zo:()=>o,kt:()=>c});var t=n(7294);function l(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function i(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function d(e){for(var a=1;a=0||(l[n]=e[n]);return l}(e,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(l[n]=e[n])}return l}var r=t.createContext({}),u=function(e){var a=t.useContext(r),n=a;return e&&(n="function"==typeof e?e(a):d(d({},a),e)),n},o=function(e){var a=u(e.components);return t.createElement(r.Provider,{value:a},e.children)},p={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},s=t.forwardRef((function(e,a){var n=e.components,l=e.mdxType,i=e.originalType,r=e.parentName,o=m(e,["components","mdxType","originalType","parentName"]),s=u(n),c=l,k=s["".concat(r,".").concat(c)]||s[c]||p[c]||i;return n?t.createElement(k,d(d({ref:a},o),{},{components:n})):t.createElement(k,d({ref:a},o))}));function c(e,a){var n=arguments,l=a&&a.mdxType;if("string"==typeof e||l){var i=n.length,d=new Array(i);d[0]=s;var m={};for(var r in a)hasOwnProperty.call(a,r)&&(m[r]=a[r]);m.originalType=e,m.mdxType="string"==typeof e?e:l,d[1]=m;for(var u=2;u{n.r(a),n.d(a,{assets:()=>r,contentTitle:()=>d,default:()=>p,frontMatter:()=>i,metadata:()=>m,toc:()=>u});var t=n(7462),l=(n(7294),n(3905));const i={sidebar_position:22,title:"A.22. Module System \u279c Scope & Akses Item",sidebar_label:"A.22. Module System \u279c Scope & Akses Item"},d=void 0,m={unversionedId:"basic/module-scope-item-access",id:"basic/module-scope-item-access",title:"A.22. Module System \u279c Scope & Akses Item",description:"Pembahasan chapter ini masih dalam lingkup module system, yaitu tentang scope dan pengakesan module item.",source:"@site/docs/basic/module-scope-item-access.md",sourceDirName:"basic",slug:"/basic/module-scope-item-access",permalink:"/basic/module-scope-item-access",draft:!1,tags:[],version:"current",sidebarPosition:22,frontMatter:{sidebar_position:22,title:"A.22. Module System \u279c Scope & Akses Item",sidebar_label:"A.22. Module System \u279c Scope & Akses Item"},sidebar:"tutorialSidebar",previous:{title:"A.21. Module System \u279c Inline Module",permalink:"/basic/module-inline"},next:{title:"A.23. Struct",permalink:"/basic/struct"}},r={},u=[{value:"A.22.1. Scope",id:"a221-scope",level:2},{value:"A.22.2. Module Scope",id:"a222-module-scope",level:2},{value:"A.22.3. Keyword self dan crate root",id:"a223-keyword-self-dan-crate-root",level:2},{value:"A.22.4. Keyword self pada module scope",id:"a224-keyword-self-pada-module-scope",level:2},{value:"A.22.5. Keyword crate pada module scope",id:"a225-keyword-crate-pada-module-scope",level:2},{value:"A.22.6. Keyword super",id:"a226-keyword-super",level:2},{value:"A.22.7. Praktik lanjutan",id:"a227-praktik-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:u};function p(e){let{components:a,...i}=e;return(0,l.kt)("wrapper",(0,t.Z)({},o,i,{components:a,mdxType:"MDXLayout"}),(0,l.kt)("p",null,"Pembahasan chapter ini masih dalam lingkup module system, yaitu tentang scope dan pengakesan module item."),(0,l.kt)("h2",{id:"a221-scope"},"A.22.1. Scope"),(0,l.kt)("p",null,"Scope bisa diartikan dengan: representasi di mana kode berada. Apapun yang ditulis dalam blok kode (ditandai dengan diapit tanda kurung kurawal ",(0,l.kt)("inlineCode",{parentName:"p"},"{ }"),") berarti dalam satu scope yang sama."),(0,l.kt)("p",null,"Agar lebih jelas, lihat kode berikut kemudian pelajari penjelasan di bawahnya:"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'const PI: f64 = 3.14;\n\nfn main() {\n my_func();\n}\n\nfn my_func() {\n let nama = "Sylvanas Windrunner";\n let occupation = "ex-Warchief of the Horde";\n\n // ...\n}\n')),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"Konstanta ",(0,l.kt)("inlineCode",{parentName:"li"},"PI"),", fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"main"),", dan juga fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," di definisikan satu level dan berada di scope terluar (yang pada ebook ini disebut sebagai ",(0,l.kt)("em",{parentName:"li"},"root"),")."),(0,l.kt)("li",{parentName:"ul"},"Statement pemanggilan fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func()")," berada dalam scope blok kode fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"main"),"."),(0,l.kt)("li",{parentName:"ul"},"Variabel ",(0,l.kt)("inlineCode",{parentName:"li"},"nama")," dan ",(0,l.kt)("inlineCode",{parentName:"li"},"occupation")," berada pada scope blok kode fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func"),".")),(0,l.kt)("blockquote",null,(0,l.kt)("p",{parentName:"blockquote"},"Sebenarnya pembahasan mengenai ",(0,l.kt)("inlineCode",{parentName:"p"},"scope")," harus diiringi dengan pembahasan tentang ",(0,l.kt)("a",{parentName:"p",href:"/basic/block-expression"},"Block expression"),", namun karena kita belum mempelajarinya, chapter ini pembahasan hanya akan fokus pada penerapan scope yang berhubungan dengan ",(0,l.kt)("strong",{parentName:"p"},"module scope"),".")),(0,l.kt)("h2",{id:"a222-module-scope"},"A.22.2. Module Scope"),(0,l.kt)("p",null,"Module scope adalah scope untuk module. Apa bedanya dengan scope secara umum? Silakan perhatikan kode berikut terlebih dahulu:"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'mod my_module {\n\n const event_one: &str = "Siege of Ogrimmar";\n\n mod my_submodule {\n \n const event_two: &str = "Battle for Azeroth";\n\n fn func_two() {\n const event_three: &str = "Sepulcher of the First Ones";\n }\n }\n}\n')),(0,l.kt)("p",null,"Pada kode di atas:"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"Konstanta ",(0,l.kt)("inlineCode",{parentName:"li"},"event_one")," berada dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_module"),"."),(0,l.kt)("li",{parentName:"ul"},"Module ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule")," adalah sebuah inline module yang berada dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_module"),". Module ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule")," berada dalam satu scope yang sama dengan variabel ",(0,l.kt)("inlineCode",{parentName:"li"},"event_one"),"."),(0,l.kt)("li",{parentName:"ul"},"Konstanta ",(0,l.kt)("inlineCode",{parentName:"li"},"event_two")," berada dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule"),"."),(0,l.kt)("li",{parentName:"ul"},"Fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"func_two")," berada dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule"),"."),(0,l.kt)("li",{parentName:"ul"},"Konstanta ",(0,l.kt)("inlineCode",{parentName:"li"},"event_three")," ",(0,l.kt)("strong",{parentName:"li"},"scope"),"-nya adalah dalam blok fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"func_two"),". Sedangkan ",(0,l.kt)("strong",{parentName:"li"},"module scope"),"-nya adalah dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule"),". Jadi kelihatan ya bedanya.")),(0,l.kt)("p",null,"Pemahaman tentang module scope ini penting karena akan berhubungan dengan apa yang akan dipelajari di ",(0,l.kt)("em",{parentName:"p"},"section")," berikutnya."),(0,l.kt)("h2",{id:"a223-keyword-self-dan-crate-root"},"A.22.3. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"self")," dan ",(0,l.kt)("em",{parentName:"h2"},"crate root")),(0,l.kt)("p",null,"Keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," merepresentasikan current module scope. Dengannya kita bisa mengakses item yang deklarasinya ada pada module scope yang sama."),(0,l.kt)("blockquote",null,(0,l.kt)("p",{parentName:"blockquote"},"Sebenarnya selain penjelasan di atas, keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," juga digunakan untuk hal lain yaitu sebagai receiver method."),(0,l.kt)("p",{parentName:"blockquote"},"Pembahasan tentang penerapan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," sebagai receiver method dibahas pada chapter ",(0,l.kt)("a",{parentName:"p",href:"/basic/method"},"Method"))),(0,l.kt)("p",null,"Selanjutnya, apa itu ",(0,l.kt)("em",{parentName:"p"},"crate root"),"? ",(0,l.kt)("strong",{parentName:"p"},"Crate root")," adalah apapun yang didefinisikan di file entrypoint crate (yaitu ",(0,l.kt)("inlineCode",{parentName:"p"},"src/main.rs")," untuk binary crate, dan ",(0,l.kt)("inlineCode",{parentName:"p"},"src/lib.rs")," untuk library crate). ",(0,l.kt)("strong",{parentName:"p"},"Crate root")," adalah module scope yang berada di root (paling atas). Apapun yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root")," berarti berada dalam module scope yang sama."),(0,l.kt)("p",null,"Sebagai contoh, pada kode program berikut, module ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod")," dan fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"main")," dideklarasikan di ",(0,l.kt)("em",{parentName:"p"},"crate root")," ",(0,l.kt)("inlineCode",{parentName:"p"},"src/main.rs"),". Artinya kedua item tersebut berada dalam satu module scope yang sama, yaitu di ",(0,l.kt)("em",{parentName:"p"},"crate root"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'mod my_mod {\n pub fn run_the_app(note: &str) {\n println!("calling `my_mod::run_the_app()`. note {}", note);\n }\n}\n\nfn main() {\n my_mod::run_the_app("1st call");\n self::my_mod::run_the_app("2nd call");\n}\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"Module item access self",src:n(7116).Z,width:"490",height:"128"})),(0,l.kt)("p",null,"Pada fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"main"),", ada dua statement pemanggilan module item ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::run_the_app"),". Kedua statement tersebut adalah mirip, pembedanya ada pada statement ke-2, yaitu keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," digunakan di segment path sebagai prefix."),(0,l.kt)("p",null,"Keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," menginstruksi program bahwa item yang dipanggil adalah item yang module scope-nya sama."),(0,l.kt)("p",null,"Pada contoh di atas, module ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod")," berada dalam satu scope yang sama dengan fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"main"),", yaitu di ",(0,l.kt)("em",{parentName:"p"},"crate root"),". Karena itu, ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod")," bisa diakses dengan menggunakan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self"),", atau lansung panggil saja nama module seperti biasanya, ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod()"),"."),(0,l.kt)("p",null,"Bisa dibilang keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," ini opsional dalam penggunaannya, boleh ikut dituliskan dan boleh juga tidak."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'my_mod::run_the_app("1st call");\nself::my_mod::run_the_app("2nd call");\n')),(0,l.kt)("h2",{id:"a224-keyword-self-pada-module-scope"},"A.22.4. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"self")," pada module scope"),(0,l.kt)("p",null,"Lanjut ke praktik berikutnya. Masih tentang keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self"),", kita akan gunakan keyword ini untuk mengakses item dalam module."),(0,l.kt)("p",null,"Silakan tulis kode berikut kemudian jalankan."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn my_func() {\n println!("calling `my_func()`");\n}\n\nmod my_mod {\n \n pub fn my_func() {\n println!("calling `my_mod::my_func()`");\n }\n \n pub fn run_the_app() {\n println!("calling `my_mod::run_the_app()`");\n my_func();\n self::my_func();\n }\n}\n\nfn main() {\n my_mod::run_the_app();\n}\n')),(0,l.kt)("p",null,"Pada contoh bisa dilihat ada dua buah fungsi dideklarasikan dengan nama yang sama persis, yang satu berada di ",(0,l.kt)("em",{parentName:"p"},"crate root"),", satunya lagi merupakan item milik ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod"),"."),(0,l.kt)("p",null,"Di dalam ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::run_the_app")," ada 2 kali pemanggilan fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func"),", satunya menggunakan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," dan satunya tidak. Fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," manakah yang dipanggil? Hasilnya bisa dilihat pada gambar di bawah ini."),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"Module item access self",src:n(576).Z,width:"681",height:"290"})),(0,l.kt)("p",null,"Kedua statement ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func()")," dan ",(0,l.kt)("inlineCode",{parentName:"p"},"self::my_func()")," dalam ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::run_the_app")," adalah mengarah ke fungsi yang sama, yaitu ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::my_func"),"."),(0,l.kt)("p",null,"Jadi kesimpulan dari penerapan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," dalam module item adalah sama seperti penerapannya pada ",(0,l.kt)("em",{parentName:"p"},"crate root"),", yaitu boleh dipakai boleh tidak. Keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," adalah opsional."),(0,l.kt)("p",null,"Lalu bagaimana cara memanggil ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root"),", dengan pemanggilan adalah dari dalam module item? Caranya adalah menggunakan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"crate"),"."),(0,l.kt)("h2",{id:"a225-keyword-crate-pada-module-scope"},"A.22.5. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"crate")," pada module scope"),(0,l.kt)("p",null,"Keyword ini digunakan untuk mengakses apapun yang ada di ",(0,l.kt)("em",{parentName:"p"},"crate root"),"."),(0,l.kt)("p",null,"Pada contoh di atas, fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root")," bisa dipanggil dari fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"main")," dengan statement ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func()")," atau ",(0,l.kt)("inlineCode",{parentName:"p"},"self::my_func()"),". Untuk bagian ini penulis rasa sudah cukup jelas."),(0,l.kt)("p",null,"Beda cerita kalau fungsi tersebut dipanggil dari dalam module item ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app"),". Kedua statement ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func()")," dan ",(0,l.kt)("inlineCode",{parentName:"p"},"self::my_func()")," mengarah ke ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::my_func"),", bukan ke fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," di ",(0,l.kt)("em",{parentName:"p"},"crate root"),". Hal ini karena ",(0,l.kt)("strong",{parentName:"p"},"current module scope")," dalam statement ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app")," adalah module ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod"),", maka pemanggilan ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," tanpa ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," atau dengannya mengarah ke fungsi yang sama yaitu ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::my_func"),"."),(0,l.kt)("p",null,"Pada ",(0,l.kt)("em",{parentName:"p"},"section")," ini kita akan belajar tentang keyword baru, yaitu ",(0,l.kt)("inlineCode",{parentName:"p"},"crate"),". Keyword tersebut digunakan untuk mengakses apapun yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root"),". Kita bisa memanfaatkannya untuk mengakses ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root")," dengan pengaksesan dari dalam module item."),(0,l.kt)("p",null,"Silakan ubah kode di atas menjadi seperti berikut. Perubahannya ada pada statement dalam fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn my_func() {\n println!("calling `my_func()`");\n}\n\nmod my_mod {\n \n pub fn my_func() {\n println!("calling `my_mod::my_func()`");\n }\n \n pub fn run_the_app() {\n println!("calling `my_mod::run_the_app()`");\n crate::my_func();\n self::my_func();\n }\n}\n\nfn main() {\n my_mod::run_the_app();\n}\n')),(0,l.kt)("p",null,"Jalankan program, hasilnya berbeda dengan eksekusi program sebelumnya. Statement ",(0,l.kt)("inlineCode",{parentName:"p"},"crate::my_func()")," dalam fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app")," mengarah ke fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," di root, sedangkan ",(0,l.kt)("inlineCode",{parentName:"p"},"self::my_func")," mengarah ke ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::my_func"),"."),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"Module item access crate",src:n(7721).Z,width:"487",height:"136"})),(0,l.kt)("h2",{id:"a226-keyword-super"},"A.22.6. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"super")),(0,l.kt)("p",null,"Selain ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," dan ",(0,l.kt)("inlineCode",{parentName:"p"},"crate")," ada juga keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"super"),", yang gunanya adalah untuk mengakses ",(0,l.kt)("em",{parentName:"p"},"parent")," module scope atau 1 scope di atas ",(0,l.kt)("em",{parentName:"p"},"current")," module scope. Agar lebih jelas silakan pelajari kode berikut:"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn my_func() {\n println!("calling `my_func()`");\n}\n\nmod my_mod {\n \n pub fn my_func() {\n println!("calling `my_mod::my_func()`");\n }\n \n pub mod my_submod {\n\n pub fn my_func() {\n println!("calling `my_mod::my_submod::my_func()`");\n }\n \n pub fn run_the_app() {\n println!("calling `my_mod::my_submod::run_the_app()`");\n crate::my_func();\n super::my_func();\n self::my_func();\n }\n }\n}\n\nfn main() {\n my_mod::my_submod::run_the_app();\n}\n')),(0,l.kt)("p",null,"Ada 3 buah fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," dideklarasikan:"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"Fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang berada di ",(0,l.kt)("em",{parentName:"li"},"crate root"),"."),(0,l.kt)("li",{parentName:"ul"},"Fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang merupakan module item milik ",(0,l.kt)("inlineCode",{parentName:"li"},"my_mod"),"."),(0,l.kt)("li",{parentName:"ul"},"Fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang merupakan module item milik submodule ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submod"),".")),(0,l.kt)("p",null,"Dalam fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app"),", ketiga fungsi dengan nama tersebut dipanggil."),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"Statement ",(0,l.kt)("inlineCode",{parentName:"li"},"crate::my_func()")," akan mengarah ke fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang ada di ",(0,l.kt)("em",{parentName:"li"},"crate root"),"."),(0,l.kt)("li",{parentName:"ul"},"Statement ",(0,l.kt)("inlineCode",{parentName:"li"},"super::my_func()")," akan mengarah ke fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang ada di parent module scope, yaitu ",(0,l.kt)("inlineCode",{parentName:"li"},"my_mod::my_func"),"."),(0,l.kt)("li",{parentName:"ul"},"Statement ",(0,l.kt)("inlineCode",{parentName:"li"},"self::my_func()")," akan mengarah ke fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang ada di current module scope, yaitu ",(0,l.kt)("inlineCode",{parentName:"li"},"my_mod::my_submod::my_func"),".")),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"Module item access super",src:n(2781).Z,width:"487",height:"152"})),(0,l.kt)("p",null,"Semoga cukup jelas ya. Silakan ulangi terus praktik di atas jika perlu, agar makin paham."),(0,l.kt)("p",null,"O iya, keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"super")," ini bisa digunakan banyak kali sesuai kebutuhan. Setiap kali keyword ditulis sebagai segment path, maka artinya 1 level parent module scope."),(0,l.kt)("p",null,"Jika mengacu ke contoh program di atas, maka kedua statement berikut adalah ekuivalen."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},"crate::my_func();\nsuper::super::my_func();\n")),(0,l.kt)("h2",{id:"a227-praktik-lanjutan"},"A.22.7. Praktik lanjutan"),(0,l.kt)("p",null,"Agar pemahaman makin mantab, silakan pelajari program berikut."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn my_func() {\n println!("call `my_func()`");\n}\n\nmod module_a {\n\n // path item \u279c `module_a::my_func`.\n pub fn my_func() {\n println!("call `module_a::my_func()`");\n }\n}\n\nmod module_b {\n\n // path item \u279c `module_b::submodule_b_one`.\n mod submodule_b_one {\n\n // path item \u279c `module_b::submodule_b_one::my_func`.\n pub fn my_func() {\n println!("call `module_b::submodule_b_one::my_func()`");\n }\n }\n\n // path item \u279c `module_b::submodule_b_two`.\n mod submodule_b_two {\n\n // path item \u279c `module_b::submodule_b_two::my_func`.\n pub fn my_func() {\n println!("call `module_b::submodule_b_two::my_func()`");\n\n // current module scope adalah module `submodule_b_two`.\n // keyword `super` di sini mengarah ke parent scope, yaitu `module`.\n //\n // statement `super::my_func()` berikut adalah\n // ekuivalen dengan `module_b::my_func()`\n // jika diakses dari crate root.\n super::my_func();\n //\n // statement `super::submodule_b_one::my_func()` berikut adalah\n // ekuivalen dengan `module_b::submodule_b_one::my_func()`\n // jika diakses dari crate root.\n super::submodule_b_one::my_func();\n }\n }\n \n // path item \u279c `module_b::my_func`.\n // fungsi ini tidak publik, jadi hanya bisa diakses dalam scope module `module_b` saja.\n fn my_func() {\n println!("call `module_b::my_func()`");\n }\n \n // path item \u279c `module_b::run_all_funcs`.\n pub fn run_all_funcs() {\n\n // semua fungsi yang didefinisikan akan di call dalam blok kode ini.\n print!("call `my::run_all_funcs()`");\n \n // keyword `self` merepresentasikan current module scope.\n // menjadikan dua statement berikut adalah ekuivalen:\n // `my_func()` adalah ekuivalen dengan `self::my_func()`.\n my_func();\n self::my_func();\n\n // current module scope adalah `module_b`.\n // keyword `super` di sini mengarah ke parent scope, yaitu root atau scope paling luar.\n //\n // statement `super::my_func()` berikut adalah\n // memanggil fungsi `my_func` yang ada di crate root,\n // yang deklarasinya satu level dengan fungsi `main`.\n super::my_func();\n //\n // statement `super::module_a::my_func()` berikut adalah\n // memanggil fungsi `my_func` milik module `module_a` yang ada di crate root.\n super::module_a::my_func();\n //\n // module `submodule_b_two` bisa diakses menggunakan self ataupun tidak\n // karena module tersebut merupakan item yang deklarasinya 1 scope dengan fungsi ini,\n // 1 level dengan `run_all_funcs`.\n submodule_b_two::my_func();\n self::submodule_b_two::my_func();\n }\n}\n\nfn main() {\n module_b::run_all_funcs();\n}\n')),(0,l.kt)("hr",null),(0,l.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,l.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,l.kt)("pre",null,(0,l.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/module_scope_item_access"},"github.com/novalagung/dasarpemrogramanrust-example/../module_scope_item_access")),(0,l.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html"},"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.self.html"},"https://doc.rust-lang.org/std/keyword.self.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.super.html"},"https://doc.rust-lang.org/std/keyword.super.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.crate.html"},"https://doc.rust-lang.org/std/keyword.crate.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/mod.html"},"https://doc.rust-lang.org/rust-by-example/mod.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/meta/doc.html"},"https://doc.rust-lang.org/rust-by-example/meta/doc.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html"))))}p.isMDXComponent=!0},7116:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},576:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/module-scope-item-access-2-d5e5b26c8180ce6a413ba9a494361526.png"},7721:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2781:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/be3f5c40.4baec74d.js b/assets/js/be3f5c40.4baec74d.js deleted file mode 100644 index b86beac3..00000000 --- a/assets/js/be3f5c40.4baec74d.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[5572],{3905:(e,a,n)=>{n.d(a,{Zo:()=>o,kt:()=>c});var t=n(7294);function l(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function i(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function d(e){for(var a=1;a=0||(l[n]=e[n]);return l}(e,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(l[n]=e[n])}return l}var r=t.createContext({}),u=function(e){var a=t.useContext(r),n=a;return e&&(n="function"==typeof e?e(a):d(d({},a),e)),n},o=function(e){var a=u(e.components);return t.createElement(r.Provider,{value:a},e.children)},p={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},s=t.forwardRef((function(e,a){var n=e.components,l=e.mdxType,i=e.originalType,r=e.parentName,o=m(e,["components","mdxType","originalType","parentName"]),s=u(n),c=l,k=s["".concat(r,".").concat(c)]||s[c]||p[c]||i;return n?t.createElement(k,d(d({ref:a},o),{},{components:n})):t.createElement(k,d({ref:a},o))}));function c(e,a){var n=arguments,l=a&&a.mdxType;if("string"==typeof e||l){var i=n.length,d=new Array(i);d[0]=s;var m={};for(var r in a)hasOwnProperty.call(a,r)&&(m[r]=a[r]);m.originalType=e,m.mdxType="string"==typeof e?e:l,d[1]=m;for(var u=2;u{n.r(a),n.d(a,{assets:()=>r,contentTitle:()=>d,default:()=>p,frontMatter:()=>i,metadata:()=>m,toc:()=>u});var t=n(7462),l=(n(7294),n(3905));const i={sidebar_position:22,title:"A.22. Module System \u279c Scope & Akses Item",sidebar_label:"A.22. Module System \u279c Scope & Akses Item"},d=void 0,m={unversionedId:"basic/module-scope-item-access",id:"basic/module-scope-item-access",title:"A.22. Module System \u279c Scope & Akses Item",description:"Pembahasan chapter ini masih dalam lingkup module system, yaitu tentang scope dan pengakesan module item.",source:"@site/docs/basic/module-scope-item-access.md",sourceDirName:"basic",slug:"/basic/module-scope-item-access",permalink:"/basic/module-scope-item-access",draft:!1,tags:[],version:"current",sidebarPosition:22,frontMatter:{sidebar_position:22,title:"A.22. Module System \u279c Scope & Akses Item",sidebar_label:"A.22. Module System \u279c Scope & Akses Item"},sidebar:"tutorialSidebar",previous:{title:"A.21. Module System \u279c Inline Module",permalink:"/basic/module-inline"},next:{title:"A.23. Struct",permalink:"/basic/struct"}},r={},u=[{value:"A.22.1. Scope",id:"a221-scope",level:2},{value:"A.22.2. Module Scope",id:"a222-module-scope",level:2},{value:"A.22.3. Keyword self dan crate root",id:"a223-keyword-self-dan-crate-root",level:2},{value:"A.22.4. Keyword self pada module scope",id:"a224-keyword-self-pada-module-scope",level:2},{value:"A.22.5. Keyword crate pada module scope",id:"a225-keyword-crate-pada-module-scope",level:2},{value:"A.22.6. Keyword super",id:"a226-keyword-super",level:2},{value:"A.22.7. Praktik lanjutan",id:"a227-praktik-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:u};function p(e){let{components:a,...i}=e;return(0,l.kt)("wrapper",(0,t.Z)({},o,i,{components:a,mdxType:"MDXLayout"}),(0,l.kt)("p",null,"Pembahasan chapter ini masih dalam lingkup module system, yaitu tentang scope dan pengakesan module item."),(0,l.kt)("h2",{id:"a221-scope"},"A.22.1. Scope"),(0,l.kt)("p",null,"Scope bisa diartikan dengan: representasi dimana kode berada. Apapun yang ditulis dalam blok kode (ditandai dengan diapit tanda kurung kurawal ",(0,l.kt)("inlineCode",{parentName:"p"},"{ }"),") berarti dalam satu scope yang sama."),(0,l.kt)("p",null,"Agar lebih jelas, lihat kode berikut kemudian pelajari penjelasan dibawahnya:"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'const PI: f64 = 3.14;\n\nfn main() {\n my_func();\n}\n\nfn my_func() {\n let nama = "Sylvanas Windrunner";\n let occupation = "ex-Warchief of the Horde";\n\n // ...\n}\n')),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"Konstanta ",(0,l.kt)("inlineCode",{parentName:"li"},"PI"),", fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"main"),", dan juga fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," di definisikan satu level dan berada di scope terluar (yang pada ebook ini disebut sebagai ",(0,l.kt)("em",{parentName:"li"},"root"),")."),(0,l.kt)("li",{parentName:"ul"},"Statement pemanggilan fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func()")," berada dalam scope blok kode fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"main"),"."),(0,l.kt)("li",{parentName:"ul"},"Variabel ",(0,l.kt)("inlineCode",{parentName:"li"},"nama")," dan ",(0,l.kt)("inlineCode",{parentName:"li"},"occupation")," berada pada scope blok kode fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func"),".")),(0,l.kt)("blockquote",null,(0,l.kt)("p",{parentName:"blockquote"},"Sebenarnya pembahasan mengenai ",(0,l.kt)("inlineCode",{parentName:"p"},"scope")," harus diiringi dengan pembahasan tentang ",(0,l.kt)("a",{parentName:"p",href:"/basic/block-expression"},"Block expression"),", namun karena kita belum mempelajarinya, chapter ini pembahasan hanya akan fokus pada penerapan scope yang berhubungan dengan ",(0,l.kt)("strong",{parentName:"p"},"module scope"),".")),(0,l.kt)("h2",{id:"a222-module-scope"},"A.22.2. Module Scope"),(0,l.kt)("p",null,"Module scope adalah scope untuk module. Apa bedanya dengan scope secara umum? Silakan perhatikan kode berikut terlebih dahulu:"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'mod my_module {\n\n const event_one: &str = "Siege of Ogrimmar";\n\n mod my_submodule {\n \n const event_two: &str = "Battle for Azeroth";\n\n fn func_two() {\n const event_three: &str = "Sepulcher of the First Ones";\n }\n }\n}\n')),(0,l.kt)("p",null,"Pada kode di atas:"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"Konstanta ",(0,l.kt)("inlineCode",{parentName:"li"},"event_one")," berada dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_module"),"."),(0,l.kt)("li",{parentName:"ul"},"Module ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule")," adalah sebuah inline module yang berada dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_module"),". Module ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule")," berada dalam satu scope yang sama dengan variabel ",(0,l.kt)("inlineCode",{parentName:"li"},"event_one"),"."),(0,l.kt)("li",{parentName:"ul"},"Konstanta ",(0,l.kt)("inlineCode",{parentName:"li"},"event_two")," berada dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule"),"."),(0,l.kt)("li",{parentName:"ul"},"Fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"func_two")," berada dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule"),"."),(0,l.kt)("li",{parentName:"ul"},"Konstanta ",(0,l.kt)("inlineCode",{parentName:"li"},"event_three")," ",(0,l.kt)("strong",{parentName:"li"},"scope"),"-nya adalah dalam blok fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"func_two"),". Sedangkan ",(0,l.kt)("strong",{parentName:"li"},"module scope"),"-nya adalah dalam module scope ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submodule"),". Jadi kelihatan ya bedanya.")),(0,l.kt)("p",null,"Pemahaman tentang module scope ini penting karena akan berhubungan dengan apa yang akan dipelajari di ",(0,l.kt)("em",{parentName:"p"},"section")," berikutnya."),(0,l.kt)("h2",{id:"a223-keyword-self-dan-crate-root"},"A.22.3. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"self")," dan ",(0,l.kt)("em",{parentName:"h2"},"crate root")),(0,l.kt)("p",null,"Keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," merepresentasikan current module scope. Dengannya kita bisa mengakses item yang deklarasinya ada pada module scope yang sama."),(0,l.kt)("blockquote",null,(0,l.kt)("p",{parentName:"blockquote"},"Sebenarnya selain penjelasan di atas, keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," juga digunakan untuk hal lain yaitu sebagai receiver method."),(0,l.kt)("p",{parentName:"blockquote"},"Pembahasan tentang penerapan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," sebagai receiver method dibahas pada chapter ",(0,l.kt)("a",{parentName:"p",href:"/basic/method"},"Method"))),(0,l.kt)("p",null,"Selanjutnya, apa itu ",(0,l.kt)("em",{parentName:"p"},"crate root"),"? ",(0,l.kt)("strong",{parentName:"p"},"Crate root")," adalah apapun yang didefinisikan di file entrypoint crate (yaitu ",(0,l.kt)("inlineCode",{parentName:"p"},"src/main.rs")," untuk binary crate, dan ",(0,l.kt)("inlineCode",{parentName:"p"},"src/lib.rs")," untuk library crate). ",(0,l.kt)("strong",{parentName:"p"},"Crate root")," adalah module scope yang berada di root (paling atas). Apapun yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root")," berarti berada dalam module scope yang sama."),(0,l.kt)("p",null,"Sebagai contoh, pada kode program berikut, module ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod")," dan fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"main")," dideklarasikan di ",(0,l.kt)("em",{parentName:"p"},"crate root")," ",(0,l.kt)("inlineCode",{parentName:"p"},"src/main.rs"),". Artinya kedua item tersebut berada dalam satu module scope yang sama, yaitu di ",(0,l.kt)("em",{parentName:"p"},"crate root"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'mod my_mod {\n pub fn run_the_app(note: &str) {\n println!("calling `my_mod::run_the_app()`. note {}", note);\n }\n}\n\nfn main() {\n my_mod::run_the_app("1st call");\n self::my_mod::run_the_app("2nd call");\n}\n')),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"Module item access self",src:n(7116).Z,width:"490",height:"128"})),(0,l.kt)("p",null,"Pada fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"main"),", ada dua statement pemanggilan module item ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::run_the_app"),". Kedua statement tersebut adalah mirip, pembedanya ada pada statement ke-2, yaitu keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," digunakan di segment path sebagai prefix."),(0,l.kt)("p",null,"Keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," menginstruksi program bahwa item yang dipanggil adalah item yang module scope-nya sama."),(0,l.kt)("p",null,"Pada contoh di atas, module ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod")," berada dalam satu scope yang sama dengan fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"main"),", yaitu di ",(0,l.kt)("em",{parentName:"p"},"crate root"),". Karena itu, ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod")," bisa diakses dengan menggunakan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self"),", atau lansung panggil saja nama module seperti biasanya, ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod()"),"."),(0,l.kt)("p",null,"Bisa dibilang keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," ini opsional dalam penggunaannya, boleh ikut dituliskan dan boleh juga tidak."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'my_mod::run_the_app("1st call");\nself::my_mod::run_the_app("2nd call");\n')),(0,l.kt)("h2",{id:"a224-keyword-self-pada-module-scope"},"A.22.4. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"self")," pada module scope"),(0,l.kt)("p",null,"Lanjut ke praktik berikutnya. Masih tentang keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self"),", kita akan gunakan keyword ini untuk mengakses item dalam module."),(0,l.kt)("p",null,"Silakan tulis kode berikut kemudian jalankan."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn my_func() {\n println!("calling `my_func()`");\n}\n\nmod my_mod {\n \n pub fn my_func() {\n println!("calling `my_mod::my_func()`");\n }\n \n pub fn run_the_app() {\n println!("calling `my_mod::run_the_app()`");\n my_func();\n self::my_func();\n }\n}\n\nfn main() {\n my_mod::run_the_app();\n}\n')),(0,l.kt)("p",null,"Pada contoh bisa dilihat ada dua buah fungsi dideklarasikan dengan nama yang sama persis, yang satu berada di ",(0,l.kt)("em",{parentName:"p"},"crate root"),", satunya lagi merupakan item milik ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod"),"."),(0,l.kt)("p",null,"Di dalam ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::run_the_app")," ada 2 kali pemanggilan fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func"),", satunya menggunakan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," dan satunya tidak. Fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," manakah yang dipanggil? Hasilnya bisa dilihat pada gambar dibawah ini."),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"Module item access self",src:n(576).Z,width:"681",height:"290"})),(0,l.kt)("p",null,"Kedua statement ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func()")," dan ",(0,l.kt)("inlineCode",{parentName:"p"},"self::my_func()")," dalam ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::run_the_app")," adalah mengarah ke fungsi yang sama, yaitu ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::my_func"),"."),(0,l.kt)("p",null,"Jadi kesimpulan dari penerapan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," dalam module item adalah sama seperti penerapannya pada ",(0,l.kt)("em",{parentName:"p"},"crate root"),", yaitu boleh dipakai boleh tidak. Keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," adalah opsional."),(0,l.kt)("p",null,"Lalu bagaimana cara memanggil ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root"),", dengan pemanggilan adalah dari dalam module item? Caranya adalah menggunakan keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"crate"),"."),(0,l.kt)("h2",{id:"a225-keyword-crate-pada-module-scope"},"A.22.5. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"crate")," pada module scope"),(0,l.kt)("p",null,"Keyword ini digunakan untuk mengakses apapun yang ada di ",(0,l.kt)("em",{parentName:"p"},"crate root"),"."),(0,l.kt)("p",null,"Pada contoh di atas, fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root")," bisa dipanggil dari fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"main")," dengan statement ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func()")," atau ",(0,l.kt)("inlineCode",{parentName:"p"},"self::my_func()"),". Untuk bagian ini penulis rasa sudah cukup jelas."),(0,l.kt)("p",null,"Beda cerita kalau fungsi tersebut dipanggil dari dalam module item ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app"),". Kedua statement ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func()")," dan ",(0,l.kt)("inlineCode",{parentName:"p"},"self::my_func()")," mengarah ke ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::my_func"),", bukan ke fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," di ",(0,l.kt)("em",{parentName:"p"},"crate root"),". Hal ini karena ",(0,l.kt)("strong",{parentName:"p"},"current module scope")," dalam statement ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app")," adalah module ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod"),", maka pemanggilan ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," tanpa ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," atau dengannya mengarah ke fungsi yang sama yaitu ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::my_func"),"."),(0,l.kt)("p",null,"Pada ",(0,l.kt)("em",{parentName:"p"},"section")," ini kita akan belajar tentang keyword baru, yaitu ",(0,l.kt)("inlineCode",{parentName:"p"},"crate"),". Keyword tersebut digunakan untuk mengakses apapun yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root"),". Kita bisa memanfaatkannya untuk mengakses ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," yang berada di ",(0,l.kt)("em",{parentName:"p"},"crate root")," dengan pengaksesan dari dalam module item."),(0,l.kt)("p",null,"Silakan ubah kode di atas menjadi seperti berikut. Perubahannya ada pada statement dalam fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app"),"."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn my_func() {\n println!("calling `my_func()`");\n}\n\nmod my_mod {\n \n pub fn my_func() {\n println!("calling `my_mod::my_func()`");\n }\n \n pub fn run_the_app() {\n println!("calling `my_mod::run_the_app()`");\n crate::my_func();\n self::my_func();\n }\n}\n\nfn main() {\n my_mod::run_the_app();\n}\n')),(0,l.kt)("p",null,"Jalankan program, hasilnya berbeda dengan eksekusi program sebelumnya. Statement ",(0,l.kt)("inlineCode",{parentName:"p"},"crate::my_func()")," dalam fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app")," mengarah ke fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," di root, sedangkan ",(0,l.kt)("inlineCode",{parentName:"p"},"self::my_func")," mengarah ke ",(0,l.kt)("inlineCode",{parentName:"p"},"my_mod::my_func"),"."),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"Module item access crate",src:n(7721).Z,width:"487",height:"136"})),(0,l.kt)("h2",{id:"a226-keyword-super"},"A.22.6. Keyword ",(0,l.kt)("inlineCode",{parentName:"h2"},"super")),(0,l.kt)("p",null,"Selain ",(0,l.kt)("inlineCode",{parentName:"p"},"self")," dan ",(0,l.kt)("inlineCode",{parentName:"p"},"crate")," ada juga keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"super"),", yang gunanya adalah untuk mengakses ",(0,l.kt)("em",{parentName:"p"},"parent")," module scope atau 1 scope di atas ",(0,l.kt)("em",{parentName:"p"},"current")," module scope. Agar lebih jelas silakan pelajari kode berikut:"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn my_func() {\n println!("calling `my_func()`");\n}\n\nmod my_mod {\n \n pub fn my_func() {\n println!("calling `my_mod::my_func()`");\n }\n \n pub mod my_submod {\n\n pub fn my_func() {\n println!("calling `my_mod::my_submod::my_func()`");\n }\n \n pub fn run_the_app() {\n println!("calling `my_mod::my_submod::run_the_app()`");\n crate::my_func();\n super::my_func();\n self::my_func();\n }\n }\n}\n\nfn main() {\n my_mod::my_submod::run_the_app();\n}\n')),(0,l.kt)("p",null,"Ada 3 buah fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"my_func")," dideklarasikan:"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"Fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang berada di ",(0,l.kt)("em",{parentName:"li"},"crate root"),"."),(0,l.kt)("li",{parentName:"ul"},"Fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang merupakan module item milik ",(0,l.kt)("inlineCode",{parentName:"li"},"my_mod"),"."),(0,l.kt)("li",{parentName:"ul"},"Fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang merupakan module item milik submodule ",(0,l.kt)("inlineCode",{parentName:"li"},"my_submod"),".")),(0,l.kt)("p",null,"Dalam fungsi ",(0,l.kt)("inlineCode",{parentName:"p"},"run_the_app"),", ketiga fungsi dengan nama tersebut dipanggil."),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"Statement ",(0,l.kt)("inlineCode",{parentName:"li"},"crate::my_func()")," akan mengarah ke fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang ada di ",(0,l.kt)("em",{parentName:"li"},"crate root"),"."),(0,l.kt)("li",{parentName:"ul"},"Statement ",(0,l.kt)("inlineCode",{parentName:"li"},"super::my_func()")," akan mengarah ke fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang ada di parent module scope, yaitu ",(0,l.kt)("inlineCode",{parentName:"li"},"my_mod::my_func"),"."),(0,l.kt)("li",{parentName:"ul"},"Statement ",(0,l.kt)("inlineCode",{parentName:"li"},"self::my_func()")," akan mengarah ke fungsi ",(0,l.kt)("inlineCode",{parentName:"li"},"my_func")," yang ada di current module scope, yaitu ",(0,l.kt)("inlineCode",{parentName:"li"},"my_mod::my_submod::my_func"),".")),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"Module item access super",src:n(2781).Z,width:"487",height:"152"})),(0,l.kt)("p",null,"Semoga cukup jelas ya. Silakan ulangi terus praktik di atas jika perlu, agar makin paham."),(0,l.kt)("p",null,"O iya, keyword ",(0,l.kt)("inlineCode",{parentName:"p"},"super")," ini bisa digunakan banyak kali sesuai kebutuhan. Setiap kali keyword ditulis sebagai segment path, maka artinya 1 level parent module scope."),(0,l.kt)("p",null,"Jika mengacu ke contoh program di atas, maka kedua statement berikut adalah ekuivalen."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},"crate::my_func();\nsuper::super::my_func();\n")),(0,l.kt)("h2",{id:"a227-praktik-lanjutan"},"A.22.7. Praktik lanjutan"),(0,l.kt)("p",null,"Agar pemahaman makin mantab, silakan pelajari program berikut."),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-rust"},'fn my_func() {\n println!("call `my_func()`");\n}\n\nmod module_a {\n\n // path item \u279c `module_a::my_func`.\n pub fn my_func() {\n println!("call `module_a::my_func()`");\n }\n}\n\nmod module_b {\n\n // path item \u279c `module_b::submodule_b_one`.\n mod submodule_b_one {\n\n // path item \u279c `module_b::submodule_b_one::my_func`.\n pub fn my_func() {\n println!("call `module_b::submodule_b_one::my_func()`");\n }\n }\n\n // path item \u279c `module_b::submodule_b_two`.\n mod submodule_b_two {\n\n // path item \u279c `module_b::submodule_b_two::my_func`.\n pub fn my_func() {\n println!("call `module_b::submodule_b_two::my_func()`");\n\n // current module scope adalah module `submodule_b_two`.\n // keyword `super` di sini mengarah ke parent scope, yaitu `module`.\n //\n // statement `super::my_func()` berikut adalah\n // ekuivalen dengan `module_b::my_func()`\n // jika diakses dari crate root.\n super::my_func();\n //\n // statement `super::submodule_b_one::my_func()` berikut adalah\n // ekuivalen dengan `module_b::submodule_b_one::my_func()`\n // jika diakses dari crate root.\n super::submodule_b_one::my_func();\n }\n }\n \n // path item \u279c `module_b::my_func`.\n // fungsi ini tidak publik, jadi hanya bisa diakses dalam scope module `module_b` saja.\n fn my_func() {\n println!("call `module_b::my_func()`");\n }\n \n // path item \u279c `module_b::run_all_funcs`.\n pub fn run_all_funcs() {\n\n // semua fungsi yang didefinisikan akan di call dalam blok kode ini.\n print!("call `my::run_all_funcs()`");\n \n // keyword `self` merepresentasikan current module scope.\n // menjadikan dua statement berikut adalah ekuivalen:\n // `my_func()` adalah ekuivalen dengan `self::my_func()`.\n my_func();\n self::my_func();\n\n // current module scope adalah `module_b`.\n // keyword `super` di sini mengarah ke parent scope, yaitu root atau scope paling luar.\n //\n // statement `super::my_func()` berikut adalah\n // memanggil fungsi `my_func` yang ada di crate root,\n // yang deklarasinya satu level dengan fungsi `main`.\n super::my_func();\n //\n // statement `super::module_a::my_func()` berikut adalah\n // memanggil fungsi `my_func` milik module `module_a` yang ada di crate root.\n super::module_a::my_func();\n //\n // module `submodule_b_two` bisa diakses menggunakan self ataupun tidak\n // karena module tersebut merupakan item yang deklarasinya 1 scope dengan fungsi ini,\n // 1 level dengan `run_all_funcs`.\n submodule_b_two::my_func();\n self::submodule_b_two::my_func();\n }\n}\n\nfn main() {\n module_b::run_all_funcs();\n}\n')),(0,l.kt)("hr",null),(0,l.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,l.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,l.kt)("pre",null,(0,l.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/module_scope_item_access"},"github.com/novalagung/dasarpemrogramanrust-example/../module_scope_item_access")),(0,l.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html"},"https://doc.rust-lang.org/book/ch07-02-defining-modules-to-control-scope-and-privacy.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.self.html"},"https://doc.rust-lang.org/std/keyword.self.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.super.html"},"https://doc.rust-lang.org/std/keyword.super.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.crate.html"},"https://doc.rust-lang.org/std/keyword.crate.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/mod.html"},"https://doc.rust-lang.org/rust-by-example/mod.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/meta/doc.html"},"https://doc.rust-lang.org/rust-by-example/meta/doc.html")),(0,l.kt)("li",{parentName:"ul"},(0,l.kt)("a",{parentName:"li",href:"https://aloso.github.io/2021/03/28/module-system.html"},"https://aloso.github.io/2021/03/28/module-system.html"))))}p.isMDXComponent=!0},7116:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},576:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/module-scope-item-access-2-d5e5b26c8180ce6a413ba9a494361526.png"},7721:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2781:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/c526a898.18163a59.js b/assets/js/c526a898.18163a59.js new file mode 100644 index 00000000..2973574a --- /dev/null +++ b/assets/js/c526a898.18163a59.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[9157],{3905:(a,e,n)=>{n.d(e,{Zo:()=>d,kt:()=>o});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function r(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var u=t.createContext({}),m=function(a){var e=t.useContext(u),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},d=function(a){var e=m(a.components);return t.createElement(u.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},s=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,l=a.originalType,u=a.parentName,d=p(a,["components","mdxType","originalType","parentName"]),s=m(n),o=i,g=s["".concat(u,".").concat(o)]||s[o]||k[o]||l;return n?t.createElement(g,r(r({ref:e},d),{},{components:n})):t.createElement(g,r({ref:e},d))}));function o(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var l=n.length,r=new Array(l);r[0]=s;var p={};for(var u in e)hasOwnProperty.call(e,u)&&(p[u]=e[u]);p.originalType=a,p.mdxType="string"==typeof a?a:i,r[1]=p;for(var m=2;m{n.r(e),n.d(e,{assets:()=>u,contentTitle:()=>r,default:()=>k,frontMatter:()=>l,metadata:()=>p,toc:()=>m});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:17,title:"A.17. Function",sidebar_label:"A.17. Function"},r=void 0,p={unversionedId:"basic/function",id:"basic/function",title:"A.17. Function",description:"Pada chapter ini kita akan belajar tentang function atau fungsi.",source:"@site/docs/basic/function.md",sourceDirName:"basic",slug:"/basic/function",permalink:"/basic/function",draft:!1,tags:[],version:"current",sidebarPosition:17,frontMatter:{sidebar_position:17,title:"A.17. Function",sidebar_label:"A.17. Function"},sidebar:"tutorialSidebar",previous:{title:"A.16. Vector",permalink:"/basic/vector"},next:{title:"A.18. Module System \u279c Path & Item",permalink:"/basic/path-item"}},u={},m=[{value:"A.17.1. Keyword fn",id:"a171-keyword-fn",level:2},{value:"A.17.2. Naming convention fungsi",id:"a172-naming-convention-fungsi",level:2},{value:"A.17.3. Parameter fungsi dan argumen fungsi",id:"a173-parameter-fungsi-dan-argumen-fungsi",level:2},{value:"A.17.4. Nilai balik fungsi (return value)",id:"a174-nilai-balik-fungsi-return-value",level:2},{value:"\u25c9 Keyword return untuk penentuan nilai balik",id:"-keyword-return-untuk-penentuan-nilai-balik",level:3},{value:"\u25c9 Return value tanpa keyword return",id:"-return-value-tanpa-keyword-return",level:3},{value:"\u25c9 Statemen sebagai return value tanpa keyword return",id:"-statemen-sebagai-return-value-tanpa-keyword-return",level:3},{value:"\u25c9 Macro format untuk membuat formatted string",id:"-macro-format-untuk-membuat-formatted-string",level:3},{value:"A.17.5. Conditional return value",id:"a175-conditional-return-value",level:2},{value:"A.17.6. Nilai balik fungsi bertipe string literal &str",id:"a176-nilai-balik-fungsi-bertipe-string-literal-str",level:2},{value:"A.17.7. Default return value",id:"a177-default-return-value",level:2},{value:"A.17.8. Pembahasan lanjutan",id:"a178-pembahasan-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function k(a){let{components:e,...l}=a;return(0,i.kt)("wrapper",(0,t.Z)({},d,l,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tentang ",(0,i.kt)("em",{parentName:"p"},"function")," atau fungsi."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Definisi fungsi dalam programming secara terminologi adalah sebuah modul atau sub-program kecil yang digunakan untuk mengeksekusi sebuah perintah, dan bisa di-",(0,i.kt)("em",{parentName:"p"},"reuse")," dalam penggunaannya.")),(0,i.kt)("h2",{id:"a171-keyword-fn"},"A.17.1. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"fn")),(0,i.kt)("p",null,"Fungsi di Rust dibuat menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"fn"),". Salah satu contoh penerapan keyword ini dalam pembuatan fungsi sudah kita praktikkan berulang kali pada definisi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", yang merupakan fungsi utama atau entrypoint sebuah program."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n\n}\n")),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," ini spesial, ketika program dijalankan maka ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," otomatis dipanggil atau dieksekusi."),(0,i.kt)("p",null,"Pembuatan fungsi lainnya-pun juga didefinisikan dengan notasi penulisan yang sama. Pada contoh berikut kita akan buat fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"greet")," yang tugasnya adalah menampilkan message ke layar output."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn greet() {\n println!("hello world");\n}\n')),(0,i.kt)("p",null,"Panggil fungsi tersebut di ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", lalu run program."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n println!("hello rust");\n greet();\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(2114).Z,width:"447",height:"86"})),(0,i.kt)("p",null,"Bisa dilihat ada 2 message muncul, yang pertama adalah pesan ",(0,i.kt)("inlineCode",{parentName:"p"},"hello rust")," yang muncul hasil eksekusi statement ",(0,i.kt)("inlineCode",{parentName:"p"},'println!("hello rust")'),", lalu diikuti ",(0,i.kt)("inlineCode",{parentName:"p"},"hello world")," yang statement print untuk menampilkan message tersebut ada dalam fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"greet"),"."),(0,i.kt)("p",null,"O iya, definisi fungsi tidak mengenal urutan ya. Sebagai contoh fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"greet")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," di atas, bebas mana yang ditulis lebih dulu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn greet() {\n // ..\n}\n\nfn main() {\n // ..\n}\n")),(0,i.kt)("p",null,"... atau ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n // ..\n}\n\nfn greet() {\n // ..\n}\n")),(0,i.kt)("h2",{id:"a172-naming-convention-fungsi"},"A.17.2. ",(0,i.kt)("em",{parentName:"h2"},"Naming convention")," fungsi"),(0,i.kt)("p",null,"Sesuai anjuran di ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", aturan penulisan nama fungsi di Rust adalah menggunakan ",(0,i.kt)("strong",{parentName:"p"},"snake case"),". Nama fungsi dituliskan dalam huruf kecil dengan separator tanda garis bawah atau underscore (",(0,i.kt)("inlineCode",{parentName:"p"},"_"),"). Contohnya seperti: ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"greet_message"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"get_report_Status"),", dll."),(0,i.kt)("h2",{id:"a173-parameter-fungsi-dan-argumen-fungsi"},"A.17.3. Parameter fungsi dan argumen fungsi"),(0,i.kt)("p",null,"Fungsi bisa didefinisikan dengan disertai parameter, dengan itu bisa menyisipkan data saat pemanggilan fungsi."),(0,i.kt)("p",null,"Parameter dituliskan di statement pendefinisian fungsi dengan notasi seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn func_name(param_a: i32) {\n // ...\n}\n\nfn func_name(param_a: i32, param_b: bool) {\n // ...\n}\n\nfn func_name(param_a: i32, param_b: bool, param_c: &str) {\n // ...\n}\n\n// ...\n")),(0,i.kt)("p",null,"Kembali ke praktik, pada bagian ini kita akan buat sebuah fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"greet_custom_message"),". Fungsi ini tugasnya masih sama seperti seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"greet")," hanya saja pesan yang ditampilkan adalah sesuai dengan value parameter fungsi."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn greet_custom_message(name: &str, message: &str) {\n println!("hi {name}, {message}");\n}\n')),(0,i.kt)("p",null,"Ada dua parameter yang didefinisikan:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"name"),", tipe datanya string literal"),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"message"),", tipe datanya juga string literal")),(0,i.kt)("p",null,"Kedua nilai parameter tersebut di-",(0,i.kt)("em",{parentName:"p"},"combine")," lalu di-print menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),"."),(0,i.kt)("p",null,"Ok, sekarang panggil fungsi tersebut dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", sisipkan argumen pertama ",(0,i.kt)("inlineCode",{parentName:"p"},"Damian")," yang ditampung parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"name"),", dan argumen ke-2 ",(0,i.kt)("inlineCode",{parentName:"p"},"welcome to the castle!")," untuk parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"message")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n greet_custom_message("Damian", "welcome to the castle!");\n}\n')),(0,i.kt)("p",null,"Jalankan program. Output bisa dilihat di gambar berikut."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(4002).Z,width:"450",height:"71"})),(0,i.kt)("p",null,"O iya, parameter bisa memiliki tipe data apa saja selama di-",(0,i.kt)("em",{parentName:"p"},"support")," oleh Rust, jadi tidak hanya string literal ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Secara terminologi, parameter adalah variabel yang didefinisikan di fungsi, sedangkan argumen adalah nilai yang disisipkan pada parameter saat pemanggilan fungsi."),(0,i.kt)("p",{parentName:"blockquote"},"Seiring berjalannya waktu, terjadi ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/book/ch03-03-how-functions-work.html#parameters"},"sedikit perubahan")," mengenai makna untuk istilah ",(0,i.kt)("inlineCode",{parentName:"p"},"parameter")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"argumen"),". Sekarang kedua istilah tersebut lebih sering dimaknai sama.")),(0,i.kt)("h2",{id:"a174-nilai-balik-fungsi-return-value"},"A.17.4. Nilai balik fungsi (",(0,i.kt)("em",{parentName:"h2"},"return value"),")"),(0,i.kt)("p",null,"Sebuah fungsi bisa di-desain memiliki nilai balik. Caranya dengan menambahkan tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"->")," diikuti tipe data dari nilai balik pada pendefinisian fungsi."),(0,i.kt)("p",null,"Pada contoh berikut, sebuah fungsi dibuat dengan nama ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume1"),", memiliki 3 buah parameter bertipe numerik (",(0,i.kt)("inlineCode",{parentName:"p"},"width"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"height"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"length"),"), dan nilai balik bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),". Tugas dari fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume1")," adalah melakukan operasi matematika perhitungan volume terhadap data yang didapat dari parameter, yang kemudian hasil kalkulasinya dijadikan nilai balik."),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," (diikuti dengan data) digunakan untuk menentukan nilai balik suatu fungsi. Pada contoh berikut, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"volume")," adalah nilai balik fungsi."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn calculate_box_volume1(width: i32, height: i32, length: i32) -> i32 {\n let volume = width * height * length;\n return volume;\n}\n")),(0,i.kt)("p",null,"Selanjutnya siapkan beberapa variabel yang nantinya digunakan saat pemanggilan ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume1"),". Lalu panggil fungsi tersebut kemudian tampilkan ",(0,i.kt)("em",{parentName:"p"},"return value"),"-nya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let width = 5;\nlet height = 8;\nlet length = 12;\n\nlet res1 = calculate_box_volume1(width, height, length);\nprintln!("result: {res1}");\n')),(0,i.kt)("p",null,"Jalankan program, hasilnya adalah sesuai harapan. Volume dari persegi panjang dengan data ",(0,i.kt)("inlineCode",{parentName:"p"},"5 x 8 x 12")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"480"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(9031).Z,width:"451",height:"64"})),(0,i.kt)("h3",{id:"-keyword-return-untuk-penentuan-nilai-balik"},"\u25c9 Keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"return")," untuk penentuan nilai balik"),(0,i.kt)("p",null,"Salah satu hal yang unik yang ada di Rust adalah beberapa variasi cara penulisan nilai balik fungsi. Cara pertama adalah menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," seperti yang sudah dipraktikkan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn calculate_box_volume1(width: i32, height: i32, length: i32) -> i32 {\n let volume = width * height * length;\n return volume;\n}\n")),(0,i.kt)("p",null,"Untuk cara ini penulis rasa cukup jelas."),(0,i.kt)("h3",{id:"-return-value-tanpa-keyword-return"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Return value")," tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"return")),(0,i.kt)("p",null,"Cara ke-2 kita langsung praktikkan menggunakan kode berikut, silakan tulis lalu jalankan program:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_box_volume2(width: i32, height: i32, length: i32) -> i32 {\n let volume = width * height * length;\n volume\n}\n\nfn main() {\n let name2 = "Damian";\n let res2 = calculate_box_volume2(width, height, length);\n println!("hi {name2}, the box volume is {res2}");\n}\n')),(0,i.kt)("p",null,"Hasilnya kurang lebih sama, hanya di-bagian outputnya beda karena pada contoh ini pesan yang ditampilkan diubah."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(3307).Z,width:"446",height:"67"})),(0,i.kt)("p",null,"Sekarang kita fokus ke kode yang sudah ditulis. Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume2")," kurang lebih struktur dan isinya sama seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume1"),", bedanya hanya pada dua hal:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"calculate_box_volume2")," nilai baliknya dituliskan tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"li"},"return")),(0,i.kt)("li",{parentName:"ul"},"Fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"calculate_box_volume2")," pada statement terakhir tidak diakhir tanda akhir statement ",(0,i.kt)("inlineCode",{parentName:"li"},";"))),(0,i.kt)("p",null,"Kedua poin tersebut merupakan syarat untuk penerapan ",(0,i.kt)("em",{parentName:"p"},"return value")," tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return"),". Caranya kurang lebih adalah dengan menuliskan data nilai balik dengan tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," dan di akhir statement tidak ditambahi tanda semicolon ",(0,i.kt)("inlineCode",{parentName:"p"},";"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Statement terakhir sebuah blok kode fungsi yang ditulis tanpa semicolon ",(0,i.kt)("inlineCode",{parentName:"p"},";")," disebut sebagai ",(0,i.kt)("inlineCode",{parentName:"p"},"tail")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"body tail"),".")),(0,i.kt)("h3",{id:"-statemen-sebagai-return-value-tanpa-keyword-return"},"\u25c9 Statemen sebagai ",(0,i.kt)("em",{parentName:"h3"},"return value")," tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"return")),(0,i.kt)("p",null,"Cara selanjutnya yang akan kita bahas ini sebenarnya sama seperti sebelumnya, hanya saja bedanya di sini tidak menggunakan nama variabel sebagai nilai balik, melainkan langsung statement, yang hasil eksekusi statement tersebut dijadikan nilai balik fungsi."),(0,i.kt)("p",null,"Langsung saja kita praktikkan. Siapkan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume3")," berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_box_volume3(width: i32, height: i32, length: i32) -> i32 {\n width * height * length\n}\n\nfn greet_custom_message(name: &str, message: &str) {\n println!("hi {name}, {message}");\n}\n')),(0,i.kt)("p",null,"Panggil fungsi tersebut, tampung nilai baliknya ke variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"res3"),". Lalu siapkan ",(0,i.kt)("inlineCode",{parentName:"p"},"message3")," yang isinya adalah sebuah pesan string literal yang dibuat menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"format"),". Terakhir tampilkan pesan tersebut menggunakan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"greet_custom_message"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let res3 = calculate_box_volume3(width, height, length);\n let message3 = format!("the box volume is {}", res3);\n\n greet_custom_message("Damian", message3.as_str());\n}\n')),(0,i.kt)("p",null,"Bisa dilihat hasilnya sama seperti pada program sebelumnya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(3307).Z,width:"446",height:"67"})),(0,i.kt)("h3",{id:"-macro-format-untuk-membuat-formatted-string"},"\u25c9 Macro ",(0,i.kt)("inlineCode",{parentName:"h3"},"format")," untuk membuat ",(0,i.kt)("em",{parentName:"h3"},"formatted")," string"),(0,i.kt)("p",null,"Bagian ini adalah penjelasan tambahan dari kode program yang baru saja dipraktikkan."),(0,i.kt)("p",null,"Macro ",(0,i.kt)("inlineCode",{parentName:"p"},"format")," digunakan untuk ",(0,i.kt)("em",{parentName:"p"},"formatting")," sebuah string. ",(0,i.kt)("em",{parentName:"p"},"Formatting")," di sini maksudnya adalah pembuatan string dengan menggunakan teknik penyusunan isi string dalam format tertentu, mirip seperti yang dilakukan menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"print")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),", bedanya hanya pada macro ",(0,i.kt)("inlineCode",{parentName:"p"},"format")," hasilnya bukan sebuah output ke console output, melainkan sebagai nilai balik."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// 2 bari statements berikut ...\nlet message3 = format!("the box volume is {}", res3);\nprintln!("{}", message3.as_str());\n\n// ... adalah sama dengan statement berikut ...\nprintln!("the box volume is {}", res3);\n')),(0,i.kt)("p",null,"Satu hal yang penting diketahui dari macro ",(0,i.kt)("inlineCode",{parentName:"p"},"format"),". Nilai balik macro ini bukanlah string literal ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),", tetapi tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),". ",(0,i.kt)("strong",{parentName:"p"},"Dua tipe tersebut adalah berbeda"),"."),(0,i.kt)("p",null,"Pada contoh di atas, method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_str")," milik tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," digunakan untuk mengambil data string literal-nya. Method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_str")," ini menghasilkan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," dari sebuah ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("h2",{id:"a175-conditional-return-value"},"A.17.5. ",(0,i.kt)("em",{parentName:"h2"},"Conditional return value")),(0,i.kt)("p",null,"Kapan harus menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," dalam penentuan nilai balik dan kapan tidak? Jawabannya mungkin adalah preferensi, tapi di luar itu ada juga case di mana keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," harus digunakan, yaitu pada fungsi yang memiliki nilai balik terkondisi. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn get_score_message(score: f32) -> &\'static str {\n if score == 100.0 {\n return "you got a perfect score!"\n }\n \n if score > 76.0 {\n return "congrats, you passed the exam!"\n }\n\n "your score is below the passing grade"\n}\n\nfn main() {\n println!("{}", get_score_message(100.0));\n println!("{}", get_score_message(98.2));\n println!("{}", get_score_message(33.12));\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(6415).Z,width:"446",height:"100"})),(0,i.kt)("p",null,"Teknik penentuan nilai balik tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," hanya bisa dipergunakan di akhir blok kode, contohnya pada fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"get_score_message")," statement terakhir blok kode adalah string literal ",(0,i.kt)("inlineCode",{parentName:"p"},'"your score is below the passing grade"'),"."),(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"Jika pada selain akhir blok ada kondisi di mana nilai balik harus ditentukan, maka wajib menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"strong"},"return")),". Bisa dilihat pada fungsi yang sudah ditulis, ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," statement dalam blok kode seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," dipergunakan."),(0,i.kt)("h2",{id:"a176-nilai-balik-fungsi-bertipe-string-literal-str"},"A.17.6. Nilai balik fungsi bertipe string literal ",(0,i.kt)("inlineCode",{parentName:"h2"},"&str")),(0,i.kt)("p",null,"Khusus untuk beberapa jenis tipe data, seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),", jika digunakan sebagai tipe data ",(0,i.kt)("em",{parentName:"p"},"return value")," fungsi harus ditambahi keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"static")," dengan penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"&'static str"),". Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn get_score_message(score: f32) -> &\'static str {\n return "you got a perfect score!"\n}\n')),(0,i.kt)("p",null,"Agar tidak makin bingung, pembahasannya dipisah pada chapter lain, yaitu ",(0,i.kt)("a",{parentName:"p",href:"/basic/static"},"Static")," dan ",(0,i.kt)("a",{parentName:"p",href:"/basic/lifetime"},"Lifetime"),"."),(0,i.kt)("h2",{id:"a177-default-return-value"},"A.17.7. ",(0,i.kt)("em",{parentName:"h2"},"Default return value")),(0,i.kt)("p",null,"Tambahan info saja, bahwa di Rust, semua fungsi itu memiliki nilai balik, yep semuanya. Untuk fungsi yang tidak didefinisikan nilai baliknya maka nilai baliknya adalah tuple kosong ",(0,i.kt)("inlineCode",{parentName:"p"},"()"),". Sebagai contoh fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"greet_custom_message")," yang sudah ditulis, coba saja tampung nilai baliknya dan print. Outputnya adalah ",(0,i.kt)("em",{parentName:"p"},"empty tuple")," atau tuple kosong ",(0,i.kt)("inlineCode",{parentName:"p"},"()"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let a = greet_custom_message("Damian", "welcome to the castle!");\nprintln!("result: {:?}", a);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(3438).Z,width:"628",height:"237"})),(0,i.kt)("h2",{id:"a178-pembahasan-lanjutan"},"A.17.8. Pembahasan lanjutan"),(0,i.kt)("p",null,"Ada beberapa topik lainnya yang relevan dengan function, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Associated function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/associated-function"},"Associated Function")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Method"),", pembahasannya juga di chapter terpisah, yaitu ",(0,i.kt)("a",{parentName:"li",href:"/basic/method"},"Method"))),(0,i.kt)("p",null,"Selain itu nantinya juga ada pembahasan mengenai topik yang cukup advance yang berhubungan dengan fungsi, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Unsafe function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"#/wip/safe-unsafe"},"Safe & Unsafe")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Async function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"#/wip/async"},"Async")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Constant function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"#/wip/constant-evaluation"},"Constant Evaluation")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Trait \u279c Function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/trait-function"},"Trait \u279c Function"))),(0,i.kt)("p",null,"Untuk sekarang silakan lanjut ke pembahasan chapter berikutnya terlebih dahulu, jangan langsung loncat ke chapter di atas."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/function"},"github.com/novalagung/dasarpemrogramanrust-example/../function")),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang diverging function (",(0,i.kt)("inlineCode",{parentName:"li"},"() -> !"),")")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-03-how-functions-work.html"},"https://doc.rust-lang.org/book/ch03-03-how-functions-work.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.fn.html"},"https://doc.rust-lang.org/std/keyword.fn.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.return.html"},"https://doc.rust-lang.org/std/keyword.return.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.static.html"},"https://doc.rust-lang.org/std/keyword.static.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/fn.html"},"https://doc.rust-lang.org/rust-by-example/fn.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"https://rust-lang.github.io/api-guidelines/naming.html"))))}k.isMDXComponent=!0},2114:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},4002:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},9031:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},3307:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},6415:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},3438:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/function-6-2ed9468eff22575ec04867323e17a7b2.png"}}]); \ No newline at end of file diff --git a/assets/js/c526a898.e25aff7f.js b/assets/js/c526a898.e25aff7f.js deleted file mode 100644 index c6eabc40..00000000 --- a/assets/js/c526a898.e25aff7f.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[9157],{3905:(a,e,n)=>{n.d(e,{Zo:()=>d,kt:()=>o});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function r(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var u=t.createContext({}),m=function(a){var e=t.useContext(u),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},d=function(a){var e=m(a.components);return t.createElement(u.Provider,{value:e},a.children)},k={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},s=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,l=a.originalType,u=a.parentName,d=p(a,["components","mdxType","originalType","parentName"]),s=m(n),o=i,g=s["".concat(u,".").concat(o)]||s[o]||k[o]||l;return n?t.createElement(g,r(r({ref:e},d),{},{components:n})):t.createElement(g,r({ref:e},d))}));function o(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var l=n.length,r=new Array(l);r[0]=s;var p={};for(var u in e)hasOwnProperty.call(e,u)&&(p[u]=e[u]);p.originalType=a,p.mdxType="string"==typeof a?a:i,r[1]=p;for(var m=2;m{n.r(e),n.d(e,{assets:()=>u,contentTitle:()=>r,default:()=>k,frontMatter:()=>l,metadata:()=>p,toc:()=>m});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:17,title:"A.17. Function",sidebar_label:"A.17. Function"},r=void 0,p={unversionedId:"basic/function",id:"basic/function",title:"A.17. Function",description:"Pada chapter ini kita akan belajar tentang function atau fungsi.",source:"@site/docs/basic/function.md",sourceDirName:"basic",slug:"/basic/function",permalink:"/basic/function",draft:!1,tags:[],version:"current",sidebarPosition:17,frontMatter:{sidebar_position:17,title:"A.17. Function",sidebar_label:"A.17. Function"},sidebar:"tutorialSidebar",previous:{title:"A.16. Vector",permalink:"/basic/vector"},next:{title:"A.18. Module System \u279c Path & Item",permalink:"/basic/path-item"}},u={},m=[{value:"A.17.1. Keyword fn",id:"a171-keyword-fn",level:2},{value:"A.17.2. Naming convention fungsi",id:"a172-naming-convention-fungsi",level:2},{value:"A.17.3. Parameter fungsi dan argumen fungsi",id:"a173-parameter-fungsi-dan-argumen-fungsi",level:2},{value:"A.17.4. Nilai balik fungsi (return value)",id:"a174-nilai-balik-fungsi-return-value",level:2},{value:"\u25c9 Keyword return untuk penentuan nilai balik",id:"-keyword-return-untuk-penentuan-nilai-balik",level:3},{value:"\u25c9 Return value tanpa keyword return",id:"-return-value-tanpa-keyword-return",level:3},{value:"\u25c9 Statemen sebagai return value tanpa keyword return",id:"-statemen-sebagai-return-value-tanpa-keyword-return",level:3},{value:"\u25c9 Macro format untuk membuat formatted string",id:"-macro-format-untuk-membuat-formatted-string",level:3},{value:"A.17.5. Conditional return value",id:"a175-conditional-return-value",level:2},{value:"A.17.6. Nilai balik fungsi bertipe string literal &str",id:"a176-nilai-balik-fungsi-bertipe-string-literal-str",level:2},{value:"A.17.7. Default return value",id:"a177-default-return-value",level:2},{value:"A.17.8. Pembahasan lanjutan",id:"a178-pembahasan-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:m};function k(a){let{components:e,...l}=a;return(0,i.kt)("wrapper",(0,t.Z)({},d,l,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tentang ",(0,i.kt)("em",{parentName:"p"},"function")," atau fungsi."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Definisi fungsi dalam programming secara terminologi adalah sebuah modul atau sub-program kecil yang digunakan untuk mengeksekusi sebuah perintah, dan bisa di-",(0,i.kt)("em",{parentName:"p"},"reuse")," dalam penggunaannya.")),(0,i.kt)("h2",{id:"a171-keyword-fn"},"A.17.1. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"fn")),(0,i.kt)("p",null,"Fungsi di Rust dibuat menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"fn"),". Salah satu contoh penerapan keyword ini dalam pembuatan fungsi sudah kita praktikkan berulang kali pada definisi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", yang merupakan fungsi utama atau entrypoint sebuah program."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n\n}\n")),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," ini spesial, ketika program dijalankan maka ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," otomatis dipanggil atau dieksekusi."),(0,i.kt)("p",null,"Pembuatan fungsi lainnya-pun juga didefinisikan dengan notasi penulisan yang sama. Pada contoh berikut kita akan buat fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"greet")," yang tugasnya adalah menampilkan message ke layar output."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn greet() {\n println!("hello world");\n}\n')),(0,i.kt)("p",null,"Panggil fungsi tersebut di ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", lalu run program."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n println!("hello rust");\n greet();\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(2114).Z,width:"447",height:"86"})),(0,i.kt)("p",null,"Bisa dilihat ada 2 message muncul, yang pertama adalah pesan ",(0,i.kt)("inlineCode",{parentName:"p"},"hello rust")," yang muncul hasil eksekusi statement ",(0,i.kt)("inlineCode",{parentName:"p"},'println!("hello rust")'),", lalu diikuti ",(0,i.kt)("inlineCode",{parentName:"p"},"hello world")," yang statement print untuk menampilkan message tersebut ada dalam fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"greet"),"."),(0,i.kt)("p",null,"O iya, definisi fungsi tidak mengenal urutan ya. Sebagai contoh fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"greet")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," di atas, bebas mana yang ditulis lebih dulu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn greet() {\n // ..\n}\n\nfn main() {\n // ..\n}\n")),(0,i.kt)("p",null,"... atau ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n // ..\n}\n\nfn greet() {\n // ..\n}\n")),(0,i.kt)("h2",{id:"a172-naming-convention-fungsi"},"A.17.2. ",(0,i.kt)("em",{parentName:"h2"},"Naming convention")," fungsi"),(0,i.kt)("p",null,"Sesuai anjuran di ",(0,i.kt)("a",{parentName:"p",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"halaman dokumentasi Rust"),", aturan penulisan nama fungsi di Rust adalah menggunakan ",(0,i.kt)("strong",{parentName:"p"},"snake case"),". Nama fungsi dituliskan dalam huruf kecil dengan separator tanda garis bawah atau underscore (",(0,i.kt)("inlineCode",{parentName:"p"},"_"),"). Contohnya seperti: ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"greet_message"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"get_report_Status"),", dll."),(0,i.kt)("h2",{id:"a173-parameter-fungsi-dan-argumen-fungsi"},"A.17.3. Parameter fungsi dan argumen fungsi"),(0,i.kt)("p",null,"Fungsi bisa didefinisikan dengan disertai parameter, dengan itu bisa menyisipkan data saat pemanggilan fungsi."),(0,i.kt)("p",null,"Parameter dituliskan di statement pendefinisian fungsi dengan notasi seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn func_name(param_a: i32) {\n // ...\n}\n\nfn func_name(param_a: i32, param_b: bool) {\n // ...\n}\n\nfn func_name(param_a: i32, param_b: bool, param_c: &str) {\n // ...\n}\n\n// ...\n")),(0,i.kt)("p",null,"Kembali ke praktik, pada bagian ini kita akan buat sebuah fungsi bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"greet_custom_message"),". Fungsi ini tugasnya masih sama seperti seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"greet")," hanya saja pesan yang ditampilkan adalah sesuai dengan value parameter fungsi."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn greet_custom_message(name: &str, message: &str) {\n println!("hi {name}, {message}");\n}\n')),(0,i.kt)("p",null,"Ada dua parameter yang didefinisikan:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"name"),", tipe datanya string literal"),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"message"),", tipe datanya juga string literal")),(0,i.kt)("p",null,"Kedua nilai parameter tersebut di-",(0,i.kt)("em",{parentName:"p"},"combine")," lalu di-print menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),"."),(0,i.kt)("p",null,"Ok, sekarang panggil fungsi tersebut dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),", sisipkan argumen pertama ",(0,i.kt)("inlineCode",{parentName:"p"},"Damian")," yang ditampung parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"name"),", dan argumen ke-2 ",(0,i.kt)("inlineCode",{parentName:"p"},"welcome to the castle!")," untuk parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"message")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n greet_custom_message("Damian", "welcome to the castle!");\n}\n')),(0,i.kt)("p",null,"Jalankan program. Output bisa dilihat di gambar berikut."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(4002).Z,width:"450",height:"71"})),(0,i.kt)("p",null,"O iya, parameter bisa memiliki tipe data apa saja selama di-",(0,i.kt)("em",{parentName:"p"},"support")," oleh Rust, jadi tidak hanya string literal ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Secara terminologi, parameter adalah variabel yang didefinisikan di fungsi, sedangkan argumen adalah nilai yang disisipkan pada parameter saat pemanggilan fungsi."),(0,i.kt)("p",{parentName:"blockquote"},"Seiring berjalannya waktu, terjadi ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/book/ch03-03-how-functions-work.html#parameters"},"sedikit perubahan")," mengenai makna untuk istilah ",(0,i.kt)("inlineCode",{parentName:"p"},"parameter")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"argumen"),". Sekarang kedua istilah tersebut lebih sering dimaknai sama.")),(0,i.kt)("h2",{id:"a174-nilai-balik-fungsi-return-value"},"A.17.4. Nilai balik fungsi (",(0,i.kt)("em",{parentName:"h2"},"return value"),")"),(0,i.kt)("p",null,"Sebuah fungsi bisa di-desain memiliki nilai balik. Caranya dengan menambahkan tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"->")," diikuti tipe data dari nilai balik pada pendefinisian fungsi."),(0,i.kt)("p",null,"Pada contoh berikut, sebuah fungsi dibuat dengan nama ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume1"),", memiliki 3 buah parameter bertipe numerik (",(0,i.kt)("inlineCode",{parentName:"p"},"width"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"height"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"length"),"), dan nilai balik bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),". Tugas dari fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume1")," adalah melakukan operasi matematika perhitungan volume terhadap data yang didapat dari parameter, yang kemudian hasil kalkulasinya dijadikan nilai balik."),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," (diikuti dengan data) digunakan untuk menentukan nilai balik suatu fungsi. Pada contoh berikut, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"volume")," adalah nilai balik fungsi."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn calculate_box_volume1(width: i32, height: i32, length: i32) -> i32 {\n let volume = width * height * length;\n return volume;\n}\n")),(0,i.kt)("p",null,"Selanjutnya siapkan beberapa variabel yang nantinya digunakan saat pemanggilan ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume1"),". Lalu panggil fungsi tersebut kemudian tampilkan ",(0,i.kt)("em",{parentName:"p"},"return value"),"-nya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let width = 5;\nlet height = 8;\nlet length = 12;\n\nlet res1 = calculate_box_volume1(width, height, length);\nprintln!("result: {res1}");\n')),(0,i.kt)("p",null,"Jalankan program, hasilnya adalah sesuai harapan. Volume dari persegi panjang dengan data ",(0,i.kt)("inlineCode",{parentName:"p"},"5 x 8 x 12")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"480"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(9031).Z,width:"451",height:"64"})),(0,i.kt)("h3",{id:"-keyword-return-untuk-penentuan-nilai-balik"},"\u25c9 Keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"return")," untuk penentuan nilai balik"),(0,i.kt)("p",null,"Salah satu hal yang unik yang ada di Rust adalah beberapa variasi cara penulisan nilai balik fungsi. Cara pertama adalah menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," seperti yang sudah dipraktikkan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn calculate_box_volume1(width: i32, height: i32, length: i32) -> i32 {\n let volume = width * height * length;\n return volume;\n}\n")),(0,i.kt)("p",null,"Untuk cara ini penulis rasa cukup jelas."),(0,i.kt)("h3",{id:"-return-value-tanpa-keyword-return"},"\u25c9 ",(0,i.kt)("em",{parentName:"h3"},"Return value")," tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"return")),(0,i.kt)("p",null,"Cara ke-2 kita langsung praktikkan menggunakan kode berikut, silakan tulis lalu jalankan program:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_box_volume2(width: i32, height: i32, length: i32) -> i32 {\n let volume = width * height * length;\n volume\n}\n\nfn main() {\n let name2 = "Damian";\n let res2 = calculate_box_volume2(width, height, length);\n println!("hi {name2}, the box volume is {res2}");\n}\n')),(0,i.kt)("p",null,"Hasilnya kurang lebih sama, hanya di-bagian outputnya beda karena pada contoh ini pesan yang ditampilkan diubah."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(3307).Z,width:"446",height:"67"})),(0,i.kt)("p",null,"Sekarang kita fokus ke kode yang sudah ditulis. Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume2")," kurang lebih struktur dan isinya sama seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume1"),", bedanya hanya pada dua hal:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"calculate_box_volume2")," nilai baliknya dituliskan tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"li"},"return")),(0,i.kt)("li",{parentName:"ul"},"Fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"calculate_box_volume2")," pada statement terakhir tidak diakhir tanda akhir statement ",(0,i.kt)("inlineCode",{parentName:"li"},";"))),(0,i.kt)("p",null,"Kedua poin tersebut merupakan syarat untuk penerapan ",(0,i.kt)("em",{parentName:"p"},"return value")," tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return"),". Caranya kurang lebih adalah dengan menuliskan data nilai balik dengan tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," dan di akhir statement tidak ditambahi tanda semicolon ",(0,i.kt)("inlineCode",{parentName:"p"},";"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Statement terakhir sebuah blok kode fungsi yang ditulis tanpa semicolon ",(0,i.kt)("inlineCode",{parentName:"p"},";")," disebut sebagai ",(0,i.kt)("inlineCode",{parentName:"p"},"tail")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"body tail"),".")),(0,i.kt)("h3",{id:"-statemen-sebagai-return-value-tanpa-keyword-return"},"\u25c9 Statemen sebagai ",(0,i.kt)("em",{parentName:"h3"},"return value")," tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"return")),(0,i.kt)("p",null,"Cara selanjutnya yang akan kita bahas ini sebenarnya sama seperti sebelumnya, hanya saja bedanya di sini tidak menggunakan nama variabel sebagai nilai balik, melainkan langsung statement, yang hasil eksekusi statement tersebut dijadikan nilai balik fungsi."),(0,i.kt)("p",null,"Langsung saja kita praktikkan. Siapkan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_box_volume3")," berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_box_volume3(width: i32, height: i32, length: i32) -> i32 {\n width * height * length\n}\n\nfn greet_custom_message(name: &str, message: &str) {\n println!("hi {name}, {message}");\n}\n')),(0,i.kt)("p",null,"Panggil fungsi tersebut, tampung nilai baliknya ke variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"res3"),". Lalu siapkan ",(0,i.kt)("inlineCode",{parentName:"p"},"message3")," yang isinya adalah sebuah pesan string literal yang dibuat menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"format"),". Terakhir tampilkan pesan tersebut menggunakan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"greet_custom_message"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let res3 = calculate_box_volume3(width, height, length);\n let message3 = format!("the box volume is {}", res3);\n\n greet_custom_message("Damian", message3.as_str());\n}\n')),(0,i.kt)("p",null,"Bisa dilihat hasilnya sama seperti pada program sebelumnya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(3307).Z,width:"446",height:"67"})),(0,i.kt)("h3",{id:"-macro-format-untuk-membuat-formatted-string"},"\u25c9 Macro ",(0,i.kt)("inlineCode",{parentName:"h3"},"format")," untuk membuat ",(0,i.kt)("em",{parentName:"h3"},"formatted")," string"),(0,i.kt)("p",null,"Bagian ini adalah penjelasan tambahan dari kode program yang baru saja dipraktikkan."),(0,i.kt)("p",null,"Macro ",(0,i.kt)("inlineCode",{parentName:"p"},"format")," digunakan untuk ",(0,i.kt)("em",{parentName:"p"},"formatting")," sebuah string. ",(0,i.kt)("em",{parentName:"p"},"Formatting")," di sini maksudnya adalah pembuatan string dengan menggunakan teknik penyusunan isi string dalam format tertentu, mirip seperti yang dilakukan menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"print")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),", bedanya hanya pada macro ",(0,i.kt)("inlineCode",{parentName:"p"},"format")," hasilnya bukan sebuah output ke console output, melainkan sebagai nilai balik."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// 2 bari statements berikut ...\nlet message3 = format!("the box volume is {}", res3);\nprintln!("{}", message3.as_str());\n\n// ... adalah sama dengan statement berikut ...\nprintln!("the box volume is {}", res3);\n')),(0,i.kt)("p",null,"Satu hal yang penting diketahui dari macro ",(0,i.kt)("inlineCode",{parentName:"p"},"format"),". Nilai balik macro ini bukanlah string literal ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),", tetapi tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),". ",(0,i.kt)("strong",{parentName:"p"},"Dua tipe tersebut adalah berbeda"),"."),(0,i.kt)("p",null,"Pada contoh di atas, method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_str")," milik tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," digunakan untuk mengambil data string literal-nya. Method ",(0,i.kt)("inlineCode",{parentName:"p"},"as_str")," ini menghasilkan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," dari sebuah ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("h2",{id:"a175-conditional-return-value"},"A.17.5. ",(0,i.kt)("em",{parentName:"h2"},"Conditional return value")),(0,i.kt)("p",null,"Kapan harus menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," dalam penentuan nilai balik dan kapan tidak? Jawabannya mungkin adalah preferensi, tapi di luar itu ada juga case dimana keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," harus digunakan, yaitu pada fungsi yang memiliki nilai balik terkondisi. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn get_score_message(score: f32) -> &\'static str {\n if score == 100.0 {\n return "you got a perfect score!"\n }\n \n if score > 76.0 {\n return "congrats, you passed the exam!"\n }\n\n "your score is below the passing grade"\n}\n\nfn main() {\n println!("{}", get_score_message(100.0));\n println!("{}", get_score_message(98.2));\n println!("{}", get_score_message(33.12));\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(6415).Z,width:"446",height:"100"})),(0,i.kt)("p",null,"Teknik penentuan nilai balik tanpa keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," hanya bisa dipergunakan di akhir blok kode, contohnya pada fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"get_score_message")," statement terakhir blok kode adalah string literal ",(0,i.kt)("inlineCode",{parentName:"p"},'"your score is below the passing grade"'),"."),(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"Jika pada selain akhir blok ada kondisi dimana nilai balik harus ditentukan, maka wajib menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"strong"},"return")),". Bisa dilihat pada fungsi yang sudah ditulis, ",(0,i.kt)("inlineCode",{parentName:"p"},"return")," statement dalam blok kode seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," dipergunakan."),(0,i.kt)("h2",{id:"a176-nilai-balik-fungsi-bertipe-string-literal-str"},"A.17.6. Nilai balik fungsi bertipe string literal ",(0,i.kt)("inlineCode",{parentName:"h2"},"&str")),(0,i.kt)("p",null,"Khusus untuk beberapa jenis tipe data, seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),", jika digunakan sebagai tipe data ",(0,i.kt)("em",{parentName:"p"},"return value")," fungsi harus ditambahi keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"static")," dengan penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"&'static str"),". Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn get_score_message(score: f32) -> &\'static str {\n return "you got a perfect score!"\n}\n')),(0,i.kt)("p",null,"Agar tidak makin bingung, pembahasannya dipisah pada chapter lain, yaitu ",(0,i.kt)("a",{parentName:"p",href:"/basic/static"},"Static")," dan ",(0,i.kt)("a",{parentName:"p",href:"/basic/lifetime"},"Lifetime"),"."),(0,i.kt)("h2",{id:"a177-default-return-value"},"A.17.7. ",(0,i.kt)("em",{parentName:"h2"},"Default return value")),(0,i.kt)("p",null,"Tambahan info saja, bahwa di Rust, semua fungsi itu memiliki nilai balik, yep semuanya. Untuk fungsi yang tidak didefinisikan nilai baliknya maka nilai baliknya adalah tuple kosong ",(0,i.kt)("inlineCode",{parentName:"p"},"()"),". Sebagai contoh fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"greet_custom_message")," yang sudah ditulis, coba saja tampung nilai baliknya dan print. Outputnya adalah ",(0,i.kt)("em",{parentName:"p"},"empty tuple")," atau tuple kosong ",(0,i.kt)("inlineCode",{parentName:"p"},"()"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let a = greet_custom_message("Damian", "welcome to the castle!");\nprintln!("result: {:?}", a);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"function",src:n(3438).Z,width:"628",height:"237"})),(0,i.kt)("h2",{id:"a178-pembahasan-lanjutan"},"A.17.8. Pembahasan lanjutan"),(0,i.kt)("p",null,"Ada beberapa topik lainnya yang relevan dengan function, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Associated function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/associated-function"},"Associated Function")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Method"),", pembahasannya juga di chapter terpisah, yaitu ",(0,i.kt)("a",{parentName:"li",href:"/basic/method"},"Method"))),(0,i.kt)("p",null,"Selain itu nantinya juga ada pembahasan mengenai topik yang cukup advance yang berhubungan dengan fungsi, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Unsafe function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"#/wip/safe-unsafe"},"Safe & Unsafe")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Async function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"#/wip/async"},"Async")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Constant function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"#/wip/constant-evaluation"},"Constant Evaluation")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("strong",{parentName:"li"},"Trait \u279c Function"),", pembahasannya ada di chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/trait-function"},"Trait \u279c Function"))),(0,i.kt)("p",null,"Untuk sekarang silakan lanjut ke pembahasan chapter berikutnya terlebih dahulu, jangan langsung loncat ke chapter di atas."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/function"},"github.com/novalagung/dasarpemrogramanrust-example/../function")),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang diverging function (",(0,i.kt)("inlineCode",{parentName:"li"},"() -> !"),")")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch03-03-how-functions-work.html"},"https://doc.rust-lang.org/book/ch03-03-how-functions-work.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.fn.html"},"https://doc.rust-lang.org/std/keyword.fn.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.return.html"},"https://doc.rust-lang.org/std/keyword.return.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.static.html"},"https://doc.rust-lang.org/std/keyword.static.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/fn.html"},"https://doc.rust-lang.org/rust-by-example/fn.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://rust-lang.github.io/api-guidelines/naming.html"},"https://rust-lang.github.io/api-guidelines/naming.html"))))}k.isMDXComponent=!0},2114:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},4002:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},9031:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},3307:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},6415:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},3438:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/function-6-2ed9468eff22575ec04867323e17a7b2.png"}}]); \ No newline at end of file diff --git a/assets/js/c76cc2e3.978efe66.js b/assets/js/c76cc2e3.978efe66.js new file mode 100644 index 00000000..764a965f --- /dev/null +++ b/assets/js/c76cc2e3.978efe66.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4729],{3905:(e,a,n)=>{n.d(a,{Zo:()=>o,kt:()=>k});var t=n(7294);function i(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function l(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function d(e){for(var a=1;a=0||(i[n]=e[n]);return i}(e,a);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var r=t.createContext({}),c=function(e){var a=t.useContext(r),n=a;return e&&(n="function"==typeof e?e(a):d(d({},a),e)),n},o=function(e){var a=c(e.components);return t.createElement(r.Provider,{value:a},e.children)},m={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},u=t.forwardRef((function(e,a){var n=e.components,i=e.mdxType,l=e.originalType,r=e.parentName,o=p(e,["components","mdxType","originalType","parentName"]),u=c(n),k=i,A=u["".concat(r,".").concat(k)]||u[k]||m[k]||l;return n?t.createElement(A,d(d({ref:a},o),{},{components:n})):t.createElement(A,d({ref:a},o))}));function k(e,a){var n=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var l=n.length,d=new Array(l);d[0]=u;var p={};for(var r in a)hasOwnProperty.call(a,r)&&(p[r]=a[r]);p.originalType=e,p.mdxType="string"==typeof e?e:i,d[1]=p;for(var c=2;c{n.r(a),n.d(a,{assets:()=>r,contentTitle:()=>d,default:()=>m,frontMatter:()=>l,metadata:()=>p,toc:()=>c});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:16,title:"A.16. Tipe Data \u279c Vector",sidebar_label:"A.16. Vector"},d=void 0,p={unversionedId:"basic/vector",id:"basic/vector",title:"A.16. Tipe Data \u279c Vector",description:"Pada chapter ini kita akan belajar tetang tipe data Vector. Vector adalah tipe data seperti array tapi dinamis. Dinamis di sini artinya bisa bertambah dan berkurang kapanpun sesuai kebutuhan.",source:"@site/docs/basic/vector.md",sourceDirName:"basic",slug:"/basic/vector",permalink:"/basic/vector",draft:!1,tags:[],version:"current",sidebarPosition:16,frontMatter:{sidebar_position:16,title:"A.16. Tipe Data \u279c Vector",sidebar_label:"A.16. Vector"},sidebar:"tutorialSidebar",previous:{title:"A.15. Tuple",permalink:"/basic/tuple"},next:{title:"A.17. Function",permalink:"/basic/function"}},r={},c=[{value:"A.16.1. Tipe data Vec<T>",id:"a161-tipe-data-vect",level:2},{value:"\u25c9 Deklarasi vektor, size, dan capacity",id:"-deklarasi-vektor-size-dan-capacity",level:3},{value:"\u25c9 Method pop \u279c menghapus elemen terakhir",id:"-method-pop--menghapus-elemen-terakhir",level:3},{value:"\u25c9 Method remove \u279c menghapus elemen index ke I",id:"-method-remove--menghapus-elemen-index-ke-i",level:3},{value:"\u25c9 Method push \u279c menambahkan elemen baru",id:"-method-push--menambahkan-elemen-baru",level:3},{value:"\u25c9 Realokasi vector",id:"-realokasi-vector",level:3},{value:"\u25c9 Mengubah value sebuah elemen menggunakan notasi [i]",id:"-mengubah-value-sebuah-elemen-menggunakan-notasi-i",level:3},{value:"\u25c9 Method is_empty \u279c mengecek apakah vector kosong",id:"-method-is_empty--mengecek-apakah-vector-kosong",level:3},{value:"\u25c9 Method clear \u279c mengosongkan isi vector",id:"-method-clear--mengosongkan-isi-vector",level:3},{value:"\u25c9 Method append \u279c concatenation/penggabungan vector",id:"-method-append--concatenationpenggabungan-vector",level:3},{value:"\u25c9 Method sort \u279c untuk mengurutkan vector",id:"-method-sort--untuk-mengurutkan-vector",level:3},{value:"A.16.2. Macam deklarasi vektor",id:"a162-macam-deklarasi-vektor",level:2},{value:"A.16.3. Iterasi data vector",id:"a163-iterasi-data-vector",level:2},{value:"A.16.4. Ownership tipe data vector",id:"a164-ownership-tipe-data-vector",level:2},{value:"A.16.5. Vector slice",id:"a165-vector-slice",level:2},{value:"A.16.6. Tipe data VecDeque<T>",id:"a166-tipe-data-vecdequet",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:c};function m(e){let{components:a,...l}=e;return(0,i.kt)("wrapper",(0,t.Z)({},o,l,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tetang tipe data ",(0,i.kt)("em",{parentName:"p"},"Vector"),". Vector adalah tipe data seperti array tapi dinamis. Dinamis di sini artinya bisa bertambah dan berkurang kapanpun sesuai kebutuhan."),(0,i.kt)("p",null,"Vector sangat mirip seperti array, yang karakteristiknya adalah tipe data elemen wajib sama, punya informasi size, elemen-nya bisa diakses atau diubah. Salah satu perbedaan vector dibanding array adalah jumlah elemen pada vector bisa bertambah lebih dari kapasitas yang sudah ditentukan."),(0,i.kt)("p",null,"Vector memiliki 3 buah atribut yg penting untuk diketahui:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"pointer ke data asli"),(0,i.kt)("li",{parentName:"ul"},"lebar atau size"),(0,i.kt)("li",{parentName:"ul"},"kapasitas (representasi dari seberapa banyak memori di-booking untuk data vector tersebut)")),(0,i.kt)("p",null,"Vector bisa bertambah jumlah isinya selama size di bawah kapasitas yang sudah dialokasikan. Jika suatu ketika vector isinya bertambah lebih banyak dari jumlah alokasi maksimal kapasitas, maka vector akan dialokasikan ulang dengan kapasitas yang lebih besar."),(0,i.kt)("h2",{id:"a161-tipe-data-vect"},"A.16.1. Tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Vec")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," adalah tipe data yang merepresentasikan vector, yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah generics. Vector datanya dialokasikan di heap memory."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai generic dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai heap dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/basic-memory-management"},"Basic Memory Management")))),(0,i.kt)("p",null,"Langsung saja kita praktikkan."),(0,i.kt)("h3",{id:"-deklarasi-vektor-size-dan-capacity"},"\u25c9 Deklarasi vektor, size, dan capacity"),(0,i.kt)("p",null,"Ada beberapa cara yang bisa dipakai untuk membuat data vector. Salah satunya adalah menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec"),", penulisannya seperti pembuatan array hanya saja perlu ditambahi prefix ",(0,i.kt)("inlineCode",{parentName:"p"},"vec!"),". Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut data_one = vec!["batman", "superman", "lobo"];\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(7662).Z,width:"453",height:"110"})),(0,i.kt)("p",null,"Pada contoh di atas variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," adalah sebuah vector yang isinya 3 elemen, yaitu literal string ",(0,i.kt)("inlineCode",{parentName:"p"},"batman"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"superman"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"lobo"),". Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," dideklarasikan menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec")," kemudiaan diikuti notasi penulisan yang mirip seperti array."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai macro dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/macro"},"Macro"))),(0,i.kt)("p",null,"Pembuatan vector menggunakan teknik ini hasilnya adalah data vector yang ",(0,i.kt)("inlineCode",{parentName:"p"},"size")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"capacity")," nya adalah sesuai dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value"),", yang pada konteks ini adalah 3."),(0,i.kt)("p",null,"Umumnya vector dideklarasikan dengan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," agar bisa diubah nilainya, karena tujuan disediakannya tipe data ini adalah untuk bisa mengakomodir tipe data array tetapi dinamis (bisa manipulasi datanya)."),(0,i.kt)("p",null,"Perihal method ",(0,i.kt)("inlineCode",{parentName:"p"},"len")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"capacity"),":"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"len")," digunakan untuk mencari tahu size atau jumlah elemen yang ada pada sebuah vector"),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"capacity")," digunakan untuk mencari tahu kapasitas atau jumlah maksimum elemen yang ada pada sebuah vector")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"O iya, kode program di atas akan menghasilkan warning saat di-run. Hal ini dikarenakan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," yang didefinisikan mutable belum diubah nilainya. Hiraukan saja, karena selanjutnya kita akan manipulasi data variabel tersebut.")),(0,i.kt)("h3",{id:"-method-pop--menghapus-elemen-terakhir"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"pop")," \u279c menghapus elemen terakhir"),(0,i.kt)("p",null,"Oke, selanjutnya mari kita oprek variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," yang sudah ditulis."),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," memiliki method ",(0,i.kt)("inlineCode",{parentName:"p"},"pop")," yang fungsinya menghapus data elemen terakhir. Mari gunakan method ini pada ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one.pop();\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(7554).Z,width:"450",height:"116"})),(0,i.kt)("p",null,"Bisa dilihat pada contoh di atas, setelah menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"pop"),", isi ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," menjadi 2 elemen saja. Elemen terakhir (yaitu string ",(0,i.kt)("inlineCode",{parentName:"p"},"lobo"),") dihapus. Efeknya, atribut size menjadi 2, tapi kapasitas tetap 3."),(0,i.kt)("h3",{id:"-method-remove--menghapus-elemen-index-ke-i"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"remove")," \u279c menghapus elemen index ke ",(0,i.kt)("inlineCode",{parentName:"h3"},"I")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"remove")," adalah salah satu method lainnya yang ada pada tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),". Kegunaan dari method ",(0,i.kt)("inlineCode",{parentName:"p"},"remove")," adalah untuk menghapus elemen pada indeks tertentu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one.remove(1);\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(41).Z,width:"450",height:"134"})),(0,i.kt)("p",null,"Dicontohkan di atas elemen indeks ke-",(0,i.kt)("inlineCode",{parentName:"p"},"1")," dihapus dengan cara memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},"remove")," lalu menyisipkan indeks elemen sebagai parameter pemanggilan method. Hasilnya elemen ",(0,i.kt)("inlineCode",{parentName:"p"},"superman")," dihapus dari vector."),(0,i.kt)("h3",{id:"-method-push--menambahkan-elemen-baru"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"push")," \u279c menambahkan elemen baru"),(0,i.kt)("p",null,"Sekarang isi dari vector ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," tinggal 1 elemen, mari kita tambahkan 3 elemen baru dengan memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"push"),"."),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"push")," fungsinya adalah untuk menambahkan elemen baru pada vector."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one.push("constantine");\ndata_one.push("trigon");\ndata_one.push("darkseid");\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(4480).Z,width:"452",height:"135"})),(0,i.kt)("p",null,"Bisa dilihat sekarang ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," isinya adalah 4 elemen dan atribut size-nya cocok, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"4"),". Namun ada yang aneh, kenapa kapasitas jadi ",(0,i.kt)("inlineCode",{parentName:"p"},"6")," padahal di awal kapasitas adalah 3."),(0,i.kt)("h3",{id:"-realokasi-vector"},"\u25c9 Realokasi vector"),(0,i.kt)("p",null,"Perubahan kapasitas atau realokasi vector terjadi ketika sebuah vector isinya bertambah lebih banyak dari jumlah alokasi maksimal kapasitas."),(0,i.kt)("p",null,"Lalu apa efeknya? secara high level bisa dibilang tidak ada, namun kalau dibahas lebih rinci, efeknya adalah di sisi alokasi space untuk menampung elemen. Proses realokasi menghasilkan vector yang baru dengan kapasitas lebih besar."),(0,i.kt)("h3",{id:"-mengubah-value-sebuah-elemen-menggunakan-notasi-i"},"\u25c9 Mengubah value sebuah elemen menggunakan notasi ",(0,i.kt)("inlineCode",{parentName:"h3"},"[i]")),(0,i.kt)("p",null,"Sama seperti array, vector juga bisa dimodifikasi nilai elemennya dengan menggunakan notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"[i]"),"."),(0,i.kt)("p",null,"Pada contoh berikut elemen indeks ke-",(0,i.kt)("inlineCode",{parentName:"p"},"2")," diubah nilainya dari ",(0,i.kt)("inlineCode",{parentName:"p"},"trigon")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"red hood"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one[2] = "red hood";\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(5216).Z,width:"449",height:"131"})),(0,i.kt)("h3",{id:"-method-is_empty--mengecek-apakah-vector-kosong"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_empty")," \u279c mengecek apakah vector kosong"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_empty")," digunakan untuk mengidentifikasi apakah sebuah vector isinya kosong atau tidak."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let is_vector_empty = data_one.is_empty();\nprintln!("result: {:?}", is_vector_empty);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(8859).Z,width:"447",height:"109"})),(0,i.kt)("h3",{id:"-method-clear--mengosongkan-isi-vector"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"clear")," \u279c mengosongkan isi vector"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"clear")," digunakan untuk mengosongkan sebuah vektor."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one.clear();\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(802).Z,width:"445",height:"94"})),(0,i.kt)("h3",{id:"-method-append--concatenationpenggabungan-vector"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"append")," \u279c concatenation/penggabungan vector"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"append")," digunakan untuk menggabungkan dua buah vector. Penggunaannya cukup mudah, panggil saja method nya lalu sisipkan ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," dari vector satunya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut result_one = vec![3, 1, 2];\n\nlet mut data_two = vec![7, 6, 8];\nresult_one.append(&mut data_two);\n\nprintln!("data: {:?}", result_one);\nprintln!("length: {}, capacity: {}", result_one.len(), result_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(2366).Z,width:"450",height:"125"})),(0,i.kt)("p",null,"Pada contoh di atas ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," adalah vector mutable dengan isi 3 elemen. Kemudian dideklarasikan ",(0,i.kt)("inlineCode",{parentName:"p"},"data_two")," yang isinya juga vector 3 elemen. Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"data_two")," dimasukkan kedalam vector ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," dengan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"append"),", dengan ini maka isi ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," adalah gabungan dari ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," yang lama dan ",(0,i.kt)("inlineCode",{parentName:"p"},"data_two"),"."),(0,i.kt)("p",null,"Proses append vector mengharuskan parameter method ",(0,i.kt)("inlineCode",{parentName:"p"},"append")," diisi dengan ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," dari vector yang ingin dimasukkan. Cara untuk mengambil ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," adalah dengan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut"),"."),(0,i.kt)("p",null,"Ok, selanjutnya tambahkan lagi isi ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," dengan vector lain."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'result_one.append(&mut vec![4, 5]);\n\nprintln!("data: {:?}", result_one);\nprintln!("length: {}, capacity: {}", result_one.len(), result_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(6241).Z,width:"447",height:"127"})),(0,i.kt)("p",null,"Proses penggabungan vector pada contoh ke-dua di atas sedikit berbeda. Method ",(0,i.kt)("inlineCode",{parentName:"p"},"append")," parameternya adalah langsung ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," dari literal vector. Ini merupakan salah satu cara yang bisa digunakan dalam penggabungan vector."),(0,i.kt)("h3",{id:"-method-sort--untuk-mengurutkan-vector"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"sort")," \u279c untuk mengurutkan vector"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"sort")," digunakan untuk mengurutkan elemen vector."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'println!("data: {:?}", result_one);\nresult_one.sort();\nprintln!("data: {:?}", result_one);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(3380).Z,width:"447",height:"127"})),(0,i.kt)("h2",{id:"a162-macam-deklarasi-vektor"},"A.16.2. Macam deklarasi vektor"),(0,i.kt)("p",null,"Ada beberapa cara deklarasi vector. Pada contoh berikut dua buah vector dideklarasikan menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec!"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut vector_4 = vec![1, 2, 3];\nlet mut vector_5: Vec = vec![1, 2, 3];\n")),(0,i.kt)("p",null,"Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"vector_4")," didefinisikan dengan cara yang sudah kita terapkan sebelumnya, yaitu menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec"),". Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"vector_5")," juga didefinisikan dengan cara yang sama, hanya saja pada vector ini tipe datanya ditentukan secara eksplisit yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),"."),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," Artinya adalah vector dengan tipe data elemen adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i64"),". Dengan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," bisa ditentukan tipe data elemen yang diinginkan."),(0,i.kt)("p",null,"Cara deklarasi vector selanjutnya adalah pembuatan vector dengan isi kosong. Deklarasi vector ini mewajibkan tipe data vector dituliskan secara eksplisit, dikarenakan tipe data tidak bisa diidentifikasi dari isinya (karena isinya kosong). Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut vector_7: Vec<&str> = vec![];\nlet mut vector_8: Vec<&str> = Vec::new();\n")),(0,i.kt)("p",null,"Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"vector_7")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"vector_8")," keduanya bertipe vector dengan isi atau elemen bertipe data literal string ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("p",null,"Deklarasi vector kosong bisa dilakukan dengan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec")," yang ditulis tanpa isi, atau bisa menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec::new()"),"."),(0,i.kt)("h2",{id:"a163-iterasi-data-vector"},"A.16.3. Iterasi data vector"),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," bisa digunakan untuk iterasi vector. Cara penerapannya seperti pada array atau slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let vec_eight = vec![1, 2, 3];\nfor e in vec_eight {\n print!("{e} ");\n}\n\n// 1 2 3\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let vec_nine = vec![1, 2, 3];\nfor i in 0..vec_nine.len() {\n print!("{} ", vec_nine[i]);\n}\n\n// 1 2 3\n')),(0,i.kt)("p",null,"Keyword perulangan lainnya juga bisa digunakan."),(0,i.kt)("h2",{id:"a164-ownership-tipe-data-vector"},"A.16.4. Ownership tipe data vector"),(0,i.kt)("p",null,"Salah satu atribut vector yang penting untuk diketahui adalah, pemilik data sebenarnya (atau owner). Agar lebih jelas, silakan coba terlebih dahulu kode berikut."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let vec_ten = vec![1, 2, 3];\nfor e in vec_ten {\n print!("{e} ");\n}\nfor i in 0..vec_ten.len() {\n print!("{} ", vec_ten[i]);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(2061).Z,width:"634",height:"626"})),(0,i.kt)("p",null,"Terlihat sekilas tidak ada kode yang bermasalah dari program di atas, tapi error, aneh."),(0,i.kt)("p",null,"Di Rust, ownership atau kepemilikan data adalah hal yang sangat penting. Saking pentingnya, beberapa orang menyebut Rust sebagai bahasa yang ",(0,i.kt)("em",{parentName:"p"},"value oriented"),"."),(0,i.kt)("p",null,"Dalam kasus kode program vector di atas, ketika keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," digunakan untuk mengiterasi vector ",(0,i.kt)("inlineCode",{parentName:"p"},"vec_ten"),", membuat pemilik data vektor berpindah ke variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"e"),". Hal ini efeknya adalah ketika kita berusaha mengakses variabel yang sama setelah perulangan selesai, maka yang muncul adalah error, karena value-nya sudah berpindah."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Perpindahan owner disebut dengan ",(0,i.kt)("em",{parentName:"p"},"move semantics"),". Lebih jelasnya nanti dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership"))),(0,i.kt)("p",null,"Solusi untuk antisipasi error ini adalah dengan cara meminjam value yang sebenarnya dari owner, untuk kemudian digunakan dalam perulangan. Caranya dengan menggunakan teknik ",(0,i.kt)("em",{parentName:"p"},"borrowing")," menggunakan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"reference")," yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"&"),". Data sebenarnya milik owner dipinjam untuk dipergunakan di perulangan."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai reference dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/pointer-references"},"Pointer & References")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai borrowing dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/borrowing"},"Borrowing")))),(0,i.kt)("p",null,"Silakan ubah kode yang sebelumnya ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'for e in vec_ten {\n print!("{e} ");\n}\n')),(0,i.kt)("p",null,"... menjadi seperti ini, kemudian run, maka error akan hilang."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'for e in &vec_ten {\n print!("{e} ");\n}\n')),(0,i.kt)("p",null,"Salah satu alternatif cara lainnya untuk antisipasi value berpindah tempat adalah dengan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," untuk mengkonversi vector menjadi iterator. Jadi yang di-iterasi bukan vector-nya, melainkan objek iterator yang dibuat dari vector tersebut."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'for e in vec_ten.iter() {\n print!("{e} ");\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai ",(0,i.kt)("em",{parentName:"li"},"ownership")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/ownership"},"Ownership")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai ",(0,i.kt)("em",{parentName:"li"},"borrowing")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/borrowing"},"Borrowing")))),(0,i.kt)("h2",{id:"a165-vector-slice"},"A.16.5. Vector slice"),(0,i.kt)("p",null,"Seperti array, slice juga bisa dibuat dari vector. Cara penerapannya juga sama persis. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let vec_population = vec![2, 1, 3];\nlet vec_sample = &vec_population[0..1];\nprintln!("{:?}", vec_sample); // [2]\n')),(0,i.kt)("p",null,"Semua operasi slice bisa diterapkan di vector."),(0,i.kt)("h2",{id:"a166-tipe-data-vecdequet"},"A.16.6. Tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"VecDeque")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque")," adalah sama seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," plus mendukung operasi menambah dan mengurangi elemen dari dua sisi secara efisien."),(0,i.kt)("p",null,"Pada tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),", ada method ",(0,i.kt)("inlineCode",{parentName:"p"},"pop")," yang fungsinya menghapus data elemen terakhir dan method ",(0,i.kt)("inlineCode",{parentName:"p"},"push")," untuk menambah elemen baru dari kanan. Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque")," memiliki bebebrapa method tambahan, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"method ",(0,i.kt)("inlineCode",{parentName:"li"},"pop_front")," untuk hapus data elemen pertama atau paling kiri (indeks ke-0)"),(0,i.kt)("li",{parentName:"ul"},"method ",(0,i.kt)("inlineCode",{parentName:"li"},"push_front")," untuk menambah data dari kiri (indeks ke-0)"),(0,i.kt)("li",{parentName:"ul"},"method ",(0,i.kt)("inlineCode",{parentName:"li"},"pop_back")," untuk hapus data elemen pertama atau paling kanan (indeks terakhir)"),(0,i.kt)("li",{parentName:"ul"},"method ",(0,i.kt)("inlineCode",{parentName:"li"},"push_back")," untuk menambah data dari kanan (indeks terakhir)")),(0,i.kt)("p",null,"Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use std::collections::VecDeque;\n\nlet mut vec_10 = VecDeque::from(vec!["a", "b", "c"]);\n\nvec_10.pop_front();\nvec_10.push_front("z");\nprintln!("data: {:?}", vec_10);\n\nvec_10.pop_back();\nvec_10.push_back("h");\nprintln!("data: {:?}", vec_10);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(4050).Z,width:"453",height:"129"})),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque")," tidak otomatis di-import. Kita perlu mengimport path di mana tipe data tersebut berada menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::collections::VecDeque;\n")),(0,i.kt)("p",null,"Cara membuat vector ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque")," bisa menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque::from")," dengan parameter diisi data vectornya, seperti pada kode program yang sudah ditulis."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/vector"},"github.com/novalagung/dasarpemrogramanrust-example/../vector")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/macro.vec.html"},"https://doc.rust-lang.org/std/macro.vec.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/vec/struct.Vec.html"},"https://doc.rust-lang.org/std/vec/struct.Vec.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/std/vec.html"},"https://doc.rust-lang.org/rust-by-example/std/vec.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/collections/struct.VecDeque.html"},"https://doc.rust-lang.org/std/collections/struct.VecDeque.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/36672845/in-rust-is-a-vector-an-iterator"},"https://stackoverflow.com/questions/36672845/in-rust-is-a-vector-an-iterator")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/28800121/what-do-i-have-to-do-to-solve-a-use-of-moved-value-error"},"https://stackoverflow.com/questions/28800121/what-do-i-have-to-do-to-solve-a-use-of-moved-value-error")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/43036279/what-does-it-mean-to-pass-in-a-vector-into-a-for-loop-versus-a-reference-to-a"},"https://stackoverflow.com/questions/43036279/what-does-it-mean-to-pass-in-a-vector-into-a-for-loop-versus-a-reference-to-a"))))}m.isMDXComponent=!0},7662:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},3380:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2061:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/vector-11-27b0884891ab00c87d90603997e1878c.png"},4050:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},7554:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},41:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},4480:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},5216:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},8859:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},802:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2366:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},6241:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/c76cc2e3.c1ae2c67.js b/assets/js/c76cc2e3.c1ae2c67.js deleted file mode 100644 index f4972cae..00000000 --- a/assets/js/c76cc2e3.c1ae2c67.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4729],{3905:(e,a,n)=>{n.d(a,{Zo:()=>o,kt:()=>k});var t=n(7294);function i(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function l(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function d(e){for(var a=1;a=0||(i[n]=e[n]);return i}(e,a);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var r=t.createContext({}),c=function(e){var a=t.useContext(r),n=a;return e&&(n="function"==typeof e?e(a):d(d({},a),e)),n},o=function(e){var a=c(e.components);return t.createElement(r.Provider,{value:a},e.children)},m={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},u=t.forwardRef((function(e,a){var n=e.components,i=e.mdxType,l=e.originalType,r=e.parentName,o=p(e,["components","mdxType","originalType","parentName"]),u=c(n),k=i,A=u["".concat(r,".").concat(k)]||u[k]||m[k]||l;return n?t.createElement(A,d(d({ref:a},o),{},{components:n})):t.createElement(A,d({ref:a},o))}));function k(e,a){var n=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var l=n.length,d=new Array(l);d[0]=u;var p={};for(var r in a)hasOwnProperty.call(a,r)&&(p[r]=a[r]);p.originalType=e,p.mdxType="string"==typeof e?e:i,d[1]=p;for(var c=2;c{n.r(a),n.d(a,{assets:()=>r,contentTitle:()=>d,default:()=>m,frontMatter:()=>l,metadata:()=>p,toc:()=>c});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:16,title:"A.16. Tipe Data \u279c Vector",sidebar_label:"A.16. Vector"},d=void 0,p={unversionedId:"basic/vector",id:"basic/vector",title:"A.16. Tipe Data \u279c Vector",description:"Pada chapter ini kita akan belajar tetang tipe data Vector. Vector adalah tipe data seperti array tapi dinamis. Dinamis di sini artinya bisa bertambah dan berkurang kapanpun sesuai kebutuhan.",source:"@site/docs/basic/vector.md",sourceDirName:"basic",slug:"/basic/vector",permalink:"/basic/vector",draft:!1,tags:[],version:"current",sidebarPosition:16,frontMatter:{sidebar_position:16,title:"A.16. Tipe Data \u279c Vector",sidebar_label:"A.16. Vector"},sidebar:"tutorialSidebar",previous:{title:"A.15. Tuple",permalink:"/basic/tuple"},next:{title:"A.17. Function",permalink:"/basic/function"}},r={},c=[{value:"A.16.1. Tipe data Vec<T>",id:"a161-tipe-data-vect",level:2},{value:"\u25c9 Deklarasi vektor, size, dan capacity",id:"-deklarasi-vektor-size-dan-capacity",level:3},{value:"\u25c9 Method pop \u279c menghapus elemen terakhir",id:"-method-pop--menghapus-elemen-terakhir",level:3},{value:"\u25c9 Method remove \u279c menghapus elemen index ke I",id:"-method-remove--menghapus-elemen-index-ke-i",level:3},{value:"\u25c9 Method push \u279c menambahkan elemen baru",id:"-method-push--menambahkan-elemen-baru",level:3},{value:"\u25c9 Realokasi vector",id:"-realokasi-vector",level:3},{value:"\u25c9 Mengubah value sebuah elemen menggunakan notasi [i]",id:"-mengubah-value-sebuah-elemen-menggunakan-notasi-i",level:3},{value:"\u25c9 Method is_empty \u279c mengecek apakah vector kosong",id:"-method-is_empty--mengecek-apakah-vector-kosong",level:3},{value:"\u25c9 Method clear \u279c mengosongkan isi vector",id:"-method-clear--mengosongkan-isi-vector",level:3},{value:"\u25c9 Method append \u279c concatenation/penggabungan vector",id:"-method-append--concatenationpenggabungan-vector",level:3},{value:"\u25c9 Method sort \u279c untuk mengurutkan vector",id:"-method-sort--untuk-mengurutkan-vector",level:3},{value:"A.16.2. Macam deklarasi vektor",id:"a162-macam-deklarasi-vektor",level:2},{value:"A.16.3. Iterasi data vector",id:"a163-iterasi-data-vector",level:2},{value:"A.16.4. Ownership tipe data vector",id:"a164-ownership-tipe-data-vector",level:2},{value:"A.16.5. Vector slice",id:"a165-vector-slice",level:2},{value:"A.16.6. Tipe data VecDeque<T>",id:"a166-tipe-data-vecdequet",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],o={toc:c};function m(e){let{components:a,...l}=e;return(0,i.kt)("wrapper",(0,t.Z)({},o,l,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tetang tipe data ",(0,i.kt)("em",{parentName:"p"},"Vector"),". Vector adalah tipe data seperti array tapi dinamis. Dinamis di sini artinya bisa bertambah dan berkurang kapanpun sesuai kebutuhan."),(0,i.kt)("p",null,"Vector sangat mirip seperti array, yang karakteristiknya adalah tipe data elemen wajib sama, punya informasi size, elemen-nya bisa diakses atau diubah. Salah satu perbedaan vector dibanding array adalah jumlah elemen pada vector bisa bertambah lebih dari kapasitas yang sudah ditentukan."),(0,i.kt)("p",null,"Vector memiliki 3 buah atribut yg penting untuk diketahui:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"pointer ke data asli"),(0,i.kt)("li",{parentName:"ul"},"lebar atau size"),(0,i.kt)("li",{parentName:"ul"},"kapasitas (representasi dari seberapa banyak memori di-booking untuk data vector tersebut)")),(0,i.kt)("p",null,"Vector bisa bertambah jumlah isinya selama size dibawah kapasitas yang sudah dialokasikan. Jika suatu ketika vector isinya bertambah lebih banyak dari jumlah alokasi maksimal kapasitas, maka vector akan dialokasikan ulang dengan kapasitas yang lebih besar."),(0,i.kt)("h2",{id:"a161-tipe-data-vect"},"A.16.1. Tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Vec")),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," adalah tipe data yang merepresentasikan vector, dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah generics. Vector datanya dialokasikan di heap memory."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai generic dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai heap dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/basic-memory-management"},"Basic Memory Management")))),(0,i.kt)("p",null,"Langsung saja kita praktikkan."),(0,i.kt)("h3",{id:"-deklarasi-vektor-size-dan-capacity"},"\u25c9 Deklarasi vektor, size, dan capacity"),(0,i.kt)("p",null,"Ada beberapa cara yang bisa dipakai untuk membuat data vector. Salah satunya adalah menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec"),", penulisannya seperti pembuatan array hanya saja perlu ditambahi prefix ",(0,i.kt)("inlineCode",{parentName:"p"},"vec!"),". Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut data_one = vec!["batman", "superman", "lobo"];\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(7662).Z,width:"453",height:"110"})),(0,i.kt)("p",null,"Pada contoh di atas variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," adalah sebuah vector yang isinya 3 elemen, yaitu literal string ",(0,i.kt)("inlineCode",{parentName:"p"},"batman"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"superman"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"lobo"),". Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," dideklarasikan menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec")," kemudiaan diikuti notasi penulisan yang mirip seperti array."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai macro dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/macro"},"Macro"))),(0,i.kt)("p",null,"Pembuatan vector menggunakan teknik ini hasilnya adalah data vector yang ",(0,i.kt)("inlineCode",{parentName:"p"},"size")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"capacity")," nya adalah sesuai dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value"),", yang pada konteks ini adalah 3."),(0,i.kt)("p",null,"Umumnya vector dideklarasikan dengan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"mut")," agar bisa diubah nilainya, karena tujuan disediakannya tipe data ini adalah untuk bisa mengakomodir tipe data array tetapi dinamis (bisa manipulasi datanya)."),(0,i.kt)("p",null,"Perihal method ",(0,i.kt)("inlineCode",{parentName:"p"},"len")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"capacity"),":"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"len")," digunakan untuk mencari tahu size atau jumlah elemen yang ada pada sebuah vector"),(0,i.kt)("li",{parentName:"ul"},"Method ",(0,i.kt)("inlineCode",{parentName:"li"},"capacity")," digunakan untuk mencari tahu kapasitas atau jumlah maksimum elemen yang ada pada sebuah vector")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"O iya, kode program di atas akan menghasilkan warning saat di-run. Hal ini dikarenakan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," yang didefinisikan mutable belum diubah nilainya. Hiraukan saja, karena selanjutnya kita akan manipulasi data variabel tersebut.")),(0,i.kt)("h3",{id:"-method-pop--menghapus-elemen-terakhir"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"pop")," \u279c menghapus elemen terakhir"),(0,i.kt)("p",null,"Oke, selanjutnya mari kita oprek variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," yang sudah ditulis."),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," memiliki method ",(0,i.kt)("inlineCode",{parentName:"p"},"pop")," yang fungsinya menghapus data elemen terakhir. Mari gunakan method ini pada ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one.pop();\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(7554).Z,width:"450",height:"116"})),(0,i.kt)("p",null,"Bisa dilihat pada contoh di atas, setelah menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"pop"),", isi ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," menjadi 2 elemen saja. Elemen terakhir (yaitu string ",(0,i.kt)("inlineCode",{parentName:"p"},"lobo"),") dihapus. Efeknya, atribut size menjadi 2, tapi kapasitas tetap 3."),(0,i.kt)("h3",{id:"-method-remove--menghapus-elemen-index-ke-i"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"remove")," \u279c menghapus elemen index ke ",(0,i.kt)("inlineCode",{parentName:"h3"},"I")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"remove")," adalah salah satu method lainnya yang ada pada tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),". Kegunaan dari method ",(0,i.kt)("inlineCode",{parentName:"p"},"remove")," adalah untuk menghapus elemen pada indeks tertentu."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one.remove(1);\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(41).Z,width:"450",height:"134"})),(0,i.kt)("p",null,"Dicontohkan di atas elemen indeks ke-",(0,i.kt)("inlineCode",{parentName:"p"},"1")," dihapus dengan cara memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},"remove")," lalu menyisipkan indeks elemen sebagai parameter pemanggilan method. Hasilnya elemen ",(0,i.kt)("inlineCode",{parentName:"p"},"superman")," dihapus dari vector."),(0,i.kt)("h3",{id:"-method-push--menambahkan-elemen-baru"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"push")," \u279c menambahkan elemen baru"),(0,i.kt)("p",null,"Sekarang isi dari vector ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," tinggal 1 elemen, mari kita tambahkan 3 elemen baru dengan memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"push"),"."),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"push")," fungsinya adalah untuk menambahkan elemen baru pada vector."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one.push("constantine");\ndata_one.push("trigon");\ndata_one.push("darkseid");\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(4480).Z,width:"452",height:"135"})),(0,i.kt)("p",null,"Bisa dilihat sekarang ",(0,i.kt)("inlineCode",{parentName:"p"},"data_one")," isinya adalah 4 elemen dan atribut size-nya cocok, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"4"),". Namun ada yang aneh, kenapa kapasitas jadi ",(0,i.kt)("inlineCode",{parentName:"p"},"6")," padahal di awal kapasitas adalah 3."),(0,i.kt)("h3",{id:"-realokasi-vector"},"\u25c9 Realokasi vector"),(0,i.kt)("p",null,"Perubahan kapasitas atau realokasi vector terjadi ketika sebuah vector isinya bertambah lebih banyak dari jumlah alokasi maksimal kapasitas."),(0,i.kt)("p",null,"Lalu apa efeknya? secara high level bisa dibilang tidak ada, namun kalau dibahas lebih rinci, efeknya adalah di sisi alokasi space untuk menampung elemen. Terjadi proses realokasi dimana vector yang baru akan memiliki kapasitas lebih besar."),(0,i.kt)("h3",{id:"-mengubah-value-sebuah-elemen-menggunakan-notasi-i"},"\u25c9 Mengubah value sebuah elemen menggunakan notasi ",(0,i.kt)("inlineCode",{parentName:"h3"},"[i]")),(0,i.kt)("p",null,"Sama seperti array, vector juga bisa dimodifikasi nilai elemennya dengan menggunakan notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"[i]"),"."),(0,i.kt)("p",null,"Pada contoh berikut elemen indeks ke-",(0,i.kt)("inlineCode",{parentName:"p"},"2")," diubah nilainya dari ",(0,i.kt)("inlineCode",{parentName:"p"},"trigon")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"red hood"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one[2] = "red hood";\n\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(5216).Z,width:"449",height:"131"})),(0,i.kt)("h3",{id:"-method-is_empty--mengecek-apakah-vector-kosong"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_empty")," \u279c mengecek apakah vector kosong"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_empty")," digunakan untuk mengidentifikasi apakah sebuah vector isinya kosong atau tidak."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let is_vector_empty = data_one.is_empty();\nprintln!("result: {:?}", is_vector_empty);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(8859).Z,width:"447",height:"109"})),(0,i.kt)("h3",{id:"-method-clear--mengosongkan-isi-vector"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"clear")," \u279c mengosongkan isi vector"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"clear")," digunakan untuk mengosongkan sebuah vektor."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'data_one.clear();\nprintln!("data: {:?}", data_one);\nprintln!("length: {}, capacity: {}", data_one.len(), data_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(802).Z,width:"445",height:"94"})),(0,i.kt)("h3",{id:"-method-append--concatenationpenggabungan-vector"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"append")," \u279c concatenation/penggabungan vector"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"append")," digunakan untuk menggabungkan dua buah vector. Penggunaannya cukup mudah, panggil saja method nya lalu sisipkan ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," dari vector satunya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut result_one = vec![3, 1, 2];\n\nlet mut data_two = vec![7, 6, 8];\nresult_one.append(&mut data_two);\n\nprintln!("data: {:?}", result_one);\nprintln!("length: {}, capacity: {}", result_one.len(), result_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(2366).Z,width:"450",height:"125"})),(0,i.kt)("p",null,"Pada contoh di atas ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," adalah vector mutable dengan isi 3 elemen. Kemudian dideklarasikan ",(0,i.kt)("inlineCode",{parentName:"p"},"data_two")," yang isinya juga vector 3 elemen. Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"data_two")," dimasukkan kedalam vector ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," dengan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"append"),", dengan ini maka isi ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," adalah gabungan dari ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," yang lama dan ",(0,i.kt)("inlineCode",{parentName:"p"},"data_two"),"."),(0,i.kt)("p",null,"Proses append vector mengharuskan parameter method ",(0,i.kt)("inlineCode",{parentName:"p"},"append")," diisi dengan ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," dari vector yang ingin dimasukkan. Cara untuk mengambil ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," adalah dengan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut"),"."),(0,i.kt)("p",null,"Ok, selanjutnya tambahkan lagi isi ",(0,i.kt)("inlineCode",{parentName:"p"},"result_one")," dengan vector lain."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'result_one.append(&mut vec![4, 5]);\n\nprintln!("data: {:?}", result_one);\nprintln!("length: {}, capacity: {}", result_one.len(), result_one.capacity());\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(6241).Z,width:"447",height:"127"})),(0,i.kt)("p",null,"Proses penggabungan vector pada contoh ke-dua di atas sedikit berbeda. Method ",(0,i.kt)("inlineCode",{parentName:"p"},"append")," parameternya adalah langsung ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," dari literal vector. Ini merupakan salah satu cara yang bisa digunakan dalam penggabungan vector."),(0,i.kt)("h3",{id:"-method-sort--untuk-mengurutkan-vector"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"sort")," \u279c untuk mengurutkan vector"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"sort")," digunakan untuk mengurutkan elemen vector."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'println!("data: {:?}", result_one);\nresult_one.sort();\nprintln!("data: {:?}", result_one);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(3380).Z,width:"447",height:"127"})),(0,i.kt)("h2",{id:"a162-macam-deklarasi-vektor"},"A.16.2. Macam deklarasi vektor"),(0,i.kt)("p",null,"Ada beberapa cara deklarasi vector. Pada contoh berikut dua buah vector dideklarasikan menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec!"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut vector_4 = vec![1, 2, 3];\nlet mut vector_5: Vec = vec![1, 2, 3];\n")),(0,i.kt)("p",null,"Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"vector_4")," didefinisikan dengan cara yang sudah kita terapkan sebelumnya, yaitu menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec"),". Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"vector_5")," juga didefinisikan dengan cara yang sama, hanya saja pada vector ini tipe datanya ditentukan secara eksplisit yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),"."),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," Artinya adalah vector dengan tipe data elemen adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i64"),". Dengan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," bisa ditentukan tipe data elemen yang diinginkan."),(0,i.kt)("p",null,"Cara deklarasi vector selanjutnya adalah pembuatan vector dengan isi kosong. Deklarasi vector ini mewajibkan tipe data vector dituliskan secara eksplisit, dikarenakan tipe data tidak bisa diidentifikasi dari isinya (karena isinya kosong). Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut vector_7: Vec<&str> = vec![];\nlet mut vector_8: Vec<&str> = Vec::new();\n")),(0,i.kt)("p",null,"Vector ",(0,i.kt)("inlineCode",{parentName:"p"},"vector_7")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"vector_8")," keduanya bertipe vector dengan isi atau elemen bertipe data literal string ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("p",null,"Deklarasi vector kosong bisa dilakukan dengan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"vec")," yang ditulis tanpa isi, atau bisa menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec::new()"),"."),(0,i.kt)("h2",{id:"a163-iterasi-data-vector"},"A.16.3. Iterasi data vector"),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," bisa digunakan untuk iterasi vector. Cara penerapannya seperti pada array atau slice."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let vec_eight = vec![1, 2, 3];\nfor e in vec_eight {\n print!("{e} ");\n}\n\n// 1 2 3\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let vec_nine = vec![1, 2, 3];\nfor i in 0..vec_nine.len() {\n print!("{} ", vec_nine[i]);\n}\n\n// 1 2 3\n')),(0,i.kt)("p",null,"Keyword perulangan lainnya juga bisa digunakan."),(0,i.kt)("h2",{id:"a164-ownership-tipe-data-vector"},"A.16.4. Ownership tipe data vector"),(0,i.kt)("p",null,"Salah satu atribut vector yang penting untuk diketahui adalah, pemilik data sebenarnya (atau owner). Agar lebih jelas, silakan coba terlebih dahulu kode berikut."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let vec_ten = vec![1, 2, 3];\nfor e in vec_ten {\n print!("{e} ");\n}\nfor i in 0..vec_ten.len() {\n print!("{} ", vec_ten[i]);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(2061).Z,width:"634",height:"626"})),(0,i.kt)("p",null,"Terlihat sekilas tidak ada kode yang bermasalah dari program di atas, tapi error, aneh."),(0,i.kt)("p",null,"Di Rust, ownership atau kepemilikan data adalah hal yang sangat penting. Saking pentingnya, beberapa orang menyebut Rust sebagai bahasa yang ",(0,i.kt)("em",{parentName:"p"},"value oriented"),"."),(0,i.kt)("p",null,"Dalam kasus kode program vector di atas, ketika keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," digunakan untuk mengiterasi vector ",(0,i.kt)("inlineCode",{parentName:"p"},"vec_ten"),", membuat pemilik data vektor berpindah ke variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"e"),". Hal ini efeknya adalah ketika kita berusaha mengakses variabel yang sama setelah perulangan selesai, maka yang muncul adalah error, karena value-nya sudah berpindah."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Perpindahan owner disebut dengan ",(0,i.kt)("em",{parentName:"p"},"move semantics"),". Lebih jelasnya nanti dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership"))),(0,i.kt)("p",null,"Solusi untuk antisipasi error ini adalah dengan cara meminjam value yang sebenarnya dari owner, untuk kemudian digunakan dalam perulangan. Caranya dengan menggunakan teknik ",(0,i.kt)("em",{parentName:"p"},"borrowing")," menggunakan operator ",(0,i.kt)("inlineCode",{parentName:"p"},"reference")," yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"&"),". Data sebenarnya milik owner dipinjam untuk dipergunakan di perulangan."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai reference dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/pointer-references"},"Pointer & References")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai borrowing dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/borrowing"},"Borrowing")))),(0,i.kt)("p",null,"Silakan ubah kode yang sebelumnya ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'for e in vec_ten {\n print!("{e} ");\n}\n')),(0,i.kt)("p",null,"... menjadi seperti ini, kemudian run, maka error akan hilang."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'for e in &vec_ten {\n print!("{e} ");\n}\n')),(0,i.kt)("p",null,"Salah satu alternatif cara lainnya untuk antisipasi value berpindah tempat adalah dengan menggunakan method ",(0,i.kt)("inlineCode",{parentName:"p"},"iter")," untuk mengkonversi vector menjadi iterator. Jadi yang di-iterasi bukan vector-nya, melainkan objek iterator yang dibuat dari vector tersebut."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'for e in vec_ten.iter() {\n print!("{e} ");\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai ",(0,i.kt)("em",{parentName:"li"},"ownership")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/ownership"},"Ownership")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai ",(0,i.kt)("em",{parentName:"li"},"borrowing")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/borrowing"},"Borrowing")))),(0,i.kt)("h2",{id:"a165-vector-slice"},"A.16.5. Vector slice"),(0,i.kt)("p",null,"Seperti array, slice juga bisa dibuat dari vector. Cara penerapannya juga sama persis. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let vec_population = vec![2, 1, 3];\nlet vec_sample = &vec_population[0..1];\nprintln!("{:?}", vec_sample); // [2]\n')),(0,i.kt)("p",null,"Semua operasi slice bisa diterapkan di vector."),(0,i.kt)("h2",{id:"a166-tipe-data-vecdequet"},"A.16.6. Tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"VecDeque")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque")," adalah sama seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec")," plus mendukung operasi menambah dan mengurangi elemen dari dua sisi secara efisien."),(0,i.kt)("p",null,"Pada tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Vec"),", ada method ",(0,i.kt)("inlineCode",{parentName:"p"},"pop")," yang fungsinya menghapus data elemen terakhir dan method ",(0,i.kt)("inlineCode",{parentName:"p"},"push")," untuk menambah elemen baru dari kanan. Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque")," memiliki bebebrapa method tambahan, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"method ",(0,i.kt)("inlineCode",{parentName:"li"},"pop_front")," untuk hapus data elemen pertama atau paling kiri (indeks ke-0)"),(0,i.kt)("li",{parentName:"ul"},"method ",(0,i.kt)("inlineCode",{parentName:"li"},"push_front")," untuk menambah data dari kiri (indeks ke-0)"),(0,i.kt)("li",{parentName:"ul"},"method ",(0,i.kt)("inlineCode",{parentName:"li"},"pop_back")," untuk hapus data elemen pertama atau paling kanan (indeks terakhir)"),(0,i.kt)("li",{parentName:"ul"},"method ",(0,i.kt)("inlineCode",{parentName:"li"},"push_back")," untuk menambah data dari kanan (indeks terakhir)")),(0,i.kt)("p",null,"Contoh penerapan:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use std::collections::VecDeque;\n\nlet mut vec_10 = VecDeque::from(vec!["a", "b", "c"]);\n\nvec_10.pop_front();\nvec_10.push_front("z");\nprintln!("data: {:?}", vec_10);\n\nvec_10.pop_back();\nvec_10.push_back("h");\nprintln!("data: {:?}", vec_10);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Vector",src:n(4050).Z,width:"453",height:"129"})),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque")," tidak otomatis di-import. Kita perlu mengimport path dimana tipe data itu berada menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::collections::VecDeque;\n")),(0,i.kt)("p",null,"Cara membuat vector ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque")," bisa menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"VecDeque::from")," dengan parameter diisi data vectornya, seperti pada kode program yang sudah ditulis."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/vector"},"github.com/novalagung/dasarpemrogramanrust-example/../vector")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/macro.vec.html"},"https://doc.rust-lang.org/std/macro.vec.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/vec/struct.Vec.html"},"https://doc.rust-lang.org/std/vec/struct.Vec.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/std/vec.html"},"https://doc.rust-lang.org/rust-by-example/std/vec.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/collections/struct.VecDeque.html"},"https://doc.rust-lang.org/std/collections/struct.VecDeque.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/36672845/in-rust-is-a-vector-an-iterator"},"https://stackoverflow.com/questions/36672845/in-rust-is-a-vector-an-iterator")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/28800121/what-do-i-have-to-do-to-solve-a-use-of-moved-value-error"},"https://stackoverflow.com/questions/28800121/what-do-i-have-to-do-to-solve-a-use-of-moved-value-error")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/43036279/what-does-it-mean-to-pass-in-a-vector-into-a-for-loop-versus-a-reference-to-a"},"https://stackoverflow.com/questions/43036279/what-does-it-mean-to-pass-in-a-vector-into-a-for-loop-versus-a-reference-to-a"))))}m.isMDXComponent=!0},7662:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},3380:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2061:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/vector-11-27b0884891ab00c87d90603997e1878c.png"},4050:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},7554:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},41:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},4480:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},5216:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},8859:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},802:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},2366:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},6241:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/cb962766.2d69a424.js b/assets/js/cb962766.2d69a424.js new file mode 100644 index 00000000..2f81dea1 --- /dev/null +++ b/assets/js/cb962766.2d69a424.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6691],{3905:(e,n,t)=>{t.d(n,{Zo:()=>s,kt:()=>c});var a=t(7294);function r(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function i(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);n&&(a=a.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,a)}return t}function l(e){for(var n=1;n=0||(r[t]=e[t]);return r}(e,n);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(r[t]=e[t])}return r}var p=a.createContext({}),o=function(e){var n=a.useContext(p),t=n;return e&&(t="function"==typeof e?e(n):l(l({},n),e)),t},s=function(e){var n=o(e.components);return a.createElement(p.Provider,{value:n},e.children)},m={inlineCode:"code",wrapper:function(e){var n=e.children;return a.createElement(a.Fragment,{},n)}},d=a.forwardRef((function(e,n){var t=e.components,r=e.mdxType,i=e.originalType,p=e.parentName,s=u(e,["components","mdxType","originalType","parentName"]),d=o(t),c=r,k=d["".concat(p,".").concat(c)]||d[c]||m[c]||i;return t?a.createElement(k,l(l({ref:n},s),{},{components:t})):a.createElement(k,l({ref:n},s))}));function c(e,n){var t=arguments,r=n&&n.mdxType;if("string"==typeof e||r){var i=t.length,l=new Array(i);l[0]=d;var u={};for(var p in n)hasOwnProperty.call(n,p)&&(u[p]=n[p]);u.originalType=e,u.mdxType="string"==typeof e?e:r,l[1]=u;for(var o=2;o{t.r(n),t.d(n,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>i,metadata:()=>u,toc:()=>o});var a=t(7462),r=(t(7294),t(3905));const i={sidebar_position:47,title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",sidebar_label:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)"},l=void 0,u={unversionedId:"basic/trait-function",id:"basic/trait-function",title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",description:"Chapter ini merupakan topik lanjutan dari chapter sebelumnya, di sini kita akan bahas beberapa jenis trait yang digunakan di closure.",source:"@site/docs/basic/trait-function.md",sourceDirName:"basic",slug:"/basic/trait-function",permalink:"/basic/trait-function",draft:!1,tags:[],version:"current",sidebarPosition:47,frontMatter:{sidebar_position:47,title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",sidebar_label:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)"},sidebar:"tutorialSidebar",previous:{title:"A.46. Closures",permalink:"/basic/closures"},next:{title:"A.48. Trait \u279c Iterator",permalink:"/basic/trait-iterator"}},p={},o=[{value:"A.47.1. Trait Fn",id:"a471-trait-fn",level:2},{value:"A.47.2. Trait FnMut",id:"a472-trait-fnmut",level:2},{value:"A.47.3. Trait FnOnce",id:"a473-trait-fnonce",level:2},{value:"A.47.4. Relasi antara trait function dengan function",id:"a474-relasi-antara-trait-function-dengan-function",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],s={toc:o};function m(e){let{components:n,...i}=e;return(0,r.kt)("wrapper",(0,a.Z)({},s,i,{components:n,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Chapter ini merupakan topik lanjutan dari chapter sebelumnya, di sini kita akan bahas beberapa jenis trait yang digunakan di closure."),(0,r.kt)("p",null,"Rust memiliki 3 buah trait yang otomatis ter-implement pada closure, yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut"),", dan ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce"),". Ketiga trait ini juga otomatis ter-implement pada fungsi yang didefinisikan menggunakan keyword ",(0,r.kt)("inlineCode",{parentName:"p"},"fn"),"."),(0,r.kt)("h2",{id:"a471-trait-fn"},"A.47.1. Trait ",(0,r.kt)("inlineCode",{parentName:"h2"},"Fn")),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"Fn")," sudah kita terapkan beberapa kali saat praktik pembuatan closure di chapter sebelumnya. Fn adalah trait (",(0,r.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/ops/trait.Fn.html"},"std::ops::Fn"),"), yang dipakai oleh Rust untuk menandai bahwa suatu closure bisa dipanggil berkali-kali dengan catatan di dalam closure tersebut tidak ada operasi mutable terhadap variabel yang scope-nya berada di luar block closure."),(0,r.kt)("p",null,"Contoh penerapan trait ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn")," bisa dilihat pada kode berikut. Trait tersebut secara otomatis ter-implement pada closure ",(0,r.kt)("inlineCode",{parentName:"p"},"pow"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'let pow_number = |n: i32| n.pow(2);\n\nprintln!("pow_number(2): {}", pow_number(2));\nprintln!("pow_number(3): {}", pow_number(3));\nprintln!("pow_number(4): {}", pow_number(4));\n')),(0,r.kt)("p",null,"Contoh penerapan trait ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn")," lainnya pada fungsi yang memiliki parameter closure:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let result = do_something_with_number_v1(13, |d: i32| d * 2);\n println!("result: {result}");\n}\n\nfn do_something_with_number_v1(n: i32, f: F) -> i32\nwhere\n F: Fn(i32) -> i32, // <----- Fn digunakan\n{\n // Statement `f(n)` bisa dipanggil berkali-kali\n return f(n);\n}\n')),(0,r.kt)("h2",{id:"a472-trait-fnmut"},"A.47.2. Trait ",(0,r.kt)("inlineCode",{parentName:"h2"},"FnMut")),(0,r.kt)("p",null,"Trait ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," (",(0,r.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/ops/trait.FnMut.html"},"std::ops::FnMut"),") merupakan trait yang menjadikan suatu closure bisa diakses berkali-kali dan bisa me-mutate atau mengubah data suatu variabel yang berada di luar scope block closure. Trait ini juga otomatis ter-implement pada closure yang di dalamnya ada kode pengaksesan variabel yang berada di luar scope block closure."),(0,r.kt)("p",null,"Contoh penerapannya silakan lihat closure ",(0,r.kt)("inlineCode",{parentName:"p"},"square_x")," berikut. Closure tersebut di-dalamnya mengubah nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," yang dideklarasikan di luar blcok closure, oleh karenanya trait ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," otomatis ter-implement."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'let mut x = 5;\n{\n let mut square_x = || x *= x;\n square_x();\n}\nprintln!("result: {}", x);\n')),(0,r.kt)("p",null,"Contoh lain penerapan ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," pada fungsi yang memiliki parameter closure:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut number = 1;\n do_something_with_number_v2(14, |x| number += x);\n println!("number: {number}");\n}\n\nfn do_something_with_number_v2(n: i32, mut f: F)\nwhere\n F: FnMut(i32), // <----- FnMut digunakan\n{\n // Statement `f(n)` berisi kode yang mengubah isi variavel `number` (mutable).\n // `f(n)` bisa dipanggil berkali-kali\n f(n);\n}\n')),(0,r.kt)("p",null,"Parameter ke-2 fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something_with_number_v2")," adalah closure yang me-mutate nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),", berarti terjadi mutable borrow karena variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," berada di luar block closure. Karena alasan tersebut maka closure harus dideklarasikan menggunakan ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," (bukan ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),")."),(0,r.kt)("p",null,"Jika dipaksa deklarasi menggunakan ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),", hasilnya pasti error."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Closure",src:t(2212).Z,width:"673",height:"274"})),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," merupakan supertrait dari ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),", artinya closure dengan trait ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn")," juga bisa digunakan sebagai argument pemanggilan fungsi di mana parameter fungsi tersebut bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai supertrait dibahas pada chpater ",(0,r.kt)("a",{parentName:"p",href:"#/wip/supertrait"},"Supertrait"))),(0,r.kt)("h2",{id:"a473-trait-fnonce"},"A.47.3. Trait ",(0,r.kt)("inlineCode",{parentName:"h2"},"FnOnce")),(0,r.kt)("p",null,"Trait ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce")," (",(0,r.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/ops/trait.FnOnce.html"},"std::ops::FnOnce"),") adalah trait yang menjadikan suatu closure hanya boleh di akses sekali saja, dan closure tersebut bisa berisi operasi mutable ataupun tidak."),(0,r.kt)("p",null,"Contoh penerapan ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce")," pada fungsi yang memiliki parameter closure."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut number = 1;\n do_something_with_number_v3(14, |x| number += x);\n println!("number: {number}");\n}\n\nfn do_something_with_number_v3(n: i32, f: F)\nwhere\n F: FnOnce(i32), // <----- FnOnce digunakan\n{\n f(n);\n}\n')),(0,r.kt)("p",null,"Jika closure dengan tipe ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce")," dipaksa diakses dua kali, pasti muncul error. Contohnya bisa dilihat pada parameter ",(0,r.kt)("inlineCode",{parentName:"p"},"f")," berikut."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Closure",src:t(1385).Z,width:"506",height:"395"})),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai supertrait dibahas pada chpater ",(0,r.kt)("a",{parentName:"p",href:"#/wip/supertrait"},"Supertrait"))),(0,r.kt)("h2",{id:"a474-relasi-antara-trait-function-dengan-function"},"A.47.4. Relasi antara trait function dengan function"),(0,r.kt)("p",null,"Trait ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut"),", dan juga ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce")," tidak hanya ter-implement pada closure, tapi juga ter-implement pada fungsi juga (secara otomatis)."),(0,r.kt)("p",null,"Contohnya bisa dilihat pada contoh berikut. Fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something_with_number_v1")," dipanggil 3 kali."),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Pemanggilan pertama, argument ",(0,r.kt)("inlineCode",{parentName:"li"},"f")," diisi dengan closure ",(0,r.kt)("inlineCode",{parentName:"li"},"|d: i32| d * 2")),(0,r.kt)("li",{parentName:"ul"},"Pemanggilan kedua, argument ",(0,r.kt)("inlineCode",{parentName:"li"},"f")," diisi dengan fungsi ",(0,r.kt)("inlineCode",{parentName:"li"},"double")),(0,r.kt)("li",{parentName:"ul"},"Pemanggilan ketiga, argument ",(0,r.kt)("inlineCode",{parentName:"li"},"f")," diisi dengan fungsi ",(0,r.kt)("inlineCode",{parentName:"li"},"pow_number"))),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let result = do_something_with_number_v1(13, |d: i32| d * 2);\n println!("result: {result}");\n \n let result = do_something_with_number_v1(13, double);\n println!("result: {result}");\n \n let result = do_something_with_number_v1(13, pow_number);\n println!("result: {result}");\n}\n\nfn do_something_with_number_v1(n: i32, f: F) -> i32\nwhere\n F: Fn(i32) -> i32,\n{\n return f(n);\n}\n\nfn double(d: i32) -> i32 {\n d * 2\n}\n\nfn pow_number(d: i32) -> i32 {\n d.pow(2)\n}\n')),(0,r.kt)("p",null,"Dari contoh di atas terbukti bahwa fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"double")," dan juga ",(0,r.kt)("inlineCode",{parentName:"p"},"pow_number")," memenuhi kriteria tipe ",(0,r.kt)("inlineCode",{parentName:"p"},"F")," yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn(i32) -> i32"),"."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Closure",src:t(2795).Z,width:"450",height:"143"})),(0,r.kt)("hr",null),(0,r.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,r.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,r.kt)("pre",null,(0,r.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/trait_function"},"github.com/novalagung/dasarpemrogramanrust-example/../trait_function")),(0,r.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"/basic/function"},"Function")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"/basic/closures"},"Closures"))),(0,r.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch13-01-closures.html"},"https://doc.rust-lang.org/book/ch13-01-closures.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/beta/rust-by-example/fn/closures.html"},"https://doc.rust-lang.org/beta/rust-by-example/fn/closures.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.Fn.html"},"https://doc.rust-lang.org/std/ops/trait.Fn.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.FnMut.html"},"https://doc.rust-lang.org/std/ops/trait.FnMut.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.FnOnce.html"},"https://doc.rust-lang.org/std/ops/trait.FnOnce.html"))))}m.isMDXComponent=!0},2212:(e,n,t)=>{t.d(n,{Z:()=>a});const a=t.p+"assets/images/trait-function-1-69d13024d5321866a37aa3bb1f72a1b2.png"},1385:(e,n,t)=>{t.d(n,{Z:()=>a});const a=t.p+"assets/images/trait-function-2-5f9002b90d061f7ad1815a678e71f312.png"},2795:(e,n,t)=>{t.d(n,{Z:()=>a});const a=""}}]); \ No newline at end of file diff --git a/assets/js/cb962766.fe11400b.js b/assets/js/cb962766.fe11400b.js deleted file mode 100644 index 90aaa437..00000000 --- a/assets/js/cb962766.fe11400b.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6691],{3905:(e,n,t)=>{t.d(n,{Zo:()=>s,kt:()=>c});var a=t(7294);function r(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}function i(e,n){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);n&&(a=a.filter((function(n){return Object.getOwnPropertyDescriptor(e,n).enumerable}))),t.push.apply(t,a)}return t}function l(e){for(var n=1;n=0||(r[t]=e[t]);return r}(e,n);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(r[t]=e[t])}return r}var p=a.createContext({}),o=function(e){var n=a.useContext(p),t=n;return e&&(t="function"==typeof e?e(n):l(l({},n),e)),t},s=function(e){var n=o(e.components);return a.createElement(p.Provider,{value:n},e.children)},m={inlineCode:"code",wrapper:function(e){var n=e.children;return a.createElement(a.Fragment,{},n)}},d=a.forwardRef((function(e,n){var t=e.components,r=e.mdxType,i=e.originalType,p=e.parentName,s=u(e,["components","mdxType","originalType","parentName"]),d=o(t),c=r,k=d["".concat(p,".").concat(c)]||d[c]||m[c]||i;return t?a.createElement(k,l(l({ref:n},s),{},{components:t})):a.createElement(k,l({ref:n},s))}));function c(e,n){var t=arguments,r=n&&n.mdxType;if("string"==typeof e||r){var i=t.length,l=new Array(i);l[0]=d;var u={};for(var p in n)hasOwnProperty.call(n,p)&&(u[p]=n[p]);u.originalType=e,u.mdxType="string"==typeof e?e:r,l[1]=u;for(var o=2;o{t.r(n),t.d(n,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>i,metadata:()=>u,toc:()=>o});var a=t(7462),r=(t(7294),t(3905));const i={sidebar_position:47,title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",sidebar_label:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)"},l=void 0,u={unversionedId:"basic/trait-function",id:"basic/trait-function",title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",description:"Chapter ini merupakan topik lanjutan dari chapter sebelumnya, di sini kita akan bahas beberapa jenis trait yang digunakan di closure.",source:"@site/docs/basic/trait-function.md",sourceDirName:"basic",slug:"/basic/trait-function",permalink:"/basic/trait-function",draft:!1,tags:[],version:"current",sidebarPosition:47,frontMatter:{sidebar_position:47,title:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)",sidebar_label:"A.47. Trait \u279c Function (Fn, FnMut, FnOnce)"},sidebar:"tutorialSidebar",previous:{title:"A.46. Closures",permalink:"/basic/closures"},next:{title:"A.48. Trait \u279c Iterator",permalink:"/basic/trait-iterator"}},p={},o=[{value:"A.47.1. Trait Fn",id:"a471-trait-fn",level:2},{value:"A.47.2. Trait FnMut",id:"a472-trait-fnmut",level:2},{value:"A.47.3. Trait FnOnce",id:"a473-trait-fnonce",level:2},{value:"A.47.4. Relasi antara trait function dengan function",id:"a474-relasi-antara-trait-function-dengan-function",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],s={toc:o};function m(e){let{components:n,...i}=e;return(0,r.kt)("wrapper",(0,a.Z)({},s,i,{components:n,mdxType:"MDXLayout"}),(0,r.kt)("p",null,"Chapter ini merupakan topik lanjutan dari chapter sebelumnya, di sini kita akan bahas beberapa jenis trait yang digunakan di closure."),(0,r.kt)("p",null,"Rust memiliki 3 buah trait yang otomatis ter-implement pada closure, yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut"),", dan ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce"),". Ketiga trait ini juga otomatis ter-implement pada fungsi yang didefinisikan menggunakan keyword ",(0,r.kt)("inlineCode",{parentName:"p"},"fn"),"."),(0,r.kt)("h2",{id:"a471-trait-fn"},"A.47.1. Trait ",(0,r.kt)("inlineCode",{parentName:"h2"},"Fn")),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"Fn")," sudah kita terapkan beberapa kali saat praktik pembuatan closure di chapter sebelumnya. Fn adalah trait (",(0,r.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/ops/trait.Fn.html"},"std::ops::Fn"),"), yang dipakai oleh Rust untuk menandai bahwa suatu closure bisa dipanggil berkali-kali dengan catatan di dalam closure tersebut tidak ada operasi mutable terhadap variabel yang scope-nya berada di luar block closure."),(0,r.kt)("p",null,"Contoh penerapan trait ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn")," bisa dilihat pada kode berikut. Trait tersebut secara otomatis ter-implement pada closure ",(0,r.kt)("inlineCode",{parentName:"p"},"pow"),"."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'let pow_number = |n: i32| n.pow(2);\n\nprintln!("pow_number(2): {}", pow_number(2));\nprintln!("pow_number(3): {}", pow_number(3));\nprintln!("pow_number(4): {}", pow_number(4));\n')),(0,r.kt)("p",null,"Contoh penerapan trait ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn")," lainnya pada fungsi yang memiliki parameter closure:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let result = do_something_with_number_v1(13, |d: i32| d * 2);\n println!("result: {result}");\n}\n\nfn do_something_with_number_v1(n: i32, f: F) -> i32\nwhere\n F: Fn(i32) -> i32, // <----- Fn digunakan\n{\n // Statement `f(n)` bisa dipanggil berkali-kali\n return f(n);\n}\n')),(0,r.kt)("h2",{id:"a472-trait-fnmut"},"A.47.2. Trait ",(0,r.kt)("inlineCode",{parentName:"h2"},"FnMut")),(0,r.kt)("p",null,"Trait ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," (",(0,r.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/ops/trait.FnMut.html"},"std::ops::FnMut"),") merupakan trait yang menjadikan suatu closure bisa diakses berkali-kali dan bisa me-mutate atau mengubah data suatu variabel yang berada di luar scope block closure. Trait ini juga otomatis ter-implement pada closure yang di dalamnya ada kode pengaksesan variabel yang berada di luar scope block closure."),(0,r.kt)("p",null,"Contoh penerapannya silakan lihat closure ",(0,r.kt)("inlineCode",{parentName:"p"},"square_x")," berikut. Closure tersebut di-dalamnya mengubah nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"x")," yang dideklarasikan di luar blcok closure, oleh karenanya trait ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," otomatis ter-implement."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'let mut x = 5;\n{\n let mut square_x = || x *= x;\n square_x();\n}\nprintln!("result: {}", x);\n')),(0,r.kt)("p",null,"Contoh lain penerapan ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," pada fungsi yang memiliki parameter closure:"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut number = 1;\n do_something_with_number_v2(14, |x| number += x);\n println!("number: {number}");\n}\n\nfn do_something_with_number_v2(n: i32, mut f: F)\nwhere\n F: FnMut(i32), // <----- FnMut digunakan\n{\n // Statement `f(n)` berisi kode yang mengubah isi variavel `number` (mutable).\n // `f(n)` bisa dipanggil berkali-kali\n f(n);\n}\n')),(0,r.kt)("p",null,"Parameter ke-2 fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something_with_number_v2")," adalah closure yang me-mutate nilai ",(0,r.kt)("inlineCode",{parentName:"p"},"number"),", berarti terjadi mutable borrow karena variabel ",(0,r.kt)("inlineCode",{parentName:"p"},"number")," berada di luar block closure. Karena alasan tersebut maka closure harus dideklarasikan menggunakan ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," (bukan ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),")."),(0,r.kt)("p",null,"Jika dipaksa deklarasi menggunakan ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),", hasilnya pasti error."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Closure",src:t(2212).Z,width:"673",height:"274"})),(0,r.kt)("p",null,(0,r.kt)("inlineCode",{parentName:"p"},"FnMut")," merupakan supertrait dari ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),", artinya closure dengan trait ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn")," juga bisa digunakan sebagai argument pemanggilan fungsi dimana parameter fungsi tersebut bertipe ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut"),"."),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai supertrait dibahas pada chpater ",(0,r.kt)("a",{parentName:"p",href:"#/wip/supertrait"},"Supertrait"))),(0,r.kt)("h2",{id:"a473-trait-fnonce"},"A.47.3. Trait ",(0,r.kt)("inlineCode",{parentName:"h2"},"FnOnce")),(0,r.kt)("p",null,"Trait ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce")," (",(0,r.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/ops/trait.FnOnce.html"},"std::ops::FnOnce"),") adalah trait yang menjadikan suatu closure hanya boleh di akses sekali saja, dan closure tersebut bisa berisi operasi mutable ataupun tidak."),(0,r.kt)("p",null,"Contoh penerapan ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce")," pada fungsi yang memiliki parameter closure."),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut number = 1;\n do_something_with_number_v3(14, |x| number += x);\n println!("number: {number}");\n}\n\nfn do_something_with_number_v3(n: i32, f: F)\nwhere\n F: FnOnce(i32), // <----- FnOnce digunakan\n{\n f(n);\n}\n')),(0,r.kt)("p",null,"Jika closure dengan tipe ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce")," dipaksa diakses dua kali, pasti muncul error. Contohnya bisa dilihat pada parameter ",(0,r.kt)("inlineCode",{parentName:"p"},"f")," berikut."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Closure",src:t(1385).Z,width:"506",height:"395"})),(0,r.kt)("blockquote",null,(0,r.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai supertrait dibahas pada chpater ",(0,r.kt)("a",{parentName:"p",href:"#/wip/supertrait"},"Supertrait"))),(0,r.kt)("h2",{id:"a474-relasi-antara-trait-function-dengan-function"},"A.47.4. Relasi antara trait function dengan function"),(0,r.kt)("p",null,"Trait ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn"),", ",(0,r.kt)("inlineCode",{parentName:"p"},"FnMut"),", dan juga ",(0,r.kt)("inlineCode",{parentName:"p"},"FnOnce")," tidak hanya ter-implement pada closure, tapi juga ter-implement pada fungsi juga (secara otomatis)."),(0,r.kt)("p",null,"Contohnya bisa dilihat pada contoh berikut. Fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"do_something_with_number_v1")," dipanggil 3 kali."),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"Pemanggilan pertama, argument ",(0,r.kt)("inlineCode",{parentName:"li"},"f")," diisi dengan closure ",(0,r.kt)("inlineCode",{parentName:"li"},"|d: i32| d * 2")),(0,r.kt)("li",{parentName:"ul"},"Pemanggilan kedua, argument ",(0,r.kt)("inlineCode",{parentName:"li"},"f")," diisi dengan fungsi ",(0,r.kt)("inlineCode",{parentName:"li"},"double")),(0,r.kt)("li",{parentName:"ul"},"Pemanggilan ketiga, argument ",(0,r.kt)("inlineCode",{parentName:"li"},"f")," diisi dengan fungsi ",(0,r.kt)("inlineCode",{parentName:"li"},"pow_number"))),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let result = do_something_with_number_v1(13, |d: i32| d * 2);\n println!("result: {result}");\n \n let result = do_something_with_number_v1(13, double);\n println!("result: {result}");\n \n let result = do_something_with_number_v1(13, pow_number);\n println!("result: {result}");\n}\n\nfn do_something_with_number_v1(n: i32, f: F) -> i32\nwhere\n F: Fn(i32) -> i32,\n{\n return f(n);\n}\n\nfn double(d: i32) -> i32 {\n d * 2\n}\n\nfn pow_number(d: i32) -> i32 {\n d.pow(2)\n}\n')),(0,r.kt)("p",null,"Dari contoh di atas terbukti bahwa fungsi ",(0,r.kt)("inlineCode",{parentName:"p"},"double")," dan juga ",(0,r.kt)("inlineCode",{parentName:"p"},"pow_number")," memenuhi kriteria tipe ",(0,r.kt)("inlineCode",{parentName:"p"},"F")," yaitu ",(0,r.kt)("inlineCode",{parentName:"p"},"Fn(i32) -> i32"),"."),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Closure",src:t(2795).Z,width:"450",height:"143"})),(0,r.kt)("hr",null),(0,r.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,r.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,r.kt)("pre",null,(0,r.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/trait_function"},"github.com/novalagung/dasarpemrogramanrust-example/../trait_function")),(0,r.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"/basic/function"},"Function")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"/basic/closures"},"Closures"))),(0,r.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch13-01-closures.html"},"https://doc.rust-lang.org/book/ch13-01-closures.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/beta/rust-by-example/fn/closures.html"},"https://doc.rust-lang.org/beta/rust-by-example/fn/closures.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.Fn.html"},"https://doc.rust-lang.org/std/ops/trait.Fn.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.FnMut.html"},"https://doc.rust-lang.org/std/ops/trait.FnMut.html")),(0,r.kt)("li",{parentName:"ul"},(0,r.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/ops/trait.FnOnce.html"},"https://doc.rust-lang.org/std/ops/trait.FnOnce.html"))))}m.isMDXComponent=!0},2212:(e,n,t)=>{t.d(n,{Z:()=>a});const a=t.p+"assets/images/trait-function-1-69d13024d5321866a37aa3bb1f72a1b2.png"},1385:(e,n,t)=>{t.d(n,{Z:()=>a});const a=t.p+"assets/images/trait-function-2-5f9002b90d061f7ad1815a678e71f312.png"},2795:(e,n,t)=>{t.d(n,{Z:()=>a});const a=""}}]); \ No newline at end of file diff --git a/assets/js/d8837387.31f2a62c.js b/assets/js/d8837387.31f2a62c.js new file mode 100644 index 00000000..674fa912 --- /dev/null +++ b/assets/js/d8837387.31f2a62c.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4622],{3905:(a,t,A)=>{A.d(t,{Zo:()=>g,kt:()=>b});var e=A(7294);function i(a,t,A){return t in a?Object.defineProperty(a,t,{value:A,enumerable:!0,configurable:!0,writable:!0}):a[t]=A,a}function n(a,t){var A=Object.keys(a);if(Object.getOwnPropertySymbols){var e=Object.getOwnPropertySymbols(a);t&&(e=e.filter((function(t){return Object.getOwnPropertyDescriptor(a,t).enumerable}))),A.push.apply(A,e)}return A}function l(a){for(var t=1;t=0||(i[A]=a[A]);return i}(a,t);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);for(e=0;e=0||Object.prototype.propertyIsEnumerable.call(a,A)&&(i[A]=a[A])}return i}var p=e.createContext({}),d=function(a){var t=e.useContext(p),A=t;return a&&(A="function"==typeof a?a(t):l(l({},t),a)),A},g=function(a){var t=d(a.components);return e.createElement(p.Provider,{value:t},a.children)},m={inlineCode:"code",wrapper:function(a){var t=a.children;return e.createElement(e.Fragment,{},t)}},k=e.forwardRef((function(a,t){var A=a.components,i=a.mdxType,n=a.originalType,p=a.parentName,g=r(a,["components","mdxType","originalType","parentName"]),k=d(A),b=i,c=k["".concat(p,".").concat(b)]||k[b]||m[b]||n;return A?e.createElement(c,l(l({ref:t},g),{},{components:A})):e.createElement(c,l({ref:t},g))}));function b(a,t){var A=arguments,i=t&&t.mdxType;if("string"==typeof a||i){var n=A.length,l=new Array(n);l[0]=k;var r={};for(var p in t)hasOwnProperty.call(t,p)&&(r[p]=t[p]);r.originalType=a,r.mdxType="string"==typeof a?a:i,l[1]=r;for(var d=2;d{A.r(t),A.d(t,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>n,metadata:()=>r,toc:()=>d});var e=A(7462),i=(A(7294),A(3905));const n={sidebar_position:6,title:"A.6. Tipe Data \u279c String Literal (&str)",sidebar_label:"A.6. Tipe Data \u279c String Literal (&str)"},l=void 0,r={unversionedId:"basic/tipe-data-string-literal",id:"basic/tipe-data-string-literal",title:"A.6. Tipe Data \u279c String Literal (&str)",description:"String adalah tipe data penting dalam pemrograman manapun. Biasanya tiap bahasa punya cara berbeda dalam meng-handle tipe ini. Di bahasa Rust, ada dua jenis tipe data string:",source:"@site/docs/basic/tipe-data-string-literal.md",sourceDirName:"basic",slug:"/basic/tipe-data-string-literal",permalink:"/basic/tipe-data-string-literal",draft:!1,tags:[],version:"current",sidebarPosition:6,frontMatter:{sidebar_position:6,title:"A.6. Tipe Data \u279c String Literal (&str)",sidebar_label:"A.6. Tipe Data \u279c String Literal (&str)"},sidebar:"tutorialSidebar",previous:{title:"A.5. Tipe Data \u279c Primitive Scalar",permalink:"/basic/tipe-data-primitive-scalar"},next:{title:"A.7. Konstanta",permalink:"/basic/konstanta"}},p={},d=[{value:"A.6.1. String literal atau &str",id:"a61-string-literal-atau-str",level:2},{value:"A.6.2. Escape karakter menggunakan ",id:"a62-escape-karakter-menggunakan-",level:2},{value:"A.6.3. Multiline string literal",id:"a63-multiline-string-literal",level:2},{value:"A.6.4. Raw string",id:"a64-raw-string",level:2},{value:"A.6.5. Pembahasan lanjutan tentang string",id:"a65-pembahasan-lanjutan-tentang-string",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],g={toc:d};function m(a){let{components:t,...n}=a;return(0,i.kt)("wrapper",(0,e.Z)({},g,n,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"String adalah tipe data penting dalam pemrograman manapun. Biasanya tiap bahasa punya cara berbeda dalam meng-handle tipe ini. Di bahasa Rust, ada dua jenis tipe data string:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Tipe string literal (kadang disebut dengan string slice, atau ",(0,i.kt)("inlineCode",{parentName:"li"},"&str"),"). Tipe data ini ada pada nilai yang dideklarasikan dengan diapit tanda petik dua (string literal), contohnya ",(0,i.kt)("inlineCode",{parentName:"li"},'"Hello, world!"'),"."),(0,i.kt)("li",{parentName:"ul"},"Tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," yang merupakan tipe data custom atau ",(0,i.kt)("em",{parentName:"li"},"custom types")," yang merupakan sebuah struct. Lebih jelasnya akan kita bahas pada chapter terpisah.")),(0,i.kt)("p",null,"Chapter ini hanya fokus pada string literal, dengan level pembahasan yang tidak terlalu advance."),(0,i.kt)("h2",{id:"a61-string-literal-atau-str"},"A.6.1. String literal atau ",(0,i.kt)("inlineCode",{parentName:"h2"},"&str")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Kita tidak akan bahas terlalu low-level untuk topik ini, karena memang masih di chapter awal-awal.")),(0,i.kt)("p",null,"Rust adalah bahasa yang ",(0,i.kt)("em",{parentName:"p"},"statically typed"),", tipe data harus diketahui saat kompilasi. Setiap pendefinisian variabel, entah itu dengan di-specify tipe datanya (contoh: ",(0,i.kt)("inlineCode",{parentName:"p"},"let x: i32 = 5"),") atau menggunakan teknik ",(0,i.kt)("em",{parentName:"p"},"type inference")," (contoh: ",(0,i.kt)("inlineCode",{parentName:"p"},"let x = 5"),"), tipe datanya akan diketahui di awal saat kompilasi program."),(0,i.kt)("p",null,"Begitu juga pada tipe string. Sebagai contoh, statement ",(0,i.kt)("inlineCode",{parentName:"p"},'let y = "hello"'),", variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"y")," di sini memiliki tipe data, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("p",null,"Apa itu tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"? Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"str")," merupakan salah satu tipe primitif yang ada di Rust. Penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," menandakan bahwa tipe tersebut adalah ",(0,i.kt)("strong",{parentName:"p"},"pointer ",(0,i.kt)("inlineCode",{parentName:"strong"},"str")),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Untuk pembahasan lebih detail mengenai tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"&")," pada tipe data akan dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"))),(0,i.kt)("p",null,"Untuk kawan-kawan yang menggunakan VSCode dengan ekstensi ",(0,i.kt)("inlineCode",{parentName:"p"},"rust-analyzer"),", tipe data variabel bisa terlihat saat definisi."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rust string literal",src:A(7031).Z,width:"299",height:"146"})),(0,i.kt)("p",null,"Ok, sampai sini yang paling penting adalah dipahami dulu bahwa string literal tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),". Jadi selanjutnya kalau melihat tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," jangan bingung, itu berarti string."),(0,i.kt)("h2",{id:"a62-escape-karakter-menggunakan-"},"A.6.2. Escape karakter menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"\\")),(0,i.kt)("p",null,"Tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"\\")," digunakan untuk ",(0,i.kt)("em",{parentName:"p"},"escape")," beberapa karakter string."),(0,i.kt)("p",null,"Silakan perhatikan contoh berikut. Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"var2")," isinya adalah string yang di dalamnya ada beberapa karakter yang di escape, yaitu tanda ",(0,i.kt)("inlineCode",{parentName:"p"},'"')," dan baris baru atau newline."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var2 = "hello \\\n \\"rust\\" \\\n and \\\n \\"world\\"";\nprintln!("{}", var2);\n')),(0,i.kt)("p",null,"Coba jalankan kode di atas."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rust string literal",src:A(6028).Z,width:"445",height:"115"})),(0,i.kt)("p",null,"Karena semua baris baru pada contoh di atas di-escape, jadinya string akan tetap 1 baris. Tanda ",(0,i.kt)("inlineCode",{parentName:"p"},'"')," juga tetap muncul karena di-escape."),(0,i.kt)("h2",{id:"a63-multiline-string-literal"},"A.6.3. Multiline string literal"),(0,i.kt)("p",null,"Penulisan string banyak baris atau ",(0,i.kt)("em",{parentName:"p"},"multiline string")," adalah sama seperti penulisan string biasa, yaitu langsung tulis saja string dengan diapit tanda ",(0,i.kt)("inlineCode",{parentName:"p"},'"'),", tambahkan baris baru di dalam string tersebut sesuai kebutuhan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var3 = "baris satu\nbaris dua\nbaris tiga";\nprintln!("{}", var3);\n')),(0,i.kt)("p",null,"Hasilnya adalah sesuai dengan string yang sudah didefinisikan."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rust string literal",src:A(1664).Z,width:"452",height:"129"})),(0,i.kt)("p",null,"Perlu diketahui bahwa karakter spasi, baris baru, dan lainnya adalah ",(0,i.kt)("strong",{parentName:"p"},"tidak dihiraukan"),", jadi jika kawan-kawan menuliskan string multiline seperti ini ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var4 = "baris satu\n baris dua\n baris tiga";\nprintln!("{}", var4);\n')),(0,i.kt)("p",null,"... maka hasilnya juga sesuai dengan yang ditulis, yaitu ada 4 karakter spasi di baris ke-2 dan ke-3."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rust string literal",src:A(8353).Z,width:"449",height:"130"})),(0,i.kt)("h2",{id:"a64-raw-string"},"A.6.4. Raw string"),(0,i.kt)("p",null,"Raw string adalah istilah untuk string yang tidak meng-",(0,i.kt)("em",{parentName:"p"},"escape")," karakter apapun. Di Rust, string literal bisa didefinisikan dengan menuliskan string diapit ",(0,i.kt)("inlineCode",{parentName:"p"},'r#"')," dan ",(0,i.kt)("inlineCode",{parentName:"p"},'"#'),". Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var5 = r#"\n {\n "name": "tim drake",\n "gender": "male"\n }\n"#;\nprintln!("{}", var5);\n')),(0,i.kt)("p",null,"Kode di atas hasilnya adalah ekuivalen dengan kode di bawah ini, yang mana string didefinisikan dengan meng-escape karakter ",(0,i.kt)("inlineCode",{parentName:"p"},'"')," menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"\\"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var6 = "\n {\n \\"name\\": \\"cassandra cain\\",\n \\"gender\\": \\"female\\"\n }\n";\nprintln!("{}", var6);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"raw string literal",src:A(8532).Z,width:"430",height:"260"})),(0,i.kt)("h2",{id:"a65-pembahasan-lanjutan-tentang-string"},"A.6.5. Pembahasan lanjutan tentang string"),(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"String adalah salah satu topik yang sangat luas cakupan pembahasannya"),", tidak cukup jika dirangkum dalam 1 chapter. Selain itu, bisa bikin makin bingung jika dibahas sekarang."),(0,i.kt)("p",null,"Penulis anjurkan untuk mempelajari chapter-chapter berikutnya secara urut terlebih dahulu."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/tipe_data_string_literal"},"github.com/novalagung/dasarpemrogramanrust-example/../tipe_data_string_literal")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pointer-references"},"Pointer & References")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-custom-type-string-slice"},"Tipe Data \u279c String Custom Type"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/tokens.html#raw-string-literals"},"https://doc.rust-lang.org/reference/tokens.html#raw-string-literals")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/str/index.html"},"https://doc.rust-lang.org/std/str/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://web.mit.edu/rust-lang_v1.25/arch/amd64_ubuntu1404/share/doc/rust/html/book/first-edition/strings.html"},"https://web.mit.edu/rust-lang_v1.25/arch/amd64_ubuntu1404/share/doc/rust/html/book/first-edition/strings.html"))))}m.isMDXComponent=!0},7031:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""},6028:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""},1664:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""},8353:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""},8532:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""}}]); \ No newline at end of file diff --git a/assets/js/d8837387.e6bc9bd5.js b/assets/js/d8837387.e6bc9bd5.js deleted file mode 100644 index a30bbcfa..00000000 --- a/assets/js/d8837387.e6bc9bd5.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4622],{3905:(a,t,A)=>{A.d(t,{Zo:()=>g,kt:()=>b});var e=A(7294);function i(a,t,A){return t in a?Object.defineProperty(a,t,{value:A,enumerable:!0,configurable:!0,writable:!0}):a[t]=A,a}function n(a,t){var A=Object.keys(a);if(Object.getOwnPropertySymbols){var e=Object.getOwnPropertySymbols(a);t&&(e=e.filter((function(t){return Object.getOwnPropertyDescriptor(a,t).enumerable}))),A.push.apply(A,e)}return A}function l(a){for(var t=1;t=0||(i[A]=a[A]);return i}(a,t);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);for(e=0;e=0||Object.prototype.propertyIsEnumerable.call(a,A)&&(i[A]=a[A])}return i}var p=e.createContext({}),d=function(a){var t=e.useContext(p),A=t;return a&&(A="function"==typeof a?a(t):l(l({},t),a)),A},g=function(a){var t=d(a.components);return e.createElement(p.Provider,{value:t},a.children)},m={inlineCode:"code",wrapper:function(a){var t=a.children;return e.createElement(e.Fragment,{},t)}},k=e.forwardRef((function(a,t){var A=a.components,i=a.mdxType,n=a.originalType,p=a.parentName,g=r(a,["components","mdxType","originalType","parentName"]),k=d(A),b=i,c=k["".concat(p,".").concat(b)]||k[b]||m[b]||n;return A?e.createElement(c,l(l({ref:t},g),{},{components:A})):e.createElement(c,l({ref:t},g))}));function b(a,t){var A=arguments,i=t&&t.mdxType;if("string"==typeof a||i){var n=A.length,l=new Array(n);l[0]=k;var r={};for(var p in t)hasOwnProperty.call(t,p)&&(r[p]=t[p]);r.originalType=a,r.mdxType="string"==typeof a?a:i,l[1]=r;for(var d=2;d{A.r(t),A.d(t,{assets:()=>p,contentTitle:()=>l,default:()=>m,frontMatter:()=>n,metadata:()=>r,toc:()=>d});var e=A(7462),i=(A(7294),A(3905));const n={sidebar_position:6,title:"A.6. Tipe Data \u279c String Literal (&str)",sidebar_label:"A.6. Tipe Data \u279c String Literal (&str)"},l=void 0,r={unversionedId:"basic/tipe-data-string-literal",id:"basic/tipe-data-string-literal",title:"A.6. Tipe Data \u279c String Literal (&str)",description:"String adalah tipe data penting dalam pemrograman manapun. Biasanya tiap bahasa punya cara berbeda dalam meng-handle tipe ini. Di bahasa Rust, ada dua jenis tipe data string:",source:"@site/docs/basic/tipe-data-string-literal.md",sourceDirName:"basic",slug:"/basic/tipe-data-string-literal",permalink:"/basic/tipe-data-string-literal",draft:!1,tags:[],version:"current",sidebarPosition:6,frontMatter:{sidebar_position:6,title:"A.6. Tipe Data \u279c String Literal (&str)",sidebar_label:"A.6. Tipe Data \u279c String Literal (&str)"},sidebar:"tutorialSidebar",previous:{title:"A.5. Tipe Data \u279c Primitive Scalar",permalink:"/basic/tipe-data-primitive-scalar"},next:{title:"A.7. Konstanta",permalink:"/basic/konstanta"}},p={},d=[{value:"A.6.1. String literal atau &str",id:"a61-string-literal-atau-str",level:2},{value:"A.6.2. Escape karakter menggunakan ",id:"a62-escape-karakter-menggunakan-",level:2},{value:"A.6.3. Multiline string literal",id:"a63-multiline-string-literal",level:2},{value:"A.6.4. Raw string",id:"a64-raw-string",level:2},{value:"A.6.5. Pembahasan lanjutan tentang string",id:"a65-pembahasan-lanjutan-tentang-string",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],g={toc:d};function m(a){let{components:t,...n}=a;return(0,i.kt)("wrapper",(0,e.Z)({},g,n,{components:t,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"String adalah tipe data penting dalam pemrograman manapun. Biasanya tiap bahasa punya cara berbeda dalam meng-handle tipe ini. Di bahasa Rust, ada dua jenis tipe data string:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Tipe string literal (kadang disebut dengan string slice, atau ",(0,i.kt)("inlineCode",{parentName:"li"},"&str"),"). Tipe data ini ada pada nilai yang dideklarasikan dengan diapit tanda petik dua (string literal), contohnya ",(0,i.kt)("inlineCode",{parentName:"li"},'"Hello, world!"'),"."),(0,i.kt)("li",{parentName:"ul"},"Tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," yang merupakan tipe data custom atau ",(0,i.kt)("em",{parentName:"li"},"custom types")," yang merupakan sebuah struct. Lebih jelasnya akan kita bahas pada chapter terpisah.")),(0,i.kt)("p",null,"Chapter ini hanya fokus pada string literal, dengan level pembahasan yang tidak terlalu advance."),(0,i.kt)("h2",{id:"a61-string-literal-atau-str"},"A.6.1. String literal atau ",(0,i.kt)("inlineCode",{parentName:"h2"},"&str")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Kita tidak akan bahas terlalu low-level untuk topik ini, karena memang masih di chapter awal-awal.")),(0,i.kt)("p",null,"Rust adalah bahasa yang ",(0,i.kt)("em",{parentName:"p"},"statically typed"),", tipe data harus diketahui saat kompilasi. Setiap pendefinisian variabel, entah itu dengan di-specify tipe datanya (contoh: ",(0,i.kt)("inlineCode",{parentName:"p"},"let x: i32 = 5"),") atau menggunakan teknik ",(0,i.kt)("em",{parentName:"p"},"type inference")," (contoh: ",(0,i.kt)("inlineCode",{parentName:"p"},"let x = 5"),"), tipe datanya akan diketahui di awal saat kompilasi program."),(0,i.kt)("p",null,"Begitu juga pada tipe string. Sebagai contoh, statement ",(0,i.kt)("inlineCode",{parentName:"p"},'let y = "hello"'),", variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"y")," di sini memiliki tipe data, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"."),(0,i.kt)("p",null,"Apa itu tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),"? Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"str")," merupakan salah satu tipe primitif yang ada di Rust. Penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," menandakan bahwa tipe tersebut adalah ",(0,i.kt)("strong",{parentName:"p"},"pointer ",(0,i.kt)("inlineCode",{parentName:"strong"},"str")),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Untuk pembahasan lebih detail mengenai tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"&")," pada tipe data akan dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"))),(0,i.kt)("p",null,"Untuk kawan-kawan yang menggunakan VSCode dengan ekstensi ",(0,i.kt)("inlineCode",{parentName:"p"},"rust-analyzer"),", tipe data variabel bisa terlihat saat definisi."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rust string literal",src:A(7031).Z,width:"299",height:"146"})),(0,i.kt)("p",null,"Ok, sampai sini yang paling penting adalah dipahami dulu bahwa string literal tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"&str"),". Jadi selanjutnya kalau melihat tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," jangan bingung, itu berarti string."),(0,i.kt)("h2",{id:"a62-escape-karakter-menggunakan-"},"A.6.2. Escape karakter menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"\\")),(0,i.kt)("p",null,"Tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"\\")," digunakan untuk ",(0,i.kt)("em",{parentName:"p"},"escape")," beberapa karakter string."),(0,i.kt)("p",null,"Silakan perhatikan contoh berikut. Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"var2")," isinya adalah string yang di dalamnya ada beberapa karakter yang di escape, yaitu tanda ",(0,i.kt)("inlineCode",{parentName:"p"},'"')," dan baris baru atau newline."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var2 = "hello \\\n \\"rust\\" \\\n and \\\n \\"world\\"";\nprintln!("{}", var2);\n')),(0,i.kt)("p",null,"Coba jalankan kode di atas."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rust string literal",src:A(6028).Z,width:"445",height:"115"})),(0,i.kt)("p",null,"Karena semua baris baru pada contoh di atas di-escape, jadinya string akan tetap 1 baris. Tanda ",(0,i.kt)("inlineCode",{parentName:"p"},'"')," juga tetap muncul karena di-escape."),(0,i.kt)("h2",{id:"a63-multiline-string-literal"},"A.6.3. Multiline string literal"),(0,i.kt)("p",null,"Penulisan string banyak baris atau ",(0,i.kt)("em",{parentName:"p"},"multiline string")," adalah sama seperti penulisan string biasa, yaitu langsung tulis saja string dengan diapit tanda ",(0,i.kt)("inlineCode",{parentName:"p"},'"'),", tambahkan baris baru di dalam string tersebut sesuai kebutuhan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var3 = "baris satu\nbaris dua\nbaris tiga";\nprintln!("{}", var3);\n')),(0,i.kt)("p",null,"Hasilnya adalah sesuai dengan string yang sudah didefinisikan."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rust string literal",src:A(1664).Z,width:"452",height:"129"})),(0,i.kt)("p",null,"Perlu diketahui bahwa karakter spasi, baris baru, dan lainnya adalah ",(0,i.kt)("strong",{parentName:"p"},"tidak dihiraukan"),", jadi jika kawan-kawan menuliskan string multiline seperti ini ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var4 = "baris satu\n baris dua\n baris tiga";\nprintln!("{}", var4);\n')),(0,i.kt)("p",null,"... maka hasilnya juga sesuai dengan yang ditulis, yaitu ada 4 karakter spasi di baris ke-2 dan ke-3."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"rust string literal",src:A(8353).Z,width:"449",height:"130"})),(0,i.kt)("h2",{id:"a64-raw-string"},"A.6.4. Raw string"),(0,i.kt)("p",null,"Raw string adalah istilah untuk string yang tidak meng-",(0,i.kt)("em",{parentName:"p"},"escape")," karakter apapun. Di Rust, string literal bisa didefinisikan dengan menuliskan string diapit ",(0,i.kt)("inlineCode",{parentName:"p"},'r#"')," dan ",(0,i.kt)("inlineCode",{parentName:"p"},'"#'),". Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var5 = r#"\n {\n "name": "tim drake",\n "gender": "male"\n }\n"#;\nprintln!("{}", var5);\n')),(0,i.kt)("p",null,"Kode di atas hasilnya adalah ekuivalen dengan kode dibawah ini, yang dimana string didefinisikan dengan meng-escape karakter ",(0,i.kt)("inlineCode",{parentName:"p"},'"')," menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"\\"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let var6 = "\n {\n \\"name\\": \\"cassandra cain\\",\n \\"gender\\": \\"female\\"\n }\n";\nprintln!("{}", var6);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"raw string literal",src:A(8532).Z,width:"430",height:"260"})),(0,i.kt)("h2",{id:"a65-pembahasan-lanjutan-tentang-string"},"A.6.5. Pembahasan lanjutan tentang string"),(0,i.kt)("p",null,(0,i.kt)("strong",{parentName:"p"},"String adalah salah satu topik yang sangat luas cakupan pembahasannya"),", tidak cukup jika dirangkum dalam 1 chapter. Selain itu, bisa bikin makin bingung jika dibahas sekarang."),(0,i.kt)("p",null,"Penulis anjurkan untuk mempelajari chapter-chapter berikutnya secara urut terlebih dahulu."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/tipe_data_string_literal"},"github.com/novalagung/dasarpemrogramanrust-example/../tipe_data_string_literal")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pointer-references"},"Pointer & References")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/string-slice-vs-string-literal"},"String Literal (&str) vs. String Custom Type")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/tipe-data-custom-type-string-slice"},"Tipe Data \u279c String Custom Type"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/tokens.html#raw-string-literals"},"https://doc.rust-lang.org/reference/tokens.html#raw-string-literals")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/str/index.html"},"https://doc.rust-lang.org/std/str/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://web.mit.edu/rust-lang_v1.25/arch/amd64_ubuntu1404/share/doc/rust/html/book/first-edition/strings.html"},"https://web.mit.edu/rust-lang_v1.25/arch/amd64_ubuntu1404/share/doc/rust/html/book/first-edition/strings.html"))))}m.isMDXComponent=!0},7031:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""},6028:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""},1664:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""},8353:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""},8532:(a,t,A)=>{A.d(t,{Z:()=>e});const e=""}}]); \ No newline at end of file diff --git a/assets/js/dadaca00.64b4e275.js b/assets/js/dadaca00.64b4e275.js new file mode 100644 index 00000000..a4082ca1 --- /dev/null +++ b/assets/js/dadaca00.64b4e275.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8051],{3905:(a,e,n)=>{n.d(e,{Zo:()=>k,kt:()=>A});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function l(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var p=t.createContext({}),m=function(a){var e=t.useContext(p),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},k=function(a){var e=m(a.components);return t.createElement(p.Provider,{value:e},a.children)},u={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},s=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,r=a.originalType,p=a.parentName,k=d(a,["components","mdxType","originalType","parentName"]),s=m(n),A=i,o=s["".concat(p,".").concat(A)]||s[A]||u[A]||r;return n?t.createElement(o,l(l({ref:e},k),{},{components:n})):t.createElement(o,l({ref:e},k))}));function A(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=n.length,l=new Array(r);l[0]=s;var d={};for(var p in e)hasOwnProperty.call(e,p)&&(d[p]=e[p]);d.originalType=a,d.mdxType="string"==typeof a?a:i,l[1]=d;for(var m=2;m{n.r(e),n.d(e,{assets:()=>p,contentTitle:()=>l,default:()=>u,frontMatter:()=>r,metadata:()=>d,toc:()=>m});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:13,title:"A.13. Tipe Data \u279c Array",sidebar_label:"A.13. Array",description:"Belajar array pada rust. Contoh penerapan array, iterasi array, dan operasi lainnya pada array",keywords:["rust","belajar rust","array rust","slice rust"]},l=void 0,d={unversionedId:"basic/array",id:"basic/array",title:"A.13. Tipe Data \u279c Array",description:"Belajar array pada rust. Contoh penerapan array, iterasi array, dan operasi lainnya pada array",source:"@site/docs/basic/array.md",sourceDirName:"basic",slug:"/basic/array",permalink:"/basic/array",draft:!1,tags:[],version:"current",sidebarPosition:13,frontMatter:{sidebar_position:13,title:"A.13. Tipe Data \u279c Array",sidebar_label:"A.13. Array",description:"Belajar array pada rust. Contoh penerapan array, iterasi array, dan operasi lainnya pada array",keywords:["rust","belajar rust","array rust","slice rust"]},sidebar:"tutorialSidebar",previous:{title:"A.12. Perulangan \u279c for in",permalink:"/basic/perulangan-for-in"},next:{title:"A.14. Slice (Basic)",permalink:"/basic/slice"}},p={},m=[{value:"A.13.1. Pengenalan array",id:"a131-pengenalan-array",level:2},{value:"\u25c9 Deklarasi variabel array (type inference)",id:"-deklarasi-variabel-array-type-inference",level:3},{value:"\u25c9 Pengaksesan elemen array",id:"-pengaksesan-elemen-array",level:3},{value:"\u25c9 Mengubah isi elemen array",id:"-mengubah-isi-elemen-array",level:3},{value:"\u25c9 Formatted print {:?}",id:"-formatted-print-",level:3},{value:"\u25c9 Formatted print {namaVariabel:?}",id:"-formatted-print-namavariabel",level:3},{value:"A.13.2. Notasi penulisan tipe data array",id:"a132-notasi-penulisan-tipe-data-array",level:2},{value:"A.13.3. Macam-macam deklarasi array",id:"a133-macam-macam-deklarasi-array",level:2},{value:"\u25c9 Deklarasi array dengan metode type inference",id:"-deklarasi-array-dengan-metode-type-inference",level:3},{value:"\u25c9 Deklarasi array dengan metode manifest typing disertai predefined value",id:"-deklarasi-array-dengan-metode-manifest-typing-disertai-predefined-value",level:3},{value:"\u25c9 Deklarasi array dengan notasi penulisan [T; N]",id:"-deklarasi-array-dengan-notasi-penulisan-t-n",level:3},{value:"A.13.4. Melihat size array menggunakan method len",id:"a134-melihat-size-array-menggunakan-method-len",level:2},{value:"A.13.5. Iterasi array menggunakan for in",id:"a135-iterasi-array-menggunakan-for-in",level:2},{value:"A.13.6. Iterasi array menggunakan while dan loop",id:"a136-iterasi-array-menggunakan-while-dan-loop",level:2},{value:"A.13.7. Iterasi array menggunakan for in dan tuple",id:"a137-iterasi-array-menggunakan-for-in-dan-tuple",level:2},{value:"A.13.8. Append elemen ke array",id:"a138-append-elemen-ke-array",level:2},{value:"A.13.9. Nested array",id:"a139-nested-array",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],k={toc:m};function u(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,t.Z)({},k,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter awal kita sudah mempelajari tipe data primitif jenis scalar. Selain ",(0,i.kt)("em",{parentName:"p"},"scalar types")," ada juga tipe data primitif lainnya yaitu ",(0,i.kt)("strong",{parentName:"p"},"compound types"),". Compound types sendiri adalah jenis tipe data kolektif yang isinya banyak data. Kesemua data tersebut memiliki tipe data yang sama dan di-group menjadi satu."),(0,i.kt)("p",null,"Array adalah salah satu tipe data compound yang tersedia di Rust, dan pada chapter ini kita akan mempelajarinya."),(0,i.kt)("h2",{id:"a131-pengenalan-array"},"A.13.1. Pengenalan array"),(0,i.kt)("p",null,"Menurut dokumentasi official Rust, array adalah:"),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"An array is a collection of objects of the same type T, stored in contiguous memory."),(0,i.kt)("p",{parentName:"blockquote"},"A fixed-size array, denoted ","[T; N]",", for the element type, T, and the non-negative compile-time constant size, N.")),(0,i.kt)("p",null,"Array (atau ",(0,i.kt)("em",{parentName:"p"},"fixed size array"),") adalah kumpulan data dengan tipe sejenis, disimpan dalam 1 variabel. Array memiliki kapasitas yang nilainya ditentukan saat deklarasi/alokasi. Jumlah data dalam array pasti tidak boleh lebih dari kapasitas yang sudah ditentukan di awal. Data dalam array biasa disebut dengan ",(0,i.kt)("em",{parentName:"p"},"element")," atau item."),(0,i.kt)("p",null,"Ada beberapa notasi deklarasi array yang bisa dipakai, kesemuanya akan dibahas pada chapter ini."),(0,i.kt)("p",null,"Ok, Sekarang kita mulai praktik dengan bermain-main terlebih dahulu dengan tipe data array ini. Silakan pelajari dan praktikkan contoh berikut, sebuah program yang isinya menampilkan data array."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut numbers = [24, 12, 32, 7];\nprintln!("array {:?}", numbers);\n\nlet data0 = numbers[0];\nprintln!("elemen array ke 0 {data0}");\n\nlet data1 = numbers[1];\nprintln!("elemen array ke 1 {data1}");\n\nnumbers[1] = 16;\nnumbers[3] = 8;\nprintln!("array {numbers:?}");\n')),(0,i.kt)("p",null,"Coba jalankan aplikasi, muncul dua baris output di console."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array",src:n(3421).Z,width:"448",height:"164"})),(0,i.kt)("p",null,"Selanjutnya mari kita bahas dengan detail contoh di atas."),(0,i.kt)("h3",{id:"-deklarasi-variabel-array-type-inference"},"\u25c9 Deklarasi variabel array (",(0,i.kt)("em",{parentName:"h3"},"type inference"),")"),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," dideklarasikan sebagai array ",(0,i.kt)("em",{parentName:"p"},"mutable")," dengan metode deklarasi type inference, yang tipe datanya didapat langsung dari nilai."),(0,i.kt)("p",null,"Value dari ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"[24, 12, 32, 7]"),", yang mana artinya sebuah array dengan size 4, bertipe numerik, dengan isi ",(0,i.kt)("inlineCode",{parentName:"p"},"24"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"12"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"32"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"7"),"."),(0,i.kt)("p",null,"Sintaks ",(0,i.kt)("inlineCode",{parentName:"p"},"[24, 12, 32, 7]")," adalah salah satu cara menulis literal array. Tulis saja data yang diinginkan dengan separator ",(0,i.kt)("inlineCode",{parentName:"p"},",")," dan diapit tanda kurung siku ",(0,i.kt)("inlineCode",{parentName:"p"},"[ ]"),"."),(0,i.kt)("p",null,"Contoh lain deklarasi array:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut alphabets = ["a", "b", "c", "d"];\nlet booleans = [true, false];\nlet floatingNumbers = [32.0000078, 3.14, 0.5];\n')),(0,i.kt)("p",null,"O iya, penulisan literal array juga bisa dituliskan dalam multi-baris, contohnya seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut alphabets = [\n "a",\n "b",\n "c",\n "d"\n];\nlet booleans = [\n true,\n false\n];\n')),(0,i.kt)("h3",{id:"-pengaksesan-elemen-array"},"\u25c9 Pengaksesan elemen array"),(0,i.kt)("p",null,"Pengaksesan elemen array dilakukan dengan menuliskan nama variabel array kemudian diikuti kurung siku yang mengapit angka indeks elemen yang diinginkan."),(0,i.kt)("p",null,"Indeks array dimulai dari 0. Pada contoh di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," size-nya adalah 4 elemen, berarti elemen array ada pada indeks ke 0, 1, 2, 3."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data0 = numbers[0];\nprintln!("elemen array ke 0 {data0}");\n\nlet data1 = numbers[1];\nprintln!("elemen array ke 1 {data1}");\n')),(0,i.kt)("p",null,"Array hanya bisa diakses elemennya sesuai size saat deklarasi. Sebagai contoh, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," yang size nya 4, jika dipaksa mengakses elemen indeks ke-7 maka akan menghasilkan error."),(0,i.kt)("h3",{id:"-mengubah-isi-elemen-array"},"\u25c9 Mengubah isi elemen array"),(0,i.kt)("p",null,"Array bisa diubah isi elemen-nya jika variabel tersebut adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"mutable"),". Pada contoh yang sudah dibuat, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," dideklarasikan mutable dengan tipe data elemen adalah numerik. Dengan ini kita bisa mengubah value elemen array dengan value baru bertipe data sama."),(0,i.kt)("p",null,"Bisa dilihat pada contoh yang sudah dipraktikkan, elemen indeks ke-1 diubah nilainya menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"16"),", dan elemen indeks ke-3 value-nya menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"8")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"numbers[1] = 16;\nnumbers[3] = 8;\n")),(0,i.kt)("p",null,"Array hanya bisa diubah elemen-nya sesuai size saat deklarasi. Sebagai contoh, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," yang size nya 4, jika dipaksa mengakses dan/atau mengubah elemen indeks ke-7 maka akan menghasilkan error."),(0,i.kt)("p",null,"Selain itu, operasi assignment atau pengubahan nilai pada elemen array hanya bisa dilakukan dengan tipe data yang sama. Pada contoh di atas ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," adalah array bertipe numerik, karenanya tidak akan bisa diubah nilai elemen-nya dengan tipe lain, dan jika dipaksa akan menghasilkan error."),(0,i.kt)("h3",{id:"-formatted-print-"},"\u25c9 Formatted print ",(0,i.kt)("inlineCode",{parentName:"h3"},"{:?}")),(0,i.kt)("p",null,"Formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}")," berguna untuk memformat macam-macam tipe data ke bentuk string, yang salah satunya adalah tipe data array. Dengan menggunakan formatted print ini kita bisa menampilkan nilai elemen array dalam bentuk string."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'println!("array {:?}", numbers);\n')),(0,i.kt)("h3",{id:"-formatted-print-namavariabel"},"\u25c9 Formatted print ",(0,i.kt)("inlineCode",{parentName:"h3"},"{namaVariabel:?}")),(0,i.kt)("p",null,"Formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{namaVariabel:?}")," kegunaannya sama seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}"),", dengan perbedaan: pada penggunaan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{namaVariabel:?}")," tidak perlu menuliskan variabel yang ingin di-format sebagai argumen. Cukup ganti ",(0,i.kt)("inlineCode",{parentName:"p"},"namaVariabel")," dengan nama variabel yang ingin di-format."),(0,i.kt)("p",null,"Silakan lihat contoh berikut. Keduanya adalah ekuivalen, menghasilkan nilai yang sama."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'println!("array {:?}", numbers);\nprintln!("array {numbers:?}");\n')),(0,i.kt)("h2",{id:"a132-notasi-penulisan-tipe-data-array"},"A.13.2. Notasi penulisan tipe data array"),(0,i.kt)("p",null,"Pada contoh program yang sudah dipraktikkan, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"number")," tipe datanya adalah array numerik, terlihat saat deklarasinya ",(0,i.kt)("em",{parentName:"p"},"predefined value")," diisi dengan literal elemen array bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," di sini adalah milik elemen array, sedang array ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," itu sendiri tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"[i32; 4]"),", yang artinya adalah ",(0,i.kt)("strong",{parentName:"p"},"sebuah array dengan elemen bertipe ",(0,i.kt)("inlineCode",{parentName:"strong"},"i32")," dengan size ",(0,i.kt)("inlineCode",{parentName:"strong"},"4")),"."),(0,i.kt)("p",null,"Jika kawan-kawan menggunakan ekstensi VSCode ",(0,i.kt)("inlineCode",{parentName:"p"},"rust-analyzer"),", akan terlihat informasi tipe data array-nya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array",src:n(2957).Z,width:"445",height:"336"})),(0,i.kt)("h2",{id:"a133-macam-macam-deklarasi-array"},"A.13.3. Macam-macam deklarasi array"),(0,i.kt)("p",null,"Array lebih mudah dideklarasikan dengan metode ",(0,i.kt)("em",{parentName:"p"},"type inference"),". Namun tak menutup kemungkinan ada kebutuhan di mana array harus dideklarasikan dengan menuliskan tipe datanya secara eksplisit. Berikut adalah macam-macam cara mendeklarasikan array."),(0,i.kt)("h3",{id:"-deklarasi-array-dengan-metode-type-inference"},"\u25c9 Deklarasi array dengan metode ",(0,i.kt)("em",{parentName:"h3"},"type inference")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let angka_integer = [24, 12, 32, 7];\nprintln!("{angka_integer:?}");\n// output: [24, 12, 32, 7]\n\nlet angka_float = [24.2, 12.5, 32.00002, 7.2];\nprintln!("{angka_float:?}");\n// output: [24.2, 12.5, 32.00002, 7.2]\n')),(0,i.kt)("h3",{id:"-deklarasi-array-dengan-metode-manifest-typing-disertai-predefined-value"},"\u25c9 Deklarasi array dengan metode ",(0,i.kt)("em",{parentName:"h3"},"manifest typing")," disertai ",(0,i.kt)("em",{parentName:"h3"},"predefined value")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_boolean: [bool; 2] = [false, true];\nprintln!("{data_boolean:?}");\n// output: [false, true]\n\nlet angka_unsigned_integer: [u32; 3] = [24, 0, 12];\nprintln!("{angka_unsigned_integer:?}");\n// output: [24, 0, 12]\n')),(0,i.kt)("h3",{id:"-deklarasi-array-dengan-notasi-penulisan-t-n"},"\u25c9 Deklarasi array dengan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"h3"},"[T; N]")),(0,i.kt)("p",null,"Pada contoh berikut, ",(0,i.kt)("inlineCode",{parentName:"p"},"data_numerik1")," dideklarasikan bertipe array dengan tipe data data elemen adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", mempunyai size ",(0,i.kt)("inlineCode",{parentName:"p"},"10"),", dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value")," untuk setiap elemen array adalah angka ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_numerik1: [i32; 10] = [0; 10];\nprintln!("{data_numerik1:?}");\n// output: [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]\n')),(0,i.kt)("p",null,"Pada contoh ke-dua ini, ",(0,i.kt)("inlineCode",{parentName:"p"},"data_numerik2")," dideklarasikan dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value")," adalah array yang size-nya ",(0,i.kt)("inlineCode",{parentName:"p"},"5")," dan nilai default tiap elemen adalah angka ",(0,i.kt)("inlineCode",{parentName:"p"},"4"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_numerik2 = [4; 5];\nprintln!("{data_numerik2:?}");\n// output: [4, 4, 4, 4, 4]\n')),(0,i.kt)("p",null,"Tambahan penjelasan mengenai notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"[T; N]"),":"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Jika digunakan pada penulisan tipe data array saat deklarasi, ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," adalah tipe data elemen, dan ",(0,i.kt)("inlineCode",{parentName:"li"},"N")," adalah lebar/size array. Contoh: ",(0,i.kt)("inlineCode",{parentName:"li"},"let data_numerik1: [i32; 10]"),"."),(0,i.kt)("li",{parentName:"ul"},"Jika digunakan pada penulisan ",(0,i.kt)("em",{parentName:"li"},"predefined value"),", ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," adalah nilai setiap elemen array, dan ",(0,i.kt)("inlineCode",{parentName:"li"},"N")," adalah lebar/size array. Contoh: ",(0,i.kt)("inlineCode",{parentName:"li"},"[4; 2]"),", yang artinya semua elemen array diisi dengan angka ",(0,i.kt)("inlineCode",{parentName:"li"},"4"),".")),(0,i.kt)("h2",{id:"a134-melihat-size-array-menggunakan-method-len"},"A.13.4. Melihat size array menggunakan method ",(0,i.kt)("inlineCode",{parentName:"h2"},"len")),(0,i.kt)("p",null,"Array secara ",(0,i.kt)("em",{parentName:"p"},"default")," properti tipe data ",(0,i.kt)("strong",{parentName:"p"},"slice"),", yang salah satunya adalah method bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"len")," yang berguna untuk melihat size dari sebuah array. Cara penggunaannya cukup dengan menuliskan ",(0,i.kt)("inlineCode",{parentName:"p"},".len()")," setelah variabel array. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names = ["jason", "grayon", "drake", "damian"];\nlet length = names.len();\nprintln!("array size is {}", length);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array for in",src:n(7674).Z,width:"449",height:"113"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai slice dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/slice"},"Slice"))),(0,i.kt)("h2",{id:"a135-iterasi-array-menggunakan-for-in"},"A.13.5. Iterasi array menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"for in")),(0,i.kt)("p",null,"Pada contoh berikut, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"names")," adalah array bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"[&str; 4]"),", dideklarasikan dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value"),". Elemen array tersebut kemudian diiterasi menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," untuk di-print ke layar console value setiap elemennya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\nfor name in names {\n println!("{name}");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array for in",src:n(2977).Z,width:"445",height:"149"})),(0,i.kt)("p",null,"Bisa juga dengan memanfaatkan ",(0,i.kt)("em",{parentName:"p"},"range syntax")," dalam perulangan array. Benefitnya adalah pengaksesan indeks array lebih mudah."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\nfor i in 0..names.len() {\n println!("array index ke-{}: {}", i, names[i]);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array for in",src:n(4097).Z,width:"447",height:"148"})),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"names.len()")," mengembalikan informasi size sebuah array, nilainya bisa dimanfaatkan dalam perulangan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," untuk iterasi indeks ke-0 hingga terakhir."),(0,i.kt)("h2",{id:"a136-iterasi-array-menggunakan-while-dan-loop"},"A.13.6. Iterasi array menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"while")," dan ",(0,i.kt)("inlineCode",{parentName:"h2"},"loop")),(0,i.kt)("p",null,"Contoh iterasi array menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"while"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\n\nlet mut i = 0;\nwhile i < names.len() {\n println!("array index ke-{}: {}", i, names[i]);\n i += 1;\n}\n')),(0,i.kt)("p",null,"Contoh iterasi array menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"loop"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\n\nlet mut i = 0;\nloop {\n if i >= names.len() {\n break;\n }\n\n println!("array index ke-{}: {}", i, names[i]);\n i += 1;\n}\n')),(0,i.kt)("p",null,"Iterasi array menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"loop")," umumnya kurang praktis jika dibandingkan dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),". Tapi pastinya ada case di mana ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," dan/atau ",(0,i.kt)("inlineCode",{parentName:"p"},"loop")," akan dibutuhkan."),(0,i.kt)("h2",{id:"a137-iterasi-array-menggunakan-for-in-dan-tuple"},"A.13.7. Iterasi array menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"for in")," dan ",(0,i.kt)("em",{parentName:"h2"},"tuple")),(0,i.kt)("p",null,"Mengiterasi value sekaligus ",(0,i.kt)("em",{parentName:"p"},"counter")," perulangan sebenarnya bisa juga dilakukan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", contohnya seperti yang sudah kita praktikkan di atas yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"for i in 0..names.len()"),"."),(0,i.kt)("p",null,"Ada juga bentuk lain pemanfaatan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," untuk mengiterasi sebuah array. Caranya dengan menggunakan teknik tuple untuk menampung data ",(0,i.kt)("em",{parentName:"p"},"counter")," sekaligus value tiap elemen. Lebih jelasnya silakan lihat dan praktikkan contoh berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\n\nfor (i, name) in names.iter().enumerate() {\n println!("array index ke-{i}: {name}");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array enumerate",src:n(4097).Z,width:"447",height:"148"})),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"names")," yang notabene bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"[&str; 4]")," perlu dikonversi ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," terlebih dahulu caranya lewat pemanggilan method ",(0,i.kt)("inlineCode",{parentName:"p"},".iter()"),". Kemudian dari tipe tersebut perlu dikonversi lagi ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Enumerate")," dengan cara memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},".enumerate()"),"."),(0,i.kt)("p",null,"Setelah mendapatkan objek bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Enumerate"),", keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," digunakan untuk menampung tiap elemen array dalam bentuk ",(0,i.kt)("em",{parentName:"p"},"tuple")," ",(0,i.kt)("inlineCode",{parentName:"p"},"(i, name)"),". Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," di situ berisi counter iterasi, dan ",(0,i.kt)("inlineCode",{parentName:"p"},"name")," adalah value-nya."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai ",(0,i.kt)("inlineCode",{parentName:"li"},"Enumerate")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/trait-iterator"},"Trait \u279c Iterator")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai tuple dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/tuple"},"Tuple")))),(0,i.kt)("h2",{id:"a138-append-elemen-ke-array"},"A.13.8. Append elemen ke array"),(0,i.kt)("p",null,"Operasi menambahkan sebuah elemen ke array yang hasilnya melebihi kapasitas ... adalah tidak bisa. Karena array memiliki size fixed, tidak dinamis. Solusinya adalah menggunakan tipe data Vector. Nantinya array perlu dikonversi ke bentu Vector terlebih dahulu kemudian di-append, lebih jelasnya kita bahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector"),"."),(0,i.kt)("h2",{id:"a139-nested-array"},"A.13.9. Nested array"),(0,i.kt)("p",null,"Data nested array bisa dibuat dengan level kedalaman tanpa batas, tetapi harus mengikuti aturan tipe data array yaitu: fixed size dan elemen bertipe data sejenis."),(0,i.kt)("p",null,"Pada contoh berikut variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_arr")," didefinisikan sebagai sebuah array bersarang atau nested dengan kedalaman 2 level."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_arr = [\n ["salad", "fried rice"],\n ["apple", "coconut"],\n ["spinach", "jalapeno"],\n];\nfor sub_arr in data_arr {\n for el in sub_arr {\n print!("{el}, ");\n }\n println!();\n}\n\n// salad, fried rice,\n// apple, coconut,\n// spinach, jalapeno,\n')),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_arr")," pada contoh di atas bertipe data",(0,i.kt)("inlineCode",{parentName:"p"},"[[&str; 2]; 3]"),", yang artinya adalah sebuah array dengan size 3, dengan isi elemen adalah juga array dengan size 2. Selalu ingat bahwa size array adalah fixed."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/array"},"github.com/novalagung/dasarpemrogramanrust-example/../array")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice"},"Slice (Basic)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/vector"},"Tipe Data \u279c Vector")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice-memory-management"},"Slice Memory Management"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.array.html"},"https://doc.rust-lang.org/std/primitive.array.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.slice.html"},"https://doc.rust-lang.org/std/primitive.slice.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/index.html"},"https://doc.rust-lang.org/std/iter/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/struct.Enumerate.html"},"https://doc.rust-lang.org/std/iter/struct.Enumerate.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/trait.Iterator.html"},"https://doc.rust-lang.org/std/iter/trait.Iterator.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/trait/iter.html"},"https://doc.rust-lang.org/rust-by-example/trait/iter.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/for.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/for.html"))))}u.isMDXComponent=!0},3421:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/array-1-919358da6705a5ced706fbcd108be794.png"},2957:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/array-2-bc70d4683afb247a4d30a6551aabec0a.png"},7674:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},4097:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},2977:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/dadaca00.d3047a10.js b/assets/js/dadaca00.d3047a10.js deleted file mode 100644 index ae2682be..00000000 --- a/assets/js/dadaca00.d3047a10.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8051],{3905:(a,e,n)=>{n.d(e,{Zo:()=>k,kt:()=>A});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function l(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var p=t.createContext({}),m=function(a){var e=t.useContext(p),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},k=function(a){var e=m(a.components);return t.createElement(p.Provider,{value:e},a.children)},u={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},s=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,r=a.originalType,p=a.parentName,k=d(a,["components","mdxType","originalType","parentName"]),s=m(n),A=i,o=s["".concat(p,".").concat(A)]||s[A]||u[A]||r;return n?t.createElement(o,l(l({ref:e},k),{},{components:n})):t.createElement(o,l({ref:e},k))}));function A(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=n.length,l=new Array(r);l[0]=s;var d={};for(var p in e)hasOwnProperty.call(e,p)&&(d[p]=e[p]);d.originalType=a,d.mdxType="string"==typeof a?a:i,l[1]=d;for(var m=2;m{n.r(e),n.d(e,{assets:()=>p,contentTitle:()=>l,default:()=>u,frontMatter:()=>r,metadata:()=>d,toc:()=>m});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:13,title:"A.13. Tipe Data \u279c Array",sidebar_label:"A.13. Array",description:"Belajar array pada rust. Contoh penerapan array, iterasi array, dan operasi lainnya pada array",keywords:["rust","belajar rust","array rust","slice rust"]},l=void 0,d={unversionedId:"basic/array",id:"basic/array",title:"A.13. Tipe Data \u279c Array",description:"Belajar array pada rust. Contoh penerapan array, iterasi array, dan operasi lainnya pada array",source:"@site/docs/basic/array.md",sourceDirName:"basic",slug:"/basic/array",permalink:"/basic/array",draft:!1,tags:[],version:"current",sidebarPosition:13,frontMatter:{sidebar_position:13,title:"A.13. Tipe Data \u279c Array",sidebar_label:"A.13. Array",description:"Belajar array pada rust. Contoh penerapan array, iterasi array, dan operasi lainnya pada array",keywords:["rust","belajar rust","array rust","slice rust"]},sidebar:"tutorialSidebar",previous:{title:"A.12. Perulangan \u279c for in",permalink:"/basic/perulangan-for-in"},next:{title:"A.14. Slice (Basic)",permalink:"/basic/slice"}},p={},m=[{value:"A.13.1. Pengenalan array",id:"a131-pengenalan-array",level:2},{value:"\u25c9 Deklarasi variabel array (type inference)",id:"-deklarasi-variabel-array-type-inference",level:3},{value:"\u25c9 Pengaksesan elemen array",id:"-pengaksesan-elemen-array",level:3},{value:"\u25c9 Mengubah isi elemen array",id:"-mengubah-isi-elemen-array",level:3},{value:"\u25c9 Formatted print {:?}",id:"-formatted-print-",level:3},{value:"\u25c9 Formatted print {namaVariabel:?}",id:"-formatted-print-namavariabel",level:3},{value:"A.13.2. Notasi penulisan tipe data array",id:"a132-notasi-penulisan-tipe-data-array",level:2},{value:"A.13.3. Macam-macam deklarasi array",id:"a133-macam-macam-deklarasi-array",level:2},{value:"\u25c9 Deklarasi array dengan metode type inference",id:"-deklarasi-array-dengan-metode-type-inference",level:3},{value:"\u25c9 Deklarasi array dengan metode manifest typing disertai predefined value",id:"-deklarasi-array-dengan-metode-manifest-typing-disertai-predefined-value",level:3},{value:"\u25c9 Deklarasi array dengan notasi penulisan [T; N]",id:"-deklarasi-array-dengan-notasi-penulisan-t-n",level:3},{value:"A.13.4. Melihat size array menggunakan method len",id:"a134-melihat-size-array-menggunakan-method-len",level:2},{value:"A.13.5. Iterasi array menggunakan for in",id:"a135-iterasi-array-menggunakan-for-in",level:2},{value:"A.13.6. Iterasi array menggunakan while dan loop",id:"a136-iterasi-array-menggunakan-while-dan-loop",level:2},{value:"A.13.7. Iterasi array menggunakan for in dan tuple",id:"a137-iterasi-array-menggunakan-for-in-dan-tuple",level:2},{value:"A.13.8. Append elemen ke array",id:"a138-append-elemen-ke-array",level:2},{value:"A.13.9. Nested array",id:"a139-nested-array",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],k={toc:m};function u(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,t.Z)({},k,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter awal kita sudah mempelajari tipe data primitif jenis scalas. Selain ",(0,i.kt)("em",{parentName:"p"},"scalar types")," ada juga tipe data primitif lainnya yaitu ",(0,i.kt)("strong",{parentName:"p"},"compound types"),". Compound types sendiri adalah jenis tipe data kolektif yang isinya banyak data. Kesemua data tersebut memiliki tipe data yang sama dan di-group menjadi satu."),(0,i.kt)("p",null,"Array adalah salah satu tipe data compound yang tersedia di Rust, dan pada chapter ini kita akan mempelajarinya."),(0,i.kt)("h2",{id:"a131-pengenalan-array"},"A.13.1. Pengenalan array"),(0,i.kt)("p",null,"Menurut dokumentasi official Rust, array adalah:"),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"An array is a collection of objects of the same type T, stored in contiguous memory."),(0,i.kt)("p",{parentName:"blockquote"},"A fixed-size array, denoted ","[T; N]",", for the element type, T, and the non-negative compile-time constant size, N.")),(0,i.kt)("p",null,"Array (atau ",(0,i.kt)("em",{parentName:"p"},"fixed size array"),") adalah kumpulan data dengan tipe sejenis, disimpan dalam 1 variabel. Array memiliki kapasitas yang nilainya ditentukan saat deklarasi/alokasi. Jumlah data dalam array pasti tidak boleh lebih dari kapasitas yang sudah ditentukan di awal. Data dalam array biasa disebut dengan ",(0,i.kt)("em",{parentName:"p"},"element")," atau item."),(0,i.kt)("p",null,"Ada beberapa notasi deklarasi array yang bisa dipakai, kesemuanya akan dibahas pada chapter ini."),(0,i.kt)("p",null,"Ok, Sekarang kita mulai praktik dengan bermain-main terlebih dahulu dengan tipe data array ini. Silakan pelajari dan praktikkan contoh berikut, sebuah program yang isinya menampilkan data array."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut numbers = [24, 12, 32, 7];\nprintln!("array {:?}", numbers);\n\nlet data0 = numbers[0];\nprintln!("elemen array ke 0 {data0}");\n\nlet data1 = numbers[1];\nprintln!("elemen array ke 1 {data1}");\n\nnumbers[1] = 16;\nnumbers[3] = 8;\nprintln!("array {numbers:?}");\n')),(0,i.kt)("p",null,"Coba jalankan aplikasi, muncul dua baris output di console."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array",src:n(3421).Z,width:"448",height:"164"})),(0,i.kt)("p",null,"Selanjutnya mari kita bahas dengan detail contoh di atas."),(0,i.kt)("h3",{id:"-deklarasi-variabel-array-type-inference"},"\u25c9 Deklarasi variabel array (",(0,i.kt)("em",{parentName:"h3"},"type inference"),")"),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," dideklarasikan sebagai array ",(0,i.kt)("em",{parentName:"p"},"mutable")," dengan metode deklarasi type inference, yang tipe datanya didapat langsung dari nilai."),(0,i.kt)("p",null,"Value dari ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"[24, 12, 32, 7]"),", yang dimana artinya sebuah array dengan size 4, bertipe numerik, dengan isi ",(0,i.kt)("inlineCode",{parentName:"p"},"24"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"12"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"32"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"7"),"."),(0,i.kt)("p",null,"Sintaks ",(0,i.kt)("inlineCode",{parentName:"p"},"[24, 12, 32, 7]")," adalah salah satu cara menulis literal array. Tulis saja data yang diinginkan dengan separator ",(0,i.kt)("inlineCode",{parentName:"p"},",")," dan diapit tanda kurung siku ",(0,i.kt)("inlineCode",{parentName:"p"},"[ ]"),"."),(0,i.kt)("p",null,"Contoh lain deklarasi array:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut alphabets = ["a", "b", "c", "d"];\nlet booleans = [true, false];\nlet floatingNumbers = [32.0000078, 3.14, 0.5];\n')),(0,i.kt)("p",null,"O iya, penulisan literal array juga bisa dituliskan dalam multi-baris, contohnya seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut alphabets = [\n "a",\n "b",\n "c",\n "d"\n];\nlet booleans = [\n true,\n false\n];\n')),(0,i.kt)("h3",{id:"-pengaksesan-elemen-array"},"\u25c9 Pengaksesan elemen array"),(0,i.kt)("p",null,"Pengaksesan elemen array dilakukan dengan menuliskan nama variabel array kemudian diikuti kurung siku yang mengapit angka indeks elemen yang diinginkan."),(0,i.kt)("p",null,"Indeks array dimulai dari 0. Pada contoh di atas, ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," size-nya adalah 4 elemen, berarti elemen array ada pada indeks ke 0, 1, 2, 3."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data0 = numbers[0];\nprintln!("elemen array ke 0 {data0}");\n\nlet data1 = numbers[1];\nprintln!("elemen array ke 1 {data1}");\n')),(0,i.kt)("p",null,"Array hanya bisa diakses elemennya sesuai size saat deklarasi. Sebagai contoh, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," yang size nya 4, jika dipaksa mengakses elemen indeks ke-7 maka akan menghasilkan error."),(0,i.kt)("h3",{id:"-mengubah-isi-elemen-array"},"\u25c9 Mengubah isi elemen array"),(0,i.kt)("p",null,"Array bisa diubah isi elemen-nya jika variabel tersebut adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"mutable"),". Pada contoh yang sudah dibuat, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," dideklarasikan mutable dengan tipe data elemen adalah numerik. Dengan ini kita bisa mengubah value elemen array dengan value baru bertipe data sama."),(0,i.kt)("p",null,"Bisa dilihat pada contoh yang sudah dipraktikkan, elemen indeks ke-1 diubah nilainya menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"16"),", dan elemen indeks ke-3 value-nya menjadi ",(0,i.kt)("inlineCode",{parentName:"p"},"8")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"numbers[1] = 16;\nnumbers[3] = 8;\n")),(0,i.kt)("p",null,"Array hanya bisa diubah elemen-nya sesuai size saat deklarasi. Sebagai contoh, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," yang size nya 4, jika dipaksa mengakses dan/atau mengubah elemen indeks ke-7 maka akan menghasilkan error."),(0,i.kt)("p",null,"Selain itu, operasi assignment atau pengubahan nilai pada elemen array hanya bisa dilakukan dengan tipe data yang sama. Pada contoh di atas ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," adalah array bertipe numerik, karenanya tidak akan bisa diubah nilai elemen-nya dengan tipe lain, dan jika dipaksa akan menghasilkan error."),(0,i.kt)("h3",{id:"-formatted-print-"},"\u25c9 Formatted print ",(0,i.kt)("inlineCode",{parentName:"h3"},"{:?}")),(0,i.kt)("p",null,"Formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}")," berguna untuk memformat macam-macam tipe data ke bentuk string, yang salah satunya adalah tipe data array. Dengan menggunakan formatted print ini kita bisa menampilkan nilai elemen array dalam bentuk string."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'println!("array {:?}", numbers);\n')),(0,i.kt)("h3",{id:"-formatted-print-namavariabel"},"\u25c9 Formatted print ",(0,i.kt)("inlineCode",{parentName:"h3"},"{namaVariabel:?}")),(0,i.kt)("p",null,"Formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{namaVariabel:?}")," kegunaannya sama seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}"),", dengan perbedaan: pada penggunaan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{namaVariabel:?}")," tidak perlu menuliskan variabel yang ingin di-format sebagai argumen. Cukup ganti ",(0,i.kt)("inlineCode",{parentName:"p"},"namaVariabel")," dengan nama variabel yang ingin di-format."),(0,i.kt)("p",null,"Silakan lihat contoh berikut. Keduanya adalah ekuivalen, menghasilkan nilai yang sama."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'println!("array {:?}", numbers);\nprintln!("array {numbers:?}");\n')),(0,i.kt)("h2",{id:"a132-notasi-penulisan-tipe-data-array"},"A.13.2. Notasi penulisan tipe data array"),(0,i.kt)("p",null,"Pada contoh program yang sudah dipraktikkan, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"number")," tipe datanya adalah array numerik, terlihat saat deklarasinya ",(0,i.kt)("em",{parentName:"p"},"predefined value")," diisi dengan literal elemen array bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," di sini adalah milik elemen array, sedang array ",(0,i.kt)("inlineCode",{parentName:"p"},"numbers")," itu sendiri tipe datanya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"[i32; 4]"),", yang artinya adalah ",(0,i.kt)("strong",{parentName:"p"},"sebuah array dengan elemen bertipe ",(0,i.kt)("inlineCode",{parentName:"strong"},"i32")," dengan size ",(0,i.kt)("inlineCode",{parentName:"strong"},"4")),"."),(0,i.kt)("p",null,"Jika kawan-kawan menggunakan ekstensi VSCode ",(0,i.kt)("inlineCode",{parentName:"p"},"rust-analyzer"),", akan terlihat informasi tipe data array-nya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array",src:n(2957).Z,width:"445",height:"336"})),(0,i.kt)("h2",{id:"a133-macam-macam-deklarasi-array"},"A.13.3. Macam-macam deklarasi array"),(0,i.kt)("p",null,"Array lebih mudah dideklarasikan dengan metode ",(0,i.kt)("em",{parentName:"p"},"type inference"),". Namun tak menutup kemungkinan ada kebutuhan dimana array harus dideklarasikan dengan menuliskan tipe datanya secara eksplisit. Berikut adalah macam-macam cara mendeklarasikan array."),(0,i.kt)("h3",{id:"-deklarasi-array-dengan-metode-type-inference"},"\u25c9 Deklarasi array dengan metode ",(0,i.kt)("em",{parentName:"h3"},"type inference")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let angka_integer = [24, 12, 32, 7];\nprintln!("{angka_integer:?}");\n// output: [24, 12, 32, 7]\n\nlet angka_float = [24.2, 12.5, 32.00002, 7.2];\nprintln!("{angka_float:?}");\n// output: [24.2, 12.5, 32.00002, 7.2]\n')),(0,i.kt)("h3",{id:"-deklarasi-array-dengan-metode-manifest-typing-disertai-predefined-value"},"\u25c9 Deklarasi array dengan metode ",(0,i.kt)("em",{parentName:"h3"},"manifest typing")," disertai ",(0,i.kt)("em",{parentName:"h3"},"predefined value")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_boolean: [bool; 2] = [false, true];\nprintln!("{data_boolean:?}");\n// output: [false, true]\n\nlet angka_unsigned_integer: [u32; 3] = [24, 0, 12];\nprintln!("{angka_unsigned_integer:?}");\n// output: [24, 0, 12]\n')),(0,i.kt)("h3",{id:"-deklarasi-array-dengan-notasi-penulisan-t-n"},"\u25c9 Deklarasi array dengan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"h3"},"[T; N]")),(0,i.kt)("p",null,"Pada contoh berikut, ",(0,i.kt)("inlineCode",{parentName:"p"},"data_numerik1")," dideklarasikan bertipe array dengan tipe data data elemen adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", mempunyai size ",(0,i.kt)("inlineCode",{parentName:"p"},"10"),", dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value")," untuk setiap elemen array adalah angka ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_numerik1: [i32; 10] = [0; 10];\nprintln!("{data_numerik1:?}");\n// output: [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]\n')),(0,i.kt)("p",null,"Pada contoh ke-dua ini, ",(0,i.kt)("inlineCode",{parentName:"p"},"data_numerik2")," dideklarasikan dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value")," adalah array yang size-nya ",(0,i.kt)("inlineCode",{parentName:"p"},"5")," dan nilai default tiap elemen adalah angka ",(0,i.kt)("inlineCode",{parentName:"p"},"4"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_numerik2 = [4; 5];\nprintln!("{data_numerik2:?}");\n// output: [4, 4, 4, 4, 4]\n')),(0,i.kt)("p",null,"Tambahan penjelasan mengenai notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"[T; N]"),":"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Jika digunakan pada penulisan tipe data array saat deklarasi, ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," adalah tipe data elemen, dan ",(0,i.kt)("inlineCode",{parentName:"li"},"N")," adalah lebar/size array. Contoh: ",(0,i.kt)("inlineCode",{parentName:"li"},"let data_numerik1: [i32; 10]"),"."),(0,i.kt)("li",{parentName:"ul"},"Jika digunakan pada penulisan ",(0,i.kt)("em",{parentName:"li"},"predefined value"),", ",(0,i.kt)("inlineCode",{parentName:"li"},"T")," adalah nilai setiap elemen array, dan ",(0,i.kt)("inlineCode",{parentName:"li"},"N")," adalah lebar/size array. Contoh: ",(0,i.kt)("inlineCode",{parentName:"li"},"[4; 2]"),", yang artinya semua elemen array diisi dengan angka ",(0,i.kt)("inlineCode",{parentName:"li"},"4"),".")),(0,i.kt)("h2",{id:"a134-melihat-size-array-menggunakan-method-len"},"A.13.4. Melihat size array menggunakan method ",(0,i.kt)("inlineCode",{parentName:"h2"},"len")),(0,i.kt)("p",null,"Array secara ",(0,i.kt)("em",{parentName:"p"},"default")," properti tipe data ",(0,i.kt)("strong",{parentName:"p"},"slice"),", yang salah satunya adalah method bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"len")," yang berguna untuk melihat size dari sebuah array. Cara penggunaannya cukup dengan menuliskan ",(0,i.kt)("inlineCode",{parentName:"p"},".len()")," setelah variabel array. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names = ["jason", "grayon", "drake", "damian"];\nlet length = names.len();\nprintln!("array size is {}", length);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array for in",src:n(7674).Z,width:"449",height:"113"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai slice dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/slice"},"Slice"))),(0,i.kt)("h2",{id:"a135-iterasi-array-menggunakan-for-in"},"A.13.5. Iterasi array menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"for in")),(0,i.kt)("p",null,"Pada contoh berikut, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"names")," adalah array bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"[&str; 4]"),", dideklarasikan dengan ",(0,i.kt)("em",{parentName:"p"},"predefined value"),". Elemen array tersebut kemudian diiterasi menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," untuk di-print ke layar console value setiap elemennya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\nfor name in names {\n println!("{name}");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array for in",src:n(2977).Z,width:"445",height:"149"})),(0,i.kt)("p",null,"Bisa juga dengan memanfaatkan ",(0,i.kt)("em",{parentName:"p"},"range syntax")," dalam perulangan array. Benefitnya adalah pengaksesan indeks array lebih mudah."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\nfor i in 0..names.len() {\n println!("array index ke-{}: {}", i, names[i]);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array for in",src:n(4097).Z,width:"447",height:"148"})),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"names.len()")," mengembalikan informasi size sebuah array, nilainya bisa dimanfaatkan dalam perulangan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," untuk iterasi indeks ke-0 hingga terakhir."),(0,i.kt)("h2",{id:"a136-iterasi-array-menggunakan-while-dan-loop"},"A.13.6. Iterasi array menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"while")," dan ",(0,i.kt)("inlineCode",{parentName:"h2"},"loop")),(0,i.kt)("p",null,"Contoh iterasi array menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"while"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\n\nlet mut i = 0;\nwhile i < names.len() {\n println!("array index ke-{}: {}", i, names[i]);\n i += 1;\n}\n')),(0,i.kt)("p",null,"Contoh iterasi array menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"loop"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\n\nlet mut i = 0;\nloop {\n if i >= names.len() {\n break;\n }\n\n println!("array index ke-{}: {}", i, names[i]);\n i += 1;\n}\n')),(0,i.kt)("p",null,"Iterasi array menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"loop")," umumnya kurang praktis jika dibandingkan dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),". Tapi pastinya ada case dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"while")," dan/atau ",(0,i.kt)("inlineCode",{parentName:"p"},"loop")," akan dibutuhkan."),(0,i.kt)("h2",{id:"a137-iterasi-array-menggunakan-for-in-dan-tuple"},"A.13.7. Iterasi array menggunakan ",(0,i.kt)("inlineCode",{parentName:"h2"},"for in")," dan ",(0,i.kt)("em",{parentName:"h2"},"tuple")),(0,i.kt)("p",null,"Mengiterasi value sekaligus ",(0,i.kt)("em",{parentName:"p"},"counter")," perulangan sebenarnya bisa juga dilakukan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in"),", contohnya seperti yang sudah kita praktikkan di atas yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"for i in 0..names.len()"),"."),(0,i.kt)("p",null,"Ada juga bentuk lain pemanfaatan ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," untuk mengiterasi sebuah array. Caranya dengan menggunakan teknik tuple untuk menampung data ",(0,i.kt)("em",{parentName:"p"},"counter")," sekaligus value tiap elemen. Lebih jelasnya silakan lihat dan praktikkan contoh berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let names: [&str; 4] = ["jason", "grayon", "drake", "damian"];\n\nfor (i, name) in names.iter().enumerate() {\n println!("array index ke-{i}: {name}");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"array enumerate",src:n(4097).Z,width:"447",height:"148"})),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"names")," yang notabene bertipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"[&str; 4]")," perlu dikonversi ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Iterator")," terlebih dahulu caranya lewat pemanggilan method ",(0,i.kt)("inlineCode",{parentName:"p"},".iter()"),". Kemudian dari tipe tersebut perlu dikonversi lagi ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Enumerate")," dengan cara memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},".enumerate()"),"."),(0,i.kt)("p",null,"Setelah mendapatkan objek bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Enumerate"),", keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," digunakan untuk menampung tiap elemen array dalam bentuk ",(0,i.kt)("em",{parentName:"p"},"tuple")," ",(0,i.kt)("inlineCode",{parentName:"p"},"(i, name)"),". Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"i")," disitu berisi counter iterasi, dan ",(0,i.kt)("inlineCode",{parentName:"p"},"name")," adalah value-nya."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai traits dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/traits"},"Traits")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai ",(0,i.kt)("inlineCode",{parentName:"li"},"Enumerate")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/trait-iterator"},"Trait \u279c Iterator")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai tuple dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/tuple"},"Tuple")))),(0,i.kt)("h2",{id:"a138-append-elemen-ke-array"},"A.13.8. Append elemen ke array"),(0,i.kt)("p",null,"Operasi menambahkan sebuah elemen ke array yang hasilnya melebihi kapasitas ... adalah tidak bisa. Karena array memiliki size fixed, tidak dinamis. Solusinya adalah menggunakan tipe data Vector. Nantinya array perlu dikonversi ke bentu Vector terlebih dahulu kemudian di-append, lebih jelasnya kita bahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/vector"},"Vector"),"."),(0,i.kt)("h2",{id:"a139-nested-array"},"A.13.9. Nested array"),(0,i.kt)("p",null,"Data nested array bisa dibuat dengan level kedalaman tanpa batas, tetapi harus mengikuti aturan tipe data array yaitu: fixed size dan elemen bertipe data sejenis."),(0,i.kt)("p",null,"Pada contoh berikut variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_arr")," didefinisikan sebagai sebuah array bersarang atau nested dengan kedalaman 2 level."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let data_arr = [\n ["salad", "fried rice"],\n ["apple", "coconut"],\n ["spinach", "jalapeno"],\n];\nfor sub_arr in data_arr {\n for el in sub_arr {\n print!("{el}, ");\n }\n println!();\n}\n\n// salad, fried rice,\n// apple, coconut,\n// spinach, jalapeno,\n')),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"data_arr")," pada contoh di atas bertipe data",(0,i.kt)("inlineCode",{parentName:"p"},"[[&str; 2] 3]"),", yang artinya adalah sebuah array dengan size 3, dengan isi elemen adalah juga array dengan size 2. Selalu ingat bahwa size array adalah fixed."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/array"},"github.com/novalagung/dasarpemrogramanrust-example/../array")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice"},"Slice (Basic)")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/vector"},"Tipe Data \u279c Vector")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/slice-memory-management"},"Slice Memory Management"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.array.html"},"https://doc.rust-lang.org/std/primitive.array.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.slice.html"},"https://doc.rust-lang.org/std/primitive.slice.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/index.html"},"https://doc.rust-lang.org/std/iter/index.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/struct.Enumerate.html"},"https://doc.rust-lang.org/std/iter/struct.Enumerate.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/iter/trait.Iterator.html"},"https://doc.rust-lang.org/std/iter/trait.Iterator.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/trait/iter.html"},"https://doc.rust-lang.org/rust-by-example/trait/iter.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/rust-by-example/flow_control/for.html"},"https://doc.rust-lang.org/rust-by-example/flow_control/for.html"))))}u.isMDXComponent=!0},3421:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/array-1-919358da6705a5ced706fbcd108be794.png"},2957:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/array-2-bc70d4683afb247a4d30a6551aabec0a.png"},7674:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},4097:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},2977:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/db8ccf76.0bea83f9.js b/assets/js/db8ccf76.0bea83f9.js new file mode 100644 index 00000000..edf9979a --- /dev/null +++ b/assets/js/db8ccf76.0bea83f9.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4861],{3905:(a,e,t)=>{t.d(e,{Zo:()=>d,kt:()=>o});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var s=n.createContext({}),u=function(a){var e=n.useContext(s),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},d=function(a){var e=u(a.components);return n.createElement(s.Provider,{value:e},a.children)},m={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},k=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,s=a.parentName,d=p(a,["components","mdxType","originalType","parentName"]),k=u(t),o=i,c=k["".concat(s,".").concat(o)]||k[o]||m[o]||r;return t?n.createElement(c,l(l({ref:e},d),{},{components:t})):n.createElement(c,l({ref:e},d))}));function o(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=k;var p={};for(var s in e)hasOwnProperty.call(e,s)&&(p[s]=e[s]);p.originalType=a,p.mdxType="string"==typeof a?a:i,l[1]=p;for(var u=2;u{t.r(e),t.d(e,{assets:()=>s,contentTitle:()=>l,default:()=>m,frontMatter:()=>r,metadata:()=>p,toc:()=>u});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:36,title:"A.36. Traits",sidebar_label:"A.36. Traits"},l=void 0,p={unversionedId:"basic/traits",id:"basic/traits",title:"A.36. Traits",description:"Trait jika diartikan dalam bahasa Indonesia artinya adalah sifat. Chapter ini akan membahas tentang apa itu trait, apa kegunaannya, dan bagaimana cara penerapannya di Rust programming.",source:"@site/docs/basic/traits.md",sourceDirName:"basic",slug:"/basic/traits",permalink:"/basic/traits",draft:!1,tags:[],version:"current",sidebarPosition:36,frontMatter:{sidebar_position:36,title:"A.36. Traits",sidebar_label:"A.36. Traits"},sidebar:"tutorialSidebar",previous:{title:"A.35. Borrowing",permalink:"/basic/borrowing"},next:{title:"A.37. Generics",permalink:"/basic/generics"}},s={},u=[{value:"A.36.1. Konsep traits",id:"a361-konsep-traits",level:2},{value:"A.36.2. Implementasi trait",id:"a362-implementasi-trait",level:2},{value:"\u25c9 Struct Circle",id:"-struct-circle",level:3},{value:"\u25c9 Block kode impl X for Y",id:"-block-kode-impl-x-for-y",level:3},{value:"\u25c9 Block kode method dalam impl",id:"-block-kode-method-dalam-impl",level:3},{value:"\u25c9 Macro write",id:"-macro-write",level:3},{value:"\u25c9 Print data menggunakan formatted print {:?}",id:"-print-data-menggunakan-formatted-print-",level:3},{value:"\u25c9 Print data menggunakan formatted print {}",id:"-print-data-menggunakan-formatted-print--1",level:3},{value:"A.36.3. Membuat custom trait",id:"a363-membuat-custom-trait",level:2},{value:"\u25c9 Method pow untuk operasi pangkat",id:"-method-pow-untuk-operasi-pangkat",level:3},{value:"\u25c9 Keyword as untuk casting tipe data",id:"-keyword-as-untuk-casting-tipe-data",level:3},{value:"A.36.4. Trait sebagai tipe parameter",id:"a364-trait-sebagai-tipe-parameter",level:2},{value:"A.36.5. Trait bound syntax",id:"a365-trait-bound-syntax",level:2},{value:"A.36.6. Trait where clause",id:"a366-trait-where-clause",level:2},{value:"A.36.7. Trait sebagai return type",id:"a367-trait-sebagai-return-type",level:2},{value:"A.36.8. Associated types pada trait",id:"a368-associated-types-pada-trait",level:2},{value:"A.36.9. Attribute derive",id:"a369-attribute-derive",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:u};function m(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},d,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Trait jika diartikan dalam bahasa Indonesia artinya adalah sifat. Chapter ini akan membahas tentang apa itu trait, apa kegunaannya, dan bagaimana cara penerapannya di Rust programming."),(0,i.kt)("p",null,"Pembahasan chapter ini cukup panjang. Makin mendekati akhir pembahasan, makin berat yang dibahas. Penulis anjurkan jika nantinya setelah section ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits#a424-trait-sebagai-tipe-parameter"},"A.36.4. Trait sebagai tipe parameter")," dirasa cukup susah untuk dipahami, silakan lanjut ke chapter berikutnya dulu, dan nanti bisa kembali ke pembahasan chapter ini lagi."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Chapter ini butuh tambahan detail")),(0,i.kt)("h2",{id:"a361-konsep-traits"},"A.36.1. Konsep traits"),(0,i.kt)("p",null,"Di Rust kita bisa mendefinisikan trait/sifat, isinya adalah definisi header method yang bisa di-share ke banyak tipe data."),(0,i.kt)("p",null,"Trait isinya hanya definisi header method (bisa diartikan method tanpa isi). Ketika ada tipe data yang meng-implement suatu trait, maka tipe tersebut wajib untuk menuliskan implementasi method sesuai dengan header method yang ada di trait."),(0,i.kt)("p",null,"Ada dua bagian penting dalam trait yang harus diketahui:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},"Deklarasi trait"),(0,i.kt)("li",{parentName:"ol"},"Implementasi trait ke tipe data")),(0,i.kt)("p",null,"Perihal point pertama, intinya kita bisa menciptakan trait sesuai kebutuhan. Terlepas dari itu, Rust juga menyediakan cukup banyak traits yang di-implement ke banyak tipe data yang ada di Rust standard library. Beberapa di antaranya:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"std::fmt::Debug"),", digunakan agar data bisa di-print menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"li"},"{:?}"),"."),(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"std::iter::Enumerate"),", digunakan agar data bisa di-iterasi menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"li"},"for"),"."),(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"std::ops::Add"),", di-implementasikan agar data bisa digunakan pada operasi aritmatik penambahan ",(0,i.kt)("inlineCode",{parentName:"li"},"+"),".")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Pada bahasa pemrograman lain, contohnya Java, konsep trait mirip dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"interface"))),(0,i.kt)("p",null,"Ok, biar lebih jelas, mari lanjut pembelajaran menggunakan contoh. Kita mulai dengan pembahasan tentang cara implementasi trait. Contoh yang digunakan adalah implementasi salah satu trait milik Rust standard library, yaitu trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug"),"."),(0,i.kt)("h2",{id:"a362-implementasi-trait"},"A.36.2. Implementasi trait"),(0,i.kt)("p",null,"Kita pilih trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug")," milik Rust standard library untuk belajar cara implementasi trait pada tipe data."),(0,i.kt)("p",null,"Kegunaan dari trait ini adalah: jika di-implement ke tipe data tertentu maka data dengan tipe tersebut bisa di-print via macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println")," atau macro printing lainnya, dengan menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}"),"."),(0,i.kt)("p",null,"Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," ini diimplementasikan ke pada banyak tipe data yang di Rust standard library, baik itu tipe primitif maupun non-primitif. Contohnya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let number = 12;\nprintln!("{:?}", number);\n\nlet text = String::from("hello");\nprintln!("{:?}", text);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(2767).Z,width:"451",height:"109"})),(0,i.kt)("p",null,"Dua variabel di atas sukses di-print tanpa error, karena tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ",(0,i.kt)("em",{parentName:"p"},"by default")," sudah implement trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Jika tertarik untuk pengecekan lebih lanjut, silakan lihat di halaman dokumentasi tipe data ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/primitive.i32.html"},"i32")," dan ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/string/struct.String.html"},"String"),".")),(0,i.kt)("p",null,"Bagaimana dengan custom type yang kita buat sendiri? Misalnya struct."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let circle_one = Circle{raidus: 6};\n println!("{:?}", circle_one);\n}\n\nstruct Circle {\n raidus: i32,\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(9856).Z,width:"841",height:"512"})),(0,i.kt)("p",null,"Hasilnya error, karena struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," yang dibuat tidak implement trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug"),"."),(0,i.kt)("p",null,"Solusi agar tidak error adalah dengan mengimplementasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug")," ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle"),", dengan itu semua data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," akan bisa di-print menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Selain via implementasi trait, tipe data custom bisa di-print dengan cara menambahkan atribut ",(0,i.kt)("inlineCode",{parentName:"p"},"#[derive(Debug)]")," pada definisi tipe data-nya. Namun kita tidak membahas itu pada chapter ini.")),(0,i.kt)("p",null,"Langkah pertama untuk implementasi trait adalah mencari tau terlebih dahulu spesifikasi trait yang ingin diimplementasikan. Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug")," adalah traits milik Rust standard library, maka harusnya spesifikasi bisa dilihat di dokumentasi Rust."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},"https://doc.rust-lang.org/std/fmt/trait.Debug.html"))),(0,i.kt)("p",null,"Pada URL dokumentasi bisa dilihat kalau trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," memiliki struktur kurang lebih seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"pub trait Debug {\n fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), Error>;\n}\n")),(0,i.kt)("p",null,"Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," mempunyai satu spesifikasi method, bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"fmt")," yang detail strukturnya bisa dilihat di atas."),(0,i.kt)("p",null,"Kita akan implement trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," ini ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle"),", maka wajib hukumnya untuk menuliskan implementasi method sesuai dengan yang ada di trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug"),"."),(0,i.kt)("p",null,"Di bawah ini adalah contoh cara implementasi trait."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct Circle {\n raidus: i32,\n}\n\nimpl std::fmt::Debug for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "Circle radius: {}", self.raidus)\n }\n}\n\nfn main() {\n let circle_one = Circle{raidus: 6};\n println!("{:?}", circle_one);\n}\n')),(0,i.kt)("p",null,"Ketika program di-run, hasilnya sukses tanpa error. Artinya implementasi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," pada tipe data struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," adalah sukses."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(4666).Z,width:"457",height:"115"})),(0,i.kt)("p",null,"Cara implementasi trait ke struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," memang step-nya agak panjang, tapi penulis yakin lama-kelamaan pasti terbiasa. Ok, sekarang kita bahas satu per satu kode di atas."),(0,i.kt)("h3",{id:"-struct-circle"},"\u25c9 Struct ",(0,i.kt)("inlineCode",{parentName:"h3"},"Circle")),(0,i.kt)("p",null,"Block kode definisi struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," cukup straightforward, isinya hanya 1 property bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"radius")," bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,i.kt)("h3",{id:"-block-kode-impl-x-for-y"},"\u25c9 Block kode ",(0,i.kt)("inlineCode",{parentName:"h3"},"impl X for Y")),(0,i.kt)("p",null,"Notasi penulisan implementasi trait adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"impl X for Y"),", yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"X")," adalah trait yang ingin diimplementasikan dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Y")," adalah tipe data tujuan implementasi."),(0,i.kt)("p",null,"Pada contoh di atas, trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," diimplementasikan ke custom type struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle"),". Maka statement-nya adalah:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl std::fmt::Debug for Circle {\n // ...\n}\n")),(0,i.kt)("h3",{id:"-block-kode-method-dalam-impl"},"\u25c9 Block kode method dalam ",(0,i.kt)("inlineCode",{parentName:"h3"},"impl")),(0,i.kt)("p",null,"Block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," harus diikuti dengan implementasi method. Pada contoh ini, method ",(0,i.kt)("inlineCode",{parentName:"p"},"fmt")," milik trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," wajib untuk diimplementasikan. Spesifikasi method ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), Error>")," (lebih jelasnya silakan lihat dokumentasi)."),(0,i.kt)("p",null,"Silakan copy method tersebut kemudian paste ke dalam block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," yang sudah ditulis, kemudian tambahkan block kurung kurawal."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl std::fmt::Debug for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {\n // ...\n }\n}\n")),(0,i.kt)("p",null,"Kemudian tulis implementasi method ",(0,i.kt)("inlineCode",{parentName:"p"},"fmt")," dalam block method. Tulis statement macro ",(0,i.kt)("inlineCode",{parentName:"p"},"write")," untuk data string (yang ingin di-print) dengan tujuan adalah variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"f"),"."),(0,i.kt)("p",null,"Di contoh, format ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle radius: {}")," digunakan. Dengan ini nantinya saat printing data, yang muncul adalah text ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle radius: {}"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'impl std::fmt::Debug for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "Circle radius: {}", self.raidus)\n }\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tips untuk pengguna visual studio code dengan rust-analyzer extension ter-install, setelah selesai menulis block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl"),", cukup jalankan ",(0,i.kt)("inlineCode",{parentName:"p"},"ctrl+space")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"cmd+space")," untuk men-trigger autocomplete suggestion. Kemudian klik opsi method yang ada di situ, maka kode implementasi method langsung muncul dengan sendirinya.")),(0,i.kt)("h3",{id:"-macro-write"},"\u25c9 Macro ",(0,i.kt)("inlineCode",{parentName:"h3"},"write")),(0,i.kt)("p",null,"Macro ini digunakan untuk menuliskan sebuah data ke object tertentu. Pada contoh kita gunakan untuk menulis string ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle radius: {}")," ke variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"f")," yang bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Formatter<'_>"),"."),(0,i.kt)("p",null,"Notasi penulisan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"write"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// notasi penulisan\nwrite!(variabel_tujuan, data_yang_ingin_di_print, arg1, arg2, ...);\n\n// contoh penerapan\nwrite!(f, "Circle radius: {}", self.raidus);\n')),(0,i.kt)("h3",{id:"-print-data-menggunakan-formatted-print-"},"\u25c9 Print data menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"h3"},"{:?}")),(0,i.kt)("p",null,"Step terakhir adalah print variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"circle")," menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Hasilnya sukses, tidak error seperti sebelumnya."),(0,i.kt)("h3",{id:"-print-data-menggunakan-formatted-print--1"},"\u25c9 Print data menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"h3"},"{}")),(0,i.kt)("p",null,"Coba tambahkan statement ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),", tetapi kali ini gunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{}"),", apakah hasilnya juga tidak error?"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(8738).Z,width:"849",height:"309"})),(0,i.kt)("p",null,"Hasilnya error, karena trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug")," hanya berguna untuk formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}"),". Agar data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," bisa di-print menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{}")," maka trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Display")," harus di-implementasikan juga."),(0,i.kt)("p",null,"Ubah kode dengan menambahkan implementasi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Display"),". Hasilnya kurang lebih seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct Circle {\n raidus: i32,\n}\n\nimpl std::fmt::Debug for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "Circle radius: {}", self.raidus)\n }\n}\n\nimpl std::fmt::Display for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "Circle radius: {}", self.raidus)\n }\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Link dokumentasi trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Debug")," ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},"https://doc.rust-lang.org/std/fmt/trait.Debug.html")),(0,i.kt)("li",{parentName:"ul"},"Link dokumentasi trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Display")," ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Display.html"},"https://doc.rust-lang.org/std/fmt/trait.Display.html")))),(0,i.kt)("h2",{id:"a363-membuat-custom-trait"},"A.36.3. Membuat custom trait"),(0,i.kt)("p",null,"Pada section di atas kita telah membahas bagaimana cara implementasi traits ke tipe data. Pada bagian ini kita akan belajar tentang cara membuat definisi trait (membuat custom trait)."),(0,i.kt)("p",null,"Masih sama dengan metode sebelumnya, pembelajaran dilakukan dengan praktik. Kita gunakan skenario praktik berikut pada program selanjutnya:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},"Buat struct bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"Circle"),"."),(0,i.kt)("li",{parentName:"ol"},"Buat struct bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"Square"),"."),(0,i.kt)("li",{parentName:"ol"},"Buat trait bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"Area")," dengan isi satu buah method untuk menghitung luas bangun datar (",(0,i.kt)("inlineCode",{parentName:"li"},"calculate"),")."),(0,i.kt)("li",{parentName:"ol"},"Implementasikan trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Area")," ke dua struct tersebut.")),(0,i.kt)("p",null,"Ok, mari mulai praktikkan skenario di atas. Pertama siapkan project dengan struktur berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 calculation_spec.rs\n \u2502\u2500\u2500\u2500 two_dimensional.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("p",null,"Module ",(0,i.kt)("inlineCode",{parentName:"p"},"calculation_spec")," berisi definisi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area"),". Trait ini punya visibility akses publik, isinya hanya satu buah definisi method header bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate"),". Trait ini nantinya diimplementasikan ke struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan juga ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),", agar nantinya kedua struct tersebut memiliki method ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate")," yang berguna untuk kalkulasi luas bangun datar 2d."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/calculation_spec.rs"',title:'"src/calculation_spec.rs"'},"pub trait Area {\n fn calculate(&self) -> f64;\n}\n")),(0,i.kt)("p",null,"Kemudian siapkan file ",(0,i.kt)("inlineCode",{parentName:"p"},"two_dimensional"),", isinya dua buah struct: ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),". Pada file yang sama, siapkan juga block kode implementasi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area"),". Dengan ini maka kedua struct tersebut wajib untuk memiliki method bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate")," dengan isi adalah operasi perhitungan aritmatika untuk mencari luas bangun datar."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/two_dimensional.rs"',title:'"src/two_dimensional.rs"'},"pub struct Circle {\n pub radius: i32,\n}\n\nimpl crate::calculation_spec::Area for Circle {\n fn calculate(&self) -> f64 {\n // PI * (r ^ 2)\n // ada operasi casting ke tipe f64 karena self.radius bertipe i32\n 3.14 * (self.radius.pow(2) as f64)\n }\n}\n\npub struct Square {\n pub length: i32,\n}\n\nimpl crate::calculation_spec::Area for Square {\n fn calculate(&self) -> f64 {\n // (s ^ 2)\n // ada operasi casting ke tipe f64 karena self.length bertipe i32\n self.length.pow(2) as f64\n }\n}\n")),(0,i.kt)("p",null,"Bisa dilihat pada kode di atas, deklarasi struct beserta property memiliki visibility publik. Idealnya, saat sturct tersebut digunakan di fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," nantinya tidak akan ada error terkait visibility akses."),(0,i.kt)("p",null,"Selanjutnya, pada file ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs")," siapkan kode yang isinya registrasi module ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_spec")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"two_dimensional"),", juga definisi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," dengan isi statement pembuatan 2 variabel object untuk masing-masing tipe data struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod calculation_spec;\nmod two_dimensional;\n\nuse crate::calculation_spec::Area;\n\nfn main() {\n let circle_one = two_dimensional::Circle{ radius: 10 };\n println!("circle area: {}", circle_one.calculate());\n\n let square_one = two_dimensional::Square{ length: 5 };\n println!("square area: {}", square_one.calculate());\n}\n')),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate")," milik object bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square")," diakses untuk kemudian di-print."),(0,i.kt)("p",null,"Coba jalankan program."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(4176).Z,width:"655",height:"756"})),(0,i.kt)("p",null,"Hmm, error. Padahal trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," sudah publik, dan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," & ",(0,i.kt)("inlineCode",{parentName:"p"},"Square")," beserta property-nya juga sudah publik. Tapi masih error."),(0,i.kt)("p",null,"Error ini disebabkan oleh trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," yang belum di-import di crate root (main). Meskipun kita tidak mengakses trait tersebut secara langsung (melainkan via method ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate")," milik struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),"), diharuskan untuk meng-import-nya juga."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Detail error beserta solusi dari error ini sebenarnya bisa dilihat di error message. Bagaimana Rust menginformasikan error sangat luar biasa informatif.")),(0,i.kt)("p",null,"Ok, sekarang ubah isi file ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs")," menjadi seperti ini, kemudian jalankan ulang program. Hasilnya tidak ada error."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod calculation_spec;\nmod two_dimensional;\n\nuse crate::calculation_spec::Area; // <------- tambahkan statement import module\n\nfn main() {\n let circle_one = two_dimensional::Circle{ radius: 10 };\n println!("circle area: {}", circle_one.calculate());\n\n let square_one = two_dimensional::Square{ length: 5 };\n println!("square area: {}", square_one.calculate());\n}\n')),(0,i.kt)("p",null,"O iya, ada beberapa hal baru pada penerapan kode di atas, berikut adalah pembahasannya:"),(0,i.kt)("h3",{id:"-method-pow-untuk-operasi-pangkat"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"pow")," untuk operasi pangkat"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"pow")," adalah item milik tipe data numerik (",(0,i.kt)("inlineCode",{parentName:"p"},"i8"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"i16"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", ...) yang fungsinya untuk operasi pangkat."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"3.pow(2); // ===> 3 pangkat 2\n8.pow(5); // ===> 8 pangkat 5\n")),(0,i.kt)("h3",{id:"-keyword-as-untuk-casting-tipe-data"},"\u25c9 Keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"as")," untuk casting tipe data"),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"as")," digunakan untuk casting tipe data. Keyword ini bisa diterapkan pada beberapa jenis tipe data, salah satunya adalah semua tipe data numerik."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"1024 as f32; // ===> 1024 dikonversi ke tipe f32, hasinya adalah 1024.0\n3.14 as i32; // ===> 3.14 dikonversi ke tipe i32, hasinya 3 karena ada pembulatan\n")),(0,i.kt)("h2",{id:"a364-trait-sebagai-tipe-parameter"},"A.36.4. Trait sebagai tipe parameter"),(0,i.kt)("p",null,"Trait bisa digunakan sebagai tipe data parameter sebuah fungsi, contoh notasi penulisannya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result(name: String, item: &impl Area) {\n println!("{} area: {}", name, item.calculate());\n}\n')),(0,i.kt)("p",null,"Manfaat penerapan trait sebagai tipe data parameter fungsi adalah saat pemanggilan fungsi, parameter tersebut bisa diisi dengan argument data bertipe apapun dengan catatan tipe dari data tersebut mengimplementasikan trait yang sama dengan yang digunakan pada parameter."),(0,i.kt)("p",null,"Misalnya, pada fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_and_print_result")," di atas yang parameter ke-2 bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&impl Area"),", nantinya saat fungsi tersebut dipanggil, kita bisa sisipi parameter ke-2 dengan object ",(0,i.kt)("inlineCode",{parentName:"p"},"circle_one")," ataupun ",(0,i.kt)("inlineCode",{parentName:"p"},"circle_two"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let circle_one = two_dimensional::Circle{ radius: 10 };\ncalculate_and_print_result("circle".to_string(), &circle_one);\n\nlet square_one = two_dimensional::Square{ length: 5 };\ncalculate_and_print_result("square".to_string(), &square_one);\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},(0,i.kt)("inlineCode",{parentName:"p"},"&impl Area")," ini tipe data pointer ya, tipe non-pointer-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"impl Area"),". Di sini digunakan tipe data pointer untuk antisipasi ",(0,i.kt)("em",{parentName:"p"},"move semantics")," pada tipe data custom type (borrowing).")),(0,i.kt)("p",null,"Dimisalkan, fungsi tersebut parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"item"),"-nya bisa menampung beberapa jenis traits, kira-kira apakah bisa dibuat seperti itu? Misalnya ada trait lain bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"Circumference"),", dan parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"item")," milik fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_and_print_result")," harus bisa menampung data baik dari tipe yang implement trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," ataupun trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Circumference"),"."),(0,i.kt)("p",null,"Hal seperti itu bisa, caranya dengan menggunakan notasi penulisan berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result(name: String, item: &(impl Area + Circumference)) {\n println!("{} area: {}", name, item.calculate());\n println!("{} circumference: {}", name, item.calculateCircumference());\n}\n')),(0,i.kt)("p",null,"Tambahkan tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"()")," sebelum ",(0,i.kt)("inlineCode",{parentName:"p"},"impl NamaTrait"),", lalu ganti ",(0,i.kt)("inlineCode",{parentName:"p"},"NamaTrait")," dengan traits apa saja yang diinginkan dengan separator tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"+"),"."),(0,i.kt)("h2",{id:"a365-trait-bound-syntax"},"A.36.5. Trait bound syntax"),(0,i.kt)("p",null,"Penerapan trait sebagai parameter fungsi juga bisa dituliskan dalam notasi yang memanfaatkan generic. Teknik penulisan ini disebut dengan ",(0,i.kt)("em",{parentName:"p"},"trait bound syntax"),"."),(0,i.kt)("p",null,"Contohnya bisa dilihat pada kode berikut. Ada generic bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," yang merepresentasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area"),", kemudian pada definisi parameter ke-2 fungsi (yaitu parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"item"),") tipenya menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"&T"),". Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&T")," di sini adalah ekuivalen dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"&impl Area"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result2(name: String, item: &T) {\n println!("{} area: {}", name, item.calculate());\n}\n')),(0,i.kt)("p",null,"Dimisalkan jika ada lebih dari satu trait yang digunakan sebagai tipe data paramater (misalnya trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Circumference"),"), maka penulisannya seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result2(name: String, item: &T) {\n println!("{} area: {}", name, item.calculate());\n println!("{} circumference: {}", name, item.calculateCircumference());\n}\n')),(0,i.kt)("p",null,"Satu tambahan contoh lagi untuk ilustrasi yang lebih kompleks:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn some_function(t: &T, u: &U) -> i32 {\n // ...\n}\n")),(0,i.kt)("p",null,"Pada contoh di atas fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"some_function")," memiliki 2 generics param, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"U"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"T")," merepresentasikan trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Display")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"Clone")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"U")," merepresentasikan trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Clone")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"Debug"))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai generics dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics"},"Generics"))),(0,i.kt)("h2",{id:"a366-trait-where-clause"},"A.36.6. Trait ",(0,i.kt)("inlineCode",{parentName:"h2"},"where")," clause"),(0,i.kt)("p",null,"Ada lagi alternatif penulisan trait bound syntax, yaitu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"where"),". Contoh pengaplikasiannya bisa dilihat pada kode berikut. Semua definisi fungsi di bawah ini adalah ekuivalen."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result(name: String, item: &(impl Area + Circumference)) {\n println!("{} area: {}", name, item.calculate());\n println!("{} circumference: {}", name, item.calculateCircumference());\n}\n\nfn calculate_and_print_result2(name: String, item: &T) {\n println!("{} area: {}", name, item.calculate());\n println!("{} circumference: {}", name, item.calculateCircumference());\n}\n\nfn calculate_and_print_result3(name: String, item: &T) where T: Area + Circumference {\n println!("{} area: {}", name, item.calculate());\n}\n\nfn calculate_and_print_result4(name: String, item: &T)\nwhere\n T: Area + Circumference,\n // ... other generic params if exists\n{\n println!("{} area: {}", name, item.calculate());\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai generics dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics"},"Generics"))),(0,i.kt)("h2",{id:"a367-trait-sebagai-return-type"},"A.36.7. Trait sebagai return type"),(0,i.kt)("p",null,"Trait bisa juga digunakan sebagai tipe data return value. Caranya gunakan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"impl NamaTrait")," sebagai tipe data."),(0,i.kt)("p",null,"Contohnya bisa dilihat pada kode berikut. Ada dua fungsi baru dideklarasikan:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},"Fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"new_circle")," dengan return type adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"impl Area"),", dan data yang dikembalikan adalah bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"two_dimensional::Circle"),"."),(0,i.kt)("li",{parentName:"ol"},"Fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"new_square")," dengan return type adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"impl Area"),", dan data yang dikembalikan adalah bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"two_dimensional::Square"),".")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let circle_one = new_circle(5);\n calculate_and_print_result4("circle".to_string(), &circle_one);\n\n let square_one = new_square(10);\n calculate_and_print_result4("square".to_string(), &square_one);\n}\n\nfn new_circle(radius: i32) -> impl Area {\n let data = two_dimensional::Circle{\n radius\n };\n data\n}\n\nfn new_square(length: i32) -> impl Area {\n two_dimensional::Square{\n length\n }\n}\n\nfn calculate_and_print_result4(name: String, item: &T)\nwhere\n T: Area,\n{\n println!("{} area: {}", name, item.calculate());\n}\n')),(0,i.kt)("p",null,"Salah satu konsekuensi dalam penerapan trait sebagai return type adalah: tipe data milik nilai yang dikembalikan terdeteksi sebagai tipe trait. Contohnya variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"circle_one")," di atas, tipe data-nya bukan ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle"),", melainkan ",(0,i.kt)("inlineCode",{parentName:"p"},"impl Area"),"."),(0,i.kt)("p",null,"Tipe data aslinya tetap bisa diakses, tapi butuh tambahan effort. Lebih jelasnya dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/trait-conversion-from-into"},"Trait \u279c Conversion (From & Into)"),"."),(0,i.kt)("h2",{id:"a368-associated-types-pada-trait"},"A.36.8. ",(0,i.kt)("em",{parentName:"h2"},"Associated types")," pada trait"),(0,i.kt)("p",null,"Associated types adalah tipe data yang didefinisikan di dalam suatu trait. Associated types tidak tidak memiliki tipe data konkret saat didefinisikan, namun ketika trait di-implementasikan maka tipe tersebut harus ditentukan tipe data konkritnya."),(0,i.kt)("p",null,"Lebih jelas silakan perhatikan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"trait Shape {\n type Area;\n\n fn area(&self) -> Self::Area;\n}\n")),(0,i.kt)("p",null,"Pada definisi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Shape")," di atas, yang disebut dengan associated types adalah tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," yang definisinya berada dalam block trait. Tipe didefinisikan tanpa assignment operator, jadi tidak ada tipe data konkretnya."),(0,i.kt)("p",null,"Associated types ini sering digunakan pada Rust programming. Contoh implementasinya bisa dilihat pada contoh di bawah ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 shape.rs\n \u2502\u2500\u2500\u2500 circle.rs\n \u2502\u2500\u2500\u2500 square.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Disiapkan suatu trait bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"shape::Shape"),".",(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},"Trait ini memiliki satu associated types bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"Area"),"."),(0,i.kt)("li",{parentName:"ul"},"Dan memiliki sebuah definisi method header ",(0,i.kt)("inlineCode",{parentName:"li"},"area")," yang gunanya untuk menghitung luas bangun datar (",(0,i.kt)("em",{parentName:"li"},"shape"),")."))),(0,i.kt)("li",{parentName:"ul"},"Disiapkan struct ",(0,i.kt)("inlineCode",{parentName:"li"},"circle::Circle")," yang mengadopsi trait ",(0,i.kt)("inlineCode",{parentName:"li"},"shape::Shape"),"."),(0,i.kt)("li",{parentName:"ul"},"Disiapkan struct ",(0,i.kt)("inlineCode",{parentName:"li"},"square::Square")," yang mengadopsi trait ",(0,i.kt)("inlineCode",{parentName:"li"},"shape::Shape"),".")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/shape.rs"',title:'"src/shape.rs"'},"pub trait Shape {\n type Area;\n\n fn area(&self) -> Self::Area;\n}\n")),(0,i.kt)("p",null,"Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Shape")," di atas spesifikasinya mirip seperti pada contoh sebelumnya, hanya saja kali ini trait-nya di set public agar bisa diakses dari ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs")," nantinya."),(0,i.kt)("p",null,"Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Shape")," kemudian di-implementasikan ke struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),", kode-nya bisa dilihat berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/circle.rs"',title:'"src/circle.rs"'},"pub struct Circle {\n pub radius: f64,\n}\n\nimpl crate::shape::Shape for Circle {\n type Area = f64;\n\n fn area(&self) -> Self::Area {\n std::f64::consts::PI * self.radius * self.radius\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/square.rs"',title:'"src/square.rs"'},"pub struct Square {\n pub side: i64,\n}\n\nimpl crate::shape::Shape for Square {\n type Area = i64;\n\n fn area(&self) -> Self::Area {\n self.side * self.side\n }\n}\n")),(0,i.kt)("p",null,"Bisa dilihat pada kedua implementasi di atas, associated type ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," diisi dengan tipe concrete, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Tipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"f64")," sebagai tipe concrete ",(0,i.kt)("inlineCode",{parentName:"li"},"circle:Circle:Area")),(0,i.kt)("li",{parentName:"ul"},"Tipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"i64")," sebagai tipe concrete ",(0,i.kt)("inlineCode",{parentName:"li"},"square:Square:Area"))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Contoh di atas adalah cara pengaplikasian associated types.")),(0,i.kt)("p",null,"Lalu pada ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs"),", tipe data struct ",(0,i.kt)("inlineCode",{parentName:"p"},"circle::Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"square::Square")," digunakan untuk membuat variabel baru, yang kemudian dari variabel tersebut, method ",(0,i.kt)("inlineCode",{parentName:"p"},".area()")," milik diakses."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod shape;\nmod circle;\nmod square;\n\nuse crate::shape::Shape;\n\nfn main() {\n let obj1 = circle::Circle{ radius: 10.0 };\n println!("area of circle: {:.2}", obj1.area());\n\n let obj2 = square::Square{ side: 10 };\n println!("area of square: {:}", obj2.area());\n}\n')),(0,i.kt)("p",null,"Silakan jalankan program dan lihat hasilnya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(6254).Z,width:"469",height:"130"})),(0,i.kt)("p",null,"O iya, pada ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs"),", module item ",(0,i.kt)("inlineCode",{parentName:"p"},"shape::Shape")," perlu di-import meskipun kita tidak menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"trait")," tersebut secara langsung. Jika tidak di-import, maka method ",(0,i.kt)("inlineCode",{parentName:"p"},".area()")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square")," tidak bisa diakses."),(0,i.kt)("h2",{id:"a369-attribute-derive"},"A.36.9. Attribute ",(0,i.kt)("inlineCode",{parentName:"h2"},"derive")),(0,i.kt)("p",null,"Ada cara lain untuk mengimplementasikan suatu trait ke dalam tipe data selain dengan menuliskan implementasinya secara eksplist, caranya menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive"),"."),(0,i.kt)("p",null,"Lebih detailnya dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/attributes"},"Attributes"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/traits"},"github.com/novalagung/dasarpemrogramanrust-example/../traits")),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang trait bounds untuk implementasi method kondisional"),(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang trait overloading")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch10-02-traits.html"},"https://doc.rust-lang.org/book/ch10-02-traits.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.i32.html"},"https://doc.rust-lang.org/std/primitive.i32.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},"https://doc.rust-lang.org/std/fmt/trait.Debug.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Display.html"},"https://doc.rust-lang.org/std/fmt/trait.Display.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/string/struct.String.html"},"https://doc.rust-lang.org/std/string/struct.String.html"))))}m.isMDXComponent=!0},2767:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},9856:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/traits-2-7c5e41bf2bd31dba0f78fc7afd81c7ba.png"},4666:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},8738:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/traits-4-fe5c6a716894397c8088bae54e7194f9.png"},4176:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/traits-5-29be804051729f23db8dd64e978e2398.png"},6254:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/traits-6-8165525fc9f8604bf115634eeb974230.png"}}]); \ No newline at end of file diff --git a/assets/js/db8ccf76.f925a746.js b/assets/js/db8ccf76.f925a746.js deleted file mode 100644 index a6c2bd45..00000000 --- a/assets/js/db8ccf76.f925a746.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[4861],{3905:(a,e,t)=>{t.d(e,{Zo:()=>d,kt:()=>o});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function l(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var s=n.createContext({}),u=function(a){var e=n.useContext(s),t=e;return a&&(t="function"==typeof a?a(e):l(l({},e),a)),t},d=function(a){var e=u(a.components);return n.createElement(s.Provider,{value:e},a.children)},m={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},k=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,s=a.parentName,d=p(a,["components","mdxType","originalType","parentName"]),k=u(t),o=i,c=k["".concat(s,".").concat(o)]||k[o]||m[o]||r;return t?n.createElement(c,l(l({ref:e},d),{},{components:t})):n.createElement(c,l({ref:e},d))}));function o(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,l=new Array(r);l[0]=k;var p={};for(var s in e)hasOwnProperty.call(e,s)&&(p[s]=e[s]);p.originalType=a,p.mdxType="string"==typeof a?a:i,l[1]=p;for(var u=2;u{t.r(e),t.d(e,{assets:()=>s,contentTitle:()=>l,default:()=>m,frontMatter:()=>r,metadata:()=>p,toc:()=>u});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:36,title:"A.36. Traits",sidebar_label:"A.36. Traits"},l=void 0,p={unversionedId:"basic/traits",id:"basic/traits",title:"A.36. Traits",description:"Trait jika diartikan dalam bahasa Indonesia artinya adalah sifat. Chapter ini akan membahas tentang apa itu trait, apa kegunaannya, dan bagaimana cara penerapannya di Rust programming.",source:"@site/docs/basic/traits.md",sourceDirName:"basic",slug:"/basic/traits",permalink:"/basic/traits",draft:!1,tags:[],version:"current",sidebarPosition:36,frontMatter:{sidebar_position:36,title:"A.36. Traits",sidebar_label:"A.36. Traits"},sidebar:"tutorialSidebar",previous:{title:"A.35. Borrowing",permalink:"/basic/borrowing"},next:{title:"A.37. Generics",permalink:"/basic/generics"}},s={},u=[{value:"A.36.1. Konsep traits",id:"a361-konsep-traits",level:2},{value:"A.36.2. Implementasi trait",id:"a362-implementasi-trait",level:2},{value:"\u25c9 Struct Circle",id:"-struct-circle",level:3},{value:"\u25c9 Block kode impl X for Y",id:"-block-kode-impl-x-for-y",level:3},{value:"\u25c9 Block kode method dalam impl",id:"-block-kode-method-dalam-impl",level:3},{value:"\u25c9 Macro write",id:"-macro-write",level:3},{value:"\u25c9 Print data menggunakan formatted print {:?}",id:"-print-data-menggunakan-formatted-print-",level:3},{value:"\u25c9 Print data menggunakan formatted print {}",id:"-print-data-menggunakan-formatted-print--1",level:3},{value:"A.36.3. Membuat custom trait",id:"a363-membuat-custom-trait",level:2},{value:"\u25c9 Method pow untuk operasi pangkat",id:"-method-pow-untuk-operasi-pangkat",level:3},{value:"\u25c9 Keyword as untuk casting tipe data",id:"-keyword-as-untuk-casting-tipe-data",level:3},{value:"A.36.4. Trait sebagai tipe parameter",id:"a364-trait-sebagai-tipe-parameter",level:2},{value:"A.36.5. Trait bound syntax",id:"a365-trait-bound-syntax",level:2},{value:"A.36.6. Trait where clause",id:"a366-trait-where-clause",level:2},{value:"A.36.7. Trait sebagai return type",id:"a367-trait-sebagai-return-type",level:2},{value:"A.36.8. Associated types pada trait",id:"a368-associated-types-pada-trait",level:2},{value:"A.36.9. Attribute derive",id:"a369-attribute-derive",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Work in progress",id:"-work-in-progress",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],d={toc:u};function m(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},d,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Trait jika diartikan dalam bahasa Indonesia artinya adalah sifat. Chapter ini akan membahas tentang apa itu trait, apa kegunaannya, dan bagaimana cara penerapannya di Rust programming."),(0,i.kt)("p",null,"Pembahasan chapter ini cukup panjang. Makin mendekati akhir pembahasan, makin berat yang dibahas. Penulis anjurkan jika nantinya setelah section ",(0,i.kt)("a",{parentName:"p",href:"/basic/traits#a424-trait-sebagai-tipe-parameter"},"A.36.4. Trait sebagai tipe parameter")," dirasa cukup susah untuk dipahami, silakan lanjut ke chapter berikutnya dulu, dan nanti bisa kembali ke pembahasan chapter ini lagi."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Chapter ini butuh tambahan detail")),(0,i.kt)("h2",{id:"a361-konsep-traits"},"A.36.1. Konsep traits"),(0,i.kt)("p",null,"Di Rust kita bisa mendefinisikan trait/sifat, isinya adalah definisi header method yang bisa di-share ke banyak tipe data."),(0,i.kt)("p",null,"Trait isinya hanya definisi header method (bisa diartikan method tanpa isi). Ketika ada tipe data yang meng-implement suatu trait, maka tipe tersebut wajib untuk menuliskan implementasi method sesuai dengan header method yang ada di trait."),(0,i.kt)("p",null,"Ada dua bagian penting dalam trait yang harus diketahui:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},"Deklarasi trait"),(0,i.kt)("li",{parentName:"ol"},"Implementasi trait ke tipe data")),(0,i.kt)("p",null,"Perihal point pertama, intinya kita bisa menciptakan trait sesuai kebutuhan. Terlepas dari itu, Rust juga menyediakan cukup banyak traits yang di-implement ke banyak tipe data yang ada di Rust standard library. Beberapa di antaranya:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"std::fmt::Debug"),", digunakan agar data bisa di-print menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"li"},"{:?}"),"."),(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"std::iter::Enumerate"),", digunakan agar data bisa di-iterasi menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"li"},"for"),"."),(0,i.kt)("li",{parentName:"ul"},"Trait ",(0,i.kt)("inlineCode",{parentName:"li"},"std::ops::Add"),", di-implementasikan agar data bisa digunakan pada operasi aritmatik penambahan ",(0,i.kt)("inlineCode",{parentName:"li"},"+"),".")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Pada bahasa pemrograman lain, contohnya Java, konsep trait mirip dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"interface"))),(0,i.kt)("p",null,"Ok, biar lebih jelas, mari lanjut pembelajaran menggunakan contoh. Kita mulai dengan pembahasan tentang cara implementasi trait. Contoh yang digunakan adalah implementasi salah satu trait milik Rust standard library, yaitu trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug"),"."),(0,i.kt)("h2",{id:"a362-implementasi-trait"},"A.36.2. Implementasi trait"),(0,i.kt)("p",null,"Kita pilih trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug")," milik Rust standard library untuk belajar cara implementasi trait pada tipe data."),(0,i.kt)("p",null,"Kegunaan dari trait ini adalah: jika di-implement ke tipe data tertentu maka data dengan tipe tersebut bisa di-print via macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println")," atau macro printing lainnya, dengan menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}"),"."),(0,i.kt)("p",null,"Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," ini diimplementasikan ke pada banyak tipe data yang di Rust standard library, baik itu tipe primitif maupun non-primitif. Contohnya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let number = 12;\nprintln!("{:?}", number);\n\nlet text = String::from("hello");\nprintln!("{:?}", text);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(2767).Z,width:"451",height:"109"})),(0,i.kt)("p",null,"Dua variabel di atas sukses di-print tanpa error, karena tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ",(0,i.kt)("em",{parentName:"p"},"by default")," sudah implement trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Jika tertarik untuk pengecekan lebih lanjut, silakan lihat di halaman dokumentasi tipe data ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/primitive.i32.html"},"i32")," dan ",(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/string/struct.String.html"},"String"),".")),(0,i.kt)("p",null,"Bagaimana dengan custom type yang kita buat sendiri? Misalnya struct."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let circle_one = Circle{raidus: 6};\n println!("{:?}", circle_one);\n}\n\nstruct Circle {\n raidus: i32,\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(9856).Z,width:"841",height:"512"})),(0,i.kt)("p",null,"Hasilnya error, karena struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," yang dibuat tidak implement trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug"),"."),(0,i.kt)("p",null,"Solusi agar tidak error adalah dengan mengimplementasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug")," ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle"),", dengan itu semua data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," akan bisa di-print menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Selain via implementasi trait, tipe data custom bisa di-print dengan cara menambahkan atribut ",(0,i.kt)("inlineCode",{parentName:"p"},"#[derive(Debug)]")," pada definisi tipe data-nya. Namun kita tidak membahas itu pada chapter ini.")),(0,i.kt)("p",null,"Langkah pertama untuk implementasi trait adalah mencari tau terlebih dahulu spesifikasi trait yang ingin diimplementasikan. Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug")," adalah traits milik Rust standard library, maka harusnya spesifikasi bisa dilihat di dokumentasi Rust."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},(0,i.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},"https://doc.rust-lang.org/std/fmt/trait.Debug.html"))),(0,i.kt)("p",null,"Pada URL dokumentasi bisa dilihat kalau trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," memiliki struktur kurang lebih seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"pub trait Debug {\n fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), Error>;\n}\n")),(0,i.kt)("p",null,"Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," mempunyai satu spesifikasi method, bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"fmt")," yang detail strukturnya bisa dilihat di atas."),(0,i.kt)("p",null,"Kita akan implement trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," ini ke tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle"),", maka wajib hukumnya untuk menuliskan implementasi method sesuai dengan yang ada di trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug"),"."),(0,i.kt)("p",null,"Di bawah ini adalah contoh cara implementasi trait."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct Circle {\n raidus: i32,\n}\n\nimpl std::fmt::Debug for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "Circle radius: {}", self.raidus)\n }\n}\n\nfn main() {\n let circle_one = Circle{raidus: 6};\n println!("{:?}", circle_one);\n}\n')),(0,i.kt)("p",null,"Ketika program di-run, hasilnya sukses tanpa error. Artinya implementasi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," pada tipe data struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," adalah sukses."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(4666).Z,width:"457",height:"115"})),(0,i.kt)("p",null,"Cara implementasi trait ke struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," memang step-nya agak panjang, tapi penulis yakin lama-kelamaan pasti terbiasa. Ok, sekarang kita bahas satu per satu kode di atas."),(0,i.kt)("h3",{id:"-struct-circle"},"\u25c9 Struct ",(0,i.kt)("inlineCode",{parentName:"h3"},"Circle")),(0,i.kt)("p",null,"Block kode definisi struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," cukup straightforward, isinya hanya 1 property bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"radius")," bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,i.kt)("h3",{id:"-block-kode-impl-x-for-y"},"\u25c9 Block kode ",(0,i.kt)("inlineCode",{parentName:"h3"},"impl X for Y")),(0,i.kt)("p",null,"Notasi penulisan implementasi trait adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"impl X for Y"),", dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"X")," adalah trait yang ingin diimplementasikan dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Y")," adalah tipe data tujuan implementasi."),(0,i.kt)("p",null,"Pada contoh di atas, trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," diimplementasikan ke custom type struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle"),". Maka statement-nya adalah:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl std::fmt::Debug for Circle {\n // ...\n}\n")),(0,i.kt)("h3",{id:"-block-kode-method-dalam-impl"},"\u25c9 Block kode method dalam ",(0,i.kt)("inlineCode",{parentName:"h3"},"impl")),(0,i.kt)("p",null,"Block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," harus diikuti dengan implementasi method. Pada contoh ini, method ",(0,i.kt)("inlineCode",{parentName:"p"},"fmt")," milik trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Debug")," wajib untuk diimplementasikan. Spesifikasi method ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), Error>")," (lebih jelasnya silakan lihat dokumentasi)."),(0,i.kt)("p",null,"Silakan copy method tersebut kemudian paste ke dalam block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl")," yang sudah ditulis, kemudian tambahkan block kurung kurawal."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"impl std::fmt::Debug for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {\n // ...\n }\n}\n")),(0,i.kt)("p",null,"Kemudian tulis implementasi method ",(0,i.kt)("inlineCode",{parentName:"p"},"fmt")," dalam block method. Tulis statement macro ",(0,i.kt)("inlineCode",{parentName:"p"},"write")," untuk data string (yang ingin di-print) dengan tujuan adalah variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"f"),"."),(0,i.kt)("p",null,"Di contoh, format ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle radius: {}")," digunakan. Dengan ini nantinya saat printing data, yang muncul adalah text ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle radius: {}"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'impl std::fmt::Debug for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "Circle radius: {}", self.raidus)\n }\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tips untuk pengguna visual studio code dengan rust-analyzer extension ter-install, setelah selesai menulis block kode ",(0,i.kt)("inlineCode",{parentName:"p"},"impl"),", cukup jalankan ",(0,i.kt)("inlineCode",{parentName:"p"},"ctrl+space")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"cmd+space")," untuk men-trigger autocomplete suggestion. Kemudian klik opsi method yang ada disitu, maka kode implementasi method langsung muncul dengan sendirinya.")),(0,i.kt)("h3",{id:"-macro-write"},"\u25c9 Macro ",(0,i.kt)("inlineCode",{parentName:"h3"},"write")),(0,i.kt)("p",null,"Macro ini digunakan untuk menuliskan sebuah data ke object tertentu. Pada contoh kita gunakan untuk menulis string ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle radius: {}")," ke variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"f")," yang bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Formatter<'_>"),"."),(0,i.kt)("p",null,"Notasi penulisan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"write"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'// notasi penulisan\nwrite!(variabel_tujuan, data_yang_ingin_di_print, arg1, arg2, ...);\n\n// contoh penerapan\nwrite!(f, "Circle radius: {}", self.raidus);\n')),(0,i.kt)("h3",{id:"-print-data-menggunakan-formatted-print-"},"\u25c9 Print data menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"h3"},"{:?}")),(0,i.kt)("p",null,"Step terakhir adalah print variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"circle")," menggunakan macro ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Hasilnya sukses, tidak error seperti sebelumnya."),(0,i.kt)("h3",{id:"-print-data-menggunakan-formatted-print--1"},"\u25c9 Print data menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"h3"},"{}")),(0,i.kt)("p",null,"Coba tambahkan statement ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),", tetapi kali ini gunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{}"),", apakah hasilnya juga tidak error?"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(8738).Z,width:"849",height:"309"})),(0,i.kt)("p",null,"Hasilnya error, karena trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Debug")," hanya berguna untuk formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{:?}"),". Agar data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," bisa di-print menggunakan formatted print ",(0,i.kt)("inlineCode",{parentName:"p"},"{}")," maka trait ",(0,i.kt)("inlineCode",{parentName:"p"},"std::fmt::Display")," harus di-implementasikan juga."),(0,i.kt)("p",null,"Ubah kode dengan menambahkan implementasi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Display"),". Hasilnya kurang lebih seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'struct Circle {\n raidus: i32,\n}\n\nimpl std::fmt::Debug for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "Circle radius: {}", self.raidus)\n }\n}\n\nimpl std::fmt::Display for Circle {\n fn fmt(&self, f: &mut std::fmt::Formatter<\'_>) -> std::fmt::Result {\n write!(f, "Circle radius: {}", self.raidus)\n }\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Link dokumentasi trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Debug")," ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},"https://doc.rust-lang.org/std/fmt/trait.Debug.html")),(0,i.kt)("li",{parentName:"ul"},"Link dokumentasi trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Display")," ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Display.html"},"https://doc.rust-lang.org/std/fmt/trait.Display.html")))),(0,i.kt)("h2",{id:"a363-membuat-custom-trait"},"A.36.3. Membuat custom trait"),(0,i.kt)("p",null,"Pada section di atas kita telah membahas bagaimana cara implementasi traits ke tipe data. Pada bagian ini kita akan belajar tentang cara membuat definisi trait (membuat custom trait)."),(0,i.kt)("p",null,"Masih sama dengan metode sebelumnya, pembelajaran dilakukan dengan praktik. Kita gunakan skenario praktik berikut pada program selanjutnya:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},"Buat struct bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"Circle"),"."),(0,i.kt)("li",{parentName:"ol"},"Buat struct bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"Square"),"."),(0,i.kt)("li",{parentName:"ol"},"Buat trait bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"Area")," dengan isi satu buah method untuk menghitung luas bangun datar (",(0,i.kt)("inlineCode",{parentName:"li"},"calculate"),")."),(0,i.kt)("li",{parentName:"ol"},"Implementasikan trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Area")," ke dua struct tersebut.")),(0,i.kt)("p",null,"Ok, mari mulai praktikkan skenario di atas. Pertama siapkan project dengan struktur berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 calculation_spec.rs\n \u2502\u2500\u2500\u2500 two_dimensional.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("p",null,"Module ",(0,i.kt)("inlineCode",{parentName:"p"},"calculation_spec")," berisi definisi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area"),". Trait ini punya visibility akses publik, isinya hanya satu buah definisi method header bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate"),". Trait ini nantinya diimplementasikan ke struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan juga ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),", agar nantinya kedua struct tersebut memiliki method ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate")," yang berguna untuk kalkulasi luas bangun datar 2d."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/calculation_spec.rs"',title:'"src/calculation_spec.rs"'},"pub trait Area {\n fn calculate(&self) -> f64;\n}\n")),(0,i.kt)("p",null,"Kemudian siapkan file ",(0,i.kt)("inlineCode",{parentName:"p"},"two_dimensional"),", isinya dua buah struct: ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),". Pada file yang sama, siapkan juga block kode implementasi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area"),". Dengan ini maka kedua struct tersebut wajib untuk memiliki method bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate")," dengan isi adalah operasi perhitungan aritmatika untuk mencari luas bangun datar."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/two_dimensional.rs"',title:'"src/two_dimensional.rs"'},"pub struct Circle {\n pub radius: i32,\n}\n\nimpl crate::calculation_spec::Area for Circle {\n fn calculate(&self) -> f64 {\n // PI * (r ^ 2)\n // ada operasi casting ke tipe f64 karena self.radius bertipe i32\n 3.14 * (self.radius.pow(2) as f64)\n }\n}\n\npub struct Square {\n pub length: i32,\n}\n\nimpl crate::calculation_spec::Area for Square {\n fn calculate(&self) -> f64 {\n // (s ^ 2)\n // ada operasi casting ke tipe f64 karena self.length bertipe i32\n self.length.pow(2) as f64\n }\n}\n")),(0,i.kt)("p",null,"Bisa dilihat pada kode di atas, deklarasi struct beserta property memiliki visibility publik. Idealnya, saat sturct tersebut digunakan di fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," nantinya tidak akan ada error terkait visibility akses."),(0,i.kt)("p",null,"Selanjutnya, pada file ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs")," siapkan kode yang isinya registrasi module ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_spec")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"two_dimensional"),", juga definisi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," dengan isi statement pembuatan 2 variabel object untuk masing-masing tipe data struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod calculation_spec;\nmod two_dimensional;\n\nuse crate::calculation_spec::Area;\n\nfn main() {\n let circle_one = two_dimensional::Circle{ radius: 10 };\n println!("circle area: {}", circle_one.calculate());\n\n let square_one = two_dimensional::Square{ length: 5 };\n println!("square area: {}", square_one.calculate());\n}\n')),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate")," milik object bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square")," diakses untuk kemudian di-print."),(0,i.kt)("p",null,"Coba jalankan program."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(4176).Z,width:"655",height:"756"})),(0,i.kt)("p",null,"Hmm, error. Padahal trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," sudah publik, dan struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," & ",(0,i.kt)("inlineCode",{parentName:"p"},"Square")," beserta property-nya juga sudah publik. Tapi masih error."),(0,i.kt)("p",null,"Error ini disebabkan oleh trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," yang belum di-import di crate root (main). Meskipun kita tidak mengakses trait tersebut secara langsung (melainkan via method ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate")," milik struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),"), diharuskan untuk meng-import-nya juga."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Detail error beserta solusi dari error ini sebenarnya bisa dilihat di error message. Bagaimana Rust menginformasikan error sangat luar biasa informatif.")),(0,i.kt)("p",null,"Ok, sekarang ubah isi file ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs")," menjadi seperti ini, kemudian jalankan ulang program. Hasilnya tidak ada error."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod calculation_spec;\nmod two_dimensional;\n\nuse crate::calculation_spec::Area; // <------- tambahkan statement import module\n\nfn main() {\n let circle_one = two_dimensional::Circle{ radius: 10 };\n println!("circle area: {}", circle_one.calculate());\n\n let square_one = two_dimensional::Square{ length: 5 };\n println!("square area: {}", square_one.calculate());\n}\n')),(0,i.kt)("p",null,"O iya, ada beberapa hal baru pada penerapan kode di atas, berikut adalah pembahasannya:"),(0,i.kt)("h3",{id:"-method-pow-untuk-operasi-pangkat"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"pow")," untuk operasi pangkat"),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"pow")," adalah item milik tipe data numerik (",(0,i.kt)("inlineCode",{parentName:"p"},"i8"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"i16"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", ...) yang fungsinya untuk operasi pangkat."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"3.pow(2); // ===> 3 pangkat 2\n8.pow(5); // ===> 8 pangkat 5\n")),(0,i.kt)("h3",{id:"-keyword-as-untuk-casting-tipe-data"},"\u25c9 Keyword ",(0,i.kt)("inlineCode",{parentName:"h3"},"as")," untuk casting tipe data"),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"as")," digunakan untuk casting tipe data. Keyword ini bisa diterapkan pada beberapa jenis tipe data, salah satunya adalah semua tipe data numerik."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"1024 as f32; // ===> 1024 dikonversi ke tipe f32, hasinya adalah 1024.0\n3.14 as i32; // ===> 3.14 dikonversi ke tipe i32, hasinya 3 karena ada pembulatan\n")),(0,i.kt)("h2",{id:"a364-trait-sebagai-tipe-parameter"},"A.36.4. Trait sebagai tipe parameter"),(0,i.kt)("p",null,"Trait bisa digunakan sebagai tipe data parameter sebuah fungsi, contoh notasi penulisannya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result(name: String, item: &impl Area) {\n println!("{} area: {}", name, item.calculate());\n}\n')),(0,i.kt)("p",null,"Manfaat penerapan trait sebagai tipe data parameter fungsi adalah saat pemanggilan fungsi, parameter tersebut bisa diisi dengan argument data bertipe apapun dengan catatan tipe dari data tersebut mengimplementasikan trait yang sama dengan yang digunakan pada parameter."),(0,i.kt)("p",null,"Misalnya, pada fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_and_print_result")," di atas yang parameter ke-2 bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&impl Area"),", nantinya saat fungsi tersebut dipanggil, kita bisa sisipi parameter ke-2 dengan object ",(0,i.kt)("inlineCode",{parentName:"p"},"circle_one")," ataupun ",(0,i.kt)("inlineCode",{parentName:"p"},"circle_two"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let circle_one = two_dimensional::Circle{ radius: 10 };\ncalculate_and_print_result("circle".to_string(), &circle_one);\n\nlet square_one = two_dimensional::Square{ length: 5 };\ncalculate_and_print_result("square".to_string(), &square_one);\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},(0,i.kt)("inlineCode",{parentName:"p"},"&impl Area")," ini tipe data pointer ya, tipe non-pointer-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"impl Area"),". Di sini digunakan tipe data pointer untuk antisipasi ",(0,i.kt)("em",{parentName:"p"},"move semantics")," pada tipe data custom type (borrowing).")),(0,i.kt)("p",null,"Dimisalkan, fungsi tersebut parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"item"),"-nya bisa menampung beberapa jenis traits, kira-kira apakah bisa dibuat seperti itu? Misalnya ada trait lain bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"Circumference"),", dan parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"item")," milik fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"calculate_and_print_result")," harus bisa menampung data baik dari tipe yang implement trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," ataupun trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Circumference"),"."),(0,i.kt)("p",null,"Hal seperti itu bisa, caranya dengan menggunakan notasi penulisan berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result(name: String, item: &(impl Area + Circumference)) {\n println!("{} area: {}", name, item.calculate());\n println!("{} circumference: {}", name, item.calculateCircumference());\n}\n')),(0,i.kt)("p",null,"Tambahkan tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"()")," sebelum ",(0,i.kt)("inlineCode",{parentName:"p"},"impl NamaTrait"),", lalu ganti ",(0,i.kt)("inlineCode",{parentName:"p"},"NamaTrait")," dengan traits apa saja yang diinginkan dengan separator tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"+"),"."),(0,i.kt)("h2",{id:"a365-trait-bound-syntax"},"A.36.5. Trait bound syntax"),(0,i.kt)("p",null,"Penerapan trait sebagai parameter fungsi juga bisa dituliskan dalam notasi yang memanfaatkan generic. Teknik penulisan ini disebut dengan ",(0,i.kt)("em",{parentName:"p"},"trait bound syntax"),"."),(0,i.kt)("p",null,"Contohnya bisa dilihat pada kode berikut. Ada generic bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," yang merepresentasikan trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area"),", kemudian pada definisi parameter ke-2 fungsi (yaitu parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"item"),") tipenya menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"&T"),". Tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"&T")," di sini adalah ekuivalen dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"&impl Area"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result2(name: String, item: &T) {\n println!("{} area: {}", name, item.calculate());\n}\n')),(0,i.kt)("p",null,"Dimisalkan jika ada lebih dari satu trait yang digunakan sebagai tipe data paramater (misalnya trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Circumference"),"), maka penulisannya seperti ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result2(name: String, item: &T) {\n println!("{} area: {}", name, item.calculate());\n println!("{} circumference: {}", name, item.calculateCircumference());\n}\n')),(0,i.kt)("p",null,"Satu tambahan contoh lagi untuk ilustrasi yang lebih kompleks:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"fn some_function(t: &T, u: &U) -> i32 {\n // ...\n}\n")),(0,i.kt)("p",null,"Pada contoh di atas fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"some_function")," memiliki 2 generics param, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"U"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"T")," merepresentasikan trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Display")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"Clone")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"U")," merepresentasikan trait ",(0,i.kt)("inlineCode",{parentName:"li"},"Clone")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"Debug"))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai generics dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics"},"Generics"))),(0,i.kt)("h2",{id:"a366-trait-where-clause"},"A.36.6. Trait ",(0,i.kt)("inlineCode",{parentName:"h2"},"where")," clause"),(0,i.kt)("p",null,"Ada lagi alternatif penulisan trait bound syntax, yaitu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"where"),". Contoh pengaplikasiannya bisa dilihat pada kode berikut. Semua definisi fungsi di bawah ini adalah ekuivalen."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn calculate_and_print_result(name: String, item: &(impl Area + Circumference)) {\n println!("{} area: {}", name, item.calculate());\n println!("{} circumference: {}", name, item.calculateCircumference());\n}\n\nfn calculate_and_print_result2(name: String, item: &T) {\n println!("{} area: {}", name, item.calculate());\n println!("{} circumference: {}", name, item.calculateCircumference());\n}\n\nfn calculate_and_print_result3(name: String, item: &T) where T: Area + Circumference {\n println!("{} area: {}", name, item.calculate());\n}\n\nfn calculate_and_print_result4(name: String, item: &T)\nwhere\n T: Area + Circumference,\n // ... other generic params if exists\n{\n println!("{} area: {}", name, item.calculate());\n}\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai generics dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/generics"},"Generics"))),(0,i.kt)("h2",{id:"a367-trait-sebagai-return-type"},"A.36.7. Trait sebagai return type"),(0,i.kt)("p",null,"Trait bisa juga digunakan sebagai tipe data return value. Caranya gunakan notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"impl NamaTrait")," sebagai tipe data."),(0,i.kt)("p",null,"Contohnya bisa dilihat pada kode berikut. Ada dua fungsi baru dideklarasikan:"),(0,i.kt)("ol",null,(0,i.kt)("li",{parentName:"ol"},"Fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"new_circle")," dengan return type adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"impl Area"),", dan data yang dikembalikan adalah bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"two_dimensional::Circle"),"."),(0,i.kt)("li",{parentName:"ol"},"Fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"new_square")," dengan return type adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"impl Area"),", dan data yang dikembalikan adalah bertipe ",(0,i.kt)("inlineCode",{parentName:"li"},"two_dimensional::Square"),".")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let circle_one = new_circle(5);\n calculate_and_print_result4("circle".to_string(), &circle_one);\n\n let square_one = new_square(10);\n calculate_and_print_result4("square".to_string(), &square_one);\n}\n\nfn new_circle(radius: i32) -> impl Area {\n let data = two_dimensional::Circle{\n radius\n };\n data\n}\n\nfn new_square(length: i32) -> impl Area {\n two_dimensional::Square{\n length\n }\n}\n\nfn calculate_and_print_result4(name: String, item: &T)\nwhere\n T: Area,\n{\n println!("{} area: {}", name, item.calculate());\n}\n')),(0,i.kt)("p",null,"Salah satu konsekuensi dalam penerapan trait sebagai return type adalah: tipe data milik nilai yang dikembalikan terdeteksi sebagai tipe trait. Contohnya variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"circle_one")," di atas, tipe data-nya bukan ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle"),", melainkan ",(0,i.kt)("inlineCode",{parentName:"p"},"impl Area"),"."),(0,i.kt)("p",null,"Tipe data aslinya tetap bisa diakses, tapi butuh tambahan effort. Lebih jelasnya dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/trait-conversion-from-into"},"Trait \u279c Conversion (From & Into)"),"."),(0,i.kt)("h2",{id:"a368-associated-types-pada-trait"},"A.36.8. ",(0,i.kt)("em",{parentName:"h2"},"Associated types")," pada trait"),(0,i.kt)("p",null,"Associated types adalah tipe data yang didefinisikan didalam suatu trait. Associated types tidak tidak memiliki tipe data konkret saat didefinisikan, namun ketika trait di-implementasikan maka tipe tersebut harus ditentukan tipe data konkritnya."),(0,i.kt)("p",null,"Lebih jelas silakan perhatikan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"trait Shape {\n type Area;\n\n fn area(&self) -> Self::Area;\n}\n")),(0,i.kt)("p",null,"Pada definisi trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Shape")," di atas, yang disebut dengan associated types adalah tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," yang definisinya berada dalam block trait. Tipe didefinisikan tanpa assignment operator, jadi tidak ada tipe data konkretnya."),(0,i.kt)("p",null,"Associated types ini sering digunakan pada Rust programming. Contoh implementasinya bisa dilihat pada contoh di bawah ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-bash",metastring:'title="package source code structure"',title:'"package',source:!0,code:!0,'structure"':!0},"my_package\n\u2502\u2500\u2500\u2500 Cargo.toml\n\u2514\u2500\u2500\u2500 src\n \u2502\u2500\u2500\u2500 shape.rs\n \u2502\u2500\u2500\u2500 circle.rs\n \u2502\u2500\u2500\u2500 square.rs\n \u2514\u2500\u2500\u2500 main.rs\n")),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Disiapkan suatu trait bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"shape::Shape"),".",(0,i.kt)("ul",{parentName:"li"},(0,i.kt)("li",{parentName:"ul"},"Trait ini memiliki satu associated types bernama ",(0,i.kt)("inlineCode",{parentName:"li"},"Area"),"."),(0,i.kt)("li",{parentName:"ul"},"Dan memiliki sebuah definisi method header ",(0,i.kt)("inlineCode",{parentName:"li"},"area")," yang gunanya untuk menghitung luas bangun datar (",(0,i.kt)("em",{parentName:"li"},"shape"),")."))),(0,i.kt)("li",{parentName:"ul"},"Disiapkan struct ",(0,i.kt)("inlineCode",{parentName:"li"},"circle::Circle")," yang mengadopsi trait ",(0,i.kt)("inlineCode",{parentName:"li"},"shape::Shape"),"."),(0,i.kt)("li",{parentName:"ul"},"Disiapkan struct ",(0,i.kt)("inlineCode",{parentName:"li"},"square::Square")," yang mengadopsi trait ",(0,i.kt)("inlineCode",{parentName:"li"},"shape::Shape"),".")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/shape.rs"',title:'"src/shape.rs"'},"pub trait Shape {\n type Area;\n\n fn area(&self) -> Self::Area;\n}\n")),(0,i.kt)("p",null,"Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Shape")," di atas spesifikasinya mirip seperti pada contoh sebelumnya, hanya saja kali ini trait-nya di set public agar bisa diakses dari ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs")," nantinya."),(0,i.kt)("p",null,"Trait ",(0,i.kt)("inlineCode",{parentName:"p"},"Shape")," kemudian di-implementasikan ke struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square"),", kode-nya bisa dilihat berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/circle.rs"',title:'"src/circle.rs"'},"pub struct Circle {\n pub radius: f64,\n}\n\nimpl crate::shape::Shape for Circle {\n type Area = f64;\n\n fn area(&self) -> Self::Area {\n std::f64::consts::PI * self.radius * self.radius\n }\n}\n")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/square.rs"',title:'"src/square.rs"'},"pub struct Square {\n pub side: i64,\n}\n\nimpl crate::shape::Shape for Square {\n type Area = i64;\n\n fn area(&self) -> Self::Area {\n self.side * self.side\n }\n}\n")),(0,i.kt)("p",null,"Bisa dilihat pada kedua implementasi di atas, associated type ",(0,i.kt)("inlineCode",{parentName:"p"},"Area")," diisi dengan tipe concrete, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Tipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"f64")," sebagai tipe concrete ",(0,i.kt)("inlineCode",{parentName:"li"},"circle:Circle:Area")),(0,i.kt)("li",{parentName:"ul"},"Tipe data ",(0,i.kt)("inlineCode",{parentName:"li"},"i64")," sebagai tipe concrete ",(0,i.kt)("inlineCode",{parentName:"li"},"square:Square:Area"))),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Contoh di atas adalah cara pengaplikasian associated types.")),(0,i.kt)("p",null,"Lalu pada ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs"),", tipe data struct ",(0,i.kt)("inlineCode",{parentName:"p"},"circle::Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"square::Square")," digunakan untuk membuat variabel baru, yang kemudian dari variabel tersebut, method ",(0,i.kt)("inlineCode",{parentName:"p"},".area()")," milik diakses."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},'mod shape;\nmod circle;\nmod square;\n\nuse crate::shape::Shape;\n\nfn main() {\n let obj1 = circle::Circle{ radius: 10.0 };\n println!("area of circle: {:.2}", obj1.area());\n\n let obj2 = square::Square{ side: 10 };\n println!("area of square: {:}", obj2.area());\n}\n')),(0,i.kt)("p",null,"Silakan jalankan program dan lihat hasilnya."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Trait",src:t(6254).Z,width:"469",height:"130"})),(0,i.kt)("p",null,"O iya, pada ",(0,i.kt)("inlineCode",{parentName:"p"},"main.rs"),", module item ",(0,i.kt)("inlineCode",{parentName:"p"},"shape::Shape")," perlu di-import meskipun kita tidak menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"trait")," tersebut secara langsung. Jika tidak di-import, maka method ",(0,i.kt)("inlineCode",{parentName:"p"},".area()")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Circle")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Square")," tidak bisa diakses."),(0,i.kt)("h2",{id:"a369-attribute-derive"},"A.36.9. Attribute ",(0,i.kt)("inlineCode",{parentName:"h2"},"derive")),(0,i.kt)("p",null,"Ada cara lain untuk mengimplementasikan suatu trait ke dalam tipe data selain dengan menuliskan implementasinya secara eksplist, caranya menggunakan attribute ",(0,i.kt)("inlineCode",{parentName:"p"},"derive"),"."),(0,i.kt)("p",null,"Lebih detailnya dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/attributes"},"Attributes"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/traits"},"github.com/novalagung/dasarpemrogramanrust-example/../traits")),(0,i.kt)("h3",{id:"-work-in-progress"},"\u25c9 Work in progress"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang trait bounds untuk implementasi method kondisional"),(0,i.kt)("li",{parentName:"ul"},"Pembahasan tentang trait overloading")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch10-02-traits.html"},"https://doc.rust-lang.org/book/ch10-02-traits.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/primitive.i32.html"},"https://doc.rust-lang.org/std/primitive.i32.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Debug.html"},"https://doc.rust-lang.org/std/fmt/trait.Debug.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/fmt/trait.Display.html"},"https://doc.rust-lang.org/std/fmt/trait.Display.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/string/struct.String.html"},"https://doc.rust-lang.org/std/string/struct.String.html"))))}m.isMDXComponent=!0},2767:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},9856:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/traits-2-7c5e41bf2bd31dba0f78fc7afd81c7ba.png"},4666:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""},8738:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/traits-4-fe5c6a716894397c8088bae54e7194f9.png"},4176:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/traits-5-29be804051729f23db8dd64e978e2398.png"},6254:(a,e,t)=>{t.d(e,{Z:()=>n});const n=t.p+"assets/images/traits-6-8165525fc9f8604bf115634eeb974230.png"}}]); \ No newline at end of file diff --git a/assets/js/e4904969.2c9c10d1.js b/assets/js/e4904969.2c9c10d1.js deleted file mode 100644 index c0d7f6a9..00000000 --- a/assets/js/e4904969.2c9c10d1.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8732],{3905:(e,a,n)=>{n.d(a,{Zo:()=>m,kt:()=>s});var t=n(7294);function i(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function l(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function r(e){for(var a=1;a=0||(i[n]=e[n]);return i}(e,a);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var d=t.createContext({}),o=function(e){var a=t.useContext(d),n=a;return e&&(n="function"==typeof e?e(a):r(r({},a),e)),n},m=function(e){var a=o(e.components);return t.createElement(d.Provider,{value:a},e.children)},u={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},k=t.forwardRef((function(e,a){var n=e.components,i=e.mdxType,l=e.originalType,d=e.parentName,m=p(e,["components","mdxType","originalType","parentName"]),k=o(n),s=i,c=k["".concat(d,".").concat(s)]||k[s]||u[s]||l;return n?t.createElement(c,r(r({ref:a},m),{},{components:n})):t.createElement(c,r({ref:a},m))}));function s(e,a){var n=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var l=n.length,r=new Array(l);r[0]=k;var p={};for(var d in a)hasOwnProperty.call(a,d)&&(p[d]=a[d]);p.originalType=e,p.mdxType="string"==typeof e?e:i,r[1]=p;for(var o=2;o{n.r(a),n.d(a,{assets:()=>d,contentTitle:()=>r,default:()=>u,frontMatter:()=>l,metadata:()=>p,toc:()=>o});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:38,title:"A.38. Tipe Data \u279c Option",sidebar_label:"A.38. Tipe Data \u279c Option"},r=void 0,p={unversionedId:"basic/option-type",id:"basic/option-type",title:"A.38. Tipe Data \u279c Option",description:"Option adalah salah satu tipe data penting pada Rust programming, digunakan untuk menampung data yang isinya bisa berpotensi kosong (None). Chapter ini membahas tentang tipe data tersebut.",source:"@site/docs/basic/option-type.md",sourceDirName:"basic",slug:"/basic/option-type",permalink:"/basic/option-type",draft:!1,tags:[],version:"current",sidebarPosition:38,frontMatter:{sidebar_position:38,title:"A.38. Tipe Data \u279c Option",sidebar_label:"A.38. Tipe Data \u279c Option"},sidebar:"tutorialSidebar",previous:{title:"A.37. Generics",permalink:"/basic/generics"},next:{title:"A.39. Tipe Data \u279c Result",permalink:"/basic/result-type"}},d={},o=[{value:"A.38.1. Konsep Option",id:"a381-konsep-option",level:2},{value:"A.38.2. Pattern matching pada tipe Option",id:"a382-pattern-matching-pada-tipe-option",level:2},{value:"\u25c9 Tips pattern matching",id:"-tips-pattern-matching",level:3},{value:"A.38.3. Method tipe data Option",id:"a383-method-tipe-data-option",level:2},{value:"\u25c9 Method unwrap",id:"-method-unwrap",level:3},{value:"\u25c9 Method is_some & is_none",id:"-method-is_some--is_none",level:3},{value:"\u25c9 Method unwrap_or_default",id:"-method-unwrap_or_default",level:3},{value:"\u25c9 Method unwrap_or",id:"-method-unwrap_or",level:3},{value:"\u25c9 Method unwrap_or_else",id:"-method-unwrap_or_else",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:o};function u(e){let{components:a,...l}=e;return(0,i.kt)("wrapper",(0,t.Z)({},m,l,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"Option")," adalah salah satu tipe data penting pada Rust programming, digunakan untuk menampung data yang isinya bisa berpotensi kosong (",(0,i.kt)("inlineCode",{parentName:"p"},"None"),"). Chapter ini membahas tentang tipe data tersebut."),(0,i.kt)("h2",{id:"a381-konsep-option"},"A.38.1. Konsep ",(0,i.kt)("inlineCode",{parentName:"h2"},"Option")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," adalah enum dengan isi 2 buah enum value:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Option::Some")," (atau ",(0,i.kt)("inlineCode",{parentName:"li"},"Some"),"), digunakan untuk menandai bahwa data memiliki value/nilai."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Option::None")," (atau ",(0,i.kt)("inlineCode",{parentName:"li"},"None"),"), digunakan untuk menandai bawah data adalah tidak ada nilainya.")),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"None")," bisa disamakan dengan nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"null")," atau ",(0,i.kt)("inlineCode",{parentName:"li"},"nil")," pada bahasa pemrograman lain."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"T")," merupakan parameter generic. Lebih jelasnya mengenai generic dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics"),"."))),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," memiliki notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah tipe data sebenarnya yang dibungkus oleh enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),"."),(0,i.kt)("p",null,"Berikut adalah contoh cara penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn divider(a: i32, b: i32) -> Option {\n if b == 0 {\n return None;\n }\n\n let result = a / b;\n return Some(result);\n}\n\nfn main() {\n let result1 = divider(10, 5);\n println!("result: {:?}", result1);\n\n let result2: Option = divider(10, 0);\n println!("result: {:?}", result2);\n}\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider")," di atas tugasnya adalah melakukan operasi aritmatika pembagian angka numerik ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"a")," dibagi ",(0,i.kt)("inlineCode",{parentName:"p"},"b"),"."),(0,i.kt)("p",null,"Pada fungsi tersebut terdapat pengecekan apabila nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"b")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),", maka yang dikembalikan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", selainnya maka hasil operasi pembagian dikembalikan dibungkus dalam enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),". Bisa dilihat pada statement return value fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider"),", nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"result")," dibungkus menggunakan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),"."),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider")," nilai baliknya bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),". Dari tipe data yang digunakan nantinya bisa diprediksi pasti akan ada 2 potensi value:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Return value adalah enum value ",(0,i.kt)("inlineCode",{parentName:"li"},"None"),", muncul ketika nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"b")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"0")),(0,i.kt)("li",{parentName:"ul"},"Return value adalah nilai hasil numerik yang dibungkus oleh enum value ",(0,i.kt)("inlineCode",{parentName:"li"},"Some"))),(0,i.kt)("p",null,"Output program di atas saat di-run:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Option type",src:n(1384).Z,width:"450",height:"127"})),(0,i.kt)("h2",{id:"a382-pattern-matching-pada-tipe-option"},"A.38.2. Pattern matching pada tipe ",(0,i.kt)("inlineCode",{parentName:"h2"},"Option")),(0,i.kt)("p",null,"Dalam penerapannya, ketika ada data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," artinya data tersebut berpotensi untuk berisi nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"None")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),", pasti antara 2 nilai tersebut."),(0,i.kt)("p",null,"Umumnya penggunaan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," selalu diikuti dengan seleksi kondisi. Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," bisa digunakan dalam seleksi kondisi, namun dalam Praktiknya lebih baik menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," karena memberikan kemudahan dalam pengaksesan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Some")," (dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah data yang kita cari dibungkus dalam enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),")."),(0,i.kt)("p",null,"Mari kita praktikkan. Ubah isi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," dengan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result1 = divider(10, 5);\nmatch result1 {\n None => println!("cannot divide by 0"),\n Some(x) => println!("result: {x}"),\n}\n\nlet result2 = divider(10, 0);\nmatch result2 {\n None => println!("cannot divide by 0"),\n Some(x) => {\n println!("result: {}", x)\n },\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Option type",src:n(4890).Z,width:"446",height:"124"})),(0,i.kt)("p",null,"Bisa dilihat cara mengambil nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dari enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some")," cukup mudah dengan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),". Penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk seleksi kondisi biasa disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"pattern matching")," dan teknik ini sangat fleksibel dan advance."),(0,i.kt)("p",null,"Sebagai contoh, dengan penerapan match yang seperti ini kita bisa meng-handle 3 skenario seleksi kondisi:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result1 = divider(10, 5);\nmatch result1 {\n None => println!("cannot divide by 0"),\n Some(2) => println!("the result is 2"),\n Some(x) => println!("result: {x}"),\n}\n')),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-1: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"None"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"cannot divide by 0")),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-2: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"2"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"the result is 2")),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-1: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Some")," selain dari ",(0,i.kt)("inlineCode",{parentName:"li"},"None")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"Some(2)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"result: {x}"))),(0,i.kt)("h3",{id:"-tips-pattern-matching"},"\u25c9 Tips pattern matching"),(0,i.kt)("p",null,"Silakan perhatikan kode yang sudah kita praktikkan berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10, 5);\nmatch result {\n None => println!("cannot divide by 0"),\n Some(x) => println!("result: {x}"),\n}\n')),(0,i.kt)("p",null,"Penerapan pattern matching seperti contoh di atas memiliki konsekuensi, yaitu variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"x")," hanya bisa diakses pada block ",(0,i.kt)("inlineCode",{parentName:"p"},"Some(x)")," saja."),(0,i.kt)("p",null,"Adakalanya kita butuh untuk mengeluarkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"x")," ke luar block. Hal seperti ini mudah untuk dilakukan, dan ada beberapa cara yang bisa dipilih, namun menurut penulis yang paling elegan adalah cara berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let result = match divider(10, 5) {\n None => {\n println!("cannot divide by 0");\n 0\n },\n Some(x) => x,\n };\n\n println!("result: {:?}", result);\n}\n')),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10, 5)")," mengembalikan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),". Data tersebut digunakan pada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," seperti biasa. Namun pada contoh di atas ada yang berbeda, yaitu return value dari statement ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," ditampung ke variabel (",(0,i.kt)("inlineCode",{parentName:"p"},"result"),")."),(0,i.kt)("p",null,"Isi dari pattern matching ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," sendiri ada dua:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Ketika block ",(0,i.kt)("inlineCode",{parentName:"li"},"None")," match, pesan error di-print kemudian nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"0")," dijadikan return statement ",(0,i.kt)("inlineCode",{parentName:"li"},"match"),"."),(0,i.kt)("li",{parentName:"ul"},"Ketika block ",(0,i.kt)("inlineCode",{parentName:"li"},"Some")," match, data ",(0,i.kt)("inlineCode",{parentName:"li"},"x")," dijadikan return value statement ",(0,i.kt)("inlineCode",{parentName:"li"},"match"),".")),(0,i.kt)("p",null,"Dengan penerapan pattern matching seperti di atas, maka variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"result")," akan selalu berisi data hasil operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10, 5)"),". Dengan pengecualian ketika ada error, pesan errornya dimunculkan kemudian hasil operasi pembagian di-set sebagai ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai pattern matching dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"))),(0,i.kt)("h2",{id:"a383-method-tipe-data-option"},"A.38.3. Method tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Option")),(0,i.kt)("h3",{id:"-method-unwrap"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap")),(0,i.kt)("p",null,"Isi dari enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some")," bisa diakses tanpa menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," dengan cara memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result1 = divider(10, 5);\nif result1 != None {\n let number = result1.unwrap();\n println!("result: {}", number);\n}\n')),(0,i.kt)("p",null,"Penggunaan method tersebut sangat dianjurkan diiringi dengan seleksi kondisi untuk memastikan data ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," tidak berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),". Jika data ternyata adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None")," dan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," diakses, hasilnya adalah panic error. Contohnya bisa dilihat pada gambar berikut:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Option type",src:n(5499).Z,width:"659",height:"262"})),(0,i.kt)("p",null,"Selain method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," ada beberapa method sejenis lainnya yang bisa dimanfaatkan untuk pengambilan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),". Kita akan bahas satu per satu."),(0,i.kt)("h3",{id:"-method-is_some--is_none"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_some")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_none")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_some")," menghasilkan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," jika data isinya adalah enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),". Sedangkan ",(0,i.kt)("inlineCode",{parentName:"p"},"is_none")," bernilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," jika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),"."),(0,i.kt)("p",null,"Contoh penerapannya bisa dilihat pada program berikut. Kesemua seleksi kondisi pada konteks ini menghasilkan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result1 = divider(10, 5);\n\nif result1 != None {\n let number = result1.unwrap();\n println!("result: {}", number);\n}\n\nif result1.is_some() {\n let number = result1.unwrap();\n println!("result: {}", number);\n}\n\nif !result1.is_none() {\n let number = result1.unwrap();\n println!("result: {}", number);\n}\n')),(0,i.kt)("h3",{id:"-method-unwrap_or_default"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or_default")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_default")," mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),". Jika data ternyata isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", maka nilai yang dikembalikan adalah ",(0,i.kt)("em",{parentName:"p"},"default value")," dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("p",null,"Sebagai contoh, pada kode berikut statement ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10, 0)")," return type-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),", sedangkan return value-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),". Pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_default")," menghasilkan ",(0,i.kt)("em",{parentName:"p"},"default value")," dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"0")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result2 = divider(10, 0);\nlet number = result2.unwrap_or_default();\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("h3",{id:"-method-unwrap_or"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or")),(0,i.kt)("p",null,"Method ini mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),", namun jika data ternyata isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", maka nilai yang dikembalikan adalah argument pemanggilan method tersebut."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result2 = divider(10, 0);\nlet number = result2.unwrap_or(0);\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("p",null,"Pada contoh di atas argument pemanggilan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or")," adalah angka ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),", artinya ketika ",(0,i.kt)("inlineCode",{parentName:"p"},"result2")," isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None")," maka angka ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," adalah return value pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or"),"."),(0,i.kt)("h3",{id:"-method-unwrap_or_else"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or_else")),(0,i.kt)("p",null,"Method ini mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),", namun jika data isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", maka nilai yang dikembalikan adalah hasil eksekusi closure yang disisipkan saat memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_else"),". Contoh pengaplikasiannya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result2 = divider(10, 0);\nlet number = result2.unwrap_or_else(|| 0);\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("p",null,"Closure harus dalam notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"FnOnce() -> T")," dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," pada konteks ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Closure ",(0,i.kt)("inlineCode",{parentName:"p"},"|| 0")," adalah kependekan dari ",(0,i.kt)("inlineCode",{parentName:"p"},"|| -> i32 { 0 }"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai closure dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/closures"},"Closures"),".")),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/option_type"},"github.com/novalagung/dasarpemrogramanrust-example/../option_type")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/closures"},"Closures"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch18-03-pattern-syntax.html"},"https://doc.rust-lang.org/book/ch18-03-pattern-syntax.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/option/index.html"},"https://doc.rust-lang.org/std/option/index.html"))))}u.isMDXComponent=!0},1384:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},4890:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},5499:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/option-type-3-e2943c7c0a60b13cd69ce8461805846c.png"}}]); \ No newline at end of file diff --git a/assets/js/e4904969.d2252a8b.js b/assets/js/e4904969.d2252a8b.js new file mode 100644 index 00000000..90a9d33b --- /dev/null +++ b/assets/js/e4904969.d2252a8b.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[8732],{3905:(e,a,n)=>{n.d(a,{Zo:()=>m,kt:()=>s});var t=n(7294);function i(e,a,n){return a in e?Object.defineProperty(e,a,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[a]=n,e}function l(e,a){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(e);a&&(t=t.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),n.push.apply(n,t)}return n}function r(e){for(var a=1;a=0||(i[n]=e[n]);return i}(e,a);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(i[n]=e[n])}return i}var d=t.createContext({}),o=function(e){var a=t.useContext(d),n=a;return e&&(n="function"==typeof e?e(a):r(r({},a),e)),n},m=function(e){var a=o(e.components);return t.createElement(d.Provider,{value:a},e.children)},u={inlineCode:"code",wrapper:function(e){var a=e.children;return t.createElement(t.Fragment,{},a)}},k=t.forwardRef((function(e,a){var n=e.components,i=e.mdxType,l=e.originalType,d=e.parentName,m=p(e,["components","mdxType","originalType","parentName"]),k=o(n),s=i,c=k["".concat(d,".").concat(s)]||k[s]||u[s]||l;return n?t.createElement(c,r(r({ref:a},m),{},{components:n})):t.createElement(c,r({ref:a},m))}));function s(e,a){var n=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var l=n.length,r=new Array(l);r[0]=k;var p={};for(var d in a)hasOwnProperty.call(a,d)&&(p[d]=a[d]);p.originalType=e,p.mdxType="string"==typeof e?e:i,r[1]=p;for(var o=2;o{n.r(a),n.d(a,{assets:()=>d,contentTitle:()=>r,default:()=>u,frontMatter:()=>l,metadata:()=>p,toc:()=>o});var t=n(7462),i=(n(7294),n(3905));const l={sidebar_position:38,title:"A.38. Tipe Data \u279c Option",sidebar_label:"A.38. Tipe Data \u279c Option"},r=void 0,p={unversionedId:"basic/option-type",id:"basic/option-type",title:"A.38. Tipe Data \u279c Option",description:"Option adalah salah satu tipe data penting pada Rust programming, digunakan untuk menampung data yang isinya bisa berpotensi kosong (None). Chapter ini membahas tentang tipe data tersebut.",source:"@site/docs/basic/option-type.md",sourceDirName:"basic",slug:"/basic/option-type",permalink:"/basic/option-type",draft:!1,tags:[],version:"current",sidebarPosition:38,frontMatter:{sidebar_position:38,title:"A.38. Tipe Data \u279c Option",sidebar_label:"A.38. Tipe Data \u279c Option"},sidebar:"tutorialSidebar",previous:{title:"A.37. Generics",permalink:"/basic/generics"},next:{title:"A.39. Tipe Data \u279c Result",permalink:"/basic/result-type"}},d={},o=[{value:"A.38.1. Konsep Option",id:"a381-konsep-option",level:2},{value:"A.38.2. Pattern matching pada tipe Option",id:"a382-pattern-matching-pada-tipe-option",level:2},{value:"\u25c9 Tips pattern matching",id:"-tips-pattern-matching",level:3},{value:"A.38.3. Method tipe data Option",id:"a383-method-tipe-data-option",level:2},{value:"\u25c9 Method unwrap",id:"-method-unwrap",level:3},{value:"\u25c9 Method is_some & is_none",id:"-method-is_some--is_none",level:3},{value:"\u25c9 Method unwrap_or_default",id:"-method-unwrap_or_default",level:3},{value:"\u25c9 Method unwrap_or",id:"-method-unwrap_or",level:3},{value:"\u25c9 Method unwrap_or_else",id:"-method-unwrap_or_else",level:3},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Chapter relevan lainnya",id:"-chapter-relevan-lainnya",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:o};function u(e){let{components:a,...l}=e;return(0,i.kt)("wrapper",(0,t.Z)({},m,l,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"Option")," adalah salah satu tipe data penting pada Rust programming, digunakan untuk menampung data yang isinya bisa berpotensi kosong (",(0,i.kt)("inlineCode",{parentName:"p"},"None"),"). Chapter ini membahas tentang tipe data tersebut."),(0,i.kt)("h2",{id:"a381-konsep-option"},"A.38.1. Konsep ",(0,i.kt)("inlineCode",{parentName:"h2"},"Option")),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," adalah enum dengan isi 2 buah enum value:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Option::Some")," (atau ",(0,i.kt)("inlineCode",{parentName:"li"},"Some"),"), digunakan untuk menandai bahwa data memiliki value/nilai."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"Option::None")," (atau ",(0,i.kt)("inlineCode",{parentName:"li"},"None"),"), digunakan untuk menandai bawah data adalah tidak ada nilainya.")),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"None")," bisa disamakan dengan nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"null")," atau ",(0,i.kt)("inlineCode",{parentName:"li"},"nil")," pada bahasa pemrograman lain."),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("inlineCode",{parentName:"li"},"T")," merupakan parameter generic. Lebih jelasnya mengenai generic dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics"),"."))),(0,i.kt)("p",null,"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," memiliki notasi penulisan ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah tipe data sebenarnya yang dibungkus oleh enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),"."),(0,i.kt)("p",null,"Berikut adalah contoh cara penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn divider(a: i32, b: i32) -> Option {\n if b == 0 {\n return None;\n }\n\n let result = a / b;\n return Some(result);\n}\n\nfn main() {\n let result1 = divider(10, 5);\n println!("result: {:?}", result1);\n\n let result2: Option = divider(10, 0);\n println!("result: {:?}", result2);\n}\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider")," di atas tugasnya adalah melakukan operasi aritmatika pembagian angka numerik ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"a")," dibagi ",(0,i.kt)("inlineCode",{parentName:"p"},"b"),"."),(0,i.kt)("p",null,"Pada fungsi tersebut terdapat pengecekan apabila nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"b")," adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),", maka yang dikembalikan adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", selainnya maka hasil operasi pembagian dikembalikan dibungkus dalam enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),". Bisa dilihat pada statement return value fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider"),", nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"result")," dibungkus menggunakan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),"."),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider")," nilai baliknya bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),". Dari tipe data yang digunakan nantinya bisa diprediksi pasti akan ada 2 potensi value:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Return value adalah enum value ",(0,i.kt)("inlineCode",{parentName:"li"},"None"),", muncul ketika nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"b")," adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"0")),(0,i.kt)("li",{parentName:"ul"},"Return value adalah nilai hasil numerik yang dibungkus oleh enum value ",(0,i.kt)("inlineCode",{parentName:"li"},"Some"))),(0,i.kt)("p",null,"Output program di atas saat di-run:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Option type",src:n(1384).Z,width:"450",height:"127"})),(0,i.kt)("h2",{id:"a382-pattern-matching-pada-tipe-option"},"A.38.2. Pattern matching pada tipe ",(0,i.kt)("inlineCode",{parentName:"h2"},"Option")),(0,i.kt)("p",null,"Dalam penerapannya, ketika ada data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," artinya data tersebut berpotensi untuk berisi nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"None")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),", pasti antara 2 nilai tersebut."),(0,i.kt)("p",null,"Umumnya penggunaan tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," selalu diikuti dengan seleksi kondisi. Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," bisa digunakan dalam seleksi kondisi, namun dalam Praktiknya lebih baik menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," karena memberikan kemudahan dalam pengaksesan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Some")," (yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," adalah data yang kita cari dibungkus dalam enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),")."),(0,i.kt)("p",null,"Mari kita praktikkan. Ubah isi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," dengan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result1 = divider(10, 5);\nmatch result1 {\n None => println!("cannot divide by 0"),\n Some(x) => println!("result: {x}"),\n}\n\nlet result2 = divider(10, 0);\nmatch result2 {\n None => println!("cannot divide by 0"),\n Some(x) => {\n println!("result: {}", x)\n },\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Option type",src:n(4890).Z,width:"446",height:"124"})),(0,i.kt)("p",null,"Bisa dilihat cara mengambil nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," dari enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some")," cukup mudah dengan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match"),". Penerapan ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," untuk seleksi kondisi biasa disebut dengan ",(0,i.kt)("strong",{parentName:"p"},"pattern matching")," dan teknik ini sangat fleksibel dan advance."),(0,i.kt)("p",null,"Sebagai contoh, dengan penerapan match yang seperti ini kita bisa meng-handle 3 skenario seleksi kondisi:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result1 = divider(10, 5);\nmatch result1 {\n None => println!("cannot divide by 0"),\n Some(2) => println!("the result is 2"),\n Some(x) => println!("result: {x}"),\n}\n')),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-1: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"None"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"cannot divide by 0")),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-2: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"2"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"the result is 2")),(0,i.kt)("li",{parentName:"ul"},"Kondisi ke-1: jika nilai adalah ",(0,i.kt)("inlineCode",{parentName:"li"},"Some")," selain dari ",(0,i.kt)("inlineCode",{parentName:"li"},"None")," dan ",(0,i.kt)("inlineCode",{parentName:"li"},"Some(2)"),", maka munculkan pesan ",(0,i.kt)("inlineCode",{parentName:"li"},"result: {x}"))),(0,i.kt)("h3",{id:"-tips-pattern-matching"},"\u25c9 Tips pattern matching"),(0,i.kt)("p",null,"Silakan perhatikan kode yang sudah kita praktikkan berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result = divider(10, 5);\nmatch result {\n None => println!("cannot divide by 0"),\n Some(x) => println!("result: {x}"),\n}\n')),(0,i.kt)("p",null,"Penerapan pattern matching seperti contoh di atas memiliki konsekuensi, yaitu variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"x")," hanya bisa diakses pada block ",(0,i.kt)("inlineCode",{parentName:"p"},"Some(x)")," saja."),(0,i.kt)("p",null,"Adakalanya kita butuh untuk mengeluarkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"x")," ke luar block. Hal seperti ini mudah untuk dilakukan, dan ada beberapa cara yang bisa dipilih, namun menurut penulis yang paling elegan adalah cara berikut ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let result = match divider(10, 5) {\n None => {\n println!("cannot divide by 0");\n 0\n },\n Some(x) => x,\n };\n\n println!("result: {:?}", result);\n}\n')),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10, 5)")," mengembalikan data bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),". Data tersebut digunakan pada keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," seperti biasa. Namun pada contoh di atas ada yang berbeda, yaitu return value dari statement ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," ditampung ke variabel (",(0,i.kt)("inlineCode",{parentName:"p"},"result"),")."),(0,i.kt)("p",null,"Isi dari pattern matching ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," sendiri ada dua:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Ketika block ",(0,i.kt)("inlineCode",{parentName:"li"},"None")," match, pesan error di-print kemudian nilai ",(0,i.kt)("inlineCode",{parentName:"li"},"0")," dijadikan return statement ",(0,i.kt)("inlineCode",{parentName:"li"},"match"),"."),(0,i.kt)("li",{parentName:"ul"},"Ketika block ",(0,i.kt)("inlineCode",{parentName:"li"},"Some")," match, data ",(0,i.kt)("inlineCode",{parentName:"li"},"x")," dijadikan return value statement ",(0,i.kt)("inlineCode",{parentName:"li"},"match"),".")),(0,i.kt)("p",null,"Dengan penerapan pattern matching seperti di atas, maka variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"result")," akan selalu berisi data hasil operasi ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10, 5)"),". Dengan pengecualian ketika ada error, pesan errornya dimunculkan kemudian hasil operasi pembagian di-set sebagai ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai pattern matching dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pattern-matching"},"Pattern Matching"))),(0,i.kt)("h2",{id:"a383-method-tipe-data-option"},"A.38.3. Method tipe data ",(0,i.kt)("inlineCode",{parentName:"h2"},"Option")),(0,i.kt)("h3",{id:"-method-unwrap"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap")),(0,i.kt)("p",null,"Isi dari enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some")," bisa diakses tanpa menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"match")," dengan cara memanfaatkan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," milik ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result1 = divider(10, 5);\nif result1 != None {\n let number = result1.unwrap();\n println!("result: {}", number);\n}\n')),(0,i.kt)("p",null,"Penggunaan method tersebut sangat dianjurkan diiringi dengan seleksi kondisi untuk memastikan data ",(0,i.kt)("inlineCode",{parentName:"p"},"Option")," tidak berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),". Jika data ternyata adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None")," dan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," diakses, hasilnya adalah panic error. Contohnya bisa dilihat pada gambar berikut:"),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Option type",src:n(5499).Z,width:"659",height:"262"})),(0,i.kt)("p",null,"Selain method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," ada beberapa method sejenis lainnya yang bisa dimanfaatkan untuk pengambilan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),". Kita akan bahas satu per satu."),(0,i.kt)("h3",{id:"-method-is_some--is_none"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_some")," & ",(0,i.kt)("inlineCode",{parentName:"h3"},"is_none")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"is_some")," menghasilkan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," jika data isinya adalah enum value ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),". Sedangkan ",(0,i.kt)("inlineCode",{parentName:"p"},"is_none")," bernilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true")," jika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),"."),(0,i.kt)("p",null,"Contoh penerapannya bisa dilihat pada program berikut. Kesemua seleksi kondisi pada konteks ini menghasilkan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result1 = divider(10, 5);\n\nif result1 != None {\n let number = result1.unwrap();\n println!("result: {}", number);\n}\n\nif result1.is_some() {\n let number = result1.unwrap();\n println!("result: {}", number);\n}\n\nif !result1.is_none() {\n let number = result1.unwrap();\n println!("result: {}", number);\n}\n')),(0,i.kt)("h3",{id:"-method-unwrap_or_default"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or_default")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_default")," mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),". Jika data ternyata isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", maka nilai yang dikembalikan adalah ",(0,i.kt)("em",{parentName:"p"},"default value")," dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"T"),"."),(0,i.kt)("p",null,"Sebagai contoh, pada kode berikut statement ",(0,i.kt)("inlineCode",{parentName:"p"},"divider(10, 0)")," return type-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),", sedangkan return value-nya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),". Pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_default")," menghasilkan ",(0,i.kt)("em",{parentName:"p"},"default value")," dari tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),", yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"0")),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result2 = divider(10, 0);\nlet number = result2.unwrap_or_default();\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("h3",{id:"-method-unwrap_or"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or")),(0,i.kt)("p",null,"Method ini mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),", namun jika data ternyata isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", maka nilai yang dikembalikan adalah argument pemanggilan method tersebut."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result2 = divider(10, 0);\nlet number = result2.unwrap_or(0);\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("p",null,"Pada contoh di atas argument pemanggilan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or")," adalah angka ",(0,i.kt)("inlineCode",{parentName:"p"},"0"),", artinya ketika ",(0,i.kt)("inlineCode",{parentName:"p"},"result2")," isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None")," maka angka ",(0,i.kt)("inlineCode",{parentName:"p"},"0")," adalah return value pengaksesan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or"),"."),(0,i.kt)("h3",{id:"-method-unwrap_or_else"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"unwrap_or_else")),(0,i.kt)("p",null,"Method ini mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," ketika data berisi ",(0,i.kt)("inlineCode",{parentName:"p"},"Some"),", namun jika data isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", maka nilai yang dikembalikan adalah hasil eksekusi closure yang disisipkan saat memanggil method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap_or_else"),". Contoh pengaplikasiannya:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let result2 = divider(10, 0);\nlet number = result2.unwrap_or_else(|| 0);\nprintln!("result: {}", number);\n// result: 0\n')),(0,i.kt)("p",null,"Closure harus dalam notasi ",(0,i.kt)("inlineCode",{parentName:"p"},"FnOnce() -> T")," yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"T")," pada konteks ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"i32"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Closure ",(0,i.kt)("inlineCode",{parentName:"p"},"|| 0")," adalah kependekan dari ",(0,i.kt)("inlineCode",{parentName:"p"},"|| -> i32 { 0 }"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai closure dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/closures"},"Closures"),".")),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/option_type"},"github.com/novalagung/dasarpemrogramanrust-example/../option_type")),(0,i.kt)("h3",{id:"-chapter-relevan-lainnya"},"\u25c9 Chapter relevan lainnya"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/generics"},"Generics")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/pattern-matching"},"Pattern Matching")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"/basic/closures"},"Closures"))),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch18-03-pattern-syntax.html"},"https://doc.rust-lang.org/book/ch18-03-pattern-syntax.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/option/index.html"},"https://doc.rust-lang.org/std/option/index.html"))))}u.isMDXComponent=!0},1384:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},4890:(e,a,n)=>{n.d(a,{Z:()=>t});const t=""},5499:(e,a,n)=>{n.d(a,{Z:()=>t});const t=n.p+"assets/images/option-type-3-e2943c7c0a60b13cd69ce8461805846c.png"}}]); \ No newline at end of file diff --git a/assets/js/e7bd9b37.38bdc3e5.js b/assets/js/e7bd9b37.38bdc3e5.js deleted file mode 100644 index 8fb7f00a..00000000 --- a/assets/js/e7bd9b37.38bdc3e5.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7539],{3905:(a,e,t)=>{t.d(e,{Zo:()=>u,kt:()=>o});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function p(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var s=n.createContext({}),m=function(a){var e=n.useContext(s),t=e;return a&&(t="function"==typeof a?a(e):p(p({},e),a)),t},u=function(a){var e=m(a.components);return n.createElement(s.Provider,{value:e},a.children)},d={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},k=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,s=a.parentName,u=l(a,["components","mdxType","originalType","parentName"]),k=m(t),o=i,h=k["".concat(s,".").concat(o)]||k[o]||d[o]||r;return t?n.createElement(h,p(p({ref:e},u),{},{components:t})):n.createElement(h,p({ref:e},u))}));function o(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,p=new Array(r);p[0]=k;var l={};for(var s in e)hasOwnProperty.call(e,s)&&(l[s]=e[s]);l.originalType=a,l.mdxType="string"==typeof a?a:i,p[1]=l;for(var m=2;m{t.r(e),t.d(e,{assets:()=>s,contentTitle:()=>p,default:()=>d,frontMatter:()=>r,metadata:()=>l,toc:()=>m});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:18,title:"A.18. Module System \u279c Path & Item",sidebar_label:"A.18. Module System \u279c Path & Item"},p=void 0,l={unversionedId:"basic/path-item",id:"basic/path-item",title:"A.18. Module System \u279c Path & Item",description:"Chapter ini membahas tentang konsep path dan item dalam Rust programming.",source:"@site/docs/basic/path-item.md",sourceDirName:"basic",slug:"/basic/path-item",permalink:"/basic/path-item",draft:!1,tags:[],version:"current",sidebarPosition:18,frontMatter:{sidebar_position:18,title:"A.18. Module System \u279c Path & Item",sidebar_label:"A.18. Module System \u279c Path & Item"},sidebar:"tutorialSidebar",previous:{title:"A.17. Function",permalink:"/basic/function"},next:{title:"A.19. Module System \u279c Package & Crate",permalink:"/basic/package-crate"}},s={},m=[{value:"A.18.1. Rust Paths",id:"a181-rust-paths",level:2},{value:"A.18.2. Absolute & relative paths",id:"a182-absolute--relative-paths",level:2},{value:"A.18.3. Rust Items",id:"a183-rust-items",level:2},{value:"A.18.4. Penerapan paths dalam penggaksesan item",id:"a184-penerapan-paths-dalam-penggaksesan-item",level:2},{value:"\u25c9 std::string::String::new()",id:"-stdstringstringnew",level:3},{value:"\u25c9 std::io::stdin()",id:"-stdiostdin",level:3},{value:"\u25c9 stdin_reader.read_line(&mut message)",id:"-stdin_readerread_linemut-message",level:3},{value:"\u25c9 Pengecekan error stdin_reader",id:"-pengecekan-error-stdin_reader",level:3},{value:"\u25c9 Menampilkan isi message",id:"-menampilkan-isi-message",level:3},{value:"A.18.5. Penggunaan keyword use untuk import path",id:"a185-penggunaan-keyword-use-untuk-import-path",level:2},{value:"\u25c9 Import beberapa items yang parent path-nya yang sama",id:"-import-beberapa-items-yang-parent-path-nya-yang-sama",level:3},{value:"\u25c9 Import semua items dalam suatu path",id:"-import-semua-items-dalam-suatu-path",level:3},{value:"A.18.6. Pembahasan lanjutan",id:"a186-pembahasan-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:m};function d(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},u,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas tentang konsep ",(0,i.kt)("em",{parentName:"p"},"path")," dan ",(0,i.kt)("em",{parentName:"p"},"item")," dalam Rust programming."),(0,i.kt)("h2",{id:"a181-rust-paths"},"A.18.1. Rust ",(0,i.kt)("em",{parentName:"h2"},"Paths")),(0,i.kt)("p",null,"Paths (atau Path) adalah notasi penulisan alamat sebuah item, contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),"."),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration")," adalah ",(0,i.kt)("em",{parentName:"p"},"path")," untuk item yang isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"struct")," bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration"),". Item bisa berupa banyak jenis, bisa saja struct, atau ",(0,i.kt)("em",{parentName:"p"},"macro"),", konstanta, atau lainnya. Lebih jelasnya silakan cek pada halaman dokumentasi ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),"."),(0,i.kt)("p",null,"Sebuah path bisa memiliki banyak bagian (biasa disebut ",(0,i.kt)("em",{parentName:"p"},"segment"),"), sebagai contoh, path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration")," memiliki 3 segmen yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"std"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"time"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration"),". Karakter ",(0,i.kt)("inlineCode",{parentName:"p"},"::")," digunakan dalam penulisan path sebagai pembatas antar segmen (jika path memiliki lebih dari 1 segmen)."),(0,i.kt)("p",null,"Dalam sebuah path, yang disebut dengan item adalah segment terakhir. Contohnya path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),", maka item yang dituju adalah struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Rust paths mirip seperti konsep ",(0,i.kt)("em",{parentName:"p"},"filesystem")," path di sistem operasi. Seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"C:\\Users\\novalagung\\Desktop")," di windows, atau ",(0,i.kt)("inlineCode",{parentName:"p"},"/etc/nginx/conf.d/nginx.conf")," di Unix/Linux."),(0,i.kt)("p",{parentName:"blockquote"},"Di Rust, path tidak menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"\\")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"/")," sebagai separator, melainkan ",(0,i.kt)("inlineCode",{parentName:"p"},"::"),".")),(0,i.kt)("p",null,"Jika di-breakdown, berikut adalah penjelasan dari setiap kombinasi segmen path pada contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Path ",(0,i.kt)("inlineCode",{parentName:"li"},"std")," \u279c adalah path untuk ",(0,i.kt)("strong",{parentName:"li"},"crate")," bernama ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/"},(0,i.kt)("strong",{parentName:"a"},"Rust Standard Library")),", isinya adalah sangat banyak item untuk keperluan umum di Rust programming. Lebih jelasnya akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"#/wip/rust-standard-library"},"Rust standard library"),"."),(0,i.kt)("li",{parentName:"ul"},"Path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::time")," \u279c adalah path untuk ",(0,i.kt)("strong",{parentName:"li"},"module")," bernama ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/time/index.html"},(0,i.kt)("inlineCode",{parentName:"a"},"time")),", isinya banyak item yang berhubungan dengan operasi waktu/time."),(0,i.kt)("li",{parentName:"ul"},"Path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::time::Duration")," \u279c adalah path untuk ",(0,i.kt)("strong",{parentName:"li"},"struct")," bernama ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/time/index.html"},(0,i.kt)("inlineCode",{parentName:"a"},"Duration")),", yang merupakan representasi dari unit waktu.")),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai crate dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/package-crate"},"Module System \u279c Package & Crate")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai module dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/module-basic"},"Module System \u279c Module")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai struct dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/struct"},"Struct")))),(0,i.kt)("h2",{id:"a182-absolute--relative-paths"},"A.18.2. Absolute & relative paths"),(0,i.kt)("p",null,"Rust mengenal dua jenis path:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Absolute path \u279c adalah path yang penulisannya lengkap dari root path, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"li"},"std::time::Duration"),"."),(0,i.kt)("li",{parentName:"ul"},"Relative path \u279c adalah path yang penulisannya relatif terhadap current path, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"li"},"self::my_func"),", ",(0,i.kt)("inlineCode",{parentName:"li"},"super::my_mod::my_consntan"),".")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai relative path dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/module-scope-item-access"},"Module System \u279c Scope & Akses Item"),".")),(0,i.kt)("h2",{id:"a183-rust-items"},"A.18.3. Rust ",(0,i.kt)("em",{parentName:"h2"},"Items")),(0,i.kt)("p",null,"Seperti yang sudah dibahas bahwa path adalah notasi penulisan alamat untuk item. Lalu apa saja yang disebut dengan item? Di Rust ada banyak hal, yang kurang lebih list-nya bisa dilihat berikut:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"modules ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/module-basic"},"Module System \u279c Module"),")")),(0,i.kt)("li",{parentName:"ul"},"extern crate declarations"),(0,i.kt)("li",{parentName:"ul"},"use declarations ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/use"},"Module System \u279c Use"),")")),(0,i.kt)("li",{parentName:"ul"},"function definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/function"},"Function"),", ",(0,i.kt)("a",{parentName:"em",href:"/basic/associated-function"},"Associated Function"),", dan ",(0,i.kt)("a",{parentName:"em",href:"/basic/method"},"Method"),")")),(0,i.kt)("li",{parentName:"ul"},"type definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter-chapter tentang tipe data)")),(0,i.kt)("li",{parentName:"ul"},"struct definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/struct"},"Struct"),")")),(0,i.kt)("li",{parentName:"ul"},"enumeration definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/enum"},"Enum"),")")),(0,i.kt)("li",{parentName:"ul"},"union definitions"),(0,i.kt)("li",{parentName:"ul"},"constant items ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/konstanta"},"Konstanta"),")")),(0,i.kt)("li",{parentName:"ul"},"static items ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/static"},"Static Items"),")")),(0,i.kt)("li",{parentName:"ul"},"trait definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/traits"},"Traits"),")")),(0,i.kt)("li",{parentName:"ul"},"implementations ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/function"},"Function"),", ",(0,i.kt)("a",{parentName:"em",href:"/basic/associated-function"},"Associated Function"),", dan ",(0,i.kt)("a",{parentName:"em",href:"/basic/method"},"Method"),")")),(0,i.kt)("li",{parentName:"ul"},"extern blocks")),(0,i.kt)("h2",{id:"a184-penerapan-paths-dalam-penggaksesan-item"},"A.18.4. Penerapan paths dalam penggaksesan item"),(0,i.kt)("p",null,"Pada bagian ini, kita akan coba terapkan path untuk mengakses beberapa item."),(0,i.kt)("p",null,"Dalam program sederhana berikut, inputan user ditampung sebagai string, kemudian ditampilkan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n // tampilkan intro untuk user agar menginput sebuah pesan\n println!("enter a message:");\n\n // variabel yang akan menampung inputan user dalam string\n let mut message = std::string::String::new();\n\n // objek reader untuk membaca inputan user\n let stdin_reader = std::io::stdin(); \n\n // proses pembacaan inputan user\n let reader_res = stdin_reader.read_line(&mut message);\n\n // pengecekan apakah ada error dalam pembacaan inputan.\n // jika iya, maka tampilkan error dan hentikan program\n if reader_res.is_err() {\n println!("error! {:?}", reader_res.err());\n return;\n }\n\n // tampilkan pesan inputan user\n println!("message: {}", message);\n}\n')),(0,i.kt)("p",null,"Jalankan program, lalu inputkan sebuah pesan, kemudian enter."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Module system path",src:t(7873).Z,width:"452",height:"153"})),(0,i.kt)("p",null,"Bisa dilihat pesan ditampilkan sesuai inputan. Oke, sekarang kita akan bahas program di atas per-barisnya."),(0,i.kt)("h3",{id:"-stdstringstringnew"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"std::string::String::new()")),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"message")," adalah string yang dibuat menggunakan tipe data custom ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),". Salah satu cara pembuatan string bertipe data ini adalah menggunakan statement ",(0,i.kt)("inlineCode",{parentName:"p"},"std::string::String::new()"),"."),(0,i.kt)("p",null,"Bisa dilihat tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"::")," digunakan pada statement tersebut. Path tersebut jika dijabarkan per segment:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Segment ",(0,i.kt)("inlineCode",{parentName:"li"},"std")," adalah crate ",(0,i.kt)("em",{parentName:"li"},"Rust Standard Library")),(0,i.kt)("li",{parentName:"ul"},"Pada path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::string"),", segment ",(0,i.kt)("inlineCode",{parentName:"li"},"string")," adalah module"),(0,i.kt)("li",{parentName:"ul"},"Pada path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::string::String"),", segment ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," adalah struct String, yang biasa disebut dengan ",(0,i.kt)("em",{parentName:"li"},"custom type ",(0,i.kt)("inlineCode",{parentName:"em"},"String"))),(0,i.kt)("li",{parentName:"ul"},"Pada path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::string::String::new"),", segment ",(0,i.kt)("inlineCode",{parentName:"li"},"new")," adalah sebuah fungsi milik struct ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," yang mengembalikan data bertipe custom string ",(0,i.kt)("inlineCode",{parentName:"li"},"String"),"."),(0,i.kt)("li",{parentName:"ul"},"Di segment terakhir, yaitu ",(0,i.kt)("inlineCode",{parentName:"li"},"new"),", ditambahkan tanda pemanggilan fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"()"),", yang menandakan bahwa fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"new")," dipanggil. Dengan itu maka nilai baliknya (yang berupa ",(0,i.kt)("inlineCode",{parentName:"li"},"String"),") ditampung oleh variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"message"),".")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Terkesan banyak sekali penjelasan dalam 1 baris kode, tapi tidak perlu dihafal, lama-kelamaan akan terbiasa.")),(0,i.kt)("p",null,"Salah satu yang menarik dari crate ",(0,i.kt)("em",{parentName:"p"},"Rust Standard Library")," adalah, beberapa segmen otomatis di-import atau dipakai. Jadi tidak perlu menuliskan path secara full."),(0,i.kt)("p",null,"Pada contoh yang sudah dibuat, statement pembuatan data ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," bisa diubah dari ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut message = std::string::String::new();\n")),(0,i.kt)("p",null,"... menjadi cukup ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut message = String::new();\n")),(0,i.kt)("h3",{id:"-stdiostdin"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"std::io::stdin()")),(0,i.kt)("p",null,"Berbeda dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io::stdin")," tidak otomatis ter-import, jadi harus dituliskan secara full meskipun sama-sama dibawah crate ",(0,i.kt)("em",{parentName:"p"},"Rust Standard Library"),"."),(0,i.kt)("p",null,"Path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io")," berisi module untuk keperluan I/O atau input output. Salah satu item yang ada dalam module ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin"),", yang merupakan sebuah fungsi berguna untuk pembuatan objek handler untuk keperluan yang berhubungan dengan console (",(0,i.kt)("em",{parentName:"p"},"stdin"),"). Objek tersebut ditampung oleh variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin_reader"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Secara terminologi, ",(0,i.kt)("em",{parentName:"p"},"stdin")," (yang merupakan kependekan dari ",(0,i.kt)("em",{parentName:"p"},"standard input"),") adalah sebuah input stream dimana data dikirim dan dibaca oleh program.")),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin_reader")," ini kemudian kita gunakan untuk berinteraksi dengan input stream, untuk menangkap inputan user."),(0,i.kt)("h3",{id:"-stdin_readerread_linemut-message"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"stdin_reader.read_line(&mut message)")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"read_line")," milik variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin_reader")," berguna untuk menangkap inputan user. Variabel yang disisipkan sebagai argumen pemanggilan method tersebut menjadi penampung inputan user, yang pada contoh ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"message"),"."),(0,i.kt)("p",null,"Perlu diperhatikan bahwa ",(0,i.kt)("inlineCode",{parentName:"p"},"message")," disisipkan ",(0,i.kt)("em",{parentName:"p"},"mutable reference"),"-nya sebagai argumen pemanggilan fungsi."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai pointer dan ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"))),(0,i.kt)("p",null,"Eksekusi dari statement ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin_reader.read_line(&mut message)")," adalah blocking, artinya program akan berhenti untuk sementara di baris tersebut, hingga ada inputan dari user dan tombol enter ditekan."),(0,i.kt)("h3",{id:"-pengecekan-error-stdin_reader"},"\u25c9 Pengecekan error ",(0,i.kt)("inlineCode",{parentName:"h3"},"stdin_reader")),(0,i.kt)("p",null,"Blok statement ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," pada contoh di atas bertugas melakukan pengecekan error. Jika ada error, maka ",(0,i.kt)("inlineCode",{parentName:"p"},"reader_res.is_err()")," bernilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),", dan pesan error-nya dimunculkan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'if reader_res.is_err() {\n println!("error! {:?}", reader_res.err());\n return;\n}\n')),(0,i.kt)("h3",{id:"-menampilkan-isi-message"},"\u25c9 Menampilkan isi ",(0,i.kt)("inlineCode",{parentName:"h3"},"message")),(0,i.kt)("p",null,"Jika program berlajan sesuai harapan, tanpa error, pada baris terakhir data dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"message")," ditampilkan ke layar."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ini tidak perlu di-konversi ke bentuk literal string ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," untuk ditampilkan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Langsung saja sisipkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ke macro tersebut dan ",(0,i.kt)("inlineCode",{parentName:"p"},"println")," akan tau harus menampilkan apa.")),(0,i.kt)("p",null,"Oke, Penulis rasa sudah cukup jelas perihal bagaimana cara menggunakan path untuk mengakses item. Cukup tulis saja path-nya. Jika path-nya panjang? ya ditulis semua."),(0,i.kt)("h2",{id:"a185-penggunaan-keyword-use-untuk-import-path"},"A.18.5. Penggunaan keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"use")," untuk import path"),(0,i.kt)("p",null,"Ada alternatif cara lain untuk memperpendek penulisan dan pengaksesan path, yaitu dengan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Penggunaan ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," juga sempat dipraktikkan pada chapter sebelumnya, yaitu ",(0,i.kt)("a",{parentName:"p",href:"/basic/perulangan-while"},"Perulangan \u279c while"),".")),(0,i.kt)("p",null,"Cara penerapannya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let stdin_reader = std::io::stdin();\n\n// ... atau ...\n\nuse std::io;\nlet stdin_reader = io::stdin();\n\n// ... atau ...\n\nuse std::io::stdin;\nlet stdin_reader = stdin();\n\n// ... atau ...\n\nuse std::io::{stdin};\nlet stdin_reader = stdin();\n")),(0,i.kt)("p",null,"Dengan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," kita bisa memperpendek pengaksesan sebuah path."),(0,i.kt)("p",null,"O iya keyword ini bisa digunakan dimana saja, artinya tidak harus di luar fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),". Bisa saja di dalam fungsi, atau di dalam blok kode seleksi kondisi atau lainnya."),(0,i.kt)("h3",{id:"-import-beberapa-items-yang-parent-path-nya-yang-sama"},"\u25c9 Import beberapa items yang parent path-nya yang sama"),(0,i.kt)("p",null,"Pada contoh di atas, path yang diimport adalah sebuah item ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io::stdin"),". Dimisalkan ada path lain (sebagai contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io::stderr"),") yang juga ikut diimport, maka penulisannya kurang lebih seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::io::stdin;\nuse std::io::stderr;\n\n// ... atau ...\n\nuse std::io::{stdin, stderr};\n")),(0,i.kt)("h3",{id:"-import-semua-items-dalam-suatu-path"},"\u25c9 Import semua items dalam suatu path"),(0,i.kt)("p",null,"Suatu path bisa saja memiliki cukup banyak item/child dibawahnya. Sebagai contoh, path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io")," merupakan parent path dari ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"stderr"),". Selain dua items tersebut, ada juga item lainnya."),(0,i.kt)("p",null,"Ada shortcut yang membuat penulisan import path lebih praktis, tidak perlu menuliskan satu-per-satu, caranya adalah menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"*"),". Sebagai contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::io::*;\n\n// ... adalah ekuivalen dengan ...\n\nuse std::io::{stdin, stderr, stdout, };\n\n// ... atau ...\n\nuse std::io::stdin;\nuse std::io::stderr;\nuse std::io::stdout;\nuse std::io::;\n")),(0,i.kt)("h2",{id:"a186-pembahasan-lanjutan"},"A.18.6. Pembahasan lanjutan"),(0,i.kt)("p",null,"Kita sudah beberapa kali menerapkan path untuk mengakses item milik crate ",(0,i.kt)("em",{parentName:"p"},"Rust Standard Library"),", lalu bagaimana penerapan path untuk internal item, pastinya pada proyek real dalam 1 program akan ada banyak item."),(0,i.kt)("p",null,"Jawabannya akan ada di beberapa chapter berikutnya. Untuk sekarang khusus pada bagian ",(0,i.kt)("strong",{parentName:"p"},"module system")," ini, penulis anjurkan untuk mengikuti urutan pembelajaran ebook terlebih dahulu."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/path_item"},"github.com/novalagung/dasarpemrogramanrust-example/../path_item")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-03-paths-for-referring-to-an-item-in-the-module-tree.html"},"https://doc.rust-lang.org/book/ch07-03-paths-for-referring-to-an-item-in-the-module-tree.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/paths.html"},"https://doc.rust-lang.org/reference/paths.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/items.html"},"https://doc.rust-lang.org/reference/items.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.use.html"},"https://doc.rust-lang.org/std/keyword.use.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/edition-guide/rust-2018/path-changes.html"},"https://doc.rust-lang.org/edition-guide/rust-2018/path-changes.html"))))}d.isMDXComponent=!0},7873:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/e7bd9b37.6bffc9e2.js b/assets/js/e7bd9b37.6bffc9e2.js new file mode 100644 index 00000000..acf68c01 --- /dev/null +++ b/assets/js/e7bd9b37.6bffc9e2.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7539],{3905:(a,e,t)=>{t.d(e,{Zo:()=>u,kt:()=>o});var n=t(7294);function i(a,e,t){return e in a?Object.defineProperty(a,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):a[e]=t,a}function r(a,e){var t=Object.keys(a);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(a);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),t.push.apply(t,n)}return t}function p(a){for(var e=1;e=0||(i[t]=a[t]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(a,t)&&(i[t]=a[t])}return i}var s=n.createContext({}),m=function(a){var e=n.useContext(s),t=e;return a&&(t="function"==typeof a?a(e):p(p({},e),a)),t},u=function(a){var e=m(a.components);return n.createElement(s.Provider,{value:e},a.children)},d={inlineCode:"code",wrapper:function(a){var e=a.children;return n.createElement(n.Fragment,{},e)}},k=n.forwardRef((function(a,e){var t=a.components,i=a.mdxType,r=a.originalType,s=a.parentName,u=l(a,["components","mdxType","originalType","parentName"]),k=m(t),o=i,h=k["".concat(s,".").concat(o)]||k[o]||d[o]||r;return t?n.createElement(h,p(p({ref:e},u),{},{components:t})):n.createElement(h,p({ref:e},u))}));function o(a,e){var t=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=t.length,p=new Array(r);p[0]=k;var l={};for(var s in e)hasOwnProperty.call(e,s)&&(l[s]=e[s]);l.originalType=a,l.mdxType="string"==typeof a?a:i,p[1]=l;for(var m=2;m{t.r(e),t.d(e,{assets:()=>s,contentTitle:()=>p,default:()=>d,frontMatter:()=>r,metadata:()=>l,toc:()=>m});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:18,title:"A.18. Module System \u279c Path & Item",sidebar_label:"A.18. Module System \u279c Path & Item"},p=void 0,l={unversionedId:"basic/path-item",id:"basic/path-item",title:"A.18. Module System \u279c Path & Item",description:"Chapter ini membahas tentang konsep path dan item dalam Rust programming.",source:"@site/docs/basic/path-item.md",sourceDirName:"basic",slug:"/basic/path-item",permalink:"/basic/path-item",draft:!1,tags:[],version:"current",sidebarPosition:18,frontMatter:{sidebar_position:18,title:"A.18. Module System \u279c Path & Item",sidebar_label:"A.18. Module System \u279c Path & Item"},sidebar:"tutorialSidebar",previous:{title:"A.17. Function",permalink:"/basic/function"},next:{title:"A.19. Module System \u279c Package & Crate",permalink:"/basic/package-crate"}},s={},m=[{value:"A.18.1. Rust Paths",id:"a181-rust-paths",level:2},{value:"A.18.2. Absolute & relative paths",id:"a182-absolute--relative-paths",level:2},{value:"A.18.3. Rust Items",id:"a183-rust-items",level:2},{value:"A.18.4. Penerapan paths dalam penggaksesan item",id:"a184-penerapan-paths-dalam-penggaksesan-item",level:2},{value:"\u25c9 std::string::String::new()",id:"-stdstringstringnew",level:3},{value:"\u25c9 std::io::stdin()",id:"-stdiostdin",level:3},{value:"\u25c9 stdin_reader.read_line(&mut message)",id:"-stdin_readerread_linemut-message",level:3},{value:"\u25c9 Pengecekan error stdin_reader",id:"-pengecekan-error-stdin_reader",level:3},{value:"\u25c9 Menampilkan isi message",id:"-menampilkan-isi-message",level:3},{value:"A.18.5. Penggunaan keyword use untuk import path",id:"a185-penggunaan-keyword-use-untuk-import-path",level:2},{value:"\u25c9 Import beberapa items yang parent path-nya yang sama",id:"-import-beberapa-items-yang-parent-path-nya-yang-sama",level:3},{value:"\u25c9 Import semua items dalam suatu path",id:"-import-semua-items-dalam-suatu-path",level:3},{value:"A.18.6. Pembahasan lanjutan",id:"a186-pembahasan-lanjutan",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],u={toc:m};function d(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,n.Z)({},u,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Chapter ini membahas tentang konsep ",(0,i.kt)("em",{parentName:"p"},"path")," dan ",(0,i.kt)("em",{parentName:"p"},"item")," dalam Rust programming."),(0,i.kt)("h2",{id:"a181-rust-paths"},"A.18.1. Rust ",(0,i.kt)("em",{parentName:"h2"},"Paths")),(0,i.kt)("p",null,"Paths (atau Path) adalah notasi penulisan alamat sebuah item, contohnya ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),"."),(0,i.kt)("p",null,(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration")," adalah ",(0,i.kt)("em",{parentName:"p"},"path")," untuk item yang isinya adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"struct")," bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration"),". Item bisa berupa banyak jenis, bisa saja struct, atau ",(0,i.kt)("em",{parentName:"p"},"macro"),", konstanta, atau lainnya. Lebih jelasnya silakan cek pada halaman dokumentasi ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),"."),(0,i.kt)("p",null,"Sebuah path bisa memiliki banyak bagian (biasa disebut ",(0,i.kt)("em",{parentName:"p"},"segment"),"), sebagai contoh, path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration")," memiliki 3 segmen yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"std"),", ",(0,i.kt)("inlineCode",{parentName:"p"},"time"),", dan ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration"),". Karakter ",(0,i.kt)("inlineCode",{parentName:"p"},"::")," digunakan dalam penulisan path sebagai pembatas antar segmen (jika path memiliki lebih dari 1 segmen)."),(0,i.kt)("p",null,"Dalam sebuah path, yang disebut dengan item adalah segment terakhir. Contohnya path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),", maka item yang dituju adalah struct ",(0,i.kt)("inlineCode",{parentName:"p"},"Duration"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Rust paths mirip seperti konsep ",(0,i.kt)("em",{parentName:"p"},"filesystem")," path di sistem operasi. Seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"C:\\Users\\novalagung\\Desktop")," di windows, atau ",(0,i.kt)("inlineCode",{parentName:"p"},"/etc/nginx/conf.d/nginx.conf")," di Unix/Linux."),(0,i.kt)("p",{parentName:"blockquote"},"Di Rust, path tidak menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"\\")," atau ",(0,i.kt)("inlineCode",{parentName:"p"},"/")," sebagai separator, melainkan ",(0,i.kt)("inlineCode",{parentName:"p"},"::"),".")),(0,i.kt)("p",null,"Jika di-breakdown, berikut adalah penjelasan dari setiap kombinasi segmen path pada contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"std::time::Duration"),"."),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Path ",(0,i.kt)("inlineCode",{parentName:"li"},"std")," \u279c adalah path untuk ",(0,i.kt)("strong",{parentName:"li"},"crate")," bernama ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/"},(0,i.kt)("strong",{parentName:"a"},"Rust Standard Library")),", isinya adalah sangat banyak item untuk keperluan umum di Rust programming. Lebih jelasnya akan dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"#/wip/rust-standard-library"},"Rust standard library"),"."),(0,i.kt)("li",{parentName:"ul"},"Path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::time")," \u279c adalah path untuk ",(0,i.kt)("strong",{parentName:"li"},"module")," bernama ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/time/index.html"},(0,i.kt)("inlineCode",{parentName:"a"},"time")),", isinya banyak item yang berhubungan dengan operasi waktu/time."),(0,i.kt)("li",{parentName:"ul"},"Path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::time::Duration")," \u279c adalah path untuk ",(0,i.kt)("strong",{parentName:"li"},"struct")," bernama ",(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/time/index.html"},(0,i.kt)("inlineCode",{parentName:"a"},"Duration")),", yang merupakan representasi dari unit waktu.")),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai crate dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/package-crate"},"Module System \u279c Package & Crate")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai module dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/module-basic"},"Module System \u279c Module")),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai struct dibahas pada chapter ",(0,i.kt)("a",{parentName:"li",href:"/basic/struct"},"Struct")))),(0,i.kt)("h2",{id:"a182-absolute--relative-paths"},"A.18.2. Absolute & relative paths"),(0,i.kt)("p",null,"Rust mengenal dua jenis path:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Absolute path \u279c adalah path yang penulisannya lengkap dari root path, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"li"},"std::time::Duration"),"."),(0,i.kt)("li",{parentName:"ul"},"Relative path \u279c adalah path yang penulisannya relatif terhadap current path, contohnya seperti ",(0,i.kt)("inlineCode",{parentName:"li"},"self::my_func"),", ",(0,i.kt)("inlineCode",{parentName:"li"},"super::my_mod::my_consntan"),".")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai relative path dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/module-scope-item-access"},"Module System \u279c Scope & Akses Item"),".")),(0,i.kt)("h2",{id:"a183-rust-items"},"A.18.3. Rust ",(0,i.kt)("em",{parentName:"h2"},"Items")),(0,i.kt)("p",null,"Seperti yang sudah dibahas bahwa path adalah notasi penulisan alamat untuk item. Lalu apa saja yang disebut dengan item? Di Rust ada banyak hal, yang kurang lebih list-nya bisa dilihat berikut:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"modules ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/module-basic"},"Module System \u279c Module"),")")),(0,i.kt)("li",{parentName:"ul"},"extern crate declarations"),(0,i.kt)("li",{parentName:"ul"},"use declarations ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/use"},"Module System \u279c Use"),")")),(0,i.kt)("li",{parentName:"ul"},"function definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/function"},"Function"),", ",(0,i.kt)("a",{parentName:"em",href:"/basic/associated-function"},"Associated Function"),", dan ",(0,i.kt)("a",{parentName:"em",href:"/basic/method"},"Method"),")")),(0,i.kt)("li",{parentName:"ul"},"type definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter-chapter tentang tipe data)")),(0,i.kt)("li",{parentName:"ul"},"struct definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/struct"},"Struct"),")")),(0,i.kt)("li",{parentName:"ul"},"enumeration definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/enum"},"Enum"),")")),(0,i.kt)("li",{parentName:"ul"},"union definitions"),(0,i.kt)("li",{parentName:"ul"},"constant items ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/konstanta"},"Konstanta"),")")),(0,i.kt)("li",{parentName:"ul"},"static items ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/static"},"Static Items"),")")),(0,i.kt)("li",{parentName:"ul"},"trait definitions ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/traits"},"Traits"),")")),(0,i.kt)("li",{parentName:"ul"},"implementations ",(0,i.kt)("em",{parentName:"li"},"(dibahas pada chapter ",(0,i.kt)("a",{parentName:"em",href:"/basic/function"},"Function"),", ",(0,i.kt)("a",{parentName:"em",href:"/basic/associated-function"},"Associated Function"),", dan ",(0,i.kt)("a",{parentName:"em",href:"/basic/method"},"Method"),")")),(0,i.kt)("li",{parentName:"ul"},"extern blocks")),(0,i.kt)("h2",{id:"a184-penerapan-paths-dalam-penggaksesan-item"},"A.18.4. Penerapan paths dalam penggaksesan item"),(0,i.kt)("p",null,"Pada bagian ini, kita akan coba terapkan path untuk mengakses beberapa item."),(0,i.kt)("p",null,"Dalam program sederhana berikut, inputan user ditampung sebagai string, kemudian ditampilkan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n // tampilkan intro untuk user agar menginput sebuah pesan\n println!("enter a message:");\n\n // variabel yang akan menampung inputan user dalam string\n let mut message = std::string::String::new();\n\n // objek reader untuk membaca inputan user\n let stdin_reader = std::io::stdin(); \n\n // proses pembacaan inputan user\n let reader_res = stdin_reader.read_line(&mut message);\n\n // pengecekan apakah ada error dalam pembacaan inputan.\n // jika iya, maka tampilkan error dan hentikan program\n if reader_res.is_err() {\n println!("error! {:?}", reader_res.err());\n return;\n }\n\n // tampilkan pesan inputan user\n println!("message: {}", message);\n}\n')),(0,i.kt)("p",null,"Jalankan program, lalu inputkan sebuah pesan, kemudian enter."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Module system path",src:t(7873).Z,width:"452",height:"153"})),(0,i.kt)("p",null,"Bisa dilihat pesan ditampilkan sesuai inputan. Oke, sekarang kita akan bahas program di atas per-barisnya."),(0,i.kt)("h3",{id:"-stdstringstringnew"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"std::string::String::new()")),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"message")," adalah string yang dibuat menggunakan tipe data custom ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),". Salah satu cara pembuatan string bertipe data ini adalah menggunakan statement ",(0,i.kt)("inlineCode",{parentName:"p"},"std::string::String::new()"),"."),(0,i.kt)("p",null,"Bisa dilihat tanda ",(0,i.kt)("inlineCode",{parentName:"p"},"::")," digunakan pada statement tersebut. Path tersebut jika dijabarkan per segment:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Segment ",(0,i.kt)("inlineCode",{parentName:"li"},"std")," adalah crate ",(0,i.kt)("em",{parentName:"li"},"Rust Standard Library")),(0,i.kt)("li",{parentName:"ul"},"Pada path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::string"),", segment ",(0,i.kt)("inlineCode",{parentName:"li"},"string")," adalah module"),(0,i.kt)("li",{parentName:"ul"},"Pada path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::string::String"),", segment ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," adalah struct String, yang biasa disebut dengan ",(0,i.kt)("em",{parentName:"li"},"custom type ",(0,i.kt)("inlineCode",{parentName:"em"},"String"))),(0,i.kt)("li",{parentName:"ul"},"Pada path ",(0,i.kt)("inlineCode",{parentName:"li"},"std::string::String::new"),", segment ",(0,i.kt)("inlineCode",{parentName:"li"},"new")," adalah sebuah fungsi milik struct ",(0,i.kt)("inlineCode",{parentName:"li"},"String")," yang mengembalikan data bertipe custom string ",(0,i.kt)("inlineCode",{parentName:"li"},"String"),"."),(0,i.kt)("li",{parentName:"ul"},"Di segment terakhir, yaitu ",(0,i.kt)("inlineCode",{parentName:"li"},"new"),", ditambahkan tanda pemanggilan fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"()"),", yang menandakan bahwa fungsi ",(0,i.kt)("inlineCode",{parentName:"li"},"new")," dipanggil. Dengan itu maka nilai baliknya (yang berupa ",(0,i.kt)("inlineCode",{parentName:"li"},"String"),") ditampung oleh variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"message"),".")),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Terkesan banyak sekali penjelasan dalam 1 baris kode, tapi tidak perlu dihafal, lama-kelamaan akan terbiasa.")),(0,i.kt)("p",null,"Salah satu yang menarik dari crate ",(0,i.kt)("em",{parentName:"p"},"Rust Standard Library")," adalah, beberapa segmen otomatis di-import atau dipakai. Jadi tidak perlu menuliskan path secara full."),(0,i.kt)("p",null,"Pada contoh yang sudah dibuat, statement pembuatan data ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," bisa diubah dari ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut message = std::string::String::new();\n")),(0,i.kt)("p",null,"... menjadi cukup ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let mut message = String::new();\n")),(0,i.kt)("h3",{id:"-stdiostdin"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"std::io::stdin()")),(0,i.kt)("p",null,"Berbeda dengan ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io::stdin")," tidak otomatis ter-import, jadi harus dituliskan secara full meskipun sama-sama di bawah crate ",(0,i.kt)("em",{parentName:"p"},"Rust Standard Library"),"."),(0,i.kt)("p",null,"Path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io")," berisi module untuk keperluan I/O atau input output. Salah satu item yang ada dalam module ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin"),", yang merupakan sebuah fungsi berguna untuk pembuatan objek handler untuk keperluan yang berhubungan dengan console (",(0,i.kt)("em",{parentName:"p"},"stdin"),"). Objek tersebut ditampung oleh variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin_reader"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Secara terminologi, ",(0,i.kt)("em",{parentName:"p"},"stdin")," (merupakan kependekan dari ",(0,i.kt)("em",{parentName:"p"},"standard input"),") adalah sebuah input stream yang nilainya dibaca oleh program sebagai inputan.")),(0,i.kt)("p",null,"Variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin_reader")," ini kemudian kita gunakan untuk berinteraksi dengan input stream, untuk menangkap inputan user."),(0,i.kt)("h3",{id:"-stdin_readerread_linemut-message"},"\u25c9 ",(0,i.kt)("inlineCode",{parentName:"h3"},"stdin_reader.read_line(&mut message)")),(0,i.kt)("p",null,"Method ",(0,i.kt)("inlineCode",{parentName:"p"},"read_line")," milik variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin_reader")," berguna untuk menangkap inputan user. Variabel yang disisipkan sebagai argumen pemanggilan method tersebut menjadi penampung inputan user, yang pada contoh ini adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"message"),"."),(0,i.kt)("p",null,"Perlu diperhatikan bahwa ",(0,i.kt)("inlineCode",{parentName:"p"},"message")," disisipkan ",(0,i.kt)("em",{parentName:"p"},"mutable reference"),"-nya sebagai argumen pemanggilan fungsi."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai pointer dan ",(0,i.kt)("em",{parentName:"p"},"mutable reference")," dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"))),(0,i.kt)("p",null,"Eksekusi dari statement ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin_reader.read_line(&mut message)")," adalah blocking, artinya program akan berhenti untuk sementara di baris tersebut, hingga ada inputan dari user dan tombol enter ditekan."),(0,i.kt)("h3",{id:"-pengecekan-error-stdin_reader"},"\u25c9 Pengecekan error ",(0,i.kt)("inlineCode",{parentName:"h3"},"stdin_reader")),(0,i.kt)("p",null,"Blok statement ",(0,i.kt)("inlineCode",{parentName:"p"},"if")," pada contoh di atas bertugas melakukan pengecekan error. Jika ada error, maka ",(0,i.kt)("inlineCode",{parentName:"p"},"reader_res.is_err()")," bernilai ",(0,i.kt)("inlineCode",{parentName:"p"},"true"),", dan pesan error-nya dimunculkan."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'if reader_res.is_err() {\n println!("error! {:?}", reader_res.err());\n return;\n}\n')),(0,i.kt)("h3",{id:"-menampilkan-isi-message"},"\u25c9 Menampilkan isi ",(0,i.kt)("inlineCode",{parentName:"h3"},"message")),(0,i.kt)("p",null,"Jika program berlajan sesuai harapan, tanpa error, pada baris terakhir data dalam ",(0,i.kt)("inlineCode",{parentName:"p"},"message")," ditampilkan ke layar."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ini tidak perlu di-konversi ke bentuk literal string ",(0,i.kt)("inlineCode",{parentName:"p"},"&str")," untuk ditampilkan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"println"),". Langsung saja sisipkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"String")," ke macro tersebut dan ",(0,i.kt)("inlineCode",{parentName:"p"},"println")," akan tau harus menampilkan apa.")),(0,i.kt)("p",null,"Oke, Penulis rasa sudah cukup jelas perihal bagaimana cara menggunakan path untuk mengakses item. Cukup tulis saja path-nya. Jika path-nya panjang? ya ditulis semua."),(0,i.kt)("h2",{id:"a185-penggunaan-keyword-use-untuk-import-path"},"A.18.5. Penggunaan keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"use")," untuk import path"),(0,i.kt)("p",null,"Ada alternatif cara lain untuk memperpendek penulisan dan pengaksesan path, yaitu dengan menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Penggunaan ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," juga sempat dipraktikkan pada chapter sebelumnya, yaitu ",(0,i.kt)("a",{parentName:"p",href:"/basic/perulangan-while"},"Perulangan \u279c while"),".")),(0,i.kt)("p",null,"Cara penerapannya bisa dilihat pada kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"let stdin_reader = std::io::stdin();\n\n// ... atau ...\n\nuse std::io;\nlet stdin_reader = io::stdin();\n\n// ... atau ...\n\nuse std::io::stdin;\nlet stdin_reader = stdin();\n\n// ... atau ...\n\nuse std::io::{stdin};\nlet stdin_reader = stdin();\n")),(0,i.kt)("p",null,"Dengan menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," kita bisa memperpendek pengaksesan sebuah path."),(0,i.kt)("p",null,"O iya keyword ini bisa digunakan di mana saja, artinya tidak harus di luar fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main"),". Bisa saja di dalam fungsi, atau di dalam blok kode seleksi kondisi atau lainnya."),(0,i.kt)("h3",{id:"-import-beberapa-items-yang-parent-path-nya-yang-sama"},"\u25c9 Import beberapa items yang parent path-nya yang sama"),(0,i.kt)("p",null,"Pada contoh di atas, path yang diimport adalah sebuah item ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io::stdin"),". Dimisalkan ada path lain (sebagai contoh ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io::stderr"),") yang juga ikut diimport, maka penulisannya kurang lebih seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::io::stdin;\nuse std::io::stderr;\n\n// ... atau ...\n\nuse std::io::{stdin, stderr};\n")),(0,i.kt)("h3",{id:"-import-semua-items-dalam-suatu-path"},"\u25c9 Import semua items dalam suatu path"),(0,i.kt)("p",null,"Suatu path bisa saja memiliki cukup banyak item/child di bawahnya. Sebagai contoh, path ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io")," merupakan parent path dari ",(0,i.kt)("inlineCode",{parentName:"p"},"stdin")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"stderr"),". Selain dua items tersebut, ada juga item lainnya."),(0,i.kt)("p",null,"Ada shortcut yang membuat penulisan import path lebih praktis, tidak perlu menuliskan satu-per-satu, caranya adalah menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"*"),". Sebagai contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::io::*;\n\n// ... adalah ekuivalen dengan ...\n\nuse std::io::{stdin, stderr, stdout, };\n\n// ... atau ...\n\nuse std::io::stdin;\nuse std::io::stderr;\nuse std::io::stdout;\nuse std::io::;\n")),(0,i.kt)("h2",{id:"a186-pembahasan-lanjutan"},"A.18.6. Pembahasan lanjutan"),(0,i.kt)("p",null,"Kita sudah beberapa kali menerapkan path untuk mengakses item milik crate ",(0,i.kt)("em",{parentName:"p"},"Rust Standard Library"),", lalu bagaimana penerapan path untuk internal item, pastinya pada proyek real dalam 1 program akan ada banyak item."),(0,i.kt)("p",null,"Jawabannya akan ada di beberapa chapter berikutnya. Untuk sekarang khusus pada bagian ",(0,i.kt)("strong",{parentName:"p"},"module system")," ini, penulis anjurkan untuk mengikuti urutan pembelajaran ebook terlebih dahulu."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/path_item"},"github.com/novalagung/dasarpemrogramanrust-example/../path_item")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch07-03-paths-for-referring-to-an-item-in-the-module-tree.html"},"https://doc.rust-lang.org/book/ch07-03-paths-for-referring-to-an-item-in-the-module-tree.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/paths.html"},"https://doc.rust-lang.org/reference/paths.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/items.html"},"https://doc.rust-lang.org/reference/items.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.use.html"},"https://doc.rust-lang.org/std/keyword.use.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/edition-guide/rust-2018/path-changes.html"},"https://doc.rust-lang.org/edition-guide/rust-2018/path-changes.html"))))}d.isMDXComponent=!0},7873:(a,e,t)=>{t.d(e,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/eb2a6637.01009699.js b/assets/js/eb2a6637.01009699.js deleted file mode 100644 index 2fe64b0c..00000000 --- a/assets/js/eb2a6637.01009699.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6388],{3905:(e,a,t)=>{t.d(a,{Zo:()=>s,kt:()=>A});var n=t(7294);function i(e,a,t){return a in e?Object.defineProperty(e,a,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[a]=t,e}function r(e,a){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);a&&(n=n.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),t.push.apply(t,n)}return t}function l(e){for(var a=1;a=0||(i[t]=e[t]);return i}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(i[t]=e[t])}return i}var u=n.createContext({}),m=function(e){var a=n.useContext(u),t=a;return e&&(t="function"==typeof e?e(a):l(l({},a),e)),t},s=function(e){var a=m(e.components);return n.createElement(u.Provider,{value:a},e.children)},d={inlineCode:"code",wrapper:function(e){var a=e.children;return n.createElement(n.Fragment,{},a)}},k=n.forwardRef((function(e,a){var t=e.components,i=e.mdxType,r=e.originalType,u=e.parentName,s=p(e,["components","mdxType","originalType","parentName"]),k=m(t),A=i,o=k["".concat(u,".").concat(A)]||k[A]||d[A]||r;return t?n.createElement(o,l(l({ref:a},s),{},{components:t})):n.createElement(o,l({ref:a},s))}));function A(e,a){var t=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var r=t.length,l=new Array(r);l[0]=k;var p={};for(var u in a)hasOwnProperty.call(a,u)&&(p[u]=a[u]);p.originalType=e,p.mdxType="string"==typeof e?e:i,l[1]=p;for(var m=2;m{t.r(a),t.d(a,{assets:()=>u,contentTitle:()=>l,default:()=>d,frontMatter:()=>r,metadata:()=>p,toc:()=>m});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:29,title:"A.29. Module System \u279c Use, Import, Re-export",sidebar_label:"A.29. Module System \u279c Use, Import, Re-export"},l=void 0,p={unversionedId:"basic/use",id:"basic/use",title:"A.29. Module System \u279c Use, Import, Re-export",description:"Keyword use digunakan untuk dua hal, yaitu import path dan re-export path. Sebenarnya kita telah mempelajari kedua penerapan tersebut pada beberapa chapter sebelumnya. Pada chapter ini kita akan ulang lagi pembahasan agar lebih jelas.",source:"@site/docs/basic/use.md",sourceDirName:"basic",slug:"/basic/use",permalink:"/basic/use",draft:!1,tags:[],version:"current",sidebarPosition:29,frontMatter:{sidebar_position:29,title:"A.29. Module System \u279c Use, Import, Re-export",sidebar_label:"A.29. Module System \u279c Use, Import, Re-export"},sidebar:"tutorialSidebar",previous:{title:"A.28. Module System \u279c Visibility & Privacy",permalink:"/basic/visibility-privacy"},next:{title:"A.30. Block Expression",permalink:"/basic/block-expression"}},u={},m=[{value:"A.29.1. Keyword use untuk import path",id:"a291-keyword-use-untuk-import-path",level:2},{value:"\u25c9 Import items dari module yang sama",id:"-import-items-dari-module-yang-sama",level:3},{value:"A.29.2. Keyword use untuk re-export path",id:"a292-keyword-use-untuk-re-export-path",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],s={toc:m};function d(e){let{components:a,...r}=e;return(0,i.kt)("wrapper",(0,n.Z)({},s,r,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," digunakan untuk dua hal, yaitu ",(0,i.kt)("em",{parentName:"p"},"import")," path dan ",(0,i.kt)("em",{parentName:"p"},"re-export")," path. Sebenarnya kita telah mempelajari kedua penerapan tersebut pada beberapa chapter sebelumnya. Pada chapter ini kita akan ulang lagi pembahasan agar lebih jelas."),(0,i.kt)("h2",{id:"a291-keyword-use-untuk-import-path"},"A.29.1. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"use")," untuk import path"),(0,i.kt)("p",null,"Untuk bisa menggunakan sebuah item dari crate lain, entah itu dari rust standard library crate maupun 3rd-party, caranya cukup dengan menuliskan path item. Contohnya bisa dilihat dibawah ini, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"current_dir")," digunakan untuk mengambil path dari current directory. Fungsi tersebut merupakan item dari module ",(0,i.kt)("inlineCode",{parentName:"p"},"std::env"),", maka untuk mengaksesnya kita harus menuliskan path secara lengkap."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let package_path = std::env::current_dir().unwrap();\nprintln!("package_path: {:?}", package_path);\n')),(0,i.kt)("p",null,"Selain cara di atas, bisa juga gunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," untuk meng-import fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"current_dir")," ke kode program, dengannya kita tidak perlu menulis path secara lengkap. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use std::env::current_dir;\n\nfn main() {\n let package_path = current_dir().unwrap();\n println!("package_path: {:?}", package_path);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Rust use",src:t(9772).Z,width:"818",height:"104"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Di contoh, pengaksesan current directory di-chain dengan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," karena nilai balik fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"current_dir")," adalah bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io::Result"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai tipe tersebut dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,i.kt)("h3",{id:"-import-items-dari-module-yang-sama"},"\u25c9 Import items dari module yang sama"),(0,i.kt)("p",null,"Pada contoh selanjutnya ini, kita akan coba praktikkan penerapan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"std::env::args")," untuk mengambil ",(0,i.kt)("em",{parentName:"p"},"argument")," saat eksekusi program, kemudian menampilkannya ke layar."),(0,i.kt)("p",null,"Ok, berarti akan ada 2 path yang akan di-import:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::env::current_dir;\nuse std::env::args;\n")),(0,i.kt)("p",null,"Ada notasi penulisan import path lainnya yang bisa digunakan jika path-nya memiliki parent segment yang sama. Contohnya seperti di atas, kedua fungsi tersebut memiliki parent segment path yang sama yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"std::env"),". Yang seperti ini bisa dituliskan juga dengan notasi penulisan import berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::env::{current_dir, args};\n")),(0,i.kt)("p",null,"Ok, lanjut ke praktik. Silakan tulis kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use std::env::{current_dir, args};\n\nfn main() {\n let package_path = current_dir().unwrap();\n println!("package_path: {:?}", package_path);\n\n for i in 1..=args().len() {\n let each_arg = args().nth(i);\n if each_arg != None {\n println!("arg{}: {:?}", i, each_arg.unwrap());\n }\n }\n}\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"std::env::args")," digunakan untuk mengambil argument eksekusi program baik eksekusi via ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run")," ataupun via pemanggilan binary. Fungsi tersebut nilai baliknya adalah iterator, jadi bisa digunakan dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," dengan mudah."),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"args().nth(i)")," mengembalikan nilai argument pada index ke-",(0,i.kt)("inlineCode",{parentName:"p"},"i")," dalam tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),", dan nilai tersebut bisa saja tidak ada (direpresentasikan dengan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),"). Oleh karena itu penting untuk dicek terlebih dahulu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if"),". Jika memang nilainya adalah selain ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," untuk mengambil nilainya dalam tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai Iterator akan dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"li",href:"/basic/trait-iterator"},"Trait \u279c Iterator"),"."),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"Option")," akan dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"li",href:"/basic/option-type"},"Tipe Data \u279c Option"),"."))),(0,i.kt)("p",null,"Silakan coba jalankan menggunakan dua command, yang pertama ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run"),", kemudian ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run tulis argumen di sini dengan pembatas spasi"),"."),(0,i.kt)("p",null,"Bisa dilihat pada gambar berikut, jika ada argument disisipkan dalam eksekusi program, maka ditampilkan. Pada gambar berikut dicontohkan argument yang dipakai adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"hello")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"world"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Rust use",src:t(4887).Z,width:"793",height:"223"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai Iterator dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/trait-iterator"},"Trait \u279c Iterator"))),(0,i.kt)("h2",{id:"a292-keyword-use-untuk-re-export-path"},"A.29.2. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"use")," untuk re-export path"),(0,i.kt)("p",null,"Re-export item adalah sebuah cara untuk mem-",(0,i.kt)("em",{parentName:"p"},"bypass")," pengaksesan item yang secara hirarki memang tidak bisa diakses dari luar module (bisa jadi karena visibility item ataupun parent module nya adalah private). Dengan teknik ini, maka item pasti bisa diakses dari luar module."),(0,i.kt)("p",null,"Item yang di-re-export akan menjadi item milik ",(0,i.kt)("em",{parentName:"p"},"current module")," dimana statement re-export tersebut ditulis."),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub use")," digunakan untuk operasi re-export. Notasi penulisannya bisa dilihat di bawah ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// pub use the_path\npub use self::sub_module::say_hello_message;\n\n// pub use the_path as exported_name;\npub use self::sub_module::say_hello_message as say_hello;\n")),(0,i.kt)("p",null,"Contoh penerapan bisa dilihat pada kode berikut. Submodule ",(0,i.kt)("inlineCode",{parentName:"p"},"sub_module")," milik module ",(0,i.kt)("inlineCode",{parentName:"p"},"messaging")," adalah private module, yang di dalamnya ada item dengan visibility publik."),(0,i.kt)("p",null,"Agar ",(0,i.kt)("inlineCode",{parentName:"p"},"say_hello_message")," tidak bisa diakses dari ",(0,i.kt)("em",{parentName:"p"},"crate root")," karena ",(0,i.kt)("inlineCode",{parentName:"p"},"sub_module")," yang merupakan module scope item tersebut adalah private. Agar item tersebut bisa diakses dari publik, maka bisa dengan menggunakan teknik ",(0,i.kt)("em",{parentName:"p"},"re-export"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'pub use self::sub_module::say_hello_message;\n\nmod sub_module {\n\n pub fn say_hello_message() {\n println!("hello rust")\n }\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::say_hello_message();\n}\n")),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Visibility & Privacy",src:t(2201).Z,width:"466",height:"109"})),(0,i.kt)("p",null,"Bisa dilihat item ",(0,i.kt)("inlineCode",{parentName:"p"},"say_hello_message")," sekarang bisa diakses dari fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," via path ",(0,i.kt)("inlineCode",{parentName:"p"},"messaging::say_hello_message")," setelah di-re-export."),(0,i.kt)("p",null,"O iya, jika suatu item ingin di-re-export dengan nama berbeda, tambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"as")," diikuti alias atau nama item export. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'pub use self::sub_module::say_hello_message as say_hello;\n\nmod sub_module {\n\n pub fn say_hello_message() {\n println!("hello rust")\n }\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::say_hello();\n}\n")),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/use"},"github.com/novalagung/dasarpemrogramanrust-example/../use")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.use.html"},"https://doc.rust-lang.org/std/keyword.use.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/items/use-declarations.html"},"https://doc.rust-lang.org/reference/items/use-declarations.html"))))}d.isMDXComponent=!0},9772:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},4887:(e,a,t)=>{t.d(a,{Z:()=>n});const n=t.p+"assets/images/use-2-14040eb5ebbc4bd6148eec31b7d9f2e7.png"},2201:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/eb2a6637.570a9def.js b/assets/js/eb2a6637.570a9def.js new file mode 100644 index 00000000..4102ce1a --- /dev/null +++ b/assets/js/eb2a6637.570a9def.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[6388],{3905:(e,a,t)=>{t.d(a,{Zo:()=>s,kt:()=>A});var n=t(7294);function i(e,a,t){return a in e?Object.defineProperty(e,a,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[a]=t,e}function r(e,a){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);a&&(n=n.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),t.push.apply(t,n)}return t}function l(e){for(var a=1;a=0||(i[t]=e[t]);return i}(e,a);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(i[t]=e[t])}return i}var u=n.createContext({}),m=function(e){var a=n.useContext(u),t=a;return e&&(t="function"==typeof e?e(a):l(l({},a),e)),t},s=function(e){var a=m(e.components);return n.createElement(u.Provider,{value:a},e.children)},d={inlineCode:"code",wrapper:function(e){var a=e.children;return n.createElement(n.Fragment,{},a)}},k=n.forwardRef((function(e,a){var t=e.components,i=e.mdxType,r=e.originalType,u=e.parentName,s=p(e,["components","mdxType","originalType","parentName"]),k=m(t),A=i,o=k["".concat(u,".").concat(A)]||k[A]||d[A]||r;return t?n.createElement(o,l(l({ref:a},s),{},{components:t})):n.createElement(o,l({ref:a},s))}));function A(e,a){var t=arguments,i=a&&a.mdxType;if("string"==typeof e||i){var r=t.length,l=new Array(r);l[0]=k;var p={};for(var u in a)hasOwnProperty.call(a,u)&&(p[u]=a[u]);p.originalType=e,p.mdxType="string"==typeof e?e:i,l[1]=p;for(var m=2;m{t.r(a),t.d(a,{assets:()=>u,contentTitle:()=>l,default:()=>d,frontMatter:()=>r,metadata:()=>p,toc:()=>m});var n=t(7462),i=(t(7294),t(3905));const r={sidebar_position:29,title:"A.29. Module System \u279c Use, Import, Re-export",sidebar_label:"A.29. Module System \u279c Use, Import, Re-export"},l=void 0,p={unversionedId:"basic/use",id:"basic/use",title:"A.29. Module System \u279c Use, Import, Re-export",description:"Keyword use digunakan untuk dua hal, yaitu import path dan re-export path. Sebenarnya kita telah mempelajari kedua penerapan tersebut pada beberapa chapter sebelumnya. Pada chapter ini kita akan ulang lagi pembahasan agar lebih jelas.",source:"@site/docs/basic/use.md",sourceDirName:"basic",slug:"/basic/use",permalink:"/basic/use",draft:!1,tags:[],version:"current",sidebarPosition:29,frontMatter:{sidebar_position:29,title:"A.29. Module System \u279c Use, Import, Re-export",sidebar_label:"A.29. Module System \u279c Use, Import, Re-export"},sidebar:"tutorialSidebar",previous:{title:"A.28. Module System \u279c Visibility & Privacy",permalink:"/basic/visibility-privacy"},next:{title:"A.30. Block Expression",permalink:"/basic/block-expression"}},u={},m=[{value:"A.29.1. Keyword use untuk import path",id:"a291-keyword-use-untuk-import-path",level:2},{value:"\u25c9 Import items dari module yang sama",id:"-import-items-dari-module-yang-sama",level:3},{value:"A.29.2. Keyword use untuk re-export path",id:"a292-keyword-use-untuk-re-export-path",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],s={toc:m};function d(e){let{components:a,...r}=e;return(0,i.kt)("wrapper",(0,n.Z)({},s,r,{components:a,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," digunakan untuk dua hal, yaitu ",(0,i.kt)("em",{parentName:"p"},"import")," path dan ",(0,i.kt)("em",{parentName:"p"},"re-export")," path. Sebenarnya kita telah mempelajari kedua penerapan tersebut pada beberapa chapter sebelumnya. Pada chapter ini kita akan ulang lagi pembahasan agar lebih jelas."),(0,i.kt)("h2",{id:"a291-keyword-use-untuk-import-path"},"A.29.1. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"use")," untuk import path"),(0,i.kt)("p",null,"Untuk bisa menggunakan sebuah item dari crate lain, entah itu dari rust standard library crate maupun 3rd-party, caranya cukup dengan menuliskan path item. Contohnya bisa dilihat di bawah ini, fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"current_dir")," digunakan untuk mengambil path dari current directory. Fungsi tersebut merupakan item dari module ",(0,i.kt)("inlineCode",{parentName:"p"},"std::env"),", maka untuk mengaksesnya kita harus menuliskan path secara lengkap."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let package_path = std::env::current_dir().unwrap();\nprintln!("package_path: {:?}", package_path);\n')),(0,i.kt)("p",null,"Selain cara di atas, bisa juga gunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"use")," untuk meng-import fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"current_dir")," ke kode program, dengannya kita tidak perlu menulis path secara lengkap. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use std::env::current_dir;\n\nfn main() {\n let package_path = current_dir().unwrap();\n println!("package_path: {:?}", package_path);\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Rust use",src:t(9772).Z,width:"818",height:"104"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Di contoh, pengaksesan current directory di-chain dengan method ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," karena nilai balik fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"current_dir")," adalah bertipe ",(0,i.kt)("inlineCode",{parentName:"p"},"std::io::Result"),"."),(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai tipe tersebut dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/result-type"},"Tipe Data \u279c Result"))),(0,i.kt)("h3",{id:"-import-items-dari-module-yang-sama"},"\u25c9 Import items dari module yang sama"),(0,i.kt)("p",null,"Pada contoh selanjutnya ini, kita akan coba praktikkan penerapan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"std::env::args")," untuk mengambil ",(0,i.kt)("em",{parentName:"p"},"argument")," saat eksekusi program, kemudian menampilkannya ke layar."),(0,i.kt)("p",null,"Ok, berarti akan ada 2 path yang akan di-import:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::env::current_dir;\nuse std::env::args;\n")),(0,i.kt)("p",null,"Ada notasi penulisan import path lainnya yang bisa digunakan jika path-nya memiliki parent segment yang sama. Contohnya seperti di atas, kedua fungsi tersebut memiliki parent segment path yang sama yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"std::env"),". Yang seperti ini bisa dituliskan juga dengan notasi penulisan import berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"use std::env::{current_dir, args};\n")),(0,i.kt)("p",null,"Ok, lanjut ke praktik. Silakan tulis kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'use std::env::{current_dir, args};\n\nfn main() {\n let package_path = current_dir().unwrap();\n println!("package_path: {:?}", package_path);\n\n for i in 1..=args().len() {\n let each_arg = args().nth(i);\n if each_arg != None {\n println!("arg{}: {:?}", i, each_arg.unwrap());\n }\n }\n}\n')),(0,i.kt)("p",null,"Fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"std::env::args")," digunakan untuk mengambil argument eksekusi program baik eksekusi via ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run")," ataupun via pemanggilan binary. Fungsi tersebut nilai baliknya adalah iterator, jadi bisa digunakan dalam blok kode ",(0,i.kt)("inlineCode",{parentName:"p"},"for in")," dengan mudah."),(0,i.kt)("p",null,"Statement ",(0,i.kt)("inlineCode",{parentName:"p"},"args().nth(i)")," mengembalikan nilai argument pada index ke-",(0,i.kt)("inlineCode",{parentName:"p"},"i")," dalam tipe ",(0,i.kt)("inlineCode",{parentName:"p"},"Option"),", dan nilai tersebut bisa saja tidak ada (direpresentasikan dengan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),"). Oleh karena itu penting untuk dicek terlebih dahulu menggunakan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"if"),". Jika memang nilainya adalah selain ",(0,i.kt)("inlineCode",{parentName:"p"},"None"),", gunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"unwrap")," untuk mengambil nilainya dalam tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"."),(0,i.kt)("blockquote",null,(0,i.kt)("ul",{parentName:"blockquote"},(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai Iterator akan dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"li",href:"/basic/trait-iterator"},"Trait \u279c Iterator"),"."),(0,i.kt)("li",{parentName:"ul"},"Lebih jelasnya mengenai tipe ",(0,i.kt)("inlineCode",{parentName:"li"},"Option")," akan dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"li",href:"/basic/option-type"},"Tipe Data \u279c Option"),"."))),(0,i.kt)("p",null,"Silakan coba jalankan menggunakan dua command, yang pertama ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run"),", kemudian ",(0,i.kt)("inlineCode",{parentName:"p"},"cargo run tulis argumen di sini dengan pembatas spasi"),"."),(0,i.kt)("p",null,"Bisa dilihat pada gambar berikut, jika ada argument disisipkan dalam eksekusi program, maka ditampilkan. Pada gambar berikut dicontohkan argument yang dipakai adalah ",(0,i.kt)("inlineCode",{parentName:"p"},"hello")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"world"),"."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Rust use",src:t(4887).Z,width:"793",height:"223"})),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai Iterator dibahas pada chapter terpisah, ",(0,i.kt)("a",{parentName:"p",href:"/basic/trait-iterator"},"Trait \u279c Iterator"))),(0,i.kt)("h2",{id:"a292-keyword-use-untuk-re-export-path"},"A.29.2. Keyword ",(0,i.kt)("inlineCode",{parentName:"h2"},"use")," untuk re-export path"),(0,i.kt)("p",null,"Re-export item adalah sebuah cara untuk mem-",(0,i.kt)("em",{parentName:"p"},"bypass")," pengaksesan item yang secara hirarki memang tidak bisa diakses dari luar module (bisa jadi karena visibility item ataupun parent module nya adalah private). Dengan teknik ini, maka item pasti bisa diakses dari luar module."),(0,i.kt)("p",null,"Item yang di-re-export akan menjadi item milik ",(0,i.kt)("em",{parentName:"p"},"current module")," di mana statement re-export tersebut ditulis."),(0,i.kt)("p",null,"Keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"pub use")," digunakan untuk operasi re-export. Notasi penulisannya bisa dilihat di bawah ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},"// pub use the_path\npub use self::sub_module::say_hello_message;\n\n// pub use the_path as exported_name;\npub use self::sub_module::say_hello_message as say_hello;\n")),(0,i.kt)("p",null,"Contoh penerapan bisa dilihat pada kode berikut. Submodule ",(0,i.kt)("inlineCode",{parentName:"p"},"sub_module")," milik module ",(0,i.kt)("inlineCode",{parentName:"p"},"messaging")," adalah private module, yang di dalamnya ada item dengan visibility publik."),(0,i.kt)("p",null,"Agar ",(0,i.kt)("inlineCode",{parentName:"p"},"say_hello_message")," tidak bisa diakses dari ",(0,i.kt)("em",{parentName:"p"},"crate root")," karena ",(0,i.kt)("inlineCode",{parentName:"p"},"sub_module")," yang merupakan module scope item tersebut adalah private. Agar item tersebut bisa diakses dari publik, maka bisa dengan menggunakan teknik ",(0,i.kt)("em",{parentName:"p"},"re-export"),"."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'pub use self::sub_module::say_hello_message;\n\nmod sub_module {\n\n pub fn say_hello_message() {\n println!("hello rust")\n }\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::say_hello_message();\n}\n")),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Visibility & Privacy",src:t(2201).Z,width:"466",height:"109"})),(0,i.kt)("p",null,"Bisa dilihat item ",(0,i.kt)("inlineCode",{parentName:"p"},"say_hello_message")," sekarang bisa diakses dari fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"main")," via path ",(0,i.kt)("inlineCode",{parentName:"p"},"messaging::say_hello_message")," setelah di-re-export."),(0,i.kt)("p",null,"O iya, jika suatu item ingin di-re-export dengan nama berbeda, tambahkan keyword ",(0,i.kt)("inlineCode",{parentName:"p"},"as")," diikuti alias atau nama item export. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/messaging.rs"',title:'"src/messaging.rs"'},'pub use self::sub_module::say_hello_message as say_hello;\n\nmod sub_module {\n\n pub fn say_hello_message() {\n println!("hello rust")\n }\n}\n')),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust",metastring:'title="src/main.rs"',title:'"src/main.rs"'},"mod messaging;\n\nfn main() {\n messaging::say_hello();\n}\n")),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/use"},"github.com/novalagung/dasarpemrogramanrust-example/../use")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/std/keyword.use.html"},"https://doc.rust-lang.org/std/keyword.use.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/reference/items/use-declarations.html"},"https://doc.rust-lang.org/reference/items/use-declarations.html"))))}d.isMDXComponent=!0},9772:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""},4887:(e,a,t)=>{t.d(a,{Z:()=>n});const n=t.p+"assets/images/use-2-14040eb5ebbc4bd6148eec31b7d9f2e7.png"},2201:(e,a,t)=>{t.d(a,{Z:()=>n});const n=""}}]); \ No newline at end of file diff --git a/assets/js/f8d4e457.88f7a231.js b/assets/js/f8d4e457.88f7a231.js deleted file mode 100644 index 1bc90308..00000000 --- a/assets/js/f8d4e457.88f7a231.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[740],{3905:(a,e,n)=>{n.d(e,{Zo:()=>p,kt:()=>o});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function r(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var d=i.createContext({}),u=function(a){var e=i.useContext(d),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},p=function(a){var e=u(a.components);return i.createElement(d.Provider,{value:e},a.children)},s={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},k=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,l=a.originalType,d=a.parentName,p=m(a,["components","mdxType","originalType","parentName"]),k=u(n),o=t,g=k["".concat(d,".").concat(o)]||k[o]||s[o]||l;return n?i.createElement(g,r(r({ref:e},p),{},{components:n})):i.createElement(g,r({ref:e},p))}));function o(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var l=n.length,r=new Array(l);r[0]=k;var m={};for(var d in e)hasOwnProperty.call(e,d)&&(m[d]=e[d]);m.originalType=a,m.mdxType="string"==typeof a?a:t,r[1]=m;for(var u=2;u{n.r(e),n.d(e,{assets:()=>d,contentTitle:()=>r,default:()=>s,frontMatter:()=>l,metadata:()=>m,toc:()=>u});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:42,title:"A.42. Lifetime",sidebar_label:"A.42. Lifetime"},r=void 0,m={unversionedId:"basic/lifetime",id:"basic/lifetime",title:"A.42. Lifetime",description:"Pada chapter ini kita akan belajar tentang lifetime. Lifetime adalah yang digunakan oleh Rust compiler untuk memonitor umur dari references agar tetap dianggap valid.",source:"@site/docs/basic/lifetime.md",sourceDirName:"basic",slug:"/basic/lifetime",permalink:"/basic/lifetime",draft:!1,tags:[],version:"current",sidebarPosition:42,frontMatter:{sidebar_position:42,title:"A.42. Lifetime",sidebar_label:"A.42. Lifetime"},sidebar:"tutorialSidebar",previous:{title:"A.41. Static Item",permalink:"/basic/static"},next:{title:"A.43. Slice Memory Management",permalink:"/basic/slice-memory-management"}},d={},u=[{value:"A.42.1. Konsep Lifetime",id:"a421-konsep-lifetime",level:2},{value:"A.42.2. Relasi antara lifetime dengan owner dan borrower",id:"a422-relasi-antara-lifetime-dengan-owner-dan-borrower",level:2},{value:"A.42.3. Lifetime annotation dan penerapannya pada return value",id:"a423-lifetime-annotation-dan-penerapannya-pada-return-value",level:2},{value:"A.42.4. Lifetime pada parameter",id:"a424-lifetime-pada-parameter",level:2},{value:"A.42.5. Lifetime elision",id:"a425-lifetime-elision",level:2},{value:"A.42.6. Lifetime pada struct",id:"a426-lifetime-pada-struct",level:2},{value:"A.42.7. Lifetime pada method",id:"a427-lifetime-pada-method",level:2},{value:"A.42.8. Generic parameter + trait bounds + lifetime",id:"a428-generic-parameter--trait-bounds--lifetime",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],p={toc:u};function s(a){let{components:e,...l}=a;return(0,t.kt)("wrapper",(0,i.Z)({},p,l,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Pada chapter ini kita akan belajar tentang lifetime. Lifetime adalah yang digunakan oleh Rust compiler untuk memonitor umur dari references agar tetap dianggap valid."),(0,t.kt)("p",null,"Normalnya kita tidak perlu berurusan dengan lifetime, karena Rust-lah yang mengelola lifetime sebuah reference. Namun di luar itu, pada beberapa case kita bisa me-manage lifetime data dengan memanfaatkan ",(0,t.kt)("em",{parentName:"p"},"annotation"),"."),(0,t.kt)("p",null,"Ketika berurusan dengan data primitif maupun non-primitif tak perlu khawatir perihal urusan lifetime. Aspek lifetime hanya perlu diperhatikan sewaktu berurusan dengan data pointer/reference, apalagi kalau data tersebut keluar masuk block scope."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Topik lifetime adalah salah satu yang paling membingungkan di Rust. Wajar jika membutuhkan waktu lebih lama untuk menguasainya. ",(0,t.kt)("em",{parentName:"p"},"Take your time"),", pelajari pelan-pelan dan ulangi berkali-kali jika perlu.")),(0,t.kt)("h2",{id:"a421-konsep-lifetime"},"A.42.1. Konsep Lifetime"),(0,t.kt)("p",null,"Lifetime adalah yang digunakan oleh Rust compiler untuk memonitor umur dari references agar tetap valid. Lifetime menempel di variabel, lebih tepatnya di reference variabel."),(0,t.kt)("p",null,"Rust menerapkan default lifetime dalam pengecekan reference. Beberapa aturan pada default lifetime sudah kita pelajari pada chapter sebelumnya, seperti variabel yang hanya akan valid di dalam block dan invalid di luar block dan data yang akan di-dealokasi ketika sudah tidak ada reference-nya."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Rust mengidentifikasi default lifetime menggunakan ",(0,t.kt)("strong",{parentName:"p"},"lifetime elision"),", yang juga akan dibahas pada chapter ini.")),(0,t.kt)("p",null,"Agar mudah memahami konsep default lifetime, mari mulai dengan kode sederhana berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let r;\n {\n let x = 5;\n r = &x;\n }\n println!("r: {}", r);\n}\n')),(0,t.kt)("p",null,"Kode di atas kalau di jalankan hasilnya error, karena ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," di-dealokasi ketika block expression selesai dieksekusi, meskipun data tersebut dipinjamkan pada ",(0,t.kt)("inlineCode",{parentName:"p"},"r")," yang scope-nya berada di atasnya."),(0,t.kt)("p",null,"Kalau diilustrasikan, lifetime variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"r")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," kurang lebih seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n let r; // -----------+-- 'lf1\n { // |\n let x = 5; // -+-- 'lf2 |\n r = &x; // | |\n } // -+ |\n println!(\"r: {}\", r); // |\n} // -----------+\n")),(0,t.kt)("p",null,"Setiap data memiliki default lifetime."),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"r")," memiliki lifetime yang pada contoh di atas diilustrasikan sebagai ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf1"),"."),(0,t.kt)("li",{parentName:"ul"},"Variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"x")," memiliki lifetime yang pada contoh di atas diilustrasikan sebagai ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf2"),"."),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf2")," milik variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"x")," menjadikan umur variabel tersebut valid mulai variabel tersebut dideklarasikan, hingga block expression selesai."),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf2")," sudah tidak valid di luar block expression. Inilah kenapa program di atas menjadi error."),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf1")," milik variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"r")," menjadikan umur variabel tersebut valid mulai variabel tersebut dideklarasikan, hingga block fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"main")," selesai.")),(0,t.kt)("p",null,"Default lifetime bisa di-override menggunakan lifetime yang kita definisikan sendiri."),(0,t.kt)("h2",{id:"a422-relasi-antara-lifetime-dengan-owner-dan-borrower"},"A.42.2. Relasi antara lifetime dengan owner dan borrower"),(0,t.kt)("p",null,"Lifetime menjadi salah satu hal yang wajib diperhatikan ketika bermain dengan references. Operasi seperti melempar reference ke luar scope, atau memasukan reference ke block scope baru berpotensi memunculkan error yang berhubungan dengan lifetime."),(0,t.kt)("p",null,"Setelah ini kita akan praktik penerapan pembuatan lifetime, namun sebelum itu mari pelajari dulu pembahasan pada section berikut agar tau kenapa dan kapan kita harus menerapkan lifetime yang kita buat sendiri."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let message = String::from("darkspear is better than zandalari");\n print_mesage(&message)\n}\n\nfn print_mesage(m: &String) {\n println!("the message: {m}")\n}\n')),(0,t.kt)("p",null,"Program di atas menampilkan pesan string via fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"print_message"),". Data string didapat dari parameter pointer ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," milik fungsi tersebut."),(0,t.kt)("p",null,"Pada fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", ada string bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"message"),", niainya dipinjamkan sebagai argument pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"print_message"),"."),(0,t.kt)("p",null,"Setelah eksekusi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"print_message")," selesai, yang terjadi di block kode fungsi tersebut adalah data ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," di-dealokasi. Hasil dari dealokasi sendiri adalah nilai sebenarnya dikembalikan ke owner (pemilik aslinya). Sampai sini harusnya cukup jelas."),(0,t.kt)("p",null,"Sekarang lanjut ke contoh ke-2 berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let m: &String = get_message();\n println!("the message: {m}");\n}\n\nfn get_message() -> &String {\n let message = String::from("darkspear is better than zandalari");\n &message\n}\n')),(0,t.kt)("p",null,"Esensi program ke-2 ini sama seperti program sebelumnya, yaitu menampilkan pesan string yang sama persis. Perbedaannya, pesan string datanya ada di dalam fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_message"),". Fungsi tersebut dipanggil kemudian reference dari pesan string dipinjamkan, maka dengan ini variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," pada fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main")," nilainya adalah data pinjaman (borrowing)."),(0,t.kt)("p",null,"Ketika di-run, hasilnya error."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(8116).Z,width:"779",height:"275"})),(0,t.kt)("p",null,"Error tersebut muncul karena setelah eksekusi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_message")," selesai, semua data dalam fungsi tersebut di-dealokasi. Termasuk variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message")," yang merupakan owner dari data yang dipinjamkan ke variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m"),"."),(0,t.kt)("p",null,"Ok, jadi owner-nya sudah di-dealokasi, lalu bagaimana nasib dari peminjam data (variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m"),")? Variabel tersebut menjadi bermasalah, karena owner data aslinya sudah tidak ada di memory, dan itulah kenapa muncul error."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Error ini disebut dengan ",(0,t.kt)("strong",{parentName:"p"},"dangling reference"),", muncul ketika data di-share ke variabel lain tapi owner-nya sudah tidak ada di-memory."),(0,t.kt)("p",{parentName:"blockquote"},"Di pemrograman Rust, error jenis ini bisa di-identifikasi saat kompilasi.")),(0,t.kt)("p",null,"Solusi pada error di atas, salah satunya adalah dengan tidak menggunakan tipe data pointer sebagai nilai balik. Gunakan saja tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"String"),". Solusi ini aman, karena tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," owner-nya selalu berpindah saat ada operasi asignment, dengan ini maka manajemen memory menjadi efisien."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'// ganti kode berikut ...\nfn get_message() -> &String {\n let message = String::from("darkspear is better than zandalari");\n message\n}\n\n// ... menjadi ...\nfn get_message() -> String {\n let message = String::from("darkspear is better than zandalari");\n message\n}\n')),(0,t.kt)("p",null,"Ok, bagaimana dengan tipe data lain yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),", misalnya ada kebutuhan untuk share reference data tersebut ke block lain. Mari kita coba."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let n = get_number();\n println!("the number: {n}");\n}\n\nfn get_number() -> &i32 {\n let number = 13;\n &number\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(5384).Z,width:"782",height:"276"})),(0,t.kt)("p",null,"Bisa dilihat, ada error, karena kondisi tersebut menimbulkan dangling reference (sama seperti contoh sebelumnya). Namun pada contoh ini kita tidak bisa menerapkan solusi yang sama, karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"number")," mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),", bukan ",(0,t.kt)("em",{parentName:"p"},"move semantics"),"."),(0,t.kt)("p",null,"Di Rust error dangling reference diantisipasi saat kompilasi, menjadikan kode tidak bisa sukses dikompilasi."),(0,t.kt)("p",null,"Solusi yang bisa diambil adalah dengan menjadikan data tersebut tidak memiliki owner (atau owner-nya adalah program), caranya dengan langsung mengembalikan reference data tanpa perlu menampungnya terlebih dahulu ke variabel, seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let n = get_number();\n println!("the number: {n}");\n}\n\nfn get_number() -> &i32 {\n &13\n}\n')),(0,t.kt)("p",null,"Ok, tapi entah kenapa ketika dilihat masih muncul error."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(6875).Z,width:"777",height:"260"})),(0,t.kt)("p",null,"Error tersebut muncul karena meskipun owner data ",(0,t.kt)("inlineCode",{parentName:"p"},"&13")," adalah program, ketika eksekusi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_number")," selesai, data borrow tersebut langsung di-dealokasi, dan tidak ada variabel lain di-luar scope yang menampung reference data tersebut."),(0,t.kt)("p",null,"Agar tidak terjadi proses dealokasi, harus ada variabel yang menampung reference tersebut di-luar scope, tapi cara ini tidak bisa dilakukan karena data-nya saja baru dideklarasikan dalam block fungsi, tidak mungkin tiba-tiba ada yang menampung di-luar scope."),(0,t.kt)("p",null,"Solusi dari masalah ini adalah menggunakan lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'static")," (yang detailnya sudah dibahas pada chapter sebelumnya). Dengan ini maka data reference ",(0,t.kt)("inlineCode",{parentName:"p"},"&13")," hidup lebih lama dari umur yang sebenarnya sudah ditakdirkan untuk data data tersebut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn get_number() -> &'static i32 {\n &13\n}\n")),(0,t.kt)("p",null,"Namun perlu diingat, bahwa data efek dari lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'static")," adalah data tidak akan pernah di-dealokasi. Data tersebut akan hidup selamanya di program hingga kecuali program dimatikan. Karena alasan itulah penggunaan ",(0,t.kt)("inlineCode",{parentName:"p"},"'static")," pada contoh ini bisa disebut berlebihan."),(0,t.kt)("p",null,"Solusi yang lebih pas adalah dengan membuat lifetime sendiri dengan cara menerapkan ",(0,t.kt)("strong",{parentName:"p"},"lifetime annotation")," (tidak menggunakan lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'static"),")."),(0,t.kt)("h2",{id:"a423-lifetime-annotation-dan-penerapannya-pada-return-value"},"A.42.3. Lifetime annotation dan penerapannya pada return value"),(0,t.kt)("p",null,"Lifetime dituliskan dengan notasi ",(0,t.kt)("inlineCode",{parentName:"p"},"'nama_lifetime"),". Dengan notasi tersebut, kita bisa menciptakan lifetime baru misalnya ",(0,t.kt)("inlineCode",{parentName:"p"},"'a"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"'b"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"'ini_lifetime"),", dst."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"&i32 // => tipe data reference i32\n&'a i32 // => tipe data reference i32 dengan lifetime 'a\n&'a mut i32 // => tipe data mutable reference i32 dengan lifetime 'a\n")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lifetime dan block label memiliki bentuk literal yang sama, keduanya diawali tanda kutip ",(0,t.kt)("inlineCode",{parentName:"p"},"'"),". Yang membedakan hanya pada tempat dimana syntax tersebut ditulis.")),(0,t.kt)("p",null,"Kegunaan dari lifetime annotation adalah untuk menginformasikan compiler agar reference tidak langsung didealokasikan setelah eksekusi block selesai. Agar lebih jelas mari kita langsung terapkan saja pada fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_number")," yang sudah ditulis. Silakan tambahkan lifetime dengan nama bebas. Di sini penulis gunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"'my_lifetime"),".nnnn"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn get_number() -> &'my_lifetime i32 {\n &13\n}\n")),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(4982).Z,width:"689",height:"184"})),(0,t.kt)("p",null,"Meski sudah ditambahkan, error tetap muncul, karena lifetime tersebut tidak dikenal. Berbeda dengan lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'static")," yang memang sudah disediakan oleh Rust."),(0,t.kt)("p",null,"Step selanjutnya adalah mengenalkan lifetime tersebut, caranya dengan menambahkannya dalam notasi parameter generic fungsi, seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn get_number<'my_lifetime>() -> &'my_lifetime i32 {\n &13\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_number")," sekarang tidak menghasilkan error, karena reference yang dikembalikan memiliki umur ",(0,t.kt)("inlineCode",{parentName:"p"},"'my_lifetime")," yang membuatnya tetap valid sampai menjadi return value, meski eksekusi block fungsi-nya sendiri sudah selesai. Efeknya, data yang di-return bisa ditampung di luar scope fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_number"),"."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(4734).Z,width:"460",height:"104"})),(0,t.kt)("p",null,"Tanpa adanya lifetime pada return value, maka data return value akan langsung di-dealokasi setelah block fungsi selesai dieksekusi. Tapi karena hal ini ",(0,t.kt)("em",{parentName:"p"},"by default")," sudah di-handle Rust, maka kita tidak perlu memikirkannya."),(0,t.kt)("h2",{id:"a424-lifetime-pada-parameter"},"A.42.4. Lifetime pada parameter"),(0,t.kt)("p",null,"Pada praktik ini kita akan bahas penerapan lifetime pada parameter."),(0,t.kt)("p",null,"Silakan perhatikan kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v1(x: &str) -> &str {\n x\n}\n")),(0,t.kt)("p",null,"Kode di atas tidak menghasilkan error, karena kalau dilihat dari sudut pandang lifetime (sesuai dengan aturan lifetime elision), yang sebenarnya terjadi adalah kurang lebih seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v2<'a>(x: &'a str) -> &'a str {\n x\n}\n")),(0,t.kt)("p",null,"Pada saat parameter ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," masuk ke fungsi, ",(0,t.kt)("em",{parentName:"p"},"by default")," variabel tersebut memiliki lifetime (yang pada contoh di atas diilustrasikan dengan `'a'). Lifetime tersebut akan aktif hingga menjadi return value karena lifetime yang sama digunakan juga pada return value. Dengan ini nantinya setelah fungsi selesai dieksekusi, nilai baliknya tetap bisa ditampung."),(0,t.kt)("p",null,"Beda lagi jika lifetime lain digunakan pada nilai balik fungsi, misalnya:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v3<'a, 'b>(x: &'a str) -> &'b str {\n x\n}\n")),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(5109).Z,width:"736",height:"204"})),(0,t.kt)("p",null,"Error muncul dengan keterangan kurang lebih: data yang dijadikan return value pada block fungsi akan valid untuk lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'b"),", namun data yang dijadikan return value hanya valid untuk lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'a"),"."),(0,t.kt)("p",null,"Dari sini bisa ditarik kesimpulan: data yang didapat dari luar scope (yang memiliki lifetime sendiri) ketika digunakan sebagai nilai balik, lifetime-nya harus sama."),(0,t.kt)("p",null,"Sekarang mari kita modifikasi lagi fungsi menjadi seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v4<'a, 'b, 'c>(x: &'a str, y: &'b str) -> &'c str {\n \"hello\"\n}\n")),(0,t.kt)("p",null,"Fungsi di atas memiliki 2 buah lifetime, yaitu:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'a"),", digunakan pada parameter ",(0,t.kt)("inlineCode",{parentName:"li"},"x")),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'b"),", digunakan pada parameter ",(0,t.kt)("inlineCode",{parentName:"li"},"y")),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'c"),", digunakan pada return value")),(0,t.kt)("p",null,"Karena yang dikembalikan ada data baru, yaitu string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello"),", maka kode di atas tidak error. Data tersebut lifetime-nya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"'c"),", dan akan tetap valid setelah pemanggilan fungsi selesai."),(0,t.kt)("p",null,"Beda situasi jika yang dikembalikan adalah data dari parameter, misalnya, ",(0,t.kt)("inlineCode",{parentName:"p"},"x"),". Jika seperti ini, maka lifetime yang sama dengan lifetime parameter ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," harus digunakan, yaitu lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'b"),". Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v5<'a, 'b, 'c>(x: &'a str, y: &'b str) -> &'b str {\n y\n}\n")),(0,t.kt)("p",null,"Pada kode di atas, lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'c")," tidak dipergunakan sama sekali, maka lifetime tersebut boleh dihapus dari fungsi. Silakan cukup definisikan lifetime yang hanya digunakan saja pada block parameter generic."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v6<'a, 'b>(x: &'a str, y: &'b str) -> &'b str {\n y\n}\n")),(0,t.kt)("p",null,"Dimisalkan ada seleksi kondisi, jadi return value bisa saja ",(0,t.kt)("inlineCode",{parentName:"p"},"x"),", bisa juga ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),". Jika seperti ini, maka ",(0,t.kt)("inlineCode",{parentName:"p"},"x"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),", dan juga return value harus memiliki lifetime yang sama."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v7<'a>(x: &'a str, y: &'a str) -> &'a str {\n if x.len() < y.len() {\n x\n } else {\n y\n }\n}\n")),(0,t.kt)("h2",{id:"a425-lifetime-elision"},"A.42.5. Lifetime elision"),(0,t.kt)("p",null,"Sampai section ini kita telah mempelajari kurang lebih 4 point berikut:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},"Setiap data, lebih tepatnya setiap reference memiliki lifetime."),(0,t.kt)("li",{parentName:"ol"},"Lifetime digunakan oleh Rust dalam penentuan kapan reference tersebut di-dealokasi."),(0,t.kt)("li",{parentName:"ol"},"Pada beberapa case, lifetime perlu di-urus secara eksplisit (contohnya seperti pada fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"do_something_vx")," di atas)."),(0,t.kt)("li",{parentName:"ol"},"Pengecekan lifetime terjadi saat kompilasi.")),(0,t.kt)("p",null,"Rust memiliki sesuatu yang disebut dengan ",(0,t.kt)("strong",{parentName:"p"},"lifetime elision"),", isinya adalah aturan yang digunakan oleh Rust dalam menganalisa reference untuk menentukan lifetime ",(0,t.kt)("em",{parentName:"p"},"default"),"-nya."),(0,t.kt)("p",null,"Namun, bukan berarti Rust akan selalu tau lifetime tiap reference. Pada beberapa case, Rust membutuhkan bantuan kita selaku programmer untuk menginformasikan lifetime reference kode yang ditulis, contohnya seperti pada fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something_v7")," di atas."),(0,t.kt)("p",null,"Meskipun demikian, tak usah terlalu khawatir, karena pengecekan lifetime reference terjadi saat kompilasi, dan ketika ada reference yang perlu dikasih ",(0,t.kt)("em",{parentName:"p"},"annotication"),", Rust akan menginformasikan ke kita via pesan error."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Untuk sekarang, pembahasan detail mengenai lifetime elision tidak dibahas pada ebook ini. Silakan gunakan dokumentasi official Rust untuk ",(0,t.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/nomicon/lifetime-elision.html"},"lifetime elision")," jika berminat untuk pengkajian yang lebih dalam.")),(0,t.kt)("p",null,"Jika kawan-kawan menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"rust-analyzer")," extension di VSCode, tak perlu meng-compiler program untuk memunculkan error-nya, karena langsung muncul saat penulisan kode program."),(0,t.kt)("h2",{id:"a426-lifetime-pada-struct"},"A.42.6. Lifetime pada struct"),(0,t.kt)("p",null,"Tak hanya pada parameter fungsi dan return value fungsi, lifetime juga bisa diterapkan pada (property) struct."),(0,t.kt)("p",null,"Ketika menggunakan tipe data reference sebagai tipe property struct, Rust langsung menginformasikan kita via error message jika tidak ada anotasi lifetime pada field tersebut."),(0,t.kt)("p",null,"Contohnya bisa dilihat pada kode di bawah ini. Pesan error muncul karena tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"&str")," pada property struct tidak ada lifetime annotation-nya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"struct Book {\n title: &str,\n}\n")),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(7106).Z,width:"725",height:"160"})),(0,t.kt)("p",null,"Solusinya, tambahkan anotasi lifetime pada tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"&str"),", dan juga daftarkan lifetime tersebut pada struct. Notasi penulisannya seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"// struct NamaStruct<'lifetime_annotication> {\n// field: &'lifetime_annotication tipe_data,\n// }\n\nstruct Book<'abc> {\n title: &'abc str,\n}\n")),(0,t.kt)("p",null,"Contoh jika ada lebih dari 1 field dengan lifetime sama:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"struct Book<'abc> {\n title: &'abc str,\n description: &'abc str,\n}\n")),(0,t.kt)("p",null,"Contoh jika ada beberapa field yang di antaranya memiliki lifetime berbeda (artinya ada lebih dari 1 lifetime):"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"struct Book<'abc, 'def> {\n title: &'abc str,\n description: &'abc str,\n price: &'def i32,\n}\n\nfn main() {\n let book = Book {\n title: \"The Silmarillion\",\n description: \"Good story, 10/10, would read again\",\n price: &99,\n };\n}\n")),(0,t.kt)("h2",{id:"a427-lifetime-pada-method"},"A.42.7. Lifetime pada method"),(0,t.kt)("p",null,"Ada 2 hal yang perlu diketahui dalam penerapan lifetime pada method. Yang pertama, lifetime annotication harus ditulis pada block ",(0,t.kt)("inlineCode",{parentName:"p"},"impl")," meskipun pada block method tidak digunakan secara langsung."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"struct Book<'abc, 'def> {\n title: &'abc str,\n description: &'abc str,\n price: &'def i32,\n}\n\nimpl<'abc, 'def> Book<'abc, 'def> {\n\n fn get_info(&self) -> String {\n let info = format!(\"{} (${}), {}\", self.title, self.price, self.description);\n info\n }\n}\n")),(0,t.kt)("p",null,"Bisa dilihat statement block ",(0,t.kt)("inlineCode",{parentName:"p"},"impl<'abc, 'def> Book<'abc, 'def>"),", lifetime annotation-nya sama persis dengan yang ada di struct. Jika ingin mengetes, silakan coba saja hapus syntax lifetime dari statement itu, pasti muncul error."),(0,t.kt)("p",null,"Hal ke-2 yang penting diketahui, pada block kode method, tidak perlu menuliskan lifetime annotation, karena sudah ditulis di block kode ",(0,t.kt)("inlineCode",{parentName:"p"},"impl"),"."),(0,t.kt)("p",null,"Sebagai contoh, dua method berikut tidak memunculkan error:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"// ...\n\nimpl<'abc, 'def> Book<'abc, 'def> {\n\n fn get_info(&self) -> String {\n let info = format!(\"{} (${}), {}\", self.title, self.price, self.description);\n info\n }\n\n fn get_price(&self) -> &i32 {\n self.price\n }\n}\n")),(0,t.kt)("p",null,"Bandingkan dengan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_book_price")," berikut, error muncul karena lifetime annotation tidak ditambahkan ke block fungsi (meskipun fungsi tersebut mengembalikan property ",(0,t.kt)("inlineCode",{parentName:"p"},"price")," milik struct ",(0,t.kt)("inlineCode",{parentName:"p"},"Book")," yang sudah memiliki lifetime sendiri). Hal ini karena data ",(0,t.kt)("inlineCode",{parentName:"p"},"&i32")," yang dikembalikan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"book.get_price()")," langsung didealokasi setelah block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_book_price")," selesai dieksekusi."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(378).Z,width:"813",height:"223"})),(0,t.kt)("p",null,"Agar error tidak muncul, tambahkan lifetime:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},"ke return value (karena tipe data return value adalah reference)"),(0,t.kt)("li",{parentName:"ol"},"lalu pada parameter ",(0,t.kt)("inlineCode",{parentName:"li"},"book")," (karena di block fungsi return value berasal dari data milik property variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"book"),"),"),(0,t.kt)("li",{parentName:"ol"},"dan juga tak lupa daftarkan lifetime annotation pada block fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"get_book_price"),".")),(0,t.kt)("p",null,"Kurang kode menjadi seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'// ...\n\nfn get_book_price<\'ghi>(book: &\'ghi Book) -> &\'ghi i32 {\n book.get_price()\n}\n\nfn main() {\n let book = Book {\n title: "The Silmarillion",\n description: "Good story, 10/10, would read again",\n price: &99,\n };\n\n println!("{}", book.get_info());\n println!("the price: {}", get_book_price(&book));\n}\n')),(0,t.kt)("p",null,"Run program, hasilnya sukses."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(7146).Z,width:"473",height:"127"})),(0,t.kt)("h2",{id:"a428-generic-parameter--trait-bounds--lifetime"},"A.42.8. Generic parameter + trait bounds + lifetime"),(0,t.kt)("p",null,"Lalu bagaimana jika ada fungsi yang disitu ada penerapan trait bounds, ada juga generic parameter, dan lifetime annotation. Cara penulisannya seperti apa? Silakan lihat contoh berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn find_greater_number<'a, T>(\n x: &'a T,\n y: &'a T,\n) -> &'a T\nwhere\n T: std::cmp::PartialOrd,\n{\n if x > y {\n x\n } else {\n y\n }\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"find_greater_number")," di atas digunakan untuk mencari angka terbesar dari dua inputan parameter (",(0,t.kt)("inlineCode",{parentName:"p"},"x")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),"), dengan tipe data yang digunakan adalah generic ",(0,t.kt)("inlineCode",{parentName:"p"},"T"),". Dari sini maka nantinya fungsi ini bisa digunakan pada data bertipe ",(0,t.kt)("inlineCode",{parentName:"p"},"i32"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"f64"),", dan data numerik lainnya."),(0,t.kt)("p",null,"Tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"T")," diasosiasikan dengan trait ",(0,t.kt)("inlineCode",{parentName:"p"},"std::cmp::PartialOrd")," agar variabel dengan tipe tersebut bisa digunakan dalam seleksi kondisi ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," yang ada dalam block fungsi tersebut."),(0,t.kt)("p",null,"Ok, sampai sini semoga cukup jelas. Lalu bagaimana dengan lifetime annotation-nya? Karena lifetime annotation definisinya berada pada tempat yang sama dengan definisi tipe generic, maka langsung saja tulis disitu tanpa memperhatikan urutan. Sebagai contoh, dua definisi block fungsi berikut adalah ekuivalen."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn say_hello_v1<'abc, T>()\nfn say_hello_v2()\n")),(0,t.kt)("p",null,"Sekarang kembali ke pembahasan di atas. Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"find_greater_number")," mengembalikan tipe data reference ",(0,t.kt)("inlineCode",{parentName:"p"},"&T")," yang nilainya bisa saja dari ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),". Dari sini maka wajib hukumnya untuk return value memiliki lifetime yang sama dengan ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),". Itulah kenapa definisi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"find_greater_number")," agak panjang."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn find_greater_number<'a, T>(x: &'a T, y: &'a T) -> &'a T\n")),(0,t.kt)("p",null,"Lanjut, mari kita panggil fungsi tersebut pada dua block expression. Block pertama untuk pengecekan data numerik ",(0,t.kt)("inlineCode",{parentName:"p"},"i32"),", dan yang kedua untuk tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"f64"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n\n {\n // i32 comparison test\n let x = 13;\n let y = 20;\n let result = find_greater_number::(&x, &y); // atau cukup `find_greater_number(&x, &y)`\n println!("result: {}", result);\n }\n \n {\n // f64 comparison test\n let x = 3.14;\n let y = 2.11;\n let result = find_greater_number::(&x, &y); // atau cukup `find_greater_number(&x, &y)`\n println!("result: {}", result);\n }\n}\n')),(0,t.kt)("p",null,"Jalankan program, hasilnya sesuai harapan, tidak ada error."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(757).Z,width:"1072",height:"200"})),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/lifetime"},"github.com/novalagung/dasarpemrogramanrust-example/../lifetime")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch10-03-lifetime-syntax.html"},"https://doc.rust-lang.org/book/ch10-03-lifetime-syntax.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/nomicon/lifetimes.html"},"https://doc.rust-lang.org/nomicon/lifetimes.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://dev.to/takaakifuruse/rust-lifetimes-a-high-wall-for-rust-newbies-3ap"},"https://dev.to/takaakifuruse/rust-lifetimes-a-high-wall-for-rust-newbies-3ap")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://anooppoommen.medium.com/lifetimes-in-rust-7f2331be998b"},"https://anooppoommen.medium.com/lifetimes-in-rust-7f2331be998b")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://blog.logrocket.com/understanding-lifetimes-in-rust/"},"https://blog.logrocket.com/understanding-lifetimes-in-rust/")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://educative.io/answers/what-are-generic-lifetime-parameters-in-a-rust-function"},"https://educative.io/answers/what-are-generic-lifetime-parameters-in-a-rust-function"))))}s.isMDXComponent=!0},8116:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-1-49980dd171b372c65a19ab2c8bd2334f.png"},5384:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-2-1116463c388b09c1a8822e3b1ff4682c.png"},6875:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-3-d0688570ad7d667d603cde8665c863fc.png"},4982:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},4734:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},5109:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-6-661acde878a7b7fc851b9e76b22e60a7.png"},7106:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},378:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-8-da413de408eba24af5e0d0db885af9fc.png"},757:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-9-fix-96c668297e128c0b8d548a6cf72dfd0c.png"},7146:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/f8d4e457.ba1644a0.js b/assets/js/f8d4e457.ba1644a0.js new file mode 100644 index 00000000..97e0bd37 --- /dev/null +++ b/assets/js/f8d4e457.ba1644a0.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[740],{3905:(a,e,n)=>{n.d(e,{Zo:()=>p,kt:()=>o});var i=n(7294);function t(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function l(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(a);e&&(i=i.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,i)}return n}function r(a){for(var e=1;e=0||(t[n]=a[n]);return t}(a,e);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(a);for(i=0;i=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(t[n]=a[n])}return t}var d=i.createContext({}),u=function(a){var e=i.useContext(d),n=e;return a&&(n="function"==typeof a?a(e):r(r({},e),a)),n},p=function(a){var e=u(a.components);return i.createElement(d.Provider,{value:e},a.children)},s={inlineCode:"code",wrapper:function(a){var e=a.children;return i.createElement(i.Fragment,{},e)}},k=i.forwardRef((function(a,e){var n=a.components,t=a.mdxType,l=a.originalType,d=a.parentName,p=m(a,["components","mdxType","originalType","parentName"]),k=u(n),o=t,g=k["".concat(d,".").concat(o)]||k[o]||s[o]||l;return n?i.createElement(g,r(r({ref:e},p),{},{components:n})):i.createElement(g,r({ref:e},p))}));function o(a,e){var n=arguments,t=e&&e.mdxType;if("string"==typeof a||t){var l=n.length,r=new Array(l);r[0]=k;var m={};for(var d in e)hasOwnProperty.call(e,d)&&(m[d]=e[d]);m.originalType=a,m.mdxType="string"==typeof a?a:t,r[1]=m;for(var u=2;u{n.r(e),n.d(e,{assets:()=>d,contentTitle:()=>r,default:()=>s,frontMatter:()=>l,metadata:()=>m,toc:()=>u});var i=n(7462),t=(n(7294),n(3905));const l={sidebar_position:42,title:"A.42. Lifetime",sidebar_label:"A.42. Lifetime"},r=void 0,m={unversionedId:"basic/lifetime",id:"basic/lifetime",title:"A.42. Lifetime",description:"Pada chapter ini kita akan belajar tentang lifetime. Lifetime adalah yang digunakan oleh Rust compiler untuk memonitor umur dari references agar tetap dianggap valid.",source:"@site/docs/basic/lifetime.md",sourceDirName:"basic",slug:"/basic/lifetime",permalink:"/basic/lifetime",draft:!1,tags:[],version:"current",sidebarPosition:42,frontMatter:{sidebar_position:42,title:"A.42. Lifetime",sidebar_label:"A.42. Lifetime"},sidebar:"tutorialSidebar",previous:{title:"A.41. Static Item",permalink:"/basic/static"},next:{title:"A.43. Slice Memory Management",permalink:"/basic/slice-memory-management"}},d={},u=[{value:"A.42.1. Konsep Lifetime",id:"a421-konsep-lifetime",level:2},{value:"A.42.2. Relasi antara lifetime dengan owner dan borrower",id:"a422-relasi-antara-lifetime-dengan-owner-dan-borrower",level:2},{value:"A.42.3. Lifetime annotation dan penerapannya pada return value",id:"a423-lifetime-annotation-dan-penerapannya-pada-return-value",level:2},{value:"A.42.4. Lifetime pada parameter",id:"a424-lifetime-pada-parameter",level:2},{value:"A.42.5. Lifetime elision",id:"a425-lifetime-elision",level:2},{value:"A.42.6. Lifetime pada struct",id:"a426-lifetime-pada-struct",level:2},{value:"A.42.7. Lifetime pada method",id:"a427-lifetime-pada-method",level:2},{value:"A.42.8. Generic parameter + trait bounds + lifetime",id:"a428-generic-parameter--trait-bounds--lifetime",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],p={toc:u};function s(a){let{components:e,...l}=a;return(0,t.kt)("wrapper",(0,i.Z)({},p,l,{components:e,mdxType:"MDXLayout"}),(0,t.kt)("p",null,"Pada chapter ini kita akan belajar tentang lifetime. Lifetime adalah yang digunakan oleh Rust compiler untuk memonitor umur dari references agar tetap dianggap valid."),(0,t.kt)("p",null,"Normalnya kita tidak perlu berurusan dengan lifetime, karena Rust-lah yang mengelola lifetime sebuah reference. Namun di luar itu, pada beberapa case kita bisa me-manage lifetime data dengan memanfaatkan ",(0,t.kt)("em",{parentName:"p"},"annotation"),"."),(0,t.kt)("p",null,"Ketika berurusan dengan data primitif maupun non-primitif tak perlu khawatir perihal urusan lifetime. Aspek lifetime hanya perlu diperhatikan sewaktu berurusan dengan data pointer/reference, apalagi kalau data tersebut keluar masuk block scope."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Topik lifetime adalah salah satu yang paling membingungkan di Rust. Wajar jika membutuhkan waktu lebih lama untuk menguasainya. ",(0,t.kt)("em",{parentName:"p"},"Take your time"),", pelajari pelan-pelan dan ulangi berkali-kali jika perlu.")),(0,t.kt)("h2",{id:"a421-konsep-lifetime"},"A.42.1. Konsep Lifetime"),(0,t.kt)("p",null,"Lifetime adalah yang digunakan oleh Rust compiler untuk memonitor umur dari references agar tetap valid. Lifetime menempel di variabel, lebih tepatnya di reference variabel."),(0,t.kt)("p",null,"Rust menerapkan default lifetime dalam pengecekan reference. Beberapa aturan pada default lifetime sudah kita pelajari pada chapter sebelumnya, seperti variabel yang hanya akan valid di dalam block dan invalid di luar block dan data yang akan di-dealokasi ketika sudah tidak ada reference-nya."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Rust mengidentifikasi default lifetime menggunakan ",(0,t.kt)("strong",{parentName:"p"},"lifetime elision"),", yang juga akan dibahas pada chapter ini.")),(0,t.kt)("p",null,"Agar mudah memahami konsep default lifetime, mari mulai dengan kode sederhana berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let r;\n {\n let x = 5;\n r = &x;\n }\n println!("r: {}", r);\n}\n')),(0,t.kt)("p",null,"Kode di atas kalau di jalankan hasilnya error, karena ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," di-dealokasi ketika block expression selesai dieksekusi, meskipun data tersebut dipinjamkan pada ",(0,t.kt)("inlineCode",{parentName:"p"},"r")," yang scope-nya berada di atasnya."),(0,t.kt)("p",null,"Kalau diilustrasikan, lifetime variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"r")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," kurang lebih seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn main() {\n let r; // -----------+-- 'lf1\n { // |\n let x = 5; // -+-- 'lf2 |\n r = &x; // | |\n } // -+ |\n println!(\"r: {}\", r); // |\n} // -----------+\n")),(0,t.kt)("p",null,"Setiap data memiliki default lifetime."),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"r")," memiliki lifetime yang pada contoh di atas diilustrasikan sebagai ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf1"),"."),(0,t.kt)("li",{parentName:"ul"},"Variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"x")," memiliki lifetime yang pada contoh di atas diilustrasikan sebagai ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf2"),"."),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf2")," milik variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"x")," menjadikan umur variabel tersebut valid mulai variabel tersebut dideklarasikan, hingga block expression selesai."),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf2")," sudah tidak valid di luar block expression. Inilah kenapa program di atas menjadi error."),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'lf1")," milik variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"r")," menjadikan umur variabel tersebut valid mulai variabel tersebut dideklarasikan, hingga block fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"main")," selesai.")),(0,t.kt)("p",null,"Default lifetime bisa di-override menggunakan lifetime yang kita definisikan sendiri."),(0,t.kt)("h2",{id:"a422-relasi-antara-lifetime-dengan-owner-dan-borrower"},"A.42.2. Relasi antara lifetime dengan owner dan borrower"),(0,t.kt)("p",null,"Lifetime menjadi salah satu hal yang wajib diperhatikan ketika bermain dengan references. Operasi seperti melempar reference ke luar scope, atau memasukan reference ke block scope baru berpotensi memunculkan error yang berhubungan dengan lifetime."),(0,t.kt)("p",null,"Setelah ini kita akan praktik penerapan pembuatan lifetime, namun sebelum itu mari pelajari dulu pembahasan pada section berikut agar tau kenapa dan kapan kita harus menerapkan lifetime yang kita buat sendiri."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let message = String::from("darkspear is better than zandalari");\n print_mesage(&message)\n}\n\nfn print_mesage(m: &String) {\n println!("the message: {m}")\n}\n')),(0,t.kt)("p",null,"Program di atas menampilkan pesan string via fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"print_message"),". Data string didapat dari parameter pointer ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," milik fungsi tersebut."),(0,t.kt)("p",null,"Pada fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main"),", ada string bernama ",(0,t.kt)("inlineCode",{parentName:"p"},"message"),", niainya dipinjamkan sebagai argument pemanggilan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"print_message"),"."),(0,t.kt)("p",null,"Setelah eksekusi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"print_message")," selesai, yang terjadi di block kode fungsi tersebut adalah data ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," di-dealokasi. Hasil dari dealokasi sendiri adalah nilai sebenarnya dikembalikan ke owner (pemilik aslinya). Sampai sini harusnya cukup jelas."),(0,t.kt)("p",null,"Sekarang lanjut ke contoh ke-2 berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let m: &String = get_message();\n println!("the message: {m}");\n}\n\nfn get_message() -> &String {\n let message = String::from("darkspear is better than zandalari");\n &message\n}\n')),(0,t.kt)("p",null,"Esensi program ke-2 ini sama seperti program sebelumnya, yaitu menampilkan pesan string yang sama persis. Perbedaannya, pesan string datanya ada di dalam fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_message"),". Fungsi tersebut dipanggil kemudian reference dari pesan string dipinjamkan, maka dengan ini variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m")," pada fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"main")," nilainya adalah data pinjaman (borrowing)."),(0,t.kt)("p",null,"Ketika di-run, hasilnya error."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(8116).Z,width:"779",height:"275"})),(0,t.kt)("p",null,"Error tersebut muncul karena setelah eksekusi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_message")," selesai, semua data dalam fungsi tersebut di-dealokasi. Termasuk variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"message")," yang merupakan owner dari data yang dipinjamkan ke variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m"),"."),(0,t.kt)("p",null,"Ok, jadi owner-nya sudah di-dealokasi, lalu bagaimana nasib dari peminjam data (variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"m"),")? Variabel tersebut menjadi bermasalah, karena owner data aslinya sudah tidak ada di memory, dan itulah kenapa muncul error."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Error ini disebut dengan ",(0,t.kt)("strong",{parentName:"p"},"dangling reference"),", muncul ketika data di-share ke variabel lain tapi owner-nya sudah tidak ada di-memory."),(0,t.kt)("p",{parentName:"blockquote"},"Di pemrograman Rust, error jenis ini bisa di-identifikasi saat kompilasi.")),(0,t.kt)("p",null,"Solusi pada error di atas, salah satunya adalah dengan tidak menggunakan tipe data pointer sebagai nilai balik. Gunakan saja tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"String"),". Solusi ini aman, karena tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"String")," owner-nya selalu berpindah saat ada operasi asignment, dengan ini maka manajemen memory menjadi efisien."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'// ganti kode berikut ...\nfn get_message() -> &String {\n let message = String::from("darkspear is better than zandalari");\n message\n}\n\n// ... menjadi ...\nfn get_message() -> String {\n let message = String::from("darkspear is better than zandalari");\n message\n}\n')),(0,t.kt)("p",null,"Ok, bagaimana dengan tipe data lain yang mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),", misalnya ada kebutuhan untuk share reference data tersebut ke block lain. Mari kita coba."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let n = get_number();\n println!("the number: {n}");\n}\n\nfn get_number() -> &i32 {\n let number = 13;\n &number\n}\n')),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(5384).Z,width:"782",height:"276"})),(0,t.kt)("p",null,"Bisa dilihat, ada error, karena kondisi tersebut menimbulkan dangling reference (sama seperti contoh sebelumnya). Namun pada contoh ini kita tidak bisa menerapkan solusi yang sama, karena variabel ",(0,t.kt)("inlineCode",{parentName:"p"},"number")," mengadopsi ",(0,t.kt)("em",{parentName:"p"},"copy semantics"),", bukan ",(0,t.kt)("em",{parentName:"p"},"move semantics"),"."),(0,t.kt)("p",null,"Di Rust error dangling reference diantisipasi saat kompilasi, menjadikan kode tidak bisa sukses dikompilasi."),(0,t.kt)("p",null,"Solusi yang bisa diambil adalah dengan menjadikan data tersebut tidak memiliki owner (atau owner-nya adalah program), caranya dengan langsung mengembalikan reference data tanpa perlu menampungnya terlebih dahulu ke variabel, seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let n = get_number();\n println!("the number: {n}");\n}\n\nfn get_number() -> &i32 {\n &13\n}\n')),(0,t.kt)("p",null,"Ok, tapi entah kenapa ketika dilihat masih muncul error."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(6875).Z,width:"777",height:"260"})),(0,t.kt)("p",null,"Error tersebut muncul karena meskipun owner data ",(0,t.kt)("inlineCode",{parentName:"p"},"&13")," adalah program, ketika eksekusi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_number")," selesai, data borrow tersebut langsung di-dealokasi, dan tidak ada variabel lain di-luar scope yang menampung reference data tersebut."),(0,t.kt)("p",null,"Agar tidak terjadi proses dealokasi, harus ada variabel yang menampung reference tersebut di-luar scope, tapi cara ini tidak bisa dilakukan karena data-nya saja baru dideklarasikan dalam block fungsi, tidak mungkin tiba-tiba ada yang menampung di-luar scope."),(0,t.kt)("p",null,"Solusi dari masalah ini adalah menggunakan lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'static")," (yang detailnya sudah dibahas pada chapter sebelumnya). Dengan ini maka data reference ",(0,t.kt)("inlineCode",{parentName:"p"},"&13")," hidup lebih lama dari umur yang sebenarnya sudah ditakdirkan untuk data data tersebut."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn get_number() -> &'static i32 {\n &13\n}\n")),(0,t.kt)("p",null,"Namun perlu diingat, bahwa data efek dari lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'static")," adalah data tidak akan pernah di-dealokasi. Data tersebut akan hidup selamanya di program hingga kecuali program dimatikan. Karena alasan itulah penggunaan ",(0,t.kt)("inlineCode",{parentName:"p"},"'static")," pada contoh ini bisa disebut berlebihan."),(0,t.kt)("p",null,"Solusi yang lebih pas adalah dengan membuat lifetime sendiri dengan cara menerapkan ",(0,t.kt)("strong",{parentName:"p"},"lifetime annotation")," (tidak menggunakan lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'static"),")."),(0,t.kt)("h2",{id:"a423-lifetime-annotation-dan-penerapannya-pada-return-value"},"A.42.3. Lifetime annotation dan penerapannya pada return value"),(0,t.kt)("p",null,"Lifetime dituliskan dengan notasi ",(0,t.kt)("inlineCode",{parentName:"p"},"'nama_lifetime"),". Dengan notasi tersebut, kita bisa menciptakan lifetime baru misalnya ",(0,t.kt)("inlineCode",{parentName:"p"},"'a"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"'b"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"'ini_lifetime"),", dst."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"&i32 // => tipe data reference i32\n&'a i32 // => tipe data reference i32 dengan lifetime 'a\n&'a mut i32 // => tipe data mutable reference i32 dengan lifetime 'a\n")),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Lifetime dan block label memiliki bentuk literal yang sama, keduanya diawali tanda kutip ",(0,t.kt)("inlineCode",{parentName:"p"},"'"),". Yang membedakan hanya pada tempat di mana syntax tersebut ditulis.")),(0,t.kt)("p",null,"Kegunaan dari lifetime annotation adalah untuk menginformasikan compiler agar reference tidak langsung didealokasikan setelah eksekusi block selesai. Agar lebih jelas mari kita langsung terapkan saja pada fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_number")," yang sudah ditulis. Silakan tambahkan lifetime dengan nama bebas. Di sini penulis gunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"'my_lifetime"),".nnnn"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn get_number() -> &'my_lifetime i32 {\n &13\n}\n")),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(4982).Z,width:"689",height:"184"})),(0,t.kt)("p",null,"Meski sudah ditambahkan, error tetap muncul, karena lifetime tersebut tidak dikenal. Berbeda dengan lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'static")," yang memang sudah disediakan oleh Rust."),(0,t.kt)("p",null,"Step selanjutnya adalah mengenalkan lifetime tersebut, caranya dengan menambahkannya dalam notasi parameter generic fungsi, seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn get_number<'my_lifetime>() -> &'my_lifetime i32 {\n &13\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_number")," sekarang tidak menghasilkan error, karena reference yang dikembalikan memiliki umur ",(0,t.kt)("inlineCode",{parentName:"p"},"'my_lifetime")," yang membuatnya tetap valid sampai menjadi return value, meski eksekusi block fungsi-nya sendiri sudah selesai. Efeknya, data yang di-return bisa ditampung di luar scope fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_number"),"."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(4734).Z,width:"460",height:"104"})),(0,t.kt)("p",null,"Tanpa adanya lifetime pada return value, maka data return value akan langsung di-dealokasi setelah block fungsi selesai dieksekusi. Tapi karena hal ini ",(0,t.kt)("em",{parentName:"p"},"by default")," sudah di-handle Rust, maka kita tidak perlu memikirkannya."),(0,t.kt)("h2",{id:"a424-lifetime-pada-parameter"},"A.42.4. Lifetime pada parameter"),(0,t.kt)("p",null,"Pada praktik ini kita akan bahas penerapan lifetime pada parameter."),(0,t.kt)("p",null,"Silakan perhatikan kode berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v1(x: &str) -> &str {\n x\n}\n")),(0,t.kt)("p",null,"Kode di atas tidak menghasilkan error, karena kalau dilihat dari sudut pandang lifetime (sesuai dengan aturan lifetime elision), yang sebenarnya terjadi adalah kurang lebih seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v2<'a>(x: &'a str) -> &'a str {\n x\n}\n")),(0,t.kt)("p",null,"Pada saat parameter ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," masuk ke fungsi, ",(0,t.kt)("em",{parentName:"p"},"by default")," variabel tersebut memiliki lifetime (yang pada contoh di atas diilustrasikan dengan `'a'). Lifetime tersebut akan aktif hingga menjadi return value karena lifetime yang sama digunakan juga pada return value. Dengan ini nantinya setelah fungsi selesai dieksekusi, nilai baliknya tetap bisa ditampung."),(0,t.kt)("p",null,"Beda lagi jika lifetime lain digunakan pada nilai balik fungsi, misalnya:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v3<'a, 'b>(x: &'a str) -> &'b str {\n x\n}\n")),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(5109).Z,width:"736",height:"204"})),(0,t.kt)("p",null,"Error muncul dengan keterangan kurang lebih: data yang dijadikan return value pada block fungsi akan valid untuk lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'b"),", namun data yang dijadikan return value hanya valid untuk lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'a"),"."),(0,t.kt)("p",null,"Dari sini bisa ditarik kesimpulan: data yang didapat dari luar scope (yang memiliki lifetime sendiri) ketika digunakan sebagai nilai balik, lifetime-nya harus sama."),(0,t.kt)("p",null,"Sekarang mari kita modifikasi lagi fungsi menjadi seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v4<'a, 'b, 'c>(x: &'a str, y: &'b str) -> &'c str {\n \"hello\"\n}\n")),(0,t.kt)("p",null,"Fungsi di atas memiliki 2 buah lifetime, yaitu:"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'a"),", digunakan pada parameter ",(0,t.kt)("inlineCode",{parentName:"li"},"x")),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'b"),", digunakan pada parameter ",(0,t.kt)("inlineCode",{parentName:"li"},"y")),(0,t.kt)("li",{parentName:"ul"},"Lifetime ",(0,t.kt)("inlineCode",{parentName:"li"},"'c"),", digunakan pada return value")),(0,t.kt)("p",null,"Karena yang dikembalikan ada data baru, yaitu string ",(0,t.kt)("inlineCode",{parentName:"p"},"hello"),", maka kode di atas tidak error. Data tersebut lifetime-nya adalah ",(0,t.kt)("inlineCode",{parentName:"p"},"'c"),", dan akan tetap valid setelah pemanggilan fungsi selesai."),(0,t.kt)("p",null,"Beda situasi jika yang dikembalikan adalah data dari parameter, misalnya, ",(0,t.kt)("inlineCode",{parentName:"p"},"x"),". Jika seperti ini, maka lifetime yang sama dengan lifetime parameter ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," harus digunakan, yaitu lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'b"),". Contoh:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v5<'a, 'b, 'c>(x: &'a str, y: &'b str) -> &'b str {\n y\n}\n")),(0,t.kt)("p",null,"Pada kode di atas, lifetime ",(0,t.kt)("inlineCode",{parentName:"p"},"'c")," tidak dipergunakan sama sekali, maka lifetime tersebut boleh dihapus dari fungsi. Silakan cukup definisikan lifetime yang hanya digunakan saja pada block parameter generic."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v6<'a, 'b>(x: &'a str, y: &'b str) -> &'b str {\n y\n}\n")),(0,t.kt)("p",null,"Dimisalkan ada seleksi kondisi, jadi return value bisa saja ",(0,t.kt)("inlineCode",{parentName:"p"},"x"),", bisa juga ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),". Jika seperti ini, maka ",(0,t.kt)("inlineCode",{parentName:"p"},"x"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),", dan juga return value harus memiliki lifetime yang sama."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn do_something_v7<'a>(x: &'a str, y: &'a str) -> &'a str {\n if x.len() < y.len() {\n x\n } else {\n y\n }\n}\n")),(0,t.kt)("h2",{id:"a425-lifetime-elision"},"A.42.5. Lifetime elision"),(0,t.kt)("p",null,"Sampai section ini kita telah mempelajari kurang lebih 4 point berikut:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},"Setiap data, lebih tepatnya setiap reference memiliki lifetime."),(0,t.kt)("li",{parentName:"ol"},"Lifetime digunakan oleh Rust dalam penentuan kapan reference tersebut di-dealokasi."),(0,t.kt)("li",{parentName:"ol"},"Pada beberapa case, lifetime perlu di-urus secara eksplisit (contohnya seperti pada fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"do_something_vx")," di atas)."),(0,t.kt)("li",{parentName:"ol"},"Pengecekan lifetime terjadi saat kompilasi.")),(0,t.kt)("p",null,"Rust memiliki sesuatu yang disebut dengan ",(0,t.kt)("strong",{parentName:"p"},"lifetime elision"),", isinya adalah aturan yang digunakan oleh Rust dalam menganalisa reference untuk menentukan lifetime ",(0,t.kt)("em",{parentName:"p"},"default"),"-nya."),(0,t.kt)("p",null,"Namun, bukan berarti Rust akan selalu tau lifetime tiap reference. Pada beberapa case, Rust membutuhkan bantuan kita selaku programmer untuk menginformasikan lifetime reference kode yang ditulis, contohnya seperti pada fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"do_something_v7")," di atas."),(0,t.kt)("p",null,"Meskipun demikian, tak usah terlalu khawatir, karena pengecekan lifetime reference terjadi saat kompilasi, dan ketika ada reference yang perlu dikasih ",(0,t.kt)("em",{parentName:"p"},"annotication"),", Rust akan menginformasikan ke kita via pesan error."),(0,t.kt)("blockquote",null,(0,t.kt)("p",{parentName:"blockquote"},"Untuk sekarang, pembahasan detail mengenai lifetime elision tidak dibahas pada ebook ini. Silakan gunakan dokumentasi official Rust untuk ",(0,t.kt)("a",{parentName:"p",href:"https://doc.rust-lang.org/nomicon/lifetime-elision.html"},"lifetime elision")," jika berminat untuk pengkajian yang lebih dalam.")),(0,t.kt)("p",null,"Jika kawan-kawan menggunakan ",(0,t.kt)("inlineCode",{parentName:"p"},"rust-analyzer")," extension di VSCode, tak perlu meng-compiler program untuk memunculkan error-nya, karena langsung muncul saat penulisan kode program."),(0,t.kt)("h2",{id:"a426-lifetime-pada-struct"},"A.42.6. Lifetime pada struct"),(0,t.kt)("p",null,"Tak hanya pada parameter fungsi dan return value fungsi, lifetime juga bisa diterapkan pada (property) struct."),(0,t.kt)("p",null,"Ketika menggunakan tipe data reference sebagai tipe property struct, Rust langsung menginformasikan kita via error message jika tidak ada anotasi lifetime pada field tersebut."),(0,t.kt)("p",null,"Contohnya bisa dilihat pada kode di bawah ini. Pesan error muncul karena tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"&str")," pada property struct tidak ada lifetime annotation-nya."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"struct Book {\n title: &str,\n}\n")),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(7106).Z,width:"725",height:"160"})),(0,t.kt)("p",null,"Solusinya, tambahkan anotasi lifetime pada tipe data ",(0,t.kt)("inlineCode",{parentName:"p"},"&str"),", dan juga daftarkan lifetime tersebut pada struct. Notasi penulisannya seperti ini:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"// struct NamaStruct<'lifetime_annotication> {\n// field: &'lifetime_annotication tipe_data,\n// }\n\nstruct Book<'abc> {\n title: &'abc str,\n}\n")),(0,t.kt)("p",null,"Contoh jika ada lebih dari 1 field dengan lifetime sama:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"struct Book<'abc> {\n title: &'abc str,\n description: &'abc str,\n}\n")),(0,t.kt)("p",null,"Contoh jika ada beberapa field yang di antaranya memiliki lifetime berbeda (artinya ada lebih dari 1 lifetime):"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"struct Book<'abc, 'def> {\n title: &'abc str,\n description: &'abc str,\n price: &'def i32,\n}\n\nfn main() {\n let book = Book {\n title: \"The Silmarillion\",\n description: \"Good story, 10/10, would read again\",\n price: &99,\n };\n}\n")),(0,t.kt)("h2",{id:"a427-lifetime-pada-method"},"A.42.7. Lifetime pada method"),(0,t.kt)("p",null,"Ada 2 hal yang perlu diketahui dalam penerapan lifetime pada method. Yang pertama, lifetime annotication harus ditulis pada block ",(0,t.kt)("inlineCode",{parentName:"p"},"impl")," meskipun pada block method tidak digunakan secara langsung."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"struct Book<'abc, 'def> {\n title: &'abc str,\n description: &'abc str,\n price: &'def i32,\n}\n\nimpl<'abc, 'def> Book<'abc, 'def> {\n\n fn get_info(&self) -> String {\n let info = format!(\"{} (${}), {}\", self.title, self.price, self.description);\n info\n }\n}\n")),(0,t.kt)("p",null,"Bisa dilihat statement block ",(0,t.kt)("inlineCode",{parentName:"p"},"impl<'abc, 'def> Book<'abc, 'def>"),", lifetime annotation-nya sama persis dengan yang ada di struct. Jika ingin mengetes, silakan coba saja hapus syntax lifetime dari statement itu, pasti muncul error."),(0,t.kt)("p",null,"Hal ke-2 yang penting diketahui, pada block kode method, tidak perlu menuliskan lifetime annotation, karena sudah ditulis di block kode ",(0,t.kt)("inlineCode",{parentName:"p"},"impl"),"."),(0,t.kt)("p",null,"Sebagai contoh, dua method berikut tidak memunculkan error:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"// ...\n\nimpl<'abc, 'def> Book<'abc, 'def> {\n\n fn get_info(&self) -> String {\n let info = format!(\"{} (${}), {}\", self.title, self.price, self.description);\n info\n }\n\n fn get_price(&self) -> &i32 {\n self.price\n }\n}\n")),(0,t.kt)("p",null,"Bandingkan dengan fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_book_price")," berikut, error muncul karena lifetime annotation tidak ditambahkan ke block fungsi (meskipun fungsi tersebut mengembalikan property ",(0,t.kt)("inlineCode",{parentName:"p"},"price")," milik struct ",(0,t.kt)("inlineCode",{parentName:"p"},"Book")," yang sudah memiliki lifetime sendiri). Hal ini karena data ",(0,t.kt)("inlineCode",{parentName:"p"},"&i32")," yang dikembalikan statement ",(0,t.kt)("inlineCode",{parentName:"p"},"book.get_price()")," langsung didealokasi setelah block fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"get_book_price")," selesai dieksekusi."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(378).Z,width:"813",height:"223"})),(0,t.kt)("p",null,"Agar error tidak muncul, tambahkan lifetime:"),(0,t.kt)("ol",null,(0,t.kt)("li",{parentName:"ol"},"ke return value (karena tipe data return value adalah reference)"),(0,t.kt)("li",{parentName:"ol"},"lalu pada parameter ",(0,t.kt)("inlineCode",{parentName:"li"},"book")," (karena di block fungsi return value berasal dari data milik property variabel ",(0,t.kt)("inlineCode",{parentName:"li"},"book"),"),"),(0,t.kt)("li",{parentName:"ol"},"dan juga tak lupa daftarkan lifetime annotation pada block fungsi ",(0,t.kt)("inlineCode",{parentName:"li"},"get_book_price"),".")),(0,t.kt)("p",null,"Kurang kode menjadi seperti berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'// ...\n\nfn get_book_price<\'ghi>(book: &\'ghi Book) -> &\'ghi i32 {\n book.get_price()\n}\n\nfn main() {\n let book = Book {\n title: "The Silmarillion",\n description: "Good story, 10/10, would read again",\n price: &99,\n };\n\n println!("{}", book.get_info());\n println!("the price: {}", get_book_price(&book));\n}\n')),(0,t.kt)("p",null,"Run program, hasilnya sukses."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(7146).Z,width:"473",height:"127"})),(0,t.kt)("h2",{id:"a428-generic-parameter--trait-bounds--lifetime"},"A.42.8. Generic parameter + trait bounds + lifetime"),(0,t.kt)("p",null,"Lalu bagaimana jika ada fungsi yang di situ ada penerapan trait bounds, ada juga generic parameter, dan lifetime annotation. Cara penulisannya seperti apa? Silakan lihat contoh berikut:"),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn find_greater_number<'a, T>(\n x: &'a T,\n y: &'a T,\n) -> &'a T\nwhere\n T: std::cmp::PartialOrd,\n{\n if x > y {\n x\n } else {\n y\n }\n}\n")),(0,t.kt)("p",null,"Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"find_greater_number")," di atas digunakan untuk mencari angka terbesar dari dua inputan parameter (",(0,t.kt)("inlineCode",{parentName:"p"},"x")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),"), dengan tipe data yang digunakan adalah generic ",(0,t.kt)("inlineCode",{parentName:"p"},"T"),". Dari sini maka nantinya fungsi ini bisa digunakan pada data bertipe ",(0,t.kt)("inlineCode",{parentName:"p"},"i32"),", ",(0,t.kt)("inlineCode",{parentName:"p"},"f64"),", dan data numerik lainnya."),(0,t.kt)("p",null,"Tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"T")," diasosiasikan dengan trait ",(0,t.kt)("inlineCode",{parentName:"p"},"std::cmp::PartialOrd")," agar variabel dengan tipe tersebut bisa digunakan dalam seleksi kondisi ",(0,t.kt)("inlineCode",{parentName:"p"},"if")," yang ada dalam block fungsi tersebut."),(0,t.kt)("p",null,"Ok, sampai sini semoga cukup jelas. Lalu bagaimana dengan lifetime annotation-nya? Karena lifetime annotation definisinya berada pada tempat yang sama dengan definisi tipe generic, maka langsung saja tulis di situ tanpa memperhatikan urutan. Sebagai contoh, dua definisi block fungsi berikut adalah ekuivalen."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn say_hello_v1<'abc, T>()\nfn say_hello_v2()\n")),(0,t.kt)("p",null,"Sekarang kembali ke pembahasan di atas. Fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"find_greater_number")," mengembalikan tipe data reference ",(0,t.kt)("inlineCode",{parentName:"p"},"&T")," yang nilainya bisa saja dari ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," atau ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),". Dari sini maka wajib hukumnya untuk return value memiliki lifetime yang sama dengan ",(0,t.kt)("inlineCode",{parentName:"p"},"x")," dan ",(0,t.kt)("inlineCode",{parentName:"p"},"y"),". Itulah kenapa definisi fungsi ",(0,t.kt)("inlineCode",{parentName:"p"},"find_greater_number")," agak panjang."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},"fn find_greater_number<'a, T>(x: &'a T, y: &'a T) -> &'a T\n")),(0,t.kt)("p",null,"Lanjut, mari kita panggil fungsi tersebut pada dua block expression. Block pertama untuk pengecekan data numerik ",(0,t.kt)("inlineCode",{parentName:"p"},"i32"),", dan yang kedua untuk tipe ",(0,t.kt)("inlineCode",{parentName:"p"},"f64"),"."),(0,t.kt)("pre",null,(0,t.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n\n {\n // i32 comparison test\n let x = 13;\n let y = 20;\n let result = find_greater_number::(&x, &y); // atau cukup `find_greater_number(&x, &y)`\n println!("result: {}", result);\n }\n \n {\n // f64 comparison test\n let x = 3.14;\n let y = 2.11;\n let result = find_greater_number::(&x, &y); // atau cukup `find_greater_number(&x, &y)`\n println!("result: {}", result);\n }\n}\n')),(0,t.kt)("p",null,"Jalankan program, hasilnya sesuai harapan, tidak ada error."),(0,t.kt)("p",null,(0,t.kt)("img",{alt:"Lifetime",src:n(757).Z,width:"1072",height:"200"})),(0,t.kt)("hr",null),(0,t.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,t.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,t.kt)("pre",null,(0,t.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/lifetime"},"github.com/novalagung/dasarpemrogramanrust-example/../lifetime")),(0,t.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,t.kt)("ul",null,(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch10-03-lifetime-syntax.html"},"https://doc.rust-lang.org/book/ch10-03-lifetime-syntax.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/nomicon/lifetimes.html"},"https://doc.rust-lang.org/nomicon/lifetimes.html")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://dev.to/takaakifuruse/rust-lifetimes-a-high-wall-for-rust-newbies-3ap"},"https://dev.to/takaakifuruse/rust-lifetimes-a-high-wall-for-rust-newbies-3ap")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://anooppoommen.medium.com/lifetimes-in-rust-7f2331be998b"},"https://anooppoommen.medium.com/lifetimes-in-rust-7f2331be998b")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://blog.logrocket.com/understanding-lifetimes-in-rust/"},"https://blog.logrocket.com/understanding-lifetimes-in-rust/")),(0,t.kt)("li",{parentName:"ul"},(0,t.kt)("a",{parentName:"li",href:"https://educative.io/answers/what-are-generic-lifetime-parameters-in-a-rust-function"},"https://educative.io/answers/what-are-generic-lifetime-parameters-in-a-rust-function"))))}s.isMDXComponent=!0},8116:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-1-49980dd171b372c65a19ab2c8bd2334f.png"},5384:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-2-1116463c388b09c1a8822e3b1ff4682c.png"},6875:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-3-d0688570ad7d667d603cde8665c863fc.png"},4982:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},4734:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},5109:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-6-661acde878a7b7fc851b9e76b22e60a7.png"},7106:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""},378:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-8-da413de408eba24af5e0d0db885af9fc.png"},757:(a,e,n)=>{n.d(e,{Z:()=>i});const i=n.p+"assets/images/lifetime-9-fix-96c668297e128c0b8d548a6cf72dfd0c.png"},7146:(a,e,n)=>{n.d(e,{Z:()=>i});const i=""}}]); \ No newline at end of file diff --git a/assets/js/fb18633e.5c83c6b6.js b/assets/js/fb18633e.5c83c6b6.js new file mode 100644 index 00000000..09ed60a2 --- /dev/null +++ b/assets/js/fb18633e.5c83c6b6.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7310],{3905:(a,e,n)=>{n.d(e,{Zo:()=>m,kt:()=>k});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function l(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var s=t.createContext({}),u=function(a){var e=t.useContext(s),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},m=function(a){var e=u(a.components);return t.createElement(s.Provider,{value:e},a.children)},p={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},d=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,r=a.originalType,s=a.parentName,m=o(a,["components","mdxType","originalType","parentName"]),d=u(n),k=i,g=d["".concat(s,".").concat(k)]||d[k]||p[k]||r;return n?t.createElement(g,l(l({ref:e},m),{},{components:n})):t.createElement(g,l({ref:e},m))}));function k(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=n.length,l=new Array(r);l[0]=d;var o={};for(var s in e)hasOwnProperty.call(e,s)&&(o[s]=e[s]);o.originalType=a,o.mdxType="string"==typeof a?a:i,l[1]=o;for(var u=2;u{n.r(e),n.d(e,{assets:()=>s,contentTitle:()=>l,default:()=>p,frontMatter:()=>r,metadata:()=>o,toc:()=>u});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:35,title:"A.35. Borrowing",sidebar_label:"A.35. Borrowing"},l=void 0,o={unversionedId:"basic/borrowing",id:"basic/borrowing",title:"A.35. Borrowing",description:"Pada chapter ini kita akan belajar tentang apa itu borrowing dalam Rust programming.",source:"@site/docs/basic/borrowing.md",sourceDirName:"basic",slug:"/basic/borrowing",permalink:"/basic/borrowing",draft:!1,tags:[],version:"current",sidebarPosition:35,frontMatter:{sidebar_position:35,title:"A.35. Borrowing",sidebar_label:"A.35. Borrowing"},sidebar:"tutorialSidebar",previous:{title:"A.34. Ownership",permalink:"/basic/ownership"},next:{title:"A.36. Traits",permalink:"/basic/traits"}},s={},u=[{value:"A.35.1. Konsep borrowing",id:"a351-konsep-borrowing",level:2},{value:"A.35.2. Rust Borrow Checker",id:"a352-rust-borrow-checker",level:2},{value:"A.35.3. Aturan borrowing",id:"a353-aturan-borrowing",level:2},{value:"A.35.4. Borrowing mutable/immutable reference",id:"a354-borrowing-mutableimmutable-reference",level:2},{value:"\u25c9 Contoh ke-1",id:"-contoh-ke-1",level:3},{value:"\u25c9 Contoh ke-2",id:"-contoh-ke-2",level:3},{value:"\u25c9 Contoh ke-3",id:"-contoh-ke-3",level:3},{value:"A.35.5. Borrowing valid/invalid reference",id:"a355-borrowing-validinvalid-reference",level:2},{value:"A.35.6. Borrowing pada block",id:"a356-borrowing-pada-block",level:2},{value:"\u25c9 Method contains milik String",id:"-method-contains-milik-string",level:3},{value:"A.35.7. Owner dan borrower data literal",id:"a357-owner-dan-borrower-data-literal",level:2},{value:"A.35.8. Borrowing pada macro println",id:"a358-borrowing-pada-macro-println",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:u};function p(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,t.Z)({},m,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tentang apa itu borrowing dalam Rust programming."),(0,i.kt)("p",null,"Sebelum masuk ke pembelajaran, penulis anjurkan untuk paham terlebih dahulu tentang ",(0,i.kt)("a",{parentName:"p",href:"/basic/basic-memory-management"},"Basic Memory Management"),", ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),", dan juga konsep ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership"),". Ketiga topik tersebut dipelajari pada chapter sebelum ini."),(0,i.kt)("h2",{id:"a351-konsep-borrowing"},"A.35.1. Konsep borrowing"),(0,i.kt)("p",null,"Pada chapter sebelumnya kita telah belajar bahwa tipe data yang mengadopsi ",(0,i.kt)("em",{parentName:"p"},"move semantics")," (seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"), ketika digunakan pada operasi assignment maka owner-nya berpindah. Variabel yang sebelumnya adalah owner, setelah proses assignment menjadi invalid, karena owner data tersebut telah berpindah ke variabel baru. Efeknya, semua operasi yang dilakukan pada variabel owner sebelumnya menghasilkan error."),(0,i.kt)("p",null,"Sekarang perhatikan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let msg_1 = String::from("hello");\nlet msg_2 = msg_1;\n\nprintln!("{:?}", msg_2);\nprintln!("{:?}", msg_1);\n')),(0,i.kt)("p",null,"Pada contoh di atas, statement print ",(0,i.kt)("inlineCode",{parentName:"p"},"msg_1")," menghasilkan error karena variabel tersebut telah invalid setelah statement ",(0,i.kt)("inlineCode",{parentName:"p"},"let msg_2 = msg_1"),"."),(0,i.kt)("p",null,"Programmer harus ekstra hati-hati dan bijak dalam pengelolaan owner data, agar memory ter-manage dengan baik. Tapi kalau dipikir-pikir, repot juga kalau owner harus dilempar-lempar setiap selesai digunakan agar tidak error."),(0,i.kt)("p",null,"Salah satu cara yang bisa digunakan agar tidak terlalu repot adalah dengan menerapkan cloning (seperti yang sudah dibahas pada chapter sebelumnya). Namun ini bukan opsi yang baik karena boros memory, sedangkan di sisi lain programmer dianjurkan untuk efisien dalam penggunaan memory."),(0,i.kt)("p",null,"Solusi yang paling pas adalah dengan menerapkan ",(0,i.kt)("strong",{parentName:"p"},"borrowing"),". Borrowing artinya adalah meminjam. Pada konteks Rust programming, borrowing berarti meminjam data milik owner, dipinjam agar bisa diakses tanpa perlu memindah owner-nya. Kemudian setelah peminjaman selesai, data dikembalikan."),(0,i.kt)("p",null,"Cara meminjam data di Rust sangat mudah, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Untuk borrowing dengan level akses immutable/read-only, gunakan operator reference ",(0,i.kt)("inlineCode",{parentName:"p"},"&")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let msg_3 = String::from("hello rust");\nlet msg_4 = &msg_3; // <----- borrow operation\n\nprintln!("{:?}", msg_4); // output => hello rust\nprintln!("{:?}", msg_3); // output => hello rust\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Untuk borrowing dengan level akses mutable, gunakan operator reference ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut msg_3 = String::from("hello");\nlet msg_4 = &mut msg_3; // <----- mutable borrow operation\n\n*msg_4 = String::from("hello rust");\n\nprintln!("{:?}", msg_4); // output => hello rust\nprintln!("{:?}", msg_3); // output => hello rust\n')))),(0,i.kt)("p",null,"Di Rust, semua statement reference (baik mutable ataupun immutable) adalah operasi ",(0,i.kt)("em",{parentName:"p"},"borrowing"),". Yang terjadi pada statement reference adalah data milik owner dipinjam dalam bentuk pointer. Pointer itu sendiri merupakan alamat memory yang mengarah ke data sebenarnya (milik owner)."),(0,i.kt)("p",null,"Dari contoh di atas, bisa ditarik kesimpulan bahwa data string ",(0,i.kt)("inlineCode",{parentName:"p"},"hello rust")," memiliki dua reference:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Yang pertama adalah owner data, yaitu variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"msg_3")),(0,i.kt)("li",{parentName:"ul"},"Yang kedua adalah peminjam data, yaitu variabel pointer ",(0,i.kt)("inlineCode",{parentName:"li"},"msg_4"))),(0,i.kt)("h2",{id:"a352-rust-borrow-checker"},"A.35.2. Rust Borrow Checker"),(0,i.kt)("p",null,"Rust compiler memiliki 1 bagian bernama ",(0,i.kt)("strong",{parentName:"p"},"borrow checker"),", tugasnya untuk melakukan pengecekan pada source code apakah ada kode yang berhubungan dengan ownership dan borrowing, dan apakah kode tersebut mengikuti aturan borrowing yang sudah ditetapkan oleh Rust, atau tidak. Jika ada yang menyalahi aturan, maka borrow checker memunculkan error."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Borrow checker sangat galak, pastikan kode yang ditulis mengikuti aturan yang berlaku di pemrograman Rust.")),(0,i.kt)("h2",{id:"a353-aturan-borrowing"},"A.35.3. Aturan borrowing"),(0,i.kt)("p",null,"Aturan borrowing atau reference sempat disinggung pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),", yang kurang lebih adalah:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Dalam waktu yang sama, hanya boleh ada satu mutable reference atau banyak immutable reference (keduanya tidak bisa bersamaan, harus salah satu)."),(0,i.kt)("li",{parentName:"ul"},"Reference harus selalu valid.")),(0,i.kt)("p",null,"Dua aturan tersebut wajib dipatuhi, jika tidak maka pasti muncul error."),(0,i.kt)("p",null,"Ok, selanjutnya mari kita test aturan tersebut."),(0,i.kt)("h2",{id:"a354-borrowing-mutableimmutable-reference"},"A.35.4. Borrowing mutable/immutable reference"),(0,i.kt)("h3",{id:"-contoh-ke-1"},"\u25c9 Contoh ke-1"),(0,i.kt)("p",null,"Pada contoh berikut kita simulasikan 1 data memiliki banyak immutable reference. Jika mengacu ke aturan di atas, maka hal seperti ini adalah boleh dan tidak menghasilkan error."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let msg_5 = String::from("hello rust");\n\nlet msg_6 = &msg_5;\nlet msg_7 = &msg_5;\nlet msg_8 = &msg_5;\n\nprintln!("{:?} {:?} {:?}", msg_6, msg_7, msg_8);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(9470).Z,width:"470",height:"301"})),(0,i.kt)("h3",{id:"-contoh-ke-2"},"\u25c9 Contoh ke-2"),(0,i.kt)("p",null,"Pada contoh ke-2 ini, kita coba simulasikan 1 data memiliki 1 mutable reference. Harusnya tidak muncul error karena diperbolehkan di aturan yang tertulis."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut msg_9 = String::from("hello rust");\n\nlet msg_10 = &mut msg_9;\n\nprintln!("{:?}", msg_10);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(5982).Z,width:"497",height:"272"})),(0,i.kt)("p",null,"Bagaimana jika ada lebih dari 1 mutable reference? mari kita test."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut msg_9 = String::from("hello rust");\n\nlet msg_10 = &mut msg_9;\nlet msg_11 = &mut msg_9;\n\nprintln!("{:?} {:?}", msg_10, msg_11);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(8301).Z,width:"652",height:"613"})),(0,i.kt)("p",null,"Hasilnya error. Hal seperti ini tidak diperbolehkan. Sebuah data tidak boleh memiliki lebih dari 1 mutable reference."),(0,i.kt)("h3",{id:"-contoh-ke-3"},"\u25c9 Contoh ke-3"),(0,i.kt)("p",null,"Sekarang mari kita coba test lagi aturan di atas dengan skenario: dalam waktu yang sama, 1 data memiliki 1 mutable reference dan 1 immutable reference. Hasilnya seperti apa, (jika mengacu ke aturan, hal seperti ini adalah tidak diperbolehkan)."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut msg_12 = String::from("hello rust");\n\nlet msg_13 = &msg_12;\nlet msg_14 = &mut msg_12;\n\nprintln!("{:?} {:?}", msg_13, msg_14);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(6389).Z,width:"652",height:"486"})),(0,i.kt)("p",null,"Hasilnya error, kenapa? karena memang tidak boleh."),(0,i.kt)("p",null,"Jadi sampai sini cukup jelas ya. Sebuah data dalam waktu yang sama hanya diperbolehkan memiliki satu atau lebih immutable reference, atau hanya 1 mutable reference, dan keduanya tidak bisa bersamaan dalam satu waktu (harus pilih salah satu)."),(0,i.kt)("h2",{id:"a355-borrowing-validinvalid-reference"},"A.35.5. Borrowing valid/invalid reference"),(0,i.kt)("p",null,"Kita telah belajar tentang ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership#a333-variable-scope"},"valid/invalid variable dan juga variable scope")," pada chapter sebelumnya. Sekarang kita akan coba gabungkan dua aspek tersebut dengan borrowing."),(0,i.kt)("p",null,"Silakan praktikkan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut fact_one = String::from("Arthas is the true lich king");\n\n change_value(&mut fact_one);\n\n let fact_two = &mut fact_one;\n println!("{:?}", fact_two);\n}\n\nfn change_value(txt: &mut String) {\n *txt = String::from("Bolvar is better lich king");\n}\n')),(0,i.kt)("p",null,"Pada kode di atas, ada satu buah mutable string bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one"),". String tersebut dipinjam menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")," (yang berarti adalah mutable borrow) sebagai argumen pemanggilan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value")," yang isinya kurang lebih adalah perubahan isi data string pada variabel pointer."),(0,i.kt)("p",null,"Kemudian, ada operasi peminjaman lagi (yang juga mutable borrow) dari variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one"),"."),(0,i.kt)("p",null,"Ketika program di-run, hasilnya sukses. Kok bisa? padahal jelas di aturan tertulis bahwa dalam waktu yang sama tidak boleh ada lebih dari satu mutable borrow."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(538).Z,width:"449",height:"88"})),(0,i.kt)("p",null,"Kata ",(0,i.kt)("strong",{parentName:"p"},"dalam waktu yang sama")," di sini adalah yang penting untuk dipahami. Arti ",(0,i.kt)("em",{parentName:"p"},"dalam waktu yang sama")," adalah dalam 1 scope yang sama. Jika ada 2 scope, maka itu sudah bukan dalam waktu yang sama lagi."),(0,i.kt)("p",null,"Pada contoh di atas, statement ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut fact_one")," terjadi pada block fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),", lebih tepatnya pada parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"txt")," fungsi tersebut."),(0,i.kt)("p",null,"Kemudian setelah eksekusi fungsi tersebut selesai, yang terjadi adalah: reference yang tadinya dipinjam, sekarang dikembalikan. Tepat setelah eksekusi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),", state data ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one")," adalah tidak dipinjam siapapun. Tadinya memang ada yang meminjam (yaitu parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"txt")," di fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),"), tapi setelah eksekusi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value")," selesai, data dikembalikan lagi ke owner."),(0,i.kt)("p",null,"Proses dealokasi pada variable scope, jika terjadi pada variable reference maka yang sebenarnya terjadi adalah pengembalian data hasil operasi borrow ke pemilik aslinya."),(0,i.kt)("p",null,"Kemudian, ada operasi pinjam lagi, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"let fact_two = &mut fact_one"),". Statement borrow ini tidak menghasilkan error karena memang kondisi data ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one")," sudah tidak ada yang meminjam. Bisa dibilang statement peminjaman ke-2 ini tidak terjadi dalam waktu yang sama dengan statement peminjaman pertama (pemanggilan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),")."),(0,i.kt)("p",null,"Pada fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),", variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"txt")," adalah valid saat fungsi dijalankan. Setelah pemanggilan fungsi selesai, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"txt")," di-dealokasi. Namun karena variabel tersebut bukanlah owner, melainkan hanya borrower yang meminjam data dari owner ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one"),", maka yang terjadi adalah: data yang dipinjam sekarang dikembalikan lagi ke pemilik aslinya."),(0,i.kt)("p",null,"Ok sampai sini semoga cukup jelas ya. Jika perlu silakan ulang-ulang penjelasan di atas."),(0,i.kt)("h2",{id:"a356-borrowing-pada-block"},"A.35.6. Borrowing pada block"),(0,i.kt)("p",null,"Kita sudah cukup paham kapan waktu terjadinya alokasi alamat memory dan juga kapan proses dealokasi terjadi, yaitu ketika variabel ",(0,i.kt)("strong",{parentName:"p"},"into scope")," dan ketika variabel ",(0,i.kt)("strong",{parentName:"p"},"out of scope"),"."),(0,i.kt)("p",null,"Scope di sini adalah variable scope, yang maksudnya adalah block scope, bisa berupa block expression, fungsi, block seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"if"),", dan juga jenis block lainnya."),(0,i.kt)("p",null,"Ok, sekarang mari kita praktikkan beberapa jenis block untuk isolasi operasi borrowing. Dengan ini maka klausul ",(0,i.kt)("strong",{parentName:"p"},"dalam satu waktu")," akan terpenuhi."),(0,i.kt)("p",null,"Silakan modifikasi program sebelumnya menjadi seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut fact_one = String::from("Arthas is the true lich king");\n println!("{:?}", fact_one);\n\n change_value(&mut fact_one);\n println!("{:?}", fact_one);\n\n {\n let fact_two = &mut fact_one;\n *fact_two = String::from("There must always be a lich king");\n println!("{:?}", fact_one);\n }\n\n if fact_one.contains("lich king") {\n let fact_three = &mut fact_one;\n *fact_three = String::from("Who is the real jailer?");\n println!("{:?}", fact_one);\n }\n\n for _ in 0..1 {\n let fact_four = &mut fact_one;\n *fact_four = String::from("Is it Zovaal or Primus?");\n println!("{:?}", fact_one);\n }\n}\n\nfn change_value(txt: &mut String) {\n *txt = String::from("Bolvar is better lich king");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(2751).Z,width:"459",height:"175"})),(0,i.kt)("p",null,"Bisa dilihat, hasilnya tidak error, meskipun terjadi beberapa kali operasi mutable borrow, tapi karena kesemuanya terjadi di waktu yang berbeda (di scope yang berbeda) maka tidak error."),(0,i.kt)("p",null,"Pada kode di atas, mutable borrow terjadi di banyak tempat, yaitu di block fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),", block expression, block seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"if"),", dan juga block perulangan ",(0,i.kt)("inlineCode",{parentName:"p"},"for"),". Variabel pointer yang menampung data pinjaman akan valid di masing-masing block, kemudian menjadi invalid setelah eksekusi block selesai, dan yang terjadi setelah itu adalah pengembalian data yang telah dipinjam."),(0,i.kt)("h3",{id:"-method-contains-milik-string"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"contains")," milik ",(0,i.kt)("inlineCode",{parentName:"h3"},"String")),(0,i.kt)("p",null,"Pada contoh di atas kita menerapkan method baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"contains"),". Method ini tersedia untuk tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", gunanya adalah untuk mengecek apakah string memiliki substring ",(0,i.kt)("inlineCode",{parentName:"p"},"x"),", yang mana ",(0,i.kt)("inlineCode",{parentName:"p"},"x")," sendiri adalah argumen pemanggilan method. Method ini mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"bool"),"."),(0,i.kt)("p",null,"Contoh penerapan method ",(0,i.kt)("inlineCode",{parentName:"p"},"contains"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let fact = String::from("There must always be a lich king");\nprintln!("{:?}", fact.contains("lich king")); // output => true\nprintln!("{:?}", fact.contains("bolvar")); // output => false\n')),(0,i.kt)("p",null,"O iya, pengecekan string-nya adalah case sensitive ya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let fact = String::from("There must always be a lich king");\nprintln!("{:?}", fact.contains("lich king")); // output => true\nprintln!("{:?}", fact.contains("Lich King")); // output => false\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai tipe data string dan method yang tersedia pada tipe tersebut akan dibahas terpisah pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/string-slice-vs-string-literal"},"Tipe Data \u279c String Custom Type vs &str"),".")),(0,i.kt)("h2",{id:"a357-owner-dan-borrower-data-literal"},"A.35.7. Owner dan borrower data literal"),(0,i.kt)("p",null,"Perhatikan statement berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let number = 12;\nlet a = &number;\n\nlet text = String::from("hello");\nlet b = &text;\n')),(0,i.kt)("p",null,"Pada kode di atas, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"number")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"text")," adalah ",(0,i.kt)("em",{parentName:"p"},"owner")," data masing-masing. Sedangkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"a")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"b")," adalah ",(0,i.kt)("em",{parentName:"p"},"borrower")," atau peminjam data (yang lebih jelasnya akan dibahas pada chapter ). Sampai sini penulis rasa cukup jelas."),(0,i.kt)("p",null,"Selanjutnya, bagaimana dengan contoh ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let c = &24;\nlet d = &false;\nlet e = &String::from("rust");\n')),(0,i.kt)("p",null,"Ketiga variabel, semuanya adalah ",(0,i.kt)("em",{parentName:"p"},"borrower")," atau peminjam data. Lalu siapa ",(0,i.kt)("em",{parentName:"p"},"owner"),"-nya? jawabannya adalah tidak ada, atau silakan juga simpulkan bawah data tersebut owner-nya adalah program."),(0,i.kt)("p",null,"Yang lebih penting untuk diurusi pada contoh di atas bukan siapa owner-nya, melainkan bagaimana caranya agar data pinjaman tersebut tidak di-dealokasi saat block kode selesai."),(0,i.kt)("p",null,"Untuk sekarang sampai sini dulu. Kita akan bahas topik ini lebih mendetail pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/static"},"Static item")," dan ",(0,i.kt)("a",{parentName:"p",href:"/basic/lifetime"},"Lifetime"),"."),(0,i.kt)("h2",{id:"a358-borrowing-pada-macro-println"},"A.35.8. Borrowing pada macro ",(0,i.kt)("inlineCode",{parentName:"h2"},"println")),(0,i.kt)("p",null,"Pada pemanggilan macro println untuk menampilkan data yang mengadopsi ",(0,i.kt)("em",{parentName:"p"},"move semantics"),", operasi borrowing tidak perlu dilakukan sewaktu pengisian argument. Karena macro tersebut secara cerdas akan melakukan opearsi peminjaman tanpa memindah owner-nya. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str1 = String::from("luwe");\nprintln!("{:?}", str1);\n')),(0,i.kt)("p",null,"... adalah ekuivalen dengan ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str2 = String::from("ngelak");\nprintln!("{:?}", &str2);\n')),(0,i.kt)("p",null,"Lebih jelasnya mengenai macro dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/macro"},"Macro"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/borrowing"},"github.com/novalagung/dasarpemrogramanrust-example/../borrowing")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("p",null,"Beberapa referensi terkait chapter ini:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-02-references-and-borrowing.html"},"https://doc.rust-lang.org/book/ch04-02-references-and-borrowing.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/57225055/in-rust-can-you-own-a-string-literal"},"https://stackoverflow.com/questions/57225055/in-rust-can-you-own-a-string-literal"))))}p.isMDXComponent=!0},9470:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/borrowing-1-6abee77c6350f8a390a807adac852003.png"},5982:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/borrowing-2-cf66c8ef1c4b76c209f6d8c7d3339a3f.png"},8301:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/borrowing-3-55cc0f7a0b551f62c748a9bf0396a671.png"},6389:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/borrowing-4-c806f4e6787a069015fe56b40e97b3ca.png"},538:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},2751:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/fb18633e.b4caf1fa.js b/assets/js/fb18633e.b4caf1fa.js deleted file mode 100644 index 0decacf4..00000000 --- a/assets/js/fb18633e.b4caf1fa.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdasarpemrogramanrust=self.webpackChunkdasarpemrogramanrust||[]).push([[7310],{3905:(a,e,n)=>{n.d(e,{Zo:()=>m,kt:()=>k});var t=n(7294);function i(a,e,n){return e in a?Object.defineProperty(a,e,{value:n,enumerable:!0,configurable:!0,writable:!0}):a[e]=n,a}function r(a,e){var n=Object.keys(a);if(Object.getOwnPropertySymbols){var t=Object.getOwnPropertySymbols(a);e&&(t=t.filter((function(e){return Object.getOwnPropertyDescriptor(a,e).enumerable}))),n.push.apply(n,t)}return n}function l(a){for(var e=1;e=0||(i[n]=a[n]);return i}(a,e);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(a);for(t=0;t=0||Object.prototype.propertyIsEnumerable.call(a,n)&&(i[n]=a[n])}return i}var u=t.createContext({}),s=function(a){var e=t.useContext(u),n=e;return a&&(n="function"==typeof a?a(e):l(l({},e),a)),n},m=function(a){var e=s(a.components);return t.createElement(u.Provider,{value:e},a.children)},p={inlineCode:"code",wrapper:function(a){var e=a.children;return t.createElement(t.Fragment,{},e)}},d=t.forwardRef((function(a,e){var n=a.components,i=a.mdxType,r=a.originalType,u=a.parentName,m=o(a,["components","mdxType","originalType","parentName"]),d=s(n),k=i,g=d["".concat(u,".").concat(k)]||d[k]||p[k]||r;return n?t.createElement(g,l(l({ref:e},m),{},{components:n})):t.createElement(g,l({ref:e},m))}));function k(a,e){var n=arguments,i=e&&e.mdxType;if("string"==typeof a||i){var r=n.length,l=new Array(r);l[0]=d;var o={};for(var u in e)hasOwnProperty.call(e,u)&&(o[u]=e[u]);o.originalType=a,o.mdxType="string"==typeof a?a:i,l[1]=o;for(var s=2;s{n.r(e),n.d(e,{assets:()=>u,contentTitle:()=>l,default:()=>p,frontMatter:()=>r,metadata:()=>o,toc:()=>s});var t=n(7462),i=(n(7294),n(3905));const r={sidebar_position:35,title:"A.35. Borrowing",sidebar_label:"A.35. Borrowing"},l=void 0,o={unversionedId:"basic/borrowing",id:"basic/borrowing",title:"A.35. Borrowing",description:"Pada chapter ini kita akan belajar tentang apa itu borrowing dalam Rust programming.",source:"@site/docs/basic/borrowing.md",sourceDirName:"basic",slug:"/basic/borrowing",permalink:"/basic/borrowing",draft:!1,tags:[],version:"current",sidebarPosition:35,frontMatter:{sidebar_position:35,title:"A.35. Borrowing",sidebar_label:"A.35. Borrowing"},sidebar:"tutorialSidebar",previous:{title:"A.34. Ownership",permalink:"/basic/ownership"},next:{title:"A.36. Traits",permalink:"/basic/traits"}},u={},s=[{value:"A.35.1. Konsep borrowing",id:"a351-konsep-borrowing",level:2},{value:"A.35.2. Rust Borrow Checker",id:"a352-rust-borrow-checker",level:2},{value:"A.35.3. Aturan borrowing",id:"a353-aturan-borrowing",level:2},{value:"A.35.4. Borrowing mutable/immutable reference",id:"a354-borrowing-mutableimmutable-reference",level:2},{value:"\u25c9 Contoh ke-1",id:"-contoh-ke-1",level:3},{value:"\u25c9 Contoh ke-2",id:"-contoh-ke-2",level:3},{value:"\u25c9 Contoh ke-3",id:"-contoh-ke-3",level:3},{value:"A.35.5. Borrowing valid/invalid reference",id:"a355-borrowing-validinvalid-reference",level:2},{value:"A.35.6. Borrowing pada block",id:"a356-borrowing-pada-block",level:2},{value:"\u25c9 Method contains milik String",id:"-method-contains-milik-string",level:3},{value:"A.35.7. Owner dan borrower data literal",id:"a357-owner-dan-borrower-data-literal",level:2},{value:"A.35.8. Borrowing pada macro println",id:"a358-borrowing-pada-macro-println",level:2},{value:"Catatan chapter \ud83d\udcd1",id:"catatan-chapter-",level:2},{value:"\u25c9 Source code praktik",id:"-source-code-praktik",level:3},{value:"\u25c9 Referensi",id:"-referensi",level:3}],m={toc:s};function p(a){let{components:e,...r}=a;return(0,i.kt)("wrapper",(0,t.Z)({},m,r,{components:e,mdxType:"MDXLayout"}),(0,i.kt)("p",null,"Pada chapter ini kita akan belajar tentang apa itu borrowing dalam Rust programming."),(0,i.kt)("p",null,"Sebelum masuk ke pembelajaran, penulis anjurkan untuk paham terlebih dahulu tentang ",(0,i.kt)("a",{parentName:"p",href:"/basic/basic-memory-management"},"Basic Memory Management"),", ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),", dan juga konsep ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership"},"Ownership"),". Ketiga topik tersebut dipelajari pada chapter sebelum ini."),(0,i.kt)("h2",{id:"a351-konsep-borrowing"},"A.35.1. Konsep borrowing"),(0,i.kt)("p",null,"Pada chapter sebelumnya kita telah belajar bahwa tipe data yang mengadopsi ",(0,i.kt)("em",{parentName:"p"},"move semantics")," (seperti ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),"), ketika digunakan pada operasi assignment maka owner-nya berpindah. Variabel yang sebelumnya adalah owner, setelah proses assignment menjadi invalid, karena owner data tersebut telah berpindah ke variabel baru. Efeknya, semua operasi yang dilakukan pada variabel owner sebelumnya menghasilkan error."),(0,i.kt)("p",null,"Sekarang perhatikan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let msg_1 = String::from("hello");\nlet msg_2 = msg_1;\n\nprintln!("{:?}", msg_2);\nprintln!("{:?}", msg_1);\n')),(0,i.kt)("p",null,"Pada contoh di atas, statement print ",(0,i.kt)("inlineCode",{parentName:"p"},"msg_1")," menghasilkan error karena variabel tersebut telah invalid setelah statement ",(0,i.kt)("inlineCode",{parentName:"p"},"let msg_2 = msg_1"),"."),(0,i.kt)("p",null,"Programmer harus ekstra hati-hati dan bijak dalam pengelolaan owner data, agar memory ter-manage dengan baik. Tapi kalau dipikir-pikir, repot juga kalau owner harus dilempar-lempar setiap selesai digunakan agar tidak error."),(0,i.kt)("p",null,"Salah satu cara yang bisa digunakan agar tidak terlalu repot adalah dengan menerapkan cloning (seperti yang sudah dibahas pada chapter sebelumnya). Namun ini bukan opsi yang baik karena boros memory, sedangkan di sisi lain programmer dianjurkan untuk efisien dalam penggunaan memory."),(0,i.kt)("p",null,"Solusi yang paling pas adalah dengan menerapkan ",(0,i.kt)("strong",{parentName:"p"},"borrowing"),". Borrowing artinya adalah meminjam. Pada konteks Rust programming, borrowing berarti meminjam data milik owner, dipinjam agar bisa diakses tanpa perlu memindah owner-nya. Kemudian setelah peminjaman selesai, data dikembalikan."),(0,i.kt)("p",null,"Cara meminjam data di Rust sangat mudah, yaitu:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Untuk borrowing dengan level akses immutable/read-only, gunakan operator reference ",(0,i.kt)("inlineCode",{parentName:"p"},"&")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let msg_3 = String::from("hello rust");\nlet msg_4 = &msg_3; // <----- borrow operation\n\nprintln!("{:?}", msg_4); // output => hello rust\nprintln!("{:?}", msg_3); // output => hello rust\n'))),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("p",{parentName:"li"},"Untuk borrowing dengan level akses mutable, gunakan operator reference ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")),(0,i.kt)("pre",{parentName:"li"},(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut msg_3 = String::from("hello");\nlet msg_4 = &mut msg_3; // <----- mutable borrow operation\n\n*msg_4 = String::from("hello rust");\n\nprintln!("{:?}", msg_4); // output => hello rust\nprintln!("{:?}", msg_3); // output => hello rust\n')))),(0,i.kt)("p",null,"Di Rust, semua statement reference (baik mutable ataupun immutable) adalah operasi ",(0,i.kt)("em",{parentName:"p"},"borrowing"),". Yang terjadi pada statement reference adalah data milik owner dipinjam dalam bentuk pointer. Pointer itu sendiri merupakan alamat memory yang mengarah ke data sebenarnya (milik owner)."),(0,i.kt)("p",null,"Dari contoh di atas, bisa ditarik kesimpulan bahwa data string ",(0,i.kt)("inlineCode",{parentName:"p"},"hello rust")," memiliki dua reference:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Yang pertama adalah owner data, yaitu variabel ",(0,i.kt)("inlineCode",{parentName:"li"},"msg_3")),(0,i.kt)("li",{parentName:"ul"},"Yang kedua adalah peminjam data, yaitu variabel pointer ",(0,i.kt)("inlineCode",{parentName:"li"},"msg_4"))),(0,i.kt)("h2",{id:"a352-rust-borrow-checker"},"A.35.2. Rust Borrow Checker"),(0,i.kt)("p",null,"Rust compiler memiliki 1 bagian bernama ",(0,i.kt)("strong",{parentName:"p"},"borrow checker"),", tugasnya untuk melakukan pengecekan pada source code apakah ada kode yang berhubungan dengan ownership dan borrowing, dan apakah kode tersebut mengikuti aturan borrowing yang sudah ditetapkan oleh Rust, atau tidak. Jika ada yang menyalahi aturan, maka borrow checker memunculkan error."),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Borrow checker sangat galak, pastikan kode yang ditulis mengikuti aturan yang berlaku di pemrograman Rust.")),(0,i.kt)("h2",{id:"a353-aturan-borrowing"},"A.35.3. Aturan borrowing"),(0,i.kt)("p",null,"Aturan borrowing atau reference sempat disinggung pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/pointer-references"},"Pointer & References"),", yang kurang lebih adalah:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},"Dalam waktu yang sama, hanya boleh ada satu mutable reference atau banyak immutable reference (keduanya tidak bisa bersamaan, harus salah satu)."),(0,i.kt)("li",{parentName:"ul"},"Reference harus selalu valid.")),(0,i.kt)("p",null,"Dua aturan tersebut wajib dipatuhi, jika tidak maka pasti muncul error."),(0,i.kt)("p",null,"Ok, selanjutnya mari kita test aturan tersebut."),(0,i.kt)("h2",{id:"a354-borrowing-mutableimmutable-reference"},"A.35.4. Borrowing mutable/immutable reference"),(0,i.kt)("h3",{id:"-contoh-ke-1"},"\u25c9 Contoh ke-1"),(0,i.kt)("p",null,"Pada contoh berikut kita simulasikan 1 data memiliki banyak immutable reference. Jika mengacu ke aturan di atas, maka hal seperti ini adalah boleh dan tidak menghasilkan error."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let msg_5 = String::from("hello rust");\n\nlet msg_6 = &msg_5;\nlet msg_7 = &msg_5;\nlet msg_8 = &msg_5;\n\nprintln!("{:?} {:?} {:?}", msg_6, msg_7, msg_8);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(9470).Z,width:"470",height:"301"})),(0,i.kt)("h3",{id:"-contoh-ke-2"},"\u25c9 Contoh ke-2"),(0,i.kt)("p",null,"Pada contoh ke-2 ini, kita coba simulasikan 1 data memiliki 1 mutable reference. Harusnya tidak muncul error karena diperbolehkan di aturan yang tertulis."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut msg_9 = String::from("hello rust");\n\nlet msg_10 = &mut msg_9;\n\nprintln!("{:?}", msg_10);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(5982).Z,width:"497",height:"272"})),(0,i.kt)("p",null,"Bagaimana jika ada lebih dari 1 mutable reference? mari kita test."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut msg_9 = String::from("hello rust");\n\nlet msg_10 = &mut msg_9;\nlet msg_11 = &mut msg_9;\n\nprintln!("{:?} {:?}", msg_10, msg_11);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(8301).Z,width:"652",height:"613"})),(0,i.kt)("p",null,"Hasilnya error. Hal seperti ini tidak diperbolehkan. Sebuah data tidak boleh memiliki lebih dari 1 mutable reference."),(0,i.kt)("h3",{id:"-contoh-ke-3"},"\u25c9 Contoh ke-3"),(0,i.kt)("p",null,"Sekarang mari kita coba test lagi aturan di atas dengan skenario: dalam waktu yang sama, 1 data memiliki 1 mutable reference dan 1 immutable reference. Hasilnya seperti apa, (jika mengacu ke aturan, hal seperti ini adalah tidak diperbolehkan)."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let mut msg_12 = String::from("hello rust");\n\nlet msg_13 = &msg_12;\nlet msg_14 = &mut msg_12;\n\nprintln!("{:?} {:?}", msg_13, msg_14);\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(6389).Z,width:"652",height:"486"})),(0,i.kt)("p",null,"Hasilnya error, kenapa? karena memang tidak boleh."),(0,i.kt)("p",null,"Jadi sampai sini cukup jelas ya. Sebuah data dalam waktu yang sama hanya diperbolehkan memiliki satu atau lebih immutable reference, atau hanya 1 mutable reference, dan keduanya tidak bisa bersamaan dalam satu waktu (harus pilih salah satu)."),(0,i.kt)("h2",{id:"a355-borrowing-validinvalid-reference"},"A.35.5. Borrowing valid/invalid reference"),(0,i.kt)("p",null,"Kita telah belajar tentang ",(0,i.kt)("a",{parentName:"p",href:"/basic/ownership#a333-variable-scope"},"valid/invalid variable dan juga variable scope")," pada chapter sebelumnya. Sekarang kita akan coba gabungkan dua aspek tersebut dengan borrowing."),(0,i.kt)("p",null,"Silakan praktikkan kode berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut fact_one = String::from("Arthas is the true lich king");\n\n change_value(&mut fact_one);\n\n let fact_two = &mut fact_one;\n println!("{:?}", fact_two);\n}\n\nfn change_value(txt: &mut String) {\n *txt = String::from("Bolvar is better lich king");\n}\n')),(0,i.kt)("p",null,"Pada kode di atas, ada satu buah mutable string bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one"),". String tersebut dipinjam menggunakan ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut")," (yang berarti adalah mutable borrow) sebagai argumen pemanggilan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value")," yang isinya kurang lebih adalah perubahan isi data string pada variabel pointer."),(0,i.kt)("p",null,"Kemudian, ada operasi peminjaman lagi (yang juga mutable borrow) dari variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one")," ke ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one"),"."),(0,i.kt)("p",null,"Ketika program di-run, hasilnya sukses. Kok bisa? padahal jelas di aturan tertulis bahwa dalam waktu yang sama tidak boleh ada lebih dari satu mutable borrow."),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(538).Z,width:"449",height:"88"})),(0,i.kt)("p",null,"Kata ",(0,i.kt)("strong",{parentName:"p"},"dalam waktu yang sama")," di sini adalah yang penting untuk dipahami. Arti ",(0,i.kt)("em",{parentName:"p"},"dalam waktu yang sama")," adalah dalam 1 scope yang sama. Jika ada 2 scope, maka itu sudah bukan dalam waktu yang sama lagi."),(0,i.kt)("p",null,"Pada contoh di atas, statement ",(0,i.kt)("inlineCode",{parentName:"p"},"&mut fact_one")," terjadi pada block fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),", lebih tepatnya pada parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"txt")," fungsi tersebut."),(0,i.kt)("p",null,"Kemudian setelah eksekusi fungsi tersebut selesai, yang terjadi adalah: reference yang tadinya dipinjam, sekarang dikembalikan. Tepat setelah eksekusi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),", state data ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one")," adalah tidak dipinjam siapapun. Tadinya memang ada yang meminjam (yaitu parameter ",(0,i.kt)("inlineCode",{parentName:"p"},"txt")," di fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),"), tapi setelah eksekusi fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value")," selesai, data dikembalikan lagi ke owner."),(0,i.kt)("p",null,"Proses dealokasi pada variable scope, jika terjadi pada variable reference maka yang sebenarnya terjadi adalah pengembalian data hasil operasi borrow ke pemilik aslinya."),(0,i.kt)("p",null,"Kemudian, ada operasi pinjam lagi, yaitu ",(0,i.kt)("inlineCode",{parentName:"p"},"let fact_two = &mut fact_one"),". Statement borrow ini tidak menghasilkan error karena memang kondisi data ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one")," sudah tidak ada yang meminjam. Bisa dibilang statement peminjaman ke-2 ini tidak terjadi dalam waktu yang sama dengan statement peminjaman pertama (pemanggilan fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),")."),(0,i.kt)("p",null,"Pada fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),", variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"txt")," adalah valid saat fungsi dijalankan. Setelah pemanggilan fungsi selesai, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"txt")," di-dealokasi. Namun karena variabel tersebut bukanlah owner, melainkan hanya borrower yang meminjam data dari owner ",(0,i.kt)("inlineCode",{parentName:"p"},"fact_one"),", maka yang terjadi adalah: data yang dipinjam sekarang dikembalikan lagi ke pemilik aslinya."),(0,i.kt)("p",null,"Ok sampai sini semoga cukup jelas ya. Jika perlu silakan ulang-ulang penjelasan di atas."),(0,i.kt)("h2",{id:"a356-borrowing-pada-block"},"A.35.6. Borrowing pada block"),(0,i.kt)("p",null,"Kita sudah cukup paham kapan waktu terjadinya alokasi alamat memory dan juga kapan proses dealokasi terjadi, yaitu ketika variabel ",(0,i.kt)("strong",{parentName:"p"},"into scope")," dan ketika variabel ",(0,i.kt)("strong",{parentName:"p"},"out of scope"),"."),(0,i.kt)("p",null,"Scope di sini adalah variable scope, yang maksudnya adalah block scope, bisa berupa block expression, fungsi, block seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"if"),", dan juga jenis block lainnya."),(0,i.kt)("p",null,"Ok, sekarang mari kita praktikkan beberapa jenis block untuk isolasi operasi borrowing. Dengan ini maka klausul ",(0,i.kt)("strong",{parentName:"p"},"dalam satu waktu")," akan terpenuhi."),(0,i.kt)("p",null,"Silakan modifikasi program sebelumnya menjadi seperti berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'fn main() {\n let mut fact_one = String::from("Arthas is the true lich king");\n println!("{:?}", fact_one);\n\n change_value(&mut fact_one);\n println!("{:?}", fact_one);\n\n {\n let fact_two = &mut fact_one;\n *fact_two = String::from("There must always be a lich king");\n println!("{:?}", fact_one);\n }\n\n if fact_one.contains("lich king") {\n let fact_three = &mut fact_one;\n *fact_three = String::from("Who is the real jailer?");\n println!("{:?}", fact_one);\n }\n\n for _ in 0..1 {\n let fact_four = &mut fact_one;\n *fact_four = String::from("Is it Zovaal or Primus?");\n println!("{:?}", fact_one);\n }\n}\n\nfn change_value(txt: &mut String) {\n *txt = String::from("Bolvar is better lich king");\n}\n')),(0,i.kt)("p",null,(0,i.kt)("img",{alt:"Borrowing",src:n(2751).Z,width:"459",height:"175"})),(0,i.kt)("p",null,"Bisa dilihat, hasilnya tidak error, meskipun terjadi beberapa kali operasi mutable borrow, tapi karena kesemuanya terjadi di waktu yang berbeda (di scope yang berbeda) maka tidak error."),(0,i.kt)("p",null,"Pada kode di atas, mutable borrow terjadi di banyak tempat, yaitu di block fungsi ",(0,i.kt)("inlineCode",{parentName:"p"},"change_value"),", block expression, block seleksi kondisi ",(0,i.kt)("inlineCode",{parentName:"p"},"if"),", dan juga block perulangan ",(0,i.kt)("inlineCode",{parentName:"p"},"for"),". Variabel pointer yang menampung data pinjaman akan valid di masing-masing block, kemudian menjadi invalid setelah eksekusi block selesai, dan yang terjadi setelah itu adalah pengembalian data yang telah dipinjam."),(0,i.kt)("h3",{id:"-method-contains-milik-string"},"\u25c9 Method ",(0,i.kt)("inlineCode",{parentName:"h3"},"contains")," milik ",(0,i.kt)("inlineCode",{parentName:"h3"},"String")),(0,i.kt)("p",null,"Pada contoh di atas kita menerapkan method baru bernama ",(0,i.kt)("inlineCode",{parentName:"p"},"contains"),". Method ini tersedia untuk tipe data ",(0,i.kt)("inlineCode",{parentName:"p"},"String"),", gunanya adalah untuk mengecek apakah string memiliki substring ",(0,i.kt)("inlineCode",{parentName:"p"},"x"),", dimana ",(0,i.kt)("inlineCode",{parentName:"p"},"x")," adalah argumen pemanggilan method. Method ini mengembalikan nilai ",(0,i.kt)("inlineCode",{parentName:"p"},"bool"),"."),(0,i.kt)("p",null,"Contoh penerapan method ",(0,i.kt)("inlineCode",{parentName:"p"},"contains"),":"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let fact = String::from("There must always be a lich king");\nprintln!("{:?}", fact.contains("lich king")); // output => true\nprintln!("{:?}", fact.contains("bolvar")); // output => false\n')),(0,i.kt)("p",null,"O iya, pengecekan string-nya adalah case sensitive ya."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let fact = String::from("There must always be a lich king");\nprintln!("{:?}", fact.contains("lich king")); // output => true\nprintln!("{:?}", fact.contains("Lich King")); // output => false\n')),(0,i.kt)("blockquote",null,(0,i.kt)("p",{parentName:"blockquote"},"Lebih jelasnya mengenai tipe data string dan method yang tersedia pada tipe tersebut akan dibahas terpisah pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/string-slice-vs-string-literal"},"Tipe Data \u279c String Custom Type vs &str"),".")),(0,i.kt)("h2",{id:"a357-owner-dan-borrower-data-literal"},"A.35.7. Owner dan borrower data literal"),(0,i.kt)("p",null,"Perhatikan statement berikut:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let number = 12;\nlet a = &number;\n\nlet text = String::from("hello");\nlet b = &text;\n')),(0,i.kt)("p",null,"Pada kode di atas, variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"number")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"text")," adalah ",(0,i.kt)("em",{parentName:"p"},"owner")," data masing-masing. Sedangkan variabel ",(0,i.kt)("inlineCode",{parentName:"p"},"a")," dan ",(0,i.kt)("inlineCode",{parentName:"p"},"b")," adalah ",(0,i.kt)("em",{parentName:"p"},"borrower")," atau peminjam data (yang lebih jelasnya akan dibahas pada chapter ). Sampai sini penulis rasa cukup jelas."),(0,i.kt)("p",null,"Selanjutnya, bagaimana dengan contoh ini:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let c = &24;\nlet d = &false;\nlet e = &String::from("rust");\n')),(0,i.kt)("p",null,"Ketiga variabel, semuanya adalah ",(0,i.kt)("em",{parentName:"p"},"borrower")," atau peminjam data. Lalu siapa ",(0,i.kt)("em",{parentName:"p"},"owner"),"-nya? jawabannya adalah tidak ada, atau silakan juga simpulkan bawah data tersebut owner-nya adalah program."),(0,i.kt)("p",null,"Yang lebih penting untuk diurusi pada contoh di atas bukan siapa owner-nya, melainkan bagaimana caranya agar data pinjaman tersebut tidak di-dealokasi saat block kode selesai."),(0,i.kt)("p",null,"Untuk sekarang sampai sini dulu. Kita akan bahas topik ini lebih mendetail pada chapter ",(0,i.kt)("a",{parentName:"p",href:"/basic/static"},"Static item")," dan ",(0,i.kt)("a",{parentName:"p",href:"/basic/lifetime"},"Lifetime"),"."),(0,i.kt)("h2",{id:"a358-borrowing-pada-macro-println"},"A.35.8. Borrowing pada macro ",(0,i.kt)("inlineCode",{parentName:"h2"},"println")),(0,i.kt)("p",null,"Pada pemanggilan macro println untuk menampilkan data yang mengadopsi ",(0,i.kt)("em",{parentName:"p"},"move semantics"),", operasi borrowing tidak perlu dilakukan sewaktu pengisian argument. Karena macro tersebut secara cerdas akan melakukan opearsi peminjaman tanpa memindah owner-nya. Contoh:"),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str1 = String::from("luwe");\nprintln!("{:?}", str1);\n')),(0,i.kt)("p",null,"... adalah ekuivalen dengan ..."),(0,i.kt)("pre",null,(0,i.kt)("code",{parentName:"pre",className:"language-rust"},'let str2 = String::from("ngelak");\nprintln!("{:?}", &str2);\n')),(0,i.kt)("p",null,"Lebih jelasnya mengenai macro dibahas pada chapter ",(0,i.kt)("a",{parentName:"p",href:"#/wip/macro"},"Macro"),"."),(0,i.kt)("hr",null),(0,i.kt)("h2",{id:"catatan-chapter-"},"Catatan chapter \ud83d\udcd1"),(0,i.kt)("h3",{id:"-source-code-praktik"},"\u25c9 Source code praktik"),(0,i.kt)("pre",null,(0,i.kt)("a",{href:"https://github.com/novalagung/dasarpemrogramanrust-example/tree/master/borrowing"},"github.com/novalagung/dasarpemrogramanrust-example/../borrowing")),(0,i.kt)("h3",{id:"-referensi"},"\u25c9 Referensi"),(0,i.kt)("p",null,"Beberapa referensi terkait chapter ini:"),(0,i.kt)("ul",null,(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://doc.rust-lang.org/book/ch04-02-references-and-borrowing.html"},"https://doc.rust-lang.org/book/ch04-02-references-and-borrowing.html")),(0,i.kt)("li",{parentName:"ul"},(0,i.kt)("a",{parentName:"li",href:"https://stackoverflow.com/questions/57225055/in-rust-can-you-own-a-string-literal"},"https://stackoverflow.com/questions/57225055/in-rust-can-you-own-a-string-literal"))))}p.isMDXComponent=!0},9470:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/borrowing-1-6abee77c6350f8a390a807adac852003.png"},5982:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/borrowing-2-cf66c8ef1c4b76c209f6d8c7d3339a3f.png"},8301:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/borrowing-3-55cc0f7a0b551f62c748a9bf0396a671.png"},6389:(a,e,n)=>{n.d(e,{Z:()=>t});const t=n.p+"assets/images/borrowing-4-c806f4e6787a069015fe56b40e97b3ca.png"},538:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""},2751:(a,e,n)=>{n.d(e,{Z:()=>t});const t=""}}]); \ No newline at end of file diff --git a/assets/js/runtime~main.36445933.js b/assets/js/runtime~main.36445933.js deleted file mode 100644 index 9e2cdc04..00000000 --- a/assets/js/runtime~main.36445933.js +++ /dev/null @@ -1 +0,0 @@ -(()=>{"use strict";var e,a,c,d,r,b={},t={};function f(e){var a=t[e];if(void 0!==a)return a.exports;var c=t[e]={exports:{}};return b[e].call(c.exports,c,c.exports,f),c.exports}f.m=b,e=[],f.O=(a,c,d,r)=>{if(!c){var b=1/0;for(i=0;i=r)&&Object.keys(f.O).every((e=>f.O[e](c[o])))?c.splice(o--,1):(t=!1,r0&&e[i-1][2]>r;i--)e[i]=e[i-1];e[i]=[c,d,r]},f.n=e=>{var a=e&&e.__esModule?()=>e.default:()=>e;return f.d(a,{a:a}),a},c=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,f.t=function(e,d){if(1&d&&(e=this(e)),8&d)return e;if("object"==typeof e&&e){if(4&d&&e.__esModule)return e;if(16&d&&"function"==typeof e.then)return e}var r=Object.create(null);f.r(r);var b={};a=a||[null,c({}),c([]),c(c)];for(var t=2&d&&e;"object"==typeof t&&!~a.indexOf(t);t=c(t))Object.getOwnPropertyNames(t).forEach((a=>b[a]=()=>e[a]));return b.default=()=>e,f.d(r,b),r},f.d=(e,a)=>{for(var c in a)f.o(a,c)&&!f.o(e,c)&&Object.defineProperty(e,c,{enumerable:!0,get:a[c]})},f.f={},f.e=e=>Promise.all(Object.keys(f.f).reduce(((a,c)=>(f.f[c](e,a),a)),[])),f.u=e=>"assets/js/"+({53:"935f2afb",274:"016a0787",620:"2bc2e36b",623:"9e6fe7cf",740:"f8d4e457",909:"bd9b9d52",1059:"7d03ccaf",1422:"62b99870",2089:"0ae704f7",2107:"8e991452",2359:"48e2ebd6",2399:"8ad68633",2706:"4c7e7ada",2870:"5f289fb7",2959:"68ec37cc",3007:"04da11da",3257:"82a9b669",3306:"164327b9",4197:"7a2568c4",4248:"8333c691",4274:"7390b81e",4452:"bc4edc5c",4622:"d8837387",4729:"c76cc2e3",4738:"496dcc30",4758:"68e1906b",4790:"b97d8256",4861:"db8ccf76",4943:"dabee7bd",5572:"be3f5c40",5673:"fef7d16c",5729:"4513d9f5",5751:"fd4f5a0c",5758:"71880105",5989:"2452d5f5",6161:"3b1dfe68",6345:"a2bea3ed",6388:"eb2a6637",6403:"475d163e",6477:"05cc1716",6531:"c01cce30",6691:"cb962766",6794:"b28a365a",6971:"c377a04b",7310:"fb18633e",7386:"3aa512bf",7505:"8d193b98",7539:"e7bd9b37",7720:"bdfc7662",7742:"87499f15",7865:"9a3629a6",7918:"17896441",7920:"1a4e3797",7949:"0e6b6dd1",8051:"dadaca00",8308:"4232c3ed",8358:"7794a932",8469:"93422b31",8732:"e4904969",8784:"d47a3664",8794:"af6aab29",9157:"c526a898",9323:"8333ea4d",9514:"1be78505",9831:"5925ce9f",9886:"8b55bf27"}[e]||e)+"."+{53:"13143cb2",205:"e9ca4e1a",274:"0ab1d3ab",620:"876e08c0",623:"c549f588",740:"88f7a231",909:"8d109af6",1059:"46ca6fcc",1422:"4801cdae",2089:"5940e697",2107:"9f077664",2359:"0b6ce575",2399:"0a69a909",2706:"f4643a6f",2870:"e0fc9497",2959:"d3961cb0",3007:"b3ba07e3",3257:"bb49d726",3306:"ce557677",4197:"cb80d87b",4248:"54569977",4274:"686f428e",4452:"f0b380ec",4622:"e6bc9bd5",4729:"c1ae2c67",4738:"2c33b3bd",4758:"f87d9410",4790:"7eff2a81",4861:"f925a746",4943:"9e39dc43",4972:"ea87a18a",5572:"4baec74d",5673:"7198ddd5",5729:"6ce8fc47",5751:"7de9809a",5758:"a92b59bc",5989:"c8d9088c",6161:"84da4a4d",6345:"c14ec614",6388:"01009699",6403:"4cbfe654",6477:"272262df",6531:"1810b6cb",6691:"fe11400b",6780:"3e547342",6794:"28a71aca",6945:"19e06702",6971:"753abc35",7310:"b4caf1fa",7386:"a1c7a02b",7505:"c7232d69",7539:"38bdc3e5",7720:"e5ac5751",7742:"679ef493",7865:"fcabd2cf",7918:"062de7c5",7920:"615c0dbb",7949:"a5062737",8051:"d3047a10",8308:"0eade53f",8358:"0807e2e7",8469:"7d2d5eae",8732:"2c9c10d1",8784:"2e54129a",8794:"fff2a138",8894:"7d968ee6",9157:"e25aff7f",9323:"d3688653",9514:"fa7071eb",9831:"d25861ea",9886:"a01d27dd"}[e]+".js",f.miniCssF=e=>{},f.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),f.o=(e,a)=>Object.prototype.hasOwnProperty.call(e,a),d={},r="dasarpemrogramanrust:",f.l=(e,a,c,b)=>{if(d[e])d[e].push(a);else{var t,o;if(void 0!==c)for(var n=document.getElementsByTagName("script"),i=0;i{t.onerror=t.onload=null,clearTimeout(s);var r=d[e];if(delete d[e],t.parentNode&&t.parentNode.removeChild(t),r&&r.forEach((e=>e(c))),a)return a(c)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:t}),12e4);t.onerror=l.bind(null,t.onerror),t.onload=l.bind(null,t.onload),o&&document.head.appendChild(t)}},f.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},f.p="/",f.gca=function(e){return e={17896441:"7918",71880105:"5758","935f2afb":"53","016a0787":"274","2bc2e36b":"620","9e6fe7cf":"623",f8d4e457:"740",bd9b9d52:"909","7d03ccaf":"1059","62b99870":"1422","0ae704f7":"2089","8e991452":"2107","48e2ebd6":"2359","8ad68633":"2399","4c7e7ada":"2706","5f289fb7":"2870","68ec37cc":"2959","04da11da":"3007","82a9b669":"3257","164327b9":"3306","7a2568c4":"4197","8333c691":"4248","7390b81e":"4274",bc4edc5c:"4452",d8837387:"4622",c76cc2e3:"4729","496dcc30":"4738","68e1906b":"4758",b97d8256:"4790",db8ccf76:"4861",dabee7bd:"4943",be3f5c40:"5572",fef7d16c:"5673","4513d9f5":"5729",fd4f5a0c:"5751","2452d5f5":"5989","3b1dfe68":"6161",a2bea3ed:"6345",eb2a6637:"6388","475d163e":"6403","05cc1716":"6477",c01cce30:"6531",cb962766:"6691",b28a365a:"6794",c377a04b:"6971",fb18633e:"7310","3aa512bf":"7386","8d193b98":"7505",e7bd9b37:"7539",bdfc7662:"7720","87499f15":"7742","9a3629a6":"7865","1a4e3797":"7920","0e6b6dd1":"7949",dadaca00:"8051","4232c3ed":"8308","7794a932":"8358","93422b31":"8469",e4904969:"8732",d47a3664:"8784",af6aab29:"8794",c526a898:"9157","8333ea4d":"9323","1be78505":"9514","5925ce9f":"9831","8b55bf27":"9886"}[e]||e,f.p+f.u(e)},(()=>{var e={1303:0,532:0};f.f.j=(a,c)=>{var d=f.o(e,a)?e[a]:void 0;if(0!==d)if(d)c.push(d[2]);else if(/^(1303|532)$/.test(a))e[a]=0;else{var r=new Promise(((c,r)=>d=e[a]=[c,r]));c.push(d[2]=r);var b=f.p+f.u(a),t=new Error;f.l(b,(c=>{if(f.o(e,a)&&(0!==(d=e[a])&&(e[a]=void 0),d)){var r=c&&("load"===c.type?"missing":c.type),b=c&&c.target&&c.target.src;t.message="Loading chunk "+a+" failed.\n("+r+": "+b+")",t.name="ChunkLoadError",t.type=r,t.request=b,d[1](t)}}),"chunk-"+a,a)}},f.O.j=a=>0===e[a];var a=(a,c)=>{var d,r,b=c[0],t=c[1],o=c[2],n=0;if(b.some((a=>0!==e[a]))){for(d in t)f.o(t,d)&&(f.m[d]=t[d]);if(o)var i=o(f)}for(a&&a(c);n{"use strict";var e,a,c,f,d,r={},t={};function b(e){var a=t[e];if(void 0!==a)return a.exports;var c=t[e]={exports:{}};return r[e].call(c.exports,c,c.exports,b),c.exports}b.m=r,e=[],b.O=(a,c,f,d)=>{if(!c){var r=1/0;for(i=0;i=d)&&Object.keys(b.O).every((e=>b.O[e](c[o])))?c.splice(o--,1):(t=!1,d0&&e[i-1][2]>d;i--)e[i]=e[i-1];e[i]=[c,f,d]},b.n=e=>{var a=e&&e.__esModule?()=>e.default:()=>e;return b.d(a,{a:a}),a},c=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,b.t=function(e,f){if(1&f&&(e=this(e)),8&f)return e;if("object"==typeof e&&e){if(4&f&&e.__esModule)return e;if(16&f&&"function"==typeof e.then)return e}var d=Object.create(null);b.r(d);var r={};a=a||[null,c({}),c([]),c(c)];for(var t=2&f&&e;"object"==typeof t&&!~a.indexOf(t);t=c(t))Object.getOwnPropertyNames(t).forEach((a=>r[a]=()=>e[a]));return r.default=()=>e,b.d(d,r),d},b.d=(e,a)=>{for(var c in a)b.o(a,c)&&!b.o(e,c)&&Object.defineProperty(e,c,{enumerable:!0,get:a[c]})},b.f={},b.e=e=>Promise.all(Object.keys(b.f).reduce(((a,c)=>(b.f[c](e,a),a)),[])),b.u=e=>"assets/js/"+({53:"935f2afb",274:"016a0787",620:"2bc2e36b",623:"9e6fe7cf",740:"f8d4e457",909:"bd9b9d52",1059:"7d03ccaf",1422:"62b99870",2089:"0ae704f7",2107:"8e991452",2359:"48e2ebd6",2399:"8ad68633",2706:"4c7e7ada",2870:"5f289fb7",2959:"68ec37cc",3007:"04da11da",3257:"82a9b669",3306:"164327b9",4197:"7a2568c4",4248:"8333c691",4274:"7390b81e",4452:"bc4edc5c",4622:"d8837387",4729:"c76cc2e3",4738:"496dcc30",4758:"68e1906b",4790:"b97d8256",4861:"db8ccf76",4943:"dabee7bd",5572:"be3f5c40",5673:"fef7d16c",5729:"4513d9f5",5751:"fd4f5a0c",5758:"71880105",5989:"2452d5f5",6161:"3b1dfe68",6345:"a2bea3ed",6388:"eb2a6637",6403:"475d163e",6477:"05cc1716",6531:"c01cce30",6691:"cb962766",6794:"b28a365a",6971:"c377a04b",7310:"fb18633e",7386:"3aa512bf",7505:"8d193b98",7539:"e7bd9b37",7720:"bdfc7662",7742:"87499f15",7865:"9a3629a6",7918:"17896441",7920:"1a4e3797",7949:"0e6b6dd1",8051:"dadaca00",8308:"4232c3ed",8358:"7794a932",8469:"93422b31",8732:"e4904969",8784:"d47a3664",8794:"af6aab29",9157:"c526a898",9323:"8333ea4d",9514:"1be78505",9831:"5925ce9f",9886:"8b55bf27"}[e]||e)+"."+{53:"13143cb2",205:"e9ca4e1a",274:"ddd4552f",620:"876e08c0",623:"c549f588",740:"ba1644a0",909:"c2ec587a",1059:"a18e440e",1422:"eac41936",2089:"12a11a9d",2107:"9f077664",2359:"0b6ce575",2399:"0a69a909",2706:"9f072125",2870:"c8228f07",2959:"d3961cb0",3007:"d61e5604",3257:"402fdaf8",3306:"ce557677",4197:"cb80d87b",4248:"d59a9952",4274:"18c2ce22",4452:"30092a0e",4622:"31f2a62c",4729:"978efe66",4738:"2c33b3bd",4758:"57a9a7b8",4790:"7eff2a81",4861:"0bea83f9",4943:"9e39dc43",4972:"ea87a18a",5572:"3675591d",5673:"7198ddd5",5729:"f78b137f",5751:"7de9809a",5758:"761a2054",5989:"c8d9088c",6161:"a3e9cea9",6345:"53e03a1b",6388:"570a9def",6403:"60520781",6477:"bb8cde02",6531:"1810b6cb",6691:"2d69a424",6780:"3e547342",6794:"28a71aca",6945:"19e06702",6971:"753abc35",7310:"5c83c6b6",7386:"35e7bf91",7505:"9b1c2896",7539:"6bffc9e2",7720:"5ed547e8",7742:"3255091a",7865:"035a9d13",7918:"062de7c5",7920:"615c0dbb",7949:"a5062737",8051:"64b4e275",8308:"32b73572",8358:"f8711f2e",8469:"2fa67f2b",8732:"d2252a8b",8784:"2e54129a",8794:"62e5b74a",8894:"7d968ee6",9157:"18163a59",9323:"79e6ada8",9514:"fa7071eb",9831:"9da92efc",9886:"c5401f96"}[e]+".js",b.miniCssF=e=>{},b.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),b.o=(e,a)=>Object.prototype.hasOwnProperty.call(e,a),f={},d="dasarpemrogramanrust:",b.l=(e,a,c,r)=>{if(f[e])f[e].push(a);else{var t,o;if(void 0!==c)for(var n=document.getElementsByTagName("script"),i=0;i{t.onerror=t.onload=null,clearTimeout(s);var d=f[e];if(delete f[e],t.parentNode&&t.parentNode.removeChild(t),d&&d.forEach((e=>e(c))),a)return a(c)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:t}),12e4);t.onerror=l.bind(null,t.onerror),t.onload=l.bind(null,t.onload),o&&document.head.appendChild(t)}},b.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},b.p="/",b.gca=function(e){return e={17896441:"7918",71880105:"5758","935f2afb":"53","016a0787":"274","2bc2e36b":"620","9e6fe7cf":"623",f8d4e457:"740",bd9b9d52:"909","7d03ccaf":"1059","62b99870":"1422","0ae704f7":"2089","8e991452":"2107","48e2ebd6":"2359","8ad68633":"2399","4c7e7ada":"2706","5f289fb7":"2870","68ec37cc":"2959","04da11da":"3007","82a9b669":"3257","164327b9":"3306","7a2568c4":"4197","8333c691":"4248","7390b81e":"4274",bc4edc5c:"4452",d8837387:"4622",c76cc2e3:"4729","496dcc30":"4738","68e1906b":"4758",b97d8256:"4790",db8ccf76:"4861",dabee7bd:"4943",be3f5c40:"5572",fef7d16c:"5673","4513d9f5":"5729",fd4f5a0c:"5751","2452d5f5":"5989","3b1dfe68":"6161",a2bea3ed:"6345",eb2a6637:"6388","475d163e":"6403","05cc1716":"6477",c01cce30:"6531",cb962766:"6691",b28a365a:"6794",c377a04b:"6971",fb18633e:"7310","3aa512bf":"7386","8d193b98":"7505",e7bd9b37:"7539",bdfc7662:"7720","87499f15":"7742","9a3629a6":"7865","1a4e3797":"7920","0e6b6dd1":"7949",dadaca00:"8051","4232c3ed":"8308","7794a932":"8358","93422b31":"8469",e4904969:"8732",d47a3664:"8784",af6aab29:"8794",c526a898:"9157","8333ea4d":"9323","1be78505":"9514","5925ce9f":"9831","8b55bf27":"9886"}[e]||e,b.p+b.u(e)},(()=>{var e={1303:0,532:0};b.f.j=(a,c)=>{var f=b.o(e,a)?e[a]:void 0;if(0!==f)if(f)c.push(f[2]);else if(/^(1303|532)$/.test(a))e[a]=0;else{var d=new Promise(((c,d)=>f=e[a]=[c,d]));c.push(f[2]=d);var r=b.p+b.u(a),t=new Error;b.l(r,(c=>{if(b.o(e,a)&&(0!==(f=e[a])&&(e[a]=void 0),f)){var d=c&&("load"===c.type?"missing":c.type),r=c&&c.target&&c.target.src;t.message="Loading chunk "+a+" failed.\n("+d+": "+r+")",t.name="ChunkLoadError",t.type=d,t.request=r,f[1](t)}}),"chunk-"+a,a)}},b.O.j=a=>0===e[a];var a=(a,c)=>{var f,d,r=c[0],t=c[1],o=c[2],n=0;if(r.some((a=>0!==e[a]))){for(f in t)b.o(t,f)&&(b.m[f]=t[f]);if(o)var i=o(b)}for(a&&a(c);n