Pemrograman berorientasi objek atau Object Oriented Programming (OOP) menjadi sebuah fenomena yang mempunyai dampak sangat besar di dunia komputasi, terlebih setelah munculnya Java sebagai bahasa pemrograman yang mengimplementasikan konsep ini secara penuh. Artinya, di Java, segala sesuatu yang menyusun aplikasi terdiri atas kumpulan objek. Tidak ada satu komponenpun di Java yang dapat berjalan tanpa melibatkan objek. Ini berbeda dengan beberapa bahasa lain yang sebenarnya berkonsep prosedural, tapi menyelipkan konsep OOP di dalamnya. Beberapa persepsi yang tidak tepat tentang OOP, antara lain:
- Menyebut hasil kompilasi dengan sebutan object, kemudian menyebut bahasa tersebut sebagai OOP
- Menganggap event-driven programming adalah OOP, misalnya seperti yang ada di VB.
Melalui artikel ini, saya mencoba untuk menggambarkan secara sederhana seperti apakah konsep OOP tersebut dan memberi contoh implementasinya dalam bahasa Java. Jika Anda belum familiar dengan Java, artikel “Belajar Java, mulai dari mana” mungkin bisa membantu.
Di dalam OOP, komponen pembangun aplikasi adalah class dan object. Secara sederhana, class dapat digambarkan sebagai cetakan, dan object adalah sesuatu yang dihasilkan dari cetakan tersebut. Contohnya, bila kita mempunyai cetakan (class) Mobil, maka kita bisa membuat object mobil sebanyak yang kita mau. Ada 2 komponen penting dalam sebuah class, property (sifat) dan method (behavior/kelakuan). Object yang dibuat dari class tertentu akan memiliki sifat dan kelakuan sesuai dengan yang didefinisikan di class tersebut.
01 | package suhearie.blog.oop; |
07 | private int kecepatan; |
09 | private int bahanBakar; |
11 | public Mobil(String model) { |
19 | public void berhenti() { |
23 | public static void main(String[] args) { |
24 | Mobil myCivic = new Mobil( "Civic" ); |
25 | Mobil myCamry = new Mobil( "Camry" ); |
Pada contoh di atas, class Mobil memiliki property model dengan tipe data String dan propertykecepatan dengan tipe data integer. Property-property ini yang akan menentukan karakteristik atau sifat dari object tersebut, apa modelnya, berapa kecepatannya, dan berapabahanBakarnya saat ini. Selain itu, class mobil juga mempunyai method untuk menyuruh mobil tersebut maju dan menyuruhnya berhenti.
Satu blok kode yang spesial dari sebuah class adalah constructor. Constructor dideklarasikan dengan tanpa return value, dan namanya harus sama persis dengan nama classnya. Kita akan memanggil constructor ini ketika akan membuat sebuah object baru dari class tersebut. Di Java, bila kita tidak mendeklarasikan satupun constructor, compiler akan secara otomatis menambahkan default constructor.
Method public static void main (String[] args) adalah entry point aplikasi Java. Di dalam method tersebut kita membuat 2 buah object dari class Mobil. Satu object kita beri nama myCivic yang kedua kita beri nama myCamry. Dua object tersebut adalah object yang berbeda, tidak terkait satu sama lain. Kita dapat menyuruh myCivic untuk maju, sementara myCamry masih diam di tempat.
Object berinteraksi dengan object lainnya dengan mengakses property nya atau memanggil method yang telah didefinisikan di class tersebut.
Ada 4 konsep yang sangat berkaitan dengan OOP. Mereka adalah Encapsulation, Inheritance, Abstraction, dan Polymorphism. Kita akan menjelajahi keempat hal tersebut dengan detail pada bagian berikut.
1. Encapsulation
Encapsulation berarti membungkus informasi atau data ke dalam sebuah object. Di dalam OOP, kita akan selalu mengakses data melalui object. Tidak ada data yang tersebar disana-sini diluar object. Konsep ini dipertegas dengan adanya penentu akses (access modifier) pada deklarasi property dan method.
Pada contoh di atas, property model dan kecepatan dideklarasikan dengan akses private, yang berarti property ini tidak dapat diakses secara langsung oleh object lain. Selain private, Java menyediakan penentu akses lainnya (dengan urutan yang semakin “terbuka”) yaitu default(tanpa modifier), protected dan public. Penentu akses public membuat property akan dapat diakses secara langsung dari semua object yang lain. Hal ini sangat tidak dianjurkan karena object lain dapat mengubah isi dari property tersebut, dan membuat object target menjadi tidak konsisten. Bayangkan bila property bahanBakar bisa langsung diubah dari class lain menjadi -10 (minus sepuluh), keadaan object mobil tersebut tentunya menjadi tidak logis.
2. Inheritance
Di dalam OOP, kita bisa membuat sebuah class dengan meng-extend class lain sehingga class baru tersebut mewarisi property dan method dari class yang di-extend atau bisa disebut parent class.
01 | package suhearie.blog.oop; |
05 | protected String nama; |
07 | protected String umur; |
10 | System.out.println( "Saya sedang makan" ); |
14 | System.out.println( "Saya sedang minum" ); |
18 | System.out.println( "Saya sedang tidur" ); |
22 | class Bayi extends Orang { |
24 | public void menangis() { |
25 | System.out.println( "Huahuahua" ); |
29 | class Karyawan extends Orang { |
31 | public void bekerja() { |
32 | System.out.println( "Sibuk nih" ); |
Class Bayi dan class Karyawan dideklarasikan dengan meng-extend class Orang, dimana class Orang tersebut sudah memiliki beberapa property dan method. Dengan demikian, class Bayi mewarisi mewarisi semua property dan method yang dideklarasikan di class Orang, kemudian menambahkan satu method menangis. Begitu juga dengan class Karyawan yang menambahkan method bekerja.
01 | public static void main(String[] args) { |
03 | Bayi bayi = new Bayi(); |
07 | Karyawan karyawan = new Karyawan(); |
Jadi, karena Bayi dan Karyawan mewarisi method-method yang udah kita declare di class Orang, kita tidak perlu menulis ulang method-method tersebut, secara otomatis Bayi dan Karyawan juga bisa makan, minum, dan tidur.
Tujuannya adalah supaya kita tidak perlu rewrite apa yang bisa di reuse. Apa keuntungan reuse dibanding rewrite ?
Kalau rewrite, kita harus copy paste ke semua class. Kemudian kalau suatu saat kita harus mengubah isi method itu, maka harus mengubah di tiga tempat (Orang, Bayi, dan Karyawan). Kalau reuse, cukup deklarasikan satu kali, kemudian jika ingin mengubah cukup ubah di satu tempat.
3. Abstraction
Abstraction merupakan konsep generalisasi dari beberapa komponen yang sejenis, dan biasanya dilakukan dengan pemisahan antara spesifikasi dan implementasi.
1 | package suhearie.blog.oop.abstraction; |
3 | public interface Televisi { |
5 | public void nyalakan(); |
01 | public class TelevisiA implements Televisi { |
03 | public void nyalakan() { |
04 | System.out.println( "1" ); |
05 | System.out.println( "2" ); |
06 | System.out.println( "3" ); |
07 | System.out.println( "Televisi A menyala" ); |
10 | public void matikan() { |
11 | System.out.println( "3" ); |
12 | System.out.println( "2" ); |
13 | System.out.println( "1" ); |
14 | System.out.println( "Televisi A mati" ); |
01 | public class TelevisiB implements Televisi { |
03 | public void nyalakan() { |
04 | System.out.println( "Televisi B langsung menyala" ); |
07 | public void matikan() { |
08 | System.out.println( "Televisi B langsung mati" ); |
01 | public class RemoteControl { |
03 | public static void main(String[] args) { |
07 | myTV = new TelevisiA(); |
12 | myTV = new TelevisiB(); |
Di atas telah dibuat sebuah interface Televisi yang mendefinisikan bahwa sebuah televisi bisa dinyalakan dan dimatikan. Interface hanya menentukan spesifikasi tanpa menyebutkan implementasinya. Class TelevisiA dan TelevisiB merupakan implementasi dari spesifikasi tersebut. Kedua class ini mengimplementasikan method nyalakan dan matikan dengan cara yang berbeda. TelevisiA menghitung 1,2,3 terlebih dahulu sebelum menyala dan sebelum mati, sementara TelevisiB langsung menyala dan langsung mati.
Class RemoteControl mencoba mengendalikan kedua televisi ini dengan mengandalkan spesifikasi yang ada. Tentunya TelevisiA dan TelevisiB berhasil dikendalikan dari remote control, meskipun implementasinya bisa berbeda.
4. Polymorphism
Polymorphism dapat diterjemahkan secara bebas mejadi”banyak bentuk”. Dalam OOP, kita bisa mendefinisikan sebuah class yang meng-extend class lain dan meng-override atau mengimplementasi ulang isi sebuah method.
02 | protected String nama; |
04 | protected String umur; |
07 | System.out.println( "Saya sedang makan" ); |
11 | System.out.println( "Saya sedang minum" ); |
15 | System.out.println( "Saya sedang tidur" ); |
18 | public static void main(String[] args) { |
19 | Orang orang = new Bayi(); |
22 | orang = new Karyawan(); |
01 | class Bayi extends Orang { |
03 | public void menangis() { |
04 | System.out.println( "Huahuahua" ); |
08 | System.out.println( "Bayi makan bubur" ); |
01 | class Karyawan extends Orang { |
03 | public void bekerja() { |
04 | System.out.println( "Sibuk nih" ); |
08 | System.out.println( "Karyawan makan pizza" ); |
Pemanggilan orang.makan() pada object bayi dan karyawan akan menghasilkan output yang berbeda karena pada saat runtime, Java akan mengeksekusi implementasi aktual dari masing-masing object.