ذاكرة التخزين المؤقت للاستعلامات واستراتيجيات التخزين المؤقت
ذاكرة التخزين المؤقت للاستعلامات واستراتيجيات التخزين المؤقت
تناول الدرس السابق ذاكرة التخزين المؤقت من المستوى الثاني (L2)، التي تخزّن كائنات الكيانات منفردةً مُفهرَسةً بمفتاحها الأساسي. تذهب ذاكرة التخزين المؤقت للاستعلامات خطوةً أبعد: فهي تخزّن مجموعة النتائج لاستعلام مسمًّى أو معياري — تحديدًا القائمة المرتّبة من المفاتيح الأساسية المُعادة — وتُعيد استخدامها عند كل تنفيذ متطابق لاحق دون الرجوع إلى قاعدة البيانات. إنّ فهم متى تُفعّلها وأيّ استراتيجية تخزين مؤقت تمنح لكل كيان هو ما يُفرّق بين طبقة بيانات مضبوطة جيدًا وأخرى تهدر الموارد دون أن يُلاحَظ.
كيف تعمل ذاكرة التخزين المؤقت للاستعلامات
عندما تُعلّم استعلامًا بأنه قابل للتخزين المؤقت وينفَّذ للمرة الأولى، يخزّن Hibernate إدخال ذاكرة مؤقت مفتاحه عبارة الاستعلام مع قيم المعاملات المرتبطة. القيمة المخزّنة ليست كائنات الكيانات المُهيدرة، بل مجرد قائمة المفاتيح الأساسية (ولاستعلامات القيم الخام: قيم الأعمدة المباشرة). في الاستدعاء التالي بالاستعلام والمعاملات ذاتها، يسترجع Hibernate قائمة المفاتيح من ذاكرة الاستعلامات، ثم يبحث عن كل كيان في ذاكرة L2 للكيانات (أو يُحمّله من قاعدة البيانات إن لم يكن مخزّنًا).
@Cacheable، سيصل Hibernate إلى قاعدة البيانات لكل كيان في كل ضربة من ذاكرة الاستعلامات — وهذا في الغالب أسوأ من غياب التخزين المؤقت تمامًا، لأنك تدفع تكلفة البحث في الذاكرة دون الاستفادة.
فعّل ذاكرة التخزين المؤقت للاستعلامات في application.properties:
علّم استعلام JPQL بأنه قابل للتخزين المؤقت عبر @QueryHints أو ثابت تلميح JPA:
من Session أو EntityManager يمكنك أيضًا تعيين التلميح برمجيًا:
مناطق الذاكرة المؤقتة
تُخزَّن كل فئة كيان وكل استعلام قابل للتخزين في منطقة ذاكرة مؤقتة مسمّاة. الاسم الافتراضي للمنطقة في الكيان هو اسم الفئة الكامل (مثل com.example.shop.Product)، والافتراضي لنتائج الاستعلام هو default-query-results-region. يمكنك تجاوز المنطقة في استعلام معيّن:
تتيح المناطق المسمّاة ضبط سياسات TTL والإخلاء المختلفة لكل مجموعة بيانات في إعداد JCache / Ehcache / Caffeine. البيانات سريعة التغيّر (مثل مستوى المخزون الحالي) تحصل على TTL مدته 30 ثانية؛ بيانات المرجعية البطيئة (مثل رموز الدول) يمكن أن تعيش ساعةً كاملة.
اختيار استراتيجية التخزين المؤقت
يعرض Hibernate أربع استراتيجيات تزامن لذاكرة L2. اختيار الاستراتيجية الصحيحة هو أهم قرار تخزين مؤقت تتخذه لكل كيان.
READ_ONLY— الكيان لا يُحدَّث بعد أول حفظ له (مثل: جداول البحث، رموز العملات، فئات المنتجات). أعلى معدل نقل بيانات، وبدون أي تكلفة قفل. أي محاولة تحديث ترمي استثناءً. استخدمها متى أمكن.NONSTRICT_READ_WRITE— الكيان يُحدَّث أحيانًا لكن القراءة المتقادمة مقبولة لفترة وجيزة. تُبطَل الذاكرة المؤقتة بعد التحديث دون الاحتفاظ بقفل خلال الإبطال. مناسبة لعمليات الكتابة منخفضة التزامن على بيانات غير حرجة.READ_WRITE— تستخدم قفلًا ناعمًا للحفاظ على دلالات read-committed: تُقفَل الإدخال أثناء الكتابة، وترى الخيوط الأخرى قيمة قاعدة البيانات (أو لا شيء) بدلًا من القيمة المتقادمة. آمنة للكيانات المُحدَّثة تحت تزامن معتدل. تكلفة أداء طفيفة لكل كتابة.TRANSACTIONAL— ذاكرة مؤقتة معاملاتية كاملة، مندمجة مع JTA. تضمن دلالات ذاكرة مؤقتة بقراءة قابلة للتكرار عبر عقد الكلاستر. تتطلب مدير معاملات JTA ومزوّد ذاكرة مؤقتة متوافقًا مع JTA. نادرًا ما تحتاجها؛ وتضيف تعقيدًا كبيرًا.
طبّق الاستراتيجية باستخدام @Cache:
READ_ONLY لبيانات المرجعية، وREAD_WRITE للكيانات القابلة للتعديل. لا تلجأ إلى NONSTRICT_READ_WRITE إلا إذا أظهر التوصيف تنافسًا على الكتابة في كيان READ_WRITE وكان نطاقك يتحمّل فعلًا قراءات متقادمة لفترة قصيرة.
تخزين المجموعات مؤقتًا
مجموعات الكيانات (مثل مجموعات @OneToMany) لها منطقة ذاكرة مؤقتة خاصة بها ويجب تعليمها بشكل منفصل:
بدون @Cache الثاني على المجموعة، يتجاوز Hibernate المجموعة عند الكتابة في الذاكرة المؤقتة، مما يُسبّب تحميل N+1 في كل ضربة من ذاكرة الكيان الأب.
إبطال الذاكرة المؤقتة
تُبطَل منطقة ذاكرة الاستعلامات تلقائيًا عند كتابة أي كيان من نوع مشارك في الاستعلام المخزّن. هذا يعني أن معدل كتابة مرتفعًا على Product سيُبطل باستمرار كل استعلام مخزّن يتعامل مع Product، محوّلًا الذاكرة المؤقتة إلى عبء. راقب معدلات الضربة في الذاكرة المؤقتة باستخدام Spring Boot Actuator أو Micrometer قبل الالتزام بذاكرة الاستعلامات في سيناريو الكتابة المكثّفة.
لإخلاء منطقة معينة برمجيًا:
متى لا تستخدم ذاكرة التخزين المؤقت للاستعلامات
يستحق تفعيل ذاكرة الاستعلامات فقط عندما:
- يُنفَّذ الاستعلام ذاته بالمعاملات ذاتها بشكل متكرر.
- تتغيّر البيانات الأساسية نادرًا نسبةً إلى تكرار القراءة.
- مجموعة النتائج صغيرة بشكل معقول (آلاف الصفوف، لا ملايين).
الخلاصة
تخزّن ذاكرة التخزين المؤقت للاستعلامات قوائم مفاتيح نتائج الاستعلامات، وهي فعّالة فقط عند اقترانها بذاكرة L2 للكيانات مُهيَّأة بشكل صحيح. اختر READ_ONLY للبيانات المرجعية غير القابلة للتغيير، وREAD_WRITE للكيانات القابلة للتعديل بأمان، واحتفظ بـ TRANSACTIONAL للبيئات الموزّعة مع JTA. علّم المجموعات صراحةً أو ستواجه تحميل N+1 عند ضربات الذاكرة المؤقتة. استخدم مقاييس Actuator للتحقق من معدل الضربة قبل التخزين المؤقت وبعده، وتذكّر أن الجداول عالية الكتابة تجعل ذاكرة الاستعلامات عبئًا لا أصلًا.