State Güncellemelerinin Kuyruğa Alınması

Bir state değişkenini değiştirmek, başka bir render’ı kuyruğa alacaktır. Ancak bazen sıradaki render’ı kuyruğa almadan önce değer üzerinde birden çok işlem gerçekleştirmek isteyebilirsiniz. Bunu yapmakta, React’in toplu state güncellemelerini nasıl ele aldığını anlamak, bunu yapmak için yardımcı olacaktır.

Bunları öğreneceksiniz

  • “Batching (Toplu işleme)” nedir ve birden fazla state güncellemesi yapmak için React tarafından nasıl kullanılır
  • Aynı state değişkenine art arda birden fazla güncelleme nasıl yapılır

React state güncellemesini toplu halde (batching) yapar

“+3” butonuna tıkladığınızda,setNumber(number + 1) üç kez çağırıldığından dolayı, sayacın üç kez artırılacağını düşünebilirsiz:

import { useState } from 'react';

export default function Counter() {
  const [number, setNumber] = useState(0);

  return (
    <>
      <h1>{number}</h1>
      <button onClick={() => {
        setNumber(number + 1);
        setNumber(number + 1);
        setNumber(number + 1);
      }}>+3</button>
    </>
  )
}

Bununla birlikte, önceki bölümden hatırlayabileceğiniz gibi her render’ın state değerleri sabittir, yani setNumber(1) fonksiyonunu kaç defa çağırırsanız çağırın ilk render’da olay yöneticisindeki number değeri her zaman 0‘dır:

setNumber(0 + 1);
setNumber(0 + 1);
setNumber(0 + 1);

Ancak göze alınması gereken başka bir faktör daha vardır. React, state güncellemelerini yapmadan önce olay yöneticilerindeki tüm kodun çalışmasını bekler. Bu nedenle yeniden render işlemi yalnızca tüm bu setNumber() çağrılarından sonra gerçekleşir.

Bu size restoranda sipariş alan bir garsonu hatırlatabilir. Garson siparişinizi alırken ilk yemeğinizi söylediğiniz zaman mutfağa koşmaz! Bunun yerine, siparişinizi tamamlamanıza, üzerinde değişiklik yapmanıza izin verirler. Hatta masadaki diğer insanların da siparişini alırlar.

An elegant cursor at a restaurant places and order multiple times with React, playing the part of the waiter. After she calls setState() multiple times, the waiter writes down the last one she requested as her final order.

Rachel Lee Nabors tarafından görselleştirilmiştir.

Bu, yeniden render tetiklemeden birden çok state değişkenini (birden çok bileşende bile) güncellemenizi sağlar. Ancak bu aynı zamanda, kullanıcı arayüzünün (UI) olay yöneticiniz ve içindeki kodlar tamamlanana kadar güncellenmeyeceği anlamına gelmektedir. Batching (toplu halde) olarak da bilinen bu davranış, React uygulamanızın çok daha hızlı çalışmasını sağlar. Ayrıca React, yalnızca bazı değişkenlerin güncellendiği kafa karıştırıcı “yarı tamamlanmış” render’larla uğraşmaktan da kaçınır.

React, tıklama gibi kasıtlı yapılmış birden fazla olayı toplu olarak işlemez- her tıklama ayrı olarak işlenir. React’in bu gruplamayı genel olarak yalnızca güvenli olduğunu düşündüğü durumlarda yaptığından emin olabilirsiniz. Böylelikle, örneğin, butona ilk defa tıklamak formu devre dışı bırakıyor ise, butona ikinci defa tıklamak formu bir daha göndermez.

Aynı state’i bir sonraki render öncesinde birden fazla kez güncelleme

Bu yaygın olmayan bir kullanım durumudur ancak aynı state değişkenini bir sonraki render’dan önce birden fazla kez güncellemek isterseniz, setNumber(number + 1) gibi sonraki state değerini iletmek yerine setNumber(n => n + 1) gibi kuyrukta önceki değerine dayanarak sıradaki değeri hesaplayan bir fonksiyon iletebilirsiniz. Böylelikle React’e state değerini değiştirmek yerine “state değeri ile bir şey yap” diyebilirsiniz.

Şimdi sayacı tekrar artırmayı deneyin:

import { useState } from 'react';

export default function Counter() {
  const [number, setNumber] = useState(0);

  return (
    <>
      <h1>{number}</h1>
      <button onClick={() => {
        setNumber(n => n + 1);
        setNumber(n => n + 1);
        setNumber(n => n + 1);
      }}>+3</button>
    </>
  )
}

Burada, n => n + 1 ifadesi güncelleyici (updater) fonksiyon olarak adlandırılır. Bu fonksiyonu bir state setter’a ilettiğiniz zaman:

  1. React, bu fonksiyonu olay yöneticisindeki tüm kodlar çalıştıktan sonra işlemek üzere kuyruğa alır.
  2. Bir sonraki render esnasında React, kuyruktaki tüm işlemleri yapar ve size güncellenmiş son state’i verir.
setNumber(n => n + 1);
setNumber(n => n + 1);
setNumber(n => n + 1);

Olay yöneticisi yürütülürken React bu kod satırlarını aşağıdaki şekilde ele alır:

  1. setNumber(n => n + 1): n => n + 1 ifadesi bir fonksiyondur. React bu ifadeyi kuyruğa alır.
  2. setNumber(n => n + 1): n => n + 1 ifadesi bir fonksiyondur. React bu ifadeyi kuyruğa alır.
  3. setNumber(n => n + 1): n => n + 1 ifadesi bir fonksiyondur. React bu ifadeyi kuyruğa alır.

Sonraki render’da useState‘i çağırdığınız zaman React, kuyruktaki işlemleri yapar. Önceki number state’i 0‘dı, dolayısıyla React bunu n argümanı olarak ilk güncelleyici fonksiyona iletir. Ardından, önceki güncelleyici fonksiyonunuzun döndürdüğü değeri alır ve bu değeri bir sonraki güncelleyici fonksiyona n olarak iletir ve bu böyle devam eder:

kuyruktaki güncellemendöndürülen değer
n => n + 100 + 1 = 1
n => n + 111 + 1 = 2
n => n + 122 + 1 = 3

React 3 değerini nihai sonuç olarak saklar ve useState‘den döndürür.

Bu yüzden yukarıdaki örnekte “+3” butonuna tıklamak, değeri doğru şekilde 3 artıracaktır.

State’i değiştirdikten sonra güncellerseniz ne olur

Peki ya bu olay yöneticisi hakkında ne düşünüyorsunuz? number‘ın değeri sonraki render’da ne olur?

<button onClick={() => {
setNumber(number + 5);
setNumber(n => n + 1);
}}>
import { useState } from 'react';

export default function Counter() {
  const [number, setNumber] = useState(0);

  return (
    <>
      <h1>{number}</h1>
      <button onClick={() => {
        setNumber(number + 5);
        setNumber(n => n + 1);
      }}>Numarayı artır</button>
    </>
  )
}

Bu olay yöneticisinin React’e yapmasını söylediği adımlar şu şekildedir:

  1. setNumber(number + 5): numberın değeri 0‘dır, yani setNumber(0 + 5). React kuyruğa 5 ile değiştir” ifadesini alır.
  2. setNumber(n => n + 1): n => n + 1 ifadesi bir güncelleyici fonksiyondur. React o fonksiyonu kuyruğa alır.

Sonraki render esnasında React, state kuyruğunu ilerletir:

kuyruktaki güncellemendöndürülen değer
5 ile değiştir”0 (kullanılmamış)5
n => n + 155 + 1 = 6

React 6 değerini son sonuç olarak saklar ve useState‘den döndürür.

Not

setState(5)‘in setState(n => 5) gibi çalıştığını ancak n‘in kullanılmadığını fark etmiş olabilirsiniz!

State’i güncelledikten sonra değiştirirseniz ne olur

Bir örnek daha deneyelim. Sonraki render’da number‘ın değeri ne olacaktır?

<button onClick={() => {
setNumber(number + 5);
setNumber(n => n + 1);
setNumber(42);
}}>
import { useState } from 'react';

export default function Counter() {
  const [number, setNumber] = useState(0);

  return (
    <>
      <h1>{number}</h1>
      <button onClick={() => {
        setNumber(number + 5);
        setNumber(n => n + 1);
        setNumber(42);
      }}>Numarayı artır</button>
    </>
  )
}

Olay yönetecisi yürütülürken React bu kod satırlarını aşağıdaki şekilde ele alır:

  1. setNumber(number + 5): number‘ın değeri 0‘dır, yani setNumber(0 + 5). React kuyruğa 5 ile değiştir” ifadesini alır.
  2. setNumber(n => n + 1): n => n + 1 ifadesi bir güncelleyici fonksiyondur. React o fonksiyonu kuyruğa alır.
  3. setNumber(42): React kuyruğa 42 ile değiştir” ifadesini alır.

Sonraki render esnasında React, state kuyruğunu ilerletir:

kuyruktaki güncellemendöndürülen değer
5 ile değiştir”0 (kullanılmamış)5
n => n + 155 + 1 = 6
42 ile değiştir”6 (kullanılmamış)42

React 42 değerini nihai sonuç olarak saklar ve useState‘den döndürür.

Özetlemek gerekirse, state setter olan setNumber‘a ne ilettiğinizi şu şekilde düşünebilirsiniz:

  • Bir güncelleyici fonksiyon (örneğin n => n + 1) kuyruğa eklenir.
  • Herhangi başka bir değer (örneğin number 5) zaten kuyrukta olanları yok sayarak kuyruğa ”5 ile değiştir” ifadesini ekler.

Olay yöneticisi tamamlandıktan sonra, React yeniden render’ı tetikleyecektir. Render esnasında, React kuyruktaki işlemleri yapacaktır. Güncelleyici fonksiyonlar render etme esnasında çalışırlar yani güncelleyici fonksiyonlar saf olmalıdır ve sadece sonucu döndürmelidirler. Güncelleyici fonksiyonların içinde state’i değiştirmeyin ya da farklı yan etkiler çalıştırmayın. Strict Mode’da React, hataları bulmanıza yardımcı olmak için her güncelleyici fonksiyonu iki defa çalıştırır (ancak ikinci sonucu göz ardı eder).

Sık kullanılan adlandırmalar

Güncelleyici fonksiyon argümanını ilgili durum değişkeninin ilk harflerine göre adlandırmak yaygın olarak kullanılmaktadır:

setEnabled(e => !e);
setLastName(ln => ln.reverse());
setFriendCount(fc => fc * 2);

Daha ayrıntılı bir kod yapısını tercih ediyorsanız, başka bir yaygın kural, setEnabled(enabled => !enabled) gibi state değişkeni adını ya da setEnabled(prevEnabled => !prevEnabled) gibi bir önad kullanmaktır.

Özet

  • State’i değiştirmek mevcut render’daki değişkeni değiştirmez, yeni bir render talep eder.
  • React, olay yönetecileri çalışmayı bitirdikten sonra state güncellemelerini yapar. Bu duruma batching (toplu işleme) denir.
  • Bir olayda bazı state’leri birden çok defa güncellemek için setNumber(n => n + 1) güncelleyici fonksiyonunu kullanabilirsiniz.

Problem 1 / 2:
İstek sayacını düzeltin

Kullanıcının bir sanat eseri için aynı anda birden fazla sipariş vermesine olanak sağlayan bir sanat marketi uygulaması üzerinde çalışıyorsunuz. Kullanıcı her “Buy (Satın al)” butonuna bastığında, “Pending (Bekleniyor)” sayacı bir artmalıdır. Üç saniye sonra “Pending (Bekleniyor)” sayacı bir azalmalı ve “Completed (Tamamlandı)” sayacı bir artmalıdır.

Ancak “Bekleniyor” sayacı amaçlandığı gibi davranmaz. “Satın Al” butonuna basıldığında, sayaç -1 olur (ki bu imkansızdır!). Aynı zamanda butona iki defa hızlıca tıklarsanız, her iki sayaç da tahmin edilemeyecek bir şekilde davranır.

Sizce buna ne sebep olmakta? İki sayacı da düzeltin.

import { useState } from 'react';

export default function RequestTracker() {
  const [pending, setPending] = useState(0);
  const [completed, setCompleted] = useState(0);

  async function handleClick() {
    setPending(pending + 1);
    await delay(3000);
    setPending(pending - 1);
    setCompleted(completed + 1);
  }

  return (
    <>
      <h3>
        Bekleniyor: {pending}
      </h3>
      <h3>
        Tamamlandı: {completed}
      </h3>
      <button onClick={handleClick}>
        Satın Al     
      </button>
    </>
  );
}

function delay(ms) {
  return new Promise(resolve => {
    setTimeout(resolve, ms);
  });
}