Rumah > hujung hadapan web > tutorial js > Daripada Kekacauan kepada Kejelasan: Pendekatan Pengisytiharan kepada Komposisi Berfungsi dan Talian Paip dalam JavaScript

Daripada Kekacauan kepada Kejelasan: Pendekatan Pengisytiharan kepada Komposisi Berfungsi dan Talian Paip dalam JavaScript

Patricia Arquette
Lepaskan: 2025-01-08 18:40:41
asal
264 orang telah melayarinya

From Chaos to Clarity: A Declarative Approach to Function Composition and Pipelines in JavaScript

Jadual Kandungan

  • Seni Kod Bersih
  • Keajaiban Fungsi Tulen
  • Membina Jambatan dengan Komposisi Fungsi
  • Memperkemas Kod dengan Talian Paip
  • Menyesuaikan Talian Paip untuk Keperluan yang Berubah
  • Mengelakkan Perangkap Komposisi Fungsi
  • Perjalanan Menuju Keanggunan

Seni Kod Bersih ?

Pernahkah anda merenung kod orang lain dan berfikir, “Apakah jenis sihir ini?” Daripada menyelesaikan masalah sebenar, anda tersesat dalam labirin gelung, keadaan dan pembolehubah. Inilah perjuangan yang dihadapi semua pembangun—pertempuran abadi antara kekacauan dan kejelasan.

Kod hendaklah ditulis untuk dibaca oleh manusia, dan hanya secara kebetulan untuk dilaksanakan oleh mesin. — Harold Abelson

Tetapi jangan takut! Kod bersih bukanlah khazanah mitos yang tersembunyi di dalam penjara pembangun—ia adalah kemahiran yang boleh anda kuasai. Pada terasnya terletak pengaturcaraan deklaratif, di mana fokus beralih kepada apa kod anda lakukan, meninggalkan bagaimana di latar belakang.

Mari kita jadikan ini nyata dengan contoh. Katakan anda perlu mencari semua nombor genap dalam senarai. Berikut ialah bilangan kita yang bermula—dengan pendekatan imperatif:

const numbers = [1, 2, 3, 4, 5];
const evenNumbers = [];
for (let i = 0; i < numbers.length; i++) {
  if (numbers[i] % 2 === 0) {
    evenNumbers.push(numbers[i]);
  }
}
console.log(evenNumbers); // Output: [2, 4]
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Sudah tentu, ia berkesan. Tetapi jujurlah—ia bising: gelung manual, penjejakan indeks dan pengurusan keadaan yang tidak perlu. Sekali imbas, sukar untuk melihat apa yang sebenarnya dilakukan oleh kod tersebut. Sekarang, mari kita bandingkan dengan pendekatan deklaratif:

const numbers = [1, 2, 3, 4, 5];
const evenNumbers = numbers.filter(num => num % 2 === 0);
console.log(evenNumbers); // Output: [2, 4]
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Satu baris, tiada kekacauan—hanya niat yang jelas: “Tapis nombor genap.” Ini adalah perbezaan antara kesederhanaan dan fokus berbanding kerumitan dan hingar.

Mengapa Kod Bersih Penting ?‍?

Kod bersih bukan hanya tentang kelihatan cantik—ia juga tentang bekerja dengan lebih bijak. Enam bulan ke depan, adakah anda lebih suka bergelut melalui kebingungan logik yang mengelirukan, atau membaca kod yang secara praktikal menerangkan dirinya?

Walaupun kod imperatif mempunyai tempatnya—terutamanya apabila prestasi kritikal—kod perisytiharan selalunya menang dengan kebolehbacaan dan kemudahan penyelenggaraan.

Berikut ialah perbandingan sebelah menyebelah pantas:

Imperative Declarative
Lots of boilerplate Clean and focused
Step-by-step instructions Expresses intent clearly
Harder to refactor or extend Easier to adjust and maintain

Setelah anda menerima kod perisytiharan yang bersih, anda akan tertanya-tanya bagaimana anda berjaya tanpa kod itu. Ia adalah kunci untuk membina sistem yang boleh diramal dan boleh diselenggara—dan semuanya bermula dengan keajaiban fungsi tulen. Jadi ambil tongkat pengekodan anda (atau kopi pekat ☕) dan sertai perjalanan ke arah kod yang lebih bersih dan berkuasa. ?✨


Keajaiban Fungsi Tulen ?

Pernahkah anda menemui fungsi yang cuba melakukan segala-galanya—mengambil data, memproses input, log keluaran dan mungkin juga membancuh kopi? Binatang berbilang tugas ini mungkin kelihatan cekap, tetapi ia artifak terkutuk: rapuh, berbelit-belit dan mimpi ngeri untuk dikekalkan. Pasti ada cara yang lebih baik.

Kesederhanaan adalah prasyarat untuk kebolehpercayaan. — Edsger W. Dijkstra

Hakikat Kesucian ⚗️

Fungsi tulen adalah seperti melafazkan mantra yang direka dengan sempurna—ia sentiasa menghasilkan hasil yang sama untuk input yang sama, tanpa kesan sampingan. Sihir ini memudahkan ujian, memudahkan penyahpepijatan dan mengabstraksi kerumitan untuk memastikan kebolehgunaan semula.

Untuk melihat perbezaannya, berikut ialah fungsi yang tidak tulen:

const numbers = [1, 2, 3, 4, 5];
const evenNumbers = [];
for (let i = 0; i < numbers.length; i++) {
  if (numbers[i] % 2 === 0) {
    evenNumbers.push(numbers[i]);
  }
}
console.log(evenNumbers); // Output: [2, 4]
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Fungsi ini mengubah suai keadaan global—seperti mantra yang hilang, ia tidak boleh dipercayai dan mengecewakan. Outputnya bergantung pada pembolehubah diskaun yang berubah-ubah, menjadikan penyahpepijatan dan penggunaan semula menjadi cabaran yang membosankan.

Sekarang, mari kita cipta fungsi tulen sebaliknya:

const numbers = [1, 2, 3, 4, 5];
const evenNumbers = numbers.filter(num => num % 2 === 0);
console.log(evenNumbers); // Output: [2, 4]
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Tanpa keadaan global, fungsi ini boleh diramal dan serba lengkap. Ujian menjadi mudah dan ia sedia untuk digunakan semula atau dilanjutkan sebagai sebahagian daripada aliran kerja yang lebih besar.

Dengan memecahkan tugas kepada fungsi kecil dan tulen, anda mencipta pangkalan kod yang teguh dan menyeronokkan untuk digunakan. Jadi, pada kali seterusnya anda menulis fungsi, tanya diri anda: "Adakah mantera ini tertumpu dan boleh dipercayai—atau adakah ia akan menjadi artifak terkutuk yang bersedia untuk mencetuskan huru-hara?"


Membina Jambatan dengan Komposisi Fungsi ?

Dengan fungsi tulen di tangan, kami telah menguasai kemahiran kesederhanaan. Seperti bata Lego ?, ia serba lengkap, tetapi batu bata sahaja tidak membina istana. Keajaibannya terletak pada menggabungkannya—intipati komposisi fungsi, di mana aliran kerja menyelesaikan masalah sambil mengabstraksi butiran pelaksanaan.

Mari lihat cara ini berfungsi dengan contoh mudah: mengira jumlah troli beli-belah. Pertama, kami mentakrifkan fungsi utiliti boleh guna semula sebagai blok binaan:

let discount = 0;   

const applyDiscount = (price: number) => {
  discount += 1; // Modifies a global variable! ?
  return price - discount;
};

// Repeated calls yield inconsistent results, even with same input!
console.log(applyDiscount(100)); // Output: 99
console.log(applyDiscount(100)); // Output: 98
discount = 100;
console.log(applyDiscount(100)); // Output: -1 ?
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Kini, kami menyusun fungsi utiliti ini menjadi satu aliran kerja:

const applyDiscount = (price: number, discountRate: number) => 
  price * (1 - discountRate);

// Always consistent for the same inputs
console.log(applyDiscount(100, 0.1)); // 90
console.log(applyDiscount(100, 0.1)); // 90
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Di sini, setiap fungsi mempunyai tujuan yang jelas: menjumlahkan harga, menggunakan diskaun dan membulatkan hasilnya. Bersama-sama, mereka membentuk aliran logik di mana output satu suapan ke seterusnya. Logik domain adalah jelas—kira jumlah pembayaran dengan diskaun.

Aliran kerja ini menangkap kuasa komposisi fungsi: memfokuskan pada apa—niat di sebalik kod anda—sambil membiarkan bagaimana—butiran pelaksanaan—pudar ke latar belakang.


Memperkemas Kod dengan Talian Paip ✨

Komposisi fungsi adalah berkuasa, tetapi apabila aliran kerja berkembang, gubahan bersarang dalam boleh menjadi sukar untuk diikuti—seperti membongkar Anak patung Rusia ?. Saluran paip mengambil abstraksi lebih jauh, menawarkan jujukan linear transformasi yang mencerminkan penaakulan semula jadi.

Membina Utiliti paip Mudah ?️

Banyak perpustakaan JavaScript (hello, peminat pengaturcaraan berfungsi! ?) menawarkan utiliti saluran paip, tetapi mencipta sendiri adalah sangat mudah:

const numbers = [1, 2, 3, 4, 5];
const evenNumbers = [];
for (let i = 0; i < numbers.length; i++) {
  if (numbers[i] % 2 === 0) {
    evenNumbers.push(numbers[i]);
  }
}
console.log(evenNumbers); // Output: [2, 4]
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Utiliti ini merangkaikan operasi kepada aliran yang jelas dan progresif. Memfaktorkan semula contoh pembayaran kami sebelum ini dengan paip memberikan kami:

const numbers = [1, 2, 3, 4, 5];
const evenNumbers = numbers.filter(num => num % 2 === 0);
console.log(evenNumbers); // Output: [2, 4]
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Hasilnya hampir puitis: setiap peringkat dibina di atas yang terakhir. Kesepaduan ini bukan sahaja cantik—ia praktikal, menjadikan aliran kerja cukup intuitif sehinggakan bukan pembangun pun boleh mengikuti dan memahami perkara yang berlaku.

Perkongsian Sempurna dengan TypeScript ?

TypeScript memastikan keselamatan jenis dalam saluran paip dengan menentukan hubungan input-output yang ketat. Menggunakan lebihan fungsi, anda boleh menaip utiliti paip seperti ini:

let discount = 0;   

const applyDiscount = (price: number) => {
  discount += 1; // Modifies a global variable! ?
  return price - discount;
};

// Repeated calls yield inconsistent results, even with same input!
console.log(applyDiscount(100)); // Output: 99
console.log(applyDiscount(100)); // Output: 98
discount = 100;
console.log(applyDiscount(100)); // Output: -1 ?
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Sekilas Tentang Masa Depan ?

Walaupun mencipta utiliti anda sendiri adalah berwawasan, operator saluran paip yang dicadangkan JavaScript (|>) akan menjadikan transformasi rantaian lebih mudah dengan sintaks asli.

const applyDiscount = (price: number, discountRate: number) => 
  price * (1 - discountRate);

// Always consistent for the same inputs
console.log(applyDiscount(100, 0.1)); // 90
console.log(applyDiscount(100, 0.1)); // 90
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Saluran paip bukan sahaja menyelaraskan aliran kerja—ia mengurangkan overhed kognitif, menawarkan kejelasan dan kesederhanaan yang bergema di luar kod.


Menyesuaikan Talian Paip untuk Keperluan yang Berubah?

Dalam pembangunan perisian, keperluan boleh berubah dalam sekelip mata. Saluran paip menjadikan penyesuaian menjadi mudah—sama ada anda menambah ciri baharu, menyusun semula proses atau memperhalusi logik. Mari terokai cara saluran paip mengendalikan keperluan yang semakin berkembang dengan beberapa senario praktikal.

Menambah Pengiraan Cukai ?️

Andaikan kita perlu memasukkan cukai jualan dalam proses pembayaran. Talian paip memudahkan perkara ini—hanya tentukan langkah baharu dan letakkannya di tempat yang betul:

type CartItem = { price: number };

const roundToTwoDecimals = (value: number) =>
  Math.round(value * 100) / 100;

const calculateTotal = (cart: CartItem[]) =>
  cart.reduce((total, item) => total + item.price, 0);

const applyDiscount = (discountRate: number) => 
  (total: number) => total * (1 - discountRate);
Salin selepas log masuk
Salin selepas log masuk

Jika keperluan berubah—seperti mengenakan cukai jualan sebelum diskaun—saluran paip menyesuaikan diri dengan mudah:

// Domain-specific logic derived from reusable utility functions
const applyStandardDiscount = applyDiscount(0.2);

const checkout = (cart: CartItem[]) =>
  roundToTwoDecimals(
    applyStandardDiscount(
      calculateTotal(cart)
    )
  );

const cart: CartItem[] = [
  { price: 19.99 },
  { price: 45.5 },
  { price: 3.49 },
];

console.log(checkout(cart)); // Output: 55.18
Salin selepas log masuk
Salin selepas log masuk

Menambah Ciri Bersyarat: Diskaun Ahli ?️

Saluran paip juga boleh mengendalikan logik bersyarat dengan mudah. Bayangkan memohon diskaun tambahan untuk ahli. Mula-mula, tentukan utiliti untuk menggunakan transformasi secara bersyarat:

const pipe =
  (...fns: Function[]) =>
  (input: any) => fns.reduce((acc, fn) => fn(acc), input);
Salin selepas log masuk
Salin selepas log masuk

Seterusnya, masukkannya ke dalam saluran paip secara dinamik:

const numbers = [1, 2, 3, 4, 5];
const evenNumbers = [];
for (let i = 0; i < numbers.length; i++) {
  if (numbers[i] % 2 === 0) {
    evenNumbers.push(numbers[i]);
  }
}
console.log(evenNumbers); // Output: [2, 4]
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Fungsi identiti bertindak sebagai no-op, menjadikannya boleh digunakan semula untuk transformasi bersyarat yang lain. Fleksibiliti ini membolehkan saluran paip menyesuaikan diri dengan lancar kepada pelbagai keadaan tanpa menambahkan kerumitan pada aliran kerja.

Memanjangkan Talian Paip untuk Nyahpepijat ?

Menyahpepijat saluran paip boleh terasa rumit—seperti mencari jarum dalam timbunan jerami—melainkan anda melengkapkan diri anda dengan alatan yang betul. Helah mudah tetapi berkesan ialah memasukkan fungsi pengelogan untuk menerangi setiap langkah:

const numbers = [1, 2, 3, 4, 5];
const evenNumbers = numbers.filter(num => num % 2 === 0);
console.log(evenNumbers); // Output: [2, 4]
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Walaupun saluran paip dan komposisi fungsi menawarkan fleksibiliti yang luar biasa, memahami kebiasaan mereka memastikan anda boleh menggunakan kuasanya tanpa terjebak dalam perangkap biasa.


Mengelakkan Perangkap Komposisi Fungsi ?️

Komposisi fungsi dan saluran paip memberikan kejelasan dan keanggunan kepada kod anda, tetapi seperti mana-mana sihir yang berkuasa, ia boleh mempunyai perangkap tersembunyi. Mari kita temui mereka dan belajar cara mengelakkannya dengan mudah.

Perangkap #1: Kesan Sampingan Yang Tidak Diingini ?

Kesan sampingan boleh menyelinap masuk ke dalam gubahan anda, mengubah aliran kerja yang boleh diramal menjadi huru-hara. Mengubah suai keadaan kongsi atau bergantung pada pembolehubah luaran boleh menjadikan kod anda tidak dapat diramalkan.

let discount = 0;   

const applyDiscount = (price: number) => {
  discount += 1; // Modifies a global variable! ?
  return price - discount;
};

// Repeated calls yield inconsistent results, even with same input!
console.log(applyDiscount(100)); // Output: 99
console.log(applyDiscount(100)); // Output: 98
discount = 100;
console.log(applyDiscount(100)); // Output: -1 ?
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Pembetulan: Pastikan semua fungsi dalam saluran paip anda adalah tulen.

const applyDiscount = (price: number, discountRate: number) => 
  price * (1 - discountRate);

// Always consistent for the same inputs
console.log(applyDiscount(100, 0.1)); // 90
console.log(applyDiscount(100, 0.1)); // 90
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Perangkap #2: Talian Paip yang Menyulitkan ?

Saluran paip bagus untuk memecahkan aliran kerja yang kompleks, tetapi tindakan berlebihan boleh menyebabkan rantaian mengelirukan yang sukar diikuti.

type CartItem = { price: number };

const roundToTwoDecimals = (value: number) =>
  Math.round(value * 100) / 100;

const calculateTotal = (cart: CartItem[]) =>
  cart.reduce((total, item) => total + item.price, 0);

const applyDiscount = (discountRate: number) => 
  (total: number) => total * (1 - discountRate);
Salin selepas log masuk
Salin selepas log masuk

Pembetulan: Kumpulan langkah berkaitan ke dalam fungsi tertib lebih tinggi yang merangkumi niat.

// Domain-specific logic derived from reusable utility functions
const applyStandardDiscount = applyDiscount(0.2);

const checkout = (cart: CartItem[]) =>
  roundToTwoDecimals(
    applyStandardDiscount(
      calculateTotal(cart)
    )
  );

const cart: CartItem[] = [
  { price: 19.99 },
  { price: 45.5 },
  { price: 3.49 },
];

console.log(checkout(cart)); // Output: 55.18
Salin selepas log masuk
Salin selepas log masuk

Perangkap #3: Menyahpepijat Bintik Buta ?

Apabila menyahpepijat saluran paip, mungkin sukar untuk menentukan langkah mana yang menyebabkan isu, terutamanya dalam rantai panjang.

Pembetulan: Menyuntik fungsi pengelogan atau pemantauan untuk menjejaki keadaan perantaraan, seperti yang kita lihat sebelum ini dengan fungsi log yang mencetak mesej dan nilai pada setiap langkah.

Perangkap #4: Kehilangan Konteks dalam Kaedah Kelas ?

Apabila mengarang kaedah daripada kelas, anda mungkin kehilangan konteks yang diperlukan untuk melaksanakannya dengan betul.

const pipe =
  (...fns: Function[]) =>
  (input: any) => fns.reduce((acc, fn) => fn(acc), input);
Salin selepas log masuk
Salin selepas log masuk

Pembetulan: Gunakan .bind(this) atau fungsi anak panah untuk mengekalkan konteks.

const checkout = pipe(
  calculateTotal,
  applyStandardDiscount,
  roundToTwoDecimals
);
Salin selepas log masuk

Dengan mengambil kira perangkap ini dan mengikut amalan terbaik, anda akan memastikan komposisi dan saluran paip anda kekal berkesan kerana ia elegan, tidak kira bagaimana keperluan anda berkembang.


Perjalanan Menuju Keanggunan ?

Menguasai komposisi fungsi dan saluran paip bukan sahaja tentang menulis kod yang lebih baik—ia mengenai mengubah minda anda untuk berfikir di luar pelaksanaan. Ia mengenai mencipta sistem yang menyelesaikan masalah, membaca seperti cerita yang diceritakan dengan baik dan memberi inspirasi dengan reka bentuk abstrak dan intuitif.

Tidak Perlu Mencipta Semula Roda?

Perpustakaan seperti RxJS, Ramda dan lodash-fp menawarkan utiliti sedia pengeluaran, diuji pertempuran yang disokong oleh komuniti aktif. Ia membebaskan anda untuk menumpukan pada menyelesaikan masalah khusus domain dan bukannya bimbang tentang butiran pelaksanaan.

Takeaways untuk membimbing amalan anda ?️

  • Kod Bersih: Kod bersih bukan hanya tentang penampilan—ia tentang bekerja dengan lebih bijak. Ia mencipta penyelesaian yang memudahkan penyahpepijatan, kerjasama dan penyelenggaraan. Enam bulan ke depan, anda akan berterima kasih kepada diri sendiri kerana menulis kod yang menerangkan dirinya secara praktikal.
  • Komposisi Fungsi: Menggabungkan fungsi tulen dan fokus untuk menghasilkan aliran kerja yang elegan menangani masalah kompleks dengan jelas.
  • Saluran Paip: Kerumitan abstrak dengan membentuk logik anda menjadi aliran yang jelas dan intuitif. Selesai dengan baik, saluran paip meningkatkan produktiviti pembangun dan menjadikan aliran kerja begitu jelas sehinggakan bukan pembangun pun dapat memahaminya.

Akhirnya, kod anda adalah lebih daripada satu siri arahan—ia adalah cerita yang anda ceritakan, mantra yang anda lemparkan. Buat dengan berhati-hati, dan biarkan keanggunan membimbing perjalanan anda. ?✨

Atas ialah kandungan terperinci Daripada Kekacauan kepada Kejelasan: Pendekatan Pengisytiharan kepada Komposisi Berfungsi dan Talian Paip dalam JavaScript. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:dev.to
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Artikel terbaru oleh pengarang
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan