1️⃣ 왜 문서화 주석이 중요한가?자바에서는 /** ... */ 형식의 주석을 JavaDoc 주석이라고 부름이건 단순한 설명이 아니라,컴파일러가 인식해서 공식 문서로 변환할 수 있는 API 계약 문서/** * 두 정수의 합을 반환한다. * * @param a 첫 번째 정수 * @param b 두 번째 정수 * @return 두 수의 합 */public int sum(int a, int b) { return a + b;} → javadoc 도구로 자동 HTML 문서 생성 가능 → IDE에서 마우스 올리면 바로 표시됨 2️⃣ “공개된 API”란 무엇인가?"공개된 API"란 단순히 public만 뜻하지 않음외부에서 사용할 수 있는 모든 요소public 클래스, public 메서드라이브러리에서 외부에 ..
Java & Spring Boot
1️⃣ “parallelStream() 붙이면 빨라지겠지?”많은 개발자가 스트림을 이렇게 사용함 list.parallelStream() .map(this::expensiveOperation) .collect(Collectors.toList()); "병렬 스트림"이니까 여러 CPU 코어를 활용해서 당연히 더 빠를거라 기대하지만,현실은 다름!병렬 스트림은 마법이 아님올바른 상황에서만 이득이 있고, 그렇지 않으면 오히려 성능이 나빠지거나 오류를 일으킬 수 있음 2️⃣ 스트림 병렬화의 원리병렬 스트림은 내부적으로 ForkJoinPool을 사용해요소들을 여러 쓰레드로 쪼개서 처리함IntStream.range(0, 1_000_000) .parallel() .forEach(i..
1️⃣ 마커란 무엇인가?"마커"란, 추가 메서드 없이 단순히 '특정 속성을 표시'하는 용도의 구조를 말함 자바에는 두 종류가 있음종류예시특징마커 인터페이스 Serializable, Cloneable타입 수준에서 표시마커 애노테이션@Override, @Deprecated소스 코드 수준에서 표시 둘 다 "이 객체는 ~한 성격을 가진다"는 의미를 전달하지만,적용 범위와 역할이 다름 2️⃣ 마커 인터페이스의 대표 예시자바 표준 라이브러리에서 이미 여러 예시를 볼 수 있음public interface Serializable {}public interface Cloneable {} 이 인터페이스들은 메서드를 하나도 가지지 않음그저 "이 객체는 직렬화/복제가 가능하다"는 타입 표시(marker)로만 쓰임 3️⃣ ..
1️⃣ 제네릭은 유연하지만, 타입 제약은 단단하다자바의 제네릭은 타입 안정성을 높이는 대신,타입 불일치에 매우 예민함List objList = new ArrayList();List strList = new ArrayList();objList = strList; // ❌ 컴파일 에러 직관적으로는 "String은 Object의 하위 타입이니까 괜찮지 않나?" 싶지만,List은 List의 하위 타입이 아님 왜냐하면, List에 Integer를 추가할 수도 있기 때문즉, 제네릭 타입은 "불변" 2️⃣ 이때 등장하는 것이 “한정적 와일드카드”?은 와일드카드(wildcard)로"어떤 타입이 올지 모른다"는 뜻 여기에 extends나 super를 붙이면 "한정(bound)"할 수 있음표현의미예시? extends TT..
1️⃣ 중첩 클래스(Nested Class)란?자바에서는 클래스 안에 클래스를 정의할 수 있음이걸 중첩 클래스(nested class)라고 함 종류는 크게 네 가지이다 👇 종류static 여부외부 클래스 참조주 사용처정적 멤버 클래스OX독립적인 유틸리티내부 로직 캡슐화비정적 멤버 클래스XO외부 인스턴스와 강하게 연결로컬 클래스XO메서드 내부의 임시 클래스익명 클래스XO짧은 1회성 구현 2️⃣ 비정적 멤버 클래스의 문제점비정적 멤버 클래스는 무조건 외부 클래스의 인스턴스에 묶여서 존재함class Outer { class Inner { void hello() { System.out.println("Hello from " + Outer.this); } ..
1️⃣ 왜 clone은 위험할까? 자바의 Cloneable 인터페이스와 Object.clone() 메서드는 객체의 복사를 쉽게 해줄 것 같지만, 실제로는 아님복사가 얕게(shallow) 이루어짐CloneNotSupportedException 예외를 신경써야 함상속 구조에서 불변식을 깨뜨리기 쉬움즉, 잘못 사용하면 버그가 생기기 쉽고, 안전한 복제를 위해선 주의 깊은 설계가 필요함 2️⃣ 얕은 복사로 생기는 문제class Person implements Cloneable { String name; List phones; @Override public Person clone() { try { return (Person) super.clone(); // ..
Java에서 병렬 처리나 비동기 작업을 구현할 때 Thread를 직접 생성해 사용하는 방식은 더 이상 일반적이지 않습니다. 요즘 Java에서는 ExecutorService를 사용하여 스레드 풀(Thread Pool)을 효율적으로 관리하는 것이 표준입니다. 특히 Executors 클래스에서 제공하는 팩토리 메서드들을 활용하면 다양한 유형의 스레드 풀을 간편하게 생성할 수 있습니다.이번 글에서는 Executors가 제공하는 대표적인 스레드 풀 4종류를 살펴보고, 각각의 사용 시나리오와 주의할 점을 예제 중심으로 정리해보겠습니다. newFixedThreadPool(int nThreads)이 메서드는 정해진 개수만큼의 스레드를 미리 생성하고, 그 스레드들이 재사용되며 작업을 처리합니다. 더 많은 작업이 들어오면..
Redis는 Java 애플리케이션에서 캐싱, 세션 관리, 실시간 데이터 저장소로 자주 사용됩니다. Redis 클라이언트에서 데이터를 저장하거나 가져올 때, 데이터는 반드시 직렬화되어야 합니다. 직렬화 방식은 성능과 코드의 유지보수에 큰 영향을 미치기 때문에 잘 선택해야 합니다. 이번 포스팅에서는 Java에서 Redis 직렬화 시 자주 사용하는 방법들을 비교하고 각 방식의 특징과 예제를 설명합니다. JdkSerializationRedisSerializerJDK 기본 직렬화는 Java 표준 라이브러리에서 제공하는 java.io.Serializable 인터페이스를 사용하여 객체를 바이트 스트림으로 변환하는 방식입니다. 이 방식은 사용하기 편리하지만, 다소 무거운 오버헤드가 있을 수 있습니다. 장점:Java ..
알림과 실시간 채팅 같은 서비스를 구현할 때, 우리는 자연스럽게 두 가지 기술 사이에서 고민에 빠지게 됩니다. SSE와 WebSocket은 각기 다른 매력을 지닌 두 명의 주인공과 같습니다. 이 포스팅에서는 SSE와 WebSocket의 특징과 사용법을 비교해 보면서, ‘내 프로젝트에서는 어떤 녀석이 주인공일까?’에 대한 답을 찾아보도록 하겠습니다. 실시간 통신의 필요성여러분의 앱이 최신 주식 가격을 보여주거나 "우리 동네에 눈이 내립니다!" 같은 알림을 띄워야 한다면, 실시간 통신은 선택이 아니라 필수입니다. 그런데 문제는 단방향이냐, 양방향이냐! 양방향 통신을 택하면 더 복잡하지만 매력적이고, 단방향 통신은 더 기능이 적지만 그만큼 단순합니다. 둘 중 어느 쪽을 고를지 결정하기 위해선 기술들의 성격을 ..
MongoDB는 NoSQL 데이터베이스로, 그 유연성 덕분에 Spring Boot와 자주 함께 사용됩니다. Spring Boot에서는 Spring Data MongoDB를 통해 MongoDB 연동을 쉽게 할 수 있습니다. Spring Data MongoDB는 두 가지 주요 방식을 제공합니다. 바로 MongoTemplate와 MongoRepository 입니다. 이번 포스팅에서는 이 두 가지 방법을 비교하고, 어떤 상황에서 각각을 사용하는 것이 좋은지 살펴보겠습니다. MongoRepository와 MongoTemplate[MongoRepository] MongoRepository는 레파지토리 패턴을 기반으로 한 인터페이스입니다. 기본적인 CRUD 메서드를 자동으로 제공합니다. Spring Data JPA와..