Gaya pribadi saya dengan C ++ selalu menempatkan deklarasi kelas dalam file sertakan, dan definisi dalam file .cpp, sangat mirip dengan yang ditentukan dalam jawaban Loki untuk File Header C ++, Pemisahan Kode . Diakui, bagian dari alasan saya suka gaya ini mungkin ada hubungannya dengan semua tahun yang saya habiskan untuk pengkodean Modula-2 dan Ada, yang keduanya memiliki skema serupa dengan spesifikasi file dan file body.
Saya memiliki rekan kerja, jauh lebih berpengetahuan di C ++ daripada saya, yang bersikeras bahwa semua deklarasi C ++ harus, jika memungkinkan, termasuk definisi di sana dalam file header. Dia tidak mengatakan ini adalah gaya alternatif yang valid, atau bahkan gaya yang sedikit lebih baik, tetapi ini adalah gaya baru yang diterima secara universal yang sekarang digunakan semua orang untuk C ++.
Aku tidak lentur seperti dulu, jadi aku tidak benar-benar ingin naik kereta musiknya sampai aku melihat beberapa orang di sana bersamanya. Jadi seberapa umum idiom ini?
Sekedar memberi struktur pada jawaban: Apakah sekarang The Way , sangat umum, agak umum, tidak umum, atau gila?
Jawaban:
Rekan kerja Anda salah, cara umum adalah dan selalu menempatkan kode dalam file .cpp (atau ekstensi apa pun yang Anda suka) dan deklarasi dalam header.
Kadang-kadang ada beberapa manfaat untuk menempatkan kode di header, ini dapat memungkinkan inlining lebih pintar oleh kompiler. Tetapi pada saat yang sama, ini dapat menghancurkan waktu kompilasi Anda karena semua kode harus diproses setiap kali dimasukkan oleh kompiler.
Akhirnya, sering menjengkelkan memiliki hubungan objek melingkar (kadang-kadang diinginkan) ketika semua kode adalah header.
Intinya, Anda benar, dia salah.
EDIT: Saya telah memikirkan pertanyaan Anda. Ada satu kasus di mana apa yang dia katakan itu benar. template. Banyak pustaka "modern" yang lebih baru seperti boost yang banyak menggunakan templat dan seringkali hanya "header". Namun, ini hanya boleh dilakukan ketika berhadapan dengan template karena itu adalah satu-satunya cara untuk melakukannya ketika berhadapan dengan template.
EDIT: Beberapa orang ingin sedikit lebih banyak klarifikasi, berikut beberapa pemikiran untuk menulis kode "hanya header":
Jika Anda mencari di sekitar, Anda akan melihat cukup banyak orang mencoba menemukan cara untuk mengurangi waktu kompilasi ketika berhadapan dengan dorongan. Misalnya: Cara mengurangi waktu kompilasi dengan Boost Asio , yaitu melihat kompilasi 14 file tunggal 1K dengan tambahan yang disertakan. 14s mungkin tidak tampak "meledak", tetapi tentu jauh lebih lama dari biasanya dan dapat bertambah dengan cepat. Saat berhadapan dengan proyek besar. Hanya pustaka header yang memengaruhi waktu kompilasi dengan cara yang cukup terukur. Kami hanya mentolerirnya karena peningkatan sangat berguna.
Selain itu, ada banyak hal yang tidak dapat dilakukan di header saja (bahkan meningkatkan memiliki perpustakaan Anda perlu menautkan ke bagian-bagian tertentu seperti utas, sistem file, dll). Contoh utama adalah bahwa Anda tidak dapat memiliki objek global sederhana di header hanya libs (kecuali Anda menggunakan kekejian yang merupakan singleton) karena Anda akan mengalami beberapa kesalahan definisi. CATATAN: Variabel inline C ++ 17 akan membuat contoh khusus ini bisa dilakukan di masa depan.
Sebagai poin terakhir, ketika menggunakan boost sebagai contoh kode hanya header, detail besar sering terlewatkan.
Boost adalah pustaka, bukan kode tingkat pengguna. jadi itu tidak sering berubah. Dalam kode pengguna, jika Anda meletakkan semuanya di header, setiap perubahan kecil akan menyebabkan Anda harus mengkompilasi ulang seluruh proyek. Itu buang-buang waktu yang monumental (dan bukan untuk perpustakaan yang tidak berubah dari kompilasi ke kompilasi). Ketika Anda membagi hal-hal antara header / sumber dan yang lebih baik lagi, gunakan deklarasi maju untuk mengurangi termasuk, Anda dapat menghemat waktu mengkompilasi ulang ketika ditambahkan dalam sehari.
sumber
Pada hari C ++ coders menyetujui The Way , domba akan berbaring dengan singa, Palestina akan merangkul orang Israel, dan kucing dan anjing akan diizinkan untuk menikah.
Pemisahan antara file h. Menurut saya, deklarasi berada di header dan definisi termasuk dalam file implementasi. Tapi, itu hanya kebiasaan, bukan agama.
sumber
Code in header umumnya merupakan ide yang buruk karena memaksa kompilasi ulang semua file yang menyertakan header ketika Anda mengubah kode aktual daripada deklarasi. Ini juga akan memperlambat kompilasi karena Anda harus mem-parsing kode di setiap file yang menyertakan header.
Alasan untuk memiliki kode dalam file header adalah umumnya diperlukan agar kata kunci inline berfungsi dengan benar dan ketika menggunakan template yang sedang dipasang pada file cpp lainnya.
sumber
Apa yang mungkin memberi tahu Anda rekan kerja adalah gagasan bahwa sebagian besar kode C ++ harus dibuat untuk memungkinkan kegunaan maksimum. Dan jika itu templated, maka semuanya harus ada dalam file header, sehingga kode klien dapat melihatnya dan membuat instance. Jika cukup baik untuk Boost dan STL, itu cukup baik untuk kita.
Saya tidak setuju dengan sudut pandang ini, tetapi mungkin dari mana asalnya.
sumber
Saya pikir rekan kerja Anda cerdas dan Anda juga benar.
Hal-hal berguna yang saya temukan bahwa meletakkan segala sesuatu di header adalah:
Tidak perlu menulis & menyinkronkan header dan sumber.
Strukturnya polos dan tidak ada ketergantungan melingkar yang memaksa pembuat kode untuk membuat struktur yang "lebih baik".
Mudah dibawa, mudah disematkan ke proyek baru.
Saya setuju dengan masalah waktu kompilasi, tetapi saya pikir kita harus memperhatikan bahwa:
Perubahan file sumber sangat mungkin untuk mengubah file header yang menyebabkan seluruh proyek dikompilasi ulang.
Kecepatan kompilasi jauh lebih cepat dari sebelumnya. Dan jika Anda memiliki proyek yang akan dibangun dengan waktu yang lama dan frekuensi tinggi, itu mungkin menunjukkan bahwa desain proyek Anda memiliki kekurangan. Memisahkan tugas ke dalam proyek dan modul yang berbeda dapat menghindari masalah ini.
Terakhir saya hanya ingin mendukung rekan kerja Anda, hanya dalam pandangan pribadi saya.
sumber
Seringkali saya akan memasukkan fungsi anggota yang sepele ke dalam file header, untuk memungkinkan mereka diuraikan. Tetapi untuk menempatkan seluruh tubuh kode di sana, hanya agar konsisten dengan template? Itu kacang biasa.
Ingat: Konsistensi bodoh adalah hobgoblin pikiran kecil .
sumber
A<B>
dalam file cpp, dan kemudian pengguna menginginkannyaA<C>
?Seperti kata Tuomas, tajuk Anda harus minimal. Agar lengkap saya akan sedikit memperluas.
Saya pribadi menggunakan 4 jenis file dalam
C++
proyek saya :Selanjutnya, saya memasangkan ini dengan aturan lain: Jangan mendefinisikan apa yang dapat Anda sampaikan. Meskipun tentu saja saya masuk akal di sana (menggunakan Pimpl di mana-mana cukup merepotkan).
Itu berarti bahwa saya lebih suka deklarasi maju daripada
#include
arahan di header saya setiap kali saya bisa lolos dengan mereka.Akhirnya, saya juga menggunakan aturan visibilitas: Saya membatasi cakupan simbol saya sebanyak mungkin sehingga mereka tidak mencemari cakupan luar.
Secara keseluruhan:
Penyelamat di sini adalah bahwa sebagian besar header tidak berguna: hanya diperlukan jika
typedef
atautemplate
begitu juga header implementasi;)sumber
Untuk menambahkan lebih banyak kesenangan, Anda dapat menambahkan
.ipp
file yang berisi implementasi templat (yang termasuk dalam.hpp
), sambil.hpp
berisi antarmuka.Selain kode templatized (tergantung pada proyek ini dapat berupa file mayoritas atau minoritas) ada kode normal dan di sini lebih baik untuk memisahkan deklarasi dan definisi. Berikan juga deklarasi maju jika diperlukan - ini mungkin berpengaruh pada waktu kompilasi.
sumber
Secara umum, ketika menulis kelas baru, saya akan meletakkan semua kode di kelas, jadi saya tidak perlu mencari di file lain untuk itu .. Setelah semuanya bekerja, saya memecah tubuh metode ke dalam file cpp , meninggalkan prototipe dalam file hpp.
sumber
Jika cara baru ini benar - benar Cara , kami mungkin telah berlari ke arah yang berbeda dalam proyek kami.
Karena kami mencoba menghindari semua hal yang tidak perlu dalam tajuk. Itu termasuk menghindari kaskade header. Kode dalam header mungkin perlu beberapa header lainnya untuk dimasukkan, yang akan membutuhkan header lain dan seterusnya. Jika kami terpaksa menggunakan templat, kami mencoba menghindari header sembarangan dengan hal-hal templat terlalu banyak.
Kami juga menggunakan "opaque pointer" -pattern saat berlaku.
Dengan praktik ini, kami dapat melakukan pembangunan lebih cepat daripada kebanyakan rekan kerja kami. Dan ya ... mengubah kode atau anggota kelas tidak akan menyebabkan pembangunan kembali yang besar.
sumber
Saya pribadi melakukan ini di file header saya:
Saya tidak suka mencampur kode untuk metode dengan kelas karena saya merasa sulit untuk melihat semuanya dengan cepat.
Saya tidak akan menempatkan SEMUA metode dalam file header. Compiler akan (biasanya) tidak dapat inline metode virtual dan akan (kemungkinan) hanya inline metode kecil tanpa loop (benar-benar tergantung pada kompiler).
Melakukan metode di kelas adalah valid ... tetapi dari sudut pandang yang dapat dibaca saya tidak menyukainya. Menempatkan metode di header berarti bahwa, jika memungkinkan, mereka akan mendapatkan inline.
sumber
IMHO, Dia pantas HANYA jika dia melakukan template dan / atau metaprogramming. Ada banyak alasan yang telah disebutkan bahwa Anda membatasi file header hanya untuk deklarasi. Itu hanya ... header. Jika Anda ingin memasukkan kode, kompilasi sebagai pustaka dan tautkan.
sumber
Saya meletakkan semua implementasi dari definisi kelas. Saya ingin mengeluarkan komentar doxygen dari definisi kelas.
sumber
Saya pikir itu benar-benar tidak masuk akal untuk menempatkan SEMUA definisi fungsi Anda ke file header. Mengapa? Karena file header digunakan sebagai antarmuka PUBLIC ke kelas Anda. Ini bagian luar "kotak hitam".
Ketika Anda perlu melihat kelas untuk referensi bagaimana menggunakannya, Anda harus melihat file header. File header harus memberikan daftar apa yang dapat dilakukan (dikomentari untuk menjelaskan detail tentang bagaimana menggunakan setiap fungsi), dan itu harus menyertakan daftar variabel anggota. HARUS TIDAK termasuk BAGAIMANA fungsi masing-masing individu diimplementasikan, karena itu memuat informasi yang tidak perlu dan hanya mengacaukan file header.
sumber
Bukankah itu benar-benar tergantung pada kompleksitas sistem, dan konvensi in-house?
Saat ini saya sedang mengerjakan simulator jaringan saraf yang sangat kompleks, dan gaya yang diterima yang saya harapkan untuk digunakan adalah:
Definisi kelas di classname.h
Kode kelas di classnameCode.h
kode yang dapat dieksekusi di classname.cpp
Ini memecah simulasi yang dibuat pengguna dari kelas dasar yang dibangun oleh pengembang, dan bekerja paling baik dalam situasi tersebut.
Namun, saya akan terkejut melihat orang-orang melakukan ini, katakanlah, aplikasi grafis, atau aplikasi lain yang tujuannya bukan untuk memberi pengguna basis kode.
sumber
Kode templat harus dalam tajuk saja. Terlepas dari itu semua definisi kecuali inline harus dalam .cpp. Argumen terbaik untuk ini adalah implementasi perpustakaan std yang mengikuti aturan yang sama. Anda tidak akan tidak setuju para pengembang std lib akan benar mengenai hal ini.
sumber
libstdc++
Tampaknya GCC (AFAICS) hampir tidak memasukkan apa pun disrc
& hampir semuanyainclude
, baik itu 'harus' ada di header. Jadi saya tidak berpikir ini kutipan yang akurat / berguna. Ngomong-ngomong, saya tidak berpikir stdlibs banyak model untuk kode pengguna: mereka jelas ditulis oleh coders yang sangat terampil, tetapi untuk digunakan , tidak membaca: mereka abstrak kompleksitas tinggi yang kebanyakan coders tidak perlu memikirkan , perlu jelek di_Reserved
__names
mana-mana untuk menghindari konflik dengan pengguna, komentar & spasi di bawah apa yang saya sarankan, dll. Mereka patut dicontoh dengan cara yang sempit.Saya pikir rekan kerja Anda benar selama dia tidak masuk dalam proses untuk menulis kode yang dapat dieksekusi di header. Keseimbangan yang tepat, saya pikir, adalah mengikuti jalur yang ditunjukkan oleh GNAT Ada di mana file .ads memberikan definisi antarmuka yang sangat memadai dari paket untuk penggunanya dan untuk anak-anaknya.
Ngomong-ngomong, Ted, pernahkah Anda melihat di forum ini pertanyaan terakhir tentang Ada yang mengikat ke perpustakaan CLIPS yang Anda tulis beberapa tahun lalu dan yang tidak tersedia lagi (halaman Web yang relevan sekarang ditutup). Bahkan jika dibuat untuk versi Klip lama, ikatan ini bisa menjadi contoh awal yang baik bagi seseorang yang bersedia menggunakan mesin inferensi CLIPS dalam program Ada 2012.
sumber