Apa itu Linked List? [Bagian 1]
Informasi ada di sekitar kita.
Dalam dunia perangkat lunak, cara yang kita pilih untuk mengatur informasi adalah setengah dari pertempuran. Begini masalahnya: ada begitu banyak cara untuk memecahkan masalah. Dan ketika harus mengatur data kami, ada banyak alat yang dapat bekerja untuk pekerjaan itu. Triknya adalah mengetahui alat mana yang tepat untuk digunakan.
Terlepas dari bahasa apa kita mulai membuat kode, salah satu hal pertama yang kita temui adalah struktur data , yang merupakan cara berbeda untuk mengatur data kita; variabel , array , hash, dan objek adalah semua jenis struktur data. Tetapi ini masih hanya puncak gunung es dalam hal struktur data; masih banyak lagi, beberapa di antaranya mulai terdengar sangat rumit saat Anda semakin sering mendengarnya.
Salah satu hal rumit bagi saya selalu terkait dengan daftar . Saya telah mengetahui tentang daftar tertaut selama beberapa tahun sekarang, tetapi saya tidak pernah bisa mengingatnya secara langsung. Saya hanya benar-benar memikirkannya ketika saya mempersiapkan (atau terkadang, di tengah) wawancara teknis, dan seseorang bertanya kepada saya tentang mereka. Saya akan melakukan sedikit riset dan berpikir bahwa saya mengerti tentang apa mereka, tetapi setelah beberapa minggu, saya melupakan mereka lagi. Semuanya sangat tidak efisien, dan itu semua berasal dari fakta bahwa saya tahu mereka ada, tetapi saya tidak secara fundamental memahaminya ! Jadi, inilah saatnya untuk mengubahnya dan menjawab pertanyaan: sebenarnya apa itu daftar tertaut?
Struktur data linier
Jika kita benar-benar ingin memahami dasar-dasar daftar tertaut, penting bagi kita untuk membicarakan jenis struktur datanya.
Salah satu karakteristik dari daftar tertaut adalah bahwa daftar tersebut merupakan struktur data linier , yang berarti terdapat urutan dan urutan bagaimana daftar tersebut dibuat dan dilintasi. Kita dapat membayangkan struktur data linier seperti permainan hopscotch : untuk mencapai akhir daftar, kita harus memeriksa semua item dalam daftar secara berurutan, atau secara berurutan . Struktur linier, bagaimanapun, adalah kebalikan dari struktur non-linier. Dalam struktur data non-linier , item tidak harus disusun secara berurutan, yang berarti kita dapat melintasi struktur data secara non-sekuensial .
Kita mungkin tidak selalu menyadarinya, tetapi kita semua bekerja dengan struktur data linier dan non-linier setiap hari! Saat kami mengatur data kami ke dalam hash (terkadang disebut kamus ), kami menerapkan struktur data non-linier. Pohon dan grafik juga merupakan struktur data non-linier yang kita lintasi dengan cara yang berbeda, tetapi kita akan membicarakannya lebih dalam di akhir tahun.
Demikian pula, saat kami menggunakan array dalam kode kami, kami menerapkan struktur data linier! Akan sangat membantu jika kita menganggap array dan daftar tertaut serupa dalam cara kita mengurutkan data. Dalam kedua struktur ini, keteraturan itu penting . Tapi apa yang membuat array dan daftar tertaut berbeda?
Manajemen memori
Pembeda terbesar antara array dan daftar tertaut adalah cara mereka menggunakan memori di mesin kami. Kita yang bekerja dengan bahasa yang diketik secara dinamis seperti Ruby, JavaScript, atau Python tidak perlu memikirkan berapa banyak memori yang digunakan array ketika kita menulis kode kita setiap hari karena ada beberapa lapisan abstraksi yang berakhir. dengan kami tidak perlu khawatir tentang alokasi memori sama sekali.
Tetapi itu tidak berarti bahwa alokasi memori tidak terjadi! Abstraksi bukanlah sihir, itu hanya kesederhanaan menyembunyikan hal-hal yang tidak perlu Anda lihat atau tangani sepanjang waktu. Bahkan jika kita tidak perlu memikirkan tentang alokasi memori saat menulis kode, jika kita ingin benar-benar memahami apa yang terjadi dalam daftar tertaut dan apa yang membuatnya kuat, kita harus turun ke tingkat dasar.
Kita telah mempelajari tentang biner dan bagaimana data dapat dipecah menjadi bit dan byte. Sama seperti karakter, angka, kata, kalimat membutuhkan byte memori untuk mewakilinya, begitu pula struktur data.
Ketika sebuah array dibuat, itu membutuhkan sejumlah memori. Jika kita memiliki 7 huruf yang perlu kita simpan dalam sebuah array, kita akan membutuhkan 7 byte memori untuk mewakili array itu. Tapi, kita membutuhkan semua memori itu dalam satu blok yang berdekatan . Artinya, komputer kita perlu mencari 7 byte memori yang kosong, satu byte di samping yang lain, semuanya bersama-sama, di satu tempat.
Di sisi lain, ketika daftar tertaut dibuat, tidak membutuhkan 7 byte memori di satu tempat. Satu byte bisa tinggal di suatu tempat, sedangkan byte berikutnya bisa disimpan di tempat lain dalam memori sama sekali! Daftar tertaut tidak perlu menggunakan satu blok memori; sebaliknya, memori yang mereka gunakan bisa tersebar .
Perbedaan mendasar antara array dan daftar tertaut adalah bahwa array adalah struktur data statis , sedangkan daftar tertaut adalah struktur data dinamis . Sebuah struktur data statis membutuhkan semua sumber dayanya untuk dialokasikan ketika struktur dibuat; ini berarti bahwa meskipun struktur akan bertambah atau menyusut dalam ukuran dan elemen yang akan ditambahkan atau dihilangkan, itu masih selalu membutuhkan ukuran dan jumlah memori tertentu. Jika lebih banyak elemen perlu ditambahkan ke struktur data statis dan tidak memiliki cukup memori, Anda perlu menyalin data dari larik tersebut, misalnya, dan membuatnya kembali dengan lebih banyak memori, sehingga Anda dapat menambahkan elemen ke saya t.
Di sisi lain, struktur data dinamis dapat menyusut dan berkembang dalam memori. Itu tidak memerlukan sejumlah memori yang ditetapkan untuk dialokasikan agar ada, dan ukuran serta bentuknya dapat berubah, dan jumlah memori yang dibutuhkannya dapat berubah juga.
Sekarang, kita sudah bisa mulai melihat beberapa perbedaan utama antara array dan daftar tertaut. Tetapi ini menimbulkan pertanyaan: apa yang memungkinkan daftar tertaut memiliki ingatannya tersebar di mana-mana? Untuk menjawab pertanyaan ini, kita perlu melihat bagaimana daftar tertaut disusun.
Bagian dari daftar tertaut
Daftar yang ditautkan bisa kecil atau besar, tetapi tidak peduli ukurannya, bagian-bagian yang menyusunnya sebenarnya cukup sederhana. Daftar tertaut terdiri dari serangkaian node , yang merupakan elemen dari daftar.
Titik awal daftar adalah referensi ke simpul pertama, yang disebut sebagai kepala . Hampir semua daftar tertaut harus memiliki kepala, karena ini adalah satu-satunya titik masuk ke daftar dan semua elemennya secara efektif, dan tanpanya, Anda tidak akan tahu harus mulai dari mana! Akhir dari daftar bukanlah node, melainkan node yang menunjuk ke null , atau nilai kosong.
Node tunggal juga cukup sederhana. Ini hanya memiliki dua bagian: data , atau informasi yang dikandung node, dan referensi ke node berikutnya .
Jika kita bisa mengatasi hal ini, maka kita sudah setengah jalan. Cara kerja node sangatlah penting, dan sangat kuat, dan dapat diringkas sebagai berikut:
Sebuah node hanya mengetahui tentang data apa yang dikandungnya, dan siapa tetangganya.
Sebuah simpul tunggal tidak tahu berapa panjang daftar tertaut itu, dan bahkan mungkin tidak tahu di mana itu dimulai, atau di mana itu berakhir. Semua node yang bersangkutan adalah data yang dikandungnya, dan node mana yang dirujuk penunjuknya - node berikutnya dalam daftar.
Dan inilah alasan utama mengapa daftar tertaut tidak membutuhkan blok memori yang berdekatan. Karena satu node memiliki "alamat" atau referensi ke node berikutnya, mereka tidak perlu berada tepat di samping satu sama lain, seperti yang dimiliki elemen dalam sebuah array. Sebaliknya, kita hanya dapat mengandalkan fakta bahwa kita dapat melintasi daftar kita dengan bersandar pada referensi penunjuk ke simpul berikutnya, yang berarti bahwa mesin kita tidak perlu memblokir satu potongan memori untuk mewakili daftar kita.
Ini juga merupakan penjelasan mengapa daftar tertaut dapat tumbuh dan menyusut secara dinamis selama eksekusi program. Menambahkan atau menghapus node dengan daftar tertaut menjadi semudah mengatur ulang beberapa pointer, daripada menyalin elemen dari sebuah array! Namun, ada juga beberapa kelemahan pada daftar tertaut yang belum saya sebutkan kepada Anda - tetapi lebih banyak tentang itu minggu depan.
Untuk saat ini, kita hanya akan menikmati betapa kerennya daftar tertaut!
Daftar untuk semua bentuk dan ukuran
Meskipun bagian dari daftar tertaut tidak berubah, cara kami menyusun daftar tertaut bisa sangat berbeda. Seperti kebanyakan hal dalam perangkat lunak, bergantung pada masalah yang kita coba selesaikan, satu jenis daftar tertaut mungkin merupakan alat yang lebih baik untuk pekerjaan itu daripada yang lain.
Daftar tertaut tunggal adalah jenis daftar tertaut yang paling sederhana, hanya berdasarkan fakta bahwa daftar tersebut hanya mengarah ke satu arah. Ada satu trek yang dapat kita lalui daftarnya; kita mulai di kepala node, dan melintasi dari akar sampai terakhir node, yang akan berakhir pada kosong nol nilai.
Tetapi seperti halnya sebuah node dapat mereferensikan node tetangga berikutnya, node tersebut juga dapat memiliki pointer referensi ke node sebelumnya! Inilah yang kami sebut daftar tertaut ganda , karena ada dua referensi yang terdapat di dalam setiap node: referensi ke node berikutnya , serta node sebelumnya . Ini dapat membantu jika kita ingin dapat melintasi struktur data kita tidak hanya dalam satu jalur atau arah, tetapi juga ke belakang.
Misalnya, jika kita ingin dapat berpindah antara satu node dan node sebelumnya tanpa harus kembali ke awal daftar , daftar tertaut ganda akan menjadi struktur data yang lebih baik daripada daftar tertaut tunggal. Namun, semuanya membutuhkan ruang dan memori, jadi jika node kita harus menyimpan dua pointer referensi, bukan hanya satu, itu akan menjadi hal lain yang perlu dipertimbangkan.
Sebuah linked list melingkar agak aneh dalam hal itu tidak berakhir dengan simpul menunjuk ke nilai nol. Sebaliknya, ia memiliki simpul yang bertindak sebagai ekor dari daftar (bukan simpul kepala konvensional), dan simpul setelah simpul ekor adalah awal dari daftar. Struktur organisasi ini membuatnya sangat mudah untuk menambahkan sesuatu ke akhir daftar, karena Anda dapat mulai melintasinya di simpul ekor , karena elemen pertama dan elemen terakhir mengarah satu sama lain. Daftar tertaut melingkar dapat mulai menjadi sangat gila karena kita dapat mengubah daftar tertaut tunggal dan daftar tertaut ganda menjadi daftar tertaut melingkar!
Tetapi betapapun rumitnya daftar tertaut, jika kita dapat mengingat dasar-dasar sebuah node dan cara kerjanya serta bagaimana referensi penunjuk yang berbeda dalam daftar kita terstruktur, tidak ada daftar tertaut yang tidak dapat kita tangani!
Minggu depan, di bagian 2 dari seri ini, kita akan tenggelam dalam kerumitan ruang waktu dari daftar terkait, dan bagaimana perbandingannya dengan sepupu mereka, array. Saya berjanji bahwa ini sebenarnya jauh lebih menyenangkan daripada kedengarannya!
Sumber daya
Jika menurut Anda daftar tertaut sangat keren, lihat sumber daya bermanfaat ini.
- Perbedaan antara Array dan Linked List , Damien Wintour
- Struktur Data: Array vs Linked List , mycodeschool
- Daftar Terkait: Dasar-dasar , Dr. Edward Gehringer
- Pengantar Daftar Terkait , Dr. Victor Adamchik
- Struktur & Implementasi Data , Dr. Jennifer Welch
- Struktur Data Statis vs. Struktur Data Dinamis , Ayoma Gayan Wijethunga