JAVA COLLECTIONS PERFORMANSI
"LinkedList ortadan ekleme için hızlıdır, ArrayList ise random access için." — Java dünyasında neredeyse herkesin ezberlediği bu cümle, modern CPU mimarilerinde büyük ölçüde yanlıştır. Big-O notasyonu size O(1) ile O(n) farkını söyler ama RAM'den L1 cache'e veri çekmenin maliyetini söylemez. Gerçekte ArrayList, çoğu pratik senaryoda LinkedList'i hem random access hem de ortadan ekleme/silme işlemlerinde geride bırakır. Bunun nedeni teorik karmaşıklık değil, cache locality denilen donanımsal gerçekliktir.
LinkedList Hakkındaki En Büyük Yanılgı
Java derslerinde standart anlatım şudur: "LinkedList düğümleri pointer ile bağlıdır, dolayısıyla ortadan ekleme O(1)'dir." Bu cümlenin gizlediği şey, "ortayı bulmak" için O(n) traversal gerektiğidir. Yani list.add(5000, x) çağrısı, 5000 düğümü tek tek dolaşıp pointer'ları takip etmek anlamına gelir. ArrayList'te ise bu işlem, ardışık bellek bloğunda System.arraycopy ile yapılır ve CPU bunu blok bazlı SIMD instruction'larla işler.
Sonuç: 10.000 elemanlı bir listede ortadan ekleme yapacaksanız, ArrayList LinkedList'ten 5-10 kat daha hızlıdır. Bu, JMH benchmark'larıyla defalarca doğrulanmış bir gerçektir.
Cache Locality: Asıl Belirleyici Faktör
Modern CPU'lar RAM'den veri çekerken tek byte değil, 64 byte'lık cache line halinde çeker. ArrayList elemanları ardışık bellekte tutulduğundan, bir elemanı okurken yanındaki 7-15 eleman da L1 cache'e gelir. LinkedList'te ise her düğüm heap'in rastgele bir yerinde durur — her erişim potansiyel bir cache miss'tir.

Bir cache miss yaklaşık 100-300 CPU cycle'a mal olur. L1 hit ise 4 cycle. Yani LinkedList traversal'ı, ArrayList traversal'ına göre teorik olarak aynı O(n) olsa da pratikte 25-75 kat daha yavaş çalışabilir. Bu fark, ölçek büyüdükçe sertleşir.
ArrayList'i Gerçekten Yavaşlatan Şey
ArrayList'in gerçek zayıf noktası ortadan ekleme değildir — başa eklemedir. add(0, element) her seferinde tüm diziyi bir slot kaydırır. Eğer iş yükünüz sürekli başa eklemeyse:
- ArrayDeque: Çift uçlu kuyruk, başa/sona O(1) ekleme, ardışık bellek avantajı korunur
- LinkedList: Sadece başa/sona iterator ile ekleme + sıralı traversal yapacaksanız mantıklı
- ArrayList + ters çevirme: Sona ekleyip sonda iterate etmek çoğu zaman en hızlısıdır
Pratikte LinkedList'in tek meşru kullanım alanı kalmıştır: Queue veya Deque arayüzü ile FIFO/LIFO işlemleri. Ama burada bile ArrayDeque neredeyse her zaman daha iyi seçimdir.
HashMap: O(1) Sözünün Küçük Yazısı
HashMap'in get ve put işlemleri ortalamada O(1)'dir — ama bu söz iki koşula bağlıdır: iyi bir hash fonksiyonu ve uygun bir load factor. Kötü hash dağılımı, bucket'ların lineer listeye (Java 8+ ile belirli eşikten sonra dengeli ağaca) dönüşmesine yol açar ve performans O(n) ya da O(log n)'e iner; bu iç yapının davranışı ve eşik değerleri resmi API dokümantasyonunda ayrıntılandırılmıştır.
HashMap'in iç yapısını ve diğer JVM seviyesi performans konularını daha derinlemesine öğrenmek için Java performans ve JVM eğitimi içeriğinden yararlanabilirsiniz.
Boyutlandırma ve Resize Maliyeti
ArrayList default kapasitesi 10'dur ve dolduğunda 1.5x büyür. HashMap default 16 bucket ile başlar ve load factor 0.75'i aştığında 2x rehash olur. Eğer veri boyutunuzu önceden biliyorsanız:
- ArrayList için
new ArrayList<>(expectedSize)kullanın — gereksiz resize'ı önler - HashMap için
new HashMap<>(expectedSize / 0.75 + 1)ile başlatın — rehash maliyetinden kurtulursunuz - Çok büyük koleksiyonlarda
trimToSize()ile fazlalık belleği geri verin
Resize işlemi ArrayList için O(n), HashMap için O(n) (tüm bucket'ların yeniden hash'lenmesi) maliyetindedir. Sık tetiklendiğinde toplam karmaşıklık amortized analizden sapar.
Karar Matrisi: Hangi Senaryoda Hangisi?
Pratik bir rehber:
- Sıralı erişim + index ile arama: ArrayList — neredeyse her zaman
- Sık başa/sona ekleme (queue/stack): ArrayDeque
- Key-value lookup: HashMap (sıra önemsizse), LinkedHashMap (insertion order korunmalıysa), TreeMap (sıralı iterasyon gerekiyorsa)
- Thread-safe lookup: ConcurrentHashMap — Hashtable veya synchronizedMap değil
- Unique elemanlar: HashSet (sırasız), LinkedHashSet (insertion order), TreeSet (sıralı)

Benchmark'a Güvenin, Sezgiye Değil
Java koleksiyon performansında "şu hep daha hızlıdır" diyebileceğiniz bir kural yoktur. JIT compiler, escape analysis, branch prediction ve cache davranışı; aynı kodun farklı veri boyutlarında farklı sıralamalar üretmesine neden olur. Bir kararı performansa dayandırıyorsanız, JMH ile ölçün — özellikle koleksiyon seçiminizi etkileyecek kadar kritik bir sıcak yolda (hot path) çalışıyorsa.
İleri seviye JVM tuning, garbage collector seçimi ve benchmark metodolojisi gibi konularda derinleşmek isteyenler Java performans eğitim içeriğini inceleyebilirsiniz.
Sonuç olarak, "LinkedList ortadan ekleme için iyidir" cümlesini unutun. Çoğu modern Java kodunda ArrayList ve HashMap, hayal ettiğinizden çok daha geniş bir senaryo yelpazesinde optimal seçimdir. Donanımı anlamadan algoritma karmaşıklığını anlamak, eksik kalan bir bilgidir.



