برنامه نویسی جاوا | Java
5.7K subscribers
1.11K photos
158 videos
379 files
1.18K links
🎓آکـــــــــادمی جاواپـــــــــــــــرو
آموزش پیش نیازهای برنامه نویسی
آموزش مقدماتی تا پیشرفته جاوا
آموزش Spring Boot
سفارش پروژه ، دوره و تدریس خصوصی: @rzutab
مشاهده دوره ها و ثبت نام👇
wwww.academyjavapro.com
گروه جاوا : @group_javapro
Download Telegram
🟢 تفاوت عمیق بین Lombok و Record

در جاوا برای ساخت کلاس‌های داده‌محور دو رویکرد رایج وجود دارد: استفاده از Lombok (مثلاً با @Data) و استفاده از Record‌های زبان. در ظاهر هر دو «کد تکراری» (boilerplate) را کم می‌کنند، اما در فلسفه، گارانتی‌ها و رفتار زمان اجرا تفاوت‌های بنیادینی دارند.

تعریف و نگاه:

در سطح زبان، Record از جاوا ۱۶ به بعد یک ساختار value-based است که به‌صورت ذاتی: اجزای داده‌ای را نهایی می‌کند، سازنده‌ی قانونی، equals/hashCode/toString و accessor‌ها را تولید می‌کند.
در سطح کتابخانه، Lombok با Annotation Processing در زمان کامپایل کد تولید می‌کند؛ یعنی زبان جاوا تغییر نمی‌کند، اما کلاس شما بر اساس annotationها «تکمیل» می‌شود.

تفاوت اصلی شماره ۱: تغییرپذیری (Mutability) و تضمین‌ها

در Record، فیلدها نهایی و کلاس اساساً ناپذیرفتار (immutable) است (البته «سطحی»؛ اگر فیلدی خودش mutable باشد، محتوایش می‌تواند عوض شود).
در Lombok با @Data، کلاس به‌طور پیش‌فرض قابل تغییر است؛ یعنی setter تولید می‌شود و می‌توان پس از ساخت شیء، وضعیت را تغییر داد.
// Lombok – کلاس پیش‌فرض قابل تغییر
import lombok.Data;

@Data
public class UserLombok {
private String name;
private int age;
}

// Record – کلاس ذاتیِ ناپذیرفتار
public record UserRecord(String name, int age) { }

public class Demo {
public static void main(String[] args) {
UserLombok u1 = new UserLombok();
u1.setName("Ali"); u1.setAge(20); // قابل تغییر

UserRecord r1 = new UserRecord("Ali", 20);
// r1.age = 21; // خطا: فیلدها نهایی‌اند و setter وجود ندارد
}
}

تفاوت اصلی شماره ۲: برابری (Equality) و استفاده به‌عنوان مقدار (Value Semantics)

در Record، برابری به‌صورت ارزش‌محور تعریف می‌شود؛ یعنی دو رکورد با اجزای برابر، برابرند و برای کلید/عضو کالکشن‌ها ایده‌آل‌اند.
در Lombok با @Data نیز equals/hashCode تولید می‌شود، اما چون کلاس معمولاً mutable است، تغییر فیلدی که در برابری دخیل است می‌تواند باعث رفتارهای خطرناک در HashMap/HashSet شود.
// خطر رایج با Lombok @Data و کلاس‌های mutable
import java.util.*;

public class EqualityPitfall {
public static void main(String[] args) {
UserLombok u = new UserLombok();
u.setName("A"); u.setAge(1);

Set<UserLombok> set = new HashSet<>();
set.add(u);
u.setAge(2); // تغییر فیلدی که در hashCode دخیل است
System.out.println(set.contains(u)); // ممکن است false شود → رفتار مشکل‌زا
}
}

// رفتار امن‌تر با Record (تا وقتی اجزا تغییر نکنند)
import java.util.*;

public class EqualitySafe {
public static void main(String[] args) {
UserRecord r = new UserRecord("A", 1);
Set<UserRecord> set = new HashSet<>();
set.add(r);
System.out.println(set.contains(r)); // همیشه true چون رکورد ناپذیرفتار است
}
}

تفاوت اصلی شماره ۳: سازنده و اعتبارسنجی (Validation)

در Record، می‌توانید سازنده‌ی فشرده (compact constructor) تعریف کنید و قواعد اعتبارسنجی را همان‌جا اعمال کنید؛ اما همچنان ناپذیرفتاری حفظ می‌شود.
public record Email(String local, String domain) {
public Email {
if (local == null || local.isBlank()) throw new IllegalArgumentException("local required");
if (domain == null || !domain.contains(".")) throw new IllegalArgumentException("invalid domain");
}
}

در Lombok، می‌توانید از annotationهایی مثل @NonNull یا سازنده‌های تولیدی (@AllArgsConstructor/@RequiredArgsConstructor) و حتی @Builder برای ساخت امن‌تر استفاده کنید، اما ماهیت کلاس لزوماً immutable نمی‌شود مگر اینکه از @Value بهره بگیرید.
import lombok.Builder;
import lombok.Value;

// Lombok – کلاس ناپذیرفتار با @Value + سازنده Builder
@Value
@Builder
public class EmailValue {
String local;
String domain;
}

تفاوت اصلی شماره ۴: وراثت و مدل شیء‌گرا

در Record، ارث‌بری از کلاس‌ها ممنوع است (همه رکوردها ضمنی final هستند) اما پیاده‌سازی اینترفیس مجاز است.
در Lombok، کلاس شما یک کلاس «عادی» جاواست؛ می‌تواند از کلاس دیگری ارث ببرد یا اینترفیس‌ها را پیاده‌سازی کند و Lombok صرفاً کدهای تکراری را تولید می‌کند.

تفاوت اصلی شماره ۵: نسخه و وابستگی
👍4
در Record، به هیچ وابستگی خارجی نیاز ندارید اما به JDK مدرن نیازمندید (۱۶+).
در Lombok، به وابستگی بیلد و معمولاً پلاگین IDE نیاز دارید، اما روی JDKهای قدیمی‌تر نیز کار می‌کند و مجموعه‌ای از قابلیت‌ها (loggerها، builder، wither، constructorها و …) را یکجا می‌دهد.

تفاوت اصلی شماره ۶: سازوکار ساخت شیء

در Record، الگوی ساخت ثابت است و خبری از سازوکار داخلیِ Builder نیست (می‌توانید factory بنویسید، اما «پیش‌ساخته» نیست).
در Lombok، استفاده از @Builder الگوی ساختِ روان و ایمن با پارامترهای زیاد را بسیار ساده می‌کند.
// Lombok Builder – مناسب برای پارامترهای زیاد/اختیاری
UserLombok u = UserLombokBuilder.builder()
.name("Sara")
.age(30)
.build();

نکته تکمیلی: الگوهای رکورد (Record Patterns) و تخریب (Deconstruction)

در نسخه‌های جدید جاوا، امکان الگوی رکورد در switch/instanceof فراهم شده است؛ یعنی می‌توانید مستقیماً اجزای رکورد را «استخراج» کنید. این قابلیت، ماهیت value-based رکوردها را تقویت می‌کند.
static String prettyPrint(Object o) {
return switch (o) {
case UserRecord(String n, int a) -> "User(name=%s, age=%d)".formatted(n, a);
default -> o.toString();
};
}

جمع‌بندی راهبردی: چه زمانی کدام‌یک؟

وقتی «شیءِ ارزش‌محور/داده‌محور» می‌خواهید که ناپذیرفتار باشد و تضمین‌های سطح‌زبان را ترجیح می‌دهید → از Record استفاده کنید (DTO، Value Object، پیام‌ها، نتایج محاسبات).

وقتی به انعطاف شیء عادی جاوا، سازنده‌های متنوع، Builder، loggerها، یا اجرای روی JDK قدیمی نیاز دارید → از Lombok استفاده کنید (به‌خصوص با @Builder, @Getter/@Setter, @With, @Slf4j).

وقتی می‌خواهید با Lombok هم ناپذیرفتاری داشته باشید → از @Value استفاده کنید یا setter تولید نکنید؛ و مراقب فیلدهای mutable باشید.

وقتی می‌خواهید کلید پایدار برای Map/Set داشته باشید → رکورد یا کلاس immutable انتخاب امن‌تری است.

وقتی در اکوسیستم‌هایی مثل JPA/ORM هستید → نیازمندی‌ها را بررسی کنید؛ رکورد برای موجودیت‌های mutable/مدیریت چرخه عمر اغلب مناسب نیست، اما برای DTOها عالی است.

نمونه‌ی درک‌محور: مقایسه‌ی رفتار در عمل
// تعریف‌ها
@Data
class MoneyLombok { private String currency; private long amount; }

record MoneyRecord(String currency, long amount) { }

public class Compare {
public static void main(String[] args) {
// برابری و تغییرپذیری
MoneyLombok m1 = new MoneyLombok(); m1.setCurrency("USD"); m1.setAmount(100);
MoneyLombok m2 = new MoneyLombok(); m2.setCurrency("USD"); m2.setAmount(100);
System.out.println(m1.equals(m2)); // true
m2.setAmount(200); // تغییر وضعیت → برابری دیگر برقرار نیست

MoneyRecord r1 = new MoneyRecord("USD", 100);
MoneyRecord r2 = new MoneyRecord("USD", 100);
System.out.println(r1.equals(r2)); // همواره value-based و پایدار
// r2.amount = 200; // ناممکن: رکورد ناپذیرفتار است
}
}

نتیجه نهایی

در مقام معماری، Record «قراردادِ ارزش» را در خود زبان تضمین می‌کند و برای مدل‌های داده‌ای شفاف و پایدار ایده‌آل است. در مقام مهندسی محصول، Lombok «چندکاره» است و برای کاهش کد تکراری در کلاس‌های معمولی و سناریوهای پیچیده (Builder، logging، سازنده‌های گوناگون) بسیار مؤثر است.
در عمل، انتخاب آگاهانه و متناسب با نیاز (immutability، نسخه‌ی JDK، وابستگی‌ها، الگوی ساخت، و الزامات فریمورک) بهترین نتیجه را می‌دهد.

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


🆔 @javapro_ir
🆔 @group_javapro
👍5