Apa praktik terbaik ketika datang ke kelas menulis yang mungkin harus tahu tentang antarmuka pengguna. Bukankah kelas yang tahu cara menggambar itu sendiri melanggar beberapa praktik terbaik karena itu tergantung pada apa antarmuka pengguna (konsol, GUI, dll)?
Dalam banyak buku pemrograman saya menemukan contoh "Shape" yang menunjukkan warisan. Bentuk kelas dasar memiliki metode draw () yang masing-masing bentuk seperti lingkaran dan menimpa persegi. Ini memungkinkan terjadinya polimorfisme. Tetapi bukankah metode draw () sangat tergantung pada apa antarmuka pengguna itu? Jika kita menulis kelas ini untuk mengatakan, Menangkan Formulir, maka kita tidak dapat menggunakannya kembali untuk aplikasi konsol atau aplikasi web. Apakah ini benar?
Alasan untuk pertanyaan ini adalah bahwa saya selalu terjebak dan menutup telepon tentang cara menggeneralisasi kelas sehingga mereka paling berguna. Ini sebenarnya bekerja melawan saya dan saya bertanya-tanya apakah saya "berusaha terlalu keras".
Shape
kelas, maka Anda mungkin menulis tumpukan grafik itu sendiri, bukan menulis klien ke tumpukan grafik.Jawaban:
Itu tergantung pada kelas dan kasus penggunaan. Unsur visual yang mengetahui cara menggambar itu sendiri belum tentu merupakan pelanggaran prinsip tanggung jawab tunggal.
Sekali lagi, belum tentu. Jika Anda bisa membuat antarmuka (drawPoint, drawLine, atur Warna, dll.), Anda dapat melewati konteks apa pun untuk menggambar sesuatu ke sesuatu ke bentuk, misalnya dalam konstruktor bentuk itu. Ini akan memungkinkan bentuk untuk menggambar diri mereka sendiri di konsol atau kanvas yang diberikan.
Ya benar. Jika Anda menulis UserControl (bukan kelas pada umumnya) untuk Formulir Windows, maka Anda tidak akan dapat menggunakannya dengan konsol. Tapi itu bukan masalah. Mengapa Anda mengharapkan UserControl untuk Formulir Windows berfungsi dengan segala jenis presentasi? UserControl harus melakukan satu hal dan melakukannya dengan baik. Ini terikat pada bentuk presentasi tertentu menurut definisi. Pada akhirnya, pengguna membutuhkan sesuatu yang konkret dan bukan abstraksi. Ini mungkin hanya sebagian benar untuk kerangka kerja, tetapi untuk aplikasi pengguna akhir, itu.
Namun, logika di baliknya harus dipisahkan, sehingga Anda dapat menggunakannya lagi dengan teknologi presentasi lainnya. Perkenalkan antarmuka jika perlu, untuk mempertahankan ortogonalitas untuk aplikasi Anda. Aturan umumnya adalah: Hal-hal konkret harus dapat ditukar dengan hal-hal konkret lainnya.
Anda tahu, programmer ekstrim menyukai sikap YAGNI mereka . Jangan mencoba menulis semuanya secara umum dan jangan berusaha terlalu keras untuk membuat semuanya menjadi tujuan umum. Ini disebut rekayasa ulang dan pada akhirnya akan mengarah pada kode yang benar-benar berbelit-belit. Berikan masing-masing komponen tugas yang tepat dan pastikan komponen itu bekerja dengan baik. Masukkan abstraksi jika perlu, di mana Anda mengharapkan hal-hal berubah (misalnya antarmuka untuk menggambar konteks, seperti yang dinyatakan di atas).
Secara umum, ketika menulis aplikasi bisnis, Anda harus selalu mencoba untuk memisahkan hal-hal. MVC dan MVVM sangat bagus untuk memisahkan logika dari presentasi, sehingga Anda dapat menggunakannya kembali untuk presentasi web atau aplikasi konsol. Perlu diingat bahwa pada akhirnya, beberapa hal harus konkret. Pengguna Anda tidak dapat bekerja dengan abstraksi, mereka membutuhkan sesuatu yang konkret. Abstraksi hanya membantu Anda, sang programmer, agar kode dapat diperluas dan dipelihara. Anda perlu memikirkan di mana Anda membutuhkan kode Anda agar fleksibel. Akhirnya semua abstraksi harus melahirkan sesuatu yang konkret.
Sunting: Jika Anda ingin membaca lebih lanjut tentang arsitektur dan teknik desain yang dapat memberikan praktik terbaik, saya sarankan Anda membaca jawaban @Catchops dan membaca tentang praktik SOLID di wikipedia.
Sebagai permulaan, saya selalu merekomendasikan buku berikut: Head First Design Patterns . Ini akan membantu Anda memahami teknik abstraksi / praktik desain OOP, lebih dari buku GoF (yang sangat bagus, hanya saja tidak cocok untuk pemula).
sumber
Anda pasti benar. Dan tidak, Anda "berusaha dengan baik" :)
Baca tentang prinsip tanggung jawab tunggal
Pekerjaan batin Anda di kelas dan cara informasi ini disajikan kepada pengguna adalah dua tanggung jawab.
Jangan takut untuk memisahkan kelas. Jarang masalahnya adalah terlalu banyak abstraksi dan decoupling :)
Dua pola yang sangat relevan adalah Model-view-controller untuk aplikasi web dan Model View ViewModel untuk Silverlight / WPF.
sumber
Saya sering menggunakan MVVM dan menurut pendapat saya, kelas objek bisnis seharusnya tidak perlu tahu apa-apa tentang antarmuka pengguna. Tentu mereka mungkin perlu mengetahui
SelectedItem
, atauIsChecked
, atauIsVisible
, dll, tetapi nilai-nilai itu tidak perlu mengikat ke UI tertentu dan dapat menjadi properti generik di kelas.Jika Anda perlu melakukan sesuatu pada antarmuka di belakang kode, seperti mengatur Fokus, menjalankan Animasi, menangani Hot Keys, dll. Maka kode tersebut harus menjadi bagian dari kode-belakang UI, bukan kelas logika bisnis Anda.
Jadi saya akan mengatakan jangan berhenti mencoba memisahkan UI dan kelas Anda. Semakin terpisah mereka, semakin mudah untuk mempertahankan dan menguji.
sumber
Ada sejumlah pola desain yang sudah terbukti dan telah dikembangkan selama bertahun-tahun untuk menjawab apa yang Anda bicarakan. Jawaban lain untuk pertanyaan Anda merujuk pada Prinsip Satu Tanggung Jawab - yang benar-benar valid - dan apa yang tampaknya mendorong pertanyaan Anda. Prinsip itu hanya menyatakan bahwa suatu kelas perlu melakukan satu hal BAIK. Dengan kata lain meningkatkan Kohesi dan menurunkan Kopling yang merupakan tujuan dari desain berorientasi objek yang baik - apakah kelas melakukan satu hal dengan baik, dan tidak memiliki banyak ketergantungan pada yang lain.
Nah ... Anda benar dalam mengamati bahwa jika Anda ingin menggambar lingkaran pada iPhone, itu akan berbeda dari menggambar satu pada PC yang menjalankan windows. Anda HARUS memiliki (dalam hal ini) kelas konkret yang menggambar satu sumur di iPhone, dan lainnya yang menarik satu sumur di PC. Di sinilah dasar OO dasar warisan yang semua contoh bentuk itu rusak. Anda tidak bisa melakukannya dengan warisan saja.
Di situlah antarmuka masuk - seperti yang dinyatakan oleh buku Gang of Four (HARUS BACA) - Selalu mendukung implementasi daripada warisan. Dengan kata lain, gunakan antarmuka untuk menyatukan arsitektur yang dapat melakukan berbagai fungsi dalam banyak cara tanpa bergantung pada dependensi kode keras.
Saya telah melihat referensi pada prinsip-prinsip SOLID . Itu bagus. 'S' adalah prinsip tanggung jawab tunggal. NAMUN, 'D' adalah singkatan dari Dependency Inversion. Pola Inversion of Control (Injeksi Ketergantungan) dapat digunakan di sini. Ini sangat kuat dan dapat digunakan untuk menjawab pertanyaan tentang bagaimana merancang sistem yang dapat menggambar lingkaran untuk iPhone serta satu untuk PC.
Dimungkinkan untuk membuat arsitektur yang berisi aturan bisnis umum dan akses data, tetapi memiliki berbagai implementasi antarmuka pengguna menggunakan konstruk ini. Namun, sangat membantu untuk benar-benar berada di tim yang mengimplementasikannya dan melihatnya beraksi untuk benar-benar memahaminya.
Ini hanyalah jawaban tingkat tinggi yang cepat untuk sebuah pertanyaan yang pantas mendapatkan jawaban yang lebih terperinci. Saya mendorong Anda untuk melihat lebih jauh ke dalam pola-pola ini. Beberapa implementasi yang lebih nyata dari patters ini dapat ditemukan sebagai nama-nama terkenal dari MVC dan MVVM.
Semoga berhasil!
sumber
Dalam hal ini Anda masih dapat menggunakan MVC / MVVM dan menyuntikkan berbagai implementasi UI menggunakan antarmuka umum:
Dengan cara ini Anda akan dapat menggunakan kembali Kontroler dan Model logika Anda sambil dapat menambahkan GUI jenis baru.
sumber
Ada beberapa pola untuk melakukannya: MVP, MVC, MVVM, dll ...
Artikel bagus dari Martin Fowler (nama besar) untuk dibaca adalah Arsitektur GUI: http://www.martinfowler.com/eaaDev/uiArchs.html
MVP belum disebutkan tetapi pasti layak untuk dikutip: lihatlah.
Ini adalah pola yang disarankan oleh pengembang Google Web Toolkit untuk digunakan, ini sangat rapi.
Anda dapat menemukan kode nyata, contoh nyata, dan alasan mengapa pendekatan ini bermanfaat di sini:
http://code.google.com/webtoolkit/articles/mvp-architecture.html
http://code.google.com/webtoolkit/articles/mvp-architecture-2.html
Salah satu keuntungan mengikuti ini atau pendekatan serupa yang belum cukup ditekankan di sini adalah testabilitas! Dalam banyak kasus saya akan mengatakan itu adalah keuntungan utama!
sumber
Ini adalah salah satu tempat di mana OOP gagal melakukan pekerjaan dengan baik di abstraksi. Polimorfisme OOP menggunakan pengiriman dinamis atas satu variabel ('ini'). Jika polimorfisme telah di-root pada Shape maka Anda tidak dapat mengirim polimorfik sekutu pada renderer (konsol, GUI, dll).
Pertimbangkan sistem pemrograman yang dapat mengirimkan dua atau lebih variabel:
dan juga anggap sistem dapat memberi Anda cara untuk mengekspresikan poly_draw untuk berbagai kombinasi tipe Shape dan tipe Renderer. Maka akan mudah untuk membuat klasifikasi bentuk dan penyaji, bukan? Pemeriksa tipe entah bagaimana akan membantu Anda mencari tahu apakah ada kombinasi bentuk dan renderer yang mungkin Anda lewatkan penerapannya.
Sebagian besar bahasa OOP tidak mendukung apa pun seperti di atas (beberapa melakukannya, tetapi mereka tidak umum). Saya sarankan Anda melihat pola Pengunjung untuk solusinya.
sumber
Di atas terdengar benar bagi saya. Menurut pemahaman saya, bisa dikatakan itu berarti kopling yang relatif ketat antara Controller dan View dalam hal pola desain MVC. Ini juga berarti bahwa untuk beralih antara desktop-konsol-webapp kita harus beralih baik Controller dan View sebagai pasangan - hanya model tetap tidak berubah.
Nah pendapat saya saat ini di atas adalah bahwa kopling View-Controller yang kita bicarakan ini OK dan bahkan lebih, itu agak trendi dalam desain modern.
Padahal, satu atau dua tahun yang lalu saya juga merasa tidak aman tentang itu. Saya berubah pikiran setelah mempelajari diskusi di forum Sun mengenai pola dan desain OO.
Jika Anda tertarik, coba sendiri forum ini - forum ini telah bermigrasi ke Oracle sekarang ( tautan ). Jika Anda sampai di sana, coba ping pria Saish - saat itu, penjelasannya tentang masalah rumit ini ternyata paling membantu saya. Saya tidak tahu apakah dia masih berpartisipasi - saya sendiri belum pernah ke sana cukup lama
sumber
Dari pandangan pragmatis, beberapa kode di sistem Anda perlu tahu cara menggambar sesuatu seperti
Rectangle
jika itu persyaratan pengguna-akhir. Dan itu akan memanas pada titik tertentu untuk melakukan hal-hal yang sangat rendah seperti rasterisasi piksel atau menampilkan sesuatu di konsol.Pertanyaan saya dari sudut pandang penggandaan adalah siapa / apa yang harus bergantung pada jenis informasi ini, dan sampai pada tingkat detail apa (seberapa abstrak, misalnya)?
Mengabstraksi Gambar / Kemampuan Rendering
Karena jika kode gambar tingkat tinggi hanya bergantung pada sesuatu yang sangat abstrak, abstraksi itu mungkin dapat bekerja (melalui substitusi implementasi konkret) pada semua platform yang Anda targetkan. Sebagai contoh yang dibuat-buat, beberapa
IDrawer
antarmuka yang sangat abstrak mungkin dapat diimplementasikan di kedua konsol dan API GUI untuk melakukan hal-hal seperti bentuk plot (implementasi konsol mungkin memperlakukan konsol seperti beberapa "gambar" 80xN dengan seni ASCII). Tentu saja itu adalah contoh yang dibuat karena biasanya bukan itu yang ingin Anda lakukan adalah memperlakukan keluaran konsol seperti penyangga gambar / bingkai; biasanya sebagian besar pengguna akhir membutuhkan panggilan untuk lebih banyak interaksi berbasis teks di konsol.Pertimbangan lain adalah seberapa mudahkah mendesain abstraksi yang stabil? Karena mungkin mudah jika semua yang Anda targetkan adalah API GUI modern untuk mengabstraksi kemampuan menggambar bentuk dasar seperti merencanakan garis, persegi panjang, jalur, teks, hal-hal semacam ini (hanya rasterisasi 2D sederhana dari kumpulan primitif terbatas) , dengan satu antarmuka abstrak yang dapat dengan mudah diimplementasikan untuk mereka semua melalui berbagai subtipe dengan sedikit biaya. Jika Anda dapat mendesain abstraksi seperti itu secara efektif dan mengimplementasikannya pada semua platform target, maka saya akan mengatakan itu adalah kejahatan yang jauh lebih kecil, jika bahkan kejahatan sama sekali, untuk bentuk atau kontrol GUI atau apa pun untuk mengetahui cara menggambar dirinya sendiri menggunakan abstraksi.
Tetapi katakan Anda mencoba untuk mengabstraksikan detail berdarah yang bervariasi antara Playstation Portable, iPhone, XBox One, dan PC gaming yang tangguh sementara kebutuhan Anda adalah memanfaatkan teknik rendering / peneduhan 3D real-time paling mutakhir pada masing-masing . Dalam hal ini mencoba untuk membuat satu antarmuka abstrak untuk mengabstraksi rincian render ketika kemampuan perangkat keras dan API yang bervariasi sangat liar hampir pasti akan menghasilkan waktu yang sangat besar dalam mendesain dan mendesain ulang, kemungkinan besar perubahan desain berulang dengan tak terduga. penemuan, dan juga solusi denominator umum terendah yang gagal mengeksploitasi keunikan dan kekuatan penuh dari perangkat keras yang mendasarinya.
Membuat Aliran Ketergantungan Menuju Desain Yang Stabil, "Mudah"
Di bidang saya, saya berada dalam skenario terakhir itu. Kami menargetkan banyak perangkat keras yang berbeda dengan kemampuan dan API yang mendasari sangat berbeda, dan untuk mencoba menghasilkan satu abstraksi rendering / menggambar untuk memerintah mereka semua adalah tanpa batas (kita mungkin menjadi terkenal di dunia hanya melakukan itu secara efektif karena itu akan menjadi permainan) changer di industri). Jadi, hal terakhir yang saya inginkan dalam kasus saya adalah seperti analog
Shape
atauModel
atauParticle Emitter
yang tahu cara menggambar itu sendiri, bahkan jika itu menyatakan bahwa menggambar dengan cara tingkat tertinggi dan paling abstrak mungkin ...... karena abstraksi-abstraksi itu terlalu sulit untuk dirancang dengan benar, dan ketika sebuah desain sulit untuk mendapatkan yang benar, dan semuanya tergantung padanya, itu adalah resep untuk perubahan desain pusat yang paling mahal yang mengacak-acak dan menghancurkan semuanya tergantung padanya. Jadi hal terakhir yang Anda inginkan adalah ketergantungan pada sistem Anda mengalir ke desain abstrak yang terlalu sulit untuk diperbaiki (terlalu sulit untuk distabilkan tanpa perubahan yang mengganggu).
Tergantung Sulit pada Mudah, Tidak Mudah Tergantung pada Sulit
Jadi yang kami lakukan adalah membuat ketergantungan mengalir ke hal-hal yang mudah dirancang. Jauh lebih mudah untuk mendesain "Model" abstrak yang hanya berfokus pada penyimpanan hal-hal seperti poligon dan material dan mendapatkan desain yang benar daripada mendesain "Renderer" abstrak yang dapat secara efektif diimplementasikan (melalui subtipe beton yang dapat diganti-ganti) untuk melayani gambar meminta secara seragam untuk perangkat keras yang berbeda seperti PSP dari PC.
Jadi kami membalikkan ketergantungan dari hal-hal yang sulit untuk dirancang. Alih-alih membuat model abstrak tahu cara menggambar diri mereka sendiri ke desain renderer abstrak yang mereka semua andalkan (dan merusak implementasinya jika desain itu berubah), kami malah memiliki renderer abstrak yang tahu cara menggambar setiap objek abstrak di adegan kami ( model, penghasil partikel, dll), dan dengan demikian kita dapat menerapkan subtipe renderer OpenGL untuk PC seperti
RendererGl
, yang lain untuk PSP sepertiRendererPsp
, yang lain untuk ponsel, dll. Dalam hal ini dependensi mengalir ke desain yang stabil, mudah untuk mendapatkan yang benar, dari renderer ke berbagai jenis entitas (model, partikel, tekstur, dll) dalam adegan kami, bukan sebaliknya.Jika Anda menemukan diri Anda mencoba untuk mengabstraksi sesuatu di tingkat pusat basis kode Anda dan itu terlalu sulit untuk dirancang, alih-alih dengan keras kepala memukul dinding dan terus-menerus membuat perubahan yang mengganggu setiap bulan / tahun yang membutuhkan memperbarui 8.000 file sumber karena itu melanggar semua yang tergantung padanya, saran nomor satu saya adalah mempertimbangkan membalikkan dependensi. Lihat apakah Anda dapat menulis kode sedemikian rupa sehingga hal yang sangat sulit untuk dirancang tergantung pada segala sesuatu yang lebih mudah untuk dirancang, tidak memiliki hal-hal yang lebih mudah untuk dirancang tergantung pada hal yang sangat sulit untuk dirancang. Perhatikan bahwa saya berbicara tentang desain (khususnya desain antarmuka) dan bukan implementasi: kadang-kadang hal-hal mudah dirancang dan sulit diimplementasikan, dan terkadang hal-hal sulit untuk dirancang tetapi mudah diimplementasikan. Ketergantungan mengalir ke arah desain, jadi fokusnya seharusnya hanya pada seberapa sulit sesuatu untuk merancang di sini untuk menentukan arah di mana dependensi mengalir.
Prinsip Tanggung Jawab Tunggal
Bagi saya SRP tidak begitu menarik di sini biasanya (meskipun tergantung pada konteksnya). Maksud saya ada tindakan menyeimbangkan tali ketat dalam mendesain hal-hal yang jelas dalam tujuan dan dapat dipelihara tetapi
Shape
objek Anda mungkin harus memaparkan informasi yang lebih rinci jika mereka tidak tahu cara menggambar diri mereka sendiri, misalnya, dan mungkin tidak ada banyak hal yang bermakna untuk lakukan dengan bentuk dalam konteks penggunaan tertentu daripada membangunnya dan menggambarnya. Ada pertukaran dengan hampir semua hal, dan itu tidak terkait dengan SRP yang dapat membuat hal-hal sadar bagaimana menggambar diri mereka mampu menjadi mimpi buruk pemeliharaan dalam pengalaman saya dalam konteks tertentu.Ini lebih terkait dengan kopling dan arah aliran dependensi dalam sistem Anda. Jika Anda mencoba untuk mem-port antarmuka rendering abstrak yang semuanya bergantung (karena mereka menggunakannya untuk menggambar sendiri) ke API / perangkat keras target baru dan menyadari bahwa Anda harus mengubah desainnya untuk membuatnya bekerja secara efektif di sana, maka itu adalah perubahan yang sangat mahal untuk dilakukan yang membutuhkan penggantian implementasi segala sesuatu di sistem Anda yang tahu cara menggambar sendiri. Dan itulah masalah perawatan paling praktis yang saya temui dengan hal-hal yang disadari tentang cara menggambar diri mereka sendiri jika itu berarti muatan perahu yang mengalir ke arah abstraksi yang terlalu sulit untuk dirancang dengan benar di muka.
Kebanggaan Pengembang
Saya menyebutkan satu poin ini karena, dalam pengalaman saya, ini sering merupakan hambatan terbesar untuk mengalirkan arah ketergantungan kepada hal-hal yang lebih mudah untuk dirancang. Sangat mudah bagi pengembang untuk menjadi sedikit ambisius di sini dan berkata, "Saya akan merancang abstraksi rendering lintas platform untuk mengatur semuanya, saya akan menyelesaikan apa yang menghabiskan waktu berbulan-bulan bagi pengembang lain dalam porting, dan saya akan mendapatkan itu benar dan itu akan bekerja seperti sulap pada setiap platform yang kami dukung dan gunakan teknik rendering canggih pada setiap platform; Saya sudah membayangkannya di kepala saya. "Dalam hal ini mereka menolak solusi praktis yaitu untuk menghindari melakukan itu dan hanya membalik arah dependensi dan menerjemahkan apa yang mungkin sangat mahal dan berulang perubahan desain pusat untuk perubahan hanya murah dan lokal untuk implementasi. Perlu ada semacam naluri "bendera putih" di pengembang untuk menyerah ketika ada sesuatu yang terlalu sulit untuk dirancang ke tingkat abstrak dan mempertimbangkan kembali seluruh strategi mereka, kalau tidak mereka akan menghadapi banyak kesedihan dan rasa sakit. Saya akan menyarankan untuk mentransfer ambisi dan semangat juang seperti itu ke implementasi canggih dari hal yang lebih mudah untuk dirancang daripada mengambil ambisi menaklukkan dunia ke tingkat desain antarmuka.
sumber
OpenGLBillboard
, Anda akan membuatOpenGLRenderer
yang tahu cara menggambar jenis apa punIBillBoard
? Tetapi akankah hal itu dilakukan dengan mendelegasikan logikaIBillBoard
, atau akankah ia memiliki saklar atauIBillBoard
tipe besar dengan persyaratan? Itulah yang sulit saya pahami, karena itu sepertinya tidak dapat dipertahankan sama sekali!RendererPsp
yang mengetahui abstraksi tingkat tinggi dari adegan permainan Anda, maka ia dapat melakukan semua keajaiban dan membalikkan yang diperlukan untuk membuat hal-hal seperti itu dengan cara yang terlihat meyakinkan di PSP ....BillBoard
bergantung pada hal itu akan sangat sulit dilakukan? SedangkanIRenderer
tingkat tinggi sudah dan bisa bergantung pada masalah ini dengan masalah jauh lebih sedikit?