Rumah > Soal Jawab > teks badan
Saya sedang menulis aplikasi jadual waktu bas yang menggunakan jenis objek untuk memodelkan "dokumen" jadual waktu.
interface Timetable { name: string stops: string[] services: string[][] }
Selain jenis, saya mempunyai banyak fungsi, dan jika saya akan menggunakan mutasi, saya biasanya menulisnya sebagai kaedah di kelas. Saya kebanyakannya menggunakan Immer jadi saya tidak perlu menulis banyak sintaks lanjutan. Contohnya,
const addStop = (timetable: Timetable, stopName: string): Timetable => { return produce(timetable, (newTimetable) => { newTimetable.stops.push(stopName) }) }
Untuk menguruskan keadaan saya menggunakan Zustand dan Immer, tetapi saya rasa jika saya menggunakan Redux masalah saya akan menjadi sama. Di kedai saya, saya mempunyai pelbagai objek Jadual Waktu dan operasi yang turut menggunakan Immer untuk menetapkan semula objek Jadual Waktu yang dipilih sekarang:
updateTt: (tt, index) => { set((state) => { state.timetables[index] = tt }) }, updateThisTt: (timetable) => { set((s) => { if (s.selectedTtIdx === null) { throw new Error("no selected timetable") } s.timetables[s.selectedTtIdx] = timetable }) },
Kemudian saya memanggil fungsi perubahan data dalam komponen React dan memanggil operasi kemas kini:
const onAddStop = (name) => { updateThisTt(addStop(timetable, name)) }
Ini berfungsi, tetapi saya tidak pasti sama ada saya melakukannya dengan betul. Saya kini mempunyai dua lapisan panggilan Immer, komponen saya kini mempunyai fungsi pengubahsuaian data yang dipanggil terus dalam pengendali acaranya, dan saya tidak begitu menyukai rupa "kaedah" walaupun ia adalah pepijat kecil secara keseluruhan.
Saya menganggap:
Timetable
类型转换为一个类,将数据修改函数重写为变异方法,并设置 [immerable] = true
saya dan biarkan Immer melakukan semua kerja untuk tindakan saya. Saya telah melakukan ini, tetapi saya lebih suka berpegang pada mod rekod tidak berubah. Untuk nilainya, dokumentasi untuk Flux, Zustand atau Immer cenderung menunjukkan pilihan pertama, dan hanya sekali-sekala tiada aplikasi semudah counter = counter + 1
. Apakah cara terbaik untuk membina aplikasi menggunakan seni bina Flux?
P粉5761849332024-01-11 16:52:03
(Saya tidak biasa dengan Zusand dan Immer, tetapi mungkin saya boleh membantu...)
Selalu ada cara yang berbeza dan saya mencadangkan cara kegemaran saya di sini.
Bezakan dengan jelas antara tindakan "penjadualan" dan "mutasi" sebenar keadaan. (Mungkintambah tahap lain di antaranya).
Saya syorkan mencipta khusus fungsi "mutasi" dan bukannya yang generik, iaitu:
updateThisTt:() => { ...
,addStop: () => { ...
. Buat seberapa banyak fungsi mutasi yang anda perlukan, setiap satu dengan tujuan.
Secara konsep, hanya gunakan immer
penjana di dalam fungsi mutasi.
(Maksud saya tentang kedai. Sudah tentu, anda masih boleh menggunakan immer
untuk tujuan lain) .
Berdasarkan contoh rasmi ini :
import { produce } from 'immer' const useLushStore = create((set) => ({ lush: { forest: { contains: { a: 'bear' } } }, clearForest: () => set( produce((state) => { // <----- create the new state here state.lush.forest.contains = null }) ), })); const clearForest = useLushStore((state) => state.clearForest); clearForest();
Di dalam komponen anda kini boleh memanggil fungsi "mutator".
const onAddStop = (name) => { updateThisTt( timetable, name ); }
Jika membina keadaan baharu menjadi rumit, anda masih boleh mengekstrak beberapa fungsi "pembina". Tetapi mula-mula pertimbangkan bahagian seterusnya, "Fail Besar dan Penduaan."
Sebagai contoh, fungsi addStop
anda juga boleh dipanggil dalam mutasi Zustand:
updateThisTt: ( timetable: Timetable, stopName: string ) => { const newTimeTable = addStop( timetable, stopName ); set( ( s ) => { if( s.selectedTtIdx === null ){ throw new Error( "no selected timetable" ) } s.timetables[ s.selectedTtIdx ] = newTimeTable; }); }
Penduaan kod pastinya harus dielakkan, tetapi sentiasa ada pertukaran.
Saya tidak akan mencadangkan pendekatan khusus, tetapi ambil perhatian bahawa kod selalunya lebih banyak dibaca daripada ditulis.
Kadang-kadangSaya rasa patut menulis beberapa huruf lagi, seperti sesuatu seperti state.timetables[index]
beberapa kali,
Jika ia menjadikan tujuan kod lebih jelas. Anda perlu menilai sendiri.
Apa pun, saya syorkan meletakkan fungsi mutasi anda ke dalam fail berasingan yang tidak melakukan apa-apa lagi, Dengan cara ini, ia mungkin kelihatan lebih mudah untuk difahami daripada yang anda fikirkan.
Jika anda mempunyai fail yang sangat besar tetapi tertumpu sepenuhnya hanya pada mengubah suai keadaan , Dan strukturnya adalah konsisten, menjadikannya mudah dibaca walaupun anda perlu menatal Beberapa muka surat.
Sebagai contoh jika ia kelihatan seperti ini:
// ... //-- Add a "stop" to the time table addStop: ( timetable: Timetable, stopName: string ) => { // .. half a page of code ... }, //-- Do some other specific state change doSomethingElse: ( arg1 ) => { // .. probably one full page of code ... }, //-- Do something else again ... doSomethingAgain: () => { // .. another half page of code ... }, // ...
Juga ambil perhatian bahawa fungsi variadic ini adalah bebas sepenuhnya (atau adakah mereka? Saya harap Zustand telah menggunakan fungsi tulen di sini, bukan?) .
Ini bermakna jika ia menjadi rumit, anda juga boleh membahagikan beberapa fungsi mutasi
Pisahkan kepada fail berasingan dalam folder
Suka /store/mutators/timeTable.js
.
Tetapi anda boleh melakukannya dengan mudah pada bila-bila masa kemudian.
Anda mungkin merasakan keperluan untuk "tahap" lain Antara pengendali peristiwa dan mutator. Saya biasanya mempunyai lapisan seperti ini, tetapi saya tidak mempunyai nama yang baik untuknya. Mari kita rujuk sementara ini sebagai "Pemanggil Operasi".
Kadangkala sukar untuk memutuskan apa itu "variogram" dan apa itu "variogram" "Pemanggil tindakan".
Bagaimanapun anda boleh "membina beberapa data" di dalam "pemanggil tindakan" tetapi anda harus melakukannya
Tiada sebarang manipulasi negeri dilakukan di sini , malah dengan immer
.
Ini adalah perbezaan yang ketara dan anda mungkin tidak perlu terlalu risau tentangnya (dan mungkin terdapat pengecualian) , tetapi sebagai contoh:
Anda boleh menggunakan beberapa bahagian keadaan lama di dalam "pemanggil tindakan", contohnya:
// OK: const { language } = oldState; const newItem = { // <-- This is ok: build a new item, use some data for that. id: 2, language: language, }; addNewItemMutator( newItem ):
Tetapi anda tidak sepatutnya tidak memetakan (atau mengubah) keadaan lama kepada keadaan baharu, contohnya:
// NOT OK: const { item } = oldState; const newItem = { // <-- This kind of update belongs inside the mutator function. ...item, updatedValue: 'new', }; addNewItemMutator( newItem ):
Tiada cadangan khusus dibuat di sini, hanya contoh:
Menyalurkan banyak nilai kepada mutator boleh mengelirukan, contohnya:
const myComponent = ( props ) =>{ const { name } = props; cosnt value = useValue(); // ... const onAddNewItem: () => { const uniqueId = createUUID(); addNewItemMutator( name, value, uniqueId, Date.now() ); },
Dalam pengendali acara, anda ingin memberi tumpuan kepada perkara yang anda mahu lakukan, seperti "tambah item baharu", Tetapi jangan pertimbangkan semua hujah. Selain itu, anda mungkin memerlukan item baharu untuk perkara lain.
Atau anda boleh menulis:
const callAddItemAction = function( name, value ){ const newItem = { name: name, value: value, uniqueId: createUUID(), dateCreated: Date.now(), }; // sendSomeRequest( url, newItem ); // maybe some side effects addNewItemMutator( newItem ); } const myComponent = ( props ) =>{ const { name } = props; cosnt value = useValue(); // ... const onAddNewItem: () => { callAddItemAction( name, value ); },