جاواپرو | برنامه نویسی جاوا | Java
5.85K subscribers
1.15K photos
158 videos
381 files
1.28K links
🎓آکـــــــــادمی جاواپـــــــــــــــرو
آموزش پیش نیازهای برنامه نویسی
آموزش مقدماتی تا پیشرفته جاوا
آموزش Spring Boot
سفارش پروژه ، دوره و تدریس خصوصی: @rzutab
مشاهده دوره ها و ثبت نام👇
wwww.academyjavapro.com
گروه جاوا : @group_javapro
Download Telegram
بعضی از عزیزان پیام دادن آیا جاوا آینده شغلی در ایران داره؟
نمونه آگهی های استخدام جاوا در جابینجا و جاب ویژن رو می تونید مشاهده نمایید.
خارج از کشور هم دیگه بدون توضیح است که چقدر به این تخصص نیاز است.

کد تخفیف 40%: ABAN
فقط تا امشب فرصت است.

📚لیست دوره های تخصصی مرتبط با جاوا
🧠 آشنایی با ابزارهای java.util.concurrent برای مدیریت Reordering و Visibility

در برنامه‌نویسی همزمانی در جاوا، یکی از چالش‌های اصلی، کنترل ترتیب اجرای عملیات و اطمینان از قابل‌مشاهده بودن تغییرات حافظه میان Threadها است. برای حل این مشکلات، زبان جاوا مجموعه‌ای از ابزارهای قدرتمند در پکیج java.util.concurrent ارائه می‌دهد که به‌طور مستقیم برای مدیریت هماهنگی، همگام‌سازی و جلوگیری از مشکلات حافظه طراحی شده‌اند. سه ابزار مهم در این زمینه CountDownLatch، CyclicBarrier و AtomicInteger هستند.


✳️ ۱. CountDownLatch — ابزار انتظار تا تکمیل عملیات

در بسیاری از شرایط، لازم است یک Thread صبر کند تا چند Thread دیگر کار خود را کامل کنند. در چنین مواقعی، CountDownLatch به کمک می‌آید.

هنگامی که Latch مقدار شمارش اولیه دارد، هر بار که یک Thread عملیات خود را تمام می‌کند، مقدار آن یک واحد کاهش می‌یابد. زمانی که مقدار به صفر برسد، همه‌ی Threadهایی که منتظر بودند اجازه‌ی ادامهٔ کار پیدا می‌کنند.

🔹 مزیت در بحث Reordering و Visibility:
CountDownLatch تضمین می‌کند که هر تغییری که قبل از کاهش شمارنده انجام شده، توسط Threadهای منتظر به‌درستی دیده می‌شود. این رفتار مطابق با *Happens-Before Relation* است.

🔹 مثال:


CountDownLatch latch = new CountDownLatch(2);
AtomicInteger result = new AtomicInteger(0);

new Thread(() -> {
result.addAndGet(10);
latch.countDown();
}).start();

new Thread(() -> {
result.addAndGet(20);
latch.countDown();
}).start();

// Thread اصلی
latch.await();
System.out.println(result.get());


در این مثال، عملیات روی result همیشه قبل از ادامه یافتن Thread اصلی اتفاق می‌افتد.


✳️ ۲. CyclicBarrier — هماهنگی میان چند Thread

در کاربردهایی مانند شبیه‌سازی یا پردازش موازی، لازم است چند Thread به یک نقطه‌ی مشترک برسند و سپس همزمان ادامه دهند. CyclicBarrier دقیقاً برای این هدف طراحی شده است.

وقتی چند Thread به مانع (Barrier) برسند، همه تا زمان تکمیل تعداد مشخص‌شده صبر می‌کنند و سپس با هم ادامه می‌دهند. این ابزار در سناریوهایی مفید است که لازم است مراحل پردازش چند Thread هماهنگ انجام شود.

🔹 مزیت در بحث Visibility:
هنگامی که همه‌ی Threadها به مانع می‌رسند، مقادیر مشترک در حافظه قبل از عبور از Barrier برای همه قابل مشاهده خواهند بود، زیرا Barrier یک نقطه‌ی همگام‌سازی ایجاد می‌کند.

🔹 مثال:


CyclicBarrier barrier = new CyclicBarrier(3, () -> {
System.out.println("All threads reached the barrier!");
});

Runnable task = () -> {
System.out.println(Thread.currentThread().getName() + " working...");
try {
barrier.await();
} catch (Exception e) {}
};

new Thread(task).start();
new Thread(task).start();
new Thread(task).start();


در این نمونه، هر سه Thread تا رسیدن همه به نقطه‌ی مشترک منتظر می‌مانند.


✳️ ۳. AtomicInteger — متغیر امن در برابر Race Condition

در شرایطی که چند Thread یک متغیر مشترک را تغییر می‌دهند، استفاده از متغیرهای معمولی باعث Race Condition و رفتار غیرقابل پیش‌بینی می‌شود. کلاس AtomicInteger عملیاتی مثل افزایش و کاهش را به صورت اتمی (Atomic) انجام می‌دهد. این یعنی عملیات به‌صورت کامل و غیرقابل تقسیم اجرا می‌شود و هیچ Thread دیگری نمی‌تواند در میانه‌ی عملیات آن را مختل کند.

🔹 مزیت در بحث Reordering:
عملیات Atomic با استفاده از دستورهای سطح پایین CPU (مانند CAS) اجرا می‌شوند که اجازه‌ی Reorder ناامن را نمی‌دهند.

🔹 مزیت در بحث Visibility:
AtomicInteger تضمین می‌کند که تغییرات بلافاصله برای تمام Threadها قابل مشاهده است.

🔹 مثال:


AtomicInteger counter = new AtomicInteger(0);

Runnable task = () -> {
for (int i = 0; i < 1000; i++) {
counter.incrementAndGet();
}
};

Thread t1 = new Thread(task);
Thread t2 = new Thread(task);

t1.start();
t2.start();

t1.join();
t2.join();

System.out.println(counter.get());


در این مثال، مقدار نهایی همیشه ۲۰۰۰ خواهد بود، بدون هیچ‌گونه Race Condition.
👍3
🧠 ۴. نتیجه‌گیری

ابزارهای موجود در java.util.concurrent نقش بسیار مهمی در مدیریت همزمانی و جلوگیری از رفتارهای غیرمنتظره حافظه دارند.

CountDownLatch
به کنترل ترتیب اجرای Threadها کمک می‌کند.
CyclicBarrier
امکان همگام‌سازی مرحله‌ای بین Threadها را فراهم می‌کند.
AtomicInteger
امنیت و قابل‌مشاهده بودن تغییرات متغیرها را تضمین می‌کند.

درک صحیح این ابزارها باعث می‌شود برنامه‌های چندریسمانی پایدارتر، ایمن‌تر و پیش‌بینی‌پذیرتری نوشته شود.

#کاربرـحرفهـای


🆔 @javapro_ir
🆔 @group_javapro
👍3
📘 آشنایی با Semaphore، Exchanger و Phaser در همزمانی (Concurrency)

در برنامه‌نویسی همزمانی در جاوا، گاهی تنها هماهنگ‌سازی ساده میان Threadها کافی نیست و نیاز داریم رفتارهای پیچیده‌تری مانند کنترل ظرفیت، تبادل داده بین Threadها یا هماهنگی مرحله‌ای ایجاد کنیم. سه ابزار Semaphore، Exchanger و Phaser برای این اهداف طراحی شده‌اند و بخش مهمی از کتابخانهٔ java.util.concurrent را تشکیل می‌دهند. این ابزارها کنترل پیشرفته‌تری بر رفتار Threadها ارائه می‌دهند.


✳️ ۱. Semaphore — کنترل دسترسی به منابع محدود

در بسیاری از برنامه‌ها، تعداد منابع محدود است؛ مانند کانکشن‌های پایگاه داده، تعداد خاصی از فایل‌ها، یا دسترسی همزمان به یک دستگاه سخت‌افزاری. Semaphore این امکان را فراهم می‌کند که فقط تعداد مشخصی از Threadها اجازهٔ ورود داشته باشند.

هنگامی که یک Thread از Semaphore اجازه می‌گیرد (acquire)، یکی از مجوزها مصرف می‌شود. زمان بازگرداندن مجوز (release)، ظرفیت دوباره آزاد می‌شود. این ساختار شبیه یک «دروازه با ظرفیت محدود» عمل می‌کند.

🔹 مثال:

Semaphore semaphore = new Semaphore(2); // فقط ۲ Thread همزمان

Runnable task = () -> {
try {
semaphore.acquire();
System.out.println(Thread.currentThread().getName() + " entered");
Thread.sleep(1000);
semaphore.release();
} catch (Exception ignored) {}
};

new Thread(task).start();
new Thread(task).start();
new Thread(task).start();


در این مثال، فقط دو Thread همزمان اجازه ورود دارند و Thread سوم منتظر می‌ماند.


✳️ ۲. Exchanger — تبادل داده بین دو Thread

گاهی دو Thread نیاز دارند داده‌های خود را با یکدیگر رد و بدل کنند؛ برای مثال یکی داده‌ای پردازش می‌کند و Thread دیگر نتیجه را بررسی می‌کند. کلاس Exchanger مکانیزمی فراهم می‌کند که دقیقاً دو Thread بتوانند داده‌های خود را با هم تبادل کنند.

Thread اول تا زمانی که Thread دوم به نقطهٔ تبادل نرسد منتظر می‌ماند و سپس داده‌ها مبادله می‌شوند.

🔹 مثال:

Exchanger<String> exchanger = new Exchanger<>();

new Thread(() -> {
try {
String result = exchanger.exchange("Data from Thread 1");
System.out.println("Thread 1 received: " + result);
} catch (Exception ignored) {}
}).start();

new Thread(() -> {
try {
String result = exchanger.exchange("Data from Thread 2");
System.out.println("Thread 2 received: " + result);
} catch (Exception ignored) {}
}).start();


در این نمونه، دو Thread پیام‌های خود را با یکدیگر جابجا می‌کنند.


✳️ ۳. Phaser — کنترل هماهنگی مرحله‌ای (Multi-Phase Synchronization)

وقتی یک عملیات از چند مرحله تشکیل شده و چند Thread باید هر مرحله را همزمان انجام دهند، Phaser ابزار مناسب این نوع هماهنگی است. این کلاس نسخهٔ پیشرفته‌تر CyclicBarrier محسوب می‌شود و امکانات بیشتری برای کنترل تعداد متغیر Threadها در طول برنامه فراهم می‌کند.

Phaser می‌تواند در هر مرحله همهٔ Threadها را متوقف کند تا زمانی که همه به پایان مرحله برسند. سپس مرحله بعدی آغاز می‌شود.

🔹 مثال:

Phaser phaser = new Phaser(3); // سه Thread ثبت شده‌اند

Runnable task = () -> {
System.out.println(Thread.currentThread().getName() + " Phase 1");
phaser.arriveAndAwaitAdvance();

System.out.println(Thread.currentThread().getName() + " Phase 2");
phaser.arriveAndAwaitAdvance();

System.out.println(Thread.currentThread().getName() + " Phase 3");
phaser.arriveAndDeregister();
};

new Thread(task).start();
new Thread(task).start();
new Thread(task).start();


در این نمونه، هر سه Thread باید مرحلهٔ ۱ را تمام کنند تا مرحلهٔ ۲ شروع شود و همین‌طور برای مراحل بعدی.


🧠 ۴. جمع‌بندی

ابزارهای معرفی‌شده نقش مهمی در ساخت برنامه‌های همزمانی پایدار و قابل‌کنترل دارند:

*در Semaphore تعداد Threadهای مجاز برای دسترسی به یک منبع را کنترل می‌کند.
Exchanger امکان تبادل دادهٔ مطمئن و هماهنگ بین دو Thread را فراهم می‌کند.
Phaser هماهنگی مرحله‌ای میان مجموعه‌ای از Threadهای متغیر را مدیریت می‌کند.

درک این ابزارها باعث می‌شود برنامه‌های چندریسمانی حرفه‌ای‌تر و قابل‌پیش‌بینی‌تری نوشته شود.

#کاربرـحرفهـای



📚لیست دوره های تخصصی مرتبط با جاوا

🆔 @javapro_ir
🆔 @group_javapro
👍5
🔥 مقایسه HashMap ،Hashtable و LinkedHashMap
جاواپرو | برنامه نویسی جاوا | Java
🔥 مقایسه HashMap ،Hashtable و LinkedHashMap
🔥 مقایسه HashMap ،Hashtable و LinkedHashMap

در جاوا، انتخاب پیاده‌سازی درست برای Map می‌تواند تأثیر زیادی روی کارایی، Thread-Safety و حفظ ترتیب در برنامه شما داشته باشد. در ادامه یک جمع‌بندی سریع و کاربردی برای انتخاب بهتر آورده شده است 👇

🟦ر ۱️⃣ HashMap – سریع و انعطاف‌پذیر

بهترین انتخاب در اکثر سناریوها که سرعت مهم است.

از کلید و مقدار null پشتیبانی می‌کند

غیر Thread-Safe → سرعت بیشتر

🔄 ترتیب درج را حفظ نمی‌کند

🎯 مناسب زمانی که سرعت مهم است و نیازی به حفظ ترتیب ندارید

🟩 ۲️⃣ ر Hashtable – Thread-Safe اما قدیمی

قدیمی‌ترین پیاده‌سازی Map در جاوا.

از null برای کلید یا مقدار پشتیبانی نمی‌کند

🔐 همه عملیات‌ها Synchronized → Thread-Safe اما کند

🔄 ترتیب مشخصی ندارد

⚠️ به‌جای آن از ConcurrentHashMap استفاده کنید


🟨ر ۳️⃣ LinkedHashMap – نسخه‌ی مرتب HashMap

ترکیبی از سرعت HashMap + قابلیت حفظ ترتیب.

ترتیب درج را حفظ می‌کند

از null key/value پشتیبانی می‌کند

غیر Thread-Safe

🎯 عالی برای کش‌ها، Mapهای مرتب

(از access-order پشتیبانی می‌کند → مناسب برای ساخت LRU Cache)


🧠 چه زمانی کدام را انتخاب کنیم؟

نیاز انتخاب مناسب

جستجوی سریع بدون نیاز به ترتیب HashMap
ر Map ایمن در برابر Thread ConcurrentHashMap (نه Hashtable)
حفظ ترتیب درج داده‌ها LinkedHashMap
ساخت LRU Cache LinkedHashMap با access-order

👉 همیشه از Hashtable در جاوای مدرن دوری کنید.
👉 برای اکثر کارها HashMap یا LinkedHashMap بهترین انتخاب‌اند.
👉 برای برنامه‌های چندریسمانی از ConcurrentHashMap استفاده کنید.

در دوره زیر Collection ها در جاوا به صورت مفصل بهش پرداخته شده است👇

مشاهده دوره[کلیک کنید]


🆔 @javapro_ir
🆔 @group_javapro
جلسه ۲۳ تهران جاگ

زمان: پنج شنبه ۲۹ آبان ۱۴۰۴ ساعت ۰۹:۰۰ الی ۱۴:۰۰

مکان: ملاصدرا، خیابان عباس شیرازی شمالی، خیابان حکیم اعظم، پلاک ۳۰، دانشگاه خاتم، همکف، سالن آمفی تأتر

لینک ثبت نام: https://evand.com/events/tehjug23


موضوع جلسه:
An Analysis of The Latest and Most Impactful JEPs Proposed in JDK 25.
ساده‌سازی توسعه جاوا با Starter Dependencies در اسپرینگ بوت!

اسپرینگ بوت توسعه جاوا را سریع‌تر، تمیزتر و آسان‌تر می‌کند، چون مجموعه‌ای از وابستگی‌های ازپیش‌پیکربندی‌شده ارائه می‌دهد — دیگر نیازی به تنظیمات دستی چندین فایل JAR یا پیکربندی‌های پیچیده نیست!

🔹 استارترها چه هستند؟
استارترهای اسپرینگ بوت مجموعه‌های آماده‌ای از وابستگی‌ها هستند که تنظیمات Maven یا Gradle را بسیار ساده می‌کنند.
به‌جای اضافه کردن ده‌ها کتابخانه به صورت دستی، کافی‌ست فقط یک استارتر را اضافه کنید — و پروژه آماده اجراست!

پرکاربردترین استارترهای اسپرینگ بوت:

1️⃣ spring-boot-starter-web →
برای ساخت REST API و برنامه‌های وب
2️⃣ spring-boot-starter-data-jpa → برای اتصال به دیتابیس با استفاده از JPA/Hibernate
3️⃣ spring-boot-starter-security → برای احراز هویت و کنترل دسترسی
4️⃣ spring-boot-starter-test →
برای تست‌های واحد و یکپارچه

💡 استارترها = افزایش بهره‌وری!
استفاده از استارترها کدهای اضافی را حذف می‌کند و باعث می‌شود پروژه شما سبک، یکپارچه و قابل نگهداری بماند.

📚لیست دوره های تخصصی مرتبط با جاوا

🆔 @javapro_ir
🆔 @group_javapro
👍2
جاواپرو | برنامه نویسی جاوا | Java
Photo
جی‌اِس‌پی،(JavaServer Pages) — ساخت محتوای وب پویا

جی‌اِس‌پی (JSP) قدرت سروالت‌ها را با این امکان گسترش می‌دهد که توسعه‌دهندگان بتوانند کد جاوا را مستقیماً داخل صفحات HTML قرار دهند؛ به این ترتیب ایجاد برنامه‌های وب پویا و داده‌محور بسیار آسان‌تر می‌شود.
در حالی که سروالت‌ها منطق اصلی را در پشت صحنه مدیریت می‌کنند، JSP بر بخش ارائه (Presentation) تمرکز دارد و ساخت رابط کاربری را ساده‌تر می‌کند.

ویژگی‌های کلیدی JSP:

1️⃣ تفکیک مسئولیت‌ها → جداسازی منطق تجاری (سروالت‌ها) از لایه نمایش (JSP).
2️⃣ کتابخانه‌های تگ (Tag Libraries) → تگ‌های قابل‌استفاده مجدد که توسعه را ساده کرده و از تکرار کد جلوگیری می‌کنند.
3️⃣ زبان عبارت‌نویسی (EL) → دسترسی ساده به JavaBeans و داده‌ها در داخل JSP.
4️⃣ دستورات و اکشن‌های JSP → کنترل رفتار صفحه و امکان استفاده از مؤلفه‌های قابل‌تکرار.

🔹 چرخهٔ حیات JSP:

ترجمه (Translation): تبدیل فایل JSP به یک کلاس Servlet

کامپایل (Compilation): کامپایل به بایت‌کد

اجرا (Execution): پردازش درخواست‌ها توسط سروالت و تولید خروجی HTML پویا


💡 ر JSP را می‌توان لایهٔ دیداری (View Layer) در نظر گرفت که در هماهنگی کامل با سروالت‌ها کار می‌کند — ترکیب منطق + نمایش در بهترین حالت!

🆔 @javapro_ir
🆔 @group_javapro
💯1
📘 آشنایی با ThreadLocal و نقش آن در Thread-Safe بودن داده‌ها

در برنامه‌نویسی همزمانی (Concurrency) در جاوا، یکی از چالش‌های مهم این است که هر Thread بتواند داده‌های مخصوص خود را داشته باشد، بدون آن‌که این داده‌ها میان Threadهای مختلف به اشتراک گذاشته شود. کلاس ThreadLocal برای حل دقیق همین مسئله طراحی شده و یکی از ابزارهای بسیار قدرتمند و ظریف در ساخت برنامه‌های Thread-Safe محسوب می‌شود.


✳️ ۱. ThreadLocal چیست؟

در حالت عادی، اگر چند Thread به یک متغیر مشترک دسترسی داشته باشند، احتمال بروز مشکلاتی مانند Race Condition وجود دارد. کلاس ThreadLocal با نگه‌داشتن یک نسخهٔ مجزا از داده برای *هر Thread*، این مشکل را کاملاً حذف می‌کند.

هر Thread یک مقدار مستقل دارد و تغییر مقدار توسط یک Thread، هیچ تأثیری روی Threadهای دیگر ندارد. این رفتار شبیه داشتن یک «کپی خصوصی» از متغیر برای هر Thread است.


✳️ ۲. کاربرد اصلی ThreadLocal

متداول‌ترین کاربرد ThreadLocal، نگهداری داده‌هایی است که هر Thread به‌صورت جداگانه به آن‌ها نیاز دارد؛ مانند:

* نگهداری Session کاربر در یک سرویس چندریسمانی
* ذخیرهٔ اطلاعات Request در وب‌سرورها
* مدیریت تاریخ و زمان با کلاس‌هایی که Thread-Safe نیستند (مثل SimpleDateFormat)
* نگهداری موقت داده در طول اجرای یک Thread بدون نیاز به Passing در پارامترها


✳️ ۳. مثال کاربردی: ذخیره‌سازی داده خصوصی برای هر Thread

در مثال زیر، هر Thread یک مقدار مستقل دارد و تغییر مقدار یک Thread روی دیگری اثر نمی‌گذارد:


public class ThreadLocalExample {
private static ThreadLocal<Integer> threadLocal = ThreadLocal.withInitial(() -> 0);

public static void main(String[] args) {
Runnable task = () -> {
int value = threadLocal.get();
threadLocal.set(value + 1);
System.out.println(Thread.currentThread().getName() + " → " + threadLocal.get());
};

new Thread(task, "Thread-A").start();
new Thread(task, "Thread-B").start();
new Thread(task, "Thread-C").start();
}
}


اجرای این برنامه خروجی زیر را می‌دهد:


Thread-A → 1
Thread-B → 1
Thread-C → 1


هر Thread مقدار جداگانه‌ای دارد و هیچ‌گونه اشتراک داده‌ای رخ نمی‌دهد.


✳️ ۴. مثال پیشرفته: حل مشکل Thread-Safety در SimpleDateFormat

کلاس SimpleDateFormat به‌صورت پیش‌فرض Thread-Safe نیست. بنابراین برای استفاده در محیط همزمانی، ThreadLocal یکی از بهترین راه‌حل‌هاست.


private static final ThreadLocal<SimpleDateFormat> dateFormat =
ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd"));

public static String format(Date date) {
return dateFormat.get().format(date);
}


در اینجا هر Thread یک نمونهٔ مستقل از SimpleDateFormat دارد و مشکل رقابت (Race Condition) کاملاً از بین می‌رود.


✳️ ۵. نکته بسیار مهم: مدیریت حافظه و جلوگیری از Memory Leak

در استفاده از ThreadLocal باید بسیار دقت کرد، زیرا اگر مقدار ذخیره‌شده در ThreadLocal پس از پایان کار Thread حذف نشود، ممکن است مشکلات Memory Leak به‌وجود بیاید؛ به‌خصوص در محیط‌هایی مانند سرورهای وب که Thread Pool استفاده می‌شود.

بهترین روش این است که همیشه پس از پایان کار مقدار ThreadLocal را پاک کنیم:


try {
// work
} finally {
threadLocal.remove();
}


این کار از باقی‌ماندن داده در ThreadPool جلوگیری می‌کند.


✳️ ۶. جمع‌بندی

درک و استفادهٔ درست از ThreadLocal یکی از مهارت‌های مهم در ساخت سیستم‌های چندریسمانی پایدار است:

* هر Thread یک نسخهٔ خصوصی از داده دارد.
* هیچ تداخلی بین Threadها ایجاد نمی‌شود.
* مناسب برای داده‌های حساس، مانند Request Context یا Formatterها.
* نیازمند مدیریت دقیق برای جلوگیری از Memory Leak.

#کاربرـحرفهـای


🆔 @javapro_ir
🆔 @group_javapro
👍41
بلک فرایدی | دوره های جاوا | 70% تخفیف

📆1 تا 10 آذر ماه

💢کد تخفیف 70%:  BLKFR

Spring, Java, Microservices

📚لیست دوره های تخصصی مرتبط با جاوا

آکادمی جاواپرو
1
📢 اطلاعیه مهم

کاربران گرامی جاواپرو
اختلال پیش‌آمده در نمایش ویدئوهای سایت مربوط به زیرساخت شرکت ارائه‌دهنده هاست است. تیم فنی آن شرکت در حال پیگیری و رفع مشکل می‌باشد.
از شکیبایی و همراهی شما سپاسگزاریم. 🙏
1