[{"data":1,"prerenderedAt":11281},["ShallowReactive",2],{"navigation_docs":3,"-java-pr2-dependency-injection-guice":3135,"-java-pr2-dependency-injection-guice-surround":11276},[4,1669,1826,2280,2461,2668,2790,2840,2897,2931,3057,3094,3131],{"title":5,"icon":6,"path":7,"stem":8,"children":9},"C#","i-devicon-csharp","\u002Fcsharp","01.csharp",[10,13,60,90,120,202,219,253,379,404,457,650,1346,1636,1665],{"title":11,"path":7,"stem":12},"C# та .NET","01.csharp\u002Findex",{"title":14,"icon":15,"path":16,"stem":17,"children":18,"page":59},"Fundamentals","i-lucide-book-open","\u002Fcsharp\u002Ffundamentals","01.csharp\u002F01.fundamentals",[19,23,27,31,35,39,43,47,51,55],{"title":20,"path":21,"stem":22},"Вступ до екосистеми .NET","\u002Fcsharp\u002Ffundamentals\u002Fintroduction-to-ecosystem","01.csharp\u002F01.fundamentals\u002F01.introduction-to-ecosystem",{"title":24,"path":25,"stem":26},"Структура програми на C#","\u002Fcsharp\u002Ffundamentals\u002Fprogram-structure","01.csharp\u002F01.fundamentals\u002F02.program-structure",{"title":28,"path":29,"stem":30},"Змінні та Типи Даних","\u002Fcsharp\u002Ffundamentals\u002Fvariables-data-types","01.csharp\u002F01.fundamentals\u002F03.variables-data-types",{"title":32,"path":33,"stem":34},"Масиви","\u002Fcsharp\u002Ffundamentals\u002Farrays","01.csharp\u002F01.fundamentals\u002F04.arrays",{"title":36,"path":37,"stem":38},"Strings & Text Handling","\u002Fcsharp\u002Ffundamentals\u002Fstrings-text-handling","01.csharp\u002F01.fundamentals\u002F05.strings-text-handling",{"title":40,"path":41,"stem":42},"Дати і Час","\u002Fcsharp\u002Ffundamentals\u002Fdates-time-handling","01.csharp\u002F01.fundamentals\u002F06.dates-time-handling",{"title":44,"path":45,"stem":46},"Потік Керування","\u002Fcsharp\u002Ffundamentals\u002Fcontrol-flow","01.csharp\u002F01.fundamentals\u002F07.control-flow",{"title":48,"path":49,"stem":50},"Методи","\u002Fcsharp\u002Ffundamentals\u002Fmethods","01.csharp\u002F01.fundamentals\u002F08.methods",{"title":52,"path":53,"stem":54},"Основи Відлагодження","\u002Fcsharp\u002Ffundamentals\u002Fdebugging-basics","01.csharp\u002F01.fundamentals\u002F09.debugging-basics",{"title":56,"path":57,"stem":58},"Інтерактивна Консоль (Classic)","\u002Fcsharp\u002Ffundamentals\u002Finteractive-console","01.csharp\u002F01.fundamentals\u002F10.interactive-console",false,{"title":61,"icon":62,"path":63,"stem":64,"children":65,"page":59},"OOP","i-lucide-box","\u002Fcsharp\u002Foop","01.csharp\u002F02.oop",[66,70,74,78,82,86],{"title":67,"path":68,"stem":69},"Package Management (Управління Пакетами)","\u002Fcsharp\u002Foop\u002Fpackage-management","01.csharp\u002F02.oop\u002F01.package-management",{"title":71,"path":72,"stem":73},"Класи та Об'єкти","\u002Fcsharp\u002Foop\u002Fclasses-objects","01.csharp\u002F02.oop\u002F02.classes-objects",{"title":75,"path":76,"stem":77},"Властивості та Поля","\u002Fcsharp\u002Foop\u002Fproperties-fields","01.csharp\u002F02.oop\u002F03.properties-fields",{"title":79,"path":80,"stem":81},"Стовпи ООП","\u002Fcsharp\u002Foop\u002Foop-pillars","01.csharp\u002F02.oop\u002F04.oop-pillars",{"title":83,"path":84,"stem":85},"Advanced Types","\u002Fcsharp\u002Foop\u002Fadvanced-types","01.csharp\u002F02.oop\u002F05.advanced-types",{"title":87,"path":88,"stem":89},"Namespaces (Простори Імен)","\u002Fcsharp\u002Foop\u002Fnamespaces","01.csharp\u002F02.oop\u002F06.namespaces",{"title":91,"icon":92,"path":93,"stem":94,"children":95,"page":59},"Advanced Core","i-lucide-zap","\u002Fcsharp\u002Fadvanced-core","01.csharp\u002F03.advanced-core",[96,100,104,108,112,116],{"title":97,"path":98,"stem":99},"Generics (Узагальнення)","\u002Fcsharp\u002Fadvanced-core\u002Fgenerics","01.csharp\u002F03.advanced-core\u002F01.generics",{"title":101,"path":102,"stem":103},"Делегати, Події та Лямбда-вирази","\u002Fcsharp\u002Fadvanced-core\u002Fdelegates-events-lambdas","01.csharp\u002F03.advanced-core\u002F02.delegates-events-lambdas",{"title":105,"path":106,"stem":107},"Interfaces Deep Dive (Інтерфейси: Поглиблений Розгляд)","\u002Fcsharp\u002Fadvanced-core\u002Finterfaces-deep-dive","01.csharp\u002F03.advanced-core\u002F03.interfaces-deep-dive",{"title":109,"path":110,"stem":111},"Обробка Винятків","\u002Fcsharp\u002Fadvanced-core\u002Fexception-handling","01.csharp\u002F03.advanced-core\u002F04.exception-handling",{"title":113,"path":114,"stem":115},"Pattern Matching","\u002Fcsharp\u002Fadvanced-core\u002Fpattern-matching","01.csharp\u002F03.advanced-core\u002F05.pattern-matching",{"title":117,"path":118,"stem":119},"Додаткові Можливості C#","\u002Fcsharp\u002Fadvanced-core\u002Fadditional-features","01.csharp\u002F03.advanced-core\u002F06.additional-features",{"title":121,"icon":122,"path":123,"stem":124,"children":125,"page":59},"Architecture Best Practices","i-lucide-building-2","\u002Fcsharp\u002Farchitecture-best-practices","01.csharp\u002F04.architecture-best-practices",[126,130,149,153,157,161,165,169],{"title":127,"path":128,"stem":129},"Software Design Principles (Частина 1)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fsoftware-design-principles","01.csharp\u002F04.architecture-best-practices\u002F01.software-design-principles",{"title":131,"icon":132,"path":133,"stem":134,"children":135,"page":59},"Design Patterns","i-lucide-folder","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns",[136],{"title":137,"icon":132,"path":138,"stem":139,"children":140,"page":59},"Creational","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns\u002Fcreational","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns\u002Fcreational",[141,145],{"title":142,"path":143,"stem":144},"Singleton (Одинак)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns\u002Fcreational\u002Fsingleton","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns\u002Fcreational\u002F01.singleton",{"title":146,"path":147,"stem":148},"Builder (Будівельник)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns\u002Fcreational\u002Fbuilder","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns\u002Fcreational\u002F02.builder",{"title":150,"path":151,"stem":152},"Building Professional CLIs","\u002Fcsharp\u002Farchitecture-best-practices\u002Fbuilding-professional-clis","01.csharp\u002F04.architecture-best-practices\u002F03.building-professional-clis",{"title":154,"path":155,"stem":156},"Validation & Flow Control","\u002Fcsharp\u002Farchitecture-best-practices\u002Fvalidation-flow-control","01.csharp\u002F04.architecture-best-practices\u002F04.validation-flow-control",{"title":158,"path":159,"stem":160},"The Modern .NET Host (Microsoft.Extensions)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fmodern-dotnet-host","01.csharp\u002F04.architecture-best-practices\u002F05.modern-dotnet-host",{"title":162,"path":163,"stem":164},"Data Mapper: Repository та DAO патерни (Частина 1)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdata-mapper-part1","01.csharp\u002F04.architecture-best-practices\u002F06.data-mapper-part1",{"title":166,"path":167,"stem":168},"Data Mapper: Repository та DAO патерни (Частина 2)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdata-mapper-part2","01.csharp\u002F04.architecture-best-practices\u002F07.data-mapper-part2",{"title":170,"icon":132,"path":171,"stem":172,"children":173,"page":59},"Di Ioc","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc",[174,178,182,186,190,194,198],{"title":175,"path":176,"stem":177},"Проблема залежностей та Інверсія Контролю","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fthe-dependency-problem","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F01.the-dependency-problem",{"title":179,"path":180,"stem":181},"Будуємо власний Service Container","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fbuild-your-own-container","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F02.build-your-own-container",{"title":183,"path":184,"stem":185},"Service Locator: Паттерн та Анти-паттерн","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fservice-locator-pattern","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F03.service-locator-pattern",{"title":187,"path":188,"stem":189},"Паттерни Dependency Injection","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fdependency-injection-patterns","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F04.dependency-injection-patterns",{"title":191,"path":192,"stem":193},"Microsoft DI: IServiceCollection та IServiceProvider","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fmicrosoft-di-deep-dive","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F05.microsoft-di-deep-dive",{"title":195,"path":196,"stem":197},"Service Lifetimes та Scopes","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fservice-lifetimes-and-scopes","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F06.service-lifetimes-and-scopes",{"title":199,"path":200,"stem":201},"DI Анти-паттерни та Найкращі Практики","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fdi-anti-patterns-and-best-practices","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F07.di-anti-patterns-and-best-practices",{"title":203,"icon":132,"path":204,"stem":205,"children":206,"page":59},"Standard Library","\u002Fcsharp\u002Fstandard-library","01.csharp\u002F05.standard-library",[207,211,215],{"title":208,"path":209,"stem":210},"Collections (Колекції)","\u002Fcsharp\u002Fstandard-library\u002Fcollections","01.csharp\u002F05.standard-library\u002F01.collections",{"title":212,"path":213,"stem":214},"High Performance Types (Високопродуктивні Типи)","\u002Fcsharp\u002Fstandard-library\u002Fhigh-performance-types","01.csharp\u002F05.standard-library\u002F02.high-performance-types",{"title":216,"path":217,"stem":218},"LINQ (Language Integrated Query)","\u002Fcsharp\u002Fstandard-library\u002Flinq","01.csharp\u002F05.standard-library\u002F03.linq",{"title":220,"icon":221,"path":222,"stem":223,"children":224,"page":59},"System Internals Concurrency","i-lucide-server","\u002Fcsharp\u002Fsystem-internals-concurrency","01.csharp\u002F06.system-internals-concurrency",[225,229,233,237,241,245,249],{"title":226,"path":227,"stem":228},"Memory Management","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fmemory-management","01.csharp\u002F06.system-internals-concurrency\u002F01.memory-management",{"title":230,"path":231,"stem":232},"Reflection API: System.Type та Метадані","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Freflection-fundamentals","01.csharp\u002F06.system-internals-concurrency\u002F02.reflection-fundamentals",{"title":234,"path":235,"stem":236},"Attributes та Dynamic Language Runtime","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fattributes-dynamic","01.csharp\u002F06.system-internals-concurrency\u002F03.attributes-dynamic",{"title":238,"path":239,"stem":240},"Expression Trees: Швидка Альтернатива Рефлексії","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fexpression-trees-compiled","01.csharp\u002F06.system-internals-concurrency\u002F04.expression-trees-compiled",{"title":242,"path":243,"stem":244},"Source Generators: Compile-Time Code Generation","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fsource-generators","01.csharp\u002F06.system-internals-concurrency\u002F05.source-generators",{"title":246,"path":247,"stem":248},"Multithreading Fundamentals","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fmultithreading-fundamentals","01.csharp\u002F06.system-internals-concurrency\u002F06.multithreading-fundamentals",{"title":250,"path":251,"stem":252},"Synchronization Primitives","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fsynchronization-primitives","01.csharp\u002F06.system-internals-concurrency\u002F07.synchronization-primitives",{"title":254,"icon":255,"path":256,"stem":257,"children":258,"page":59},"System Programming Windows","i-lucide-cpu","\u002Fcsharp\u002Fsystem-programming-windows","01.csharp\u002F07.system-programming-windows",[259,263,267,271,275,279,283,287,291,295,299,303,307,311,315,319,323,327,331,335,339,343,347,351,355,359,363,367,371,375],{"title":260,"path":261,"stem":262},"Як Працює Операційна Система","\u002Fcsharp\u002Fsystem-programming-windows\u002Fhow-os-works","01.csharp\u002F07.system-programming-windows\u002F01.how-os-works",{"title":264,"path":265,"stem":266},"Процеси в .NET — API та Запуск","\u002Fcsharp\u002Fsystem-programming-windows\u002Fprocesses-in-dotnet","01.csharp\u002F07.system-programming-windows\u002F02.processes-in-dotnet",{"title":268,"path":269,"stem":270},"Процеси в .NET — IPC та Міжпроцесна Комунікація","\u002Fcsharp\u002Fsystem-programming-windows\u002F02a.processes-ipc","01.csharp\u002F07.system-programming-windows\u002F02a.processes-ipc",{"title":272,"path":273,"stem":274},"Application Domains та Збірки — AppDomain і AssemblyLoadContext","\u002Fcsharp\u002Fsystem-programming-windows\u002Fappdomains-assemblies","01.csharp\u002F07.system-programming-windows\u002F03.appdomains-assemblies",{"title":276,"path":277,"stem":278},"Application Domains та Збірки — Plug-in Система з Hot-Reload","\u002Fcsharp\u002Fsystem-programming-windows\u002F03a.appdomains-plugin-system","01.csharp\u002F07.system-programming-windows\u002F03a.appdomains-plugin-system",{"title":280,"path":281,"stem":282},"Потоки — Основи та API Thread","\u002Fcsharp\u002Fsystem-programming-windows\u002Fthread-fundamentals","01.csharp\u002F07.system-programming-windows\u002F04.thread-fundamentals",{"title":284,"path":285,"stem":286},"Потоки — Lifecycle, Пріоритети та Безпечне Завершення","\u002Fcsharp\u002Fsystem-programming-windows\u002F04a.thread-lifecycle-priorities","01.csharp\u002F07.system-programming-windows\u002F04a.thread-lifecycle-priorities",{"title":288,"path":289,"stem":290},"Проблеми Спільного Стану — Race Condition та Data Race","\u002Fcsharp\u002Fsystem-programming-windows\u002Fshared-state-problems","01.csharp\u002F07.system-programming-windows\u002F05.shared-state-problems",{"title":292,"path":293,"stem":294},"Проблеми Спільного Стану — Memory Model та volatile","\u002Fcsharp\u002Fsystem-programming-windows\u002F05a.shared-state-memory-model","01.csharp\u002F07.system-programming-windows\u002F05a.shared-state-memory-model",{"title":296,"path":297,"stem":298},"Синхронізація — Monitor, lock та еволюція примітивів","\u002Fcsharp\u002Fsystem-programming-windows\u002Fsynchronization-fundamentals","01.csharp\u002F07.system-programming-windows\u002F06.synchronization-fundamentals",{"title":300,"path":301,"stem":302},"Синхронізація — Наскрізний Приклад та Deadlock Detection","\u002Fcsharp\u002Fsystem-programming-windows\u002F06a.synchronization-walkthrough","01.csharp\u002F07.system-programming-windows\u002F06a.synchronization-walkthrough",{"title":304,"path":305,"stem":306},"Синхронізація — Mutex, Semaphore та Event-Based Primitives","\u002Fcsharp\u002Fsystem-programming-windows\u002Fsynchronization-advanced","01.csharp\u002F07.system-programming-windows\u002F07.synchronization-advanced",{"title":308,"path":309,"stem":310},"Синхронізація — Interlocked, Volatile та Lock-Free Структури","\u002Fcsharp\u002Fsystem-programming-windows\u002F07a.synchronization-advanced-walkthrough","01.csharp\u002F07.system-programming-windows\u002F07a.synchronization-advanced-walkthrough",{"title":312,"path":313,"stem":314},"Interlocked, CAS та Lock-Free Структури","\u002Fcsharp\u002Fsystem-programming-windows\u002Finterlocked-cas-lockfree","01.csharp\u002F07.system-programming-windows\u002F08.interlocked-cas-lockfree",{"title":316,"path":317,"stem":318},"Volatile, Memory Model та Spinning","\u002Fcsharp\u002Fsystem-programming-windows\u002F08a.volatile-memory-model","01.csharp\u002F07.system-programming-windows\u002F08a.volatile-memory-model",{"title":320,"path":321,"stem":322},"ThreadPool — Пул Потоків для Ефективного Виконання","\u002Fcsharp\u002Fsystem-programming-windows\u002Fthread-pool","01.csharp\u002F07.system-programming-windows\u002F09.thread-pool",{"title":324,"path":325,"stem":326},"ThreadPool — Просунуті Сценарії та Внутрішня Будова","\u002Fcsharp\u002Fsystem-programming-windows\u002F09a.thread-pool-advanced","01.csharp\u002F07.system-programming-windows\u002F09a.thread-pool-advanced",{"title":328,"path":329,"stem":330},"Concurrent та Immutable Collections","\u002Fcsharp\u002Fsystem-programming-windows\u002Fconcurrent-collections","01.csharp\u002F07.system-programming-windows\u002F10.concurrent-collections",{"title":332,"path":333,"stem":334},"TPL, Task та Композиція — Від Thread до Task","\u002Fcsharp\u002Fsystem-programming-windows\u002Ftpl-parallel-plinq","01.csharp\u002F07.system-programming-windows\u002F11.tpl-parallel-plinq",{"title":336,"path":337,"stem":338},"Parallel Class та PLINQ — Data Parallelism","\u002Fcsharp\u002Fsystem-programming-windows\u002F11a.tpl-parallel-plinq-advanced","01.csharp\u002F07.system-programming-windows\u002F11a.tpl-parallel-plinq-advanced",{"title":340,"path":341,"stem":342},"Async\u002FAwait — Фундамент Асинхронного Програмування","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-fundamentals","01.csharp\u002F07.system-programming-windows\u002F12.async-fundamentals",{"title":344,"path":345,"stem":346},"SynchronizationContext та ConfigureAwait — Контекст Виконання","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-context-configureawait","01.csharp\u002F07.system-programming-windows\u002F13.async-context-configureawait",{"title":348,"path":349,"stem":350},"Async — Просунуті Паттерни","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-advanced","01.csharp\u002F07.system-programming-windows\u002F14.async-advanced",{"title":352,"path":353,"stem":354},"System.Threading.Channels — Async Producer-Consumer","\u002Fcsharp\u002Fsystem-programming-windows\u002Fchannels","01.csharp\u002F07.system-programming-windows\u002F15.channels",{"title":356,"path":357,"stem":358},"Асинхронна Синхронізація","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-synchronization","01.csharp\u002F07.system-programming-windows\u002F16.async-synchronization",{"title":360,"path":361,"stem":362},"Unsafe Code та Вказівники","\u002Fcsharp\u002Fsystem-programming-windows\u002Funsafe-code","01.csharp\u002F07.system-programming-windows\u002F17.unsafe-code",{"title":364,"path":365,"stem":366},"P\u002FInvoke та Windows API — Міст між .NET та Native Code","\u002Fcsharp\u002Fsystem-programming-windows\u002Fpinvoke-winapi","01.csharp\u002F07.system-programming-windows\u002F18.pinvoke-winapi",{"title":368,"path":369,"stem":370},"Реєстр Windows — Центральна База Конфігурації Системи","\u002Fcsharp\u002Fsystem-programming-windows\u002Fwindows-registry","01.csharp\u002F07.system-programming-windows\u002F19.windows-registry",{"title":372,"path":373,"stem":374},"Windows Hooks, Hotkeys та Services — Глибока Інтеграція з ОС","\u002Fcsharp\u002Fsystem-programming-windows\u002Fwindows-hooks-services","01.csharp\u002F07.system-programming-windows\u002F20.windows-hooks-services",{"title":376,"path":377,"stem":378},"Системне Програмування C# (Windows) — 07.system-programming-windows","\u002Fcsharp\u002Fsystem-programming-windows\u002Fimplementation_plan","01.csharp\u002F07.system-programming-windows\u002Fimplementation_plan",{"title":380,"icon":132,"path":381,"stem":382,"children":383,"page":59},"Io","\u002Fcsharp\u002Fio","01.csharp\u002F08.io",[384,388,392,396,400],{"title":385,"path":386,"stem":387},"8.1.1. Основи роботи з файловою системою","\u002Fcsharp\u002Fio\u002Ffile-system-basics","01.csharp\u002F08.io\u002F01.file-system-basics",{"title":389,"path":390,"stem":391},"8.1.2. Потоки (Streams) та Серіалізація Даних","\u002Fcsharp\u002Fio\u002Fstreams-serialization","01.csharp\u002F08.io\u002F02.streams-serialization",{"title":393,"path":394,"stem":395},"8.2.1. JSON Serialization з System.Text.Json","\u002Fcsharp\u002Fio\u002Fjson-serialization","01.csharp\u002F08.io\u002F03.json-serialization",{"title":397,"path":398,"stem":399},"8.2.2. XML Serialization та LINQ to XML","\u002Fcsharp\u002Fio\u002Fxml-serialization","01.csharp\u002F08.io\u002F04.xml-serialization",{"title":401,"path":402,"stem":403},"8.2.3. Binary Serialization: MessagePack та Protocol Buffers","\u002Fcsharp\u002Fio\u002Fbinary-serialization","01.csharp\u002F08.io\u002F05.binary-serialization",{"title":405,"icon":132,"path":406,"stem":407,"children":408,"page":59},"Ado Net","\u002Fcsharp\u002Fado-net","01.csharp\u002F09.ado-net",[409,413,417,421,425,429,433,437,441,445,449,453],{"title":410,"path":411,"stem":412},"9.1. Введення в ADO.NET","\u002Fcsharp\u002Fado-net\u002Fintroduction-to-adonet","01.csharp\u002F09.ado-net\u002F01.introduction-to-adonet",{"title":414,"path":415,"stem":416},"9.2. Клас DbConnection — з'єднання з базою даних","\u002Fcsharp\u002Fado-net\u002Fconnection","01.csharp\u002F09.ado-net\u002F02.connection",{"title":418,"path":419,"stem":420},"9.3. Клас DbCommand — виконання SQL-запитів","\u002Fcsharp\u002Fado-net\u002Fcommand-and-queries","01.csharp\u002F09.ado-net\u002F03.command-and-queries",{"title":422,"path":423,"stem":424},"9.4. Клас DbDataReader — ефективне читання даних","\u002Fcsharp\u002Fado-net\u002Fdatareader","01.csharp\u002F09.ado-net\u002F04.datareader",{"title":426,"path":427,"stem":428},"9.5. Параметризовані запити та захист від SQL Injection","\u002Fcsharp\u002Fado-net\u002Fparameters-and-sql-injection","01.csharp\u002F09.ado-net\u002F05.parameters-and-sql-injection",{"title":430,"path":431,"stem":432},"9.6. Транзакції в ADO.NET","\u002Fcsharp\u002Fado-net\u002Ftransactions","01.csharp\u002F09.ado-net\u002F06.transactions",{"title":434,"path":435,"stem":436},"9.7. DbProviderFactory — провайдер-незалежний код","\u002Fcsharp\u002Fado-net\u002Fprovider-factory","01.csharp\u002F09.ado-net\u002F07.provider-factory",{"title":438,"path":439,"stem":440},"9.8. Асинхронний доступ до даних","\u002Fcsharp\u002Fado-net\u002Fasync-data-access","01.csharp\u002F09.ado-net\u002F08.async-data-access",{"title":442,"path":443,"stem":444},"9.9. Від'єднаний режим: DataSet, DataTable, DataRow","\u002Fcsharp\u002Fado-net\u002Fdisconnected-mode-dataset","01.csharp\u002F09.ado-net\u002F09.disconnected-mode-dataset",{"title":446,"path":447,"stem":448},"9.10. DataAdapter — міст між DataSet та базою даних","\u002Fcsharp\u002Fado-net\u002Fdata-adapter","01.csharp\u002F09.ado-net\u002F10.data-adapter",{"title":450,"path":451,"stem":452},"9.11. Data Mapper та Repository: Архітектура доступу до даних","\u002Fcsharp\u002Fado-net\u002Fdata-mapper-repository","01.csharp\u002F09.ado-net\u002F11.data-mapper-repository",{"title":454,"path":455,"stem":456},"9.12. Identity Map, Unit of Work та Specification Pattern","\u002Fcsharp\u002Fado-net\u002Fadvanced-patterns","01.csharp\u002F09.ado-net\u002F12.advanced-patterns",{"title":458,"icon":255,"path":459,"stem":460,"children":461,"page":59},"Ef Core","\u002Fcsharp\u002Fef-core","01.csharp\u002F10.ef-core",[462,466,470,474,478,482,486,490,494,498,502,506,510,514,518,522,526,532,538,542,546,550,554,558,562,566,570,574,578,582,586,590,594,598,602,606,610,614,618,622,626,630,634,638,642,646],{"title":463,"path":464,"stem":465},"Що таке ORM? Від SQL до об'єктів","\u002Fcsharp\u002Fef-core\u002Fwhat-is-orm","01.csharp\u002F10.ef-core\u002F01.what-is-orm",{"title":467,"path":468,"stem":469},"Перший проєкт — від нуля до CRUD","\u002Fcsharp\u002Fef-core\u002Ffirst-project","01.csharp\u002F10.ef-core\u002F02.first-project",{"title":471,"path":472,"stem":473},"DbContext — Серце EF Core","\u002Fcsharp\u002Fef-core\u002Fdbcontext-deep-dive","01.csharp\u002F10.ef-core\u002F03.dbcontext-deep-dive",{"title":475,"path":476,"stem":477},"Провайдери баз даних — Архітектура та Вибір СУБД","\u002Fcsharp\u002Fef-core\u002Fdatabase-providers","01.csharp\u002F10.ef-core\u002F04.database-providers",{"title":479,"path":480,"stem":481},"Конвенції EF Core — Магія без конфігурації","\u002Fcsharp\u002Fef-core\u002Fconventions","01.csharp\u002F10.ef-core\u002F05.conventions",{"title":483,"path":484,"stem":485},"Fluent API та Data Annotations — Явна конфігурація моделі","\u002Fcsharp\u002Fef-core\u002Ffluent-api-vs-annotations","01.csharp\u002F10.ef-core\u002F06.fluent-api-vs-annotations",{"title":487,"path":488,"stem":489},"Зв'язки — One-to-One та One-to-Many","\u002Fcsharp\u002Fef-core\u002Frelationships-basics","01.csharp\u002F10.ef-core\u002F07.relationships-basics",{"title":491,"path":492,"stem":493},"Зв'язки Advanced — Many-to-Many та Складні Сценарії","\u002Fcsharp\u002Fef-core\u002Frelationships-advanced","01.csharp\u002F10.ef-core\u002F08.relationships-advanced",{"title":495,"path":496,"stem":497},"Властивості — Типи, Конвертери, Компаратори (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fproperty-configuration-part1","01.csharp\u002F10.ef-core\u002F09.property-configuration-part1",{"title":499,"path":500,"stem":501},"Властивості — Value Comparers, Generators, Shadow Properties (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fproperty-configuration-part2","01.csharp\u002F10.ef-core\u002F09.property-configuration-part2",{"title":503,"path":504,"stem":505},"Складні типи — Owned Types та Complex Types (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fcomplex-types-owned-part1","01.csharp\u002F10.ef-core\u002F10.complex-types-owned-part1",{"title":507,"path":508,"stem":509},"Складні типи — Complex Types, Keyless Entities, Порівняння (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fcomplex-types-owned-part2","01.csharp\u002F10.ef-core\u002F10.complex-types-owned-part2",{"title":511,"path":512,"stem":513},"JSON Columns — Складні дані у JSON (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fjson-columns-part1","01.csharp\u002F10.ef-core\u002F11.json-columns-part1",{"title":515,"path":516,"stem":517},"JSON Columns — Value Comparers, Індекси, Провайдери (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fjson-columns-part2","01.csharp\u002F10.ef-core\u002F11.json-columns-part2",{"title":519,"path":520,"stem":521},"Успадкування — Абстрактні класи та TPH (Частина 1)","\u002Fcsharp\u002Fef-core\u002Finheritance-part1","01.csharp\u002F10.ef-core\u002F12.inheritance-part1",{"title":523,"path":524,"stem":525},"Успадкування — TPT, TPC та Порівняння Стратегій (Частина 2)","\u002Fcsharp\u002Fef-core\u002Finheritance-part2","01.csharp\u002F10.ef-core\u002F12.inheritance-part2",{"title":527,"path":528,"stem":529,"children":530},"Індекси, Обмеження та Схема (Частина 1)","\u002Fcsharp\u002Fef-core\u002Findexes-constraints-part1","01.csharp\u002F10.ef-core\u002F13.indexes-constraints-part1",[531],{"title":527,"path":528,"stem":529},{"title":533,"path":534,"stem":535,"children":536},"Індекси, Обмеження та Схема (Частина 2)","\u002Fcsharp\u002Fef-core\u002Findexes-constraints-part2","01.csharp\u002F10.ef-core\u002F13.indexes-constraints-part2",[537],{"title":533,"path":534,"stem":535},{"title":539,"path":540,"stem":541},"Seed Data — Початкові Дані (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fseeding-part1","01.csharp\u002F10.ef-core\u002F14.seeding-part1",{"title":543,"path":544,"stem":545},"Seed Data — SQL-скрипти, Bogus та Стратегії (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fseeding-part2","01.csharp\u002F10.ef-core\u002F14.seeding-part2",{"title":547,"path":548,"stem":549},"Global Query Filters — Глобальні Фільтри (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fglobal-query-filters-part1","01.csharp\u002F10.ef-core\u002F15.global-query-filters-part1",{"title":551,"path":552,"stem":553},"Global Query Filters — Підводні камені та Інтеграція (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fglobal-query-filters-part2","01.csharp\u002F10.ef-core\u002F15.global-query-filters-part2",{"title":555,"path":556,"stem":557},"LINQ-запити в EF Core (Частина 1)","\u002Fcsharp\u002Fef-core\u002Flinq-queries-part1","01.csharp\u002F10.ef-core\u002F16.linq-queries-part1",{"title":559,"path":560,"stem":561},"LINQ-запити в EF Core (Частина 2)","\u002Fcsharp\u002Fef-core\u002Flinq-queries-part2","01.csharp\u002F10.ef-core\u002F16.linq-queries-part2",{"title":563,"path":564,"stem":565},"Завантаження Пов'язаних Даних (Частина 1)","\u002Fcsharp\u002Fef-core\u002Floading-related-data-part1","01.csharp\u002F10.ef-core\u002F17.loading-related-data-part1",{"title":567,"path":568,"stem":569},"Завантаження Пов'язаних Даних (Частина 2)","\u002Fcsharp\u002Fef-core\u002Floading-related-data-part2","01.csharp\u002F10.ef-core\u002F17.loading-related-data-part2",{"title":571,"path":572,"stem":573},"Raw SQL, Views та Stored Procedures (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fraw-sql-part1","01.csharp\u002F10.ef-core\u002F18.raw-sql-part1",{"title":575,"path":576,"stem":577},"Raw SQL — Stored Procedures, DbFunction та Bulk Operations (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fraw-sql-part2","01.csharp\u002F10.ef-core\u002F18.raw-sql-part2",{"title":579,"path":580,"stem":581},"Продвинуті Запити — Compiled Queries, Bulk та Оптимізація (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fadvanced-queries-part1","01.csharp\u002F10.ef-core\u002F19.advanced-queries-part1",{"title":583,"path":584,"stem":585},"Продвинуті Запити — Query Tags, Bulk та Interceptors (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fadvanced-queries-part2","01.csharp\u002F10.ef-core\u002F19.advanced-queries-part2",{"title":587,"path":588,"stem":589},"Change Tracker — Відстеження Змін (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fchange-tracking-part1","01.csharp\u002F10.ef-core\u002F20.change-tracking-part1",{"title":591,"path":592,"stem":593},"Change Tracker — Графи Об'єктів та Disconnected (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fchange-tracking-part2","01.csharp\u002F10.ef-core\u002F20.change-tracking-part2",{"title":595,"path":596,"stem":597},"Збереження Даних та Транзакції (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fsaving-data-part1","01.csharp\u002F10.ef-core\u002F21.saving-data-part1",{"title":599,"path":600,"stem":601},"Збереження Даних — Concurrency та Outbox (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fsaving-data-part2","01.csharp\u002F10.ef-core\u002F21.saving-data-part2",{"title":603,"path":604,"stem":605},"Конкурентність та Блокування (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fconcurrency-part1","01.csharp\u002F10.ef-core\u002F22.concurrency-part1",{"title":607,"path":608,"stem":609},"Конкурентність — Дедлоки та Queue Processing (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fconcurrency-part2","01.csharp\u002F10.ef-core\u002F22.concurrency-part2",{"title":611,"path":612,"stem":613},"Міграції в EF Core — Основи (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fmigrations-basics-part1","01.csharp\u002F10.ef-core\u002F23.migrations-basics-part1",{"title":615,"path":616,"stem":617},"Міграції в EF Core — Основи (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fmigrations-basics-part2","01.csharp\u002F10.ef-core\u002F23.migrations-basics-part2",{"title":619,"path":620,"stem":621},"Міграції — Просунуті Сценарії (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fmigrations-advanced-part1","01.csharp\u002F10.ef-core\u002F24.migrations-advanced-part1",{"title":623,"path":624,"stem":625},"Міграції — Просунуті Сценарії (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fmigrations-advanced-part2","01.csharp\u002F10.ef-core\u002F24.migrations-advanced-part2",{"title":627,"path":628,"stem":629},"Управління Схемою та Database-First (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fschema-management-part1","01.csharp\u002F10.ef-core\u002F25.schema-management-part1",{"title":631,"path":632,"stem":633},"Управління Схемою та Database-First (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fschema-management-part2","01.csharp\u002F10.ef-core\u002F25.schema-management-part2",{"title":635,"path":636,"stem":637},"Продуктивність EF Core — Основи (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fperformance-fundamentals-part1","01.csharp\u002F10.ef-core\u002F26.performance-fundamentals-part1",{"title":639,"path":640,"stem":641},"Interceptors в EF Core (Частина 1)","\u002Fcsharp\u002Fef-core\u002Finterceptors-part1","01.csharp\u002F10.ef-core\u002F29.interceptors-part1",{"title":643,"path":644,"stem":645},"Interceptors в EF Core — Connection, Transaction та Materialization (Частина 2)","\u002Fcsharp\u002Fef-core\u002Finterceptors-part2","01.csharp\u002F10.ef-core\u002F29.interceptors-part2",{"title":647,"path":648,"stem":649},"План вивчення Entity Framework Core — Повний курс","\u002Fcsharp\u002Fef-core\u002Fimplementation_plan","01.csharp\u002F10.ef-core\u002Fimplementation_plan",{"title":651,"icon":652,"path":653,"stem":654,"children":655,"page":59},"ASP.NET","i-devicon-dotnetcore","\u002Fcsharp\u002Faspnet","01.csharp\u002F11.aspnet",[656,730,791,869,927,941,967,1057,1111,1182,1212,1289],{"title":657,"icon":658,"path":659,"stem":660,"children":661,"page":59},"Minimal API","i-lucide-network","\u002Fcsharp\u002Faspnet\u002Fminimal-api","01.csharp\u002F11.aspnet\u002F01.minimal-api",[662,666,670,674,678,682,686,690,694,698,702,706,710,714,718,722,726],{"title":663,"path":664,"stem":665},"Вступ до ASP.NET та еволюція фреймворку","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fintroduction","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F01.introduction",{"title":667,"path":668,"stem":669},"Перший додаток на ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Ffirst-application","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F02.first-application",{"title":671,"path":672,"stem":673},"WebApplication, Builder та Dependency Injection","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fwebapplication-builder","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F03.webapplication-builder",{"title":675,"path":676,"stem":677},"Конвеєр запитів та Middleware","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Frequest-pipeline-middleware","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F04.request-pipeline-middleware",{"title":679,"path":680,"stem":681},"Маршрутизація в ASP.NET Core: Основи","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Frouting-basics","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F05.routing-basics",{"title":683,"path":684,"stem":685},"Маршрутизація в ASP.NET Core: Розширені можливості","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Frouting-advanced","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F06.routing-advanced",{"title":687,"path":688,"stem":689},"Статичні файли в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fstatic-files","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F07.static-files",{"title":691,"path":692,"stem":693},"Статичні Активи: MapStaticAssets (ASP.NET Core 9.0)","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fstatic-assets","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F08.static-assets",{"title":695,"path":696,"stem":697},"Конфігурація в ASP.NET Core: Основи","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fconfiguration-fundamentals","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F09.configuration-fundamentals",{"title":699,"path":700,"stem":701},"Конфігурація: Паттерн Options","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fconfiguration-options","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F10.configuration-options",{"title":703,"path":704,"stem":705},"Логування в ASP.NET Core: Основи","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Flogging-basics","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F11.logging-basics",{"title":707,"path":708,"stem":709},"Логування: Serilog та Middleware","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Flogging-advanced","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F12.logging-advanced",{"title":711,"path":712,"stem":713},"Управління станом: HttpContext.Items та Cookies","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fstate-management","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F13.state-management",{"title":715,"path":716,"stem":717},"Стан сесії: Sessions","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fsession-state","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F14.session-state",{"title":719,"path":720,"stem":721},"Структура проєкту: від хаосу до архітектури","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fproject-structure","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F15.project-structure",{"title":723,"path":724,"stem":725},"Scalar у Minimal API: повний проєкт і Fluent OpenAPI","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fscalar-openapi-fluent","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F16.scalar-openapi-fluent",{"title":727,"path":728,"stem":729},"Swagger \u002F Swashbuckle у Minimal API: окремий класичний шлях","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fswagger-swashbuckle","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F17.swagger-swashbuckle",{"title":731,"icon":658,"path":732,"stem":733,"children":734,"page":59},"API","\u002Fcsharp\u002Faspnet\u002Fapi","01.csharp\u002F11.aspnet\u002F02.api",[735,739,743,747,751,755,759,763,767,771,775,779,783,787],{"title":736,"path":737,"stem":738},"Що таке API. Клієнт-серверна архітектура","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fwhat-is-api","01.csharp\u002F11.aspnet\u002F02.api\u002F01.what-is-api",{"title":740,"path":741,"stem":742},"Формати даних: JSON, XML, TOML та бінарні формати","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fdata-formats","01.csharp\u002F11.aspnet\u002F02.api\u002F02.data-formats",{"title":744,"path":745,"stem":746},"Парадигми API та концепція REST","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-paradigms-rest","01.csharp\u002F11.aspnet\u002F02.api\u002F03.api-paradigms-rest",{"title":748,"path":749,"stem":750},"HTTP-методи, статус-коди та заголовки","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fhttp-methods-status-codes","01.csharp\u002F11.aspnet\u002F02.api\u002F04.http-methods-status-codes",{"title":752,"path":753,"stem":754},"Організація HTTP API за принципами REST","\u002Fcsharp\u002Faspnet\u002Fapi\u002Frest-organizing","01.csharp\u002F11.aspnet\u002F02.api\u002F05.rest-organizing",{"title":756,"path":757,"stem":758},"Номенклатура URL та CRUD-операції","\u002Fcsharp\u002Faspnet\u002Fapi\u002Furl-nomenclature-crud","01.csharp\u002F11.aspnet\u002F02.api\u002F06.url-nomenclature-crud",{"title":760,"path":761,"stem":762},"Правила дизайну: іменування та стандарти","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-design-naming","01.csharp\u002F11.aspnet\u002F02.api\u002F07.api-design-naming",{"title":764,"path":765,"stem":766},"Валідація, ліміти та обробка помилок","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-design-validation","01.csharp\u002F11.aspnet\u002F02.api\u002F08.api-design-validation",{"title":768,"path":769,"stem":770},"Обробка помилок у Minimal API","\u002Fcsharp\u002Faspnet\u002Fapi\u002Ferror-handling-http","01.csharp\u002F11.aspnet\u002F02.api\u002F09.error-handling-http",{"title":772,"path":773,"stem":774},"Ідемпотентність та синхронізація стану","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fidempotency-sync","01.csharp\u002F11.aspnet\u002F02.api\u002F10.idempotency-sync",{"title":776,"path":777,"stem":778},"Пагінація та організація списків","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fpagination-lists","01.csharp\u002F11.aspnet\u002F02.api\u002F11.pagination-lists",{"title":780,"path":781,"stem":782},"Безпека API, кешування та інтернаціоналізація","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fsecurity-auth","01.csharp\u002F11.aspnet\u002F02.api\u002F12.security-auth",{"title":784,"path":785,"stem":786},"Процес проєктування API та документування","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-design-process","01.csharp\u002F11.aspnet\u002F02.api\u002F13.api-design-process",{"title":788,"path":789,"stem":790},"OpenAPI: контракт, специфікація та документація API","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fopenapi","01.csharp\u002F11.aspnet\u002F02.api\u002F14.openapi",{"title":792,"icon":793,"path":794,"stem":795,"children":796,"page":59},"Auth","i-lucide-shield-check","\u002Fcsharp\u002Faspnet\u002Fauth","01.csharp\u002F11.aspnet\u002F03.auth",[797,801,805,809,813,817,821,825,829,833,837,841,845,849,853,857,861,865],{"title":798,"path":799,"stem":800},"Основи аутентифікації та авторизації","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fauth-fundamentals","01.csharp\u002F11.aspnet\u002F03.auth\u002F01.auth-fundamentals",{"title":802,"path":803,"stem":804},"JWT-аутентифікація","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fjwt-authentication","01.csharp\u002F11.aspnet\u002F03.auth\u002F02.jwt-authentication",{"title":806,"path":807,"stem":808},"Авторизація: ролі, політики та resource-based доступ","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fauthorization-policies","01.csharp\u002F11.aspnet\u002F03.auth\u002F03.authorization-policies",{"title":810,"path":811,"stem":812},"Cookie-аутентифікація та ASP.NET Core Identity","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fcookie-auth-identity","01.csharp\u002F11.aspnet\u002F03.auth\u002F04.cookie-auth-identity",{"title":814,"path":815,"stem":816},"JWT + Refresh Tokens (HttpOnly Cookie)","\u002Fcsharp\u002Faspnet\u002Fauth\u002F04b.identity-auth-jwt","01.csharp\u002F11.aspnet\u002F03.auth\u002F04b.identity-auth-jwt",{"title":818,"path":819,"stem":820},"Identity: Підтвердження Email та Скидання Пароля","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fidentity-email-confirmation","01.csharp\u002F11.aspnet\u002F03.auth\u002F05.identity-email-confirmation",{"title":822,"path":823,"stem":824},"Identity: Двофакторна Аутентифікація (2FA)","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fidentity-two-factor","01.csharp\u002F11.aspnet\u002F03.auth\u002F06.identity-two-factor",{"title":826,"path":827,"stem":828},"Identity: Внутрішня Архітектура та Кастомізація","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fidentity-internals","01.csharp\u002F11.aspnet\u002F03.auth\u002F07.identity-internals",{"title":830,"path":831,"stem":832},"OAuth 2.0 та зовнішні провайдери","\u002Fcsharp\u002Faspnet\u002Fauth\u002Foauth-external-providers","01.csharp\u002F11.aspnet\u002F03.auth\u002F08.oauth-external-providers",{"title":834,"path":835,"stem":836},"Безпека на практиці: CORS, HTTPS та захист від атак","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fsecurity-hardening","01.csharp\u002F11.aspnet\u002F03.auth\u002F09.security-hardening",{"title":838,"path":839,"stem":840},"Теорія OAuth 2.0: Поняття, Аналогії та Флоу","\u002Fcsharp\u002Faspnet\u002Fauth\u002Foauth-theory","01.csharp\u002F11.aspnet\u002F03.auth\u002F10.oauth-theory",{"title":842,"path":843,"stem":844},"OIDC, OAuth 2.0 та Keycloak в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Foidc-keycloak","01.csharp\u002F11.aspnet\u002F03.auth\u002F10.oidc-keycloak",{"title":846,"path":847,"stem":848},"API Keys аутентифікація в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fapi-keys","01.csharp\u002F11.aspnet\u002F03.auth\u002F11.api-keys",{"title":850,"path":851,"stem":852},"Rate Limiting та Throttling в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Frate-limiting","01.csharp\u002F11.aspnet\u002F03.auth\u002F12.rate-limiting",{"title":854,"path":855,"stem":856},"Refresh Token Rotation в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Frefresh-token-rotation","01.csharp\u002F11.aspnet\u002F03.auth\u002F13.refresh-token-rotation",{"title":858,"path":859,"stem":860},"Certificate Authentication та mTLS в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fcertificate-auth","01.csharp\u002F11.aspnet\u002F03.auth\u002F14.certificate-auth",{"title":862,"path":863,"stem":864},"RBAC, ABAC та ReBAC в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Frbac-abac-rebac","01.csharp\u002F11.aspnet\u002F03.auth\u002F15.rbac-abac-rebac",{"title":866,"path":867,"stem":868},"Multi-tenancy та ізоляція даних в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fmulti-tenancy","01.csharp\u002F11.aspnet\u002F03.auth\u002F16.multi-tenancy",{"title":870,"icon":871,"path":872,"stem":873,"children":874,"page":59},"Нотифікації","i-lucide-bell","\u002Fcsharp\u002Faspnet\u002Fnotifications","01.csharp\u002F11.aspnet\u002F04.notifications",[875,879,883,887,891,895,899,903,907,911,915,919,923],{"title":876,"path":877,"stem":878},"In-App нотифікації через базу даних","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fin-app-database-notifications","01.csharp\u002F11.aspnet\u002F04.notifications\u002F01.in-app-database-notifications",{"title":880,"path":881,"stem":882},"Polling: Регулярний запит оновлень","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fpolling","01.csharp\u002F11.aspnet\u002F04.notifications\u002F02.polling",{"title":884,"path":885,"stem":886},"Server-Sent Events: Однострімовий push від сервера","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fserver-sent-events","01.csharp\u002F11.aspnet\u002F04.notifications\u002F03.server-sent-events",{"title":888,"path":889,"stem":890},"WebSockets: Двостороннє з'єднання в реальному часі","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fwebsockets","01.csharp\u002F11.aspnet\u002F04.notifications\u002F04.websockets",{"title":892,"path":893,"stem":894},"SignalR: Абстракція над транспортами реального часу","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fsignalr","01.csharp\u002F11.aspnet\u002F04.notifications\u002F05.signalr",{"title":896,"path":897,"stem":898},"Background Services: Фонові задачі в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fbackground-services","01.csharp\u002F11.aspnet\u002F04.notifications\u002F06.background-services",{"title":900,"path":901,"stem":902},"Web Push нотифікації","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fweb-push","01.csharp\u002F11.aspnet\u002F04.notifications\u002F07.web-push",{"title":904,"path":905,"stem":906},"Email нотифікації","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Femail-notifications","01.csharp\u002F11.aspnet\u002F04.notifications\u002F08.email-notifications",{"title":908,"path":909,"stem":910},"Порівняння підходів: Як вибрати правильну технологію нотифікацій","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fchoosing-the-right-approach","01.csharp\u002F11.aspnet\u002F04.notifications\u002F09.choosing-the-right-approach",{"title":912,"path":913,"stem":914},"Hangfire: Надійне планування фонових задач","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fhangfire","01.csharp\u002F11.aspnet\u002F04.notifications\u002F10.hangfire",{"title":916,"path":917,"stem":918},"Практика: Конвертація зображень у WebP через Hangfire","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fhangfire-image-webp","01.csharp\u002F11.aspnet\u002F04.notifications\u002F11.hangfire-image-webp",{"title":920,"path":921,"stem":922},"Практика: Підготовка відео до HLS-стрімінгу через Hangfire","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fhangfire-video-hls","01.csharp\u002F11.aspnet\u002F04.notifications\u002F12.hangfire-video-hls",{"title":924,"path":925,"stem":926},"Telegram-нотифікації: від одного повідомлення до масових розсилок і мульти-канального підходу","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Ftelegram-notifications","01.csharp\u002F11.aspnet\u002F04.notifications\u002F13.telegram-notifications",{"title":928,"icon":929,"path":930,"stem":931,"children":932,"page":59},"Інтернаціоналізація","i-lucide-languages","\u002Fcsharp\u002Faspnet\u002Fi18n","01.csharp\u002F11.aspnet\u002F05.i18n",[933,937],{"title":934,"path":935,"stem":936},"Інтернаціоналізація (i18n) у Minimal API: від A до Я","\u002Fcsharp\u002Faspnet\u002Fi18n\u002Finternationalization","01.csharp\u002F11.aspnet\u002F05.i18n\u002F01.internationalization",{"title":938,"path":939,"stem":940},"Humanizer: людиномовні рядки у .NET","\u002Fcsharp\u002Faspnet\u002Fi18n\u002Fhumanizer","01.csharp\u002F11.aspnet\u002F05.i18n\u002F02.humanizer",{"title":942,"icon":943,"path":944,"stem":945,"children":946,"page":59},"Кешування","i-lucide-layers","\u002Fcsharp\u002Faspnet\u002Fcaching","01.csharp\u002F11.aspnet\u002F06.caching",[947,951,955,959,963],{"title":948,"path":949,"stem":950},"Огляд кешування: чотири рівні і коли що обирати","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fcaching","01.csharp\u002F11.aspnet\u002F06.caching\u002F01.caching",{"title":952,"path":953,"stem":954},"IMemoryCache: кеш в оперативній пам'яті","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fmemory-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F02.memory-cache",{"title":956,"path":957,"stem":958},"IDistributedCache і Redis: розподілений кеш","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fdistributed-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F03.distributed-cache",{"title":960,"path":961,"stem":962},"Response Cache: HTTP-кешування через Cache-Control","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fresponse-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F04.response-cache",{"title":964,"path":965,"stem":966},"Output Cache: серверний кеш HTTP-відповідей (.NET 7+)","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Foutput-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F05.output-cache",{"title":968,"icon":969,"path":970,"stem":971,"children":972,"page":59},"Тестування","i-lucide-test-tube","\u002Fcsharp\u002Faspnet\u002Ftesting","01.csharp\u002F11.aspnet\u002F07.testing",[973,977,981,985,989,993,997,1001,1005,1009,1013,1017,1021,1025,1029,1033,1037,1041,1045,1049,1053],{"title":974,"path":975,"stem":976},"Що таке тестування? Від інтуїції до науки","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fwhat-is-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F01.what-is-testing",{"title":978,"path":979,"stem":980},"Піраміда тестування — Стратегія, а не Догма","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftesting-pyramid","01.csharp\u002F11.aspnet\u002F07.testing\u002F02.testing-pyramid",{"title":982,"path":983,"stem":984},"Дві Школи Тестування — Лондон проти Детройту","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftesting-schools","01.csharp\u002F11.aspnet\u002F07.testing\u002F03.testing-schools",{"title":986,"path":987,"stem":988},"TDD та BDD — Тести як Дизайн-інструмент","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftdd-and-bdd","01.csharp\u002F11.aspnet\u002F07.testing\u002F04.tdd-and-bdd",{"title":990,"path":991,"stem":992},"Що саме тестувати — Техніки аналізу та Циклomatична складність","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fwhat-to-test","01.csharp\u002F11.aspnet\u002F07.testing\u002F05.what-to-test",{"title":994,"path":995,"stem":996},"Тестові Фреймворки — Навіщо вони і що всередині","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftest-frameworks","01.csharp\u002F11.aspnet\u002F07.testing\u002F06.test-frameworks",{"title":998,"path":999,"stem":1000},"xUnit — Факти, Теорії та Lifecycle тестів","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fxunit-basics","01.csharp\u002F11.aspnet\u002F07.testing\u002F07.xunit-basics",{"title":1002,"path":1003,"stem":1004},"xUnit Advanced — Fixtures, Кастомізація та Розширення","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fxunit-advanced","01.csharp\u002F11.aspnet\u002F07.testing\u002F08.xunit-advanced",{"title":1006,"path":1007,"stem":1008},"Moq — Глибоке занурення в мокування","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fmocking-with-moq","01.csharp\u002F11.aspnet\u002F07.testing\u002F09.mocking-with-moq",{"title":1010,"path":1011,"stem":1012},"Тестування Баз Даних — EF Core, SQLite та Testcontainers","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fdatabase-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F10.database-testing",{"title":1014,"path":1015,"stem":1016},"Integration Testing — Частина 1 [Теорія та WebApplicationFactory]","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fintegration-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F11.integration-testing",{"title":1018,"path":1019,"stem":1020},"Інтеграційне тестування — Практика","\u002Fcsharp\u002Faspnet\u002Ftesting\u002F11a.integration-testing-practice","01.csharp\u002F11.aspnet\u002F07.testing\u002F11a.integration-testing-practice",{"title":1022,"path":1023,"stem":1024},"Integration Testing — Частина 2 [Просунуті Сценарії та Testcontainers]","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fintegration-testing-advanced","01.csharp\u002F11.aspnet\u002F07.testing\u002F12.integration-testing-advanced",{"title":1026,"path":1027,"stem":1028},"Професійний Postman: Колекції, Змінні та GitHub Інтеграція","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fpostman-professional","01.csharp\u002F11.aspnet\u002F07.testing\u002F13.postman-professional",{"title":1030,"path":1031,"stem":1032},"HttpClient у Тестах Частина 1: Архітектура та MockHttpMessageHandler","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fhttpclient-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F14.httpclient-testing",{"title":1034,"path":1035,"stem":1036},"HttpClient у Тестах Частина 2: WireMock.Net та Resilience","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fwiremock-net","01.csharp\u002F11.aspnet\u002F07.testing\u002F15.wiremock-net",{"title":1038,"path":1039,"stem":1040},"Патерни та Анти-патерни Тестування: Test Smells","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftesting-patterns","01.csharp\u002F11.aspnet\u002F07.testing\u002F16.testing-patterns",{"title":1042,"path":1043,"stem":1044},"Просунуті інструменти: Time, Snapshots та Властивості","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fadvanced-testing-tools","01.csharp\u002F11.aspnet\u002F07.testing\u002F17.advanced-testing-tools",{"title":1046,"path":1047,"stem":1048},"Тестування Архітектури з NetArchTest","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Farchitecture-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F18.architecture-testing",{"title":1050,"path":1051,"stem":1052},"Тестування Продуктивності: BenchmarkDotNet, NBomber та k6","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fperformance-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F19.performance-testing",{"title":1054,"path":1055,"stem":1056},"Залишок плану для курсу \"Тестування ASP.NET Minimal API\"","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fremaining_plan","01.csharp\u002F11.aspnet\u002F07.testing\u002Fremaining_plan",{"title":1058,"icon":1059,"path":1060,"stem":1061,"children":1062,"page":59},"Платежі","i-lucide-credit-card","\u002Fcsharp\u002Faspnet\u002Fpayments","01.csharp\u002F11.aspnet\u002F08.payments",[1063,1067,1071,1075,1079,1083,1087,1091,1095,1099,1103,1107],{"title":1064,"path":1065,"stem":1066},"Основи платіжної інфраструктури","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpayment-fundamentals","01.csharp\u002F11.aspnet\u002F08.payments\u002F01.payment-fundamentals",{"title":1068,"path":1069,"stem":1070},"Методи оплати в Україні","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpayment-methods-ukraine","01.csharp\u002F11.aspnet\u002F08.payments\u002F02.payment-methods-ukraine",{"title":1072,"path":1073,"stem":1074},"PCI DSS та безпека платежів","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpci-dss-security","01.csharp\u002F11.aspnet\u002F08.payments\u002F03.pci-dss-security",{"title":1076,"path":1077,"stem":1078},"Архітектура платіжної підсистеми","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpayment-architecture","01.csharp\u002F11.aspnet\u002F08.payments\u002F04.payment-architecture",{"title":1080,"path":1081,"stem":1082},"Інтеграція LiqPay (ПриватБанк)","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fliqpay-integration","01.csharp\u002F11.aspnet\u002F08.payments\u002F05.liqpay-integration",{"title":1084,"path":1085,"stem":1086},"Інтеграція Monobank Acquiring API","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fmonobank-acquiring","01.csharp\u002F11.aspnet\u002F08.payments\u002F06.monobank-acquiring",{"title":1088,"path":1089,"stem":1090},"Інтеграція Stripe","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fstripe-integration","01.csharp\u002F11.aspnet\u002F08.payments\u002F07.stripe-integration",{"title":1092,"path":1093,"stem":1094},"Webhooks — глибоке занурення","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fwebhooks-deep-dive","01.csharp\u002F11.aspnet\u002F08.payments\u002F08.webhooks-deep-dive",{"title":1096,"path":1097,"stem":1098},"Підписки та рекурентні платежі","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fsubscriptions-recurring","01.csharp\u002F11.aspnet\u002F08.payments\u002F09.subscriptions-recurring",{"title":1100,"path":1101,"stem":1102},"Повернення коштів та диспути","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Frefunds-disputes","01.csharp\u002F11.aspnet\u002F08.payments\u002F10.refunds-disputes",{"title":1104,"path":1105,"stem":1106},"Тестування платіжних інтеграцій","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Ftesting-payments","01.csharp\u002F11.aspnet\u002F08.payments\u002F11.testing-payments",{"title":1108,"path":1109,"stem":1110},"Чекліст виходу в Production","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fproduction-checklist","01.csharp\u002F11.aspnet\u002F08.payments\u002F12.production-checklist",{"title":1112,"icon":1113,"items":1114,"path":1127,"stem":1128,"children":1129,"page":59},"Популярні бібліотеки","lucide:box",[1115,1116,1117,1118,1119,1120,1121,1122,1123,1124,1125,1126],"01.fluent-validation","02.mapster","03.erroror-result-pattern","04.serilog","05.mediatr","06.polly","07.health-checks","08.feature-management","09.fluent-email","10.quest-pdf","11.bogus","12.humanizer-guard","\u002Fcsharp\u002Faspnet\u002Flibraries","01.csharp\u002F11.aspnet\u002F09.libraries",[1130,1134,1138,1142,1146,1150,1154,1158,1162,1166,1170,1174,1178],{"title":1131,"path":1132,"stem":1133},"Валідація з FluentValidation в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ffluent-validation","01.csharp\u002F11.aspnet\u002F09.libraries\u002F01.fluent-validation",{"title":1135,"path":1136,"stem":1137},"Маппінг об","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fmapster","01.csharp\u002F11.aspnet\u002F09.libraries\u002F02.mapster",{"title":1139,"path":1140,"stem":1141},"Обробка помилок з ErrorOr та Result Pattern в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ferroror-result-pattern","01.csharp\u002F11.aspnet\u002F09.libraries\u002F03.erroror-result-pattern",{"title":1143,"path":1144,"stem":1145},"Структуроване логування з Serilog в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fserilog","01.csharp\u002F11.aspnet\u002F09.libraries\u002F04.serilog",{"title":1147,"path":1148,"stem":1149},"CQRS та Mediator з MediatR в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fmediatr","01.csharp\u002F11.aspnet\u002F09.libraries\u002F05.mediatr",{"title":1151,"path":1152,"stem":1153},"Відмовостійкість з Polly в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fpolly","01.csharp\u002F11.aspnet\u002F09.libraries\u002F06.polly",{"title":1155,"path":1156,"stem":1157},"Health Checks в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fhealth-checks","01.csharp\u002F11.aspnet\u002F09.libraries\u002F07.health-checks",{"title":1159,"path":1160,"stem":1161},"Feature Management та Feature Flags в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ffeature-management","01.csharp\u002F11.aspnet\u002F09.libraries\u002F08.feature-management",{"title":1163,"path":1164,"stem":1165},"Відправка Email з FluentEmail в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ffluent-email","01.csharp\u002F11.aspnet\u002F09.libraries\u002F09.fluent-email",{"title":1167,"path":1168,"stem":1169},"Генерація PDF з QuestPDF в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fquest-pdf","01.csharp\u002F11.aspnet\u002F09.libraries\u002F10.quest-pdf",{"title":1171,"path":1172,"stem":1173},"Генерація тестових даних з Bogus в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fbogus","01.csharp\u002F11.aspnet\u002F09.libraries\u002F11.bogus",{"title":1175,"path":1176,"stem":1177},"Humanizer та Guard Clauses в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fhumanizer-guard","01.csharp\u002F11.aspnet\u002F09.libraries\u002F12.humanizer-guard",{"title":1179,"path":1180,"stem":1181},"План модуля 10.libraries — Популярні бібліотеки ASP.NET","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fplan","01.csharp\u002F11.aspnet\u002F09.libraries\u002Fplan",{"title":1183,"icon":1184,"path":1185,"stem":1186,"children":1187,"page":59},"Razor Pages","i-lucide-layout-template","\u002Fcsharp\u002Faspnet\u002Frazor-pages","01.csharp\u002F11.aspnet\u002F10.razor-pages",[1188,1192,1196,1200,1204,1208],{"title":1189,"path":1190,"stem":1191},"Від Minimal API до Razor Pages: концептуальний перехід","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Ffrom-minimal-api","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F01.from-minimal-api",{"title":1193,"path":1194,"stem":1195},"PageModel: логіка сторінки Razor Pages","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Fpage-model","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F02.page-model",{"title":1197,"path":1198,"stem":1199},"Razor синтаксис: шаблонізатор у .cshtml","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Frazor-syntax","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F03.razor-syntax",{"title":1201,"path":1202,"stem":1203},"Tag Helpers: типізований HTML","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Ftag-helpers","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F04.tag-helpers",{"title":1205,"path":1206,"stem":1207},"Форми і валідація: повний цикл обробки даних","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Fforms-validation","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F05.forms-validation",{"title":1209,"path":1210,"stem":1211},"Практичний проєкт: TaskManager на Razor Pages","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Fproject-task-manager","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F06.project-task-manager",{"title":1213,"path":1214,"stem":1215,"children":1216,"page":59},"ASP.NET Core MVC","\u002Fcsharp\u002Faspnet\u002Fmvc","01.csharp\u002F11.aspnet\u002F11.mvc",[1217,1221,1225,1229,1233,1237,1241,1245,1249,1253,1257,1261,1265,1269,1273,1277,1281,1285],{"title":1218,"path":1219,"stem":1220},"Патерн MVC: архітектура, що змінила веб","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fmvc-pattern","01.csharp\u002F11.aspnet\u002F11.mvc\u002F01.mvc-pattern",{"title":1222,"path":1223,"stem":1224},"Від Razor Pages до MVC: концептуальний перехід","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Ffrom-razor-pages","01.csharp\u002F11.aspnet\u002F11.mvc\u002F02.from-razor-pages",{"title":1226,"path":1227,"stem":1228},"Controllers та Actions: серце MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fcontrollers-actions","01.csharp\u002F11.aspnet\u002F11.mvc\u002F03.controllers-actions",{"title":1230,"path":1231,"stem":1232},"Маршрутизація в MVC: Convention vs Attribute Routing","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Frouting-mvc","01.csharp\u002F11.aspnet\u002F11.mvc\u002F04.routing-mvc",{"title":1234,"path":1235,"stem":1236},"Model Binding: від HTTP до C#","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fmodel-binding","01.csharp\u002F11.aspnet\u002F11.mvc\u002F05.model-binding",{"title":1238,"path":1239,"stem":1240},"Views, ViewData, ViewBag, TempData і ViewModel","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fviews-viewdata-tempdata","01.csharp\u002F11.aspnet\u002F11.mvc\u002F06.views-viewdata-tempdata",{"title":1242,"path":1243,"stem":1244},"Filters: аспектно-орієнтоване програмування в MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Ffilters","01.csharp\u002F11.aspnet\u002F11.mvc\u002F07.filters",{"title":1246,"path":1247,"stem":1248},"Areas: структурування великих застосунків","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fareas","01.csharp\u002F11.aspnet\u002F11.mvc\u002F08.areas",{"title":1250,"path":1251,"stem":1252},"View Components: повторювані незалежні блоки UI","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fview-components","01.csharp\u002F11.aspnet\u002F11.mvc\u002F09.view-components",{"title":1254,"path":1255,"stem":1256},"Display та Editor Templates","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fdisplay-editor-templates","01.csharp\u002F11.aspnet\u002F11.mvc\u002F10.display-editor-templates",{"title":1258,"path":1259,"stem":1260},"Валідація: IValidatableObject та FluentValidation","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fvalidation-advanced","01.csharp\u002F11.aspnet\u002F11.mvc\u002F11.validation-advanced",{"title":1262,"path":1263,"stem":1264},"HTMX: інтерактивність через HTML-атрибути","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fhtmx","01.csharp\u002F11.aspnet\u002F11.mvc\u002F12.htmx",{"title":1266,"path":1267,"stem":1268},"HTMX у ASP.NET Core MVC: серверна інтеграція","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fajax-htmx-mvc","01.csharp\u002F11.aspnet\u002F11.mvc\u002F13.ajax-htmx-mvc",{"title":1270,"path":1271,"stem":1272},"Практичний проєкт: Каталог товарів з HTMX","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fhtmx-project","01.csharp\u002F11.aspnet\u002F11.mvc\u002F14.htmx-project",{"title":1274,"path":1275,"stem":1276},"Завантаження та обробка файлів","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Ffile-upload","01.csharp\u002F11.aspnet\u002F11.mvc\u002F15.file-upload",{"title":1278,"path":1279,"stem":1280},"Глобалізація та Локалізація MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fglobalization-localization","01.csharp\u002F11.aspnet\u002F11.mvc\u002F16.globalization-localization",{"title":1282,"path":1283,"stem":1284},"Підсумковий проєкт: Блог-платформа","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fmvc-project","01.csharp\u002F11.aspnet\u002F11.mvc\u002F17.mvc-project",{"title":1286,"path":1287,"stem":1288},"План курсу: ASP.NET Core MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fplan","01.csharp\u002F11.aspnet\u002F11.mvc\u002Fplan",{"title":1290,"path":1291,"stem":1292,"children":1293,"page":59},"Web Api","\u002Fcsharp\u002Faspnet\u002Fweb-api","01.csharp\u002F11.aspnet\u002F12.web-api",[1294,1298,1302,1306,1310,1314,1318,1322,1326,1330,1334,1338,1342],{"title":1295,"path":1296,"stem":1297},"Від Minimal API до Controller-based API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Ffrom-minimal-api-to-controllers","01.csharp\u002F11.aspnet\u002F12.web-api\u002F01.from-minimal-api-to-controllers",{"title":1299,"path":1300,"stem":1301},"ControllerBase, ActionResult\u003CT> та Response Types","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fcontroller-base-actionresult","01.csharp\u002F11.aspnet\u002F12.web-api\u002F02.controller-base-actionresult",{"title":1303,"path":1304,"stem":1305},"Content Negotiation - JSON, XML та власні форматери","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fcontent-negotiation","01.csharp\u002F11.aspnet\u002F12.web-api\u002F03.content-negotiation",{"title":1307,"path":1308,"stem":1309},"Версіонування API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fapi-versioning","01.csharp\u002F11.aspnet\u002F12.web-api\u002F04.api-versioning",{"title":1311,"path":1312,"stem":1313},"ProblemDetails та структурована обробка помилок","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fproblemdetails-error-handling","01.csharp\u002F11.aspnet\u002F12.web-api\u002F05.problemdetails-error-handling",{"title":1315,"path":1316,"stem":1317},"Фільтри у Web API контексті","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Ffilters-for-api","01.csharp\u002F11.aspnet\u002F12.web-api\u002F06.filters-for-api",{"title":1319,"path":1320,"stem":1321},"Пагінація, фільтрація та сортування","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fpagination-filtering-sorting","01.csharp\u002F11.aspnet\u002F12.web-api\u002F07.pagination-filtering-sorting",{"title":1323,"path":1324,"stem":1325},"HATEOAS та Resource Expansion","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fhateoas-resource-expansion","01.csharp\u002F11.aspnet\u002F12.web-api\u002F08.hateoas-resource-expansion",{"title":1327,"path":1328,"stem":1329},"Гібридна архітектура - Minimal API + Controllers","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fminimal-api-vs-controllers-hybrid","01.csharp\u002F11.aspnet\u002F12.web-api\u002F09.minimal-api-vs-controllers-hybrid",{"title":1331,"path":1332,"stem":1333},"Документація API - Swashbuckle, NSwag та генерація клієнтів","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fapi-documentation-generation","01.csharp\u002F11.aspnet\u002F12.web-api\u002F10.api-documentation-generation",{"title":1335,"path":1336,"stem":1337},"Health Checks та моніторинг API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fhealth-checks-monitoring","01.csharp\u002F11.aspnet\u002F12.web-api\u002F11.health-checks-monitoring",{"title":1339,"path":1340,"stem":1341},"Підсумковий проєкт - Production-Ready REST API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fweb-api-project","01.csharp\u002F11.aspnet\u002F12.web-api\u002F12.web-api-project",{"title":1343,"path":1344,"stem":1345},"План курсу: ASP.NET Core Web API (Controllers)","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fplan","01.csharp\u002F11.aspnet\u002F12.web-api\u002Fplan",{"title":1347,"icon":1348,"path":1349,"stem":1350,"children":1351,"page":59},"Desktop UI","i-lucide-app-window","\u002Fcsharp\u002Fdesktop-ui","01.csharp\u002F12.desktop-ui",[1352,1356,1360,1364,1368,1372,1376,1380,1384,1388,1392,1396,1400,1404,1408,1412,1416,1420,1424,1428,1432,1436,1440,1444,1448,1452,1456,1460,1464,1468,1472,1476,1480,1484,1488,1492,1496,1500,1504,1508,1512,1516,1520,1524,1528,1532,1536,1540,1544,1548,1552,1556,1560,1564,1568,1572,1576,1580,1584,1588,1592,1596,1600,1604,1608,1612,1616,1620,1624,1628,1632],{"title":1353,"path":1354,"stem":1355},"Що таке десктопна розробка?","\u002Fcsharp\u002Fdesktop-ui\u002Fwhat-is-desktop-dev","01.csharp\u002F12.desktop-ui\u002F01.what-is-desktop-dev",{"title":1357,"path":1358,"stem":1359},"Архітектура WPF — як влаштований графічний інтерфейс","\u002Fcsharp\u002Fdesktop-ui\u002Fwpf-architecture","01.csharp\u002F12.desktop-ui\u002F02.wpf-architecture",{"title":1361,"path":1362,"stem":1363},"Перший WPF-проєкт — від нуля до вікна","\u002Fcsharp\u002Fdesktop-ui\u002Ffirst-wpf-app","01.csharp\u002F12.desktop-ui\u002F03.first-wpf-app",{"title":1365,"path":1366,"stem":1367},"Перший Avalonia-проєкт: WPF для всіх платформ","\u002Fcsharp\u002Fdesktop-ui\u002F03a.first-avalonia-app","01.csharp\u002F12.desktop-ui\u002F03a.first-avalonia-app",{"title":1369,"path":1370,"stem":1371},"XAML: декларативний інтерфейс","\u002Fcsharp\u002Fdesktop-ui\u002Fxaml-basics","01.csharp\u002F12.desktop-ui\u002F04.xaml-basics",{"title":1373,"path":1374,"stem":1375},"Fluent UI у WPF — сучасний дизайн Windows 11","\u002Fcsharp\u002Fdesktop-ui\u002F04a.wpf-fluent-ui","01.csharp\u002F12.desktop-ui\u002F04a.wpf-fluent-ui",{"title":1377,"path":1378,"stem":1379},"WPF UI — сучасна бібліотека Fluent контролів","\u002Fcsharp\u002Fdesktop-ui\u002F04b.wpf-ui-library","01.csharp\u002F12.desktop-ui\u002F04b.wpf-ui-library",{"title":1381,"path":1382,"stem":1383},"HandyControl — велика бібліотека UI контролів для WPF","\u002Fcsharp\u002Fdesktop-ui\u002F04c.handycontrol-library","01.csharp\u002F12.desktop-ui\u002F04c.handycontrol-library",{"title":1385,"path":1386,"stem":1387},"Простори імен та ресурси XAML","\u002Fcsharp\u002Fdesktop-ui\u002Fxaml-namespaces-resources","01.csharp\u002F12.desktop-ui\u002F05.xaml-namespaces-resources",{"title":1389,"path":1390,"stem":1391},"XAML в Avalonia: ключові відмінності від WPF","\u002Fcsharp\u002Fdesktop-ui\u002F05a.avalonia-xaml-differences","01.csharp\u002F12.desktop-ui\u002F05a.avalonia-xaml-differences",{"title":1393,"path":1394,"stem":1395},"Розширення розмітки XAML (Markup Extensions)","\u002Fcsharp\u002Fdesktop-ui\u002Fxaml-markup-extensions","01.csharp\u002F12.desktop-ui\u002F06.xaml-markup-extensions",{"title":1397,"path":1398,"stem":1399},"Панелі Layout: StackPanel, WrapPanel, DockPanel","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-panels-part1","01.csharp\u002F12.desktop-ui\u002F07.layout-panels-part1",{"title":1401,"path":1402,"stem":1403},"Grid, Canvas, UniformGrid","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-panels-part2","01.csharp\u002F12.desktop-ui\u002F07.layout-panels-part2",{"title":1405,"path":1406,"stem":1407},"Просунуті техніки Layout","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-advanced","01.csharp\u002F12.desktop-ui\u002F08.layout-advanced",{"title":1409,"path":1410,"stem":1411},"Адаптивний Layout та найкращі практики","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-responsive","01.csharp\u002F12.desktop-ui\u002F09.layout-responsive",{"title":1413,"path":1414,"stem":1415},"Layout в Avalonia: відмінності та нові можливості","\u002Fcsharp\u002Fdesktop-ui\u002F09a.layout-avalonia","01.csharp\u002F12.desktop-ui\u002F09a.layout-avalonia",{"title":1417,"path":1418,"stem":1419},"Button, Image, ProgressBar та інші базові контроли","\u002Fcsharp\u002Fdesktop-ui\u002Fbasic-controls","01.csharp\u002F12.desktop-ui\u002F10.basic-controls",{"title":1421,"path":1422,"stem":1423},"Контроли в Avalonia: відмінності від WPF","\u002Fcsharp\u002Fdesktop-ui\u002F10a.controls-avalonia","01.csharp\u002F12.desktop-ui\u002F10a.controls-avalonia",{"title":1425,"path":1426,"stem":1427},"Текстові контроли — TextBlock, TextBox, RichTextBox","\u002Fcsharp\u002Fdesktop-ui\u002Ftext-controls","01.csharp\u002F12.desktop-ui\u002F11.text-controls",{"title":1429,"path":1430,"stem":1431},"Контроли вибору — CheckBox, RadioButton, ComboBox, ListBox, DatePicker","\u002Fcsharp\u002Fdesktop-ui\u002Fselection-controls","01.csharp\u002F12.desktop-ui\u002F12.selection-controls",{"title":1433,"path":1434,"stem":1435},"Content Model — GroupBox, Expander, TabControl, StatusBar","\u002Fcsharp\u002Fdesktop-ui\u002Fcontent-controls","01.csharp\u002F12.desktop-ui\u002F13.content-controls",{"title":1437,"path":1438,"stem":1439},"UI\u002FUX принципи десктопних застосунків","\u002Fcsharp\u002Fdesktop-ui\u002F13a.ui-ux-principles","01.csharp\u002F12.desktop-ui\u002F13a.ui-ux-principles",{"title":1441,"path":1442,"stem":1443},"Dependency Properties — Концепція та Value Resolution","\u002Fcsharp\u002Fdesktop-ui\u002Fdependency-properties-part1","01.csharp\u002F12.desktop-ui\u002F14.dependency-properties-part1",{"title":1445,"path":1446,"stem":1447},"Avalonia Property System — StyledProperty та DirectProperty","\u002Fcsharp\u002Fdesktop-ui\u002F14a.avalonia-property-system","01.csharp\u002F12.desktop-ui\u002F14a.avalonia-property-system",{"title":1449,"path":1450,"stem":1451},"Attached Properties — Властивості без меж","\u002Fcsharp\u002Fdesktop-ui\u002Fattached-properties","01.csharp\u002F12.desktop-ui\u002F15.attached-properties",{"title":1453,"path":1454,"stem":1455},"Routed Events — Маршрутизація подій у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Frouted-events","01.csharp\u002F12.desktop-ui\u002F16.routed-events",{"title":1457,"path":1458,"stem":1459},"Data Binding — Від Code-Behind до Декларативності","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-binding-basics-part1","01.csharp\u002F12.desktop-ui\u002F17.data-binding-basics-part1",{"title":1461,"path":1462,"stem":1463},"INotifyPropertyChanged — Живе оновлення UI","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-binding-basics-part2","01.csharp\u002F12.desktop-ui\u002F17.data-binding-basics-part2",{"title":1465,"path":1466,"stem":1467},"Compiled Bindings в Avalonia — Безпека на етапі компіляції","\u002Fcsharp\u002Fdesktop-ui\u002F17a.avalonia-compiled-bindings","01.csharp\u002F12.desktop-ui\u002F17a.avalonia-compiled-bindings",{"title":1469,"path":1470,"stem":1471},"Просунутий Data Binding — ElementName, RelativeSource, MultiBinding","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-binding-advanced","01.csharp\u002F12.desktop-ui\u002F18.data-binding-advanced",{"title":1473,"path":1474,"stem":1475},"Value Converters — Перетворення типів даних у Data Binding","\u002Fcsharp\u002Fdesktop-ui\u002Fvalue-converters","01.csharp\u002F12.desktop-ui\u002F19.value-converters",{"title":1477,"path":1478,"stem":1479},"Data Templates — Візуалізація об'єктів у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-templates","01.csharp\u002F12.desktop-ui\u002F20.data-templates",{"title":1481,"path":1482,"stem":1483},"Collections Binding Part 1 — ObservableCollection та ItemsControl","\u002Fcsharp\u002Fdesktop-ui\u002Fcollections-binding-part1","01.csharp\u002F12.desktop-ui\u002F21.collections-binding-part1",{"title":1485,"path":1486,"stem":1487},"Collections Binding Part 2 — ICollectionView, Filtering, Sorting та Virtualization","\u002Fcsharp\u002Fdesktop-ui\u002Fcollections-binding-part2","01.csharp\u002F12.desktop-ui\u002F21.collections-binding-part2",{"title":1489,"path":1490,"stem":1491},"MVVM Pattern — Від Spaghetti Code до архітектури","\u002Fcsharp\u002Fdesktop-ui\u002Fmvvm-pattern","01.csharp\u002F12.desktop-ui\u002F22.mvvm-pattern",{"title":1493,"path":1494,"stem":1495},"ViewModel Implementation — Від BaseViewModel до валідації","\u002Fcsharp\u002Fdesktop-ui\u002Fviewmodel-implementation","01.csharp\u002F12.desktop-ui\u002F23.viewmodel-implementation",{"title":1497,"path":1498,"stem":1499},"Commands — Від event handlers до декларативних команд","\u002Fcsharp\u002Fdesktop-ui\u002Fcommands","01.csharp\u002F12.desktop-ui\u002F24.commands",{"title":1501,"path":1502,"stem":1503},"MVVM Toolkit — MVVM без boilerplate через Source Generators","\u002Fcsharp\u002Fdesktop-ui\u002Fmvvm-toolkit","01.csharp\u002F12.desktop-ui\u002F25.mvvm-toolkit",{"title":1505,"path":1506,"stem":1507},"Messenger Pattern — Комунікація між ViewModel без прямих посилань","\u002Fcsharp\u002Fdesktop-ui\u002Fmessenger-pattern","01.csharp\u002F12.desktop-ui\u002F26.messenger-pattern",{"title":1509,"path":1510,"stem":1511},"Стилі WPF — CSS для десктопу","\u002Fcsharp\u002Fdesktop-ui\u002Fstyles-basics","01.csharp\u002F12.desktop-ui\u002F27.styles-basics",{"title":1513,"path":1514,"stem":1515},"CSS-like стилі Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002F27a.avalonia-css-styling","01.csharp\u002F12.desktop-ui\u002F27a.avalonia-css-styling",{"title":1517,"path":1518,"stem":1519},"Control Templates — Частина 1. Концепція та TemplateBinding","\u002Fcsharp\u002Fdesktop-ui\u002Fcontrol-templates-part1","01.csharp\u002F12.desktop-ui\u002F28.control-templates-part1",{"title":1521,"path":1522,"stem":1523},"Control Templates — Частина 2. Named Parts та ContentPresenter","\u002Fcsharp\u002Fdesktop-ui\u002Fcontrol-templates-part2","01.csharp\u002F12.desktop-ui\u002F28.control-templates-part2",{"title":1525,"path":1526,"stem":1527},"Control Themes в Avalonia — нова ера стилізації","\u002Fcsharp\u002Fdesktop-ui\u002F28a.avalonia-control-themes","01.csharp\u002F12.desktop-ui\u002F28a.avalonia-control-themes",{"title":1529,"path":1530,"stem":1531},"Triggers та Visual State Manager у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Ftriggers-visual-states","01.csharp\u002F12.desktop-ui\u002F29.triggers-visual-states",{"title":1533,"path":1534,"stem":1535},"Pseudo-classes в Avalonia — замість WPF Triggers","\u002Fcsharp\u002Fdesktop-ui\u002F29a.avalonia-pseudo-classes","01.csharp\u002F12.desktop-ui\u002F29a.avalonia-pseudo-classes",{"title":1537,"path":1538,"stem":1539},"Теми та ресурсні словники у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fresources-themes","01.csharp\u002F12.desktop-ui\u002F30.resources-themes",{"title":1541,"path":1542,"stem":1543},"Avalonia Themes — Fluent Design та система тематизації","\u002Fcsharp\u002Fdesktop-ui\u002F30a.avalonia-themes-fluent","01.csharp\u002F12.desktop-ui\u002F30a.avalonia-themes-fluent",{"title":1545,"path":1546,"stem":1547},"Контроли колекцій — глибоке занурення","\u002Fcsharp\u002Fdesktop-ui\u002Fcollection-controls","01.csharp\u002F12.desktop-ui\u002F31.collection-controls",{"title":1549,"path":1550,"stem":1551},"DataGrid — колонки та базове відображення","\u002Fcsharp\u002Fdesktop-ui\u002Fdatagrid-part1","01.csharp\u002F12.desktop-ui\u002F32.datagrid-part1",{"title":1553,"path":1554,"stem":1555},"DataGrid — сортування, фільтрація, редагування","\u002Fcsharp\u002Fdesktop-ui\u002Fdatagrid-part2","01.csharp\u002F12.desktop-ui\u002F32.datagrid-part2",{"title":1557,"path":1558,"stem":1559},"TreeView та GridView","\u002Fcsharp\u002Fdesktop-ui\u002Ftreeview-listview","01.csharp\u002F12.desktop-ui\u002F33.treeview-listview",{"title":1561,"path":1562,"stem":1563},"Меню, Toolbar, ContextMenu, StatusBar","\u002Fcsharp\u002Fdesktop-ui\u002Fmenus-toolbars","01.csharp\u002F12.desktop-ui\u002F34.menus-toolbars",{"title":1565,"path":1566,"stem":1567},"Навігація та керування вікнами. Частина 1: вікна та сторінки","\u002Fcsharp\u002Fdesktop-ui\u002Fnavigation-windows-part1","01.csharp\u002F12.desktop-ui\u002F35.navigation-windows-part1",{"title":1569,"path":1570,"stem":1571},"Навігація та керування вікнами. Частина 2: MVVM-навігація","\u002Fcsharp\u002Fdesktop-ui\u002Fnavigation-windows-part2","01.csharp\u002F12.desktop-ui\u002F35.navigation-windows-part2",{"title":1573,"path":1574,"stem":1575},"Avalonia — Навігація та діалоги","\u002Fcsharp\u002Fdesktop-ui\u002F35a.avalonia-navigation-dialogs","01.csharp\u002F12.desktop-ui\u002F35a.avalonia-navigation-dialogs",{"title":1577,"path":1578,"stem":1579},"Діалоги та File Pickers у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fdialogs-file-pickers","01.csharp\u002F12.desktop-ui\u002F36.dialogs-file-pickers",{"title":1581,"path":1582,"stem":1583},"UserControl: компонентний підхід у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fuser-controls","01.csharp\u002F12.desktop-ui\u002F37.user-controls",{"title":1585,"path":1586,"stem":1587},"Custom Controls: Lookless Controls у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fcustom-controls","01.csharp\u002F12.desktop-ui\u002F38.custom-controls",{"title":1589,"path":1590,"stem":1591},"Avalonia TemplatedControl — Lookless Controls","\u002Fcsharp\u002Fdesktop-ui\u002F38a.avalonia-templated-controls","01.csharp\u002F12.desktop-ui\u002F38a.avalonia-templated-controls",{"title":1593,"path":1594,"stem":1595},"Анімації у WPF: Storyboard та Easing Functions","\u002Fcsharp\u002Fdesktop-ui\u002Fanimations-transitions","01.csharp\u002F12.desktop-ui\u002F39.animations-transitions",{"title":1597,"path":1598,"stem":1599},"Анімації в Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002F39a.avalonia-animations","01.csharp\u002F12.desktop-ui\u002F39a.avalonia-animations",{"title":1601,"path":1602,"stem":1603},"2D Графіка та Мультимедіа у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fmedia-graphics","01.csharp\u002F12.desktop-ui\u002F40.media-graphics",{"title":1605,"path":1606,"stem":1607},"Dependency Injection у WPF та Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002Fdi-integration","01.csharp\u002F12.desktop-ui\u002F41.di-integration",{"title":1609,"path":1610,"stem":1611},"SQLite та EF Core у десктопних додатках","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-persistence-part1","01.csharp\u002F12.desktop-ui\u002F42.data-persistence-part1",{"title":1613,"path":1614,"stem":1615},"Repository Pattern та Unit of Work","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-persistence-part2","01.csharp\u002F12.desktop-ui\u002F43.data-persistence-part2",{"title":1617,"path":1618,"stem":1619},"Тестування ViewModels","\u002Fcsharp\u002Fdesktop-ui\u002Fviewmodel-testing","01.csharp\u002F12.desktop-ui\u002F44.viewmodel-testing",{"title":1621,"path":1622,"stem":1623},"Avalonia Headless Testing — тестування UI без вікон","\u002Fcsharp\u002Fdesktop-ui\u002F44a.avalonia-headless-testing","01.csharp\u002F12.desktop-ui\u002F44a.avalonia-headless-testing",{"title":1625,"path":1626,"stem":1627},"Кросплатформна розробка з Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002Favalonia-cross-platform","01.csharp\u002F12.desktop-ui\u002F45.avalonia-cross-platform",{"title":1629,"path":1630,"stem":1631},"Пакування та розгортання Avalonia додатків","\u002Fcsharp\u002Fdesktop-ui\u002Favalonia-packaging-deployment","01.csharp\u002F12.desktop-ui\u002F46.avalonia-packaging-deployment",{"title":1633,"path":1634,"stem":1635},"Розгортання WPF застосунків","\u002Fcsharp\u002Fdesktop-ui\u002Fwpf-packaging-deployment","01.csharp\u002F12.desktop-ui\u002F47.wpf-packaging-deployment",{"title":1637,"icon":658,"path":1638,"stem":1639,"children":1640,"page":59},"Network Programming","\u002Fcsharp\u002Fnetwork-programming","01.csharp\u002F13.network-programming",[1641,1645,1649,1653,1657,1661],{"title":1642,"path":1643,"stem":1644},"Основи комп'ютерних мереж","\u002Fcsharp\u002Fnetwork-programming\u002Ffoundations","01.csharp\u002F13.network-programming\u002F01.foundations",{"title":1646,"path":1647,"stem":1648},"Модель OSI та стек TCP\u002FIP","\u002Fcsharp\u002Fnetwork-programming\u002Fosi-model","01.csharp\u002F13.network-programming\u002F02.osi-model",{"title":1650,"path":1651,"stem":1652},"IP-протокол та адресація","\u002Fcsharp\u002Fnetwork-programming\u002Fip-addressing","01.csharp\u002F13.network-programming\u002F03.ip-addressing",{"title":1654,"path":1655,"stem":1656},"UDP — протокол без з'єднання","\u002Fcsharp\u002Fnetwork-programming\u002Fudp","01.csharp\u002F13.network-programming\u002F05.udp",{"title":1658,"path":1659,"stem":1660},"UDP Broadcast та Multicast","\u002Fcsharp\u002Fnetwork-programming\u002Fudp-broadcast-multicast","01.csharp\u002F13.network-programming\u002F06.udp-broadcast-multicast",{"title":1662,"path":1663,"stem":1664},"HTTP — протокол вебу","\u002Fcsharp\u002Fnetwork-programming\u002Fhttp-fundamentals","01.csharp\u002F13.network-programming\u002F07.http-fundamentals",{"title":1666,"path":1667,"stem":1668},"C# & .NET: The Ultimate Roadmap","\u002Fcsharp\u002Froadmap","01.csharp\u002Froadmap",{"title":1670,"icon":1671,"path":1672,"stem":1673,"children":1674,"page":59},"C++","i-devicon-cplusplus","\u002Fcpp","02.cpp",[1675,1679,1683,1687,1691,1695,1699,1703,1707,1710,1714,1718,1722,1726,1730,1734,1738,1742,1746,1750,1754,1758,1762,1766,1770,1774,1778,1782,1786,1790,1794,1798,1802,1806,1810,1814,1818,1822],{"title":1676,"path":1677,"stem":1678},"Вступ у програмування та алгоритми","\u002Fcpp\u002Fintro-algorithms","02.cpp\u002F01.intro-algorithms",{"title":1680,"path":1681,"stem":1682},"Code Style: угоди про оформлення коду","\u002Fcpp\u002Fcode-style","02.cpp\u002F02.code-style",{"title":1684,"path":1685,"stem":1686},"Середовище розробки та перший проєкт","\u002Fcpp\u002Fide-setup","02.cpp\u002F03.ide-setup",{"title":1688,"path":1689,"stem":1690},"Вивід даних на екран","\u002Fcpp\u002Fdata-output","02.cpp\u002F04.data-output",{"title":1692,"path":1693,"stem":1694},"Типи даних, змінні та константи","\u002Fcpp\u002Fdata-types-variables","02.cpp\u002F05.data-types-variables",{"title":1696,"path":1697,"stem":1698},"Ввід даних з клавіатури","\u002Fcpp\u002Fdata-input","02.cpp\u002F06.data-input",{"title":1700,"path":1701,"stem":1702},"Оператори, перетворення типів та логічні операції","\u002Fcpp\u002Foperators-type-conversion","02.cpp\u002F07.operators-type-conversion",{"title":1704,"path":1705,"stem":1706},"Цикли","\u002Fcpp\u002Floops","02.cpp\u002F08.loops",{"title":32,"path":1708,"stem":1709},"\u002Fcpp\u002Farrays","02.cpp\u002F09.arrays",{"title":1711,"path":1712,"stem":1713},"Алгоритми сортування та аналіз складності","\u002Fcpp\u002Fsorting","02.cpp\u002F10.sorting",{"title":1715,"path":1716,"stem":1717},"Алгоритми пошуку","\u002Fcpp\u002Fsearching","02.cpp\u002F11.searching",{"title":1719,"path":1720,"stem":1721},"Функції: основи","\u002Fcpp\u002Ffunctions-basics","02.cpp\u002F12.functions-basics",{"title":1723,"path":1724,"stem":1725},"Функції: прототипи, область видимості та додаткові можливості","\u002Fcpp\u002Ffunctions-scope","02.cpp\u002F13.functions-scope",{"title":1727,"path":1728,"stem":1729},"Функції: перевантаження та шаблони","\u002Fcpp\u002Ffunctions-overloading-templates","02.cpp\u002F14.functions-overloading-templates",{"title":1731,"path":1732,"stem":1733},"Вказівники: основи","\u002Fcpp\u002Fpointers-basics","02.cpp\u002F15.pointers-basics",{"title":1735,"path":1736,"stem":1737},"Посилання (References)","\u002Fcpp\u002Freferences","02.cpp\u002F16.references",{"title":1739,"path":1740,"stem":1741},"Вказівники, const і масиви","\u002Fcpp\u002Fpointers-const-arrays","02.cpp\u002F17.pointers-const-arrays",{"title":1743,"path":1744,"stem":1745},"Адресна арифметика","\u002Fcpp\u002Fpointer-arithmetic","02.cpp\u002F18.pointer-arithmetic",{"title":1747,"path":1748,"stem":1749},"Динамічна пам'ять","\u002Fcpp\u002Fdynamic-memory","02.cpp\u002F19.dynamic-memory",{"title":1751,"path":1752,"stem":1753},"Вказівники типу void","\u002Fcpp\u002Fvoid-pointers","02.cpp\u002F20.void-pointers",{"title":1755,"path":1756,"stem":1757},"Вказівники на вказівники","\u002Fcpp\u002Fpointers-to-pointers","02.cpp\u002F21.pointers-to-pointers",{"title":1759,"path":1760,"stem":1761},"Оператор доступу до членів через вказівник (->)","\u002Fcpp\u002Fmember-access-operator","02.cpp\u002F22.member-access-operator",{"title":1763,"path":1764,"stem":1765},"Цикл for-each (Range-based for)","\u002Fcpp\u002Fforeach-loop","02.cpp\u002F23.foreach-loop",{"title":1767,"path":1768,"stem":1769},"Вказівники на функції","\u002Fcpp\u002Ffunction-pointers","02.cpp\u002F24.function-pointers",{"title":1771,"path":1772,"stem":1773},"Лямбда-вирази","\u002Fcpp\u002Flambdas","02.cpp\u002F25.lambdas",{"title":1775,"path":1776,"stem":1777},"Лямбда-захоплення","\u002Fcpp\u002Flambda-captures","02.cpp\u002F26.lambda-captures",{"title":1779,"path":1780,"stem":1781},"Еліпсис","\u002Fcpp\u002Fellipsis","02.cpp\u002F27.ellipsis",{"title":1783,"path":1784,"stem":1785},"Безпечні альтернативи еліпсису","\u002Fcpp\u002F27a.ellipsis","02.cpp\u002F27a.ellipsis",{"title":1787,"path":1788,"stem":1789},"Аргументи командного рядка","\u002Fcpp\u002Fcommand-line-arguments","02.cpp\u002F28.command-line-arguments",{"title":1791,"path":1792,"stem":1793},"Перерахування (enum)","\u002Fcpp\u002Fenum","02.cpp\u002F29.enum",{"title":1795,"path":1796,"stem":1797},"Класи-перерахування (enum class)","\u002Fcpp\u002Fenum-class","02.cpp\u002F30.enum-class",{"title":1799,"path":1800,"stem":1801},"Псевдоніми типів (typedef і using)","\u002Fcpp\u002Ftype-aliases","02.cpp\u002F31.type-aliases",{"title":1803,"path":1804,"stem":1805},"Системи числення та двійкова арифметика","\u002Fcpp\u002Fnumber-systems","02.cpp\u002F32.number-systems",{"title":1807,"path":1808,"stem":1809},"Структури (struct): агрегування даних","\u002Fcpp\u002Fstruct","02.cpp\u002F33.struct",{"title":1811,"path":1812,"stem":1813},"Структури у функціях","\u002Fcpp\u002Fstruct-functions","02.cpp\u002F34.struct-functions",{"title":1815,"path":1816,"stem":1817},"Масиви структур і вкладені структури","\u002Fcpp\u002Fstruct-arrays","02.cpp\u002F35.struct-arrays",{"title":1819,"path":1820,"stem":1821},"Патерни struct та межі застосування","\u002Fcpp\u002Fstruct-patterns","02.cpp\u002F36.struct-patterns",{"title":1823,"path":1824,"stem":1825},"План навчання: Курс C++ — Продовження (Статті 29–60+)","\u002Fcpp\u002Fcurriculum-plan","02.cpp\u002Fcurriculum-plan",{"title":1827,"icon":1828,"path":1829,"stem":1830,"children":1831,"page":59},"JavaScript","i-devicon-javascript","\u002Fjavascript","03.javascript",[1832,1858,1912,1934,2238,2276],{"title":1833,"icon":1834,"path":1835,"stem":1836,"children":1837,"page":59},"Events","i-lucide-mouse-pointer-click","\u002Fjavascript\u002Fevents","03.javascript\u002F01.events",[1838,1842,1846,1850,1854],{"title":1839,"path":1840,"stem":1841},"Вступ до подій браузера","\u002Fjavascript\u002Fevents\u002Fintro","03.javascript\u002F01.events\u002F01.intro",{"title":1843,"path":1844,"stem":1845},"Бульбашковий механізм (Bubbling) та занурення (Capturing)","\u002Fjavascript\u002Fevents\u002Fbubbling-capturing","03.javascript\u002F01.events\u002F02.bubbling-capturing",{"title":1847,"path":1848,"stem":1849},"Делегування подій (Event Delegation)","\u002Fjavascript\u002Fevents\u002Fdelegate-events","03.javascript\u002F01.events\u002F03.delegate-events",{"title":1851,"path":1852,"stem":1853},"Типові дії браузера та preventDefault()","\u002Fjavascript\u002Fevents\u002Fprevent-default","03.javascript\u002F01.events\u002F04.prevent-default",{"title":1855,"path":1856,"stem":1857},"Запуск користувацьких подій (Custom Events)","\u002Fjavascript\u002Fevents\u002Fcustom-events","03.javascript\u002F01.events\u002F05.custom-events",{"title":1859,"icon":1860,"path":1861,"stem":1862,"children":1863,"page":59},"Network","i-lucide-globe","\u002Fjavascript\u002Fnetwork","03.javascript\u002F02.network",[1864,1868,1872,1876,1880,1884,1888,1892,1896,1900,1904,1908],{"title":1865,"path":1866,"stem":1867},"Fetch API - Сучасний підхід до HTTP-запитів","\u002Fjavascript\u002Fnetwork\u002F01-fetch-api","03.javascript\u002F02.network\u002F01-fetch-api",{"title":1869,"path":1870,"stem":1871},"FormData - Робота з формами та файлами","\u002Fjavascript\u002Fnetwork\u002F02-formdata","03.javascript\u002F02.network\u002F02-formdata",{"title":1873,"path":1874,"stem":1875},"Відстеження прогресу завантаження","\u002Fjavascript\u002Fnetwork\u002F03-download-progress","03.javascript\u002F02.network\u002F03-download-progress",{"title":1877,"path":1878,"stem":1879},"Переривання fetch-запитів","\u002Fjavascript\u002Fnetwork\u002F04-abort-requests","03.javascript\u002F02.network\u002F04-abort-requests",{"title":1881,"path":1882,"stem":1883},"CORS - Запити між різними джерелами","\u002Fjavascript\u002Fnetwork\u002F05-cors","03.javascript\u002F02.network\u002F05-cors",{"title":1885,"path":1886,"stem":1887},"Fetch API - Повний довідник опцій","\u002Fjavascript\u002Fnetwork\u002F06-fetch-options","03.javascript\u002F02.network\u002F06-fetch-options",{"title":1889,"path":1890,"stem":1891},"URL Objects - Робота з посиланнями","\u002Fjavascript\u002Fnetwork\u002F07-url-objects","03.javascript\u002F02.network\u002F07-url-objects",{"title":1893,"path":1894,"stem":1895},"XMLHttpRequest - AJAX та низькорівневі запити","\u002Fjavascript\u002Fnetwork\u002F08-xmlhttprequest","03.javascript\u002F02.network\u002F08-xmlhttprequest",{"title":1897,"path":1898,"stem":1899},"Відновлюване завантаження файлів","\u002Fjavascript\u002Fnetwork\u002F09-resumable-upload","03.javascript\u002F02.network\u002F09-resumable-upload",{"title":1901,"path":1902,"stem":1903},"Cookies, document.cookie та світ після \"Cookiepocalypse\"","\u002Fjavascript\u002Fnetwork\u002F10-cookies","03.javascript\u002F02.network\u002F10-cookies",{"title":1905,"path":1906,"stem":1907},"js-cookie: Керування Cookies без Болю","\u002Fjavascript\u002Fnetwork\u002F11-js-cookie","03.javascript\u002F02.network\u002F11-js-cookie",{"title":1909,"path":1910,"stem":1911},"Axios: Потужний HTTP-клієнт для JavaScript","\u002Fjavascript\u002Fnetwork\u002F12-axios","03.javascript\u002F02.network\u002F12-axios",{"title":1913,"icon":1914,"path":1915,"stem":1916,"children":1917,"page":59},"Bom","i-lucide-monitor","\u002Fjavascript\u002Fbom","03.javascript\u002F03.bom",[1918,1922,1926,1930],{"title":1919,"path":1920,"stem":1921},"LocalStorage, SessionStorage та patterns збереження даних","\u002Fjavascript\u002Fbom\u002F01-localstorage","03.javascript\u002F03.bom\u002F01-localstorage",{"title":1923,"path":1924,"stem":1925},"Location Object - Керування адресою сторінки","\u002Fjavascript\u002Fbom\u002F02-location-object","03.javascript\u002F03.bom\u002F02-location-object",{"title":1927,"path":1928,"stem":1929},"History API - Керування історією браузера","\u002Fjavascript\u002Fbom\u002F03-history-api","03.javascript\u002F03.bom\u002F03-history-api",{"title":1931,"path":1932,"stem":1933},"Navigator Object - Ідентифікація та Можливості Пристрою","\u002Fjavascript\u002Fbom\u002F04-navigator-object","03.javascript\u002F03.bom\u002F04-navigator-object",{"title":1935,"icon":1936,"path":1937,"stem":1938,"children":1939},"React","i-devicon-react","\u002Fjavascript\u002Freact","03.javascript\u002F04.react\u002Findex",[1940,1941,1945,1949,1953,1957,2020,2055,2207],{"title":1935,"path":1937,"stem":1938},{"title":1942,"path":1943,"stem":1944},"Робота з Формами в React","\u002Fjavascript\u002Freact\u002Freact-forms","03.javascript\u002F04.react\u002F01.react-forms",{"title":1946,"path":1947,"stem":1948},"React Hook Form: Професійна Робота з Формами","\u002Fjavascript\u002Freact\u002Freact-hook-form","03.javascript\u002F04.react\u002F02.react-hook-form",{"title":1950,"path":1951,"stem":1952},"React Hook Form: Глибоке Розуміння Архітектури та Оптимізації","\u002Fjavascript\u002Freact\u002Freact-hook-form-new","03.javascript\u002F04.react\u002F02.react-hook-form-new",{"title":1954,"path":1955,"stem":1956},"Axios та React: Професійна Архітектура Запитів","\u002Fjavascript\u002Freact\u002Fdata-fetching-axios","03.javascript\u002F04.react\u002F03.data-fetching-axios",{"title":1958,"icon":132,"path":1959,"stem":1960,"children":1961},"Tanstack Query","\u002Fjavascript\u002Freact\u002Ftanstack-query","03.javascript\u002F04.react\u002F04.tanstack-query\u002Findex",[1962,1964,1968,1972,1976,1980,1984,1988,1992,1996,2000,2004,2008,2012,2016],{"title":1963,"path":1959,"stem":1960},"TanStack Query: Майстерність Керування Станом Сервера",{"title":1965,"path":1966,"stem":1967},"Парадигма Server State: Чому useEffect недостатньо","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fserver-state-paradigm","03.javascript\u002F04.react\u002F04.tanstack-query\u002F01.server-state-paradigm",{"title":1969,"path":1970,"stem":1971},"Встановлення та Налаштування: Фундамент","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Finstallation-and-devtools","03.javascript\u002F04.react\u002F04.tanstack-query\u002F02.installation-and-devtools",{"title":1973,"path":1974,"stem":1975},"Основи Запитів та Магія Ключів","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fquery-basics-and-keys","03.javascript\u002F04.react\u002F04.tanstack-query\u002F03.query-basics-and-keys",{"title":1977,"path":1978,"stem":1979},"Синхронізація Даних: Життєвий Цикл Запиту","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fdata-synchronization","03.javascript\u002F04.react\u002F04.tanstack-query\u002F04.data-synchronization",{"title":1981,"path":1982,"stem":1983},"Мутації та Інвалідація: Зміна Даних","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fmutations-and-invalidation","03.javascript\u002F04.react\u002F04.tanstack-query\u002F05.mutations-and-invalidation",{"title":1985,"path":1986,"stem":1987},"Оптимістичні Оновлення: Швидше за Світло","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Foptimistic-updates","03.javascript\u002F04.react\u002F04.tanstack-query\u002F06.optimistic-updates",{"title":1989,"path":1990,"stem":1991},"Пагінація та Infinite Scroll","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fpagination-and-load-more","03.javascript\u002F04.react\u002F04.tanstack-query\u002F07.pagination-and-load-more",{"title":1993,"path":1994,"stem":1995},"Просунуті Патерни та Оптимізація","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fadvanced-patterns","03.javascript\u002F04.react\u002F04.tanstack-query\u002F08.advanced-patterns",{"title":1997,"path":1998,"stem":1999},"Архітектура та Best Practices","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Farchitecture-and-best-practices","03.javascript\u002F04.react\u002F04.tanstack-query\u002F09.architecture-and-best-practices",{"title":2001,"path":2002,"stem":2003},"Server-Side Rendering (SSR) та Гідратація","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fserver-side-rendering","03.javascript\u002F04.react\u002F04.tanstack-query\u002F10.server-side-rendering",{"title":2005,"path":2006,"stem":2007},"Стратегії Тестування","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Ftesting-strategies","03.javascript\u002F04.react\u002F04.tanstack-query\u002F11.testing-strategies",{"title":2009,"path":2010,"stem":2011},"Аутентифікація та Обробка Помилок","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fauthentication-and-errors","03.javascript\u002F04.react\u002F04.tanstack-query\u002F12.authentication-and-errors",{"title":2013,"path":2014,"stem":2015},"React Suspense та Майбутнє","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Freact-suspense","03.javascript\u002F04.react\u002F04.tanstack-query\u002F13.react-suspense",{"title":2017,"path":2018,"stem":2019},"Глибоке Занурення в Продуктивність","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fperformance-deep-dive","03.javascript\u002F04.react\u002F04.tanstack-query\u002F14.performance-deep-dive",{"title":2021,"icon":1936,"path":2022,"stem":2023,"children":2024},"React Router","\u002Fjavascript\u002Freact\u002Freact-router","03.javascript\u002F04.react\u002F05.react-router\u002Findex",[2025,2027,2031,2035,2039,2043,2047,2051],{"title":2026,"path":2022,"stem":2023},"React Router: Навігаційна система сучасного вебу",{"title":2028,"path":2029,"stem":2030},"Налаштування та Базовий Роутинг","\u002Fjavascript\u002Freact\u002Freact-router\u002Fsetup-and-basic-routing","03.javascript\u002F04.react\u002F05.react-router\u002F01.setup-and-basic-routing",{"title":2032,"path":2033,"stem":2034},"Динамічна Навігація","\u002Fjavascript\u002Freact\u002Freact-router\u002Fnavigation-and-links","03.javascript\u002F04.react\u002F05.react-router\u002F02.navigation-and-links",{"title":2036,"path":2037,"stem":2038},"Вкладені Маршрути та Макети","\u002Fjavascript\u002Freact\u002Freact-router\u002Fnested-routes-and-layouts","03.javascript\u002F04.react\u002F05.react-router\u002F03.nested-routes-and-layouts",{"title":2040,"path":2041,"stem":2042},"Динамічні Маршрути та Параметри","\u002Fjavascript\u002Freact\u002Freact-router\u002Fdynamic-routing","03.javascript\u002F04.react\u002F05.react-router\u002F04.dynamic-routing",{"title":2044,"path":2045,"stem":2046},"Data APIs: Loaders та Actions","\u002Fjavascript\u002Freact\u002Freact-router\u002Fdata-loading","03.javascript\u002F04.react\u002F05.react-router\u002F05.data-loading",{"title":2048,"path":2049,"stem":2050},"Просунуті Патерни","\u002Fjavascript\u002Freact\u002Freact-router\u002Fadvanced-patterns","03.javascript\u002F04.react\u002F05.react-router\u002F06.advanced-patterns",{"title":2052,"path":2053,"stem":2054},"Legacy Routing: Компонентний підхід","\u002Fjavascript\u002Freact\u002Freact-router\u002Flegacy-routing","03.javascript\u002F04.react\u002F05.react-router\u002F07.legacy-routing",{"title":2056,"icon":132,"path":2057,"stem":2058,"children":2059},"Redux","\u002Fjavascript\u002Freact\u002Fredux","03.javascript\u002F04.react\u002F06.redux\u002Findex",[2060,2062,2078,2107,2116,2137,2153,2182],{"title":2061,"path":2057,"stem":2058},"Redux: Еволюція управління станом",{"title":14,"icon":15,"path":2063,"stem":2064,"children":2065,"page":59},"\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals",[2066,2070,2074],{"title":2067,"path":2068,"stem":2069},"Вступ до State Management","\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals\u002Fintro-state-management","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals\u002F01.intro-state-management",{"title":2071,"path":2072,"stem":2073},"Філософія Redux та Три Принципи","\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals\u002Fredux-philosophy","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals\u002F02.redux-philosophy",{"title":2075,"path":2076,"stem":2077},"Чисті функції та Іммутабельність","\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals\u002Fpure-functions-immutability","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals\u002F03.pure-functions-immutability",{"title":2079,"icon":132,"path":2080,"stem":2081,"children":2082,"page":59},"Classic Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux",[2083,2087,2091,2095,2099,2103],{"title":2084,"path":2085,"stem":2086},"Створення Store (Classic Redux)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Fstore-setup","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F01.store-setup",{"title":2088,"path":2089,"stem":2090},"Actions, Constants та Action Creators","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Factions-constants","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F02.actions-constants",{"title":2092,"path":2093,"stem":2094},"Логіка Reducers","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Freducers","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F03.reducers",{"title":2096,"path":2097,"stem":2098},"Комбінування Reducers (Root Reducer)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Fdata-flow","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F04.data-flow",{"title":2100,"path":2101,"stem":2102},"Підключення до React (React-Redux)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Freact-redux-connection","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F05.react-redux-connection",{"title":2104,"path":2105,"stem":2106},"Middleware та Асинхронність (Redux Thunk)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Fmiddleware-thunk","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F06.middleware-thunk",{"title":2108,"icon":132,"path":2109,"stem":2110,"children":2111,"page":59},"Transition To Rtk","\u002Fjavascript\u002Freact\u002Fredux\u002Ftransition-to-rtk","03.javascript\u002F04.react\u002F06.redux\u002F03.transition-to-rtk",[2112],{"title":2113,"path":2114,"stem":2115},"Проблеми класичного Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Ftransition-to-rtk\u002Fproblems-with-classic","03.javascript\u002F04.react\u002F06.redux\u002F03.transition-to-rtk\u002F01.problems-with-classic",{"title":2117,"icon":132,"path":2118,"stem":2119,"children":2120,"page":59},"Redux Toolkit","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit",[2121,2125,2129,2133],{"title":2122,"path":2123,"stem":2124},"Налаштування Store з configureStore","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fconfigure-store","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F01.configure-store",{"title":2126,"path":2127,"stem":2128},"createSlice: Революція в Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fcreate-slice","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F02.create-slice",{"title":2130,"path":2131,"stem":2132},"Асинхронність з createAsyncThunk","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fasync-thunks","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F03.async-thunks",{"title":2134,"path":2135,"stem":2136},"04. Entity Adapter: Керування нормалізованим станом","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fentity-adapter","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F04.entity-adapter",{"title":2138,"icon":92,"path":2139,"stem":2140,"children":2141,"page":59},"Advanced","\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced",[2142,2146,2150],{"title":2143,"path":2144,"stem":2145},"Мемоізація та Селектори: Повний Гайд по Reselect","\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced\u002Fselectors-reselect","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced\u002F01.selectors-reselect",{"title":2147,"path":2148,"stem":2149},"RTK Query: Архітектура Серверного Кешу","\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced\u002Frtk-query-intro","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced\u002F02.rtk-query-intro",{"title":1997,"path":2151,"stem":2152},"\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced\u002Farchitecture-best-practices","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced\u002F03.architecture-best-practices",{"title":2154,"icon":132,"path":2155,"stem":2156,"children":2157,"page":59},"Project Kanban","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban",[2158,2162,2166,2170,2174,2178],{"title":2159,"path":2160,"stem":2161},"Проєкт: Kanban Board (Trello Clone)","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fproject-overview","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F01.project-overview",{"title":2163,"path":2164,"stem":2165},"Налаштування та Типізація","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fsetup-and-types","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F02.setup-and-types",{"title":2167,"path":2168,"stem":2169},"Board Slice: Серце Дошки","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fboard-slice","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F03.board-slice",{"title":2171,"path":2172,"stem":2173},"Логіка Drag & Drop","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fdrag-and-drop-logic","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F04.drag-and-drop-logic",{"title":2175,"path":2176,"stem":2177},"Інтеграція з RTK Query","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Frtk-query-integration","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F05.rtk-query-integration",{"title":2179,"path":2180,"stem":2181},"Optimistic Updates","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Foptimistic-updates","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F06.optimistic-updates",{"title":2183,"icon":132,"path":2184,"stem":2185,"children":2186,"page":59},"Testing","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting","03.javascript\u002F04.react\u002F06.redux\u002F07.testing",[2187,2191,2195,2199,2203],{"title":2188,"path":2189,"stem":2190},"Тестування Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Fintro-testing","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F01.intro-testing",{"title":2192,"path":2193,"stem":2194},"Тестування Reducers","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-reducers","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F02.testing-reducers",{"title":2196,"path":2197,"stem":2198},"Тестування Селекторів","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-selectors","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F03.testing-selectors",{"title":2200,"path":2201,"stem":2202},"Тестування Компонентів (Integration)","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-components","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F04.testing-components",{"title":2204,"path":2205,"stem":2206},"Тестування Async Thunks","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-thunks","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F05.testing-thunks",{"title":2208,"icon":132,"path":2209,"stem":2210,"children":2211},"Ui Libraries","\u002Fjavascript\u002Freact\u002Fui-libraries","03.javascript\u002F04.react\u002F07.ui-libraries\u002Findex",[2212,2214,2218,2222,2226,2230,2234],{"title":2213,"path":2209,"stem":2210},"UI Бібліотеки в React",{"title":2215,"path":2216,"stem":2217},"Вступ до UI Бібліотек: Навіщо Винаходити Велосипед Двічі?","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fintroduction-to-ui-libraries","03.javascript\u002F04.react\u002F07.ui-libraries\u002F01.introduction-to-ui-libraries",{"title":2219,"path":2220,"stem":2221},"Філософія shadcn\u002Fui: \"Not a Component Library\"","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-philosophy","03.javascript\u002F04.react\u002F07.ui-libraries\u002F02.shadcn-philosophy",{"title":2223,"path":2224,"stem":2225},"Установка та Налаштування shadcn\u002Fui","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-installation","03.javascript\u002F04.react\u002F07.ui-libraries\u002F03.shadcn-installation",{"title":2227,"path":2228,"stem":2229},"Базові Компоненти shadcn\u002Fui: Фундамент Інтерфейсу","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-components-basics","03.javascript\u002F04.react\u002F07.ui-libraries\u002F04.shadcn-components-basics",{"title":2231,"path":2232,"stem":2233},"Компоненти Форм: Побудова Інтерактивних Form","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-components-forms","03.javascript\u002F04.react\u002F07.ui-libraries\u002F05.shadcn-components-forms",{"title":2235,"path":2236,"stem":2237},"Складні Компоненти: Dialog, Dropdown, Table та Command","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-components-advanced","03.javascript\u002F04.react\u002F07.ui-libraries\u002F06.shadcn-components-advanced",{"title":2239,"icon":2240,"path":2241,"stem":2242,"children":2243,"page":59},"TypeScript","i-devicon-typescript","\u002Fjavascript\u002Ftypescript","03.javascript\u002F05.typescript",[2244,2248,2252,2256,2260,2264,2268,2272],{"title":2245,"path":2246,"stem":2247},"TypeScript: Броня для вашого коду","\u002Fjavascript\u002Ftypescript\u002Fintro-and-basic-types","03.javascript\u002F05.typescript\u002F01.intro-and-basic-types",{"title":2249,"path":2250,"stem":2251},"Майстерність Моделювання Даних: Інтерфейси та Просунуті Типи","\u002Fjavascript\u002Ftypescript\u002Finterfaces-and-advanced-types","03.javascript\u002F05.typescript\u002F02.interfaces-and-advanced-types",{"title":2253,"path":2254,"stem":2255},"Алхімія Типів: Generics та Utility Types","\u002Fjavascript\u002Ftypescript\u002Fgenerics-and-utilities","03.javascript\u002F05.typescript\u002F03.generics-and-utilities",{"title":2257,"path":2258,"stem":2259},"Архітектура та Шаблони: Класи в TypeScript","\u002Fjavascript\u002Ftypescript\u002Fclasses-and-oop","03.javascript\u002F05.typescript\u002F04.classes-and-oop",{"title":2261,"path":2262,"stem":2263},"Продакшн та Екосистема: Advanced Config & Workflow","\u002Fjavascript\u002Ftypescript\u002Fadvanced-patterns-and-config","03.javascript\u002F05.typescript\u002F05.advanced-patterns-and-config",{"title":2265,"path":2266,"stem":2267},"TypeScript у світі React","\u002Fjavascript\u002Ftypescript\u002Freact-basics","03.javascript\u002F05.typescript\u002F06.react-basics",{"title":2269,"path":2270,"stem":2271},"React + TypeScript: Продвинуті патерни","\u002Fjavascript\u002Ftypescript\u002Freact-advanced","03.javascript\u002F05.typescript\u002F07.react-advanced",{"title":2273,"path":2274,"stem":2275},"React + TypeScript: Екосистема та бібліотеки","\u002Fjavascript\u002Ftypescript\u002Freact-ecosystem","03.javascript\u002F05.typescript\u002F08.react-ecosystem",{"title":2277,"path":2278,"stem":2279},"Atomic Design","\u002Fjavascript\u002Fatomic-design","03.javascript\u002F2.atomic-design",{"title":2281,"icon":2282,"path":2283,"stem":2284,"children":2285,"page":59},"Java","i-devicon-java","\u002Fjava","04.java",[2286,2289,2292,2296,2300,2304,2308],{"title":162,"path":2287,"stem":2288},"\u002Fjava\u002Fdata-mapper-part1","04.java\u002F01.data-mapper-part1",{"title":166,"path":2290,"stem":2291},"\u002Fjava\u002Fdata-mapper-part2","04.java\u002F02.data-mapper-part2",{"title":2293,"path":2294,"stem":2295},"Service Layer: Організація бізнес-логіки","\u002Fjava\u002Fservice-layer","04.java\u002F03.service-layer",{"title":2297,"path":2298,"stem":2299},"Rich Domain Model та State Pattern","\u002Fjava\u002Frich-domain-model","04.java\u002F04.rich-domain-model",{"title":2301,"path":2302,"stem":2303},"Патерни для складної бізнес-логіки","\u002Fjava\u002Fbusiness-logic-patterns","04.java\u002F05.business-logic-patterns",{"title":2305,"path":2306,"stem":2307},"Обробка помилок та валідація","\u002Fjava\u002Ferror-handling-validation","04.java\u002F06.error-handling-validation",{"title":2309,"path":2310,"stem":2311,"children":2312,"page":59},"Проектування баз даних","\u002Fjava\u002Fpr2","04.java\u002Fpr2",[2313,2317,2321,2325,2329,2333,2337,2341,2345,2349,2353,2357,2361,2365,2369,2373,2377,2381,2385,2389,2393,2397,2401,2405,2409,2413,2417,2421,2425,2429,2433,2437,2441,2445,2449,2453,2457],{"title":2314,"path":2315,"stem":2316},"Концептуальне моделювання: Мистецтво розуміння предметної області","\u002Fjava\u002Fpr2\u002Fconceptual-modeling","04.java\u002Fpr2\u002F01.conceptual-modeling",{"title":2318,"path":2319,"stem":2320},"Логічне моделювання: Від бізнес-ідей до структур даних","\u002Fjava\u002Fpr2\u002Flogical-modeling","04.java\u002Fpr2\u002F02.logical-modeling",{"title":2322,"path":2323,"stem":2324},"Нормалізація: Гігієна даних та боротьба з аномаліями","\u002Fjava\u002Fpr2\u002Fnormalization","04.java\u002Fpr2\u002F03.normalization",{"title":2326,"path":2327,"stem":2328},"Фізична схема: Від абстракції до DDL","\u002Fjava\u002Fpr2\u002Fphysical-schema","04.java\u002Fpr2\u002F04.physical-schema",{"title":2330,"path":2331,"stem":2332},"Архітектурна класифікація таблиць","\u002Fjava\u002Fpr2\u002Ftable-classification","04.java\u002Fpr2\u002F05.table-classification",{"title":2334,"path":2335,"stem":2336},"Database Migrations: Версіонування схеми з Flyway","\u002Fjava\u002Fpr2\u002Fdatabase-migrations","04.java\u002Fpr2\u002F06.database-migrations",{"title":2338,"path":2339,"stem":2340},"А що, якби це була не реляційна БД?","\u002Fjava\u002Fpr2\u002Fbeyond-relational","04.java\u002Fpr2\u002F07.beyond-relational",{"title":2342,"path":2343,"stem":2344},"Object-Relational Impedance Mismatch: Два світи, що не хочуть дружити","\u002Fjava\u002Fpr2\u002Fimpedance-mismatch","04.java\u002Fpr2\u002F09.impedance-mismatch",{"title":2346,"path":2347,"stem":2348},"JDBC: Перший контакт із базою даних","\u002Fjava\u002Fpr2\u002Fjdbc-fundamentals","04.java\u002Fpr2\u002F10.jdbc-fundamentals",{"title":2350,"path":2351,"stem":2352},"Якість коду: Spotless, SpotBugs та SonarQube","\u002Fjava\u002Fpr2\u002F10a.code-quality","04.java\u002Fpr2\u002F10a.code-quality",{"title":2354,"path":2355,"stem":2356},"Connection Pool: Патерн Object Pool для JDBC-з'єднань","\u002Fjava\u002Fpr2\u002Fconnection-pool","04.java\u002Fpr2\u002F11.connection-pool",{"title":2358,"path":2359,"stem":2360},"Row Data Gateway: Об'єкт як обгортка рядка таблиці","\u002Fjava\u002Fpr2\u002Frow-data-gateway","04.java\u002Fpr2\u002F12.row-data-gateway",{"title":2362,"path":2363,"stem":2364},"Table Data Gateway: Фасад таблиці як архітектурний відступ","\u002Fjava\u002Fpr2\u002Ftable-data-gateway","04.java\u002Fpr2\u002F13.table-data-gateway",{"title":2366,"path":2367,"stem":2368},"Repository + Data Mapper: Правильна шарова архітектура з JDBC","\u002Fjava\u002Fpr2\u002Frepository-data-mapper","04.java\u002Fpr2\u002F14.repository-data-mapper",{"title":2370,"path":2371,"stem":2372},"Identity Map: Кешування сутностей у рамках сесії","\u002Fjava\u002Fpr2\u002Fidentity-map","04.java\u002Fpr2\u002F15.identity-map",{"title":2374,"path":2375,"stem":2376},"Unit of Work: Відстеження змін і координація JDBC-транзакцій","\u002Fjava\u002Fpr2\u002Funit-of-work","04.java\u002Fpr2\u002F16.unit-of-work",{"title":2378,"path":2379,"stem":2380},"Strategy: Замінювані SQL-стратегії для підтримки різних СУБД","\u002Fjava\u002Fpr2\u002Fstrategy-sql","04.java\u002Fpr2\u002F17.strategy-sql",{"title":2382,"path":2383,"stem":2384},"Proxy: Lazy Loading для One-To-Many колекцій","\u002Fjava\u002Fpr2\u002Fproxy-lazy-loading","04.java\u002Fpr2\u002F18.proxy-lazy-loading",{"title":2386,"path":2387,"stem":2388},"Generic Repository через Java Reflection: анотації та динамічний SQL","\u002Fjava\u002Fpr2\u002Fgeneric-repository-reflection","04.java\u002Fpr2\u002F19.generic-repository-reflection",{"title":2390,"path":2391,"stem":2392},"Specification Pattern: Композиція бізнес-правил для складних запитів","\u002Fjava\u002Fpr2\u002Fspecification-pattern","04.java\u002Fpr2\u002F20.specification-pattern",{"title":2394,"path":2395,"stem":2396},"Розширені можливості Specification Pattern: підзапити, агрегації та гібридний підхід","\u002Fjava\u002Fpr2\u002F20a.advanced-specifications","04.java\u002Fpr2\u002F20a.advanced-specifications",{"title":2398,"path":2399,"stem":2400},"Асинхронність у JDBC: Від блокуючих викликів до CompletableFuture","\u002Fjava\u002Fpr2\u002Fasynchronous-jdbc","04.java\u002Fpr2\u002F21.asynchronous-jdbc",{"title":2402,"path":2403,"stem":2404},"Інтеграційне тестування JDBC-репозиторіїв: Embedded H2 та патерн AAA","\u002Fjava\u002Fpr2\u002Fintegration-testing-h2","04.java\u002Fpr2\u002F22.integration-testing-h2",{"title":2406,"path":2407,"stem":2408},"Testcontainers: Тестування з реальною PostgreSQL у Docker-контейнерах","\u002Fjava\u002Fpr2\u002Fintegration-testing-testcontainers","04.java\u002Fpr2\u002F23.integration-testing-testcontainers",{"title":2410,"path":2411,"stem":2412},"Google Guice: Впровадження залежностей у JavaFX-проєкті","\u002Fjava\u002Fpr2\u002Fdependency-injection-guice","04.java\u002Fpr2\u002F24.dependency-injection-guice",{"title":2414,"path":2415,"stem":2416},"JavaFX: Основи побудови графічних інтерфейсів","\u002Fjava\u002Fpr2\u002Fjavafx-fundamentals","04.java\u002Fpr2\u002F25.javafx-fundamentals",{"title":2418,"path":2419,"stem":2420},"Properties та Bindings: Реактивність у JavaFX","\u002Fjava\u002Fpr2\u002Fjavafx-properties-bindings","04.java\u002Fpr2\u002F26.javafx-properties-bindings",{"title":2422,"path":2423,"stem":2424},"MVC vs MVP vs MVVM: Еволюція архітектурних патернів UI","\u002Fjava\u002Fpr2\u002Fui-architecture-patterns","04.java\u002Fpr2\u002F27.ui-architecture-patterns",{"title":2426,"path":2427,"stem":2428},"MVVM на практиці: Побудова ViewModel","\u002Fjava\u002Fpr2\u002Fmvvm-viewmodel-implementation","04.java\u002Fpr2\u002F28.mvvm-viewmodel-implementation",{"title":2430,"path":2431,"stem":2432},"View та Controller: Зв'язування з ViewModel через FXML","\u002Fjava\u002Fpr2\u002Fmvvm-view-controller","04.java\u002Fpr2\u002F29.mvvm-view-controller",{"title":2434,"path":2435,"stem":2436},"Інтеграція MVVM з Guice: Автоматична ін'єкція залежностей","\u002Fjava\u002Fpr2\u002Fmvvm-guice-integration","04.java\u002Fpr2\u002F30.mvvm-guice-integration",{"title":2438,"path":2439,"stem":2440},"Валідація та обробка помилок у MVVM","\u002Fjava\u002Fpr2\u002Fmvvm-validation-error-handling","04.java\u002Fpr2\u002F31.mvvm-validation-error-handling",{"title":2442,"path":2443,"stem":2444},"Навігація та управління екранами у JavaFX MVVM","\u002Fjava\u002Fpr2\u002Fmvvm-navigation-screen-management","04.java\u002Fpr2\u002F32.mvvm-navigation-screen-management",{"title":2446,"path":2447,"stem":2448},"Тестування JavaFX MVVM-додатків","\u002Fjava\u002Fpr2\u002Fmvvm-testing","04.java\u002Fpr2\u002F33.mvvm-testing",{"title":2450,"path":2451,"stem":2452},"Стилізація та теми у JavaFX: CSS та User Experience","\u002Fjava\u002Fpr2\u002Fjavafx-styling-themes","04.java\u002Fpr2\u002F34.javafx-styling-themes",{"title":2454,"path":2455,"stem":2456},"AtlantaFX: Сучасні теми для JavaFX додатків","\u002Fjava\u002Fpr2\u002Fatlantafx-modern-themes","04.java\u002Fpr2\u002F35.atlantafx-modern-themes",{"title":2458,"path":2459,"stem":2460},"Пакування та розповсюдження JavaFX-додатків","\u002Fjava\u002Fpr2\u002Fjar-packaging-distribution","04.java\u002Fpr2\u002F36.jar-packaging-distribution",{"title":2462,"icon":2463,"path":2464,"stem":2465,"children":2466,"page":59},"Бази даних","i-lucide-database","\u002Fdatabases","06.databases",[2467,2497,2520,2557,2586,2604,2638,2650,2659],{"title":2468,"icon":2469,"path":2470,"stem":2471,"children":2472,"page":59},"Intro","i-lucide-play","\u002Fdatabases\u002Fintro","06.databases\u002F01.intro",[2473,2477,2481,2485,2489,2493],{"title":2474,"path":2475,"stem":2476},"Введення в теорію баз даних","\u002Fdatabases\u002Fintro\u002Fintroduction-to-databases","06.databases\u002F01.intro\u002F01.introduction-to-databases",{"title":2478,"path":2479,"stem":2480},"Реляційна модель даних","\u002Fdatabases\u002Fintro\u002Frelational-model-theory","06.databases\u002F01.intro\u002F02.relational-model-theory",{"title":2482,"path":2483,"stem":2484},"ER-моделювання","\u002Fdatabases\u002Fintro\u002Fer-modeling","06.databases\u002F01.intro\u002F03.er-modeling",{"title":2486,"path":2487,"stem":2488},"Логічне проектування БД","\u002Fdatabases\u002Fintro\u002Flogical-schema","06.databases\u002F01.intro\u002F04.logical-schema",{"title":2490,"path":2491,"stem":2492},"Класифікація таблиць","\u002Fdatabases\u002Fintro\u002Ftable-classification","06.databases\u002F01.intro\u002F05.table-classification",{"title":2494,"path":2495,"stem":2496},"PlantUML для баз даних","\u002Fdatabases\u002Fintro\u002Fplantuml-diagrams","06.databases\u002F01.intro\u002F06.plantuml-diagrams",{"title":2498,"icon":2463,"path":2499,"stem":2500,"children":2501,"page":59},"MS SQL Server Start","\u002Fdatabases\u002Fms-sql-server-start","06.databases\u002F02.ms-sql-server-start",[2502,2506,2512,2516],{"title":2503,"path":2504,"stem":2505},"Типи даних у MS SQL Server","\u002Fdatabases\u002Fms-sql-server-start\u002Fdata-types","06.databases\u002F02.ms-sql-server-start\u002F01.data-types",{"title":2507,"path":2508,"stem":2509,"children":2510},"Індекси у MS SQL Server","\u002Fdatabases\u002Fms-sql-server-start\u002Fsql-indexes","06.databases\u002F02.ms-sql-server-start\u002F02.sql-indexes",[2511],{"title":2507,"path":2508,"stem":2509},{"title":2513,"path":2514,"stem":2515},"Системні бази даних MS SQL Server","\u002Fdatabases\u002Fms-sql-server-start\u002Fsystem-databases","06.databases\u002F02.ms-sql-server-start\u002F03.system-databases",{"title":2517,"path":2518,"stem":2519},"Огляд мови SQL та запитів","\u002Fdatabases\u002Fms-sql-server-start\u002Fsql-queries-overview","06.databases\u002F02.ms-sql-server-start\u002F04.sql-queries-overview",{"title":2521,"icon":2463,"path":2522,"stem":2523,"children":2524,"page":59},"SQL","\u002Fdatabases\u002Fsql","06.databases\u002F03.sql",[2525,2529,2533,2537,2541,2545,2549,2553],{"title":2526,"path":2527,"stem":2528},"Налаштування демонстраційної бази даних","\u002Fdatabases\u002Fsql\u002Fsample-database-setup","06.databases\u002F03.sql\u002F00.sample-database-setup",{"title":2530,"path":2531,"stem":2532},"DDL - Створення таблиць (CREATE TABLE)","\u002Fdatabases\u002Fsql\u002Fddl-create-table","06.databases\u002F03.sql\u002F01.ddl-create-table",{"title":2534,"path":2535,"stem":2536},"DDL - Зміна та видалення таблиць (ALTER, DROP)","\u002Fdatabases\u002Fsql\u002Fddl-alter-drop-table","06.databases\u002F03.sql\u002F02.ddl-alter-drop-table",{"title":2538,"path":2539,"stem":2540},"SELECT запити - Основи","\u002Fdatabases\u002Fsql\u002Fselect-queries-fundamentals","06.databases\u002F03.sql\u002F03.select-queries-fundamentals",{"title":2542,"path":2543,"stem":2544},"SELECT запити - Розширені можливості","\u002Fdatabases\u002Fsql\u002Fselect-queries-advanced","06.databases\u002F03.sql\u002F04.select-queries-advanced",{"title":2546,"path":2547,"stem":2548},"INSERT запити - Додавання даних","\u002Fdatabases\u002Fsql\u002Finsert-queries","06.databases\u002F03.sql\u002F05.insert-queries",{"title":2550,"path":2551,"stem":2552},"UPDATE та DELETE запити","\u002Fdatabases\u002Fsql\u002Fupdate-delete-queries","06.databases\u002F03.sql\u002F06.update-delete-queries",{"title":2554,"path":2555,"stem":2556},"Транзакції в SQL","\u002Fdatabases\u002Fsql\u002Ftransactions","06.databases\u002F03.sql\u002F07.transactions",{"title":2558,"icon":2463,"path":2559,"stem":2560,"children":2561,"page":59},"Multi Table Databases","\u002Fdatabases\u002Fmulti-table-databases","06.databases\u002F04.multi-table-databases",[2562,2566,2570,2574,2578,2582],{"title":2563,"path":2564,"stem":2565},"Зв'язки та нормалізація БД","\u002Fdatabases\u002Fmulti-table-databases\u002Frelationships-and-normalization","06.databases\u002F04.multi-table-databases\u002F00.relationships-and-normalization",{"title":2567,"path":2568,"stem":2569},"INNER JOIN - З'єднання таблиць","\u002Fdatabases\u002Fmulti-table-databases\u002Finner-join","06.databases\u002F04.multi-table-databases\u002F01.inner-join",{"title":2571,"path":2572,"stem":2573},"OUTER JOINs - LEFT, RIGHT, FULL","\u002Fdatabases\u002Fmulti-table-databases\u002Fouter-joins","06.databases\u002F04.multi-table-databases\u002F02.outer-joins",{"title":2575,"path":2576,"stem":2577},"CROSS та SELF JOINs","\u002Fdatabases\u002Fmulti-table-databases\u002Fcross-self-joins","06.databases\u002F04.multi-table-databases\u002F03.cross-self-joins",{"title":2579,"path":2580,"stem":2581},"Підзапити (Subqueries)","\u002Fdatabases\u002Fmulti-table-databases\u002Fsubqueries","06.databases\u002F04.multi-table-databases\u002F04.subqueries",{"title":2583,"path":2584,"stem":2585},"Агрегації з JOIN","\u002Fdatabases\u002Fmulti-table-databases\u002Faggregations-with-joins","06.databases\u002F04.multi-table-databases\u002F05.aggregations-with-joins",{"title":2587,"icon":2588,"path":2589,"stem":2590,"children":2591,"page":59},"Aggregate Functions","i-lucide-calculator","\u002Fdatabases\u002Faggregate-functions","06.databases\u002F05.aggregate-functions",[2592,2596,2600],{"title":2593,"path":2594,"stem":2595},"Функції агрегування в MS SQL Server","\u002Fdatabases\u002Faggregate-functions\u002Fintroduction-aggregate-functions","06.databases\u002F05.aggregate-functions\u002F01.introduction-aggregate-functions",{"title":2597,"path":2598,"stem":2599},"Групування даних в MS SQL Server","\u002Fdatabases\u002Faggregate-functions\u002Fgrouping-data","06.databases\u002F05.aggregate-functions\u002F02.grouping-data",{"title":2601,"path":2602,"stem":2603},"Підзапити з агрегатними функціями","\u002Fdatabases\u002Faggregate-functions\u002Fsubqueries-aggregates","06.databases\u002F05.aggregate-functions\u002F03.subqueries-aggregates",{"title":2605,"icon":2606,"path":2607,"stem":2608,"children":2609,"page":59},"Тригери та зберігаємі процедури","i-lucide-database-zap","\u002Fdatabases\u002Ftriggers-stored-procedures","06.databases\u002F07.triggers-stored-procedures",[2610,2614,2618,2622,2626,2630,2634],{"title":2611,"path":2612,"stem":2613},"DML-тригери","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fdml-triggers","06.databases\u002F07.triggers-stored-procedures\u002F01.dml-triggers",{"title":2615,"path":2616,"stem":2617},"DDL-тригери","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fddl-triggers","06.databases\u002F07.triggers-stored-procedures\u002F02.ddl-triggers",{"title":2619,"path":2620,"stem":2621},"Transact-SQL розширення","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Ftransact-sql-extensions","06.databases\u002F07.triggers-stored-procedures\u002F03.transact-sql-extensions",{"title":2623,"path":2624,"stem":2625},"Транзакції","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Ftransactions","06.databases\u002F07.triggers-stored-procedures\u002F04.transactions",{"title":2627,"path":2628,"stem":2629},"Зберігаємі процедури","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fstored-procedures","06.databases\u002F07.triggers-stored-procedures\u002F05.stored-procedures",{"title":2631,"path":2632,"stem":2633},"Користувацькі функції","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fuser-defined-functions","06.databases\u002F07.triggers-stored-procedures\u002F06.user-defined-functions",{"title":2635,"path":2636,"stem":2637},"Безпека баз даних","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fsecurity","06.databases\u002F07.triggers-stored-procedures\u002F08.security",{"title":2635,"icon":793,"path":2639,"stem":2640,"children":2641,"page":59},"\u002Fdatabases\u002Fsecurity","06.databases\u002F08.security",[2642,2646],{"title":2643,"path":2644,"stem":2645},"Вступ до безпеки баз даних","\u002Fdatabases\u002Fsecurity\u002Fintroduction","06.databases\u002F08.security\u002F01.introduction",{"title":2647,"path":2648,"stem":2649},"Системні представлення та метадані","\u002Fdatabases\u002Fsecurity\u002Fsystem-views","06.databases\u002F08.security\u002F02.system-views",{"title":2651,"icon":2652,"path":2653,"stem":2654,"children":2655,"page":59},"Резервне копіювання та відновлення","i-lucide-database-backup","\u002Fdatabases\u002Fbackup-recovery","06.databases\u002F09.backup-recovery",[2656],{"title":2651,"path":2657,"stem":2658},"\u002Fdatabases\u002Fbackup-recovery\u002Fbackup-restore","06.databases\u002F09.backup-recovery\u002F01.backup-restore",{"title":2660,"icon":2661,"path":2662,"stem":2663,"children":2664,"page":59},"Повнотекстовий пошук","i-lucide-search","\u002Fdatabases\u002Ffull-text-search","06.databases\u002F10.full-text-search",[2665],{"title":2660,"path":2666,"stem":2667},"\u002Fdatabases\u002Ffull-text-search\u002Ffull-text-search","06.databases\u002F10.full-text-search\u002F01.full-text-search",{"title":2669,"icon":2670,"path":2671,"stem":2672,"children":2673,"page":59},"Tools","i-lucide-wrench","\u002Ftools","07.tools",[2674,2750],{"title":2675,"icon":2676,"path":2677,"stem":2678,"children":2679},"Docker","i-simple-icons-docker","\u002Ftools\u002Fdocker","07.tools\u002F01.docker\u002Findex",[2680,2682,2686,2690,2694,2698,2702,2706,2710,2714,2718,2722,2726,2730,2734,2738,2742,2746],{"title":2681,"path":2677,"stem":2678},"Docker: від нуля до production",{"title":2683,"path":2684,"stem":2685},"Контейнеризація — від проблеми до рішення","\u002Ftools\u002Fdocker\u002Fcontainerization-concept","07.tools\u002F01.docker\u002F01.containerization-concept",{"title":2687,"path":2688,"stem":2689},"Docker — що це і навіщо?","\u002Ftools\u002Fdocker\u002Fdocker-what-and-why","07.tools\u002F01.docker\u002F02.docker-what-and-why",{"title":2691,"path":2692,"stem":2693},"Архітектура Docker Engine","\u002Ftools\u002Fdocker\u002Fdocker-architecture","07.tools\u002F01.docker\u002F03.docker-architecture",{"title":2695,"path":2696,"stem":2697},"Встановлення Docker","\u002Ftools\u002Fdocker\u002Finstallation","07.tools\u002F01.docker\u002F04.installation",{"title":2699,"path":2700,"stem":2701},"Перший контейнер — docker run","\u002Ftools\u002Fdocker\u002Ffirst-container","07.tools\u002F01.docker\u002F05.first-container",{"title":2703,"path":2704,"stem":2705},"Життєвий цикл контейнера","\u002Ftools\u002Fdocker\u002Fcontainer-lifecycle","07.tools\u002F01.docker\u002F06.container-lifecycle",{"title":2707,"path":2708,"stem":2709},"Docker Images — фундаментальні концепції","\u002Ftools\u002Fdocker\u002Fdocker-images-fundamentals","07.tools\u002F01.docker\u002F07.docker-images-fundamentals",{"title":2711,"path":2712,"stem":2713},"Dockerfile — основи","\u002Ftools\u002Fdocker\u002Fdockerfile-basics","07.tools\u002F01.docker\u002F08.dockerfile-basics",{"title":2715,"path":2716,"stem":2717},"Dockerfile — просунуті техніки","\u002Ftools\u002Fdocker\u002Fdockerfile-advanced","07.tools\u002F01.docker\u002F09.dockerfile-advanced",{"title":2719,"path":2720,"stem":2721},"Build Context та кешування шарів","\u002Ftools\u002Fdocker\u002Fbuild-context-and-cache","07.tools\u002F01.docker\u002F10.build-context-and-cache",{"title":2723,"path":2724,"stem":2725},"Реєстри Docker-образів","\u002Ftools\u002Fdocker\u002Fimage-registries","07.tools\u002F01.docker\u002F11.image-registries",{"title":2727,"path":2728,"stem":2729},"Контейнеризація .NET додатків","\u002Ftools\u002Fdocker\u002Fdotnet-containerization","07.tools\u002F01.docker\u002F12.dotnet-containerization",{"title":2731,"path":2732,"stem":2733},"Томи та збереження даних","\u002Ftools\u002Fdocker\u002Fvolumes-and-data","07.tools\u002F01.docker\u002F13.volumes-and-data",{"title":2735,"path":2736,"stem":2737},"Основи мережі в Docker","\u002Ftools\u002Fdocker\u002Fnetworking-basics","07.tools\u002F01.docker\u002F14.networking-basics",{"title":2739,"path":2740,"stem":2741},"Змінні оточення та конфігурація","\u002Ftools\u002Fdocker\u002Fenvironment-and-configuration","07.tools\u002F01.docker\u002F15.environment-and-configuration",{"title":2743,"path":2744,"stem":2745},"Docker Compose — оркестрація контейнерів","\u002Ftools\u002Fdocker\u002Fdocker-compose-basics","07.tools\u002F01.docker\u002F16.docker-compose-basics",{"title":2747,"path":2748,"stem":2749},"Docker Compose — Multi-Service застосунки","\u002Ftools\u002Fdocker\u002Fcompose-multi-service","07.tools\u002F01.docker\u002F17.compose-multi-service",{"title":2751,"icon":2752,"path":2753,"stem":2754,"children":2755},"Kubernetes","simple-icons:kubernetes","\u002Ftools\u002Fkubernetes","07.tools\u002F02.kubernetes\u002Findex",[2756,2758,2762,2766,2770,2774,2778,2782,2786],{"title":2757,"path":2753,"stem":2754},"Kubernetes: від розробки до production",{"title":2759,"path":2760,"stem":2761},"Kubernetes — коли Docker Compose більше не вистачає","\u002Ftools\u002Fkubernetes\u002Fwhy-kubernetes","07.tools\u002F02.kubernetes\u002F01.why-kubernetes",{"title":2763,"path":2764,"stem":2765},"Архітектура Kubernetes — анатомія кластера","\u002Ftools\u002Fkubernetes\u002Fkubernetes-architecture","07.tools\u002F02.kubernetes\u002F02.kubernetes-architecture",{"title":2767,"path":2768,"stem":2769},"Локальне середовище — minikube, kind та k3s","\u002Ftools\u002Fkubernetes\u002Flocal-environment","07.tools\u002F02.kubernetes\u002F03.local-environment",{"title":2771,"path":2772,"stem":2773},"Pod — атомарна одиниця Kubernetes","\u002Ftools\u002Fkubernetes\u002Fpods-and-containers","07.tools\u002F02.kubernetes\u002F04.pods-and-containers",{"title":2775,"path":2776,"stem":2777},"Патерни використання Pod","\u002Ftools\u002Fkubernetes\u002Fpod-patterns","07.tools\u002F02.kubernetes\u002F05.pod-patterns",{"title":2779,"path":2780,"stem":2781},"Deployment — декларативне управління Pod","\u002Ftools\u002Fkubernetes\u002Fdeployment-basics","07.tools\u002F02.kubernetes\u002F06.deployment-basics",{"title":2783,"path":2784,"stem":2785},"Rolling Updates та управління життєвим циклом Deployment","\u002Ftools\u002Fkubernetes\u002Fdeployment-rolling-updates","07.tools\u002F02.kubernetes\u002F07.deployment-rolling-updates",{"title":2787,"path":2788,"stem":2789},"Service — мережева абстракція для Pod","\u002Ftools\u002Fkubernetes\u002Fservices-networking","07.tools\u002F02.kubernetes\u002F08.services-networking",{"title":2791,"icon":2792,"path":2793,"stem":2794,"children":2795,"page":59},"Software Engineering","i-lucide-code-2","\u002Fsoftware-engineering","09.software-engineering",[2796,2800,2804,2808,2812,2816,2820,2824,2828,2832,2836],{"title":2797,"path":2798,"stem":2799},"1. Аналіз предметної області. Експертні знання та складність","\u002Fsoftware-engineering\u002Fintro-subdomains","09.software-engineering\u002F01.intro-subdomains",{"title":2801,"path":2802,"stem":2803},"2. Обмежені контексти. Інтеграція обмежених контекстів","\u002Fsoftware-engineering\u002Fintegrating-limited-contexts","09.software-engineering\u002F02.integrating-limited-contexts",{"title":2805,"path":2806,"stem":2807},"3. Реалізація простої бізнес-логіки","\u002Fsoftware-engineering\u002Fsimple","09.software-engineering\u002F03.simple",{"title":2809,"path":2810,"stem":2811},"4. Опрацювання складної бізнес-логіки","\u002Fsoftware-engineering\u002Fcomplex-business-logic","09.software-engineering\u002F04.complex-business-logic",{"title":2813,"path":2814,"stem":2815},"5. Моделювання фактора часу. Подієво-орієнтована архітектура.","\u002Fsoftware-engineering\u002Fmodelling-the-time-factor","09.software-engineering\u002F05.modelling-the-time-factor",{"title":2817,"path":2818,"stem":2819},"6. Архітектурні патерни","\u002Fsoftware-engineering\u002Farchitectural-patterns","09.software-engineering\u002F06.architectural-patterns",{"title":2821,"path":2822,"stem":2823},"Паттерни взаємодії","\u002Fsoftware-engineering\u002Fpatterns-of-interaction","09.software-engineering\u002F07.patterns-of-interaction",{"title":2825,"path":2826,"stem":2827},"Евристика проєктування","\u002Fsoftware-engineering\u002Fdesign-heuristics","09.software-engineering\u002F08.design-heuristics",{"title":2829,"path":2830,"stem":2831},"Еволюція проєктних рішень","\u002Fsoftware-engineering\u002Fevolution-of-design-solutions","09.software-engineering\u002F09.evolution-of-design-solutions",{"title":2833,"path":2834,"stem":2835},"EventStorming","\u002Fsoftware-engineering\u002Feventstorming","09.software-engineering\u002F10.eventstorming",{"title":2837,"path":2838,"stem":2839},"DDD на практиці","\u002Fsoftware-engineering\u002Fddd-in-practice","09.software-engineering\u002F11.ddd-in-practice",{"title":2841,"icon":943,"path":2842,"stem":2843,"children":2844,"page":59},"DDD","\u002Fddd","10.ddd",[2845,2849,2853,2857,2861,2865,2869,2873,2877,2881,2885,2889,2893],{"title":2846,"path":2847,"stem":2848},"Аналіз предметної області","\u002Fddd\u002Fdomain-analysis","10.ddd\u002F01.domain-analysis",{"title":2850,"path":2851,"stem":2852},"Експертні знання про предметну область","\u002Fddd\u002Fdomain-expert-knowledge","10.ddd\u002F02.domain-expert-knowledge",{"title":2854,"path":2855,"stem":2856},"Як осмислити складність предметної області","\u002Fddd\u002Fmanaging-domain-complexity","10.ddd\u002F03.managing-domain-complexity",{"title":2858,"path":2859,"stem":2860},"Інтеграція обмежених контекстів","\u002Fddd\u002Fbounded-context-integration","10.ddd\u002F04.bounded-context-integration",{"title":2862,"path":2863,"stem":2864},"Реалізація простої бізнес-логіки","\u002Fddd\u002Fsimple-business-logic","10.ddd\u002F05.simple-business-logic",{"title":2866,"path":2867,"stem":2868},"Обробка складної бізнес-логіки","\u002Fddd\u002Fcomplex-business-logic","10.ddd\u002F06.complex-business-logic",{"title":2870,"path":2871,"stem":2872},"Моделювання фактора часу","\u002Fddd\u002Ftime-modeling","10.ddd\u002F07.time-modeling",{"title":2874,"path":2875,"stem":2876},"Глава 8. Архітектурні Патерни","\u002Fddd\u002Farchitectural-patterns","10.ddd\u002F08.architectural-patterns",{"title":2878,"path":2879,"stem":2880},"Глава 9. Патерни Взаємодії","\u002Fddd\u002Finteraction-patterns","10.ddd\u002F09.interaction-patterns",{"title":2882,"path":2883,"stem":2884},"Глава 10. Проектні Евристики","\u002Fddd\u002Fdesign-heuristics","10.ddd\u002F10.design-heuristics",{"title":2886,"path":2887,"stem":2888},"Глава 11. Еволюція Проектних Рішень","\u002Fddd\u002Fevolution-of-design-decisions","10.ddd\u002F11.evolution-of-design-decisions",{"title":2890,"path":2891,"stem":2892},"Глава 12. EventStorming","\u002Fddd\u002Fevent-storming","10.ddd\u002F12.event-storming",{"title":2894,"path":2895,"stem":2896},"Глава 13. DDD на Практиці","\u002Fddd\u002Fddd-in-practice","10.ddd\u002F13.ddd-in-practice",{"title":2898,"icon":2899,"path":2900,"stem":2901,"children":2902,"page":59},"Media Streaming","i-lucide-video","\u002Fmedia-streaming","11.media-streaming",[2903,2907,2911,2915,2919,2923,2927],{"title":2904,"path":2905,"stem":2906},"01. Магія Стрімінгу: Що відбувається, коли ви натискаєте \"Play\"","\u002Fmedia-streaming\u002Fintroduction","11.media-streaming\u002F01.introduction",{"title":2908,"path":2909,"stem":2910},"02. Анатомія Медіа: Кодеки, Контейнери та Стиснення","\u002Fmedia-streaming\u002Faudio-video-anatomy","11.media-streaming\u002F02.audio-video-anatomy",{"title":2912,"path":2913,"stem":2914},"03. The Gym: FFmpeg Deep Dive","\u002Fmedia-streaming\u002Fffmpeg-gym","11.media-streaming\u002F03.ffmpeg-gym",{"title":2916,"path":2917,"stem":2918},"04. HLS Protocol: HTTP Live Streaming у Деталях","\u002Fmedia-streaming\u002Fhls-protocol","11.media-streaming\u002F04.hls-protocol",{"title":2920,"path":2921,"stem":2922},"05. DASH Protocol: Відкритий Стандарт","\u002Fmedia-streaming\u002Fdash-protocol","11.media-streaming\u002F05.dash-protocol",{"title":2924,"path":2925,"stem":2926},"06. Масштабування: CDN та Adaptive Bitrate","\u002Fmedia-streaming\u002Fcdn-and-adaptive-bitrate","11.media-streaming\u002F06.cdn-and-adaptive-bitrate",{"title":2928,"path":2929,"stem":2930},"07. Війна із Затримкою (Latency)","\u002Fmedia-streaming\u002Frealtime-latency","11.media-streaming\u002F07.realtime-latency",{"title":2932,"icon":2933,"path":2934,"stem":2935,"children":2936,"page":59},"HTML & CSS","i-devicon-html5","\u002Fhtml-css","12.html-css",[2937,2941,2945,2949,2953,2957,2961,2965,2969,2973,2977,2981,2985,2989,2993,2997,3001,3005,3009,3013,3017,3021,3025,3029,3033,3037,3041,3045,3049,3053],{"title":2938,"path":2939,"stem":2940},"Вступ до HTML. Структура документа","\u002Fhtml-css\u002Fintro-html-structure","12.html-css\u002F01.intro-html-structure",{"title":2942,"path":2943,"stem":2944},"Форматування тексту в HTML","\u002Fhtml-css\u002Fhtml-text-formatting","12.html-css\u002F02.html-text-formatting",{"title":2946,"path":2947,"stem":2948},"Посилання та зображення в HTML","\u002Fhtml-css\u002Fhtml-links-images","12.html-css\u002F03.html-links-images",{"title":2950,"path":2951,"stem":2952},"Списки та таблиці в HTML","\u002Fhtml-css\u002Fhtml-lists-tables","12.html-css\u002F04.html-lists-tables",{"title":2954,"path":2955,"stem":2956},"Форми в HTML","\u002Fhtml-css\u002Fhtml-forms","12.html-css\u002F05.html-forms",{"title":2958,"path":2959,"stem":2960},"Семантичні елементи HTML5","\u002Fhtml-css\u002Fhtml-semantic-elements","12.html-css\u002F06.html-semantic-elements",{"title":2962,"path":2963,"stem":2964},"Мультимедіа та розширені елементи HTML","\u002Fhtml-css\u002Fhtml-multimedia-advanced","12.html-css\u002F07.html-multimedia-advanced",{"title":2966,"path":2967,"stem":2968},"Мікророзмітка та SEO в HTML","\u002Fhtml-css\u002Fhtml-microdata-seo","12.html-css\u002F08.html-microdata-seo",{"title":2970,"path":2971,"stem":2972},"Вступ до CSS. Селектори та специфічність","\u002Fhtml-css\u002Fcss-intro-selectors","12.html-css\u002F09.css-intro-selectors",{"title":2974,"path":2975,"stem":2976},"Блокова модель CSS. Відступи. Box Sizing","\u002Fhtml-css\u002Fcss-box-model","12.html-css\u002F10.css-box-model",{"title":2978,"path":2979,"stem":2980},"Розміри у CSS: повний довідник одиниць і ключових слів","\u002Fhtml-css\u002F10a.css-sizing","12.html-css\u002F10a.css-sizing",{"title":2982,"path":2983,"stem":2984},"Типографіка в CSS. Шрифти та текст","\u002Fhtml-css\u002Fcss-typography","12.html-css\u002F11.css-typography",{"title":2986,"path":2987,"stem":2988},"Кольори та фони в CSS","\u002Fhtml-css\u002Fcss-colors-backgrounds","12.html-css\u002F12.css-colors-backgrounds",{"title":2990,"path":2991,"stem":2992},"Тіні та фільтри в CSS","\u002Fhtml-css\u002F12b.css-shadows-filters","12.html-css\u002F12b.css-shadows-filters",{"title":2994,"path":2995,"stem":2996},"CSS Flexbox: Фундамент гнучких макетів","\u002Fhtml-css\u002Fcss-flexbox-fundamentals","12.html-css\u002F13.css-flexbox-fundamentals",{"title":2998,"path":2999,"stem":3000},"CSS Flexbox: Вирівнювання та Позиціонування","\u002Fhtml-css\u002Fcss-flexbox-alignment-sizing-and-patterns","12.html-css\u002F14.css-flexbox-alignment-sizing-and-patterns",{"title":3002,"path":3003,"stem":3004},"CSS Grid. Двовимірний макет. Частина 1","\u002Fhtml-css\u002Fcss-layout-grid","12.html-css\u002F15.css-layout-grid",{"title":3006,"path":3007,"stem":3008},"CSS Grid. Двовимірний макет. Частина 2","\u002Fhtml-css\u002Fcss-layout-grid-advanced","12.html-css\u002F16.css-layout-grid-advanced",{"title":3010,"path":3011,"stem":3012},"Позиціонування в CSS. Z-index. Stacking Context","\u002Fhtml-css\u002Fcss-positioning","12.html-css\u002F17.css-positioning",{"title":3014,"path":3015,"stem":3016},"CSS Анімації та Переходи","\u002Fhtml-css\u002Fcss-animations-transitions","12.html-css\u002F18.css-animations-transitions",{"title":3018,"path":3019,"stem":3020},"Адаптивний дизайн. Media Queries. Частина 1","\u002Fhtml-css\u002Fcss-responsive-media-queries","12.html-css\u002F19.css-responsive-media-queries",{"title":3022,"path":3023,"stem":3024},"Адаптивний дизайн. Частина 2: clamp(), Container Queries, @layer","\u002Fhtml-css\u002Fcss-responsive-advanced","12.html-css\u002F20.css-responsive-advanced",{"title":3026,"path":3027,"stem":3028},"CSS Custom Properties. Методології. Сучасний CSS","\u002Fhtml-css\u002Fcss-variables-methodologies","12.html-css\u002F21.css-variables-methodologies",{"title":3030,"path":3031,"stem":3032},"Сучасний CSS 2023–2025: Нові можливості","\u002Fhtml-css\u002Fcss-modern-features","12.html-css\u002F22.css-modern-features",{"title":3034,"path":3035,"stem":3036},"CSS Nesting, @layer, @scope та @property: нативний препроцесор","\u002Fhtml-css\u002F22a.css-nesting-modern-syntax","12.html-css\u002F22a.css-nesting-modern-syntax",{"title":3038,"path":3039,"stem":3040},"CSS для форм та інтерактивних станів","\u002Fhtml-css\u002Fcss-forms-interactive-states","12.html-css\u002F23.css-forms-interactive-states",{"title":3042,"path":3043,"stem":3044},"Доступність у CSS (CSS Accessibility)","\u002Fhtml-css\u002Fcss-accessibility","12.html-css\u002F24.css-accessibility",{"title":3046,"path":3047,"stem":3048},"CSS-функції та сучасні sizing primitives","\u002Fhtml-css\u002Fcss-functions-sizing","12.html-css\u002F25.css-functions-sizing",{"title":3050,"path":3051,"stem":3052},"Rendering Pipeline і CSS Performance","\u002Fhtml-css\u002Fcss-rendering-performance","12.html-css\u002F26.css-rendering-performance",{"title":3054,"path":3055,"stem":3056},"CSS Best Practices: типові ситуації та правильні рішення","\u002Fhtml-css\u002Fcss-best-practices","12.html-css\u002F27.css-best-practices",{"title":3058,"path":3059,"stem":3060,"children":3061,"page":59},"AWS","\u002Faws","13.aws",[3062,3066,3070,3074,3078,3082,3086,3090],{"title":3063,"path":3064,"stem":3065},"Реєстрація AWS акаунту та студентські програми","\u002Faws\u002Faccount-registration","13.aws\u002F00.account-registration",{"title":3067,"path":3068,"stem":3069},"Вступ до хмарних обчислень та AWS","\u002Faws\u002Fintroduction-to-cloud","13.aws\u002F01.introduction-to-cloud",{"title":3071,"path":3072,"stem":3073},"AWS IAM — Identity and Access Management","\u002Faws\u002Fiam","13.aws\u002F02.iam",{"title":3075,"path":3076,"stem":3077},"Docker та контейнеризація в AWS — ECR, ECS та Fargate","\u002Faws\u002Fdocker-ecs","13.aws\u002F03.docker-ecs",{"title":3079,"path":3080,"stem":3081},"Amazon EC2 — Elastic Compute Cloud","\u002Faws\u002Fec2","13.aws\u002F04.ec2",{"title":3083,"path":3084,"stem":3085},"Elastic Load Balancing та Auto Scaling","\u002Faws\u002Falb-asg","13.aws\u002F05.alb-asg",{"title":3087,"path":3088,"stem":3089},"Amazon S3 — Simple Storage Service","\u002Faws\u002Fs3","13.aws\u002F06.s3",{"title":3091,"path":3092,"stem":3093},"Amazon CloudFront — Content Delivery Network","\u002Faws\u002Fcloudfront","13.aws\u002F07.cloudfront",{"title":3095,"path":3096,"stem":3097,"children":3098,"page":59},"Tailwind","\u002Ftailwind","21.tailwind",[3099,3103,3107,3111,3115,3119,3123,3127],{"title":3100,"path":3101,"stem":3102},"Що таке Tailwind CSS і навіщо він потрібен","\u002Ftailwind\u002Ftailwind-intro-philosophy","21.tailwind\u002F01.tailwind-intro-philosophy",{"title":3104,"path":3105,"stem":3106},"Встановлення та налаштування Tailwind CSS v4","\u002Ftailwind\u002Ftailwind-installation-setup","21.tailwind\u002F02.tailwind-installation-setup",{"title":3108,"path":3109,"stem":3110},"Utility-класи: основи та система Tailwind","\u002Ftailwind\u002Ftailwind-utility-classes-core","21.tailwind\u002F03.tailwind-utility-classes-core",{"title":3112,"path":3113,"stem":3114},"Layout: Flexbox та Grid через Tailwind","\u002Ftailwind\u002Ftailwind-flexbox-grid","21.tailwind\u002F04.tailwind-flexbox-grid",{"title":3116,"path":3117,"stem":3118},"Кастомізація теми через @theme у Tailwind v4","\u002Ftailwind\u002Ftailwind-theme-customization","21.tailwind\u002F05.tailwind-theme-customization",{"title":3120,"path":3121,"stem":3122},"Варіанти: hover, focus, responsive, dark mode та нові v4","\u002Ftailwind\u002Ftailwind-variants-states","21.tailwind\u002F06.tailwind-variants-states",{"title":3124,"path":3125,"stem":3126},"Типографіка та система кольорів у Tailwind v4","\u002Ftailwind\u002Ftailwind-typography-colors","21.tailwind\u002F07.tailwind-typography-colors",{"title":3128,"path":3129,"stem":3130},"Компоненти та повторюваність: @apply, @utility та патерни","\u002Ftailwind\u002Ftailwind-components-patterns","21.tailwind\u002F08.tailwind-components-patterns",{"title":3132,"path":3133,"stem":3134},"Тестування компонентів діаграм","\u002Ftest-components","98.test-components",{"id":3136,"title":2410,"body":3137,"description":11270,"extension":11271,"links":11272,"meta":11273,"navigation":3192,"path":2411,"seo":11274,"stem":2412,"__hash__":11275},"docs\u002F04.java\u002Fpr2\u002F24.dependency-injection-guice.md",{"type":3138,"value":3139,"toc":11217},"minimark",[3140,3144,3149,3158,3376,3392,3395,3409,3422,3431,3446,3459,3506,3509,3513,3518,3524,3535,3543,3559,3563,3569,3589,3603,3759,3763,3770,4086,4100,4102,4106,4109,4131,4147,4156,4171,4186,4355,4359,4362,4457,4486,4488,4492,4497,5613,5616,5682,5700,5702,5706,5715,5719,5722,5808,5817,5821,5824,5910,5916,5920,5927,6222,6234,6240,6247,6255,6435,6445,6616,6624,6759,6808,6814,6820,7025,7040,7054,7056,7060,7066,7070,7095,7098,7137,7152,7159,7168,7260,7281,7419,7423,7426,8766,8769,8791,8812,8839,8856,8950,8952,8956,8967,8971,8989,9077,9084,9090,9102,9161,9164,9233,9240,9244,9257,9296,9301,9445,9449,9465,9479,9515,9517,9521,9528,9532,9538,9608,9619,9626,9637,9762,10146,10166,10170,10180,10506,10515,10530,10532,10536,10539,10543,10559,10592,10598,10620,10626,10720,10726,10728,10732,10771,10810,10830,10848,10861,10863,10867,10870,10876,10923,10933,10962,10975,10988,11022,11024,11028,11034,11175,11213],[3141,3142,2410],"h1",{"id":3143},"google-guice-впровадження-залежностей-у-javafx-проєкті",[3145,3146,3148],"h2",{"id":3147},"вступ-коли-граф-залежностей-стає-лабіринтом","Вступ: Коли граф залежностей стає лабіринтом",[3150,3151,3152,3153,3157],"p",{},"Уявіть типову сцену розробки. Ви відкриваєте контролер JavaFX, що відповідає за список треків аудіоплатформи. Там, всередині конструктора або методу ",[3154,3155,3156],"code",{},"initialize()",", ховається ось такий код:",[3159,3160,3165],"pre",{"className":3161,"code":3162,"language":3163,"meta":3164,"style":3164},"language-java shiki shiki-themes light-plus dark-plus dark-plus","public class TrackListController {\n\n    private final TrackService trackService;\n\n    public TrackListController() {\n        ConnectionManager connectionManager = new ConnectionManager(\n            \"jdbc:h2:.\u002Fdata\u002Faudiobook;MODE=PostgreSQL\",\n            \"sa\", \"\"\n        );\n        TrackRepository trackRepository = new JdbcTrackRepository(connectionManager);\n        TrackValidator validator = new TrackValidator();\n        AudioFileProcessor processor = new AudioFileProcessor(\"\u002Ftmp\u002Fuploads\");\n        this.trackService = new TrackService(trackRepository, validator, processor);\n    }\n}\n","java","",[3154,3166,3167,3187,3194,3213,3218,3230,3252,3262,3274,3280,3299,3318,3343,3364,3370],{"__ignoreMap":3164},[3168,3169,3172,3176,3179,3183],"span",{"class":3170,"line":3171},"line",1,[3168,3173,3175],{"class":3174},"su1O8","public",[3168,3177,3178],{"class":3174}," class",[3168,3180,3182],{"class":3181},"sN1BT"," TrackListController",[3168,3184,3186],{"class":3185},"sHH4Y"," {\n",[3168,3188,3190],{"class":3170,"line":3189},2,[3168,3191,3193],{"emptyLinePlaceholder":3192},true,"\n",[3168,3195,3197,3200,3203,3206,3210],{"class":3170,"line":3196},3,[3168,3198,3199],{"class":3174},"    private",[3168,3201,3202],{"class":3174}," final",[3168,3204,3205],{"class":3181}," TrackService",[3168,3207,3209],{"class":3208},"siwwj"," trackService",[3168,3211,3212],{"class":3185},";\n",[3168,3214,3216],{"class":3170,"line":3215},4,[3168,3217,3193],{"emptyLinePlaceholder":3192},[3168,3219,3221,3224,3227],{"class":3170,"line":3220},5,[3168,3222,3223],{"class":3174},"    public",[3168,3225,3182],{"class":3226},"s8Opu",[3168,3228,3229],{"class":3185},"() {\n",[3168,3231,3233,3236,3239,3242,3246,3249],{"class":3170,"line":3232},6,[3168,3234,3235],{"class":3181},"        ConnectionManager",[3168,3237,3238],{"class":3208}," connectionManager",[3168,3240,3241],{"class":3185}," = ",[3168,3243,3245],{"class":3244},"s8xlr","new",[3168,3247,3248],{"class":3226}," ConnectionManager",[3168,3250,3251],{"class":3185},"(\n",[3168,3253,3255,3259],{"class":3170,"line":3254},7,[3168,3256,3258],{"class":3257},"sbdoH","            \"jdbc:h2:.\u002Fdata\u002Faudiobook;MODE=PostgreSQL\"",[3168,3260,3261],{"class":3185},",\n",[3168,3263,3265,3268,3271],{"class":3170,"line":3264},8,[3168,3266,3267],{"class":3257},"            \"sa\"",[3168,3269,3270],{"class":3185},", ",[3168,3272,3273],{"class":3257},"\"\"\n",[3168,3275,3277],{"class":3170,"line":3276},9,[3168,3278,3279],{"class":3185},"        );\n",[3168,3281,3283,3286,3289,3291,3293,3296],{"class":3170,"line":3282},10,[3168,3284,3285],{"class":3181},"        TrackRepository",[3168,3287,3288],{"class":3208}," trackRepository",[3168,3290,3241],{"class":3185},[3168,3292,3245],{"class":3244},[3168,3294,3295],{"class":3226}," JdbcTrackRepository",[3168,3297,3298],{"class":3185},"(connectionManager);\n",[3168,3300,3302,3305,3308,3310,3312,3315],{"class":3170,"line":3301},11,[3168,3303,3304],{"class":3181},"        TrackValidator",[3168,3306,3307],{"class":3208}," validator",[3168,3309,3241],{"class":3185},[3168,3311,3245],{"class":3244},[3168,3313,3314],{"class":3226}," TrackValidator",[3168,3316,3317],{"class":3185},"();\n",[3168,3319,3321,3324,3327,3329,3331,3334,3337,3340],{"class":3170,"line":3320},12,[3168,3322,3323],{"class":3181},"        AudioFileProcessor",[3168,3325,3326],{"class":3208}," processor",[3168,3328,3241],{"class":3185},[3168,3330,3245],{"class":3244},[3168,3332,3333],{"class":3226}," AudioFileProcessor",[3168,3335,3336],{"class":3185},"(",[3168,3338,3339],{"class":3257},"\"\u002Ftmp\u002Fuploads\"",[3168,3341,3342],{"class":3185},");\n",[3168,3344,3346,3349,3352,3355,3357,3359,3361],{"class":3170,"line":3345},13,[3168,3347,3348],{"class":3174},"        this",[3168,3350,3351],{"class":3185},".",[3168,3353,3354],{"class":3208},"trackService",[3168,3356,3241],{"class":3185},[3168,3358,3245],{"class":3244},[3168,3360,3205],{"class":3226},[3168,3362,3363],{"class":3185},"(trackRepository, validator, processor);\n",[3168,3365,3367],{"class":3170,"line":3366},14,[3168,3368,3369],{"class":3185},"    }\n",[3168,3371,3373],{"class":3170,"line":3372},15,[3168,3374,3375],{"class":3185},"}\n",[3150,3377,3378,3379,3383,3384,3387,3388,3391],{},"З першого погляду — нічого катастрофічного. Але подивіться уважніше: контролер ",[3380,3381,3382],"strong",{},"сам вирішує",", як саме побудувати весь граф своїх залежностей. Він знає про ",[3154,3385,3386],{},"ConnectionManager",", про шлях до файлів завантаження, про ",[3154,3389,3390],{},"TrackValidator",". Він прив'язаний до конкретних реалізацій усіх своїх залежностей так міцно, наче виготовлений разом з ними на одному заводі.",[3150,3393,3394],{},"Проблема не в тому, що цей код не працює. Він працює. Проблема в тому, що відбувається далі.",[3150,3396,3397,3400,3401,3404,3405,3408],{},[3380,3398,3399],{},"Перший симптом — неможливість тестування."," Щоб написати unit-тест для ",[3154,3402,3403],{},"TrackListController",", вам доведеться або підіймати реальну базу даних H2 (це вже інтеграційний тест), або вдаватись до складних маніпуляцій з рефлексією. Підмінити ",[3154,3406,3407],{},"TrackRepository"," на mock-об'єкт неможливо: він жорстко закодований усередині конструктора.",[3150,3410,3411,3414,3415,3418,3419,3421],{},[3380,3412,3413],{},"Другий симптом — жорсткість при зміні реалізацій."," Вирішили замінити H2 на PostgreSQL? Або ",[3154,3416,3417],{},"AudioFileProcessor"," на хмарне сховище? Вам потрібно знайти кожне місце, де ці класи інстанціюються через ",[3154,3420,3245],{},", і оновити їх вручну. У великому проєкті таких місць можуть бути десятки.",[3150,3423,3424,3427,3428,3430],{},[3380,3425,3426],{},"Третій симптом — дублювання ініціалізації."," ",[3154,3429,3386],{}," зі своїм URL та обліковими даними створюється у кожному DAO-класі, у кожному тесті, у кожному місці, де потрібен доступ до бази. Жодної гарантії узгодженості.",[3150,3432,3433,3434,3437,3438,3441,3442,3445],{},"Це не погані розробники — це ",[3380,3435,3436],{},"природний наслідок"," відсутності систематичного підходу до управління залежностями. І саме цю проблему вирішує патерн ",[3380,3439,3440],{},"Dependency Injection (Впровадження залежностей)"," та бібліотека ",[3380,3443,3444],{},"Google Guice",", яку ми вивчатимемо у цій статті.",[3447,3448,3449,3451,3452],"note",{},[3380,3450,3444],{}," (вимовляється «джус») — легковагий фреймворк для впровадження залежностей у Java, розроблений командою Google у 2006 році. Він є основою для внутрішньої інфраструктури Google та використовується у тисячах відкритих і закритих проєктів. ",[3453,3454,3458],"a",{"href":3455,"rel":3456},"https:\u002F\u002Fgithub.com\u002Fgoogle\u002Fguice\u002Fwiki",[3457],"nofollow","Офіційна документація",[3460,3461,3462,3478,3492],"card-group",{},[3463,3464,3467,3475],"card",{"icon":3465,"title":3466},"i-heroicons-wrench-screwdriver","Ручне зв'язування",[3150,3468,3469,3427,3472],{},[3380,3470,3471],{},"Підхід:",[3154,3473,3474],{},"new ServiceA(new RepoA(new ConnMgr(...)))",[3150,3476,3477],{},"Кожен клас сам створює залежності. Швидко для маленьких проєктів, але стає некерованим при зростанні. Тестування — лише інтеграційне.",[3463,3479,3482,3489],{"icon":3480,"title":3481},"i-heroicons-building-storefront","Фабрики та Сервіс-локатор",[3150,3483,3484,3427,3486],{},[3380,3485,3471],{},[3154,3487,3488],{},"ServiceFactory.create(\"trackService\")",[3150,3490,3491],{},"Централізоване створення об'єктів, але клієнт сам шукає залежності. Залежність від фабрики — прихована зв'язаність.",[3463,3493,3496,3503],{"icon":3494,"title":3495},"i-heroicons-cube-transparent","IoC-контейнер (Guice)",[3150,3497,3498,3427,3500],{},[3380,3499,3471],{},[3154,3501,3502],{},"@Inject TrackService trackService",[3150,3504,3505],{},"Контейнер будує весь граф залежностей автоматично. Клас лише декларує потреби — хто і як їх задовольнить, вирішує зовні.",[3507,3508],"hr",{},[3145,3510,3512],{"id":3511},"фундаментальні-концепції-ioc-di-та-принцип-інверсії-залежностей","Фундаментальні концепції: IoC, DI та принцип інверсії залежностей",[3514,3515,3517],"h3",{"id":3516},"інверсія-управління-inversion-of-control","Інверсія управління (Inversion of Control)",[3150,3519,3520,3523],{},[3380,3521,3522],{},"Інверсія управління (IoC)"," — це архітектурний принцип, згідно з яким управління потоком виконання програми передається від самого коду до зовнішнього фреймворку або контейнера. Термін систематизований Мартіном Фаулером у 2004 році, хоча сам принцип існував значно раніше.",[3150,3525,3526,3527,3530,3531,3534],{},"Щоб зрозуміти «інверсію», необхідно спочатку усвідомити те, що саме інвертується. У традиційній програмі ",[3380,3528,3529],{},"ваш код"," контролює все: він викликає бібліотечні функції, вирішує, коли і що створювати, сам організовує порядок дій. У IoC-підході ",[3380,3532,3533],{},"фреймворк"," контролює загальний потік виконання, а ваш код лише реагує на події та надає конкретні реалізації через інтерфейси або анотації.",[3150,3536,3537,3538,3542],{},"Голлівудський принцип (Hollywood Principle) — влучна метафора для IoC: ",[3539,3540,3541],"em",{},"«Не телефонуйте нам, ми самі зателефонуємо вам»",". Ваш код не «шукає» залежності — контейнер сам «передзвонює» й надає їх.",[3150,3544,3545,3547,3548,3550,3551,3554,3555,3558],{},[3380,3546,3440],{}," — це конкретний механізм реалізації IoC. Замість того щоб клас самостійно створював свої залежності через ",[3154,3549,3245],{},", він ",[3380,3552,3553],{},"декларує"," їх через конструктор, сеттери або поля, а зовнішній код (контейнер) ",[3380,3556,3557],{},"впроваджує"," їх ззовні.",[3514,3560,3562],{"id":3561},"принцип-інверсії-залежностей-dip","Принцип інверсії залежностей (DIP)",[3150,3564,3565,3568],{},[3380,3566,3567],{},"Принцип інверсії залежностей"," (Dependency Inversion Principle, DIP) — п'ятий принцип SOLID — формулюється так:",[3570,3571,3572,3583],"ol",{},[3573,3574,3575,3576,3579,3580,3351],"li",{},"Модулі верхнього рівня ",[3380,3577,3578],{},"не повинні залежати"," від модулів нижнього рівня. Обидва повинні залежати від ",[3380,3581,3582],{},"абстракцій",[3573,3584,3585,3586,3588],{},"Абстракції ",[3380,3587,3578],{}," від деталей. Деталі повинні залежати від абстракцій.",[3150,3590,3591,3592,3595,3596,3599,3600,3602],{},"На практиці це означає: ",[3154,3593,3594],{},"TrackService"," (бізнес-логіка, верхній рівень) не повинен залежати від ",[3154,3597,3598],{},"JdbcTrackRepository"," (деталь реалізації, нижній рівень). Обидва мають залежати від інтерфейсу ",[3154,3601,3407],{},". Це та «абстракція», що дозволяє замінити реалізацію без зміни споживача.",[3150,3604,3605,3606,3616],{},"Рис. 1 ілюструє трансформацію архітектури:\n",[3607,3608,3615],"text",{"x":3609,"y":3610,"style":3611,"font-size":3612,"font-weight":3613,"fill":3614},"645","24","text-anchor: middle;","12","bold","#34d399","✅ Після DI: IoC-контейнер",[3170,3617,3624],{"x1":3618,"y1":3619,"x2":3618,"y2":3620,"stroke":3621,"stroke-width":3622,"stroke-dasharray":3623},"430","10","280","#6b7280","1","6,4",[3625,3626,3633,3638,3644,3648],"rect",{"x":3619,"y":3627,"width":3628,"height":3629,"rx":3619,"fill":3630,"stroke":3631,"stroke-width":3632},"38",200,56,"rgba(239,68,68,0.08)","#f87171","1.5",[3607,3634,3403],{"x":3635,"y":3636,"style":3611,"font-size":3637,"font-weight":3613,"fill":3631},"110","58","11",[3607,3639,3643],{"x":3635,"y":3640,"style":3611,"font-size":3641,"fill":3642},"74","9","#9ca3af","new TrackService(...)",[3607,3645,3647],{"x":3635,"y":3646,"style":3611,"font-size":3641,"fill":3642},"87","new JdbcTrackRepository(...)",[3625,3649,3657,3662],{"x":3650,"y":3651,"width":3652,"height":3653,"rx":3654,"fill":3655,"stroke":3656,"stroke-width":3622},"250","42",170,28,"7","rgba(239,68,68,0.06)","rgba(239,68,68,0.4)",[3607,3658,3594],{"x":3659,"y":3660,"style":3611,"font-size":3619,"fill":3661},"335","61","#fca5a5",[3170,3663,3669],{"x1":3664,"y1":3665,"x2":3650,"y2":3666,"stroke":3631,"stroke-width":3667,"marker-end":3668},"210","64","56","1.2","url(#arr-red)",[3625,3670,3672,3675],{"x":3650,"y":3671,"width":3652,"height":3653,"rx":3654,"fill":3655,"stroke":3656,"stroke-width":3622},"82",[3607,3673,3598],{"x":3659,"y":3674,"style":3611,"font-size":3619,"fill":3661},"101",[3170,3676,3679],{"x1":3664,"y1":3677,"x2":3650,"y2":3678,"stroke":3631,"stroke-width":3667,"marker-end":3668},"70","91",[3625,3680,3682,3685],{"x":3650,"y":3681,"width":3652,"height":3653,"rx":3654,"fill":3655,"stroke":3656,"stroke-width":3622},"122",[3607,3683,3386],{"x":3659,"y":3684,"style":3611,"font-size":3619,"fill":3661},"141",[3170,3686,3687],{"x1":3659,"y1":3635,"x2":3659,"y2":3681,"stroke":3631,"stroke-width":3667,"marker-end":3668},[3625,3688,3690,3693],{"x":3650,"y":3689,"width":3652,"height":3653,"rx":3654,"fill":3655,"stroke":3656,"stroke-width":3622},"162",[3607,3691,3417],{"x":3659,"y":3692,"style":3611,"font-size":3619,"fill":3661},"181",[3170,3694,3697,3702],{"x1":3664,"y1":3695,"x2":3650,"y2":3696,"stroke":3631,"stroke-width":3667,"marker-end":3668},"78","171",[3607,3698,3701],{"x":3699,"y":3700,"style":3611,"font-size":3641,"fill":3631},"215","225","Контролер знає ВСЕ про реалізації",[3625,3703,3708,3713,3717,3721],{"x":3704,"y":3627,"width":3652,"height":3629,"rx":3619,"fill":3705,"stroke":3706,"stroke-width":3707},"440","rgba(96,165,250,0.08)","#60a5fa","2",[3607,3709,3712],{"x":3710,"y":3711,"style":3611,"font-size":3637,"font-weight":3613,"fill":3706},"525","60","Guice Injector",[3607,3714,3716],{"x":3710,"y":3715,"style":3611,"font-size":3641,"fill":3642},"76","Module + Bindings",[3607,3718,3720],{"x":3710,"y":3719,"style":3611,"font-size":3641,"fill":3642},"88","будує граф автоматично",[3625,3722,3726,3729],{"x":3723,"y":3651,"width":3628,"height":3653,"rx":3654,"fill":3724,"stroke":3725,"stroke-width":3632},"650","rgba(52,211,153,0.08)","rgba(52,211,153,0.4)",[3607,3727,3403],{"x":3728,"y":3660,"style":3611,"font-size":3619,"fill":3614},"750",[3170,3730,3733],{"x1":3731,"y1":3711,"x2":3723,"y2":3666,"stroke":3614,"stroke-width":3632,"marker-end":3732},"610","url(#arr-green)",[3625,3734,3735,3737],{"x":3723,"y":3671,"width":3628,"height":3653,"rx":3654,"fill":3724,"stroke":3725,"stroke-width":3632},[3607,3736,3594],{"x":3728,"y":3674,"style":3611,"font-size":3619,"fill":3614},[3170,3738,3740],{"x1":3731,"y1":3739,"x2":3723,"y2":3678,"stroke":3614,"stroke-width":3632,"marker-end":3732},"65",[3625,3741,3742,3744],{"x":3723,"y":3681,"width":3628,"height":3653,"rx":3654,"fill":3724,"stroke":3725,"stroke-width":3632},[3607,3743,3598],{"x":3728,"y":3684,"style":3611,"font-size":3619,"fill":3614},[3170,3745,3748],{"x1":3731,"y1":3746,"x2":3723,"y2":3747,"stroke":3614,"stroke-width":3632,"marker-end":3732},"67","131",[3625,3749,3750,3752],{"x":3723,"y":3689,"width":3628,"height":3653,"rx":3654,"fill":3724,"stroke":3725,"stroke-width":3632},[3607,3751,3386],{"x":3728,"y":3692,"style":3611,"font-size":3619,"fill":3614},[3170,3753,3755,3758],{"x1":3731,"y1":3754,"x2":3723,"y2":3696,"stroke":3614,"stroke-width":3632,"marker-end":3732},"69",[3607,3756,3757],{"x":3609,"y":3700,"style":3611,"font-size":3641,"fill":3614},"Контролер знає лише інтерфейси","\n\n::",[3514,3760,3762],{"id":3761},"три-типи-впровадження-залежностей","Три типи впровадження залежностей",[3150,3764,3765,3766,3769],{},"Специфікація Jakarta Inject (колишня ",[3154,3767,3768],{},"javax.inject",") визначає три місця, де можуть впроваджуватись залежності. Guice підтримує всі три, але надає різні рекомендації щодо їх застосування.",[3771,3772,3773,3902,3983],"tabs",{},[3774,3775,3777,3887],"tabs-item",{"label":3776},"Constructor Injection (рекомендований)",[3159,3778,3780],{"className":3161,"code":3779,"language":3163,"meta":3164,"style":3164},"public class TrackService {\n\n    private final TrackRepository repository;\n    private final TrackValidator validator;\n\n    @Inject\n    public TrackService(TrackRepository repository, TrackValidator validator) {\n        this.repository = repository;\n        this.validator  = validator;\n    }\n}\n",[3154,3781,3782,3792,3796,3810,3822,3826,3834,3855,3867,3879,3883],{"__ignoreMap":3164},[3168,3783,3784,3786,3788,3790],{"class":3170,"line":3171},[3168,3785,3175],{"class":3174},[3168,3787,3178],{"class":3174},[3168,3789,3205],{"class":3181},[3168,3791,3186],{"class":3185},[3168,3793,3794],{"class":3170,"line":3189},[3168,3795,3193],{"emptyLinePlaceholder":3192},[3168,3797,3798,3800,3802,3805,3808],{"class":3170,"line":3196},[3168,3799,3199],{"class":3174},[3168,3801,3202],{"class":3174},[3168,3803,3804],{"class":3181}," TrackRepository",[3168,3806,3807],{"class":3208}," repository",[3168,3809,3212],{"class":3185},[3168,3811,3812,3814,3816,3818,3820],{"class":3170,"line":3215},[3168,3813,3199],{"class":3174},[3168,3815,3202],{"class":3174},[3168,3817,3314],{"class":3181},[3168,3819,3307],{"class":3208},[3168,3821,3212],{"class":3185},[3168,3823,3824],{"class":3170,"line":3220},[3168,3825,3193],{"emptyLinePlaceholder":3192},[3168,3827,3828,3831],{"class":3170,"line":3232},[3168,3829,3830],{"class":3185},"    @",[3168,3832,3833],{"class":3181},"Inject\n",[3168,3835,3836,3838,3840,3842,3844,3846,3848,3850,3852],{"class":3170,"line":3254},[3168,3837,3223],{"class":3174},[3168,3839,3205],{"class":3226},[3168,3841,3336],{"class":3185},[3168,3843,3407],{"class":3181},[3168,3845,3807],{"class":3208},[3168,3847,3270],{"class":3185},[3168,3849,3390],{"class":3181},[3168,3851,3307],{"class":3208},[3168,3853,3854],{"class":3185},") {\n",[3168,3856,3857,3859,3861,3864],{"class":3170,"line":3264},[3168,3858,3348],{"class":3174},[3168,3860,3351],{"class":3185},[3168,3862,3863],{"class":3208},"repository",[3168,3865,3866],{"class":3185}," = repository;\n",[3168,3868,3869,3871,3873,3876],{"class":3170,"line":3276},[3168,3870,3348],{"class":3174},[3168,3872,3351],{"class":3185},[3168,3874,3875],{"class":3208},"validator",[3168,3877,3878],{"class":3185},"  = validator;\n",[3168,3880,3881],{"class":3170,"line":3282},[3168,3882,3369],{"class":3185},[3168,3884,3885],{"class":3170,"line":3301},[3168,3886,3375],{"class":3185},[3150,3888,3889,3890,3893,3894,3897,3898,3901],{},"Guice бачить анотацію ",[3154,3891,3892],{},"@Inject"," на конструкторі та автоматично передає всі параметри. Поля оголошені ",[3154,3895,3896],{},"final"," — залежності незмінні після ініціалізації. Клас ",[3380,3899,3900],{},"неможливо"," створити без усіх залежностей: це унеможливлює часткову ініціалізацію. Для тестування достатньо передати mock у конструктор напряму, без жодних фреймворків.",[3774,3903,3905,3969],{"label":3904},"Field Injection (зручний, але з нюансами)",[3159,3906,3908],{"className":3161,"code":3907,"language":3163,"meta":3164,"style":3164},"public class TrackService {\n\n    @Inject\n    private TrackRepository repository;  \u002F\u002F Guice встановлює через рефлексію\n\n    @Inject\n    private TrackValidator validator;\n}\n",[3154,3909,3910,3920,3924,3930,3945,3949,3955,3965],{"__ignoreMap":3164},[3168,3911,3912,3914,3916,3918],{"class":3170,"line":3171},[3168,3913,3175],{"class":3174},[3168,3915,3178],{"class":3174},[3168,3917,3205],{"class":3181},[3168,3919,3186],{"class":3185},[3168,3921,3922],{"class":3170,"line":3189},[3168,3923,3193],{"emptyLinePlaceholder":3192},[3168,3925,3926,3928],{"class":3170,"line":3196},[3168,3927,3830],{"class":3185},[3168,3929,3833],{"class":3181},[3168,3931,3932,3934,3936,3938,3941],{"class":3170,"line":3215},[3168,3933,3199],{"class":3174},[3168,3935,3804],{"class":3181},[3168,3937,3807],{"class":3208},[3168,3939,3940],{"class":3185},";  ",[3168,3942,3944],{"class":3943},"spJ8K","\u002F\u002F Guice встановлює через рефлексію\n",[3168,3946,3947],{"class":3170,"line":3220},[3168,3948,3193],{"emptyLinePlaceholder":3192},[3168,3950,3951,3953],{"class":3170,"line":3232},[3168,3952,3830],{"class":3185},[3168,3954,3833],{"class":3181},[3168,3956,3957,3959,3961,3963],{"class":3170,"line":3254},[3168,3958,3199],{"class":3174},[3168,3960,3314],{"class":3181},[3168,3962,3307],{"class":3208},[3168,3964,3212],{"class":3185},[3168,3966,3967],{"class":3170,"line":3264},[3168,3968,3375],{"class":3185},[3150,3970,3971,3972,3974,3975,3978,3979,3982],{},"Синтаксично коротший, але має суттєві недоліки: поля не можуть бути ",[3154,3973,3896],{},", залежності невидимі в публічному API класу, а для тестування без Guice потрібна рефлексія або ",[3154,3976,3977],{},"@InjectMocks"," від Mockito. Guice-документація ",[3380,3980,3981],{},"не рекомендує"," цей підхід для нових проєктів.",[3774,3984,3986,4076],{"label":3985},"Method Injection (для опціональних залежностей)",[3159,3987,3989],{"className":3161,"code":3988,"language":3163,"meta":3164,"style":3164},"public class TrackService {\n\n    private TrackEventListener listener;\n\n    @Inject(optional = true)\n    public void setEventListener(TrackEventListener listener) {\n        this.listener = listener;\n    }\n}\n",[3154,3990,3991,4001,4005,4017,4021,4037,4056,4068,4072],{"__ignoreMap":3164},[3168,3992,3993,3995,3997,3999],{"class":3170,"line":3171},[3168,3994,3175],{"class":3174},[3168,3996,3178],{"class":3174},[3168,3998,3205],{"class":3181},[3168,4000,3186],{"class":3185},[3168,4002,4003],{"class":3170,"line":3189},[3168,4004,3193],{"emptyLinePlaceholder":3192},[3168,4006,4007,4009,4012,4015],{"class":3170,"line":3196},[3168,4008,3199],{"class":3174},[3168,4010,4011],{"class":3181}," TrackEventListener",[3168,4013,4014],{"class":3208}," listener",[3168,4016,3212],{"class":3185},[3168,4018,4019],{"class":3170,"line":3215},[3168,4020,3193],{"emptyLinePlaceholder":3192},[3168,4022,4023,4025,4028,4031,4034],{"class":3170,"line":3220},[3168,4024,3830],{"class":3185},[3168,4026,4027],{"class":3181},"Inject",[3168,4029,4030],{"class":3185},"(optional = ",[3168,4032,4033],{"class":3174},"true",[3168,4035,4036],{"class":3185},")\n",[3168,4038,4039,4041,4044,4047,4049,4052,4054],{"class":3170,"line":3232},[3168,4040,3223],{"class":3174},[3168,4042,4043],{"class":3181}," void",[3168,4045,4046],{"class":3226}," setEventListener",[3168,4048,3336],{"class":3185},[3168,4050,4051],{"class":3181},"TrackEventListener",[3168,4053,4014],{"class":3208},[3168,4055,3854],{"class":3185},[3168,4057,4058,4060,4062,4065],{"class":3170,"line":3254},[3168,4059,3348],{"class":3174},[3168,4061,3351],{"class":3185},[3168,4063,4064],{"class":3208},"listener",[3168,4066,4067],{"class":3185}," = listener;\n",[3168,4069,4070],{"class":3170,"line":3264},[3168,4071,3369],{"class":3185},[3168,4073,4074],{"class":3170,"line":3276},[3168,4075,3375],{"class":3185},[3150,4077,4078,4079,4082,4083,4085],{},"Атрибут ",[3154,4080,4081],{},"optional = true"," дозволяє Guice пропустити впровадження, якщо прив'язки для ",[3154,4084,4051],{}," не знайдено у жодному модулі. Доречний для залежностей, що підключаються лише в окремих конфігураціях — наприклад, listener для метрик у production, відсутній у тестах.",[4087,4088,4089,4092,4093,4096,4097,4099],"tip",{},[3380,4090,4091],{},"Рекомендація Guice:"," завжди надавайте перевагу ",[3380,4094,4095],{},"Constructor Injection",". Він є найексплікитнішим, найбезпечнішим і найзручнішим для тестування. Field Injection виправданий лише для ситуацій, коли конструктор недоступний (наприклад, при успадкуванні від класу без ",[3154,4098,3892],{}," конструктора).",[3507,4101],{},[3145,4103,4105],{"id":4104},"архітектура-google-guice-ключові-компоненти","Архітектура Google Guice: ключові компоненти",[3150,4107,4108],{},"Перш ніж писати перший код з Guice, необхідно зрозуміти концептуальну модель бібліотеки. Guice побудований навколо чотирьох фундаментальних абстракцій, що взаємодіють у суворо визначеному порядку.",[3150,4110,4111,4117,4118,4120,4121,4123,4124,4127,4128,3351],{},[3380,4112,4113,4116],{},[3154,4114,4115],{},"Module"," (Модуль)"," — це декларативний опис зв'язувань (bindings). Модуль — це місце, де ви як розробник кажете Guice: «якщо хтось попросить ",[3154,4119,3407],{},", надай йому ",[3154,4122,3598],{},"». Модуль реалізується через успадкування від ",[3154,4125,4126],{},"AbstractModule"," та перевизначення методу ",[3154,4129,4130],{},"configure()",[3150,4132,4133,4139,4140,4143,4144,4146],{},[3380,4134,4135,4138],{},[3154,4136,4137],{},"Injector"," (Інжектор)"," — центральний реєстр усіх прив'язок і фабрика об'єктів. Він створюється один раз, на старті застосунку, через ",[3154,4141,4142],{},"Guice.createInjector(module)",". Після цього ",[3154,4145,4137],{}," є єдиним джерелом правди щодо того, які об'єкти і як створювати. У JavaFX-застосунку він живе протягом усього часу роботи програми.",[3150,4148,4149,4155],{},[3380,4150,4151,4154],{},[3154,4152,4153],{},"Binding"," (Прив'язка)"," — зв'язок між ключем (зазвичай інтерфейсом або класом) та способом його задоволення (реалізацією, провайдером або екземпляром). Bindings описуються у Module і реєструються в Injector.",[3150,4157,4158,4162,4163,4166,4167,4170],{},[3380,4159,4160],{},[3154,4161,3892],{}," — анотація з пакету ",[3154,4164,4165],{},"jakarta.inject"," (або ",[3154,4168,4169],{},"com.google.inject","), що позначає точки впровадження: конструктори, поля або методи. Це єдиний маркер, якого потребує клас; він не прив'язує клас до Guice як до фреймворку.",[3150,4172,4173,4178,4179,4182,4183,4185],{},[3380,4174,4175],{},[3154,4176,4177],{},"Provider\u003CT>"," — функціональний інтерфейс для ліниво або параметризовано створюваних об'єктів. Guice може впроваджувати ",[3154,4180,4181],{},"Provider\u003CTrackService>"," замість ",[3154,4184,3594],{}," — це дає отримувачу контроль над часом та кількістю створення екземплярів.",[4187,4188,4195,4196,4195,4223,4195,4229,4195,4234,4195,4238,4195,4245,4195,4250,4195,4252,4195,4256,4195,4259,4195,4263,4195,4266,4195,4270,4195,4278,4195,4283,4195,4286,4195,4289,4195,4293,4195,4298,4195,4301,4195,4305,4195,4308,4195,4310,4195,4313,4195,4316,4195,4319,4195,4322,4195,4325,4195,4328,4195,4332,4195,4338,4195,4342,4195,4345,4195,4347,4195,4351],"svg",{"viewBox":4189,"className":4190,"xmlns":4194},"0 0 860 300",[4191,4192,4193],"w-full","h-auto","block","http:\u002F\u002Fwww.w3.org\u002F2000\u002Fsvg","\n  ",[4197,4198,4199,4200,4199,4212,4199,4217,4195],"defs",{},"\n    ",[4201,4202,4208],"marker",{"id":4203,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-b2","8","6","3","auto",[4209,4210],"path",{"d":4211,"fill":3706},"M0,0 L0,6 L8,3 z",[4201,4213,4215],{"id":4214,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-g2",[4209,4216],{"d":4211,"fill":3614},[4201,4218,4220],{"id":4219,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-a2",[4209,4221],{"d":4211,"fill":4222},"#f59e0b",[3625,4224],{"x":3619,"y":4225,"width":4226,"height":4227,"rx":3612,"fill":4228,"stroke":3706,"strokeWidth":3632},"30","260","200","rgba(96,165,250,0.06)",[3607,4230,4233],{"x":4231,"y":4232,"style":3611,"fontSize":3612,"fontWeight":3613,"fill":3706},"140","52","AppModule",[3607,4235,4237],{"x":4231,"y":4236,"style":3611,"fontSize":3641,"fill":3642},"68","extends AbstractModule",[3625,4239],{"x":4240,"y":3695,"width":4241,"height":3610,"rx":4242,"fill":4243,"stroke":4244,"strokeWidth":3622},"25","230","5","rgba(96,165,250,0.1)","rgba(96,165,250,0.3)",[3607,4246,4249],{"x":4231,"y":4247,"style":3611,"fontSize":3641,"fill":4248},"94","#93c5fd","TrackRepository → JdbcTrackRepository",[3625,4251],{"x":4240,"y":3635,"width":4241,"height":3610,"rx":4242,"fill":4243,"stroke":4244,"strokeWidth":3622},[3607,4253,4255],{"x":4231,"y":4254,"style":3611,"fontSize":3641,"fill":4248},"126","AuthorRepository → JdbcAuthorRepository",[3625,4257],{"x":4240,"y":4258,"width":4241,"height":3610,"rx":4242,"fill":4243,"stroke":4244,"strokeWidth":3622},"142",[3607,4260,4262],{"x":4231,"y":4261,"style":3611,"fontSize":3641,"fill":4248},"158","\"dbUrl\" → \"jdbc:h2:.\u002Fdata\u002F...\"",[3625,4264],{"x":4240,"y":4265,"width":4241,"height":3610,"rx":4242,"fill":4243,"stroke":4244,"strokeWidth":3622},"174",[3607,4267,4269],{"x":4231,"y":4268,"style":3611,"fontSize":3641,"fill":4248},"190","ConnectionManager → @Singleton",[3625,4271],{"x":4272,"y":4273,"width":4274,"height":4275,"rx":4276,"fill":4277,"stroke":4222,"strokeWidth":3707},"330","80","170","100","14","rgba(245,158,11,0.08)",[3607,4279,4137],{"x":4280,"y":4281,"style":3611,"fontSize":4282,"fontWeight":3613,"fill":4222},"415","108","13",[3607,4284,4285],{"x":4280,"y":4254,"style":3611,"fontSize":3641,"fill":3642},"Guice.createInjector",[3607,4287,4288],{"x":4280,"y":4231,"style":3611,"fontSize":3641,"fill":3642},"(new AppModule())",[3607,4290,4292],{"x":4280,"y":4261,"style":3611,"fontSize":3641,"fill":4291},"#fbbf24","реєстр + фабрика",[3170,4294],{"x1":4295,"y1":4296,"x2":4272,"y2":4296,"stroke":3706,"strokeWidth":3632,"markerEnd":4297},"270","130","url(#arr-b2)",[3607,4299,4130],{"x":4300,"y":3681,"style":3611,"fontSize":4204,"fill":3706},"300",[3625,4302],{"x":4303,"y":4304,"width":3664,"height":4225,"rx":3654,"fill":3724,"stroke":3725,"strokeWidth":3632},"620","40",[3607,4306,3403],{"x":4307,"y":3711,"style":3611,"fontSize":3619,"fill":3614},"725",[3625,4309],{"x":4303,"y":3671,"width":3664,"height":4225,"rx":3654,"fill":3724,"stroke":3725,"strokeWidth":3632},[3607,4311,3594],{"x":4307,"y":4312,"style":3611,"fontSize":3619,"fill":3614},"102",[3625,4314],{"x":4303,"y":4315,"width":3664,"height":4225,"rx":3654,"fill":3724,"stroke":3725,"strokeWidth":3632},"124",[3607,4317,3598],{"x":4307,"y":4318,"style":3611,"fontSize":3619,"fill":3614},"144",[3625,4320],{"x":4303,"y":4321,"width":3664,"height":4225,"rx":3654,"fill":3724,"stroke":3725,"strokeWidth":3632},"166",[3607,4323,3386],{"x":4307,"y":4324,"style":3611,"fontSize":3619,"fill":3614},"186",[3625,4326],{"x":4303,"y":4327,"width":3664,"height":4225,"rx":3654,"fill":3724,"stroke":3725,"strokeWidth":3632},"208",[3607,4329,4331],{"x":4307,"y":4330,"style":3611,"fontSize":3619,"fill":3614},"228","JdbcAuthorRepository",[3170,4333],{"x1":4334,"y1":3635,"x2":4303,"y2":4335,"stroke":3614,"strokeWidth":4336,"markerEnd":4337},"500","55","1.3","url(#arr-g2)",[3170,4339],{"x1":4334,"y1":4340,"x2":4303,"y2":4341,"stroke":3614,"strokeWidth":4336,"markerEnd":4337},"120","97",[3170,4343],{"x1":4334,"y1":4296,"x2":4303,"y2":4344,"stroke":3614,"strokeWidth":4336,"markerEnd":4337},"139",[3170,4346],{"x1":4334,"y1":4231,"x2":4303,"y2":3692,"stroke":3614,"strokeWidth":4336,"markerEnd":4337},[3170,4348],{"x1":4334,"y1":4349,"x2":4303,"y2":4350,"stroke":3614,"strokeWidth":4336,"markerEnd":4337},"148","223",[3607,4352,4354],{"x":4353,"y":3695,"style":3611,"fontSize":4204,"fill":3614},"560","getInstance() \u002F @Inject",[3514,4356,4358],{"id":4357},"послідовність-роботи-guice","Послідовність роботи Guice",[3150,4360,4361],{},"Розуміння порядку, в якому Guice виконує свою роботу, є критично важливим для правильного проєктування застосунку. Цей порядок незмінний і не залежить від того, скільки модулів та прив'язок ви визначаєте.",[4363,4364,4365],"mermaid",{},[3159,4366,4369],{"className":4367,"code":4368,"language":4363,"meta":3164,"style":3164},"language-mermaid shiki shiki-themes light-plus dark-plus dark-plus","sequenceDiagram\n    participant Main as main()\n    participant Guice as Guice\n    participant Module as AppModule\n    participant Injector as Injector\n    participant App as JavaFX App\n\n    Main->>Guice: createInjector(new AppModule())\n    Guice->>Module: configure()\n    Module-->>Guice: bindings зареєстровано\n    Guice-->>Main: injector (готовий реєстр)\n    Main->>Injector: getInstance(TrackListController.class)\n    Injector->>Injector: визначити залежності @Inject\n    Injector->>Injector: рекурсивно вирішити TrackService\n    Injector->>Injector: рекурсивно вирішити JdbcTrackRepository\n    Injector->>Injector: вирішити ConnectionManager (@Singleton)\n    Injector-->>App: готовий контролер з усіма залежностями\n",[3154,4370,4371,4376,4381,4386,4391,4396,4401,4405,4410,4415,4420,4425,4430,4435,4440,4445,4451],{"__ignoreMap":3164},[3168,4372,4373],{"class":3170,"line":3171},[3168,4374,4375],{},"sequenceDiagram\n",[3168,4377,4378],{"class":3170,"line":3189},[3168,4379,4380],{},"    participant Main as main()\n",[3168,4382,4383],{"class":3170,"line":3196},[3168,4384,4385],{},"    participant Guice as Guice\n",[3168,4387,4388],{"class":3170,"line":3215},[3168,4389,4390],{},"    participant Module as AppModule\n",[3168,4392,4393],{"class":3170,"line":3220},[3168,4394,4395],{},"    participant Injector as Injector\n",[3168,4397,4398],{"class":3170,"line":3232},[3168,4399,4400],{},"    participant App as JavaFX App\n",[3168,4402,4403],{"class":3170,"line":3254},[3168,4404,3193],{"emptyLinePlaceholder":3192},[3168,4406,4407],{"class":3170,"line":3264},[3168,4408,4409],{},"    Main->>Guice: createInjector(new AppModule())\n",[3168,4411,4412],{"class":3170,"line":3276},[3168,4413,4414],{},"    Guice->>Module: configure()\n",[3168,4416,4417],{"class":3170,"line":3282},[3168,4418,4419],{},"    Module-->>Guice: bindings зареєстровано\n",[3168,4421,4422],{"class":3170,"line":3301},[3168,4423,4424],{},"    Guice-->>Main: injector (готовий реєстр)\n",[3168,4426,4427],{"class":3170,"line":3320},[3168,4428,4429],{},"    Main->>Injector: getInstance(TrackListController.class)\n",[3168,4431,4432],{"class":3170,"line":3345},[3168,4433,4434],{},"    Injector->>Injector: визначити залежності @Inject\n",[3168,4436,4437],{"class":3170,"line":3366},[3168,4438,4439],{},"    Injector->>Injector: рекурсивно вирішити TrackService\n",[3168,4441,4442],{"class":3170,"line":3372},[3168,4443,4444],{},"    Injector->>Injector: рекурсивно вирішити JdbcTrackRepository\n",[3168,4446,4448],{"class":3170,"line":4447},16,[3168,4449,4450],{},"    Injector->>Injector: вирішити ConnectionManager (@Singleton)\n",[3168,4452,4454],{"class":3170,"line":4453},17,[3168,4455,4456],{},"    Injector-->>App: готовий контролер з усіма залежностями\n",[3150,4458,4459,4460,4462,4463,4465,4466,4468,4469,4465,4471,4468,4473,4475,4476,4479,4480,4482,4483,4485],{},"Зверніть увагу на рекурсивне вирішення: щоб створити ",[3154,4461,3403],{},", Guice спочатку створює ",[3154,4464,3594],{},"; щоб створити ",[3154,4467,3594],{}," — ",[3154,4470,3598],{},[3154,4472,3598],{},[3154,4474,3386],{},". Весь цей граф вирішується ",[3380,4477,4478],{},"автоматично",", на основі анотацій ",[3154,4481,3892],{}," у конструкторах та прив'язок у ",[3154,4484,4233],{},". Розробнику не потрібно описувати порядок ініціалізації — Guice визначає його сам.",[3507,4487],{},[3145,4489,4491],{"id":4490},"налаштування-проєкту-від-нуля-до-першого-injector","Налаштування проєкту: від нуля до першого Injector",[3150,4493,4494,4495,3351],{},"Перейдемо від теорії до практики. У цьому розділі ми налаштуємо Google Guice у JavaFX-проєкті аудіоплатформи та побудуємо фундамент: підключимо залежності, опишемо перший модуль та отримаємо перший об'єкт через ",[3154,4496,4137],{},[4498,4499,4500,4504,4520,4730,4745,4752,4755,4857,4872,4876,5137,5152,5156,5427,5443,5447],"steps",{},[3514,4501,4503],{"id":4502},"крок-1-додати-залежності-до-системи-збірки","Крок 1: Додати залежності до системи збірки",[3150,4505,4506,4507,4510,4511,4514,4515,3270,4517,3351],{},"Guice доступний через Maven Central. Нам потрібні два артефакти: сам ",[3154,4508,4509],{},"guice"," та ",[3154,4512,4513],{},"jakarta.inject-api"," — специфікація анотацій ",[3154,4516,3892],{},[3154,4518,4519],{},"@Singleton",[4521,4522,4523,4703],"code-group",{},[3159,4524,4529],{"className":4525,"code":4526,"filename":4527,"language":4528,"meta":3164,"style":3164},"language-xml shiki shiki-themes light-plus dark-plus dark-plus","\u003Cdependencies>\n    \u003C!-- Google Guice — IoC-контейнер -->\n    \u003Cdependency>\n        \u003CgroupId>com.google.inject\u003C\u002FgroupId>\n        \u003CartifactId>guice\u003C\u002FartifactId>\n        \u003Cversion>7.0.0\u003C\u002Fversion>\n    \u003C\u002Fdependency>\n    \u003C!-- JavaFX FXML-підтримка -->\n    \u003Cdependency>\n        \u003CgroupId>org.openjfx\u003C\u002FgroupId>\n        \u003CartifactId>javafx-fxml\u003C\u002FartifactId>\n        \u003Cversion>21.0.3\u003C\u002Fversion>\n    \u003C\u002Fdependency>\n\u003C\u002Fdependencies>\n","pom.xml (Maven)","xml",[3154,4530,4531,4544,4549,4559,4579,4596,4614,4623,4628,4636,4653,4670,4687,4695],{"__ignoreMap":3164},[3168,4532,4533,4537,4541],{"class":3170,"line":3171},[3168,4534,4536],{"class":4535},"s0P7L","\u003C",[3168,4538,4540],{"class":4539},"sKtos","dependencies",[3168,4542,4543],{"class":4535},">\n",[3168,4545,4546],{"class":3170,"line":3189},[3168,4547,4548],{"class":3943},"    \u003C!-- Google Guice — IoC-контейнер -->\n",[3168,4550,4551,4554,4557],{"class":3170,"line":3196},[3168,4552,4553],{"class":4535},"    \u003C",[3168,4555,4556],{"class":4539},"dependency",[3168,4558,4543],{"class":4535},[3168,4560,4561,4564,4567,4570,4572,4575,4577],{"class":3170,"line":3215},[3168,4562,4563],{"class":4535},"        \u003C",[3168,4565,4566],{"class":4539},"groupId",[3168,4568,4569],{"class":4535},">",[3168,4571,4169],{"class":3185},[3168,4573,4574],{"class":4535},"\u003C\u002F",[3168,4576,4566],{"class":4539},[3168,4578,4543],{"class":4535},[3168,4580,4581,4583,4586,4588,4590,4592,4594],{"class":3170,"line":3220},[3168,4582,4563],{"class":4535},[3168,4584,4585],{"class":4539},"artifactId",[3168,4587,4569],{"class":4535},[3168,4589,4509],{"class":3185},[3168,4591,4574],{"class":4535},[3168,4593,4585],{"class":4539},[3168,4595,4543],{"class":4535},[3168,4597,4598,4600,4603,4605,4608,4610,4612],{"class":3170,"line":3232},[3168,4599,4563],{"class":4535},[3168,4601,4602],{"class":4539},"version",[3168,4604,4569],{"class":4535},[3168,4606,4607],{"class":3185},"7.0.0",[3168,4609,4574],{"class":4535},[3168,4611,4602],{"class":4539},[3168,4613,4543],{"class":4535},[3168,4615,4616,4619,4621],{"class":3170,"line":3254},[3168,4617,4618],{"class":4535},"    \u003C\u002F",[3168,4620,4556],{"class":4539},[3168,4622,4543],{"class":4535},[3168,4624,4625],{"class":3170,"line":3264},[3168,4626,4627],{"class":3943},"    \u003C!-- JavaFX FXML-підтримка -->\n",[3168,4629,4630,4632,4634],{"class":3170,"line":3276},[3168,4631,4553],{"class":4535},[3168,4633,4556],{"class":4539},[3168,4635,4543],{"class":4535},[3168,4637,4638,4640,4642,4644,4647,4649,4651],{"class":3170,"line":3282},[3168,4639,4563],{"class":4535},[3168,4641,4566],{"class":4539},[3168,4643,4569],{"class":4535},[3168,4645,4646],{"class":3185},"org.openjfx",[3168,4648,4574],{"class":4535},[3168,4650,4566],{"class":4539},[3168,4652,4543],{"class":4535},[3168,4654,4655,4657,4659,4661,4664,4666,4668],{"class":3170,"line":3301},[3168,4656,4563],{"class":4535},[3168,4658,4585],{"class":4539},[3168,4660,4569],{"class":4535},[3168,4662,4663],{"class":3185},"javafx-fxml",[3168,4665,4574],{"class":4535},[3168,4667,4585],{"class":4539},[3168,4669,4543],{"class":4535},[3168,4671,4672,4674,4676,4678,4681,4683,4685],{"class":3170,"line":3320},[3168,4673,4563],{"class":4535},[3168,4675,4602],{"class":4539},[3168,4677,4569],{"class":4535},[3168,4679,4680],{"class":3185},"21.0.3",[3168,4682,4574],{"class":4535},[3168,4684,4602],{"class":4539},[3168,4686,4543],{"class":4535},[3168,4688,4689,4691,4693],{"class":3170,"line":3345},[3168,4690,4618],{"class":4535},[3168,4692,4556],{"class":4539},[3168,4694,4543],{"class":4535},[3168,4696,4697,4699,4701],{"class":3170,"line":3366},[3168,4698,4574],{"class":4535},[3168,4700,4540],{"class":4539},[3168,4702,4543],{"class":4535},[3159,4704,4709],{"className":4705,"code":4706,"filename":4707,"language":4708,"meta":3164,"style":3164},"language-groovy shiki shiki-themes light-plus dark-plus dark-plus","dependencies {\n    implementation 'com.google.inject:guice:7.0.0'\n    implementation 'org.openjfx:javafx-fxml:21.0.3'\n}\n","build.gradle (Groovy DSL)","groovy",[3154,4710,4711,4716,4721,4726],{"__ignoreMap":3164},[3168,4712,4713],{"class":3170,"line":3171},[3168,4714,4715],{},"dependencies {\n",[3168,4717,4718],{"class":3170,"line":3189},[3168,4719,4720],{},"    implementation 'com.google.inject:guice:7.0.0'\n",[3168,4722,4723],{"class":3170,"line":3196},[3168,4724,4725],{},"    implementation 'org.openjfx:javafx-fxml:21.0.3'\n",[3168,4727,4728],{"class":3170,"line":3215},[3168,4729,3375],{},[3447,4731,4732,4735,4736,4738,4739,4741,4742,4744],{},[3380,4733,4734],{},"Версія Guice 7.x:"," починаючи з версії 7.0, Guice перейшов з пакету ",[3154,4737,3768],{}," на ",[3154,4740,4165],{},". Якщо ваш проєкт залежить від старих бібліотек зі ",[3154,4743,3768],{},", використовуйте Guice 6.x, що підтримує обидва простори імен.",[3514,4746,4748,4749],{"id":4747},"крок-2-налаштувати-module-infojava","Крок 2: Налаштувати ",[3154,4750,4751],{},"module-info.java",[3150,4753,4754],{},"Якщо проєкт використовує Java Platform Module System (JPMS), необхідно відкрити пакети для Guice-рефлексії:",[3159,4756,4758],{"className":3161,"code":4757,"filename":4751,"language":3163,"meta":3164,"style":3164},"module com.example.audiobook {\n    requires javafx.controls;\n    requires javafx.fxml;\n    requires com.google.guice;\n    requires jakarta.inject;\n\n    \u002F\u002F opens — глибокий рефлексивний доступ у runtime (потрібен Guice)\n    opens com.example.audiobook.controller to javafx.fxml, com.google.guice;\n    opens com.example.audiobook.service    to com.google.guice;\n    opens com.example.audiobook.repository to com.google.guice;\n    exports com.example.audiobook;\n}\n",[3154,4759,4760,4771,4779,4786,4793,4800,4804,4809,4823,4834,4845,4853],{"__ignoreMap":3164},[3168,4761,4762,4765,4768],{"class":3170,"line":3171},[3168,4763,4764],{"class":3174},"module",[3168,4766,4767],{"class":3181}," com",[3168,4769,4770],{"class":3185},".example.audiobook {\n",[3168,4772,4773,4776],{"class":3170,"line":3189},[3168,4774,4775],{"class":3174},"    requires",[3168,4777,4778],{"class":3185}," javafx.controls;\n",[3168,4780,4781,4783],{"class":3170,"line":3196},[3168,4782,4775],{"class":3174},[3168,4784,4785],{"class":3185}," javafx.fxml;\n",[3168,4787,4788,4790],{"class":3170,"line":3215},[3168,4789,4775],{"class":3174},[3168,4791,4792],{"class":3185}," com.google.guice;\n",[3168,4794,4795,4797],{"class":3170,"line":3220},[3168,4796,4775],{"class":3174},[3168,4798,4799],{"class":3185}," jakarta.inject;\n",[3168,4801,4802],{"class":3170,"line":3232},[3168,4803,3193],{"emptyLinePlaceholder":3192},[3168,4805,4806],{"class":3170,"line":3254},[3168,4807,4808],{"class":3943},"    \u002F\u002F opens — глибокий рефлексивний доступ у runtime (потрібен Guice)\n",[3168,4810,4811,4814,4817,4820],{"class":3170,"line":3264},[3168,4812,4813],{"class":3174},"    opens",[3168,4815,4816],{"class":3185}," com.example.audiobook.controller ",[3168,4818,4819],{"class":3174},"to",[3168,4821,4822],{"class":3185}," javafx.fxml, com.google.guice;\n",[3168,4824,4825,4827,4830,4832],{"class":3170,"line":3276},[3168,4826,4813],{"class":3174},[3168,4828,4829],{"class":3185}," com.example.audiobook.service    ",[3168,4831,4819],{"class":3174},[3168,4833,4792],{"class":3185},[3168,4835,4836,4838,4841,4843],{"class":3170,"line":3282},[3168,4837,4813],{"class":3174},[3168,4839,4840],{"class":3185}," com.example.audiobook.repository ",[3168,4842,4819],{"class":3174},[3168,4844,4792],{"class":3185},[3168,4846,4847,4850],{"class":3170,"line":3301},[3168,4848,4849],{"class":3174},"    exports",[3168,4851,4852],{"class":3185}," com.example.audiobook;\n",[3168,4854,4855],{"class":3170,"line":3320},[3168,4856,3375],{"class":3185},[3150,4858,4859,4860,4863,4864,4867,4868,4871],{},"Ключове слово ",[3154,4861,4862],{},"opens"," (на відміну від ",[3154,4865,4866],{},"exports",") дозволяє саме глибокий рефлексивний доступ у runtime — це необхідно Guice для впровадження залежностей. Якщо пакет не відкрито, Guice викине ",[3154,4869,4870],{},"InaccessibleObjectException"," при спробі створити об'єкт.",[3514,4873,4875],{"id":4874},"крок-3-описати-перший-module","Крок 3: Описати перший Module",[3159,4877,4880],{"className":3161,"code":4878,"filename":4879,"language":3163,"meta":3164,"style":3164},"package com.example.audiobook;\n\nimport com.example.audiobook.repository.*;\nimport com.google.inject.AbstractModule;\nimport com.google.inject.name.Names;\n\npublic class AppModule extends AbstractModule {\n\n    @Override\n    protected void configure() {\n        bind(TrackRepository.class).to(JdbcTrackRepository.class);\n        bind(AuthorRepository.class).to(JdbcAuthorRepository.class);\n\n        bindConstant().annotatedWith(Names.named(\"dbUrl\"))\n            .to(\"jdbc:h2:.\u002Fdata\u002Faudiobook;MODE=PostgreSQL;DB_CLOSE_DELAY=-1\");\n        bindConstant().annotatedWith(Names.named(\"dbUser\")).to(\"sa\");\n        bindConstant().annotatedWith(Names.named(\"dbPassword\")).to(\"\");\n    }\n}\n","AppModule.java",[3154,4881,4882,4889,4893,4901,4908,4915,4919,4936,4940,4947,4959,4988,5015,5019,5048,5062,5095,5127,5132],{"__ignoreMap":3164},[3168,4883,4884,4887],{"class":3170,"line":3171},[3168,4885,4886],{"class":3174},"package",[3168,4888,4852],{"class":3185},[3168,4890,4891],{"class":3170,"line":3189},[3168,4892,3193],{"emptyLinePlaceholder":3192},[3168,4894,4895,4898],{"class":3170,"line":3196},[3168,4896,4897],{"class":3174},"import",[3168,4899,4900],{"class":3185}," com.example.audiobook.repository.*;\n",[3168,4902,4903,4905],{"class":3170,"line":3215},[3168,4904,4897],{"class":3174},[3168,4906,4907],{"class":3185}," com.google.inject.AbstractModule;\n",[3168,4909,4910,4912],{"class":3170,"line":3220},[3168,4911,4897],{"class":3174},[3168,4913,4914],{"class":3185}," com.google.inject.name.Names;\n",[3168,4916,4917],{"class":3170,"line":3232},[3168,4918,3193],{"emptyLinePlaceholder":3192},[3168,4920,4921,4923,4925,4928,4931,4934],{"class":3170,"line":3254},[3168,4922,3175],{"class":3174},[3168,4924,3178],{"class":3174},[3168,4926,4927],{"class":3181}," AppModule",[3168,4929,4930],{"class":3174}," extends",[3168,4932,4933],{"class":3181}," AbstractModule",[3168,4935,3186],{"class":3185},[3168,4937,4938],{"class":3170,"line":3264},[3168,4939,3193],{"emptyLinePlaceholder":3192},[3168,4941,4942,4944],{"class":3170,"line":3276},[3168,4943,3830],{"class":3185},[3168,4945,4946],{"class":3181},"Override\n",[3168,4948,4949,4952,4954,4957],{"class":3170,"line":3282},[3168,4950,4951],{"class":3174},"    protected",[3168,4953,4043],{"class":3181},[3168,4955,4956],{"class":3226}," configure",[3168,4958,3229],{"class":3185},[3168,4960,4961,4964,4966,4968,4970,4973,4976,4978,4980,4982,4984,4986],{"class":3170,"line":3301},[3168,4962,4963],{"class":3226},"        bind",[3168,4965,3336],{"class":3185},[3168,4967,3407],{"class":3208},[3168,4969,3351],{"class":3185},[3168,4971,4972],{"class":3208},"class",[3168,4974,4975],{"class":3185},").",[3168,4977,4819],{"class":3226},[3168,4979,3336],{"class":3185},[3168,4981,3598],{"class":3208},[3168,4983,3351],{"class":3185},[3168,4985,4972],{"class":3208},[3168,4987,3342],{"class":3185},[3168,4989,4990,4992,4994,4997,4999,5001,5003,5005,5007,5009,5011,5013],{"class":3170,"line":3320},[3168,4991,4963],{"class":3226},[3168,4993,3336],{"class":3185},[3168,4995,4996],{"class":3208},"AuthorRepository",[3168,4998,3351],{"class":3185},[3168,5000,4972],{"class":3208},[3168,5002,4975],{"class":3185},[3168,5004,4819],{"class":3226},[3168,5006,3336],{"class":3185},[3168,5008,4331],{"class":3208},[3168,5010,3351],{"class":3185},[3168,5012,4972],{"class":3208},[3168,5014,3342],{"class":3185},[3168,5016,5017],{"class":3170,"line":3345},[3168,5018,3193],{"emptyLinePlaceholder":3192},[3168,5020,5021,5024,5027,5030,5032,5035,5037,5040,5042,5045],{"class":3170,"line":3366},[3168,5022,5023],{"class":3226},"        bindConstant",[3168,5025,5026],{"class":3185},"().",[3168,5028,5029],{"class":3226},"annotatedWith",[3168,5031,3336],{"class":3185},[3168,5033,5034],{"class":3208},"Names",[3168,5036,3351],{"class":3185},[3168,5038,5039],{"class":3226},"named",[3168,5041,3336],{"class":3185},[3168,5043,5044],{"class":3257},"\"dbUrl\"",[3168,5046,5047],{"class":3185},"))\n",[3168,5049,5050,5053,5055,5057,5060],{"class":3170,"line":3372},[3168,5051,5052],{"class":3185},"            .",[3168,5054,4819],{"class":3226},[3168,5056,3336],{"class":3185},[3168,5058,5059],{"class":3257},"\"jdbc:h2:.\u002Fdata\u002Faudiobook;MODE=PostgreSQL;DB_CLOSE_DELAY=-1\"",[3168,5061,3342],{"class":3185},[3168,5063,5064,5066,5068,5070,5072,5074,5076,5078,5080,5083,5086,5088,5090,5093],{"class":3170,"line":4447},[3168,5065,5023],{"class":3226},[3168,5067,5026],{"class":3185},[3168,5069,5029],{"class":3226},[3168,5071,3336],{"class":3185},[3168,5073,5034],{"class":3208},[3168,5075,3351],{"class":3185},[3168,5077,5039],{"class":3226},[3168,5079,3336],{"class":3185},[3168,5081,5082],{"class":3257},"\"dbUser\"",[3168,5084,5085],{"class":3185},")).",[3168,5087,4819],{"class":3226},[3168,5089,3336],{"class":3185},[3168,5091,5092],{"class":3257},"\"sa\"",[3168,5094,3342],{"class":3185},[3168,5096,5097,5099,5101,5103,5105,5107,5109,5111,5113,5116,5118,5120,5122,5125],{"class":3170,"line":4453},[3168,5098,5023],{"class":3226},[3168,5100,5026],{"class":3185},[3168,5102,5029],{"class":3226},[3168,5104,3336],{"class":3185},[3168,5106,5034],{"class":3208},[3168,5108,3351],{"class":3185},[3168,5110,5039],{"class":3226},[3168,5112,3336],{"class":3185},[3168,5114,5115],{"class":3257},"\"dbPassword\"",[3168,5117,5085],{"class":3185},[3168,5119,4819],{"class":3226},[3168,5121,3336],{"class":3185},[3168,5123,5124],{"class":3257},"\"\"",[3168,5126,3342],{"class":3185},[3168,5128,5130],{"class":3170,"line":5129},18,[3168,5131,3369],{"class":3185},[3168,5133,5135],{"class":3170,"line":5134},19,[3168,5136,3375],{"class":3185},[3150,5138,5139,5140,5143,5144,5147,5148,5151],{},"Кожен виклик ",[3154,5141,5142],{},"bind(...).to(...)"," — одна прив'язка. ",[3154,5145,5146],{},"bindConstant().annotatedWith(Names.named(...)).to(...)"," дозволяє впроваджувати прості значення (рядки, числа) через анотацію ",[3154,5149,5150],{},"@Named"," на параметрах конструктора.",[3514,5153,5155],{"id":5154},"крок-4-впровадити-залежності-у-клас","Крок 4: Впровадити залежності у клас",[3159,5157,5160],{"className":3161,"code":5158,"filename":5159,"language":3163,"meta":3164,"style":3164},"import com.google.inject.Inject;\nimport com.google.inject.Singleton;\nimport com.google.inject.name.Named;\n\n@Singleton\npublic class ConnectionManager {\n\n    private final String url;\n    private final String user;\n    private final String password;\n\n    @Inject\n    public ConnectionManager(\n        @Named(\"dbUrl\")      String url,\n        @Named(\"dbUser\")     String user,\n        @Named(\"dbPassword\") String password\n    ) {\n        this.url      = url;\n        this.user     = user;\n        this.password = password;\n    }\n\n    public Connection getConnection() throws SQLException {\n        return DriverManager.getConnection(url, user, password);\n    }\n}\n","ConnectionManager.java",[3154,5161,5162,5169,5176,5183,5187,5195,5205,5209,5223,5236,5249,5253,5259,5267,5289,5308,5326,5331,5343,5355,5368,5373,5378,5400,5417,5422],{"__ignoreMap":3164},[3168,5163,5164,5166],{"class":3170,"line":3171},[3168,5165,4897],{"class":3174},[3168,5167,5168],{"class":3185}," com.google.inject.Inject;\n",[3168,5170,5171,5173],{"class":3170,"line":3189},[3168,5172,4897],{"class":3174},[3168,5174,5175],{"class":3185}," com.google.inject.Singleton;\n",[3168,5177,5178,5180],{"class":3170,"line":3196},[3168,5179,4897],{"class":3174},[3168,5181,5182],{"class":3185}," com.google.inject.name.Named;\n",[3168,5184,5185],{"class":3170,"line":3215},[3168,5186,3193],{"emptyLinePlaceholder":3192},[3168,5188,5189,5192],{"class":3170,"line":3220},[3168,5190,5191],{"class":3185},"@",[3168,5193,5194],{"class":3181},"Singleton\n",[3168,5196,5197,5199,5201,5203],{"class":3170,"line":3232},[3168,5198,3175],{"class":3174},[3168,5200,3178],{"class":3174},[3168,5202,3248],{"class":3181},[3168,5204,3186],{"class":3185},[3168,5206,5207],{"class":3170,"line":3254},[3168,5208,3193],{"emptyLinePlaceholder":3192},[3168,5210,5211,5213,5215,5218,5221],{"class":3170,"line":3264},[3168,5212,3199],{"class":3174},[3168,5214,3202],{"class":3174},[3168,5216,5217],{"class":3181}," String",[3168,5219,5220],{"class":3208}," url",[3168,5222,3212],{"class":3185},[3168,5224,5225,5227,5229,5231,5234],{"class":3170,"line":3276},[3168,5226,3199],{"class":3174},[3168,5228,3202],{"class":3174},[3168,5230,5217],{"class":3181},[3168,5232,5233],{"class":3208}," user",[3168,5235,3212],{"class":3185},[3168,5237,5238,5240,5242,5244,5247],{"class":3170,"line":3282},[3168,5239,3199],{"class":3174},[3168,5241,3202],{"class":3174},[3168,5243,5217],{"class":3181},[3168,5245,5246],{"class":3208}," password",[3168,5248,3212],{"class":3185},[3168,5250,5251],{"class":3170,"line":3301},[3168,5252,3193],{"emptyLinePlaceholder":3192},[3168,5254,5255,5257],{"class":3170,"line":3320},[3168,5256,3830],{"class":3185},[3168,5258,3833],{"class":3181},[3168,5260,5261,5263,5265],{"class":3170,"line":3345},[3168,5262,3223],{"class":3174},[3168,5264,3248],{"class":3226},[3168,5266,3251],{"class":3185},[3168,5268,5269,5272,5275,5277,5279,5282,5285,5287],{"class":3170,"line":3366},[3168,5270,5271],{"class":3185},"        @",[3168,5273,5274],{"class":3181},"Named",[3168,5276,3336],{"class":3185},[3168,5278,5044],{"class":3257},[3168,5280,5281],{"class":3185},")      ",[3168,5283,5284],{"class":3181},"String",[3168,5286,5220],{"class":3208},[3168,5288,3261],{"class":3185},[3168,5290,5291,5293,5295,5297,5299,5302,5304,5306],{"class":3170,"line":3372},[3168,5292,5271],{"class":3185},[3168,5294,5274],{"class":3181},[3168,5296,3336],{"class":3185},[3168,5298,5082],{"class":3257},[3168,5300,5301],{"class":3185},")     ",[3168,5303,5284],{"class":3181},[3168,5305,5233],{"class":3208},[3168,5307,3261],{"class":3185},[3168,5309,5310,5312,5314,5316,5318,5321,5323],{"class":3170,"line":4447},[3168,5311,5271],{"class":3185},[3168,5313,5274],{"class":3181},[3168,5315,3336],{"class":3185},[3168,5317,5115],{"class":3257},[3168,5319,5320],{"class":3185},") ",[3168,5322,5284],{"class":3181},[3168,5324,5325],{"class":3208}," password\n",[3168,5327,5328],{"class":3170,"line":4453},[3168,5329,5330],{"class":3185},"    ) {\n",[3168,5332,5333,5335,5337,5340],{"class":3170,"line":5129},[3168,5334,3348],{"class":3174},[3168,5336,3351],{"class":3185},[3168,5338,5339],{"class":3208},"url",[3168,5341,5342],{"class":3185},"      = url;\n",[3168,5344,5345,5347,5349,5352],{"class":3170,"line":5134},[3168,5346,3348],{"class":3174},[3168,5348,3351],{"class":3185},[3168,5350,5351],{"class":3208},"user",[3168,5353,5354],{"class":3185},"     = user;\n",[3168,5356,5358,5360,5362,5365],{"class":3170,"line":5357},20,[3168,5359,3348],{"class":3174},[3168,5361,3351],{"class":3185},[3168,5363,5364],{"class":3208},"password",[3168,5366,5367],{"class":3185}," = password;\n",[3168,5369,5371],{"class":3170,"line":5370},21,[3168,5372,3369],{"class":3185},[3168,5374,5376],{"class":3170,"line":5375},22,[3168,5377,3193],{"emptyLinePlaceholder":3192},[3168,5379,5381,5383,5386,5389,5392,5395,5398],{"class":3170,"line":5380},23,[3168,5382,3223],{"class":3174},[3168,5384,5385],{"class":3181}," Connection",[3168,5387,5388],{"class":3226}," getConnection",[3168,5390,5391],{"class":3185},"() ",[3168,5393,5394],{"class":3174},"throws",[3168,5396,5397],{"class":3181}," SQLException",[3168,5399,3186],{"class":3185},[3168,5401,5403,5406,5409,5411,5414],{"class":3170,"line":5402},24,[3168,5404,5405],{"class":3244},"        return",[3168,5407,5408],{"class":3208}," DriverManager",[3168,5410,3351],{"class":3185},[3168,5412,5413],{"class":3226},"getConnection",[3168,5415,5416],{"class":3185},"(url, user, password);\n",[3168,5418,5420],{"class":3170,"line":5419},25,[3168,5421,3369],{"class":3185},[3168,5423,5425],{"class":3170,"line":5424},26,[3168,5426,3375],{"class":3185},[3150,5428,5429,5431,5432,5434,5435,5438,5439,5442],{},[3154,5430,4519],{}," на класі декларує єдиний екземпляр на весь ",[3154,5433,4137],{},". Анотації ",[3154,5436,5437],{},"@Named(...)"," на параметрах вказують Guice знайти саме ту константу, що зв'язана у модулі через ",[3154,5440,5441],{},"Names.named(...)"," з відповідним ім'ям.",[3514,5444,5446],{"id":5445},"крок-5-створити-injector-і-отримати-перший-обєкт","Крок 5: Створити Injector і отримати перший об'єкт",[3159,5448,5451],{"className":3161,"code":5449,"filename":5450,"language":3163,"meta":3164,"style":3164},"import com.google.inject.Guice;\nimport com.google.inject.Injector;\n\npublic class Main {\n    public static void main(String[] args) {\n        \u002F\u002F Єдина точка створення Injector у всьому застосунку\n        Injector injector = Guice.createInjector(new AppModule());\n\n        \u002F\u002F Guice рекурсивно вирішує весь граф залежностей\n        TrackService trackService = injector.getInstance(TrackService.class);\n        trackService.listAll().forEach(System.out::println);\n    }\n}\n","Main.java",[3154,5452,5453,5460,5467,5471,5482,5506,5511,5538,5542,5547,5574,5605,5609],{"__ignoreMap":3164},[3168,5454,5455,5457],{"class":3170,"line":3171},[3168,5456,4897],{"class":3174},[3168,5458,5459],{"class":3185}," com.google.inject.Guice;\n",[3168,5461,5462,5464],{"class":3170,"line":3189},[3168,5463,4897],{"class":3174},[3168,5465,5466],{"class":3185}," com.google.inject.Injector;\n",[3168,5468,5469],{"class":3170,"line":3196},[3168,5470,3193],{"emptyLinePlaceholder":3192},[3168,5472,5473,5475,5477,5480],{"class":3170,"line":3215},[3168,5474,3175],{"class":3174},[3168,5476,3178],{"class":3174},[3168,5478,5479],{"class":3181}," Main",[3168,5481,3186],{"class":3185},[3168,5483,5484,5486,5489,5491,5494,5496,5498,5501,5504],{"class":3170,"line":3220},[3168,5485,3223],{"class":3174},[3168,5487,5488],{"class":3174}," static",[3168,5490,4043],{"class":3181},[3168,5492,5493],{"class":3226}," main",[3168,5495,3336],{"class":3185},[3168,5497,5284],{"class":3181},[3168,5499,5500],{"class":3185},"[] ",[3168,5502,5503],{"class":3208},"args",[3168,5505,3854],{"class":3185},[3168,5507,5508],{"class":3170,"line":3232},[3168,5509,5510],{"class":3943},"        \u002F\u002F Єдина точка створення Injector у всьому застосунку\n",[3168,5512,5513,5516,5519,5521,5524,5526,5529,5531,5533,5535],{"class":3170,"line":3254},[3168,5514,5515],{"class":3181},"        Injector",[3168,5517,5518],{"class":3208}," injector",[3168,5520,3241],{"class":3185},[3168,5522,5523],{"class":3208},"Guice",[3168,5525,3351],{"class":3185},[3168,5527,5528],{"class":3226},"createInjector",[3168,5530,3336],{"class":3185},[3168,5532,3245],{"class":3244},[3168,5534,4927],{"class":3226},[3168,5536,5537],{"class":3185},"());\n",[3168,5539,5540],{"class":3170,"line":3264},[3168,5541,3193],{"emptyLinePlaceholder":3192},[3168,5543,5544],{"class":3170,"line":3276},[3168,5545,5546],{"class":3943},"        \u002F\u002F Guice рекурсивно вирішує весь граф залежностей\n",[3168,5548,5549,5552,5554,5556,5559,5561,5564,5566,5568,5570,5572],{"class":3170,"line":3282},[3168,5550,5551],{"class":3181},"        TrackService",[3168,5553,3209],{"class":3208},[3168,5555,3241],{"class":3185},[3168,5557,5558],{"class":3208},"injector",[3168,5560,3351],{"class":3185},[3168,5562,5563],{"class":3226},"getInstance",[3168,5565,3336],{"class":3185},[3168,5567,3594],{"class":3208},[3168,5569,3351],{"class":3185},[3168,5571,4972],{"class":3208},[3168,5573,3342],{"class":3185},[3168,5575,5576,5579,5581,5584,5586,5589,5591,5594,5596,5599,5602],{"class":3170,"line":3301},[3168,5577,5578],{"class":3208},"        trackService",[3168,5580,3351],{"class":3185},[3168,5582,5583],{"class":3226},"listAll",[3168,5585,5026],{"class":3185},[3168,5587,5588],{"class":3226},"forEach",[3168,5590,3336],{"class":3185},[3168,5592,5593],{"class":3208},"System",[3168,5595,3351],{"class":3185},[3168,5597,5598],{"class":3208},"out",[3168,5600,5601],{"class":3244},"::",[3168,5603,5604],{"class":3185},"println);\n",[3168,5606,5607],{"class":3170,"line":3320},[3168,5608,3369],{"class":3185},[3168,5610,5611],{"class":3170,"line":3345},[3168,5612,3375],{"class":3185},[3150,5614,5615],{},"Структура проєкту після налаштування:",[5617,5618,5619,5627,5634,5641,5648,5655,5662,5669,5676],"code-tree",{},[3159,5620,5625],{"className":5621,"code":5623,"filename":5624,"language":3607,"meta":3164},[5622],"language-text","\u002F\u002F Guice-модуль: всі прив'язки\n","src\u002Fmain\u002Fjava\u002Fcom\u002Fexample\u002Faudiobook\u002FAppModule.java",[3154,5626,5623],{"__ignoreMap":3164},[3159,5628,5632],{"className":5629,"code":5630,"filename":5631,"language":3607,"meta":3164},[5622],"\u002F\u002F Точка входу: Injector\n","src\u002Fmain\u002Fjava\u002Fcom\u002Fexample\u002Faudiobook\u002FMain.java",[3154,5633,5630],{"__ignoreMap":3164},[3159,5635,5639],{"className":5636,"code":5637,"filename":5638,"language":3607,"meta":3164},[5622],"\u002F\u002F @Singleton\n","src\u002Fmain\u002Fjava\u002Fcom\u002Fexample\u002Faudiobook\u002Fdb\u002FConnectionManager.java",[3154,5640,5637],{"__ignoreMap":3164},[3159,5642,5646],{"className":5643,"code":5644,"filename":5645,"language":3607,"meta":3164},[5622],"\u002F\u002F Інтерфейс\n","src\u002Fmain\u002Fjava\u002Fcom\u002Fexample\u002Faudiobook\u002Frepository\u002FTrackRepository.java",[3154,5647,5644],{"__ignoreMap":3164},[3159,5649,5653],{"className":5650,"code":5651,"filename":5652,"language":3607,"meta":3164},[5622],"\u002F\u002F Реалізація\n","src\u002Fmain\u002Fjava\u002Fcom\u002Fexample\u002Faudiobook\u002Frepository\u002FJdbcTrackRepository.java",[3154,5654,5651],{"__ignoreMap":3164},[3159,5656,5660],{"className":5657,"code":5658,"filename":5659,"language":3607,"meta":3164},[5622],"\u002F\u002F Бізнес-логіка\n","src\u002Fmain\u002Fjava\u002Fcom\u002Fexample\u002Faudiobook\u002Fservice\u002FTrackService.java",[3154,5661,5658],{"__ignoreMap":3164},[3159,5663,5667],{"className":5664,"code":5665,"filename":5666,"language":3607,"meta":3164},[5622],"\u002F\u002F JavaFX-контролер (@Inject)\n","src\u002Fmain\u002Fjava\u002Fcom\u002Fexample\u002Faudiobook\u002Fcontroller\u002FTrackListController.java",[3154,5668,5665],{"__ignoreMap":3164},[3159,5670,5674],{"className":5671,"code":5672,"filename":5673,"language":3607,"meta":3164},[5622],"\u002F\u002F FXML-розмітка\n","src\u002Fmain\u002Fresources\u002Ffxml\u002Ftrack-list.fxml",[3154,5675,5672],{"__ignoreMap":3164},[3159,5677,5680],{"className":5678,"code":5679,"filename":4751,"language":3607,"meta":3164},[5622],"\u002F\u002F JPMS: opens для Guice\n",[3154,5681,5679],{"__ignoreMap":3164},[5683,5684,5685,5686,5688,5689,5691,5692,5694,5695,4510,5697,3351],"warning",{},"Ніколи не зберігайте ",[3154,5687,4137],{}," у статичному полі та не передавайте його класам як залежність — це антипатерн Service Locator. Клас, що отримує ",[3154,5690,4137],{},", сам витягує залежності й унеможливлює тестування. Правильна практика: ",[3154,5693,4137],{}," живе лише у ",[3154,5696,5450],{},[3154,5698,5699],{},"App.java",[3507,5701],{},[3145,5703,5705],{"id":5704},"типи-привязок-bindings-та-анотації-guice","Типи прив'язок (Bindings) та анотації Guice",[3150,5707,5708,5709,5711,5712,5714],{},"Метод ",[3154,5710,4130],{}," у ",[3154,5713,4126],{}," — це не просто перелік рядків коду. Це повноцінна декларативна мова опису залежностей. Guice підтримує кілька типів прив'язок, кожен з яких вирішує конкретний сценарій.",[3514,5716,5718],{"id":5717},"linked-binding-звязування-інтерфейсу-з-реалізацією","Linked Binding — зв'язування інтерфейсу з реалізацією",[3150,5720,5721],{},"Найпоширеніший тип. Дозволяє замінити реалізацію без зміни жодного класу-споживача.",[3159,5723,5725],{"className":3161,"code":5724,"language":3163,"meta":3164,"style":3164},"\u002F\u002F Коли хтось запросить TrackRepository — отримає JdbcTrackRepository\nbind(TrackRepository.class).to(JdbcTrackRepository.class);\n\n\u002F\u002F З явним Scope: лише один JdbcTrackRepository на весь Injector\nbind(TrackRepository.class).to(JdbcTrackRepository.class).in(Singleton.class);\n",[3154,5726,5727,5732,5759,5763,5768],{"__ignoreMap":3164},[3168,5728,5729],{"class":3170,"line":3171},[3168,5730,5731],{"class":3943},"\u002F\u002F Коли хтось запросить TrackRepository — отримає JdbcTrackRepository\n",[3168,5733,5734,5737,5739,5741,5743,5745,5747,5749,5751,5753,5755,5757],{"class":3170,"line":3189},[3168,5735,5736],{"class":3226},"bind",[3168,5738,3336],{"class":3185},[3168,5740,3407],{"class":3208},[3168,5742,3351],{"class":3185},[3168,5744,4972],{"class":3208},[3168,5746,4975],{"class":3185},[3168,5748,4819],{"class":3226},[3168,5750,3336],{"class":3185},[3168,5752,3598],{"class":3208},[3168,5754,3351],{"class":3185},[3168,5756,4972],{"class":3208},[3168,5758,3342],{"class":3185},[3168,5760,5761],{"class":3170,"line":3196},[3168,5762,3193],{"emptyLinePlaceholder":3192},[3168,5764,5765],{"class":3170,"line":3215},[3168,5766,5767],{"class":3943},"\u002F\u002F З явним Scope: лише один JdbcTrackRepository на весь Injector\n",[3168,5769,5770,5772,5774,5776,5778,5780,5782,5784,5786,5788,5790,5792,5794,5797,5799,5802,5804,5806],{"class":3170,"line":3220},[3168,5771,5736],{"class":3226},[3168,5773,3336],{"class":3185},[3168,5775,3407],{"class":3208},[3168,5777,3351],{"class":3185},[3168,5779,4972],{"class":3208},[3168,5781,4975],{"class":3185},[3168,5783,4819],{"class":3226},[3168,5785,3336],{"class":3185},[3168,5787,3598],{"class":3208},[3168,5789,3351],{"class":3185},[3168,5791,4972],{"class":3208},[3168,5793,4975],{"class":3185},[3168,5795,5796],{"class":3226},"in",[3168,5798,3336],{"class":3185},[3168,5800,5801],{"class":3208},"Singleton",[3168,5803,3351],{"class":3185},[3168,5805,4972],{"class":3208},[3168,5807,3342],{"class":3185},[3150,5809,5810,5811,5813,5814,5816],{},"Зверніть увагу: ",[3154,5812,3598],{}," сам також може мати ",[3154,5815,3892],{},"-конструктор і власні залежності — Guice рекурсивно вирішить їх автоматично.",[3514,5818,5820],{"id":5819},"instance-binding-впровадження-готового-екземпляра","Instance Binding — впровадження готового екземпляра",[3150,5822,5823],{},"Коли об'єкт уже створений (або не може бути створений Guice) — передайте його напряму:",[3159,5825,5827],{"className":3161,"code":5826,"language":3163,"meta":3164,"style":3164},"\u002F\u002F Передаємо готовий об'єкт — Guice не створює його сам\nConnectionManager manager = ConnectionManager.forH2(\".\u002Fdata\u002Faudiobook\");\nbind(ConnectionManager.class).toInstance(manager);\n\n\u002F\u002F Або через статичну фабрику\nbind(DataSource.class).toInstance(createHikariDataSource());\n",[3154,5828,5829,5834,5857,5877,5881,5886],{"__ignoreMap":3164},[3168,5830,5831],{"class":3170,"line":3171},[3168,5832,5833],{"class":3943},"\u002F\u002F Передаємо готовий об'єкт — Guice не створює його сам\n",[3168,5835,5836,5838,5841,5843,5845,5847,5850,5852,5855],{"class":3170,"line":3189},[3168,5837,3386],{"class":3181},[3168,5839,5840],{"class":3208}," manager",[3168,5842,3241],{"class":3185},[3168,5844,3386],{"class":3208},[3168,5846,3351],{"class":3185},[3168,5848,5849],{"class":3226},"forH2",[3168,5851,3336],{"class":3185},[3168,5853,5854],{"class":3257},"\".\u002Fdata\u002Faudiobook\"",[3168,5856,3342],{"class":3185},[3168,5858,5859,5861,5863,5865,5867,5869,5871,5874],{"class":3170,"line":3196},[3168,5860,5736],{"class":3226},[3168,5862,3336],{"class":3185},[3168,5864,3386],{"class":3208},[3168,5866,3351],{"class":3185},[3168,5868,4972],{"class":3208},[3168,5870,4975],{"class":3185},[3168,5872,5873],{"class":3226},"toInstance",[3168,5875,5876],{"class":3185},"(manager);\n",[3168,5878,5879],{"class":3170,"line":3215},[3168,5880,3193],{"emptyLinePlaceholder":3192},[3168,5882,5883],{"class":3170,"line":3220},[3168,5884,5885],{"class":3943},"\u002F\u002F Або через статичну фабрику\n",[3168,5887,5888,5890,5892,5895,5897,5899,5901,5903,5905,5908],{"class":3170,"line":3232},[3168,5889,5736],{"class":3226},[3168,5891,3336],{"class":3185},[3168,5893,5894],{"class":3208},"DataSource",[3168,5896,3351],{"class":3185},[3168,5898,4972],{"class":3208},[3168,5900,4975],{"class":3185},[3168,5902,5873],{"class":3226},[3168,5904,3336],{"class":3185},[3168,5906,5907],{"class":3226},"createHikariDataSource",[3168,5909,5537],{"class":3185},[3150,5911,5912,5915],{},[3154,5913,5914],{},"toInstance()"," автоматично реєструє переданий об'єкт як Singleton — він завжди повертає той самий екземпляр.",[3514,5917,5919],{"id":5918},"provider-binding-відкладене-або-параметризоване-створення","Provider Binding — відкладене або параметризоване створення",[3150,5921,5922,5924,5925,3351],{},[3154,5923,4177],{}," — це фабрика, що викликається Guice щоразу, коли потрібен новий об'єкт. Використовується, коли логіка створення складніша за простий ",[3154,5926,3245],{},[3159,5928,5930],{"className":3161,"code":5929,"language":3163,"meta":3164,"style":3164},"\u002F\u002F Клас-провайдер для DataSource (наприклад, HikariCP)\npublic class HikariDataSourceProvider implements Provider\u003CDataSource> {\n\n    private final String url;\n    private final String user;\n\n    @Inject\n    public HikariDataSourceProvider(\n        @Named(\"dbUrl\")  String url,\n        @Named(\"dbUser\") String user\n    ) {\n        this.url  = url;\n        this.user = user;\n    }\n\n    @Override\n    public DataSource get() {\n        HikariConfig config = new HikariConfig();\n        config.setJdbcUrl(url);\n        config.setUsername(user);\n        config.setMaximumPoolSize(10);\n        return new HikariDataSource(config);\n    }\n}\n\n\u002F\u002F У модулі:\nbind(DataSource.class).toProvider(HikariDataSourceProvider.class).in(Singleton.class);\n",[3154,5931,5932,5937,5959,5963,5975,5987,5991,5997,6005,6024,6041,6045,6056,6067,6071,6075,6081,6093,6110,6123,6135,6151,6164,6168,6172,6176,6181],{"__ignoreMap":3164},[3168,5933,5934],{"class":3170,"line":3171},[3168,5935,5936],{"class":3943},"\u002F\u002F Клас-провайдер для DataSource (наприклад, HikariCP)\n",[3168,5938,5939,5941,5943,5946,5949,5952,5954,5956],{"class":3170,"line":3189},[3168,5940,3175],{"class":3174},[3168,5942,3178],{"class":3174},[3168,5944,5945],{"class":3181}," HikariDataSourceProvider",[3168,5947,5948],{"class":3174}," implements",[3168,5950,5951],{"class":3181}," Provider",[3168,5953,4536],{"class":3185},[3168,5955,5894],{"class":3181},[3168,5957,5958],{"class":3185},"> {\n",[3168,5960,5961],{"class":3170,"line":3196},[3168,5962,3193],{"emptyLinePlaceholder":3192},[3168,5964,5965,5967,5969,5971,5973],{"class":3170,"line":3215},[3168,5966,3199],{"class":3174},[3168,5968,3202],{"class":3174},[3168,5970,5217],{"class":3181},[3168,5972,5220],{"class":3208},[3168,5974,3212],{"class":3185},[3168,5976,5977,5979,5981,5983,5985],{"class":3170,"line":3220},[3168,5978,3199],{"class":3174},[3168,5980,3202],{"class":3174},[3168,5982,5217],{"class":3181},[3168,5984,5233],{"class":3208},[3168,5986,3212],{"class":3185},[3168,5988,5989],{"class":3170,"line":3232},[3168,5990,3193],{"emptyLinePlaceholder":3192},[3168,5992,5993,5995],{"class":3170,"line":3254},[3168,5994,3830],{"class":3185},[3168,5996,3833],{"class":3181},[3168,5998,5999,6001,6003],{"class":3170,"line":3264},[3168,6000,3223],{"class":3174},[3168,6002,5945],{"class":3226},[3168,6004,3251],{"class":3185},[3168,6006,6007,6009,6011,6013,6015,6018,6020,6022],{"class":3170,"line":3276},[3168,6008,5271],{"class":3185},[3168,6010,5274],{"class":3181},[3168,6012,3336],{"class":3185},[3168,6014,5044],{"class":3257},[3168,6016,6017],{"class":3185},")  ",[3168,6019,5284],{"class":3181},[3168,6021,5220],{"class":3208},[3168,6023,3261],{"class":3185},[3168,6025,6026,6028,6030,6032,6034,6036,6038],{"class":3170,"line":3282},[3168,6027,5271],{"class":3185},[3168,6029,5274],{"class":3181},[3168,6031,3336],{"class":3185},[3168,6033,5082],{"class":3257},[3168,6035,5320],{"class":3185},[3168,6037,5284],{"class":3181},[3168,6039,6040],{"class":3208}," user\n",[3168,6042,6043],{"class":3170,"line":3301},[3168,6044,5330],{"class":3185},[3168,6046,6047,6049,6051,6053],{"class":3170,"line":3320},[3168,6048,3348],{"class":3174},[3168,6050,3351],{"class":3185},[3168,6052,5339],{"class":3208},[3168,6054,6055],{"class":3185},"  = url;\n",[3168,6057,6058,6060,6062,6064],{"class":3170,"line":3345},[3168,6059,3348],{"class":3174},[3168,6061,3351],{"class":3185},[3168,6063,5351],{"class":3208},[3168,6065,6066],{"class":3185}," = user;\n",[3168,6068,6069],{"class":3170,"line":3366},[3168,6070,3369],{"class":3185},[3168,6072,6073],{"class":3170,"line":3372},[3168,6074,3193],{"emptyLinePlaceholder":3192},[3168,6076,6077,6079],{"class":3170,"line":4447},[3168,6078,3830],{"class":3185},[3168,6080,4946],{"class":3181},[3168,6082,6083,6085,6088,6091],{"class":3170,"line":4453},[3168,6084,3223],{"class":3174},[3168,6086,6087],{"class":3181}," DataSource",[3168,6089,6090],{"class":3226}," get",[3168,6092,3229],{"class":3185},[3168,6094,6095,6098,6101,6103,6105,6108],{"class":3170,"line":5129},[3168,6096,6097],{"class":3181},"        HikariConfig",[3168,6099,6100],{"class":3208}," config",[3168,6102,3241],{"class":3185},[3168,6104,3245],{"class":3244},[3168,6106,6107],{"class":3226}," HikariConfig",[3168,6109,3317],{"class":3185},[3168,6111,6112,6115,6117,6120],{"class":3170,"line":5134},[3168,6113,6114],{"class":3208},"        config",[3168,6116,3351],{"class":3185},[3168,6118,6119],{"class":3226},"setJdbcUrl",[3168,6121,6122],{"class":3185},"(url);\n",[3168,6124,6125,6127,6129,6132],{"class":3170,"line":5357},[3168,6126,6114],{"class":3208},[3168,6128,3351],{"class":3185},[3168,6130,6131],{"class":3226},"setUsername",[3168,6133,6134],{"class":3185},"(user);\n",[3168,6136,6137,6139,6141,6144,6146,6149],{"class":3170,"line":5370},[3168,6138,6114],{"class":3208},[3168,6140,3351],{"class":3185},[3168,6142,6143],{"class":3226},"setMaximumPoolSize",[3168,6145,3336],{"class":3185},[3168,6147,3619],{"class":6148},"sJj4R",[3168,6150,3342],{"class":3185},[3168,6152,6153,6155,6158,6161],{"class":3170,"line":5375},[3168,6154,5405],{"class":3244},[3168,6156,6157],{"class":3244}," new",[3168,6159,6160],{"class":3226}," HikariDataSource",[3168,6162,6163],{"class":3185},"(config);\n",[3168,6165,6166],{"class":3170,"line":5380},[3168,6167,3369],{"class":3185},[3168,6169,6170],{"class":3170,"line":5402},[3168,6171,3375],{"class":3185},[3168,6173,6174],{"class":3170,"line":5419},[3168,6175,3193],{"emptyLinePlaceholder":3192},[3168,6177,6178],{"class":3170,"line":5424},[3168,6179,6180],{"class":3943},"\u002F\u002F У модулі:\n",[3168,6182,6184,6186,6188,6190,6192,6194,6196,6199,6201,6204,6206,6208,6210,6212,6214,6216,6218,6220],{"class":3170,"line":6183},27,[3168,6185,5736],{"class":3226},[3168,6187,3336],{"class":3185},[3168,6189,5894],{"class":3208},[3168,6191,3351],{"class":3185},[3168,6193,4972],{"class":3208},[3168,6195,4975],{"class":3185},[3168,6197,6198],{"class":3226},"toProvider",[3168,6200,3336],{"class":3185},[3168,6202,6203],{"class":3208},"HikariDataSourceProvider",[3168,6205,3351],{"class":3185},[3168,6207,4972],{"class":3208},[3168,6209,4975],{"class":3185},[3168,6211,5796],{"class":3226},[3168,6213,3336],{"class":3185},[3168,6215,5801],{"class":3208},[3168,6217,3351],{"class":3185},[3168,6219,4972],{"class":3208},[3168,6221,3342],{"class":3185},[3150,6223,6224,6226,6227,6229,6230,6233],{},[3154,6225,6203],{}," сам також отримує ",[3154,6228,3892],{},"-залежності. Поєднання з ",[3154,6231,6232],{},"in(Singleton.class)"," гарантує, що пул з'єднань створюється рівно один раз.",[3514,6235,6237,6239],{"id":6236},"named-та-кастомні-кваліфікатори",[3154,6238,5150],{}," та кастомні кваліфікатори",[3150,6241,6242,6243,6246],{},"Коли один інтерфейс має кілька реалізацій — Guice потребує підказки, яку саме надавати. Для цього існують ",[3380,6244,6245],{},"кваліфікатори"," (qualifiers).",[3150,6248,6249,6254],{},[3380,6250,6251,6252],{},"Варіант 1 — ",[3154,6253,5150],{}," (найпростіший):",[3159,6256,6258],{"className":3161,"code":6257,"language":3163,"meta":3164,"style":3164},"\u002F\u002F У модулі — дві реалізації одного інтерфейсу\nbind(TrackRepository.class)\n    .annotatedWith(Names.named(\"jdbc\"))\n    .to(JdbcTrackRepository.class);\n\nbind(TrackRepository.class)\n    .annotatedWith(Names.named(\"cache\"))\n    .to(InMemoryCacheTrackRepository.class);\n\n\u002F\u002F У споживачі — явно вказуємо, яка потрібна\n@Inject\npublic TrackService(\n    @Named(\"jdbc\")  TrackRepository primary,\n    @Named(\"cache\") TrackRepository cache\n) { ... }\n",[3154,6259,6260,6265,6279,6301,6317,6321,6335,6356,6373,6377,6382,6388,6396,6413,6430],{"__ignoreMap":3164},[3168,6261,6262],{"class":3170,"line":3171},[3168,6263,6264],{"class":3943},"\u002F\u002F У модулі — дві реалізації одного інтерфейсу\n",[3168,6266,6267,6269,6271,6273,6275,6277],{"class":3170,"line":3189},[3168,6268,5736],{"class":3226},[3168,6270,3336],{"class":3185},[3168,6272,3407],{"class":3208},[3168,6274,3351],{"class":3185},[3168,6276,4972],{"class":3208},[3168,6278,4036],{"class":3185},[3168,6280,6281,6284,6286,6288,6290,6292,6294,6296,6299],{"class":3170,"line":3196},[3168,6282,6283],{"class":3185},"    .",[3168,6285,5029],{"class":3226},[3168,6287,3336],{"class":3185},[3168,6289,5034],{"class":3208},[3168,6291,3351],{"class":3185},[3168,6293,5039],{"class":3226},[3168,6295,3336],{"class":3185},[3168,6297,6298],{"class":3257},"\"jdbc\"",[3168,6300,5047],{"class":3185},[3168,6302,6303,6305,6307,6309,6311,6313,6315],{"class":3170,"line":3215},[3168,6304,6283],{"class":3185},[3168,6306,4819],{"class":3226},[3168,6308,3336],{"class":3185},[3168,6310,3598],{"class":3208},[3168,6312,3351],{"class":3185},[3168,6314,4972],{"class":3208},[3168,6316,3342],{"class":3185},[3168,6318,6319],{"class":3170,"line":3220},[3168,6320,3193],{"emptyLinePlaceholder":3192},[3168,6322,6323,6325,6327,6329,6331,6333],{"class":3170,"line":3232},[3168,6324,5736],{"class":3226},[3168,6326,3336],{"class":3185},[3168,6328,3407],{"class":3208},[3168,6330,3351],{"class":3185},[3168,6332,4972],{"class":3208},[3168,6334,4036],{"class":3185},[3168,6336,6337,6339,6341,6343,6345,6347,6349,6351,6354],{"class":3170,"line":3254},[3168,6338,6283],{"class":3185},[3168,6340,5029],{"class":3226},[3168,6342,3336],{"class":3185},[3168,6344,5034],{"class":3208},[3168,6346,3351],{"class":3185},[3168,6348,5039],{"class":3226},[3168,6350,3336],{"class":3185},[3168,6352,6353],{"class":3257},"\"cache\"",[3168,6355,5047],{"class":3185},[3168,6357,6358,6360,6362,6364,6367,6369,6371],{"class":3170,"line":3264},[3168,6359,6283],{"class":3185},[3168,6361,4819],{"class":3226},[3168,6363,3336],{"class":3185},[3168,6365,6366],{"class":3208},"InMemoryCacheTrackRepository",[3168,6368,3351],{"class":3185},[3168,6370,4972],{"class":3208},[3168,6372,3342],{"class":3185},[3168,6374,6375],{"class":3170,"line":3276},[3168,6376,3193],{"emptyLinePlaceholder":3192},[3168,6378,6379],{"class":3170,"line":3282},[3168,6380,6381],{"class":3943},"\u002F\u002F У споживачі — явно вказуємо, яка потрібна\n",[3168,6383,6384,6386],{"class":3170,"line":3301},[3168,6385,5191],{"class":3185},[3168,6387,3833],{"class":3181},[3168,6389,6390,6392,6394],{"class":3170,"line":3320},[3168,6391,3175],{"class":3174},[3168,6393,3205],{"class":3226},[3168,6395,3251],{"class":3185},[3168,6397,6398,6400,6402,6404,6406,6408,6410],{"class":3170,"line":3345},[3168,6399,3830],{"class":3185},[3168,6401,5274],{"class":3181},[3168,6403,3336],{"class":3185},[3168,6405,6298],{"class":3257},[3168,6407,6017],{"class":3185},[3168,6409,3407],{"class":3181},[3168,6411,6412],{"class":3185}," primary,\n",[3168,6414,6415,6417,6419,6421,6423,6425,6427],{"class":3170,"line":3366},[3168,6416,3830],{"class":3185},[3168,6418,5274],{"class":3181},[3168,6420,3336],{"class":3185},[3168,6422,6353],{"class":3257},[3168,6424,5320],{"class":3185},[3168,6426,3407],{"class":3181},[3168,6428,6429],{"class":3185}," cache\n",[3168,6431,6432],{"class":3170,"line":3372},[3168,6433,6434],{"class":3185},") { ... }\n",[3150,6436,6437,6440,6441,6444],{},[3380,6438,6439],{},"Варіант 2 — кастомний кваліфікатор"," (",[3154,6442,6443],{},"@BindingAnnotation","):",[3159,6446,6448],{"className":3161,"code":6447,"language":3163,"meta":3164,"style":3164},"\u002F\u002F Оголошення кваліфікатора — окрема анотація\n@Retention(RetentionPolicy.RUNTIME)\n@Target({ElementType.FIELD, ElementType.PARAMETER})\n@BindingAnnotation\npublic @interface ReadOnly {}\n\n\u002F\u002F У модулі\nbind(TrackRepository.class)\n    .annotatedWith(ReadOnly.class)\n    .to(ReadOnlyJdbcTrackRepository.class);\n\n\u002F\u002F У споживачі\n@Inject\npublic AnalyticsService(@ReadOnly TrackRepository repo) { ... }\n",[3154,6449,6450,6455,6474,6504,6511,6527,6531,6536,6550,6567,6584,6588,6593,6599],{"__ignoreMap":3164},[3168,6451,6452],{"class":3170,"line":3171},[3168,6453,6454],{"class":3943},"\u002F\u002F Оголошення кваліфікатора — окрема анотація\n",[3168,6456,6457,6459,6462,6464,6467,6469,6472],{"class":3170,"line":3189},[3168,6458,5191],{"class":3185},[3168,6460,6461],{"class":3181},"Retention",[3168,6463,3336],{"class":3185},[3168,6465,6466],{"class":3208},"RetentionPolicy",[3168,6468,3351],{"class":3185},[3168,6470,6471],{"class":3208},"RUNTIME",[3168,6473,4036],{"class":3185},[3168,6475,6476,6478,6481,6484,6487,6489,6492,6494,6496,6498,6501],{"class":3170,"line":3196},[3168,6477,5191],{"class":3185},[3168,6479,6480],{"class":3181},"Target",[3168,6482,6483],{"class":3185},"({",[3168,6485,6486],{"class":3208},"ElementType",[3168,6488,3351],{"class":3185},[3168,6490,6491],{"class":3208},"FIELD",[3168,6493,3270],{"class":3185},[3168,6495,6486],{"class":3208},[3168,6497,3351],{"class":3185},[3168,6499,6500],{"class":3208},"PARAMETER",[3168,6502,6503],{"class":3185},"})\n",[3168,6505,6506,6508],{"class":3170,"line":3215},[3168,6507,5191],{"class":3185},[3168,6509,6510],{"class":3181},"BindingAnnotation\n",[3168,6512,6513,6515,6518,6521,6524],{"class":3170,"line":3220},[3168,6514,3175],{"class":3174},[3168,6516,6517],{"class":3185}," @",[3168,6519,6520],{"class":3174},"interface",[3168,6522,6523],{"class":3181}," ReadOnly",[3168,6525,6526],{"class":3185}," {}\n",[3168,6528,6529],{"class":3170,"line":3232},[3168,6530,3193],{"emptyLinePlaceholder":3192},[3168,6532,6533],{"class":3170,"line":3254},[3168,6534,6535],{"class":3943},"\u002F\u002F У модулі\n",[3168,6537,6538,6540,6542,6544,6546,6548],{"class":3170,"line":3264},[3168,6539,5736],{"class":3226},[3168,6541,3336],{"class":3185},[3168,6543,3407],{"class":3208},[3168,6545,3351],{"class":3185},[3168,6547,4972],{"class":3208},[3168,6549,4036],{"class":3185},[3168,6551,6552,6554,6556,6558,6561,6563,6565],{"class":3170,"line":3276},[3168,6553,6283],{"class":3185},[3168,6555,5029],{"class":3226},[3168,6557,3336],{"class":3185},[3168,6559,6560],{"class":3208},"ReadOnly",[3168,6562,3351],{"class":3185},[3168,6564,4972],{"class":3208},[3168,6566,4036],{"class":3185},[3168,6568,6569,6571,6573,6575,6578,6580,6582],{"class":3170,"line":3282},[3168,6570,6283],{"class":3185},[3168,6572,4819],{"class":3226},[3168,6574,3336],{"class":3185},[3168,6576,6577],{"class":3208},"ReadOnlyJdbcTrackRepository",[3168,6579,3351],{"class":3185},[3168,6581,4972],{"class":3208},[3168,6583,3342],{"class":3185},[3168,6585,6586],{"class":3170,"line":3301},[3168,6587,3193],{"emptyLinePlaceholder":3192},[3168,6589,6590],{"class":3170,"line":3320},[3168,6591,6592],{"class":3943},"\u002F\u002F У споживачі\n",[3168,6594,6595,6597],{"class":3170,"line":3345},[3168,6596,5191],{"class":3185},[3168,6598,3833],{"class":3181},[3168,6600,6601,6603,6606,6609,6611,6613],{"class":3170,"line":3366},[3168,6602,3175],{"class":3174},[3168,6604,6605],{"class":3226}," AnalyticsService",[3168,6607,6608],{"class":3185},"(@",[3168,6610,6560],{"class":3181},[3168,6612,3804],{"class":3181},[3168,6614,6615],{"class":3185}," repo) { ... }\n",[3150,6617,6618,6619,6621,6622,4975],{},"Кастомні кваліфікатори кращі за ",[3154,6620,5150],{}," у великих проєктах: вони рефакторяться разом з кодом (компілятор перевірить назву анотації, але не перевірить рядок у ",[3154,6623,5150],{},[4187,6625,4195,6627,4195,6634,4195,6638,4195,6642,4195,6647,4195,6650,4195,6653,4195,6656,4195,6660,4195,6664,4195,6667,4195,6670,4195,6673,4195,6676,4195,6679,4195,6683,4195,6687,4195,6689,4195,6692,4195,6695,4195,6698,4195,6704,4195,6708,4195,6711,4195,6714,4195,6717,4195,6720,4195,6726,4195,6730,4195,6734,4195,6738,4195,6742,4195,6746,4195,6749,4195,6752,4195,6756],{"viewBox":4189,"className":6626,"xmlns":4194},[4191,4192,4193],[4197,6628,4199,6629,4195],{},[4201,6630,6632],{"id":6631,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-b3",[4209,6633],{"d":4211,"fill":3706},[3607,6635,6637],{"x":3618,"y":6636,"style":3611,"fontSize":4282,"fontWeight":3613,"fill":4222},"22","Типи Bindings у Guice",[3625,6639],{"x":6640,"y":4304,"width":4268,"height":4275,"rx":3619,"fill":6641,"stroke":3706,"strokeWidth":3632},"20","rgba(96,165,250,0.07)",[3607,6643,6646],{"x":6644,"y":6645,"style":3611,"fontSize":3637,"fontWeight":3613,"fill":3706},"115","62","Linked Binding",[3607,6648,6649],{"x":6644,"y":4273,"style":3611,"fontSize":3641,"fill":3642},"bind(Interface.class)",[3607,6651,6652],{"x":6644,"y":4247,"style":3611,"fontSize":3641,"fill":3642},".to(Implementation.class)",[3607,6654,6655],{"x":6644,"y":6644,"style":3611,"fontSize":4204,"fill":3621},"Заміна реалізації без зміни",[3607,6657,6659],{"x":6644,"y":6658,"style":3611,"fontSize":4204,"fill":3621},"128","споживачів. Найпоширеніший.",[3625,6661],{"x":6662,"y":4304,"width":4268,"height":4275,"rx":3619,"fill":6663,"stroke":3614,"strokeWidth":3632},"235","rgba(52,211,153,0.07)",[3607,6665,6666],{"x":4272,"y":6645,"style":3611,"fontSize":3637,"fontWeight":3613,"fill":3614},"Instance Binding",[3607,6668,6669],{"x":4272,"y":4273,"style":3611,"fontSize":3641,"fill":3642},"bind(Cls.class)",[3607,6671,6672],{"x":4272,"y":4247,"style":3611,"fontSize":3641,"fill":3642},".toInstance(obj)",[3607,6674,6675],{"x":4272,"y":6644,"style":3611,"fontSize":4204,"fill":3621},"Готовий об'єкт. Автоматично",[3607,6677,6678],{"x":4272,"y":6658,"style":3611,"fontSize":4204,"fill":3621},"є Singleton.",[3625,6680],{"x":6681,"y":4304,"width":4268,"height":4275,"rx":3619,"fill":6682,"stroke":4222,"strokeWidth":3632},"450","rgba(245,158,11,0.07)",[3607,6684,6686],{"x":6685,"y":6645,"style":3611,"fontSize":3637,"fontWeight":3613,"fill":4222},"545","Provider Binding",[3607,6688,6669],{"x":6685,"y":4273,"style":3611,"fontSize":3641,"fill":3642},[3607,6690,6691],{"x":6685,"y":4247,"style":3611,"fontSize":3641,"fill":3642},".toProvider(Provider.class)",[3607,6693,6694],{"x":6685,"y":6644,"style":3611,"fontSize":4204,"fill":3621},"Складна логіка створення.",[3607,6696,6697],{"x":6685,"y":6658,"style":3611,"fontSize":4204,"fill":3621},"HikariCP, кешування.",[3625,6699],{"x":6700,"y":4304,"width":6701,"height":4275,"rx":3619,"fill":6702,"stroke":6703,"strokeWidth":3632},"665","175","rgba(167,139,250,0.07)","#a78bfa",[3607,6705,6707],{"x":6706,"y":6645,"style":3611,"fontSize":3637,"fontWeight":3613,"fill":6703},"752","Constant Binding",[3607,6709,6710],{"x":6706,"y":4273,"style":3611,"fontSize":3641,"fill":3642},"bindConstant()",[3607,6712,6713],{"x":6706,"y":4247,"style":3611,"fontSize":3641,"fill":3642},".annotatedWith(@Named)",[3607,6715,6716],{"x":6706,"y":6644,"style":3611,"fontSize":4204,"fill":3621},"Рядки, числа, enum-значення.",[3607,6718,6719],{"x":6706,"y":6658,"style":3611,"fontSize":4204,"fill":3621},"URL, порти, ліміти.",[3625,6721],{"x":6640,"y":6722,"width":6723,"height":3635,"rx":3619,"fill":6724,"stroke":6725,"strokeWidth":3632},"168","390","rgba(239,68,68,0.05)","rgba(239,68,68,0.3)",[3607,6727,6729],{"x":3699,"y":6728,"style":3611,"fontSize":3637,"fontWeight":3613,"fill":3631},"188","@Named (рядковий кваліфікатор)",[3607,6731,6733],{"x":3699,"y":6732,"style":3611,"fontSize":3641,"fill":3642},"206","bind(Repo.class).annotatedWith(Names.named(\"jdbc\")).to(...)",[3607,6735,6737],{"x":3699,"y":6736,"style":3611,"fontSize":3641,"fill":3642},"224","bind(Repo.class).annotatedWith(Names.named(\"cache\")).to(...)",[3607,6739,6741],{"x":3699,"y":6740,"style":3611,"fontSize":4204,"fill":3621},"248","Швидко, але refactoring-небезпечно: рядок не перевіряє компілятор",[3625,6743],{"x":6681,"y":6722,"width":6723,"height":3635,"rx":3619,"fill":6744,"stroke":6745,"strokeWidth":3632},"rgba(52,211,153,0.05)","rgba(52,211,153,0.3)",[3607,6747,6748],{"x":3609,"y":6728,"style":3611,"fontSize":3637,"fontWeight":3613,"fill":3614},"@BindingAnnotation (кастомний кваліфікатор)",[3607,6750,6751],{"x":3609,"y":6732,"style":3611,"fontSize":3641,"fill":3642},"@Retention(RUNTIME) @BindingAnnotation",[3607,6753,6755],{"x":3609,"y":6754,"style":3611,"fontSize":3641,"fill":3642},"220","public @interface ReadOnly {}",[3607,6757,6758],{"x":3609,"y":6740,"style":3611,"fontSize":4204,"fill":3621},"Compile-time безпека. Рекомендовано для великих проєктів.",[6760,6761,6762,6773,6783,6796,6801],"field-group",{},[6763,6764,6766,6767,6769,6770,6772],"field",{"name":3892,"type":6765},"annotation","Позначає конструктор, поле або метод як точку впровадження. Guice автоматично надає значення при створенні об'єкта. Є у пакеті ",[3154,6768,4165],{}," (стандарт) та ",[3154,6771,4169],{}," (Guice-специфічний).",[6763,6774,6775,6776,6778,6779,6782],{"name":4519,"type":6765},"Декларує, що Guice повинен створити рівно один екземпляр класу на весь ",[3154,6777,4137],{},". Можна застосувати до класу або у виклику ",[3154,6780,6781],{},"bind(...).in(Singleton.class)",". Потоко-безпечний: Guice гарантує атомарність створення.",[6763,6784,6785,6786,5711,6788,6790,6791,4510,6794,3351],{"name":5150,"type":6765},"Кваліфікатор для розрізнення кількох прив'язок одного типу. Використовується разом з ",[3154,6787,5441],{},[3154,6789,4130],{},". Доступний з ",[3154,6792,6793],{},"com.google.inject.name",[3154,6795,4165],{},[6763,6797,6798,6799,3351],{"name":6443,"type":6765},"Мета-анотація для створення кастомних кваліфікаторів. Оголошена на вашій анотації дає Guice знати, що вона є кваліфікатором. Перевіряється компілятором — безпечніша за ",[3154,6800,5150],{},[6763,6802,6804,6805,6807],{"name":6803,"type":6765},"@Provides","Альтернатива класу-провайдеру: метод у Module з анотацією ",[3154,6806,6803],{}," стає фабрикою. Зручніше для простих випадків.",[3514,6809,5708,6811,6813],{"id":6810},"метод-provides-як-зручна-альтернатива-provider-класу",[3154,6812,6803],{}," як зручна альтернатива Provider-класу",[3150,6815,6816,6817,6819],{},"Замість окремого класу, що реалізує ",[3154,6818,4177],{},", можна написати метод прямо у Module:",[3159,6821,6823],{"className":3161,"code":6822,"language":3163,"meta":3164,"style":3164},"public class AppModule extends AbstractModule {\n\n    @Override\n    protected void configure() {\n        bind(TrackRepository.class).to(JdbcTrackRepository.class);\n    }\n\n    \u002F\u002F Guice автоматично визнає цей метод як провайдер для DataSource\n    @Provides\n    @Singleton\n    DataSource provideDataSource(\n        @Named(\"dbUrl\")  String url,\n        @Named(\"dbUser\") String user\n    ) {\n        HikariConfig config = new HikariConfig();\n        config.setJdbcUrl(url);\n        config.setUsername(user);\n        config.setMaximumPoolSize(10);\n        return new HikariDataSource(config);\n    }\n}\n",[3154,6824,6825,6839,6843,6849,6859,6885,6889,6893,6898,6905,6911,6921,6939,6955,6959,6973,6983,6993,7007,7017,7021],{"__ignoreMap":3164},[3168,6826,6827,6829,6831,6833,6835,6837],{"class":3170,"line":3171},[3168,6828,3175],{"class":3174},[3168,6830,3178],{"class":3174},[3168,6832,4927],{"class":3181},[3168,6834,4930],{"class":3174},[3168,6836,4933],{"class":3181},[3168,6838,3186],{"class":3185},[3168,6840,6841],{"class":3170,"line":3189},[3168,6842,3193],{"emptyLinePlaceholder":3192},[3168,6844,6845,6847],{"class":3170,"line":3196},[3168,6846,3830],{"class":3185},[3168,6848,4946],{"class":3181},[3168,6850,6851,6853,6855,6857],{"class":3170,"line":3215},[3168,6852,4951],{"class":3174},[3168,6854,4043],{"class":3181},[3168,6856,4956],{"class":3226},[3168,6858,3229],{"class":3185},[3168,6860,6861,6863,6865,6867,6869,6871,6873,6875,6877,6879,6881,6883],{"class":3170,"line":3220},[3168,6862,4963],{"class":3226},[3168,6864,3336],{"class":3185},[3168,6866,3407],{"class":3208},[3168,6868,3351],{"class":3185},[3168,6870,4972],{"class":3208},[3168,6872,4975],{"class":3185},[3168,6874,4819],{"class":3226},[3168,6876,3336],{"class":3185},[3168,6878,3598],{"class":3208},[3168,6880,3351],{"class":3185},[3168,6882,4972],{"class":3208},[3168,6884,3342],{"class":3185},[3168,6886,6887],{"class":3170,"line":3232},[3168,6888,3369],{"class":3185},[3168,6890,6891],{"class":3170,"line":3254},[3168,6892,3193],{"emptyLinePlaceholder":3192},[3168,6894,6895],{"class":3170,"line":3264},[3168,6896,6897],{"class":3943},"    \u002F\u002F Guice автоматично визнає цей метод як провайдер для DataSource\n",[3168,6899,6900,6902],{"class":3170,"line":3276},[3168,6901,3830],{"class":3185},[3168,6903,6904],{"class":3181},"Provides\n",[3168,6906,6907,6909],{"class":3170,"line":3282},[3168,6908,3830],{"class":3185},[3168,6910,5194],{"class":3181},[3168,6912,6913,6916,6919],{"class":3170,"line":3301},[3168,6914,6915],{"class":3181},"    DataSource",[3168,6917,6918],{"class":3226}," provideDataSource",[3168,6920,3251],{"class":3185},[3168,6922,6923,6925,6927,6929,6931,6933,6935,6937],{"class":3170,"line":3320},[3168,6924,5271],{"class":3185},[3168,6926,5274],{"class":3181},[3168,6928,3336],{"class":3185},[3168,6930,5044],{"class":3257},[3168,6932,6017],{"class":3185},[3168,6934,5284],{"class":3181},[3168,6936,5220],{"class":3208},[3168,6938,3261],{"class":3185},[3168,6940,6941,6943,6945,6947,6949,6951,6953],{"class":3170,"line":3345},[3168,6942,5271],{"class":3185},[3168,6944,5274],{"class":3181},[3168,6946,3336],{"class":3185},[3168,6948,5082],{"class":3257},[3168,6950,5320],{"class":3185},[3168,6952,5284],{"class":3181},[3168,6954,6040],{"class":3208},[3168,6956,6957],{"class":3170,"line":3366},[3168,6958,5330],{"class":3185},[3168,6960,6961,6963,6965,6967,6969,6971],{"class":3170,"line":3372},[3168,6962,6097],{"class":3181},[3168,6964,6100],{"class":3208},[3168,6966,3241],{"class":3185},[3168,6968,3245],{"class":3244},[3168,6970,6107],{"class":3226},[3168,6972,3317],{"class":3185},[3168,6974,6975,6977,6979,6981],{"class":3170,"line":4447},[3168,6976,6114],{"class":3208},[3168,6978,3351],{"class":3185},[3168,6980,6119],{"class":3226},[3168,6982,6122],{"class":3185},[3168,6984,6985,6987,6989,6991],{"class":3170,"line":4453},[3168,6986,6114],{"class":3208},[3168,6988,3351],{"class":3185},[3168,6990,6131],{"class":3226},[3168,6992,6134],{"class":3185},[3168,6994,6995,6997,6999,7001,7003,7005],{"class":3170,"line":5129},[3168,6996,6114],{"class":3208},[3168,6998,3351],{"class":3185},[3168,7000,6143],{"class":3226},[3168,7002,3336],{"class":3185},[3168,7004,3619],{"class":6148},[3168,7006,3342],{"class":3185},[3168,7008,7009,7011,7013,7015],{"class":3170,"line":5134},[3168,7010,5405],{"class":3244},[3168,7012,6157],{"class":3244},[3168,7014,6160],{"class":3226},[3168,7016,6163],{"class":3185},[3168,7018,7019],{"class":3170,"line":5357},[3168,7020,3369],{"class":3185},[3168,7022,7023],{"class":3170,"line":5370},[3168,7024,3375],{"class":3185},[3150,7026,7027,7028,7030,7031,7033,7034,7036,7037,7039],{},"Параметри методу ",[3154,7029,6803],{}," автоматично впроваджуються Guice — він вирішує їх так само, як і параметри ",[3154,7032,3892],{},"-конструкторів. Поєднання ",[3154,7035,6803],{}," + ",[3154,7038,4519],{}," гарантує, що HikariCP-пул буде створено рівно один раз.",[4087,7041,7042,7048,7049,7051,7052,3351],{},[3380,7043,7044,7045,7047],{},"Коли обирати ",[3154,7046,6803],{}," vs клас-провайдер:"," якщо логіка провайдера проста та умістилась в один метод — використовуйте ",[3154,7050,6803],{},". Якщо провайдер складний, потребує стану або хочеться тестувати його окремо — винесіть у власний клас, що реалізує ",[3154,7053,4177],{},[3507,7055],{},[3145,7057,7059],{"id":7058},"інтеграція-з-javafx-вирішення-ключової-суперечності","Інтеграція з JavaFX: вирішення ключової суперечності",[3150,7061,7062,7063,7065],{},"Переходимо до центрального практичного питання цієї статті: як поєднати Guice з JavaFX? На перший погляд завдання виглядає тривіально — просто додайте ",[3154,7064,3892],{}," до контролерів. Але реальність складніша.",[3514,7067,7069],{"id":7068},"проблема-fxmlloader-прихований-конкурент","Проблема: FXMLLoader — прихований конкурент",[3150,7071,7072,7073,7076,7077,7080,7081,7084,7085,7088,7089,7091,7092,3351],{},"JavaFX використовує ",[3154,7074,7075],{},"FXMLLoader"," для завантаження ",[3154,7078,7079],{},".fxml","-файлів та автоматичного створення контролерів, описаних у ",[3154,7082,7083],{},"fx:controller",". Цей механізм ",[3380,7086,7087],{},"сам інстанціює контролери"," через рефлексію, викликаючи конструктор за замовчуванням (без параметрів). Guice при цьому не задіяний — він нічого не знає про ці об'єкти, і всі ",[3154,7090,3892],{},"-поля залишаються ",[3154,7093,7094],{},"null",[3150,7096,7097],{},"Якщо у контролері написати:",[3159,7099,7101],{"className":3161,"code":7100,"language":3163,"meta":3164,"style":3164},"public class TrackListController {\n    @Inject\n    private TrackService trackService; \u002F\u002F завжди null при стандартному FXMLLoader!\n}\n",[3154,7102,7103,7113,7119,7133],{"__ignoreMap":3164},[3168,7104,7105,7107,7109,7111],{"class":3170,"line":3171},[3168,7106,3175],{"class":3174},[3168,7108,3178],{"class":3174},[3168,7110,3182],{"class":3181},[3168,7112,3186],{"class":3185},[3168,7114,7115,7117],{"class":3170,"line":3189},[3168,7116,3830],{"class":3185},[3168,7118,3833],{"class":3181},[3168,7120,7121,7123,7125,7127,7130],{"class":3170,"line":3196},[3168,7122,3199],{"class":3174},[3168,7124,3205],{"class":3181},[3168,7126,3209],{"class":3208},[3168,7128,7129],{"class":3185},"; ",[3168,7131,7132],{"class":3943},"\u002F\u002F завжди null при стандартному FXMLLoader!\n",[3168,7134,7135],{"class":3170,"line":3215},[3168,7136,3375],{"class":3185},[3150,7138,5708,7139,7141,7142,7144,7145,7147,7148,7151],{},[3154,7140,3156],{}," викличеться, ",[3154,7143,3354],{}," буде ",[3154,7146,7094],{},", і застосунок впаде з ",[3154,7149,7150],{},"NullPointerException",". Це класична пастка, у яку потрапляє кожен, хто вперше намагається поєднати JavaFX та DI.",[3514,7153,7155,7156],{"id":7154},"рішення-setcontrollerfactory","Рішення: ",[3154,7157,7158],{},"setControllerFactory()",[3150,7160,7161,7163,7164,7167],{},[3154,7162,7075],{}," передбачає цю ситуацію. Метод ",[3154,7165,7166],{},"setControllerFactory(Callback\u003CClass\u003C?>, Object> factory)"," дозволяє замінити стандартний механізм створення контролерів на довільну фабрику. І саме тут з'являється Guice:",[3159,7169,7171],{"className":3161,"code":7170,"language":3163,"meta":3164,"style":3164},"FXMLLoader loader = new FXMLLoader(getClass().getResource(\"\u002Ffxml\u002Ftrack-list.fxml\"));\n\n\u002F\u002F Передаємо Injector як фабрику контролерів\n\u002F\u002F injector::getInstance — це method reference на Injector.getInstance(Class)\nloader.setControllerFactory(injector::getInstance);\n\nParent root = loader.load();\n",[3154,7172,7173,7205,7209,7214,7219,7237,7241],{"__ignoreMap":3164},[3168,7174,7175,7177,7180,7182,7184,7187,7189,7192,7194,7197,7199,7202],{"class":3170,"line":3171},[3168,7176,7075],{"class":3181},[3168,7178,7179],{"class":3208}," loader",[3168,7181,3241],{"class":3185},[3168,7183,3245],{"class":3244},[3168,7185,7186],{"class":3226}," FXMLLoader",[3168,7188,3336],{"class":3185},[3168,7190,7191],{"class":3226},"getClass",[3168,7193,5026],{"class":3185},[3168,7195,7196],{"class":3226},"getResource",[3168,7198,3336],{"class":3185},[3168,7200,7201],{"class":3257},"\"\u002Ffxml\u002Ftrack-list.fxml\"",[3168,7203,7204],{"class":3185},"));\n",[3168,7206,7207],{"class":3170,"line":3189},[3168,7208,3193],{"emptyLinePlaceholder":3192},[3168,7210,7211],{"class":3170,"line":3196},[3168,7212,7213],{"class":3943},"\u002F\u002F Передаємо Injector як фабрику контролерів\n",[3168,7215,7216],{"class":3170,"line":3215},[3168,7217,7218],{"class":3943},"\u002F\u002F injector::getInstance — це method reference на Injector.getInstance(Class)\n",[3168,7220,7221,7224,7226,7229,7232,7234],{"class":3170,"line":3220},[3168,7222,7223],{"class":3208},"loader",[3168,7225,3351],{"class":3185},[3168,7227,7228],{"class":3226},"setControllerFactory",[3168,7230,7231],{"class":3185},"(injector",[3168,7233,5601],{"class":3244},[3168,7235,7236],{"class":3185},"getInstance);\n",[3168,7238,7239],{"class":3170,"line":3232},[3168,7240,3193],{"emptyLinePlaceholder":3192},[3168,7242,7243,7246,7249,7251,7253,7255,7258],{"class":3170,"line":3254},[3168,7244,7245],{"class":3181},"Parent",[3168,7247,7248],{"class":3208}," root",[3168,7250,3241],{"class":3185},[3168,7252,7223],{"class":3208},[3168,7254,3351],{"class":3185},[3168,7256,7257],{"class":3226},"load",[3168,7259,3317],{"class":3185},[3150,7261,7262,7263,7265,7266,7269,7270,7273,7274,7277,7278,7280],{},"Тепер, коли ",[3154,7264,7075],{}," зустрічає ",[3154,7267,7268],{},"fx:controller=\"...TrackListController\"",", він не викликає ",[3154,7271,7272],{},"new TrackListController()",", а передає клас у ",[3154,7275,7276],{},"injector.getInstance(TrackListController.class)",". Guice будує контролер зі всіма залежностями через ",[3154,7279,3892],{},"-конструктор.",[4187,7282,4195,7285,4195,7302,4195,7304,4195,7308,4195,7311,4195,7315,4195,7317,4195,7320,4195,7323,4195,7326,4195,7330,4195,7333,4195,7336,4195,7339,4195,7343,4195,7345,4195,7348,4195,7353,4195,7358,4195,7361,4195,7363,4195,7367,4195,7371,4195,7375,4195,7378,4195,7382,4195,7385,4195,7389,4195,7391,4195,7393,4195,7396,4195,7399,4195,7402,4195,7406,4195,7409,4195,7412,4195,7416],{"viewBox":7283,"className":7284,"xmlns":4194},"0 0 860 330",[4191,4192,4193],[4197,7286,4199,7287,4199,7292,4199,7297,4195],{},[4201,7288,7290],{"id":7289,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-b4",[4209,7291],{"d":4211,"fill":3706},[4201,7293,7295],{"id":7294,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-g4",[4209,7296],{"d":4211,"fill":3614},[4201,7298,7300],{"id":7299,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-a4",[4209,7301],{"d":4211,"fill":4222},[3625,7303],{"x":6640,"y":6640,"width":4340,"height":4304,"rx":4204,"fill":4243,"stroke":3706,"strokeWidth":3632},[3607,7305,7307],{"x":4273,"y":7306,"style":3611,"fontSize":3619,"fontWeight":3613,"fill":3706},"37","main()",[3607,7309,7310],{"x":4273,"y":4232,"style":3611,"fontSize":4204,"fill":3642},"Application.launch()",[3170,7312],{"x1":4231,"y1":4304,"x2":7313,"y2":4304,"stroke":3706,"strokeWidth":3632,"markerEnd":7314},"180","url(#arr-b4)",[3625,7316],{"x":7313,"y":6640,"width":4231,"height":4304,"rx":4204,"fill":4243,"stroke":3706,"strokeWidth":3632},[3607,7318,7319],{"x":3650,"y":7306,"style":3611,"fontSize":3619,"fontWeight":3613,"fill":3706},"App.start()",[3607,7321,7322],{"x":3650,"y":4232,"style":3611,"fontSize":4204,"fill":3642},"extends Application",[3170,7324],{"x1":3650,"y1":3711,"x2":3650,"y2":4275,"stroke":4222,"strokeWidth":3632,"markerEnd":7325},"url(#arr-a4)",[3625,7327],{"x":7328,"y":4275,"width":7313,"height":7329,"rx":4204,"fill":4277,"stroke":4222,"strokeWidth":3707},"160","44",[3607,7331,4285],{"x":3650,"y":7332,"style":3611,"fontSize":3619,"fontWeight":3613,"fill":4222},"118",[3607,7334,4288],{"x":3650,"y":7335,"style":3611,"fontSize":3641,"fill":3642},"133",[3170,7337],{"x1":3650,"y1":4318,"x2":3650,"y2":7338,"stroke":4222,"strokeWidth":3632,"markerEnd":7325},"184",[3625,7340],{"x":7341,"y":7338,"width":3664,"height":4304,"rx":4204,"fill":7342,"stroke":4222,"strokeWidth":3632},"145","rgba(245,158,11,0.06)",[3607,7344,7075],{"x":3650,"y":4227,"style":3611,"fontSize":3641,"fontWeight":3613,"fill":4222},[3607,7346,7347],{"x":3650,"y":3699,"style":3611,"fontSize":4204,"fill":3642},".setControllerFactory(injector::getInstance)",[3170,7349],{"x1":7350,"y1":7351,"x2":3704,"y2":7351,"stroke":3614,"strokeWidth":3632,"markerEnd":7352},"355","204","url(#arr-g4)",[3607,7354,7357],{"x":7355,"y":7356,"style":3611,"fontSize":4204,"fill":3614},"397","196","fx:controller=",[3607,7359,7360],{"x":7355,"y":4327,"style":3611,"fontSize":4204,"fill":3614},"\"TrackListCtrl\"",[3625,7362],{"x":3704,"y":4265,"width":6754,"height":3711,"rx":4204,"fill":3724,"stroke":3614,"strokeWidth":3707},[3607,7364,7366],{"x":7365,"y":7356,"style":3611,"fontSize":3619,"fontWeight":3613,"fill":3614},"550","Injector.getInstance",[3607,7368,7370],{"x":7365,"y":7369,"style":3611,"fontSize":4204,"fill":3642},"212","(TrackListController.class)",[3607,7372,7374],{"x":7365,"y":7373,"style":3611,"fontSize":4204,"fill":3642},"226","@Inject → вирішення залежностей",[3170,7376],{"x1":7365,"y1":7377,"x2":7365,"y2":4295,"stroke":3614,"strokeWidth":3632,"markerEnd":7352},"234",[3625,7379],{"x":3618,"y":4295,"width":7380,"height":7329,"rx":4204,"fill":7381,"stroke":3614,"strokeWidth":3632},"240","rgba(52,211,153,0.1)",[3607,7383,3403],{"x":7365,"y":7384,"style":3611,"fontSize":3619,"fontWeight":3613,"fill":3614},"288",[3607,7386,7388],{"x":7365,"y":7387,"style":3611,"fontSize":4204,"fill":3642},"304","з TrackService, NavigationService…",[3170,7390],{"x1":3650,"y1":6736,"x2":3650,"y2":4295,"stroke":3706,"strokeWidth":3632,"markerEnd":7314},[3625,7392],{"x":4231,"y":4295,"width":7313,"height":7329,"rx":4204,"fill":3705,"stroke":3706,"strokeWidth":3632},[3607,7394,7395],{"x":4241,"y":7384,"style":3611,"fontSize":3619,"fontWeight":3613,"fill":3706},"Scene \u002F Stage",[3607,7397,7398],{"x":4241,"y":7387,"style":3611,"fontSize":4204,"fill":3642},"primaryStage.setScene(...)",[3607,7400,7401],{"x":4304,"y":4340,"fontSize":4204,"fill":3621},"① createInjector",[3607,7403,7405],{"x":4304,"y":7404,"fontSize":4204,"fill":3621},"134","② setControllerFactory",[3607,7407,7408],{"x":4304,"y":4349,"fontSize":4204,"fill":3621},"③ loader.load()",[3607,7410,7411],{"x":4304,"y":3689,"fontSize":4204,"fill":3621},"④ getInstance(ctrl)",[3607,7413,7415],{"x":4304,"y":7414,"fontSize":4204,"fill":3621},"176","⑤ @Inject вирішено",[3607,7417,7418],{"x":4304,"y":4268,"fontSize":4204,"fill":3621},"⑥ show()",[3514,7420,7422],{"id":7421},"повна-реалізація-app-guicemodule-контролер","Повна реалізація: App + GuiceModule + контролер",[3150,7424,7425],{},"Розглянемо повний, робочий приклад для нашої аудіоплатформи.",[4521,7427,7428,7808,8207,8554],{},[3159,7429,7431],{"className":3161,"code":7430,"filename":5699,"language":3163,"meta":3164,"style":3164},"package com.example.audiobook;\n\nimport com.google.inject.Guice;\nimport com.google.inject.Injector;\nimport javafx.application.Application;\nimport javafx.fxml.FXMLLoader;\nimport javafx.scene.Parent;\nimport javafx.scene.Scene;\nimport javafx.stage.Stage;\n\npublic class App extends Application {\n\n    \u002F\u002F Injector доступний у межах App — і нікуди далі\n    private Injector injector;\n\n    @Override\n    public void init() {\n        \u002F\u002F init() — правильне місце для Guice. Викликається до start(),\n        \u002F\u002F у окремому потоці JavaFX Launcher Thread, до JavaFX Application Thread.\n        injector = Guice.createInjector(new AppModule());\n    }\n\n    @Override\n    public void start(Stage primaryStage) throws Exception {\n        FXMLLoader loader = new FXMLLoader(\n            getClass().getResource(\"\u002Ffxml\u002Ftrack-list.fxml\")\n        );\n        \u002F\u002F Підключаємо Guice як фабрику контролерів\n        loader.setControllerFactory(injector::getInstance);\n\n        Parent root = loader.load();\n        primaryStage.setTitle(\"Аудіоплатформа\");\n        primaryStage.setScene(new Scene(root, 900, 600));\n        primaryStage.show();\n    }\n\n    public static void main(String[] args) {\n        launch(args);\n    }\n}\n",[3154,7432,7433,7439,7443,7449,7455,7462,7469,7476,7483,7490,7494,7510,7514,7519,7530,7534,7540,7551,7556,7561,7580,7584,7588,7594,7620,7635,7650,7654,7659,7675,7680,7698,7716,7746,7758,7763,7768,7789,7798,7803],{"__ignoreMap":3164},[3168,7434,7435,7437],{"class":3170,"line":3171},[3168,7436,4886],{"class":3174},[3168,7438,4852],{"class":3185},[3168,7440,7441],{"class":3170,"line":3189},[3168,7442,3193],{"emptyLinePlaceholder":3192},[3168,7444,7445,7447],{"class":3170,"line":3196},[3168,7446,4897],{"class":3174},[3168,7448,5459],{"class":3185},[3168,7450,7451,7453],{"class":3170,"line":3215},[3168,7452,4897],{"class":3174},[3168,7454,5466],{"class":3185},[3168,7456,7457,7459],{"class":3170,"line":3220},[3168,7458,4897],{"class":3174},[3168,7460,7461],{"class":3185}," javafx.application.Application;\n",[3168,7463,7464,7466],{"class":3170,"line":3232},[3168,7465,4897],{"class":3174},[3168,7467,7468],{"class":3185}," javafx.fxml.FXMLLoader;\n",[3168,7470,7471,7473],{"class":3170,"line":3254},[3168,7472,4897],{"class":3174},[3168,7474,7475],{"class":3185}," javafx.scene.Parent;\n",[3168,7477,7478,7480],{"class":3170,"line":3264},[3168,7479,4897],{"class":3174},[3168,7481,7482],{"class":3185}," javafx.scene.Scene;\n",[3168,7484,7485,7487],{"class":3170,"line":3276},[3168,7486,4897],{"class":3174},[3168,7488,7489],{"class":3185}," javafx.stage.Stage;\n",[3168,7491,7492],{"class":3170,"line":3282},[3168,7493,3193],{"emptyLinePlaceholder":3192},[3168,7495,7496,7498,7500,7503,7505,7508],{"class":3170,"line":3301},[3168,7497,3175],{"class":3174},[3168,7499,3178],{"class":3174},[3168,7501,7502],{"class":3181}," App",[3168,7504,4930],{"class":3174},[3168,7506,7507],{"class":3181}," Application",[3168,7509,3186],{"class":3185},[3168,7511,7512],{"class":3170,"line":3320},[3168,7513,3193],{"emptyLinePlaceholder":3192},[3168,7515,7516],{"class":3170,"line":3345},[3168,7517,7518],{"class":3943},"    \u002F\u002F Injector доступний у межах App — і нікуди далі\n",[3168,7520,7521,7523,7526,7528],{"class":3170,"line":3366},[3168,7522,3199],{"class":3174},[3168,7524,7525],{"class":3181}," Injector",[3168,7527,5518],{"class":3208},[3168,7529,3212],{"class":3185},[3168,7531,7532],{"class":3170,"line":3372},[3168,7533,3193],{"emptyLinePlaceholder":3192},[3168,7535,7536,7538],{"class":3170,"line":4447},[3168,7537,3830],{"class":3185},[3168,7539,4946],{"class":3181},[3168,7541,7542,7544,7546,7549],{"class":3170,"line":4453},[3168,7543,3223],{"class":3174},[3168,7545,4043],{"class":3181},[3168,7547,7548],{"class":3226}," init",[3168,7550,3229],{"class":3185},[3168,7552,7553],{"class":3170,"line":5129},[3168,7554,7555],{"class":3943},"        \u002F\u002F init() — правильне місце для Guice. Викликається до start(),\n",[3168,7557,7558],{"class":3170,"line":5134},[3168,7559,7560],{"class":3943},"        \u002F\u002F у окремому потоці JavaFX Launcher Thread, до JavaFX Application Thread.\n",[3168,7562,7563,7566,7568,7570,7572,7574,7576,7578],{"class":3170,"line":5357},[3168,7564,7565],{"class":3185},"        injector = ",[3168,7567,5523],{"class":3208},[3168,7569,3351],{"class":3185},[3168,7571,5528],{"class":3226},[3168,7573,3336],{"class":3185},[3168,7575,3245],{"class":3244},[3168,7577,4927],{"class":3226},[3168,7579,5537],{"class":3185},[3168,7581,7582],{"class":3170,"line":5370},[3168,7583,3369],{"class":3185},[3168,7585,7586],{"class":3170,"line":5375},[3168,7587,3193],{"emptyLinePlaceholder":3192},[3168,7589,7590,7592],{"class":3170,"line":5380},[3168,7591,3830],{"class":3185},[3168,7593,4946],{"class":3181},[3168,7595,7596,7598,7600,7603,7605,7608,7611,7613,7615,7618],{"class":3170,"line":5402},[3168,7597,3223],{"class":3174},[3168,7599,4043],{"class":3181},[3168,7601,7602],{"class":3226}," start",[3168,7604,3336],{"class":3185},[3168,7606,7607],{"class":3181},"Stage",[3168,7609,7610],{"class":3208}," primaryStage",[3168,7612,5320],{"class":3185},[3168,7614,5394],{"class":3174},[3168,7616,7617],{"class":3181}," Exception",[3168,7619,3186],{"class":3185},[3168,7621,7622,7625,7627,7629,7631,7633],{"class":3170,"line":5419},[3168,7623,7624],{"class":3181},"        FXMLLoader",[3168,7626,7179],{"class":3208},[3168,7628,3241],{"class":3185},[3168,7630,3245],{"class":3244},[3168,7632,7186],{"class":3226},[3168,7634,3251],{"class":3185},[3168,7636,7637,7640,7642,7644,7646,7648],{"class":3170,"line":5424},[3168,7638,7639],{"class":3226},"            getClass",[3168,7641,5026],{"class":3185},[3168,7643,7196],{"class":3226},[3168,7645,3336],{"class":3185},[3168,7647,7201],{"class":3257},[3168,7649,4036],{"class":3185},[3168,7651,7652],{"class":3170,"line":6183},[3168,7653,3279],{"class":3185},[3168,7655,7656],{"class":3170,"line":3653},[3168,7657,7658],{"class":3943},"        \u002F\u002F Підключаємо Guice як фабрику контролерів\n",[3168,7660,7662,7665,7667,7669,7671,7673],{"class":3170,"line":7661},29,[3168,7663,7664],{"class":3208},"        loader",[3168,7666,3351],{"class":3185},[3168,7668,7228],{"class":3226},[3168,7670,7231],{"class":3185},[3168,7672,5601],{"class":3244},[3168,7674,7236],{"class":3185},[3168,7676,7678],{"class":3170,"line":7677},30,[3168,7679,3193],{"emptyLinePlaceholder":3192},[3168,7681,7683,7686,7688,7690,7692,7694,7696],{"class":3170,"line":7682},31,[3168,7684,7685],{"class":3181},"        Parent",[3168,7687,7248],{"class":3208},[3168,7689,3241],{"class":3185},[3168,7691,7223],{"class":3208},[3168,7693,3351],{"class":3185},[3168,7695,7257],{"class":3226},[3168,7697,3317],{"class":3185},[3168,7699,7701,7704,7706,7709,7711,7714],{"class":3170,"line":7700},32,[3168,7702,7703],{"class":3208},"        primaryStage",[3168,7705,3351],{"class":3185},[3168,7707,7708],{"class":3226},"setTitle",[3168,7710,3336],{"class":3185},[3168,7712,7713],{"class":3257},"\"Аудіоплатформа\"",[3168,7715,3342],{"class":3185},[3168,7717,7719,7721,7723,7726,7728,7730,7733,7736,7739,7741,7744],{"class":3170,"line":7718},33,[3168,7720,7703],{"class":3208},[3168,7722,3351],{"class":3185},[3168,7724,7725],{"class":3226},"setScene",[3168,7727,3336],{"class":3185},[3168,7729,3245],{"class":3244},[3168,7731,7732],{"class":3226}," Scene",[3168,7734,7735],{"class":3185},"(root, ",[3168,7737,7738],{"class":6148},"900",[3168,7740,3270],{"class":3185},[3168,7742,7743],{"class":6148},"600",[3168,7745,7204],{"class":3185},[3168,7747,7749,7751,7753,7756],{"class":3170,"line":7748},34,[3168,7750,7703],{"class":3208},[3168,7752,3351],{"class":3185},[3168,7754,7755],{"class":3226},"show",[3168,7757,3317],{"class":3185},[3168,7759,7761],{"class":3170,"line":7760},35,[3168,7762,3369],{"class":3185},[3168,7764,7766],{"class":3170,"line":7765},36,[3168,7767,3193],{"emptyLinePlaceholder":3192},[3168,7769,7771,7773,7775,7777,7779,7781,7783,7785,7787],{"class":3170,"line":7770},37,[3168,7772,3223],{"class":3174},[3168,7774,5488],{"class":3174},[3168,7776,4043],{"class":3181},[3168,7778,5493],{"class":3226},[3168,7780,3336],{"class":3185},[3168,7782,5284],{"class":3181},[3168,7784,5500],{"class":3185},[3168,7786,5503],{"class":3208},[3168,7788,3854],{"class":3185},[3168,7790,7792,7795],{"class":3170,"line":7791},38,[3168,7793,7794],{"class":3226},"        launch",[3168,7796,7797],{"class":3185},"(args);\n",[3168,7799,7801],{"class":3170,"line":7800},39,[3168,7802,3369],{"class":3185},[3168,7804,7806],{"class":3170,"line":7805},40,[3168,7807,3375],{"class":3185},[3159,7809,7812],{"className":3161,"code":7810,"filename":7811,"language":3163,"meta":3164,"style":3164},"package com.example.audiobook.controller;\n\nimport com.example.audiobook.service.TrackService;\nimport com.example.audiobook.service.NavigationService;\nimport com.google.inject.Inject;\nimport javafx.fxml.FXML;\nimport javafx.fxml.Initializable;\nimport javafx.scene.control.ListView;\n\nimport java.net.URL;\nimport java.util.ResourceBundle;\n\npublic class TrackListController implements Initializable {\n\n    \u002F\u002F Залежності впроваджуються через конструктор — не через поля!\n    private final TrackService trackService;\n    private final NavigationService navigationService;\n\n    @FXML\n    private ListView\u003CString> trackListView;\n\n    @Inject\n    public TrackListController(\n        TrackService trackService,\n        NavigationService navigationService\n    ) {\n        this.trackService     = trackService;\n        this.navigationService = navigationService;\n    }\n\n    @Override\n    public void initialize(URL location, ResourceBundle resources) {\n        \u002F\u002F trackService вже ін'єктований — NullPointerException неможливий\n        trackService.listAll()\n            .stream()\n            .map(t -> t.getTitle() + \" — \" + t.getArtist())\n            .forEach(trackListView.getItems()::add);\n    }\n\n    @FXML\n    private void onAddTrack() {\n        navigationService.openAddTrackDialog();\n    }\n}\n","TrackListController.java",[3154,7813,7814,7821,7825,7832,7839,7845,7852,7859,7866,7870,7877,7884,7888,7903,7907,7912,7924,7938,7942,7949,7968,7972,7978,7986,7994,8002,8006,8017,8029,8033,8037,8043,8070,8075,8086,8095,8135,8158,8162,8166,8172,8184,8197,8202],{"__ignoreMap":3164},[3168,7815,7816,7818],{"class":3170,"line":3171},[3168,7817,4886],{"class":3174},[3168,7819,7820],{"class":3185}," com.example.audiobook.controller;\n",[3168,7822,7823],{"class":3170,"line":3189},[3168,7824,3193],{"emptyLinePlaceholder":3192},[3168,7826,7827,7829],{"class":3170,"line":3196},[3168,7828,4897],{"class":3174},[3168,7830,7831],{"class":3185}," com.example.audiobook.service.TrackService;\n",[3168,7833,7834,7836],{"class":3170,"line":3215},[3168,7835,4897],{"class":3174},[3168,7837,7838],{"class":3185}," com.example.audiobook.service.NavigationService;\n",[3168,7840,7841,7843],{"class":3170,"line":3220},[3168,7842,4897],{"class":3174},[3168,7844,5168],{"class":3185},[3168,7846,7847,7849],{"class":3170,"line":3232},[3168,7848,4897],{"class":3174},[3168,7850,7851],{"class":3185}," javafx.fxml.FXML;\n",[3168,7853,7854,7856],{"class":3170,"line":3254},[3168,7855,4897],{"class":3174},[3168,7857,7858],{"class":3185}," javafx.fxml.Initializable;\n",[3168,7860,7861,7863],{"class":3170,"line":3264},[3168,7862,4897],{"class":3174},[3168,7864,7865],{"class":3185}," javafx.scene.control.ListView;\n",[3168,7867,7868],{"class":3170,"line":3276},[3168,7869,3193],{"emptyLinePlaceholder":3192},[3168,7871,7872,7874],{"class":3170,"line":3282},[3168,7873,4897],{"class":3174},[3168,7875,7876],{"class":3185}," java.net.URL;\n",[3168,7878,7879,7881],{"class":3170,"line":3301},[3168,7880,4897],{"class":3174},[3168,7882,7883],{"class":3185}," java.util.ResourceBundle;\n",[3168,7885,7886],{"class":3170,"line":3320},[3168,7887,3193],{"emptyLinePlaceholder":3192},[3168,7889,7890,7892,7894,7896,7898,7901],{"class":3170,"line":3345},[3168,7891,3175],{"class":3174},[3168,7893,3178],{"class":3174},[3168,7895,3182],{"class":3181},[3168,7897,5948],{"class":3174},[3168,7899,7900],{"class":3181}," Initializable",[3168,7902,3186],{"class":3185},[3168,7904,7905],{"class":3170,"line":3366},[3168,7906,3193],{"emptyLinePlaceholder":3192},[3168,7908,7909],{"class":3170,"line":3372},[3168,7910,7911],{"class":3943},"    \u002F\u002F Залежності впроваджуються через конструктор — не через поля!\n",[3168,7913,7914,7916,7918,7920,7922],{"class":3170,"line":4447},[3168,7915,3199],{"class":3174},[3168,7917,3202],{"class":3174},[3168,7919,3205],{"class":3181},[3168,7921,3209],{"class":3208},[3168,7923,3212],{"class":3185},[3168,7925,7926,7928,7930,7933,7936],{"class":3170,"line":4453},[3168,7927,3199],{"class":3174},[3168,7929,3202],{"class":3174},[3168,7931,7932],{"class":3181}," NavigationService",[3168,7934,7935],{"class":3208}," navigationService",[3168,7937,3212],{"class":3185},[3168,7939,7940],{"class":3170,"line":5129},[3168,7941,3193],{"emptyLinePlaceholder":3192},[3168,7943,7944,7946],{"class":3170,"line":5134},[3168,7945,3830],{"class":3185},[3168,7947,7948],{"class":3181},"FXML\n",[3168,7950,7951,7953,7956,7958,7960,7963,7966],{"class":3170,"line":5357},[3168,7952,3199],{"class":3174},[3168,7954,7955],{"class":3181}," ListView",[3168,7957,4536],{"class":3185},[3168,7959,5284],{"class":3181},[3168,7961,7962],{"class":3185},"> ",[3168,7964,7965],{"class":3208},"trackListView",[3168,7967,3212],{"class":3185},[3168,7969,7970],{"class":3170,"line":5370},[3168,7971,3193],{"emptyLinePlaceholder":3192},[3168,7973,7974,7976],{"class":3170,"line":5375},[3168,7975,3830],{"class":3185},[3168,7977,3833],{"class":3181},[3168,7979,7980,7982,7984],{"class":3170,"line":5380},[3168,7981,3223],{"class":3174},[3168,7983,3182],{"class":3226},[3168,7985,3251],{"class":3185},[3168,7987,7988,7990,7992],{"class":3170,"line":5402},[3168,7989,5551],{"class":3181},[3168,7991,3209],{"class":3208},[3168,7993,3261],{"class":3185},[3168,7995,7996,7999],{"class":3170,"line":5419},[3168,7997,7998],{"class":3181},"        NavigationService",[3168,8000,8001],{"class":3208}," navigationService\n",[3168,8003,8004],{"class":3170,"line":5424},[3168,8005,5330],{"class":3185},[3168,8007,8008,8010,8012,8014],{"class":3170,"line":6183},[3168,8009,3348],{"class":3174},[3168,8011,3351],{"class":3185},[3168,8013,3354],{"class":3208},[3168,8015,8016],{"class":3185},"     = trackService;\n",[3168,8018,8019,8021,8023,8026],{"class":3170,"line":3653},[3168,8020,3348],{"class":3174},[3168,8022,3351],{"class":3185},[3168,8024,8025],{"class":3208},"navigationService",[3168,8027,8028],{"class":3185}," = navigationService;\n",[3168,8030,8031],{"class":3170,"line":7661},[3168,8032,3369],{"class":3185},[3168,8034,8035],{"class":3170,"line":7677},[3168,8036,3193],{"emptyLinePlaceholder":3192},[3168,8038,8039,8041],{"class":3170,"line":7682},[3168,8040,3830],{"class":3185},[3168,8042,4946],{"class":3181},[3168,8044,8045,8047,8049,8052,8054,8057,8060,8062,8065,8068],{"class":3170,"line":7700},[3168,8046,3223],{"class":3174},[3168,8048,4043],{"class":3181},[3168,8050,8051],{"class":3226}," initialize",[3168,8053,3336],{"class":3185},[3168,8055,8056],{"class":3181},"URL",[3168,8058,8059],{"class":3208}," location",[3168,8061,3270],{"class":3185},[3168,8063,8064],{"class":3181},"ResourceBundle",[3168,8066,8067],{"class":3208}," resources",[3168,8069,3854],{"class":3185},[3168,8071,8072],{"class":3170,"line":7718},[3168,8073,8074],{"class":3943},"        \u002F\u002F trackService вже ін'єктований — NullPointerException неможливий\n",[3168,8076,8077,8079,8081,8083],{"class":3170,"line":7748},[3168,8078,5578],{"class":3208},[3168,8080,3351],{"class":3185},[3168,8082,5583],{"class":3226},[3168,8084,8085],{"class":3185},"()\n",[3168,8087,8088,8090,8093],{"class":3170,"line":7760},[3168,8089,5052],{"class":3185},[3168,8091,8092],{"class":3226},"stream",[3168,8094,8085],{"class":3185},[3168,8096,8097,8099,8102,8105,8108,8111,8113,8116,8119,8122,8124,8127,8129,8132],{"class":3170,"line":7765},[3168,8098,5052],{"class":3185},[3168,8100,8101],{"class":3226},"map",[3168,8103,8104],{"class":3185},"(t ",[3168,8106,8107],{"class":3174},"->",[3168,8109,8110],{"class":3208}," t",[3168,8112,3351],{"class":3185},[3168,8114,8115],{"class":3226},"getTitle",[3168,8117,8118],{"class":3185},"() + ",[3168,8120,8121],{"class":3257},"\" — \"",[3168,8123,7036],{"class":3185},[3168,8125,8126],{"class":3208},"t",[3168,8128,3351],{"class":3185},[3168,8130,8131],{"class":3226},"getArtist",[3168,8133,8134],{"class":3185},"())\n",[3168,8136,8137,8139,8141,8143,8145,8147,8150,8153,8155],{"class":3170,"line":7770},[3168,8138,5052],{"class":3185},[3168,8140,5588],{"class":3226},[3168,8142,3336],{"class":3185},[3168,8144,7965],{"class":3208},[3168,8146,3351],{"class":3185},[3168,8148,8149],{"class":3226},"getItems",[3168,8151,8152],{"class":3185},"()",[3168,8154,5601],{"class":3244},[3168,8156,8157],{"class":3185},"add);\n",[3168,8159,8160],{"class":3170,"line":7791},[3168,8161,3369],{"class":3185},[3168,8163,8164],{"class":3170,"line":7800},[3168,8165,3193],{"emptyLinePlaceholder":3192},[3168,8167,8168,8170],{"class":3170,"line":7805},[3168,8169,3830],{"class":3185},[3168,8171,7948],{"class":3181},[3168,8173,8175,8177,8179,8182],{"class":3170,"line":8174},41,[3168,8176,3199],{"class":3174},[3168,8178,4043],{"class":3181},[3168,8180,8181],{"class":3226}," onAddTrack",[3168,8183,3229],{"class":3185},[3168,8185,8187,8190,8192,8195],{"class":3170,"line":8186},42,[3168,8188,8189],{"class":3208},"        navigationService",[3168,8191,3351],{"class":3185},[3168,8193,8194],{"class":3226},"openAddTrackDialog",[3168,8196,3317],{"class":3185},[3168,8198,8200],{"class":3170,"line":8199},43,[3168,8201,3369],{"class":3185},[3168,8203,8205],{"class":3170,"line":8204},44,[3168,8206,3375],{"class":3185},[3159,8208,8211],{"className":3161,"code":8209,"filename":8210,"language":3163,"meta":3164,"style":3164},"package com.example.audiobook.service;\n\nimport com.google.inject.Inject;\nimport com.google.inject.Injector;\nimport com.google.inject.Singleton;\nimport javafx.fxml.FXMLLoader;\nimport javafx.scene.Parent;\nimport javafx.scene.Scene;\nimport javafx.stage.Stage;\n\n@Singleton\npublic class NavigationService {\n\n    \u002F\u002F NavigationService сам отримує Injector для створення нових сцен\n    \u002F\u002F Це ЄДИНИЙ клас у проєкті, що може зберігати Injector\n    private final Injector injector;\n\n    @Inject\n    public NavigationService(Injector injector) {\n        this.injector = injector;\n    }\n\n    public void openAddTrackDialog() {\n        try {\n            FXMLLoader loader = new FXMLLoader(\n                getClass().getResource(\"\u002Ffxml\u002Fadd-track.fxml\")\n            );\n            loader.setControllerFactory(injector::getInstance);\n\n            Stage dialog = new Stage();\n            dialog.setScene(new Scene(loader.load(), 600, 400));\n            dialog.setTitle(\"Додати трек\");\n            dialog.show();\n        } catch (Exception e) {\n            throw new RuntimeException(\"Неможливо відкрити діалог\", e);\n        }\n    }\n}\n","NavigationService.java",[3154,8212,8213,8220,8224,8230,8236,8242,8248,8254,8260,8266,8270,8276,8286,8290,8295,8300,8312,8316,8322,8336,8347,8351,8355,8366,8373,8388,8404,8409,8424,8428,8445,8480,8495,8505,8523,8541,8546,8550],{"__ignoreMap":3164},[3168,8214,8215,8217],{"class":3170,"line":3171},[3168,8216,4886],{"class":3174},[3168,8218,8219],{"class":3185}," com.example.audiobook.service;\n",[3168,8221,8222],{"class":3170,"line":3189},[3168,8223,3193],{"emptyLinePlaceholder":3192},[3168,8225,8226,8228],{"class":3170,"line":3196},[3168,8227,4897],{"class":3174},[3168,8229,5168],{"class":3185},[3168,8231,8232,8234],{"class":3170,"line":3215},[3168,8233,4897],{"class":3174},[3168,8235,5466],{"class":3185},[3168,8237,8238,8240],{"class":3170,"line":3220},[3168,8239,4897],{"class":3174},[3168,8241,5175],{"class":3185},[3168,8243,8244,8246],{"class":3170,"line":3232},[3168,8245,4897],{"class":3174},[3168,8247,7468],{"class":3185},[3168,8249,8250,8252],{"class":3170,"line":3254},[3168,8251,4897],{"class":3174},[3168,8253,7475],{"class":3185},[3168,8255,8256,8258],{"class":3170,"line":3264},[3168,8257,4897],{"class":3174},[3168,8259,7482],{"class":3185},[3168,8261,8262,8264],{"class":3170,"line":3276},[3168,8263,4897],{"class":3174},[3168,8265,7489],{"class":3185},[3168,8267,8268],{"class":3170,"line":3282},[3168,8269,3193],{"emptyLinePlaceholder":3192},[3168,8271,8272,8274],{"class":3170,"line":3301},[3168,8273,5191],{"class":3185},[3168,8275,5194],{"class":3181},[3168,8277,8278,8280,8282,8284],{"class":3170,"line":3320},[3168,8279,3175],{"class":3174},[3168,8281,3178],{"class":3174},[3168,8283,7932],{"class":3181},[3168,8285,3186],{"class":3185},[3168,8287,8288],{"class":3170,"line":3345},[3168,8289,3193],{"emptyLinePlaceholder":3192},[3168,8291,8292],{"class":3170,"line":3366},[3168,8293,8294],{"class":3943},"    \u002F\u002F NavigationService сам отримує Injector для створення нових сцен\n",[3168,8296,8297],{"class":3170,"line":3372},[3168,8298,8299],{"class":3943},"    \u002F\u002F Це ЄДИНИЙ клас у проєкті, що може зберігати Injector\n",[3168,8301,8302,8304,8306,8308,8310],{"class":3170,"line":4447},[3168,8303,3199],{"class":3174},[3168,8305,3202],{"class":3174},[3168,8307,7525],{"class":3181},[3168,8309,5518],{"class":3208},[3168,8311,3212],{"class":3185},[3168,8313,8314],{"class":3170,"line":4453},[3168,8315,3193],{"emptyLinePlaceholder":3192},[3168,8317,8318,8320],{"class":3170,"line":5129},[3168,8319,3830],{"class":3185},[3168,8321,3833],{"class":3181},[3168,8323,8324,8326,8328,8330,8332,8334],{"class":3170,"line":5134},[3168,8325,3223],{"class":3174},[3168,8327,7932],{"class":3226},[3168,8329,3336],{"class":3185},[3168,8331,4137],{"class":3181},[3168,8333,5518],{"class":3208},[3168,8335,3854],{"class":3185},[3168,8337,8338,8340,8342,8344],{"class":3170,"line":5357},[3168,8339,3348],{"class":3174},[3168,8341,3351],{"class":3185},[3168,8343,5558],{"class":3208},[3168,8345,8346],{"class":3185}," = injector;\n",[3168,8348,8349],{"class":3170,"line":5370},[3168,8350,3369],{"class":3185},[3168,8352,8353],{"class":3170,"line":5375},[3168,8354,3193],{"emptyLinePlaceholder":3192},[3168,8356,8357,8359,8361,8364],{"class":3170,"line":5380},[3168,8358,3223],{"class":3174},[3168,8360,4043],{"class":3181},[3168,8362,8363],{"class":3226}," openAddTrackDialog",[3168,8365,3229],{"class":3185},[3168,8367,8368,8371],{"class":3170,"line":5402},[3168,8369,8370],{"class":3244},"        try",[3168,8372,3186],{"class":3185},[3168,8374,8375,8378,8380,8382,8384,8386],{"class":3170,"line":5419},[3168,8376,8377],{"class":3181},"            FXMLLoader",[3168,8379,7179],{"class":3208},[3168,8381,3241],{"class":3185},[3168,8383,3245],{"class":3244},[3168,8385,7186],{"class":3226},[3168,8387,3251],{"class":3185},[3168,8389,8390,8393,8395,8397,8399,8402],{"class":3170,"line":5424},[3168,8391,8392],{"class":3226},"                getClass",[3168,8394,5026],{"class":3185},[3168,8396,7196],{"class":3226},[3168,8398,3336],{"class":3185},[3168,8400,8401],{"class":3257},"\"\u002Ffxml\u002Fadd-track.fxml\"",[3168,8403,4036],{"class":3185},[3168,8405,8406],{"class":3170,"line":6183},[3168,8407,8408],{"class":3185},"            );\n",[3168,8410,8411,8414,8416,8418,8420,8422],{"class":3170,"line":3653},[3168,8412,8413],{"class":3208},"            loader",[3168,8415,3351],{"class":3185},[3168,8417,7228],{"class":3226},[3168,8419,7231],{"class":3185},[3168,8421,5601],{"class":3244},[3168,8423,7236],{"class":3185},[3168,8425,8426],{"class":3170,"line":7661},[3168,8427,3193],{"emptyLinePlaceholder":3192},[3168,8429,8430,8433,8436,8438,8440,8443],{"class":3170,"line":7677},[3168,8431,8432],{"class":3181},"            Stage",[3168,8434,8435],{"class":3208}," dialog",[3168,8437,3241],{"class":3185},[3168,8439,3245],{"class":3244},[3168,8441,8442],{"class":3226}," Stage",[3168,8444,3317],{"class":3185},[3168,8446,8447,8450,8452,8454,8456,8458,8460,8462,8464,8466,8468,8471,8473,8475,8478],{"class":3170,"line":7682},[3168,8448,8449],{"class":3208},"            dialog",[3168,8451,3351],{"class":3185},[3168,8453,7725],{"class":3226},[3168,8455,3336],{"class":3185},[3168,8457,3245],{"class":3244},[3168,8459,7732],{"class":3226},[3168,8461,3336],{"class":3185},[3168,8463,7223],{"class":3208},[3168,8465,3351],{"class":3185},[3168,8467,7257],{"class":3226},[3168,8469,8470],{"class":3185},"(), ",[3168,8472,7743],{"class":6148},[3168,8474,3270],{"class":3185},[3168,8476,8477],{"class":6148},"400",[3168,8479,7204],{"class":3185},[3168,8481,8482,8484,8486,8488,8490,8493],{"class":3170,"line":7700},[3168,8483,8449],{"class":3208},[3168,8485,3351],{"class":3185},[3168,8487,7708],{"class":3226},[3168,8489,3336],{"class":3185},[3168,8491,8492],{"class":3257},"\"Додати трек\"",[3168,8494,3342],{"class":3185},[3168,8496,8497,8499,8501,8503],{"class":3170,"line":7718},[3168,8498,8449],{"class":3208},[3168,8500,3351],{"class":3185},[3168,8502,7755],{"class":3226},[3168,8504,3317],{"class":3185},[3168,8506,8507,8510,8513,8515,8518,8521],{"class":3170,"line":7748},[3168,8508,8509],{"class":3185},"        } ",[3168,8511,8512],{"class":3244},"catch",[3168,8514,6440],{"class":3185},[3168,8516,8517],{"class":3181},"Exception",[3168,8519,8520],{"class":3208}," e",[3168,8522,3854],{"class":3185},[3168,8524,8525,8528,8530,8533,8535,8538],{"class":3170,"line":7760},[3168,8526,8527],{"class":3244},"            throw",[3168,8529,6157],{"class":3244},[3168,8531,8532],{"class":3226}," RuntimeException",[3168,8534,3336],{"class":3185},[3168,8536,8537],{"class":3257},"\"Неможливо відкрити діалог\"",[3168,8539,8540],{"class":3185},", e);\n",[3168,8542,8543],{"class":3170,"line":7765},[3168,8544,8545],{"class":3185},"        }\n",[3168,8547,8548],{"class":3170,"line":7770},[3168,8549,3369],{"class":3185},[3168,8551,8552],{"class":3170,"line":7791},[3168,8553,3375],{"class":3185},[3159,8555,8558],{"className":4525,"code":8556,"filename":8557,"language":4528,"meta":3164,"style":3164},"\u003C?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\n\u003C?import javafx.scene.control.*?>\n\u003C?import javafx.scene.layout.*?>\n\n\u003C!-- fx:controller вказує клас контролера.\n     FXMLLoader передасть його у setControllerFactory → Guice -->\n\u003CBorderPane\n    xmlns=\"http:\u002F\u002Fjavafx.com\u002Fjavafx\"\n    xmlns:fx=\"http:\u002F\u002Fjavafx.com\u002Ffxml\"\n    fx:controller=\"com.example.audiobook.controller.TrackListController\">\n\n    \u003Ccenter>\n        \u003CListView fx:id=\"trackListView\" \u002F>\n    \u003C\u002Fcenter>\n\n    \u003Cbottom>\n        \u003CButton text=\"Додати трек\" onAction=\"#onAddTrack\" \u002F>\n    \u003C\u002Fbottom>\n\u003C\u002FBorderPane>\n","track-list.fxml",[3154,8559,8560,8589,8593,8607,8620,8624,8629,8634,8641,8651,8661,8673,8677,8686,8704,8712,8716,8725,8749,8757],{"__ignoreMap":3164},[3168,8561,8562,8565,8567,8571,8574,8578,8581,8583,8586],{"class":3170,"line":3171},[3168,8563,8564],{"class":4535},"\u003C?",[3168,8566,4528],{"class":4539},[3168,8568,8570],{"class":8569},"sa4r_"," version",[3168,8572,8573],{"class":3185},"=",[3168,8575,8577],{"class":8576},"su9tN","\"1.0\"",[3168,8579,8580],{"class":8569}," encoding",[3168,8582,8573],{"class":3185},[3168,8584,8585],{"class":8576},"\"UTF-8\"",[3168,8587,8588],{"class":4535},"?>\n",[3168,8590,8591],{"class":3170,"line":3189},[3168,8592,3193],{"emptyLinePlaceholder":3192},[3168,8594,8595,8597,8599,8602,8605],{"class":3170,"line":3196},[3168,8596,8564],{"class":4535},[3168,8598,4897],{"class":4539},[3168,8600,8601],{"class":8569}," javafx",[3168,8603,8604],{"class":3185},".scene.control.*",[3168,8606,8588],{"class":4535},[3168,8608,8609,8611,8613,8615,8618],{"class":3170,"line":3215},[3168,8610,8564],{"class":4535},[3168,8612,4897],{"class":4539},[3168,8614,8601],{"class":8569},[3168,8616,8617],{"class":3185},".scene.layout.*",[3168,8619,8588],{"class":4535},[3168,8621,8622],{"class":3170,"line":3220},[3168,8623,3193],{"emptyLinePlaceholder":3192},[3168,8625,8626],{"class":3170,"line":3232},[3168,8627,8628],{"class":3943},"\u003C!-- fx:controller вказує клас контролера.\n",[3168,8630,8631],{"class":3170,"line":3254},[3168,8632,8633],{"class":3943},"     FXMLLoader передасть його у setControllerFactory → Guice -->\n",[3168,8635,8636,8638],{"class":3170,"line":3264},[3168,8637,4536],{"class":4535},[3168,8639,8640],{"class":4539},"BorderPane\n",[3168,8642,8643,8646,8648],{"class":3170,"line":3276},[3168,8644,8645],{"class":8569},"    xmlns",[3168,8647,8573],{"class":3185},[3168,8649,8650],{"class":8576},"\"http:\u002F\u002Fjavafx.com\u002Fjavafx\"\n",[3168,8652,8653,8656,8658],{"class":3170,"line":3282},[3168,8654,8655],{"class":8569},"    xmlns:fx",[3168,8657,8573],{"class":3185},[3168,8659,8660],{"class":8576},"\"http:\u002F\u002Fjavafx.com\u002Ffxml\"\n",[3168,8662,8663,8666,8668,8671],{"class":3170,"line":3301},[3168,8664,8665],{"class":8569},"    fx:controller",[3168,8667,8573],{"class":3185},[3168,8669,8670],{"class":8576},"\"com.example.audiobook.controller.TrackListController\"",[3168,8672,4543],{"class":4535},[3168,8674,8675],{"class":3170,"line":3320},[3168,8676,3193],{"emptyLinePlaceholder":3192},[3168,8678,8679,8681,8684],{"class":3170,"line":3345},[3168,8680,4553],{"class":4535},[3168,8682,8683],{"class":4539},"center",[3168,8685,4543],{"class":4535},[3168,8687,8688,8690,8693,8696,8698,8701],{"class":3170,"line":3366},[3168,8689,4563],{"class":4535},[3168,8691,8692],{"class":4539},"ListView",[3168,8694,8695],{"class":8569}," fx:id",[3168,8697,8573],{"class":3185},[3168,8699,8700],{"class":8576},"\"trackListView\"",[3168,8702,8703],{"class":4535}," \u002F>\n",[3168,8705,8706,8708,8710],{"class":3170,"line":3372},[3168,8707,4618],{"class":4535},[3168,8709,8683],{"class":4539},[3168,8711,4543],{"class":4535},[3168,8713,8714],{"class":3170,"line":4447},[3168,8715,3193],{"emptyLinePlaceholder":3192},[3168,8717,8718,8720,8723],{"class":3170,"line":4453},[3168,8719,4553],{"class":4535},[3168,8721,8722],{"class":4539},"bottom",[3168,8724,4543],{"class":4535},[3168,8726,8727,8729,8732,8735,8737,8739,8742,8744,8747],{"class":3170,"line":5129},[3168,8728,4563],{"class":4535},[3168,8730,8731],{"class":4539},"Button",[3168,8733,8734],{"class":8569}," text",[3168,8736,8573],{"class":3185},[3168,8738,8492],{"class":8576},[3168,8740,8741],{"class":8569}," onAction",[3168,8743,8573],{"class":3185},[3168,8745,8746],{"class":8576},"\"#onAddTrack\"",[3168,8748,8703],{"class":4535},[3168,8750,8751,8753,8755],{"class":3170,"line":5134},[3168,8752,4618],{"class":4535},[3168,8754,8722],{"class":4539},[3168,8756,4543],{"class":4535},[3168,8758,8759,8761,8764],{"class":3170,"line":5357},[3168,8760,4574],{"class":4535},[3168,8762,8763],{"class":4539},"BorderPane",[3168,8765,4543],{"class":4535},[3150,8767,8768],{},"Розберемо ключові архітектурні рішення у прикладі вище.",[3150,8770,8771,8780,8781,8783,8784,8787,8788,8790],{},[3380,8772,8773,4182,8776,8779],{},[3154,8774,8775],{},"init()",[3154,8777,8778],{},"start()",":"," метод ",[3154,8782,8775],{}," класу ",[3154,8785,8786],{},"Application"," викликається до відображення будь-якого UI і є правильним місцем для ініціалізації важких ресурсів (бази даних, пулів, Guice-контейнера). Якщо ініціалізація ",[3154,8789,4137],{}," займає час (наприклад, перевіряє з'єднання з БД), вона не блокує JavaFX Application Thread.",[3150,8792,8793,8801,8802,8804,8805,8807,8808,8811],{},[3380,8794,8795,8798,8799,8779],{},[3154,8796,8797],{},"NavigationService"," з ",[3154,8800,4137],{}," це єдиний виправданий випадок, коли клас зберігає посилання на ",[3154,8803,4137],{},". ",[3154,8806,8797],{}," виступає централізованим сервісом навігації — він відкриває нові вікна\u002Fдіалоги, завантажуючи FXML і передаючи ",[3154,8809,8810],{},"injector::getInstance"," як фабрику. Всі інші класи отримують залежності виключно через конструктор.",[3150,8813,8814,8817,8818,8820,8821,8823,8824,8827,8828,8830,8831,8833,8834,8836,8837,4975],{},[3380,8815,8816],{},"Constructor Injection у контролері:"," незважаючи на те, що ",[3154,8819,7075],{}," традиційно очікує конструктор без параметрів, після підключення ",[3154,8822,7228],{}," Guice передає повністю сформований об'єкт. Поля ",[3154,8825,8826],{},"@FXML"," заповнює сам ",[3154,8829,7075],{}," після отримання контролера від Guice, тому ",[3154,8832,3156],{}," отримує і залежності (",[3154,8835,3892],{},"), і UI-компоненти (",[3154,8838,8826],{},[5683,8840,8841,8842,3270,8844,8847,8848,8850,8851,8853,8854,3351],{},"Не впроваджуйте ",[3154,8843,7607],{},[3154,8845,8846],{},"Scene"," або ",[3154,8849,7245],{}," через Guice ",[3154,8852,3892],{},". Ці об'єкти мають бути створені у JavaFX Application Thread, але Guice може будувати граф залежностей у будь-якому потоці. Передавайте UI-об'єкти явно через параметри методів або через ",[3154,8855,8797],{},[8857,8858,8860,8872,8882,8891,8898,8905,8912,8919,8928,8936,8943],"terminal-preview",{"title":8859},"App startup — Guice initialization log",[8861,8862,8864,3427,8869],"div",{"className":8863},[3170],[3168,8865,8868],{"className":8866},[8867],"opacity-40","$",[3380,8870,8871],{},"mvn javafx:run",[8861,8873,8875,8881],{"className":8874},[3170],[3168,8876,8880],{"className":8877},[8878,8879],"text-blue-400","font-bold","[INFO]"," Building audiobook-platform 1.0-SNAPSHOT",[8861,8883,8885,8890],{"className":8884},[3170],[3168,8886,8889],{"className":8887},[8888,8879],"text-green-400","[Guice]"," Stage 1: Module configuration started",[8861,8892,8894,8897],{"className":8893},[3170],[3168,8895,8889],{"className":8896},[8888],"   Binding: TrackRepository → JdbcTrackRepository",[8861,8899,8901,8904],{"className":8900},[3170],[3168,8902,8889],{"className":8903},[8888],"   Binding: AuthorRepository → JdbcAuthorRepository",[8861,8906,8908,8911],{"className":8907},[3170],[3168,8909,8889],{"className":8910},[8888],"   Binding: @Named(\"dbUrl\") → \"jdbc:h2:.\u002Fdata\u002Faudiobook...\"",[8861,8913,8915,8918],{"className":8914},[3170],[3168,8916,8889],{"className":8917},[8888,8879]," Stage 2: Injector created (12 bindings registered)",[8861,8920,8922,8927],{"className":8921},[3170],[3168,8923,8926],{"className":8924},[8925],"text-yellow-400","[H2]","   Connection established: jdbc:h2:.\u002Fdata\u002Faudiobook",[8861,8929,8931,8935],{"className":8930},[3170],[3168,8932,8934],{"className":8933},[8878],"[JavaFX]"," Loading FXML: \u002Ffxml\u002Ftrack-list.fxml",[8861,8937,8939,8942],{"className":8938},[3170],[3168,8940,8889],{"className":8941},[8888],"   getInstance(TrackListController) — resolved in 3ms",[8861,8944,8946,8949],{"className":8945},[3170],[3168,8947,8934],{"className":8948},[8878,8879]," Stage shown — application ready ✓",[3507,8951],{},[3145,8953,8955],{"id":8954},"скопи-scopes-та-час-життя-обєктів","Скопи (Scopes) та час життя об'єктів",[3150,8957,8958,8959,8962,8963,8966],{},"Одне з найважливіших рішень при проєктуванні DI-архітектури — ",[3380,8960,8961],{},"скільки екземплярів"," певного класу повинно існувати у системі та ",[3380,8964,8965],{},"як довго",". Guice вирішує цю задачу через механізм скопів (Scopes).",[3514,8968,8970],{"id":8969},"prototype-скоп-за-замовчуванням","Prototype — скоп за замовчуванням",[3150,8972,8973,8974,8977,8978,8981,8982,8984,8985,8988],{},"Якщо жодного скопу не вказано, Guice поводиться як ",[3380,8975,8976],{},"фабрика без пам'яті",": кожен виклик ",[3154,8979,8980],{},"getInstance()"," або кожне впровадження ",[3154,8983,3892],{}," отримує ",[3380,8986,8987],{},"новий"," екземпляр. Цей режим називається Prototype або No Scope.",[3159,8990,8992],{"className":3161,"code":8991,"language":3163,"meta":3164,"style":3164},"\u002F\u002F Guice створює новий TrackValidator при кожному впровадженні\npublic class TrackService {\n    @Inject\n    public TrackService(TrackValidator validator) { ... }\n}\n\npublic class AudioProcessor {\n    @Inject\n    public AudioProcessor(TrackValidator validator) { ... }\n}\n\u002F\u002F TrackService та AudioProcessor отримають РІЗНІ екземпляри TrackValidator\n",[3154,8993,8994,8999,9009,9015,9029,9033,9037,9048,9054,9068,9072],{"__ignoreMap":3164},[3168,8995,8996],{"class":3170,"line":3171},[3168,8997,8998],{"class":3943},"\u002F\u002F Guice створює новий TrackValidator при кожному впровадженні\n",[3168,9000,9001,9003,9005,9007],{"class":3170,"line":3189},[3168,9002,3175],{"class":3174},[3168,9004,3178],{"class":3174},[3168,9006,3205],{"class":3181},[3168,9008,3186],{"class":3185},[3168,9010,9011,9013],{"class":3170,"line":3196},[3168,9012,3830],{"class":3185},[3168,9014,3833],{"class":3181},[3168,9016,9017,9019,9021,9023,9025,9027],{"class":3170,"line":3215},[3168,9018,3223],{"class":3174},[3168,9020,3205],{"class":3226},[3168,9022,3336],{"class":3185},[3168,9024,3390],{"class":3181},[3168,9026,3307],{"class":3208},[3168,9028,6434],{"class":3185},[3168,9030,9031],{"class":3170,"line":3220},[3168,9032,3375],{"class":3185},[3168,9034,9035],{"class":3170,"line":3232},[3168,9036,3193],{"emptyLinePlaceholder":3192},[3168,9038,9039,9041,9043,9046],{"class":3170,"line":3254},[3168,9040,3175],{"class":3174},[3168,9042,3178],{"class":3174},[3168,9044,9045],{"class":3181}," AudioProcessor",[3168,9047,3186],{"class":3185},[3168,9049,9050,9052],{"class":3170,"line":3264},[3168,9051,3830],{"class":3185},[3168,9053,3833],{"class":3181},[3168,9055,9056,9058,9060,9062,9064,9066],{"class":3170,"line":3276},[3168,9057,3223],{"class":3174},[3168,9059,9045],{"class":3226},[3168,9061,3336],{"class":3185},[3168,9063,3390],{"class":3181},[3168,9065,3307],{"class":3208},[3168,9067,6434],{"class":3185},[3168,9069,9070],{"class":3170,"line":3282},[3168,9071,3375],{"class":3185},[3168,9073,9074],{"class":3170,"line":3301},[3168,9075,9076],{"class":3943},"\u002F\u002F TrackService та AudioProcessor отримають РІЗНІ екземпляри TrackValidator\n",[3150,9078,9079,9080,9083],{},"Prototype ідеально підходить для ",[3380,9081,9082],{},"stateful"," об'єктів, де кожен споживач повинен мати власний, незалежний стан: form-objects, команди (Command pattern), тимчасові обчислювальні об'єкти.",[3514,9085,9087,9089],{"id":9086},"singleton-один-на-весь-injector",[3154,9088,4519],{}," — один на весь Injector",[3150,9091,9092,9094,9095,9098,9099,9101],{},[3154,9093,4519],{}," гарантує, що Guice створить клас ",[3380,9096,9097],{},"рівно один раз"," протягом усього часу життя ",[3154,9100,4137],{},". Усі, хто залежать від цього класу, отримають той самий екземпляр.",[3159,9103,9105],{"className":3161,"code":9104,"language":3163,"meta":3164,"style":3164},"@Singleton\npublic class ConnectionManager {\n    \u002F\u002F Конструктор викликається один раз — при першому запиті\n    @Inject\n    public ConnectionManager(@Named(\"dbUrl\") String url, ...) { ... }\n}\n",[3154,9106,9107,9113,9123,9128,9134,9157],{"__ignoreMap":3164},[3168,9108,9109,9111],{"class":3170,"line":3171},[3168,9110,5191],{"class":3185},[3168,9112,5194],{"class":3181},[3168,9114,9115,9117,9119,9121],{"class":3170,"line":3189},[3168,9116,3175],{"class":3174},[3168,9118,3178],{"class":3174},[3168,9120,3248],{"class":3181},[3168,9122,3186],{"class":3185},[3168,9124,9125],{"class":3170,"line":3196},[3168,9126,9127],{"class":3943},"    \u002F\u002F Конструктор викликається один раз — при першому запиті\n",[3168,9129,9130,9132],{"class":3170,"line":3215},[3168,9131,3830],{"class":3185},[3168,9133,3833],{"class":3181},[3168,9135,9136,9138,9140,9142,9144,9146,9148,9150,9152,9154],{"class":3170,"line":3220},[3168,9137,3223],{"class":3174},[3168,9139,3248],{"class":3226},[3168,9141,6608],{"class":3185},[3168,9143,5274],{"class":3181},[3168,9145,3336],{"class":3185},[3168,9147,5044],{"class":3257},[3168,9149,5320],{"class":3185},[3168,9151,5284],{"class":3181},[3168,9153,5220],{"class":3208},[3168,9155,9156],{"class":3185},", ...) { ... }\n",[3168,9158,9159],{"class":3170,"line":3232},[3168,9160,3375],{"class":3185},[3150,9162,9163],{},"Альтернативний спосіб — вказати скоп у модулі, не чіпаючи клас:",[3159,9165,9167],{"className":3161,"code":9166,"language":3163,"meta":3164,"style":3164},"\u002F\u002F У AppModule.configure() — корисно, коли клас з бібліотеки\nbind(ConnectionManager.class).in(Singleton.class);\n\u002F\u002F або через статичний імпорт:\nbind(ConnectionManager.class).in(Scopes.SINGLETON);\n",[3154,9168,9169,9174,9200,9205],{"__ignoreMap":3164},[3168,9170,9171],{"class":3170,"line":3171},[3168,9172,9173],{"class":3943},"\u002F\u002F У AppModule.configure() — корисно, коли клас з бібліотеки\n",[3168,9175,9176,9178,9180,9182,9184,9186,9188,9190,9192,9194,9196,9198],{"class":3170,"line":3189},[3168,9177,5736],{"class":3226},[3168,9179,3336],{"class":3185},[3168,9181,3386],{"class":3208},[3168,9183,3351],{"class":3185},[3168,9185,4972],{"class":3208},[3168,9187,4975],{"class":3185},[3168,9189,5796],{"class":3226},[3168,9191,3336],{"class":3185},[3168,9193,5801],{"class":3208},[3168,9195,3351],{"class":3185},[3168,9197,4972],{"class":3208},[3168,9199,3342],{"class":3185},[3168,9201,9202],{"class":3170,"line":3196},[3168,9203,9204],{"class":3943},"\u002F\u002F або через статичний імпорт:\n",[3168,9206,9207,9209,9211,9213,9215,9217,9219,9221,9223,9226,9228,9231],{"class":3170,"line":3215},[3168,9208,5736],{"class":3226},[3168,9210,3336],{"class":3185},[3168,9212,3386],{"class":3208},[3168,9214,3351],{"class":3185},[3168,9216,4972],{"class":3208},[3168,9218,4975],{"class":3185},[3168,9220,5796],{"class":3226},[3168,9222,3336],{"class":3185},[3168,9224,9225],{"class":3208},"Scopes",[3168,9227,3351],{"class":3185},[3168,9229,9230],{"class":3208},"SINGLETON",[3168,9232,3342],{"class":3185},[3150,9234,9235,9237,9238,3351],{},[3154,9236,4519],{}," ідеальний для: репозиторіїв, сервісів, пулів з'єднань, кешів, ",[3154,9239,8797],{},[3514,9241,9243],{"id":9242},"eager-vs-lazy-singleton","Eager vs Lazy Singleton",[3150,9245,9246,9247,9249,9250,9253,9254,8779],{},"За замовчуванням ",[3154,9248,4519],{}," є ",[3380,9251,9252],{},"lazy"," (ліниво створюваним): Guice створює екземпляр лише при першому зверненні. Якщо потрібно створити його при старті застосунку (наприклад, щоб перевірити з'єднання з БД негайно), використовуйте ",[3154,9255,9256],{},"asEagerSingleton()",[3159,9258,9260],{"className":3161,"code":9259,"language":3163,"meta":3164,"style":3164},"\u002F\u002F У AppModule.configure()\nbind(ConnectionManager.class).asEagerSingleton();\n\u002F\u002F Guice створить ConnectionManager відразу при Guice.createInjector()\n\u002F\u002F Якщо ініціалізація провалиться — застосунок не запуститься\n",[3154,9261,9262,9267,9286,9291],{"__ignoreMap":3164},[3168,9263,9264],{"class":3170,"line":3171},[3168,9265,9266],{"class":3943},"\u002F\u002F У AppModule.configure()\n",[3168,9268,9269,9271,9273,9275,9277,9279,9281,9284],{"class":3170,"line":3189},[3168,9270,5736],{"class":3226},[3168,9272,3336],{"class":3185},[3168,9274,3386],{"class":3208},[3168,9276,3351],{"class":3185},[3168,9278,4972],{"class":3208},[3168,9280,4975],{"class":3185},[3168,9282,9283],{"class":3226},"asEagerSingleton",[3168,9285,3317],{"class":3185},[3168,9287,9288],{"class":3170,"line":3196},[3168,9289,9290],{"class":3943},"\u002F\u002F Guice створить ConnectionManager відразу при Guice.createInjector()\n",[3168,9292,9293],{"class":3170,"line":3215},[3168,9294,9295],{"class":3943},"\u002F\u002F Якщо ініціалізація провалиться — застосунок не запуститься\n",[3150,9297,9298,9300],{},[3154,9299,9256],{}," підходить для критичних ресурсів, відсутність яких є фатальною для застосунку: бази даних, черги повідомлень, ліцензійні перевірки.",[4187,9302,4195,9305,4195,9322,4195,9325,4195,9330,4195,9333,4195,9335,4195,9338,4195,9340,4195,9344,4195,9347,4195,9351,4195,9354,4195,9358,4195,9362,4195,9367,4195,9370,4195,9374,4195,9376,4195,9380,4195,9382,4195,9386,4195,9389,4195,9392,4195,9396,4195,9400,4195,9406,4195,9409,4195,9413,4195,9417,4195,9422,4195,9425,4195,9430,4195,9433,4195,9435,4195,9438,4195,9442],{"viewBox":9303,"className":9304,"xmlns":4194},"0 0 860 310",[4191,4192,4193],[4197,9306,4199,9307,4199,9312,4199,9317,4195],{},[4201,9308,9310],{"id":9309,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-b5",[4209,9311],{"d":4211,"fill":3706},[4201,9313,9315],{"id":9314,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-g5",[4209,9316],{"d":4211,"fill":3614},[4201,9318,9320],{"id":9319,"markerWidth":4204,"markerHeight":4204,"refX":4205,"refY":4206,"orient":4207},"arr-a5",[4209,9321],{"d":4211,"fill":4222},[3607,9323,9324],{"x":3618,"y":6636,"style":3611,"fontSize":3612,"fontWeight":3613,"fill":4222},"Lifecycle об'єктів у Guice",[3170,9326],{"x1":3711,"y1":9327,"x2":9328,"y2":9327,"stroke":9329,"strokeWidth":3632},"50","820","#374151",[3607,9331,9332],{"x":3704,"y":7329,"style":3611,"fontSize":3641,"fill":3621},"час виконання застосунку →",[3625,9334],{"x":3711,"y":4335,"width":3612,"height":3612,"rx":3707,"fill":4222},[3607,9336,9337],{"x":3695,"y":3739,"fontSize":3641,"fill":4222},"Injector.createInjector()",[3625,9339],{"x":3620,"y":4335,"width":3612,"height":3612,"rx":3707,"fill":3706},[3607,9341,9343],{"x":9342,"y":3739,"fontSize":3641,"fill":3706},"298","Запит 1",[3625,9345],{"x":9346,"y":4335,"width":3612,"height":3612,"rx":3707,"fill":3706},"480",[3607,9348,9350],{"x":9349,"y":3739,"fontSize":3641,"fill":3706},"498","Запит 2",[3625,9352],{"x":9353,"y":4335,"width":3612,"height":3612,"rx":3707,"fill":3706},"680",[3607,9355,9357],{"x":9356,"y":3739,"fontSize":3641,"fill":3706},"698","Запит 3",[3607,9359,9361],{"x":4225,"y":6644,"style":3611,"fontSize":3619,"fontWeight":3613,"fill":6703,"transform":9360},"rotate(-90, 30, 115)","Prototype",[3170,9363],{"x1":3711,"y1":4275,"x2":9328,"y2":4275,"stroke":9364,"strokeWidth":3622,"strokeDashArray":9365},"#4b5563",[9366,4206],"4",[3625,9368],{"x":3620,"y":3671,"width":4273,"height":6636,"rx":4242,"fill":9369,"stroke":3706,"strokeWidth":3632},"rgba(96,165,250,0.15)",[3607,9371,9373],{"x":9372,"y":4341,"style":3611,"fontSize":3641,"fill":4248},"320","new Instance #1",[3625,9375],{"x":9346,"y":3671,"width":4273,"height":6636,"rx":4242,"fill":9369,"stroke":3706,"strokeWidth":3632},[3607,9377,9379],{"x":9378,"y":4341,"style":3611,"fontSize":3641,"fill":4248},"520","new Instance #2",[3625,9381],{"x":9353,"y":3671,"width":4273,"height":6636,"rx":4242,"fill":9369,"stroke":3706,"strokeWidth":3632},[3607,9383,9385],{"x":9384,"y":4341,"style":3611,"fontSize":3641,"fill":4248},"720","new Instance #3",[3607,9387,9388],{"x":3704,"y":7332,"style":3611,"fontSize":4204,"fill":3621},"Кожен запит отримує новий об'єкт. Немає спільного стану.",[3607,9390,5801],{"x":4225,"y":7313,"style":3611,"fontSize":3619,"fontWeight":3613,"fill":3614,"transform":9391},"rotate(-90, 30, 180)",[3170,9393],{"x1":3711,"y1":9394,"x2":9328,"y2":9394,"stroke":9364,"strokeWidth":3622,"strokeDashArray":9395},"165",[9366,4206],[3625,9397],{"x":3711,"y":4349,"width":9398,"height":6636,"rx":4242,"fill":9399,"stroke":3614,"strokeWidth":3632},"90","rgba(52,211,153,0.12)",[3607,9401,9405],{"x":9402,"y":9403,"style":3611,"fontSize":3641,"fill":9404},"105","163","#6ee7b7","Lazy: перший запит",[3625,9407],{"x":3620,"y":4349,"width":4334,"height":6636,"rx":4242,"fill":7381,"stroke":3614,"strokeWidth":3632,"strokeDashArray":9408},[4242,4206],[3607,9410,9412],{"x":9411,"y":9403,"style":3611,"fontSize":3641,"fill":9404},"530","той самий екземпляр — завжди",[3607,9414,9416],{"x":3704,"y":9415,"style":3611,"fontSize":4204,"fill":3621},"183","Один екземпляр на весь Injector. Потоко-безпечний.",[3607,9418,9421],{"x":4225,"y":9419,"style":3611,"fontSize":3619,"fontWeight":3613,"fill":4222,"transform":9420},"245","rotate(-90, 30, 245)","Eager",[3170,9423],{"x1":3711,"y1":4241,"x2":9328,"y2":4241,"stroke":9364,"strokeWidth":3622,"strokeDashArray":9424},[9366,4206],[3625,9426],{"x":3711,"y":9427,"width":9428,"height":6636,"rx":4242,"fill":9429,"stroke":4222,"strokeWidth":3632},"213","740","rgba(245,158,11,0.1)",[3607,9431,9432],{"x":3711,"y":4330,"fontSize":3641,"fill":4291},"↑ створено при createInjector()",[3607,9434,9412],{"x":9411,"y":4330,"style":3611,"fontSize":3641,"fill":4291},[3607,9436,9437],{"x":3704,"y":6740,"style":3611,"fontSize":4204,"fill":3621},"asEagerSingleton(): ініціалізація при старті. Збій = застосунок не запуститься.",[3607,9439,9441],{"x":4296,"y":9440,"style":3611,"fontSize":3641,"fill":3642},"290","💡 Repository, Service, Pool → @Singleton",[3607,9443,9444],{"x":9411,"y":9440,"style":3611,"fontSize":3641,"fill":3642},"💡 FormObject, Command, DTO → Prototype",[3514,9446,9448],{"id":9447},"thread-safety-у-singleton","Thread Safety у Singleton",[3150,9450,9451,9453,9454,9457,9458,9461,9462,9464],{},[3154,9452,4519],{}," у Guice є потоко-безпечним з точки зору ",[3380,9455,9456],{},"створення",": Guice гарантує, що конструктор викличеться рівно один раз, навіть при одночасних запитах з кількох потоків. Але ",[3380,9459,9460],{},"стан"," синглтона — ваша відповідальність. Якщо ",[3154,9463,4519],{},"-клас зберігає змінні поля, вони повинні бути синхронізовані або використовуватись лише з JavaFX Application Thread.",[5683,9466,9467,9468,9471,9472,9474,9475,9478],{},"JavaFX-контролери ",[3380,9469,9470],{},"не повинні"," бути ",[3154,9473,4519],{},". Кожне відкриття FXML-сцени створює новий контролер — і Guice повинен мати можливість це зробити. Якщо контролер є Singleton, другий виклик ",[3154,9476,9477],{},"loader.load()"," поверне той самий об'єкт зі старим UI-станом, що спричинить непередбачувану поведінку.",[9480,9481,9482,9491,9495],"accordion",{},[9483,9484,9487,9488,9490],"accordion-item",{"icon":9485,"label":9486},"i-lucide-circle-help","Коли обирати @Singleton?","Використовуйте ",[3154,9489,4519],{}," для класів, що: (1) містять дорогу ініціалізацію (пул з'єднань, кеш); (2) зберігають спільний стан між різними частинами застосунку (NavigationService, EventBus); (3) є stateless — не мають змінних полів, лише методи. Сервіси і репозиторії зазвичай є Singleton.",[9483,9492,9494],{"icon":9485,"label":9493},"Чому Prototype за замовчуванням?","Singleton — небезпечний за замовчуванням, бо розробник може забути про thread safety або ненавмисно зберегти стан між запитами. Prototype безпечніший: кожен споживач ізольований. Guice обирає обережніший варіант, залишаючи оголошення Singleton явним рішенням.",[9483,9496,9498,9499,9502,9503,9505,9506,9508,9509,4182,9512,9514],{"icon":9485,"label":9497},"Чи може Singleton залежати від Prototype?","Так, але з застереженням. Якщо ",[3154,9500,9501],{},"@Singleton TrackService"," залежить від ",[3154,9504,3390],{}," (Prototype), то при створенні Singleton отримає один екземпляр ",[3154,9507,3390],{}," і зберігатиме його назавжди — фактично перетворюючи його на Singleton. Це «Scope Widening» — небезпечний антипатерн. Якщо потрібно щоразу отримувати новий Prototype — впроваджуйте ",[3154,9510,9511],{},"Provider\u003CTrackValidator>",[3154,9513,3390],{}," напряму.",[3507,9516],{},[3145,9518,9520],{"id":9519},"тестування-з-google-guice","Тестування з Google Guice",[3150,9522,9523,9524,9527],{},"Одна з головних переваг DI — ",[3380,9525,9526],{},"тестованість",". Guice надає елегантний механізм підміни залежностей у тестах без зміни продуктивного коду.",[3514,9529,9531],{"id":9530},"проблема-без-di","Проблема без DI",[3150,9533,9534,9535,9537],{},"Уявімо, що ",[3154,9536,3594],{}," без Guice сам створює свій репозиторій:",[3159,9539,9541],{"className":3161,"code":9540,"language":3163,"meta":3164,"style":3164},"public class TrackService {\n    \u002F\u002F Жорстка залежність — підмінити неможливо без рефлексії\n    private final TrackRepository repo = new JdbcTrackRepository(\n        new ConnectionManager(\"jdbc:h2:.\u002Fdata\u002Ftest\", \"sa\", \"\")\n    );\n}\n",[3154,9542,9543,9553,9558,9577,9599,9604],{"__ignoreMap":3164},[3168,9544,9545,9547,9549,9551],{"class":3170,"line":3171},[3168,9546,3175],{"class":3174},[3168,9548,3178],{"class":3174},[3168,9550,3205],{"class":3181},[3168,9552,3186],{"class":3185},[3168,9554,9555],{"class":3170,"line":3189},[3168,9556,9557],{"class":3943},"    \u002F\u002F Жорстка залежність — підмінити неможливо без рефлексії\n",[3168,9559,9560,9562,9564,9566,9569,9571,9573,9575],{"class":3170,"line":3196},[3168,9561,3199],{"class":3174},[3168,9563,3202],{"class":3174},[3168,9565,3804],{"class":3181},[3168,9567,9568],{"class":3208}," repo",[3168,9570,3241],{"class":3185},[3168,9572,3245],{"class":3244},[3168,9574,3295],{"class":3226},[3168,9576,3251],{"class":3185},[3168,9578,9579,9582,9584,9586,9589,9591,9593,9595,9597],{"class":3170,"line":3215},[3168,9580,9581],{"class":3244},"        new",[3168,9583,3248],{"class":3226},[3168,9585,3336],{"class":3185},[3168,9587,9588],{"class":3257},"\"jdbc:h2:.\u002Fdata\u002Ftest\"",[3168,9590,3270],{"class":3185},[3168,9592,5092],{"class":3257},[3168,9594,3270],{"class":3185},[3168,9596,5124],{"class":3257},[3168,9598,4036],{"class":3185},[3168,9600,9601],{"class":3170,"line":3220},[3168,9602,9603],{"class":3185},"    );\n",[3168,9605,9606],{"class":3170,"line":3232},[3168,9607,3375],{"class":3185},[3150,9609,9610,9611,9614,9615,9618],{},"Щоб протестувати ",[3154,9612,9613],{},"TrackService.getTopRatedTracks()",", ми ",[3380,9616,9617],{},"змушені"," підняти H2-базу, наповнити її тестовими даними, і виконати реальний SQL-запит. Це інтеграційний тест, а не unit-тест — він повільніший, нестабільніший і залежить від середовища.",[3514,9620,9622,9625],{"id":9621},"modulesoverride-хірургічна-підміна",[3154,9623,9624],{},"Modules.override()"," — хірургічна підміна",[3150,9627,9628,9629,9632,9633,9636],{},"Guice надає метод ",[3154,9630,9631],{},"Modules.override(base).with(override)",", що дозволяє замінити ",[3380,9634,9635],{},"конкретні"," прив'язки базового модуля тестовими, не змінюючи жодного рядка продуктивного коду:",[3159,9638,9640],{"className":3161,"code":9639,"language":3163,"meta":3164,"style":3164},"\u002F\u002F Тестовий модуль — лише ті прив'язки, що потрібно замінити\npublic class TestAppModule extends AbstractModule {\n    @Override\n    protected void configure() {\n        \u002F\u002F Замінюємо JDBC-репозиторій на in-memory заглушку\n        bind(TrackRepository.class).to(InMemoryTrackRepository.class);\n        \u002F\u002F Замінюємо реальний URL на тестовий (H2 in-memory)\n        bindConstant()\n            .annotatedWith(Names.named(\"dbUrl\"))\n            .to(\"jdbc:h2:mem:testdb;DB_CLOSE_DELAY=-1\");\n    }\n}\n",[3154,9641,9642,9647,9662,9668,9678,9683,9710,9715,9721,9741,9754,9758],{"__ignoreMap":3164},[3168,9643,9644],{"class":3170,"line":3171},[3168,9645,9646],{"class":3943},"\u002F\u002F Тестовий модуль — лише ті прив'язки, що потрібно замінити\n",[3168,9648,9649,9651,9653,9656,9658,9660],{"class":3170,"line":3189},[3168,9650,3175],{"class":3174},[3168,9652,3178],{"class":3174},[3168,9654,9655],{"class":3181}," TestAppModule",[3168,9657,4930],{"class":3174},[3168,9659,4933],{"class":3181},[3168,9661,3186],{"class":3185},[3168,9663,9664,9666],{"class":3170,"line":3196},[3168,9665,3830],{"class":3185},[3168,9667,4946],{"class":3181},[3168,9669,9670,9672,9674,9676],{"class":3170,"line":3215},[3168,9671,4951],{"class":3174},[3168,9673,4043],{"class":3181},[3168,9675,4956],{"class":3226},[3168,9677,3229],{"class":3185},[3168,9679,9680],{"class":3170,"line":3220},[3168,9681,9682],{"class":3943},"        \u002F\u002F Замінюємо JDBC-репозиторій на in-memory заглушку\n",[3168,9684,9685,9687,9689,9691,9693,9695,9697,9699,9701,9704,9706,9708],{"class":3170,"line":3232},[3168,9686,4963],{"class":3226},[3168,9688,3336],{"class":3185},[3168,9690,3407],{"class":3208},[3168,9692,3351],{"class":3185},[3168,9694,4972],{"class":3208},[3168,9696,4975],{"class":3185},[3168,9698,4819],{"class":3226},[3168,9700,3336],{"class":3185},[3168,9702,9703],{"class":3208},"InMemoryTrackRepository",[3168,9705,3351],{"class":3185},[3168,9707,4972],{"class":3208},[3168,9709,3342],{"class":3185},[3168,9711,9712],{"class":3170,"line":3254},[3168,9713,9714],{"class":3943},"        \u002F\u002F Замінюємо реальний URL на тестовий (H2 in-memory)\n",[3168,9716,9717,9719],{"class":3170,"line":3264},[3168,9718,5023],{"class":3226},[3168,9720,8085],{"class":3185},[3168,9722,9723,9725,9727,9729,9731,9733,9735,9737,9739],{"class":3170,"line":3276},[3168,9724,5052],{"class":3185},[3168,9726,5029],{"class":3226},[3168,9728,3336],{"class":3185},[3168,9730,5034],{"class":3208},[3168,9732,3351],{"class":3185},[3168,9734,5039],{"class":3226},[3168,9736,3336],{"class":3185},[3168,9738,5044],{"class":3257},[3168,9740,5047],{"class":3185},[3168,9742,9743,9745,9747,9749,9752],{"class":3170,"line":3282},[3168,9744,5052],{"class":3185},[3168,9746,4819],{"class":3226},[3168,9748,3336],{"class":3185},[3168,9750,9751],{"class":3257},"\"jdbc:h2:mem:testdb;DB_CLOSE_DELAY=-1\"",[3168,9753,3342],{"class":3185},[3168,9755,9756],{"class":3170,"line":3301},[3168,9757,3369],{"class":3185},[3168,9759,9760],{"class":3170,"line":3320},[3168,9761,3375],{"class":3185},[3159,9763,9765],{"className":3161,"code":9764,"language":3163,"meta":3164,"style":3164},"\u002F\u002F Тест сервісу — без реальної бази даних\nclass TrackServiceTest {\n\n    private TrackService trackService;\n    private InMemoryTrackRepository fakeRepository;\n\n    @BeforeEach\n    void setUp() {\n        \u002F\u002F Базовий модуль + тестові override\n        Injector injector = Guice.createInjector(\n            Modules.override(new AppModule()).with(new TestAppModule())\n        );\n        trackService   = injector.getInstance(TrackService.class);\n        fakeRepository = (InMemoryTrackRepository)\n            injector.getInstance(TrackRepository.class);\n    }\n\n    @Test\n    void shouldReturnTopRatedTracks() {\n        \u002F\u002F Arrange: наповнюємо in-memory сховище тестовими даними\n        fakeRepository.save(new Track(\"Beethoven - Symphony No.9\", 5.0));\n        fakeRepository.save(new Track(\"Mozart - Requiem\",          4.8));\n        fakeRepository.save(new Track(\"Bach - Goldberg Variations\", 3.2));\n\n        \u002F\u002F Act\n        List\u003CTrack> top = trackService.getTopRatedTracks(2);\n\n        \u002F\u002F Assert\n        assertThat(top).hasSize(2);\n        assertThat(top.get(0).getTitle()).contains(\"Beethoven\");\n    }\n}\n",[3154,9766,9767,9772,9781,9785,9795,9807,9811,9818,9828,9833,9849,9879,9883,9904,9909,9928,9932,9936,9943,9952,9957,9986,10013,10039,10043,10048,10078,10082,10087,10104,10138,10142],{"__ignoreMap":3164},[3168,9768,9769],{"class":3170,"line":3171},[3168,9770,9771],{"class":3943},"\u002F\u002F Тест сервісу — без реальної бази даних\n",[3168,9773,9774,9776,9779],{"class":3170,"line":3189},[3168,9775,4972],{"class":3174},[3168,9777,9778],{"class":3181}," TrackServiceTest",[3168,9780,3186],{"class":3185},[3168,9782,9783],{"class":3170,"line":3196},[3168,9784,3193],{"emptyLinePlaceholder":3192},[3168,9786,9787,9789,9791,9793],{"class":3170,"line":3215},[3168,9788,3199],{"class":3174},[3168,9790,3205],{"class":3181},[3168,9792,3209],{"class":3208},[3168,9794,3212],{"class":3185},[3168,9796,9797,9799,9802,9805],{"class":3170,"line":3220},[3168,9798,3199],{"class":3174},[3168,9800,9801],{"class":3181}," InMemoryTrackRepository",[3168,9803,9804],{"class":3208}," fakeRepository",[3168,9806,3212],{"class":3185},[3168,9808,9809],{"class":3170,"line":3232},[3168,9810,3193],{"emptyLinePlaceholder":3192},[3168,9812,9813,9815],{"class":3170,"line":3254},[3168,9814,3830],{"class":3185},[3168,9816,9817],{"class":3181},"BeforeEach\n",[3168,9819,9820,9823,9826],{"class":3170,"line":3264},[3168,9821,9822],{"class":3181},"    void",[3168,9824,9825],{"class":3226}," setUp",[3168,9827,3229],{"class":3185},[3168,9829,9830],{"class":3170,"line":3276},[3168,9831,9832],{"class":3943},"        \u002F\u002F Базовий модуль + тестові override\n",[3168,9834,9835,9837,9839,9841,9843,9845,9847],{"class":3170,"line":3282},[3168,9836,5515],{"class":3181},[3168,9838,5518],{"class":3208},[3168,9840,3241],{"class":3185},[3168,9842,5523],{"class":3208},[3168,9844,3351],{"class":3185},[3168,9846,5528],{"class":3226},[3168,9848,3251],{"class":3185},[3168,9850,9851,9854,9856,9859,9861,9863,9865,9868,9871,9873,9875,9877],{"class":3170,"line":3301},[3168,9852,9853],{"class":3208},"            Modules",[3168,9855,3351],{"class":3185},[3168,9857,9858],{"class":3226},"override",[3168,9860,3336],{"class":3185},[3168,9862,3245],{"class":3244},[3168,9864,4927],{"class":3226},[3168,9866,9867],{"class":3185},"()).",[3168,9869,9870],{"class":3226},"with",[3168,9872,3336],{"class":3185},[3168,9874,3245],{"class":3244},[3168,9876,9655],{"class":3226},[3168,9878,8134],{"class":3185},[3168,9880,9881],{"class":3170,"line":3320},[3168,9882,3279],{"class":3185},[3168,9884,9885,9888,9890,9892,9894,9896,9898,9900,9902],{"class":3170,"line":3345},[3168,9886,9887],{"class":3185},"        trackService   = ",[3168,9889,5558],{"class":3208},[3168,9891,3351],{"class":3185},[3168,9893,5563],{"class":3226},[3168,9895,3336],{"class":3185},[3168,9897,3594],{"class":3208},[3168,9899,3351],{"class":3185},[3168,9901,4972],{"class":3208},[3168,9903,3342],{"class":3185},[3168,9905,9906],{"class":3170,"line":3366},[3168,9907,9908],{"class":3185},"        fakeRepository = (InMemoryTrackRepository)\n",[3168,9910,9911,9914,9916,9918,9920,9922,9924,9926],{"class":3170,"line":3372},[3168,9912,9913],{"class":3208},"            injector",[3168,9915,3351],{"class":3185},[3168,9917,5563],{"class":3226},[3168,9919,3336],{"class":3185},[3168,9921,3407],{"class":3208},[3168,9923,3351],{"class":3185},[3168,9925,4972],{"class":3208},[3168,9927,3342],{"class":3185},[3168,9929,9930],{"class":3170,"line":4447},[3168,9931,3369],{"class":3185},[3168,9933,9934],{"class":3170,"line":4453},[3168,9935,3193],{"emptyLinePlaceholder":3192},[3168,9937,9938,9940],{"class":3170,"line":5129},[3168,9939,3830],{"class":3185},[3168,9941,9942],{"class":3181},"Test\n",[3168,9944,9945,9947,9950],{"class":3170,"line":5134},[3168,9946,9822],{"class":3181},[3168,9948,9949],{"class":3226}," shouldReturnTopRatedTracks",[3168,9951,3229],{"class":3185},[3168,9953,9954],{"class":3170,"line":5357},[3168,9955,9956],{"class":3943},"        \u002F\u002F Arrange: наповнюємо in-memory сховище тестовими даними\n",[3168,9958,9959,9962,9964,9967,9969,9971,9974,9976,9979,9981,9984],{"class":3170,"line":5370},[3168,9960,9961],{"class":3208},"        fakeRepository",[3168,9963,3351],{"class":3185},[3168,9965,9966],{"class":3226},"save",[3168,9968,3336],{"class":3185},[3168,9970,3245],{"class":3244},[3168,9972,9973],{"class":3226}," Track",[3168,9975,3336],{"class":3185},[3168,9977,9978],{"class":3257},"\"Beethoven - Symphony No.9\"",[3168,9980,3270],{"class":3185},[3168,9982,9983],{"class":6148},"5.0",[3168,9985,7204],{"class":3185},[3168,9987,9988,9990,9992,9994,9996,9998,10000,10002,10005,10008,10011],{"class":3170,"line":5375},[3168,9989,9961],{"class":3208},[3168,9991,3351],{"class":3185},[3168,9993,9966],{"class":3226},[3168,9995,3336],{"class":3185},[3168,9997,3245],{"class":3244},[3168,9999,9973],{"class":3226},[3168,10001,3336],{"class":3185},[3168,10003,10004],{"class":3257},"\"Mozart - Requiem\"",[3168,10006,10007],{"class":3185},",          ",[3168,10009,10010],{"class":6148},"4.8",[3168,10012,7204],{"class":3185},[3168,10014,10015,10017,10019,10021,10023,10025,10027,10029,10032,10034,10037],{"class":3170,"line":5380},[3168,10016,9961],{"class":3208},[3168,10018,3351],{"class":3185},[3168,10020,9966],{"class":3226},[3168,10022,3336],{"class":3185},[3168,10024,3245],{"class":3244},[3168,10026,9973],{"class":3226},[3168,10028,3336],{"class":3185},[3168,10030,10031],{"class":3257},"\"Bach - Goldberg Variations\"",[3168,10033,3270],{"class":3185},[3168,10035,10036],{"class":6148},"3.2",[3168,10038,7204],{"class":3185},[3168,10040,10041],{"class":3170,"line":5402},[3168,10042,3193],{"emptyLinePlaceholder":3192},[3168,10044,10045],{"class":3170,"line":5419},[3168,10046,10047],{"class":3943},"        \u002F\u002F Act\n",[3168,10049,10050,10053,10055,10058,10060,10063,10065,10067,10069,10072,10074,10076],{"class":3170,"line":5424},[3168,10051,10052],{"class":3181},"        List",[3168,10054,4536],{"class":3185},[3168,10056,10057],{"class":3181},"Track",[3168,10059,7962],{"class":3185},[3168,10061,10062],{"class":3208},"top",[3168,10064,3241],{"class":3185},[3168,10066,3354],{"class":3208},[3168,10068,3351],{"class":3185},[3168,10070,10071],{"class":3226},"getTopRatedTracks",[3168,10073,3336],{"class":3185},[3168,10075,3707],{"class":6148},[3168,10077,3342],{"class":3185},[3168,10079,10080],{"class":3170,"line":6183},[3168,10081,3193],{"emptyLinePlaceholder":3192},[3168,10083,10084],{"class":3170,"line":3653},[3168,10085,10086],{"class":3943},"        \u002F\u002F Assert\n",[3168,10088,10089,10092,10095,10098,10100,10102],{"class":3170,"line":7661},[3168,10090,10091],{"class":3226},"        assertThat",[3168,10093,10094],{"class":3185},"(top).",[3168,10096,10097],{"class":3226},"hasSize",[3168,10099,3336],{"class":3185},[3168,10101,3707],{"class":6148},[3168,10103,3342],{"class":3185},[3168,10105,10106,10108,10110,10112,10114,10117,10119,10122,10124,10126,10128,10131,10133,10136],{"class":3170,"line":7677},[3168,10107,10091],{"class":3226},[3168,10109,3336],{"class":3185},[3168,10111,10062],{"class":3208},[3168,10113,3351],{"class":3185},[3168,10115,10116],{"class":3226},"get",[3168,10118,3336],{"class":3185},[3168,10120,10121],{"class":6148},"0",[3168,10123,4975],{"class":3185},[3168,10125,8115],{"class":3226},[3168,10127,9867],{"class":3185},[3168,10129,10130],{"class":3226},"contains",[3168,10132,3336],{"class":3185},[3168,10134,10135],{"class":3257},"\"Beethoven\"",[3168,10137,3342],{"class":3185},[3168,10139,10140],{"class":3170,"line":7682},[3168,10141,3369],{"class":3185},[3168,10143,10144],{"class":3170,"line":7700},[3168,10145,3375],{"class":3185},[3150,10147,5708,10148,10150,10151,10153,10154,10157,10158,10160,10161,3270,10163,10165],{},[3154,10149,9631],{}," будує новий об'єднаний модуль. Усі прив'язки з ",[3154,10152,9858],{}," мають вищий пріоритет і перекривають відповідні прив'язки з ",[3154,10155,10156],{},"base",". Решта прив'язок з ",[3154,10159,4233],{}," залишаються незмінними — наприклад, ",[3154,10162,3594],{},[3154,10164,3390],{}," та всі інші класи, що не потребують підміни у тесті.",[3514,10167,10169],{"id":10168},"тестування-з-mockito-без-testmodule","Тестування з Mockito без TestModule",[3150,10171,10172,10173,10176,10177,10179],{},"Якщо ви використовуєте Mockito, можна не писати ",[3154,10174,10175],{},"TestModule"," взагалі — використайте ",[3154,10178,5914],{}," з mock-об'єктом:",[3159,10181,10183],{"className":3161,"code":10182,"language":3163,"meta":3164,"style":3164},"@ExtendWith(MockitoExtension.class)\nclass TrackServiceTest {\n\n    @Mock\n    private TrackRepository mockRepository;\n\n    private TrackService trackService;\n\n    @BeforeEach\n    void setUp() {\n        \u002F\u002F Передаємо Mockito-mock напряму через toInstance()\n        Injector injector = Guice.createInjector(binder ->\n            binder.bind(TrackRepository.class).toInstance(mockRepository)\n        );\n        trackService = injector.getInstance(TrackService.class);\n    }\n\n    @Test\n    void shouldDelegateToRepository() {\n        \u002F\u002F Arrange\n        when(mockRepository.findAll()).thenReturn(List.of(\n            new Track(\"Test Track\", 4.0)\n        ));\n\n        \u002F\u002F Act\n        List\u003CTrack> result = trackService.listAll();\n\n        \u002F\u002F Assert\n        assertThat(result).hasSize(1);\n        verify(mockRepository, times(1)).findAll();\n    }\n}\n",[3154,10184,10185,10203,10211,10215,10222,10233,10237,10247,10251,10257,10265,10270,10290,10314,10318,10339,10343,10347,10353,10362,10367,10399,10418,10423,10427,10431,10454,10458,10462,10477,10498,10502],{"__ignoreMap":3164},[3168,10186,10187,10189,10192,10194,10197,10199,10201],{"class":3170,"line":3171},[3168,10188,5191],{"class":3185},[3168,10190,10191],{"class":3181},"ExtendWith",[3168,10193,3336],{"class":3185},[3168,10195,10196],{"class":3208},"MockitoExtension",[3168,10198,3351],{"class":3185},[3168,10200,4972],{"class":3208},[3168,10202,4036],{"class":3185},[3168,10204,10205,10207,10209],{"class":3170,"line":3189},[3168,10206,4972],{"class":3174},[3168,10208,9778],{"class":3181},[3168,10210,3186],{"class":3185},[3168,10212,10213],{"class":3170,"line":3196},[3168,10214,3193],{"emptyLinePlaceholder":3192},[3168,10216,10217,10219],{"class":3170,"line":3215},[3168,10218,3830],{"class":3185},[3168,10220,10221],{"class":3181},"Mock\n",[3168,10223,10224,10226,10228,10231],{"class":3170,"line":3220},[3168,10225,3199],{"class":3174},[3168,10227,3804],{"class":3181},[3168,10229,10230],{"class":3208}," mockRepository",[3168,10232,3212],{"class":3185},[3168,10234,10235],{"class":3170,"line":3232},[3168,10236,3193],{"emptyLinePlaceholder":3192},[3168,10238,10239,10241,10243,10245],{"class":3170,"line":3254},[3168,10240,3199],{"class":3174},[3168,10242,3205],{"class":3181},[3168,10244,3209],{"class":3208},[3168,10246,3212],{"class":3185},[3168,10248,10249],{"class":3170,"line":3264},[3168,10250,3193],{"emptyLinePlaceholder":3192},[3168,10252,10253,10255],{"class":3170,"line":3276},[3168,10254,3830],{"class":3185},[3168,10256,9817],{"class":3181},[3168,10258,10259,10261,10263],{"class":3170,"line":3282},[3168,10260,9822],{"class":3181},[3168,10262,9825],{"class":3226},[3168,10264,3229],{"class":3185},[3168,10266,10267],{"class":3170,"line":3301},[3168,10268,10269],{"class":3943},"        \u002F\u002F Передаємо Mockito-mock напряму через toInstance()\n",[3168,10271,10272,10274,10276,10278,10280,10282,10284,10287],{"class":3170,"line":3320},[3168,10273,5515],{"class":3181},[3168,10275,5518],{"class":3208},[3168,10277,3241],{"class":3185},[3168,10279,5523],{"class":3208},[3168,10281,3351],{"class":3185},[3168,10283,5528],{"class":3226},[3168,10285,10286],{"class":3185},"(binder ",[3168,10288,10289],{"class":3174},"->\n",[3168,10291,10292,10295,10297,10299,10301,10303,10305,10307,10309,10311],{"class":3170,"line":3345},[3168,10293,10294],{"class":3208},"            binder",[3168,10296,3351],{"class":3185},[3168,10298,5736],{"class":3226},[3168,10300,3336],{"class":3185},[3168,10302,3407],{"class":3208},[3168,10304,3351],{"class":3185},[3168,10306,4972],{"class":3208},[3168,10308,4975],{"class":3185},[3168,10310,5873],{"class":3226},[3168,10312,10313],{"class":3185},"(mockRepository)\n",[3168,10315,10316],{"class":3170,"line":3366},[3168,10317,3279],{"class":3185},[3168,10319,10320,10323,10325,10327,10329,10331,10333,10335,10337],{"class":3170,"line":3372},[3168,10321,10322],{"class":3185},"        trackService = ",[3168,10324,5558],{"class":3208},[3168,10326,3351],{"class":3185},[3168,10328,5563],{"class":3226},[3168,10330,3336],{"class":3185},[3168,10332,3594],{"class":3208},[3168,10334,3351],{"class":3185},[3168,10336,4972],{"class":3208},[3168,10338,3342],{"class":3185},[3168,10340,10341],{"class":3170,"line":4447},[3168,10342,3369],{"class":3185},[3168,10344,10345],{"class":3170,"line":4453},[3168,10346,3193],{"emptyLinePlaceholder":3192},[3168,10348,10349,10351],{"class":3170,"line":5129},[3168,10350,3830],{"class":3185},[3168,10352,9942],{"class":3181},[3168,10354,10355,10357,10360],{"class":3170,"line":5134},[3168,10356,9822],{"class":3181},[3168,10358,10359],{"class":3226}," shouldDelegateToRepository",[3168,10361,3229],{"class":3185},[3168,10363,10364],{"class":3170,"line":5357},[3168,10365,10366],{"class":3943},"        \u002F\u002F Arrange\n",[3168,10368,10369,10372,10374,10377,10379,10382,10384,10387,10389,10392,10394,10397],{"class":3170,"line":5370},[3168,10370,10371],{"class":3226},"        when",[3168,10373,3336],{"class":3185},[3168,10375,10376],{"class":3208},"mockRepository",[3168,10378,3351],{"class":3185},[3168,10380,10381],{"class":3226},"findAll",[3168,10383,9867],{"class":3185},[3168,10385,10386],{"class":3226},"thenReturn",[3168,10388,3336],{"class":3185},[3168,10390,10391],{"class":3208},"List",[3168,10393,3351],{"class":3185},[3168,10395,10396],{"class":3226},"of",[3168,10398,3251],{"class":3185},[3168,10400,10401,10404,10406,10408,10411,10413,10416],{"class":3170,"line":5375},[3168,10402,10403],{"class":3244},"            new",[3168,10405,9973],{"class":3226},[3168,10407,3336],{"class":3185},[3168,10409,10410],{"class":3257},"\"Test Track\"",[3168,10412,3270],{"class":3185},[3168,10414,10415],{"class":6148},"4.0",[3168,10417,4036],{"class":3185},[3168,10419,10420],{"class":3170,"line":5380},[3168,10421,10422],{"class":3185},"        ));\n",[3168,10424,10425],{"class":3170,"line":5402},[3168,10426,3193],{"emptyLinePlaceholder":3192},[3168,10428,10429],{"class":3170,"line":5419},[3168,10430,10047],{"class":3943},[3168,10432,10433,10435,10437,10439,10441,10444,10446,10448,10450,10452],{"class":3170,"line":5424},[3168,10434,10052],{"class":3181},[3168,10436,4536],{"class":3185},[3168,10438,10057],{"class":3181},[3168,10440,7962],{"class":3185},[3168,10442,10443],{"class":3208},"result",[3168,10445,3241],{"class":3185},[3168,10447,3354],{"class":3208},[3168,10449,3351],{"class":3185},[3168,10451,5583],{"class":3226},[3168,10453,3317],{"class":3185},[3168,10455,10456],{"class":3170,"line":6183},[3168,10457,3193],{"emptyLinePlaceholder":3192},[3168,10459,10460],{"class":3170,"line":3653},[3168,10461,10086],{"class":3943},[3168,10463,10464,10466,10469,10471,10473,10475],{"class":3170,"line":7661},[3168,10465,10091],{"class":3226},[3168,10467,10468],{"class":3185},"(result).",[3168,10470,10097],{"class":3226},[3168,10472,3336],{"class":3185},[3168,10474,3622],{"class":6148},[3168,10476,3342],{"class":3185},[3168,10478,10479,10482,10485,10488,10490,10492,10494,10496],{"class":3170,"line":7677},[3168,10480,10481],{"class":3226},"        verify",[3168,10483,10484],{"class":3185},"(mockRepository, ",[3168,10486,10487],{"class":3226},"times",[3168,10489,3336],{"class":3185},[3168,10491,3622],{"class":6148},[3168,10493,5085],{"class":3185},[3168,10495,10381],{"class":3226},[3168,10497,3317],{"class":3185},[3168,10499,10500],{"class":3170,"line":7682},[3168,10501,3369],{"class":3185},[3168,10503,10504],{"class":3170,"line":7700},[3168,10505,3375],{"class":3185},[3150,10507,5810,10508,10511,10512,10514],{},[3154,10509,10510],{},"binder -> binder.bind(...).toInstance(...)"," — це лямбда, що реалізує інтерфейс ",[3154,10513,4115],{},". Guice підтримує цей скорочений синтаксис, що робить тестові конфігурації надзвичайно компактними.",[4087,10516,10517,10520,10521,10524,10525,8847,10527,10529],{},[3380,10518,10519],{},"Золоте правило тестування з Guice:"," тест повинен перевіряти ",[3380,10522,10523],{},"один"," клас у ізоляції. Підміняйте через ",[3154,10526,9624],{},[3154,10528,5914],{}," лише ті залежності, що є зовнішніми по відношенню до тестованого класу (БД, файлова система, HTTP-клієнти). Внутрішні залежності (валідатори, маппери, утиліти) залишайте реальними — вони є частиною контракту тестованого класу.",[3507,10531],{},[3145,10533,10535],{"id":10534},"практичні-завдання","Практичні завдання",[3150,10537,10538],{},"Завдання розбиті на три рівні складності. Рівні 1 і 2 є самостійними, рівень 3 утворює міні-проєкт — повноцінну JavaFX-сцену з Guice від нуля до готового застосунку.",[3514,10540,10542],{"id":10541},"рівень-1-базовий-синтаксис-та-анотації","Рівень 1 — Базовий: Синтаксис та анотації",[3150,10544,10545,10548,10549,10552,10553,10555,10556,10558],{},[3380,10546,10547],{},"Завдання 1.1 — Перший Module."," Напишіть клас ",[3154,10550,10551],{},"AudioModule",", що розширює ",[3154,10554,4126],{},". У методі ",[3154,10557,4130],{}," зв'яжіть:",[10560,10561,10562,10569,10583],"ul",{},[3573,10563,10564,10566,10567],{},[3154,10565,4996],{}," → ",[3154,10568,4331],{},[3573,10570,10571,10572,10575,10576,10578,10579,10582],{},"Константу ",[3154,10573,10574],{},"@Named(\"maxPoolSize\")"," → значення ",[3154,10577,3619],{}," (тип ",[3154,10580,10581],{},"int",")",[3573,10584,10571,10585,10588,10589],{},[3154,10586,10587],{},"@Named(\"uploadPath\")"," → рядок ",[3154,10590,10591],{},"\"\u002Fvar\u002Faudiobook\u002Fuploads\"",[3150,10593,10594,10595,10597],{},"Перевірте, що Guice може створити ",[3154,10596,4137],{}," без помилок.",[3150,10599,10600,10603,10604,10607,10608,4510,10611,10613,10614,10616,10617,10619],{},[3380,10601,10602],{},"Завдання 1.2 — Впровадження через конструктор."," Візьміть клас ",[3154,10605,10606],{},"AuthorService"," з попередньої статті, що має поля ",[3154,10609,10610],{},"authorRepository",[3154,10612,3875],{},". Перепишіть його так, щоб обидві залежності впроваджувались через ",[3154,10615,3892],{},"-конструктор із ",[3154,10618,3896],{}," полями. Переконайтесь, що клас неможливо інстанціювати без Guice без явної передачі залежностей у конструктор.",[3150,10621,10622,10625],{},[3380,10623,10624],{},"Завдання 1.3 — Виправлення антипатерну."," Знайдіть помилку у такому коді та поясніть, чому вона є проблемою:",[3159,10627,10629],{"className":3161,"code":10628,"language":3163,"meta":3164,"style":3164},"@Singleton\npublic class PlaybackService {\n\n    @Inject\n    private Injector injector; \u002F\u002F впроваджуємо сам контейнер\n\n    public TrackPlayer createPlayer() {\n        return injector.getInstance(TrackPlayer.class); \u002F\u002F Service Locator!\n    }\n}\n",[3154,10630,10631,10637,10648,10652,10658,10671,10675,10687,10712,10716],{"__ignoreMap":3164},[3168,10632,10633,10635],{"class":3170,"line":3171},[3168,10634,5191],{"class":3185},[3168,10636,5194],{"class":3181},[3168,10638,10639,10641,10643,10646],{"class":3170,"line":3189},[3168,10640,3175],{"class":3174},[3168,10642,3178],{"class":3174},[3168,10644,10645],{"class":3181}," PlaybackService",[3168,10647,3186],{"class":3185},[3168,10649,10650],{"class":3170,"line":3196},[3168,10651,3193],{"emptyLinePlaceholder":3192},[3168,10653,10654,10656],{"class":3170,"line":3215},[3168,10655,3830],{"class":3185},[3168,10657,3833],{"class":3181},[3168,10659,10660,10662,10664,10666,10668],{"class":3170,"line":3220},[3168,10661,3199],{"class":3174},[3168,10663,7525],{"class":3181},[3168,10665,5518],{"class":3208},[3168,10667,7129],{"class":3185},[3168,10669,10670],{"class":3943},"\u002F\u002F впроваджуємо сам контейнер\n",[3168,10672,10673],{"class":3170,"line":3232},[3168,10674,3193],{"emptyLinePlaceholder":3192},[3168,10676,10677,10679,10682,10685],{"class":3170,"line":3254},[3168,10678,3223],{"class":3174},[3168,10680,10681],{"class":3181}," TrackPlayer",[3168,10683,10684],{"class":3226}," createPlayer",[3168,10686,3229],{"class":3185},[3168,10688,10689,10691,10693,10695,10697,10699,10702,10704,10706,10709],{"class":3170,"line":3264},[3168,10690,5405],{"class":3244},[3168,10692,5518],{"class":3208},[3168,10694,3351],{"class":3185},[3168,10696,5563],{"class":3226},[3168,10698,3336],{"class":3185},[3168,10700,10701],{"class":3208},"TrackPlayer",[3168,10703,3351],{"class":3185},[3168,10705,4972],{"class":3208},[3168,10707,10708],{"class":3185},"); ",[3168,10710,10711],{"class":3943},"\u002F\u002F Service Locator!\n",[3168,10713,10714],{"class":3170,"line":3276},[3168,10715,3369],{"class":3185},[3168,10717,10718],{"class":3170,"line":3282},[3168,10719,3375],{"class":3185},[3150,10721,10722,10723,3351],{},"Перепишіть клас правильно — через ",[3154,10724,10725],{},"Provider\u003CTrackPlayer>",[3507,10727],{},[3514,10729,10731],{"id":10730},"рівень-2-логіка-провайдери-та-тестування","Рівень 2 — Логіка: Провайдери та тестування",[3150,10733,10734,10737,10738,10741,10742,10744,10745,3270,10748,3270,10751,3270,10754,10555,10757,10760,10761,10764,10765,10767,10768,3351],{},[3380,10735,10736],{},"Завдання 2.1 — Provider для пулу з'єднань."," Реалізуйте клас ",[3154,10739,10740],{},"HikariConnectionPoolProvider implements Provider\u003CDataSource>",". Він повинен отримувати через ",[3154,10743,3892],{}," такі параметри: ",[3154,10746,10747],{},"@Named(\"dbUrl\") String url",[3154,10749,10750],{},"@Named(\"dbUser\") String user",[3154,10752,10753],{},"@Named(\"dbPassword\") String password",[3154,10755,10756],{},"@Named(\"maxPoolSize\") int maxPoolSize",[3154,10758,10759],{},"get()"," створюйте і налаштовуйте ",[3154,10762,10763],{},"HikariDataSource",". Зареєструйте провайдер у ",[3154,10766,10551],{}," через ",[3154,10769,10770],{},"bind(DataSource.class).toProvider(HikariConnectionPoolProvider.class).in(Singleton.class)",[3150,10772,10773,10548,10778,10781,10782,10785,10786,10767,10788,10791,10792,10795,10796,4738,10798,10801,10802,10805,10806,10809],{},[3380,10774,10775,10776,3351],{},"Завдання 2.2 — Тест з ",[3154,10777,9624],{},[3154,10779,10780],{},"AudioServiceTest"," з JUnit 5. У методі ",[3154,10783,10784],{},"setUp()"," створіть ",[3154,10787,4137],{},[3154,10789,10790],{},"Modules.override(new AudioModule()).with(testModule)",", де ",[3154,10793,10794],{},"testModule"," підміняє ",[3154,10797,4996],{},[3154,10799,10800],{},"InMemoryAuthorRepository"," (проста реалізація у пам'яті на основі ",[3154,10803,10804],{},"HashMap","). Напишіть тест ",[3154,10807,10808],{},"shouldFindAuthorByName()",", що перевіряє логіку пошуку без жодного підключення до бази даних.",[3150,10811,10812,10815,10816,6440,10819,3270,10821,3270,10824,10827,10828,8779],{},[3380,10813,10814],{},"Завдання 2.3 — Кастомний кваліфікатор."," Створіть анотацію ",[3154,10817,10818],{},"@Primary",[3154,10820,6443],{},[3154,10822,10823],{},"@Retention(RUNTIME)",[3154,10825,10826],{},"@Target(FIELD, PARAMETER)","). Зареєструйте у модулі дві реалізації ",[3154,10829,3407],{},[10560,10831,10832,10840],{},[3573,10833,10834,10566,10837,10839],{},[3154,10835,10836],{},"@Primary TrackRepository",[3154,10838,3598],{}," (основна)",[3573,10841,10842,10844,10845,10847],{},[3154,10843,3407],{}," без кваліфікатора → ",[3154,10846,6366],{}," (кеш)",[3150,10849,10850,10851,10853,10854,10857,10858,10860],{},"Впровадьте обидві у ",[3154,10852,3594],{}," і реалізуйте метод ",[3154,10855,10856],{},"findById(UUID id)",", що спочатку перевіряє кеш, і лише якщо запис не знайдено — звертається до ",[3154,10859,10818],{},"-репозиторію.",[3507,10862],{},[3514,10864,10866],{"id":10865},"рівень-3-архітектура-javafx-застосунок-з-guice","Рівень 3 — Архітектура: JavaFX-застосунок з Guice",[3150,10868,10869],{},"Ці завдання є послідовними кроками до реалізації повноцінного JavaFX-застосунку аудіоплатформи з Guice.",[3150,10871,10872,10875],{},[3380,10873,10874],{},"Завдання 3.1 — Базова інтеграція."," Налаштуйте JavaFX-застосунок з Guice за схемою, описаною у цій статті:",[10560,10877,10878,10890,10908,10917],{},[3573,10879,10880,10881,10884,10885,10887,10888,3351],{},"Створіть клас ",[3154,10882,10883],{},"App extends Application"," з методом ",[3154,10886,8775],{},", де ініціалізується ",[3154,10889,4137],{},[3573,10891,10892,10893,10896,10897,10900,10901,10903,10904,4510,10906,3351],{},"Реалізуйте головну сцену ",[3154,10894,10895],{},"main-view.fxml"," з контролером ",[3154,10898,10899],{},"MainController",", що має ",[3154,10902,3892],{},"-конструктор з ",[3154,10905,3594],{},[3154,10907,10606],{},[3573,10909,10910,10911,10914,10915,3351],{},"Підключіть ",[3154,10912,10913],{},"setControllerFactory(injector::getInstance)"," і переконайтесь, що сцена відображається без ",[3154,10916,7150],{},[3573,10918,10919,10920,10922],{},"Додайте лог-повідомлення в конструктор ",[3154,10921,10899],{}," та всіх сервісів — переконайтесь у правильному порядку ініціалізації.",[3150,10924,10925,10928,10929,10932],{},[3380,10926,10927],{},"Завдання 3.2 — NavigationService."," Реалізуйте ",[3154,10930,10931],{},"NavigationService @Singleton",", що підтримує такі операції:",[10560,10934,10935,10941,10947,10953],{},[3573,10936,10937,10940],{},[3154,10938,10939],{},"showTrackList()"," — відкрити сцену зі списком треків",[3573,10942,10943,10946],{},[3154,10944,10945],{},"showAuthorList()"," — відкрити сцену зі списком авторів",[3573,10948,10949,10952],{},[3154,10950,10951],{},"openAddTrackDialog()"," — відкрити модальний діалог додавання треку",[3573,10954,10955,10958,10959,10582],{},[3154,10956,10957],{},"goBack()"," — повернутись до попередньої сцени (зберігайте стек ",[3154,10960,10961],{},"Deque\u003CStage>",[3150,10963,10964,10965,10967,10968,10971,10972,10974],{},"Впровадьте ",[3154,10966,8797],{}," у три різних контролери. Переконайтесь, що всі три отримують ",[3380,10969,10970],{},"той самий"," екземпляр (",[3154,10973,4519],{},"), і навігація між сценами працює коректно.",[3150,10976,10977,10983,10984,10987],{},[3380,10978,10979,10980,3351],{},"Завдання 3.3 — Кастомний Scope ",[3154,10981,10982],{},"@PerWindow"," Реалізуйте власний Guice Scope, що гарантує: кожне нове вікно JavaFX отримує свій екземпляр певного сервісу (наприклад, ",[3154,10985,10986],{},"TrackEditState",", що зберігає стан форми редагування), але в межах одного вікна — завжди той самий. Для цього:",[10560,10989,10990,10998,11011,11017],{},[3573,10991,10992,10993,6440,10995,4975],{},"Оголосіть анотацію ",[3154,10994,10982],{},[3154,10996,10997],{},"@ScopeAnnotation",[3573,10999,11000,11001,11004,11005,4510,11008,3351],{},"Реалізуйте клас ",[3154,11002,11003],{},"WindowScope implements Scope"," з методами ",[3154,11006,11007],{},"enter()",[3154,11009,11010],{},"exit()",[3573,11012,11013,11014,3351],{},"Зареєструйте scope у модулі: ",[3154,11015,11016],{},"bindScope(PerWindow.class, windowScope)",[3573,11018,11019,11020,3351],{},"Протестуйте: відкрийте два вікна редагування і переконайтесь, що кожне має власний ",[3154,11021,10986],{},[3507,11023],{},[3145,11025,11027],{"id":11026},"підсумок","Підсумок",[3150,11029,11030,11031,11033],{},"У цій статті ми пройшли шлях від ручного зв'язування залежностей через ",[3154,11032,3245],{}," до повноцінного IoC-контейнера, інтегрованого у JavaFX-застосунок. Ключові ідеї, які варто зафіксувати:",[3460,11035,11036,11069,11100,11140],{},[3463,11037,11040],{"icon":11038,"title":11039},"i-heroicons-academic-cap","Що вивчили",[10560,11041,11042,11048,11054,11059,11064],{},[3573,11043,11044,11047],{},[3380,11045,11046],{},"DI\u002FIoC\u002FDIP"," — три концепції, що доповнюють одна одну",[3573,11049,11050,11053],{},[3380,11051,11052],{},"Guice Module"," — декларативний опис залежностей",[3573,11055,11056,11058],{},[3380,11057,4137],{}," — центральна фабрика об'єктів",[3573,11060,11061,11063],{},[3380,11062,7228],{}," — ключ до JavaFX-інтеграції",[3573,11065,11066,11068],{},[3380,11067,9225],{}," — управління lifecycle об'єктів",[3463,11070,11073],{"icon":11071,"title":11072},"i-heroicons-at-symbol","Ключові анотації",[10560,11074,11075,11080,11085,11090,11095],{},[3573,11076,11077,11079],{},[3154,11078,3892],{}," — точка впровадження (конструктор!)",[3573,11081,11082,11084],{},[3154,11083,4519],{}," — один екземпляр на Injector",[3573,11086,11087,11089],{},[3154,11088,5150],{}," — рядковий кваліфікатор",[3573,11091,11092,11094],{},[3154,11093,6443],{}," — кастомний кваліфікатор",[3573,11096,11097,11099],{},[3154,11098,6803],{}," — метод-фабрика у Module",[3463,11101,11104],{"icon":11102,"title":11103},"i-heroicons-exclamation-triangle","Типові помилки",[10560,11105,11106,11118,11123,11126,11135],{},[3573,11107,11108,11109,11111,11112,11115,11116],{},"Зберігати ",[3154,11110,4137],{}," поза ",[3154,11113,11114],{},"App"," \u002F ",[3154,11117,8797],{},[3573,11119,11120,11122],{},[3154,11121,4519],{}," на JavaFX-контролерах",[3573,11124,11125],{},"Field Injection замість Constructor Injection",[3573,11127,11128,11129,11131,11132,11134],{},"Впроваджувати ",[3154,11130,7607],{},"\u002F",[3154,11133,8846],{}," через Guice",[3573,11136,11137,11138],{},"Не відкривати пакети у ",[3154,11139,4751],{},[3463,11141,11144],{"icon":11142,"title":11143},"i-heroicons-arrow-right-circle","Що далі",[10560,11145,11146,11156,11166],{},[3573,11147,11148,4468,11152,11155],{},[3453,11149,11151],{"href":11150},"\u002F04.java\u002Fpr2\u002F11.connection-pool","Connection Pool →",[3154,11153,11154],{},"@Provides DataSource"," з HikariCP",[3573,11157,11158,11162,11163],{},[3453,11159,11161],{"href":11160},"\u002F04.java\u002Fpr2\u002F14.repository-data-mapper","Repository Pattern →"," — інтерфейси для ",[3154,11164,11165],{},"bind().to()",[3573,11167,11168,4468,11172,11174],{},[3453,11169,11171],{"href":11170},"\u002F04.java\u002Fpr2\u002F22.integration-testing-h2","Integration Testing →",[3154,11173,9624],{}," на практиці",[3447,11176,11177,11182],{},[3150,11178,11179],{},[3380,11180,11181],{},"Офіційні ресурси:",[10560,11183,11184,11191,11199],{},[3573,11185,11186,11190],{},[3453,11187,11189],{"href":3455,"rel":11188},[3457],"Google Guice Wiki"," — повна документація",[3573,11192,11193,11198],{},[3453,11194,11197],{"href":11195,"rel":11196},"https:\u002F\u002Fgithub.com\u002Fgoogle\u002Fguice\u002Fwiki\u002FBestPractices",[3457],"Guice Best Practices"," — рекомендовані патерни",[3573,11200,11201,11206,11207,3270,11209,3270,11211],{},[3453,11202,11205],{"href":11203,"rel":11204},"https:\u002F\u002Fjakarta.ee\u002Fspecifications\u002Fdependency-injection\u002F",[3457],"Jakarta Inject API"," — стандарт анотацій ",[3154,11208,3892],{},[3154,11210,4519],{},[3154,11212,5150],{},[11214,11215,11216],"style",{},"html pre.shiki code .su1O8, html code.shiki .su1O8{--shiki-light:#0000FF;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .sN1BT, html code.shiki .sN1BT{--shiki-light:#267F99;--shiki-default:#4EC9B0;--shiki-dark:#4EC9B0}html pre.shiki code .sHH4Y, html code.shiki .sHH4Y{--shiki-light:#000000;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .siwwj, html code.shiki .siwwj{--shiki-light:#001080;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .s8Opu, html code.shiki .s8Opu{--shiki-light:#795E26;--shiki-default:#DCDCAA;--shiki-dark:#DCDCAA}html pre.shiki code .s8xlr, html code.shiki .s8xlr{--shiki-light:#AF00DB;--shiki-default:#C586C0;--shiki-dark:#C586C0}html pre.shiki code .sbdoH, html code.shiki .sbdoH{--shiki-light:#A31515;--shiki-default:#CE9178;--shiki-dark:#CE9178}html .light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html.light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .spJ8K, html code.shiki .spJ8K{--shiki-light:#008000;--shiki-default:#6A9955;--shiki-dark:#6A9955}html pre.shiki code .s0P7L, html code.shiki .s0P7L{--shiki-light:#800000;--shiki-default:#808080;--shiki-dark:#808080}html pre.shiki code .sKtos, html code.shiki .sKtos{--shiki-light:#800000;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .sJj4R, html code.shiki .sJj4R{--shiki-light:#098658;--shiki-default:#B5CEA8;--shiki-dark:#B5CEA8}html pre.shiki code .sa4r_, html code.shiki .sa4r_{--shiki-light:#E50000;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .su9tN, html code.shiki .su9tN{--shiki-light:#0000FF;--shiki-default:#CE9178;--shiki-dark:#CE9178}",{"title":3164,"searchDepth":3189,"depth":3189,"links":11218},[11219,11220,11225,11228,11236,11245,11251,11258,11264,11269],{"id":3147,"depth":3189,"text":3148},{"id":3511,"depth":3189,"text":3512,"children":11221},[11222,11223,11224],{"id":3516,"depth":3196,"text":3517},{"id":3561,"depth":3196,"text":3562},{"id":3761,"depth":3196,"text":3762},{"id":4104,"depth":3189,"text":4105,"children":11226},[11227],{"id":4357,"depth":3196,"text":4358},{"id":4490,"depth":3189,"text":4491,"children":11229},[11230,11231,11233,11234,11235],{"id":4502,"depth":3196,"text":4503},{"id":4747,"depth":3196,"text":11232},"Крок 2: Налаштувати module-info.java",{"id":4874,"depth":3196,"text":4875},{"id":5154,"depth":3196,"text":5155},{"id":5445,"depth":3196,"text":5446},{"id":5704,"depth":3189,"text":5705,"children":11237},[11238,11239,11240,11241,11243],{"id":5717,"depth":3196,"text":5718},{"id":5819,"depth":3196,"text":5820},{"id":5918,"depth":3196,"text":5919},{"id":6236,"depth":3196,"text":11242},"@Named та кастомні кваліфікатори",{"id":6810,"depth":3196,"text":11244},"Метод @Provides як зручна альтернатива Provider-класу",{"id":7058,"depth":3189,"text":7059,"children":11246},[11247,11248,11250],{"id":7068,"depth":3196,"text":7069},{"id":7154,"depth":3196,"text":11249},"Рішення: setControllerFactory()",{"id":7421,"depth":3196,"text":7422},{"id":8954,"depth":3189,"text":8955,"children":11252},[11253,11254,11256,11257],{"id":8969,"depth":3196,"text":8970},{"id":9086,"depth":3196,"text":11255},"@Singleton — один на весь Injector",{"id":9242,"depth":3196,"text":9243},{"id":9447,"depth":3196,"text":9448},{"id":9519,"depth":3189,"text":9520,"children":11259},[11260,11261,11263],{"id":9530,"depth":3196,"text":9531},{"id":9621,"depth":3196,"text":11262},"Modules.override() — хірургічна підміна",{"id":10168,"depth":3196,"text":10169},{"id":10534,"depth":3189,"text":10535,"children":11265},[11266,11267,11268],{"id":10541,"depth":3196,"text":10542},{"id":10730,"depth":3196,"text":10731},{"id":10865,"depth":3196,"text":10866},{"id":11026,"depth":3189,"text":11027},"Від ручного зв'язування об'єктів до повноцінного IoC-контейнера: архітектура Google Guice, анотаційне впровадження, модулі, скопи та інтеграція з JavaFX через ControllerFactory.","md",null,{},{"title":2410,"description":11270},"HZzvdfG66MHBRO2FujNsoVqABXGTNT2V-LY2QiAKshM",[11277,11279],{"title":2406,"path":2407,"stem":2408,"description":11278,"children":-1},"Від емуляції до реальності: архітектура Testcontainers, lifecycle Docker-контейнерів у тестах, тестування PostgreSQL-специфічних функцій (ENUM, JSON, full-text search), паралельне виконання тестів та інтеграція з CI\u002FCD.",{"title":2414,"path":2415,"stem":2416,"description":11280,"children":-1},"Від консольних додатків до вікон з кнопками: архітектура JavaFX, Scene Graph, життєвий цикл Application, layout-контейнери та FXML як декларативний опис UI.",1778998388345]