<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom"><channel><title>Tutorial on TECHFOR by Suriya Sonphu</title><link>http://suriyasonphu.com/categories/tutorial/</link><description>Recent content in Tutorial on TECHFOR by Suriya Sonphu</description><generator>Hugo -- gohugo.io</generator><language>th</language><lastBuildDate>Fri, 08 Aug 2025 00:00:00 +0000</lastBuildDate><atom:link href="http://suriyasonphu.com/categories/tutorial/index.xml" rel="self" type="application/rss+xml"/><item><title>การสร้าง Full Stack Web Application: Vue.js + Tailwind CSS + ASP.NET Core 9 Minimal API ด้วย Modular Monolith Architecture</title><link>http://suriyasonphu.com/post/2025-08-08-fullstack-vue-tailwind-dotnet-modular-monolith/</link><pubDate>Fri, 08 Aug 2025 00:00:00 +0000</pubDate><guid>http://suriyasonphu.com/post/2025-08-08-fullstack-vue-tailwind-dotnet-modular-monolith/</guid><description>&lt;img src="http://suriyasonphu.com/post/2025-08-08-fullstack-vue-tailwind-dotnet-modular-monolith/fullstack-cover.svg" alt="Featured image of post การสร้าง Full Stack Web Application: Vue.js + Tailwind CSS + ASP.NET Core 9 Minimal API ด้วย Modular Monolith Architecture" />&lt;h2 id="บทนำ-full-stack-development-คออะไร">บทนำ: Full Stack Development คืออะไร?
&lt;/h2>&lt;p>&lt;strong>Full Stack Development&lt;/strong> หมายถึงการพัฒนาทั้งส่วน frontend (ฝั่งผู้ใช้) และ backend (ฝั่งเซิร์ฟเวอร์) ของเว็บแอปพลิเคชัน นักพัฒนา full stack คือผู้ที่สามารถทำงานได้ทั้งส่วนของ user interface ที่ผู้ใช้มองเห็นและโต้ตอบได้ และส่วนของ server logic ที่ทำงานเบื้องหลัง&lt;/p>
&lt;h3 id="องคประกอบหลกของ-full-stack-development">องค์ประกอบหลักของ Full Stack Development:
&lt;/h3>&lt;ul>
&lt;li>
&lt;p>&lt;strong>Frontend (Client-side)&lt;/strong>: สิ่งที่ผู้ใช้เห็นและโต้ตอบ&lt;/p>
&lt;ul>
&lt;li>User Interface (UI)&lt;/li>
&lt;li>User Experience (UX)&lt;/li>
&lt;li>Client-side logic และการทำงานโต้ตอบ&lt;/li>
&lt;/ul>
&lt;/li>
&lt;li>
&lt;p>&lt;strong>Backend (Server-side)&lt;/strong>: เครื่องยนต์ที่ขับเคลื่อนแอปพลิเคชัน&lt;/p>
&lt;ul>
&lt;li>Server logic และ APIs&lt;/li>
&lt;li>การจัดการฐานข้อมูล&lt;/li>
&lt;li>Authentication และความปลอดภัย&lt;/li>
&lt;li>Business logic&lt;/li>
&lt;/ul>
&lt;/li>
&lt;li>
&lt;p>&lt;strong>Database&lt;/strong>: ที่เก็บและจัดการข้อมูล&lt;/p>
&lt;/li>
&lt;li>
&lt;p>&lt;strong>DevOps&lt;/strong>: การ deploy และจัดการ infrastructure&lt;/p>
&lt;/li>
&lt;/ul>
&lt;p>ในบทความนี้ เราจะสร้าง full stack application ที่สมบูรณ์โดยใช้เทคโนโลยีสมัยใหม่: &lt;strong>Vue.js&lt;/strong> สำหรับ frontend, &lt;strong>Tailwind CSS&lt;/strong> สำหรับการจัดแต่งหน้าตา และ &lt;strong>ASP.NET Core 9 Minimal API&lt;/strong> สำหรับ backend โดยจัดระบบด้วย &lt;strong>Modular Monolith&lt;/strong> architecture&lt;/p>
&lt;h2 id="vuejs-คออะไร">Vue.js คืออะไร?
&lt;/h2>&lt;p>&lt;strong>Vue.js&lt;/strong> เป็น progressive JavaScript framework สำหรับสร้าง user interfaces และ single-page applications (SPAs) สร้างโดย Evan You ในปี 2014 Vue.js ได้กลายเป็นหนึ่งใน frontend frameworks ที่ได้รับความนิยมมากที่สุด เคียงข้างกับ React และ Angular&lt;/p>
&lt;h3 id="คณสมบตหลกของ-vuejs">คุณสมบัติหลักของ Vue.js:
&lt;/h3>&lt;ol>
&lt;li>&lt;strong>Progressive Framework&lt;/strong>: สามารถใช้ Vue.js แบบค่อยเป็นค่อยไป - ใช้เพียงส่วนหนึ่งของหน้าเว็บหรือสร้างแอปพลิเคชันทั้งหมด&lt;/li>
&lt;li>&lt;strong>Reactive Data Binding&lt;/strong>: UI อัปเดตอัตโนมัติเมื่อข้อมูลเปลี่ยน&lt;/li>
&lt;li>&lt;strong>Component-Based Architecture&lt;/strong>: สร้าง components ที่ห่อหุ้มและจัดการ state ของตัวเอง&lt;/li>
&lt;li>&lt;strong>Virtual DOM&lt;/strong>: การ render ที่มีประสิทธิภาพสำหรับความเร็วที่เหมาะสม&lt;/li>
&lt;li>&lt;strong>ง่ายต่อการเรียนรู้&lt;/strong>: syntax ที่เรียบง่าย เข้าใจง่ายสำหรับผู้เริ่มต้น&lt;/li>
&lt;/ol>
&lt;h3 id="ขอดของ-vuejs">ข้อดีของ Vue.js:
&lt;/h3>&lt;ul>
&lt;li>&lt;strong>ขนาด bundle เล็ก&lt;/strong>: โหลดเร็ว&lt;/li>
&lt;li>&lt;strong>เอกสารที่เยี่ยม&lt;/strong>: ครอบคลุมและเข้าใจง่ายสำหรับผู้เริ่มต้น&lt;/li>
&lt;li>&lt;strong>ระบบนิเวศที่ดี&lt;/strong>: เครื่องมือและไลบรารีที่หลากหลาย&lt;/li>
&lt;li>&lt;strong>รองรับ TypeScript&lt;/strong>: การผสานรวม TypeScript อย่างเต็มรูปแบบ&lt;/li>
&lt;li>&lt;strong>Composition API&lt;/strong>: วิธีสมัยใหม่ในการจัดระเบียบ component logic&lt;/li>
&lt;/ul>
&lt;h3 id="ตวอยาง-vuejs-component">ตัวอย่าง Vue.js Component:
&lt;/h3>&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-vue" data-lang="vue">&lt;span style="display:flex;">&lt;span>&amp;lt;&lt;span style="color:#f92672">template&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">div&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;user-profile&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">h2&lt;/span>&amp;gt;{{ &lt;span style="color:#a6e22e">user&lt;/span>.&lt;span style="color:#a6e22e">name&lt;/span> }}&amp;lt;/&lt;span style="color:#f92672">h2&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">p&lt;/span>&amp;gt;{{ &lt;span style="color:#a6e22e">user&lt;/span>.&lt;span style="color:#a6e22e">email&lt;/span> }}&amp;lt;/&lt;span style="color:#f92672">p&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">button&lt;/span> &lt;span style="color:#f92672">@click&lt;/span>&lt;span style="color:#e6db74">=&amp;#34;updateProfile&amp;#34;&lt;/span>&amp;gt;&lt;span style="color:#a6e22e">อัปเดตโปรไฟล์&lt;/span>&amp;lt;/&lt;span style="color:#f92672">button&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">div&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;/&lt;span style="color:#f92672">template&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;&lt;span style="color:#f92672">script&lt;/span> &lt;span style="color:#a6e22e">setup&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">import&lt;/span> { &lt;span style="color:#a6e22e">ref&lt;/span>, &lt;span style="color:#a6e22e">reactive&lt;/span> } &lt;span style="color:#a6e22e">from&lt;/span> &lt;span style="color:#e6db74">&amp;#39;vue&amp;#39;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">const&lt;/span> &lt;span style="color:#a6e22e">user&lt;/span> &lt;span style="color:#f92672">=&lt;/span> &lt;span style="color:#a6e22e">reactive&lt;/span>({
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#a6e22e">name&lt;/span>&lt;span style="color:#f92672">:&lt;/span> &lt;span style="color:#e6db74">&amp;#39;สมชาย ใจดี&amp;#39;&lt;/span>,
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#a6e22e">email&lt;/span>&lt;span style="color:#f92672">:&lt;/span> &lt;span style="color:#e6db74">&amp;#39;somchai@example.com&amp;#39;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>})
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">const&lt;/span> &lt;span style="color:#a6e22e">updateProfile&lt;/span> &lt;span style="color:#f92672">=&lt;/span> () =&amp;gt; {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#75715e">// อัปเดตโปรไฟล์ผู้ใช้
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">&lt;/span> &lt;span style="color:#a6e22e">console&lt;/span>.&lt;span style="color:#a6e22e">log&lt;/span>(&lt;span style="color:#e6db74">&amp;#39;โปรไฟล์ถูกอัปเดตแล้ว!&amp;#39;&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;/&lt;span style="color:#f92672">script&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>อ้างอิง&lt;/strong>: &lt;a class="link" href="https://vuejs.org/" target="_blank" rel="noopener"
>Vue.js Official Documentation&lt;/a>&lt;/p>
&lt;h2 id="tailwind-css-คออะไร">Tailwind CSS คืออะไร?
&lt;/h2>&lt;p>&lt;strong>Tailwind CSS&lt;/strong> เป็น utility-first CSS framework ที่ให้ utility classes ระดับต่ำสำหรับสร้างการออกแบบที่กำหนดเองได้โดยตรงใน markup แทนที่จะเขียน custom CSS คุณจะประกอบการออกแบบด้วยการรวม utility classes ขนาดเล็กที่มีจุดประสงค์เดียว&lt;/p>
&lt;h3 id="คณสมบตหลกของ-tailwind-css">คุณสมบัติหลักของ Tailwind CSS:
&lt;/h3>&lt;ol>
&lt;li>&lt;strong>Utility-First&lt;/strong>: classes ที่สร้างมาล่วงหน้าสำหรับ CSS properties ทั่วไป&lt;/li>
&lt;li>&lt;strong>Responsive Design&lt;/strong>: utilities สำหรับ responsive design ที่สร้างไว้แล้ว&lt;/li>
&lt;li>&lt;strong>ปรับแต่งได้&lt;/strong>: ระบบการออกแบบที่กำหนดค่าได้สูง&lt;/li>
&lt;li>&lt;strong>ประสิทธิภาพ&lt;/strong>: ลบ CSS ที่ไม่ใช้อัตโนมัติ&lt;/li>
&lt;li>&lt;strong>ประสบการณ์นักพัฒนา&lt;/strong>: รองรับ IDE และเครื่องมือ debugging ที่เยี่ยม&lt;/li>
&lt;/ol>
&lt;h3 id="ขอดของ-tailwind-css">ข้อดีของ Tailwind CSS:
&lt;/h3>&lt;ul>
&lt;li>&lt;strong>การพัฒนาอย่างรวดเร็ว&lt;/strong>: สร้าง interfaces ได้เร็วโดยไม่ต้องเขียน custom CSS&lt;/li>
&lt;li>&lt;strong>การออกแบบที่สอดคล้อง&lt;/strong>: ระบบการออกแบบที่สร้างไว้แล้วรับประกันความสอดคล้อง&lt;/li>
&lt;li>&lt;strong>Mobile-First&lt;/strong>: responsive design ตั้งแต่ต้น&lt;/li>
&lt;li>&lt;strong>ขนาด Bundle เล็ก&lt;/strong>: รวมเฉพาะ CSS ที่คุณใช้จริง&lt;/li>
&lt;li>&lt;strong>ไม่ต้องเปลี่ยนบริบท&lt;/strong>: จัดแต่ง components โดยไม่ต้องออกจาก HTML&lt;/li>
&lt;/ul>
&lt;h3 id="ตวอยางการใช-tailwind-css">ตัวอย่างการใช้ Tailwind CSS:
&lt;/h3>&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-html" data-lang="html">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">&amp;lt;!-- วิธี CSS แบบดั้งเดิม --&amp;gt;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;&lt;span style="color:#f92672">div&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;card&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">h2&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;card-title&amp;#34;&lt;/span>&amp;gt;สวัสดีครับ&amp;lt;/&lt;span style="color:#f92672">h2&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">p&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;card-description&amp;#34;&lt;/span>&amp;gt;นี่คือคำอธิบาย&amp;lt;/&lt;span style="color:#f92672">p&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">button&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;btn btn-primary&amp;#34;&lt;/span>&amp;gt;คลิกที่นี่&amp;lt;/&lt;span style="color:#f92672">button&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;/&lt;span style="color:#f92672">div&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">&amp;lt;!-- วิธี Tailwind CSS --&amp;gt;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;&lt;span style="color:#f92672">div&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;bg-white p-6 rounded-lg shadow-md&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">h2&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;text-2xl font-bold text-gray-900 mb-2&amp;#34;&lt;/span>&amp;gt;สวัสดีครับ&amp;lt;/&lt;span style="color:#f92672">h2&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">p&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;text-gray-600 mb-4&amp;#34;&lt;/span>&amp;gt;นี่คือคำอธิบาย&amp;lt;/&lt;span style="color:#f92672">p&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">button&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> คลิกที่นี่
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">button&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;/&lt;span style="color:#f92672">div&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="tailwind-css-classes-ทใชบอย">Tailwind CSS Classes ที่ใช้บ่อย:
&lt;/h3>&lt;ul>
&lt;li>&lt;strong>Layout&lt;/strong>: &lt;code>flex&lt;/code>, &lt;code>grid&lt;/code>, &lt;code>container&lt;/code>, &lt;code>mx-auto&lt;/code>&lt;/li>
&lt;li>&lt;strong>Spacing&lt;/strong>: &lt;code>p-4&lt;/code> (padding), &lt;code>m-2&lt;/code> (margin), &lt;code>space-y-4&lt;/code>&lt;/li>
&lt;li>&lt;strong>Typography&lt;/strong>: &lt;code>text-xl&lt;/code>, &lt;code>font-bold&lt;/code>, &lt;code>text-center&lt;/code>&lt;/li>
&lt;li>&lt;strong>Colors&lt;/strong>: &lt;code>bg-blue-500&lt;/code>, &lt;code>text-red-600&lt;/code>, &lt;code>border-gray-300&lt;/code>&lt;/li>
&lt;li>&lt;strong>Responsive&lt;/strong>: &lt;code>sm:text-lg&lt;/code>, &lt;code>md:grid-cols-2&lt;/code>, &lt;code>lg:px-8&lt;/code>&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>อ้างอิง&lt;/strong>: &lt;a class="link" href="https://tailwindcss.com/" target="_blank" rel="noopener"
>Tailwind CSS Official Documentation&lt;/a>&lt;/p>
&lt;h2 id="aspnet-core-9-minimal-api-คออะไร">ASP.NET Core 9 Minimal API คืออะไร?
&lt;/h2>&lt;p>&lt;strong>ASP.NET Core Minimal API&lt;/strong> เป็นแนวทางที่เรียบง่ายในการสร้าง HTTP APIs ด้วย ASP.NET Core เปิดตัวใน .NET 6 และปรับปรุงในเวอร์ชันต่อๆ มารวมถึง .NET 9 Minimal APIs ช่วยให้คุณสร้าง APIs ได้โดยใช้ code น้อยและไม่ซับซ้อน&lt;/p>
&lt;h3 id="คณสมบตหลกของ-minimal-api">คุณสมบัติหลักของ Minimal API:
&lt;/h3>&lt;ol>
&lt;li>&lt;strong>Boilerplate น้อย&lt;/strong>: เขียน code น้อยกว่าแบบ controllers ดั้งเดิม&lt;/li>
&lt;li>&lt;strong>ประสิทธิภาพ&lt;/strong>: ประสิทธิภาพสูงด้วย overhead ที่น้อย&lt;/li>
&lt;li>&lt;strong>Syntax เรียบง่าย&lt;/strong>: เข้าใจและเขียนง่าย&lt;/li>
&lt;li>&lt;strong>คุณสมบัติที่สร้างไว้&lt;/strong>: model binding, validation และรองรับ OpenAPI อัตโนมัติ&lt;/li>
&lt;li>&lt;strong>Hosting&lt;/strong>: ผสานรวมกับ ASP.NET Core hosting model&lt;/li>
&lt;/ol>
&lt;h3 id="ขอดของ-minimal-api">ข้อดีของ Minimal API:
&lt;/h3>&lt;ul>
&lt;li>&lt;strong>การพัฒนาอย่างรวดเร็ว&lt;/strong>: สร้าง APIs ได้เร็วด้วยการตั้งค่าที่น้อย&lt;/li>
&lt;li>&lt;strong>เหมาะกับ Microservices&lt;/strong>: เหมาะสำหรับ services ขนาดเล็กที่มีจุดมุ่งหมายเฉพาะ&lt;/li>
&lt;li>&lt;strong>Cloud Native&lt;/strong>: เหมาะสำหรับสภาพแวดล้อม containerized&lt;/li>
&lt;li>&lt;strong>คุณสมบัติ C# สมัยใหม่&lt;/strong>: ใช้ประโยชน์จากคุณสมบัติล่าสุดของภาษา C#&lt;/li>
&lt;li>&lt;strong>การผสานรวม&lt;/strong>: ทำงานร่วมกับไลบรารี .NET อื่นๆ ได้อย่างราบรื่น&lt;/li>
&lt;/ul>
&lt;h3 id="ตวอยาง-minimal-api">ตัวอย่าง Minimal API:
&lt;/h3>&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">using&lt;/span> Microsoft.EntityFrameworkCore;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">var&lt;/span> builder = WebApplication.CreateBuilder(args);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// เพิ่ม services ลงใน container&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>builder.Services.AddDbContext&amp;lt;AppDbContext&amp;gt;(options =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> options.UseSqlite(&lt;span style="color:#e6db74">&amp;#34;DataSource=app.db&amp;#34;&lt;/span>));
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>builder.Services.AddEndpointsApiExplorer();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>builder.Services.AddSwaggerGen();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">var&lt;/span> app = builder.Build();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// กำหนดค่า HTTP request pipeline&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">if&lt;/span> (app.Environment.IsDevelopment())
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> app.UseSwagger();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> app.UseSwaggerUI();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.UseHttpsRedirection();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// กำหนด API endpoints&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.MapGet(&lt;span style="color:#e6db74">&amp;#34;/api/products&amp;#34;&lt;/span>, &lt;span style="color:#66d9ef">async&lt;/span> (AppDbContext db) =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">await&lt;/span> db.Products.ToListAsync())
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .WithName(&lt;span style="color:#e6db74">&amp;#34;GetProducts&amp;#34;&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .WithOpenApi();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.MapGet(&lt;span style="color:#e6db74">&amp;#34;/api/products/{id}&amp;#34;&lt;/span>, &lt;span style="color:#66d9ef">async&lt;/span> (&lt;span style="color:#66d9ef">int&lt;/span> id, AppDbContext db) =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">await&lt;/span> db.Products.FindAsync(id) &lt;span style="color:#66d9ef">is&lt;/span> Product product
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> ? Results.Ok(product)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> : Results.NotFound())
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .WithName(&lt;span style="color:#e6db74">&amp;#34;GetProduct&amp;#34;&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .WithOpenApi();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.MapPost(&lt;span style="color:#e6db74">&amp;#34;/api/products&amp;#34;&lt;/span>, &lt;span style="color:#66d9ef">async&lt;/span> (Product product, AppDbContext db) =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> db.Products.Add(product);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">await&lt;/span> db.SaveChangesAsync();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> Results.Created(&lt;span style="color:#e6db74">$&amp;#34;/api/products/{product.Id}&amp;#34;&lt;/span>, product);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>})
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>.WithName(&lt;span style="color:#e6db74">&amp;#34;CreateProduct&amp;#34;&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>.WithOpenApi();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.Run();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// Models&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">Product&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">int&lt;/span> Id { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">string&lt;/span> Name { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">set&lt;/span>; } = &lt;span style="color:#66d9ef">string&lt;/span>.Empty;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">decimal&lt;/span> Price { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">string&lt;/span> Description { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">set&lt;/span>; } = &lt;span style="color:#66d9ef">string&lt;/span>.Empty;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">AppDbContext&lt;/span> : DbContext
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> AppDbContext(DbContextOptions&amp;lt;AppDbContext&amp;gt; options) : &lt;span style="color:#66d9ef">base&lt;/span>(options) { }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> DbSet&amp;lt;Product&amp;gt; Products { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>อ้างอิง&lt;/strong>: &lt;a class="link" href="https://docs.microsoft.com/en-us/aspnet/core/fundamentals/minimal-apis" target="_blank" rel="noopener"
>ASP.NET Core Minimal APIs Documentation&lt;/a>&lt;/p>
&lt;h2 id="ตวอยางโปรเจค-e-commerce-product-catalog">ตัวอย่างโปรเจค: E-commerce Product Catalog
&lt;/h2>&lt;p>มาสร้าง e-commerce product catalog อย่างง่ายที่แสดงให้เห็นว่าเทคโนโลยีทั้งหมดนี้ทำงานร่วมกันอย่างไรใน modular monolith architecture&lt;/p>
&lt;h3 id="โครงสรางโปรเจค">โครงสร้างโปรเจค:
&lt;/h3>&lt;pre tabindex="0">&lt;code>ECommerceApp/
├── frontend/ # Vue.js + Tailwind CSS
│ ├── src/
│ │ ├── components/
│ │ ├── views/
│ │ ├── services/
│ │ └── main.js
│ ├── index.html
│ └── package.json
├── backend/ # ASP.NET Core 9 Minimal API
│ ├── Modules/
│ │ ├── Products/
│ │ ├── Orders/
│ │ └── Users/
│ ├── Shared/
│ │ ├── Database/
│ │ └── Infrastructure/
│ ├── Program.cs
│ └── ECommerceApp.csproj
└── README.md
&lt;/code>&lt;/pre>&lt;h3 id="การตดตง-frontend-vuejs--tailwind">การติดตั้ง Frontend (Vue.js + Tailwind):
&lt;/h3>&lt;p>&lt;strong>ProductList.vue:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-vue" data-lang="vue">&lt;span style="display:flex;">&lt;span>&amp;lt;&lt;span style="color:#f92672">template&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">div&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;container mx-auto px-4 py-8&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">h1&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;text-3xl font-bold text-gray-900 mb-8&amp;#34;&lt;/span>&amp;gt;&lt;span style="color:#a6e22e">แคตตาล&lt;/span>&lt;span style="color:#960050;background-color:#1e0010">็&lt;/span>&lt;span style="color:#a6e22e">อกส&lt;/span>&lt;span style="color:#960050;background-color:#1e0010">ิ&lt;/span>&lt;span style="color:#a6e22e">นค&lt;/span>&lt;span style="color:#960050;background-color:#1e0010">้&lt;/span>&lt;span style="color:#a6e22e">า&lt;/span>&amp;lt;/&lt;span style="color:#f92672">h1&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">div&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">div&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#f92672">v-for&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;product in products&amp;#34;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#f92672">:key&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;product.id&amp;#34;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;bg-white rounded-lg shadow-md overflow-hidden hover:shadow-lg transition-shadow&amp;#34;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">div&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;p-6&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">h3&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;text-xl font-semibold text-gray-900 mb-2&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {{ &lt;span style="color:#a6e22e">product&lt;/span>.&lt;span style="color:#a6e22e">name&lt;/span> }}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">h3&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">p&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;text-gray-600 mb-4&amp;#34;&lt;/span>&amp;gt;{{ &lt;span style="color:#a6e22e">product&lt;/span>.&lt;span style="color:#a6e22e">description&lt;/span> }}&amp;lt;/&lt;span style="color:#f92672">p&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">div&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;flex justify-between items-center&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">span&lt;/span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;text-2xl font-bold text-blue-600&amp;#34;&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#960050;background-color:#1e0010">฿&lt;/span>{{ &lt;span style="color:#a6e22e">product&lt;/span>.&lt;span style="color:#a6e22e">price&lt;/span> }}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">span&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;&lt;span style="color:#f92672">button&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#f92672">@click&lt;/span>&lt;span style="color:#e6db74">=&amp;#34;addToCart(product)&amp;#34;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#a6e22e">class&lt;/span>&lt;span style="color:#f92672">=&lt;/span>&lt;span style="color:#e6db74">&amp;#34;bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded&amp;#34;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#a6e22e">เพ&lt;/span>&lt;span style="color:#960050;background-color:#1e0010">ิ่&lt;/span>&lt;span style="color:#a6e22e">มลงตะกร&lt;/span>&lt;span style="color:#960050;background-color:#1e0010">้&lt;/span>&lt;span style="color:#a6e22e">า&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">button&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">div&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">div&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">div&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">div&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &amp;lt;/&lt;span style="color:#f92672">div&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;/&lt;span style="color:#f92672">template&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;&lt;span style="color:#f92672">script&lt;/span> &lt;span style="color:#a6e22e">setup&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">import&lt;/span> { &lt;span style="color:#a6e22e">ref&lt;/span>, &lt;span style="color:#a6e22e">onMounted&lt;/span> } &lt;span style="color:#a6e22e">from&lt;/span> &lt;span style="color:#e6db74">&amp;#39;vue&amp;#39;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">import&lt;/span> { &lt;span style="color:#a6e22e">productService&lt;/span> } &lt;span style="color:#a6e22e">from&lt;/span> &lt;span style="color:#e6db74">&amp;#39;../services/productService&amp;#39;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">const&lt;/span> &lt;span style="color:#a6e22e">products&lt;/span> &lt;span style="color:#f92672">=&lt;/span> &lt;span style="color:#a6e22e">ref&lt;/span>([])
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">const&lt;/span> &lt;span style="color:#a6e22e">fetchProducts&lt;/span> &lt;span style="color:#f92672">=&lt;/span> &lt;span style="color:#a6e22e">async&lt;/span> () =&amp;gt; {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">try&lt;/span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">const&lt;/span> &lt;span style="color:#a6e22e">response&lt;/span> &lt;span style="color:#f92672">=&lt;/span> &lt;span style="color:#a6e22e">await&lt;/span> &lt;span style="color:#a6e22e">productService&lt;/span>.&lt;span style="color:#a6e22e">getAll&lt;/span>()
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#a6e22e">products&lt;/span>.&lt;span style="color:#a6e22e">value&lt;/span> &lt;span style="color:#f92672">=&lt;/span> &lt;span style="color:#a6e22e">response&lt;/span>.&lt;span style="color:#a6e22e">data&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> } &lt;span style="color:#66d9ef">catch&lt;/span> (&lt;span style="color:#a6e22e">error&lt;/span>) {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#a6e22e">console&lt;/span>.&lt;span style="color:#a6e22e">error&lt;/span>(&lt;span style="color:#e6db74">&amp;#39;เกิดข้อผิดพลาดในการดึงข้อมูลสินค้า:&amp;#39;&lt;/span>, &lt;span style="color:#a6e22e">error&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">const&lt;/span> &lt;span style="color:#a6e22e">addToCart&lt;/span> &lt;span style="color:#f92672">=&lt;/span> (&lt;span style="color:#a6e22e">product&lt;/span>) =&amp;gt; {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#75715e">// เพิ่มลงตะกร้า logic
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">&lt;/span> &lt;span style="color:#a6e22e">console&lt;/span>.&lt;span style="color:#a6e22e">log&lt;/span>(&lt;span style="color:#e6db74">&amp;#39;เพิ่มลงตะกร้า:&amp;#39;&lt;/span>, &lt;span style="color:#a6e22e">product&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#a6e22e">onMounted&lt;/span>(() =&amp;gt; {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#a6e22e">fetchProducts&lt;/span>()
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>})
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&amp;lt;/&lt;span style="color:#f92672">script&lt;/span>&amp;gt;
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>Product Service:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-javascript" data-lang="javascript">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// services/productService.js
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">&lt;/span>&lt;span style="color:#66d9ef">import&lt;/span> &lt;span style="color:#a6e22e">axios&lt;/span> &lt;span style="color:#a6e22e">from&lt;/span> &lt;span style="color:#e6db74">&amp;#39;axios&amp;#39;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">const&lt;/span> &lt;span style="color:#a6e22e">API_BASE_URL&lt;/span> &lt;span style="color:#f92672">=&lt;/span> &lt;span style="color:#e6db74">&amp;#39;https://localhost:7070/api&amp;#39;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">export&lt;/span> &lt;span style="color:#66d9ef">const&lt;/span> &lt;span style="color:#a6e22e">productService&lt;/span> &lt;span style="color:#f92672">=&lt;/span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">async&lt;/span> &lt;span style="color:#a6e22e">getAll&lt;/span>() {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> &lt;span style="color:#66d9ef">await&lt;/span> &lt;span style="color:#a6e22e">axios&lt;/span>.&lt;span style="color:#a6e22e">get&lt;/span>(&lt;span style="color:#e6db74">`&lt;/span>&lt;span style="color:#e6db74">${&lt;/span>&lt;span style="color:#a6e22e">API_BASE_URL&lt;/span>&lt;span style="color:#e6db74">}&lt;/span>&lt;span style="color:#e6db74">/products`&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> },
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">async&lt;/span> &lt;span style="color:#a6e22e">getById&lt;/span>(&lt;span style="color:#a6e22e">id&lt;/span>) {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> &lt;span style="color:#66d9ef">await&lt;/span> &lt;span style="color:#a6e22e">axios&lt;/span>.&lt;span style="color:#a6e22e">get&lt;/span>(&lt;span style="color:#e6db74">`&lt;/span>&lt;span style="color:#e6db74">${&lt;/span>&lt;span style="color:#a6e22e">API_BASE_URL&lt;/span>&lt;span style="color:#e6db74">}&lt;/span>&lt;span style="color:#e6db74">/products/&lt;/span>&lt;span style="color:#e6db74">${&lt;/span>&lt;span style="color:#a6e22e">id&lt;/span>&lt;span style="color:#e6db74">}&lt;/span>&lt;span style="color:#e6db74">`&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> },
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">async&lt;/span> &lt;span style="color:#a6e22e">create&lt;/span>(&lt;span style="color:#a6e22e">product&lt;/span>) {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> &lt;span style="color:#66d9ef">await&lt;/span> &lt;span style="color:#a6e22e">axios&lt;/span>.&lt;span style="color:#a6e22e">post&lt;/span>(&lt;span style="color:#e6db74">`&lt;/span>&lt;span style="color:#e6db74">${&lt;/span>&lt;span style="color:#a6e22e">API_BASE_URL&lt;/span>&lt;span style="color:#e6db74">}&lt;/span>&lt;span style="color:#e6db74">/products`&lt;/span>, &lt;span style="color:#a6e22e">product&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="การตดตง-backend-aspnet-core-9-minimal-api">การติดตั้ง Backend (ASP.NET Core 9 Minimal API):
&lt;/h3>&lt;p>&lt;strong>Program.cs:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">using&lt;/span> Microsoft.EntityFrameworkCore;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">using&lt;/span> ECommerceApp.Modules.Products;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">using&lt;/span> ECommerceApp.Modules.Orders;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">using&lt;/span> ECommerceApp.Shared.Database;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">var&lt;/span> builder = WebApplication.CreateBuilder(args);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// เพิ่ม services ลงใน container&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>builder.Services.AddDbContext&amp;lt;AppDbContext&amp;gt;(options =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> options.UseSqlite(builder.Configuration.GetConnectionString(&lt;span style="color:#e6db74">&amp;#34;DefaultConnection&amp;#34;&lt;/span>)));
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// เพิ่ม CORS&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>builder.Services.AddCors(options =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> options.AddPolicy(&lt;span style="color:#e6db74">&amp;#34;AllowFrontend&amp;#34;&lt;/span>, policy =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> policy.WithOrigins(&lt;span style="color:#e6db74">&amp;#34;http://localhost:3000&amp;#34;&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .AllowAnyHeader()
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .AllowAnyMethod();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> });
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>});
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>builder.Services.AddEndpointsApiExplorer();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>builder.Services.AddSwaggerGen();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// ลงทะเบียน module services&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>builder.Services.AddProductModule();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>builder.Services.AddOrderModule();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">var&lt;/span> app = builder.Build();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// กำหนดค่า HTTP request pipeline&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">if&lt;/span> (app.Environment.IsDevelopment())
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> app.UseSwagger();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> app.UseSwaggerUI();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.UseHttpsRedirection();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.UseCors(&lt;span style="color:#e6db74">&amp;#34;AllowFrontend&amp;#34;&lt;/span>);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// แมป module endpoints&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.MapProductEndpoints();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.MapOrderEndpoints();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>app.Run();
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>Product Module:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// Modules/Products/ProductModule.cs&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">using&lt;/span> Microsoft.EntityFrameworkCore;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">using&lt;/span> ECommerceApp.Shared.Database;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">namespace&lt;/span> ECommerceApp.Modules.Products;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">ProductModule&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> IServiceCollection AddProductModule(&lt;span style="color:#66d9ef">this&lt;/span> IServiceCollection services)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> services.AddScoped&amp;lt;IProductService, ProductService&amp;gt;();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> services;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> WebApplication MapProductEndpoints(&lt;span style="color:#66d9ef">this&lt;/span> WebApplication app)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> &lt;span style="color:#66d9ef">group&lt;/span> = app.MapGroup(&lt;span style="color:#e6db74">&amp;#34;/api/products&amp;#34;&lt;/span>).WithTags(&lt;span style="color:#e6db74">&amp;#34;Products&amp;#34;&lt;/span>);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">group&lt;/span>.MapGet(&lt;span style="color:#e6db74">&amp;#34;/&amp;#34;&lt;/span>, GetProducts);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">group&lt;/span>.MapGet(&lt;span style="color:#e6db74">&amp;#34;/{id}&amp;#34;&lt;/span>, GetProduct);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">group&lt;/span>.MapPost(&lt;span style="color:#e6db74">&amp;#34;/&amp;#34;&lt;/span>, CreateProduct);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">group&lt;/span>.MapPut(&lt;span style="color:#e6db74">&amp;#34;/{id}&amp;#34;&lt;/span>, UpdateProduct);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">group&lt;/span>.MapDelete(&lt;span style="color:#e6db74">&amp;#34;/{id}&amp;#34;&lt;/span>, DeleteProduct);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> app;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task&amp;lt;IResult&amp;gt; GetProducts(IProductService productService)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> products = &lt;span style="color:#66d9ef">await&lt;/span> productService.GetAllAsync();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> Results.Ok(products);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task&amp;lt;IResult&amp;gt; GetProduct(&lt;span style="color:#66d9ef">int&lt;/span> id, IProductService productService)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> product = &lt;span style="color:#66d9ef">await&lt;/span> productService.GetByIdAsync(id);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> product != &lt;span style="color:#66d9ef">null&lt;/span> ? Results.Ok(product) : Results.NotFound();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task&amp;lt;IResult&amp;gt; CreateProduct(Product product, IProductService productService)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> createdProduct = &lt;span style="color:#66d9ef">await&lt;/span> productService.CreateAsync(product);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> Results.Created(&lt;span style="color:#e6db74">$&amp;#34;/api/products/{createdProduct.Id}&amp;#34;&lt;/span>, createdProduct);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task&amp;lt;IResult&amp;gt; UpdateProduct(&lt;span style="color:#66d9ef">int&lt;/span> id, Product product, IProductService productService)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">if&lt;/span> (id != product.Id) &lt;span style="color:#66d9ef">return&lt;/span> Results.BadRequest();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> updatedProduct = &lt;span style="color:#66d9ef">await&lt;/span> productService.UpdateAsync(product);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> updatedProduct != &lt;span style="color:#66d9ef">null&lt;/span> ? Results.Ok(updatedProduct) : Results.NotFound();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task&amp;lt;IResult&amp;gt; DeleteProduct(&lt;span style="color:#66d9ef">int&lt;/span> id, IProductService productService)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> success = &lt;span style="color:#66d9ef">await&lt;/span> productService.DeleteAsync(id);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> success ? Results.NoContent() : Results.NotFound();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h2 id="ทำไม-modular-monolith-ถงสำคญ">ทำไม Modular Monolith ถึงสำคัญ?
&lt;/h2>&lt;p>&lt;strong>Modular Monolith&lt;/strong> รวมข้อดีของทั้ง monolithic และ microservices architectures ทำให้เป็นทางเลือกที่ยอดเยี่ยมสำหรับแอปพลิเคชันหลายประเภท โดยเฉพาะเมื่อเริ่มต้นโปรเจคใหม่&lt;/p>
&lt;h3 id="ขอดของ-modular-monolith">ข้อดีของ Modular Monolith:
&lt;/h3>&lt;ol>
&lt;li>&lt;strong>ความเรียบง่าย&lt;/strong>: deployment unit เดียว ง่ายต่อการพัฒนาและ debug&lt;/li>
&lt;li>&lt;strong>ประสิทธิภาพ&lt;/strong>: ไม่มี network latency ระหว่าง modules&lt;/li>
&lt;li>&lt;strong>ความสอดคล้อง&lt;/strong>: database transactions เดียวข้าม modules&lt;/li>
&lt;li>&lt;strong>ความเร็วในการพัฒนา&lt;/strong>: การพัฒนาที่เร็วขึ้นด้วยการแชร์ code และไลบรารี&lt;/li>
&lt;li>&lt;strong>การปรับปรุงง่าย&lt;/strong>: สามารถแยกเป็น microservices ทีหลังได้หากจำเป็น&lt;/li>
&lt;/ol>
&lt;h3 id="หลกการสำคญ">หลักการสำคัญ:
&lt;/h3>&lt;ol>
&lt;li>&lt;strong>ขอบเขต Module ที่ชัดเจน&lt;/strong>: แต่ละ module มีความรับผิดชอบที่กำหนดไว้อย่างชัดเจน&lt;/li>
&lt;li>&lt;strong>การเชื่อมต่อแบบหลวม&lt;/strong>: modules สื่อสารผ่าน interfaces ที่กำหนดไว้อย่างดี&lt;/li>
&lt;li>&lt;strong>ความเหนียวแน่นสูง&lt;/strong>: functionality ที่เกี่ยวข้องอยู่ด้วยกันภายใน module&lt;/li>
&lt;li>&lt;strong>Infrastructure ร่วม&lt;/strong>: เรื่องทั่วไปเช่น logging, configuration, database&lt;/li>
&lt;/ol>
&lt;h3 id="เมอไรควรใช-modular-monolith">เมื่อไรควรใช้ Modular Monolith:
&lt;/h3>&lt;ul>
&lt;li>&lt;strong>เริ่มโปรเจคใหม่&lt;/strong>: เริ่มด้วย monolith แล้วแยกทีหลังหากจำเป็น&lt;/li>
&lt;li>&lt;strong>ทีมขนาดเล็กถึงกลาง&lt;/strong>: การประสานงานและการพัฒนาที่ง่ายขึ้น&lt;/li>
&lt;li>&lt;strong>ขอบเขต domain ไม่แน่นอน&lt;/strong>: เรียนรู้ domain ก่อนแยก&lt;/li>
&lt;li>&lt;strong>ความต้องการประสิทธิภาพ&lt;/strong>: latency ต่ำกว่าระบบแบบกระจาย&lt;/li>
&lt;li>&lt;strong>ความต้องการความสอดคล้อง&lt;/strong>: ต้องการ ACID transactions ข้าม modules&lt;/li>
&lt;/ul>
&lt;h3 id="ตวอยางโครงสราง-module">ตัวอย่างโครงสร้าง Module:
&lt;/h3>&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// Modules/Products/&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> ProductModule.cs &lt;span style="color:#960050;background-color:#1e0010">#&lt;/span> &lt;span style="color:#960050;background-color:#1e0010">การกำหนดค่า&lt;/span> module &lt;span style="color:#960050;background-color:#1e0010">และ&lt;/span> endpoints
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> Models/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">│&lt;/span> &lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> Product.cs &lt;span style="color:#960050;background-color:#1e0010">#&lt;/span> Product entity
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">│&lt;/span> &lt;span style="color:#960050;background-color:#1e0010">└──&lt;/span> ProductDto.cs &lt;span style="color:#960050;background-color:#1e0010">#&lt;/span> Data transfer objects
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> Services/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">│&lt;/span> &lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> IProductService.cs &lt;span style="color:#960050;background-color:#1e0010">#&lt;/span> Service &lt;span style="color:#66d9ef">interface&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">│&lt;/span> &lt;span style="color:#960050;background-color:#1e0010">└──&lt;/span> ProductService.cs &lt;span style="color:#960050;background-color:#1e0010">#&lt;/span> Service implementation
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">└──&lt;/span> Data/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#960050;background-color:#1e0010">└──&lt;/span> ProductRepository.cs &lt;span style="color:#960050;background-color:#1e0010">#&lt;/span> Data access layer
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// Modules/Orders/&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> OrderModule.cs
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> Models/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> Services/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">└──&lt;/span> Data/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// Shared/&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> Database/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">│&lt;/span> &lt;span style="color:#960050;background-color:#1e0010">└──&lt;/span> AppDbContext.cs &lt;span style="color:#960050;background-color:#1e0010">#&lt;/span> Shared database context
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> Infrastructure/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">│&lt;/span> &lt;span style="color:#960050;background-color:#1e0010">├──&lt;/span> Logging/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">│&lt;/span> &lt;span style="color:#960050;background-color:#1e0010">└──&lt;/span> Configuration/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#960050;background-color:#1e0010">└──&lt;/span> Common/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#960050;background-color:#1e0010">└──&lt;/span> BaseEntity.cs &lt;span style="color:#960050;background-color:#1e0010">#&lt;/span> Shared &lt;span style="color:#66d9ef">base&lt;/span> classes
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="เสนทางการยายไป-microservices">เส้นทางการย้ายไป Microservices:
&lt;/h3>&lt;p>หากแอปพลิเคชันของคุณเติบโตและคุณต้องการแยกเป็น microservices โครงสร้างแบบ modular ทำให้การเปลี่ยนแปลงนี้ง่ายขึ้น:&lt;/p>
&lt;ol>
&lt;li>&lt;strong>ระบุขอบเขต Module&lt;/strong>: modules กลายเป็นขอบเขต service&lt;/li>
&lt;li>&lt;strong>แยกฐานข้อมูล&lt;/strong>: ให้แต่ละ service มีฐานข้อมูลของตัวเอง&lt;/li>
&lt;li>&lt;strong>เพิ่มชั้น Communication&lt;/strong>: แทนที่การเรียกตรงด้วย HTTP/messaging&lt;/li>
&lt;li>&lt;strong>Deploy อิสระ&lt;/strong>: แต่ละ service ได้ deployment pipeline ของตัวเอง&lt;/li>
&lt;/ol>
&lt;h2 id="สรป">สรุป
&lt;/h2>&lt;p>การสร้าง full stack applications ด้วย &lt;strong>Vue.js&lt;/strong>, &lt;strong>Tailwind CSS&lt;/strong> และ &lt;strong>ASP.NET Core 9 Minimal API&lt;/strong> โดยใช้ &lt;strong>Modular Monolith&lt;/strong> architecture ให้:&lt;/p>
&lt;ul>
&lt;li>&lt;strong>ประสบการณ์การพัฒนาสมัยใหม่&lt;/strong>: เทคโนโลยีล่าสุดและแนวปฏิบัติที่ดีที่สุด&lt;/li>
&lt;li>&lt;strong>การพัฒนาอย่างรวดเร็ว&lt;/strong>: การทำซ้ำและ deployment ที่รวดเร็ว&lt;/li>
&lt;li>&lt;strong>สถาปัตยกรรมที่ขยายได้&lt;/strong>: สามารถเติบโตจาก monolith ไป microservices&lt;/li>
&lt;li>&lt;strong>ประสบการณ์นักพัฒนาที่ยอดเยี่ยม&lt;/strong>: เครื่องมือและเอกสารที่ยอดเยี่ยม&lt;/li>
&lt;li>&lt;strong>ประสิทธิภาพ&lt;/strong>: แอปพลิเคชันที่รวดเร็วและมีประสิทธิภาพ&lt;/li>
&lt;/ul>
&lt;p>วิธีการนี้เหมาะสำหรับผู้เริ่มต้นเขียนโปรแกรมเพราะให้:&lt;/p>
&lt;ul>
&lt;li>การแยกส่วนงานที่ชัดเจน&lt;/li>
&lt;li>แนวปฏิบัติการพัฒนาสมัยใหม่&lt;/li>
&lt;li>สถาปัตยกรรมที่ขยายได้&lt;/li>
&lt;li>แหล่งเรียนรู้ที่ยอดเยี่ยม&lt;/li>
&lt;/ul>
&lt;p>เริ่มต้นด้วย stack นี้ เรียนรู้พื้นฐาน และค่อยขยายความรู้ของคุณเมื่อแอปพลิเคชันมีความซับซ้อนมากขึ้น&lt;/p>
&lt;h2 id="อางอง">อ้างอิง
&lt;/h2>&lt;ul>
&lt;li>&lt;a class="link" href="https://vuejs.org/" target="_blank" rel="noopener"
>Vue.js Official Documentation&lt;/a>&lt;/li>
&lt;li>&lt;a class="link" href="https://tailwindcss.com/" target="_blank" rel="noopener"
>Tailwind CSS Documentation&lt;/a>&lt;/li>
&lt;li>&lt;a class="link" href="https://docs.microsoft.com/en-us/aspnet/core/fundamentals/minimal-apis" target="_blank" rel="noopener"
>ASP.NET Core Minimal APIs&lt;/a>&lt;/li>
&lt;li>&lt;a class="link" href="https://docs.microsoft.com/en-us/dotnet/" target="_blank" rel="noopener"
>.NET 9 Documentation&lt;/a>&lt;/li>
&lt;li>&lt;a class="link" href="https://www.milanjovanovic.tech/blog/modular-monolith-architecture" target="_blank" rel="noopener"
>Modular Monolith Architecture&lt;/a>&lt;/li>
&lt;/ul></description></item></channel></rss>