∙Java & Spring

자바의 정석 정리(2)

coor 2022. 3. 9. 21:17

● Java 장점

- 운영체제가 독립적
- 객체지향언어
- 비교적 배우기 쉬움
- 네트워크와 분산처리를 지원
- 자동 메모리 관리
>> 가비지컬렉터(garbage collector)
- 멀티쓰레드 지원
>> 여러 쓰레드에 대한 스케줄링을 자바 인터프리터가 담당
- 동적 로딩 지원

 

● JVM 역할(Java Virtual Machine)

- 자바 애플리케이션을 실행시키기 위한 소프트웨어
기능 1 -> 하드웨어 기계어로 변환해주어 속도 향상해주는 JIT 컴파일러
기능 2 -> 각 운영체제에 맞게 실행

 

JDK 역할(Java Development Kit)

- JVM과 자바클래스 라이브러리 외에 자바를 개발하는데 필요한 프로그램이 설치

JDK bin 역할
- javac.exe >> 자바 컴파일러, 자바 소스코드를 바이트코드로 컴파일한다.
- java.exe   >> 자바 인터프리터, 컴파일러가 생성한 바이트코드를 해석하고 실행한다.
- javap.exe >> 역어셈블러, 컴파일된 클래스 파일을 원래의 소스로 변환한다.
- jar.exe    >> 압축프로그램, 클래스 파일과 프로그램의 실행에 관련된 파일을 하나의 jar. 압축하거나 해체한다.

 

JRE

- 자바실행환경, 자바로 작성된 응용프로그램이 실행되기 위한 최소환경

 

JVM 메모리 구조

1. 메서드 영역(method area)
- 해당 클래스의 클래스 파일(*. class)을 읽어서 분석하여 클래스에 대한 정보를 저장하는 영역

2. 힙(heap)
- 인스턴스가 생성되는 공간

3. 호출 스택(call stack 또는 execution stack)
- 메서드가 작업에 필요한 메모리 공간을 제공한다.
- 지역변수들과 연산의 중간결과 등을 저장하는데 사용
- 메서드가 작업을 마치면 할당되었던 메모리 공간은 비워진다.

 

● 오버로딩

- 하나의 메서드 이름으로 여러 기능을 구현하는 것

void println( )
void printlnBoolean(boolean x)
void printlnChar(char x)
void printlnDouble(double x)
void printlnString(String x)

하나의 메서드로 정의하면
>> void println( )

1. 오버로딩 조건
- 메서드 이름이 같아야 함
- 매개변수의 개수 또는 타입이 달라야 한다. (타입은 상관 X)

예시)

int add(int a, int b)
int add(int x, int y)

- 매개 변수 타입이 같아서 X

int add(int a, int b)
long add(int a, int b)

- 리턴 타입만 다른 경우 X

 

올바른 예시)

long add(int a, long b)
long add(long a, int b)

- 같은 메서드 O , 다른 매개변수 타입 O

 

2. 오버로딩 장점
- 메서드 하나의 이름으로 정의할 수 있어서 기억하기 쉽고 짧게 지을 수 있어서 오류 가능성 줄인다.
- 여러 개의 메서드를 지을 필요가 없어서 다른 이름을 짓는데 고민할 필요 없다.

 

 

● 오버라이딩

- 자식이 상속받은 부모 메서드의 내용을 변경하는 것
- 상속받은 메서드를 그대로 사용하기도 하지만, 자신이 쓰고 싶은 값에 맞게 쓰는 경우에 쓰인다.

class Point {
	int x=10;
	int y=11;
	
	int getLocation() {
		return x+y;
	}
	
}
class Points extends Point {
	int z=3;
	
	int getLocation() {   // 오버라이딩
		return x+y+z;
	}
}

 

조건
1. 이름이 같아야 한다.
2. 매개 변수가 같아야 한다.
3. 반환 타입이 같아야 한다.
4. 자식이 많은 수의 예외를 선언 X (예외 레벨에 따라)

 

오버로딩 vs 오버라이딩 차이

- 오버로딩은 기존에 없는 메서드를 추가하는 것
- 오버라이딩은 부모로부터 상속받은 메서드의 내용을 변경하는 것

class Parent {
	void parentMethod( ) { }
}
class Child extends Parent {
	void parentMethod( ) { }           // 오버라이딩
	void parentMethod(int i) { }       // 오버로딩
}

 

생성자

- 생성자는 인스턴스가 생성될 때 호출되는 '인스턴스 초기화 메서드' 이다.
- 주로 연산자 new에 의해서 메모리(heap)에 인스턴스가 생성된다.
- JPA 제공하는 어노테이션
1. @NoArgsConstructor - 파라미터가 없는 기본 생성자
2. @AllArgsConstructor - 모든 필드 값을 파라미터로 받는 생성자
3. @RequiredArgsConstructor - final이나 @NonNull인 필드 값만 파라미터로 받는 생성자

 

인스턴스

- 어떤 클래스로부터 만들어진 객체를 그 클래스의 '인스턴스'라고 한다.
- Tv클래스로부터 인스턴스를 생성하고 인스턴스의 속성과 메서드를 사용한다.

class Tv {
   int channel;
   void channelDown( ) { --channel; }
}
class TvTest {
   Tv t = new Tv( );  // 인스턴스 생성
   t.channel = 7;
   t.channelDown( );
}

 

인터페이스 장점

- 개발 시간을 단축시킬 수 있다.
- 표준화가 가능하다.
- 서로 관계없는 클래스들에게 관계를 맺어 줄 수 있다.
- 독립적인 프로그래밍이 가능하다.

 

1. 인터페이스를 사용하는 이유
>> 첫번째 이유

- 건물을 표현하는 클래스중에서 Barrack 클래스만 건물 이동 메서드를 추가할 경우가 필요하다. 
이럴 경우 인터페이스를 통해 메소드를 추가할 수 있다.

interface Liftable {
	void move(int x, int y);
}

class LiftableImpl implements Liftable {
	public void move(int x, int y) { ... }  // 이동 메소드 추가
}

class Barrack extends Building implements Liftable { 
	LiftableImpl l = new LiftableImpl( );
	
	void move(int x, int y) { 
          l.move(x,y); 
   }
}

상속은 그대로이지만 인터페이스를 통해 따로 메소드를 추가할 수 있는 것을 볼 수 있다.

 

>> 두번째 이유

class A {
	public void methodA(B b) {
		b.methodB();
	}
}
class B {
	public void methodB() {
		System.out.println("methodB()");
	}
}

클래스 A, B는 직접적인 관계이므로 B의 methodB( ) 선언부가 변경되면 클래스 A도 변경되어야 한다.
하나가 변경되면 다른 하나가 변경되는 문제점이 발생된다.
이 때 클래스 A가 클래스 B를 직접 호출하지 않고 인터페이스를 매개체로 해서 접근할 수 있다.
그러면 B가 변경되어도 A는 전혀 영향을 받지 않는다.

interface I {
	void methodB( );
}
class B implements I {
	public void methodB( ) {
		System.out.println("methodB( )");
	}
}

 

 

얕은 복사 vs 깊은 복사

- 얕은 복사는 '주소값'을 복사한다. 즉, 객체는 참조하지 않고 데이터만 참조한다. 그래서 실제 값이 같다.
- 깊은 복사는 '실제값'을 새로운 메모리 공간에 복사한다. 즉, 실제값이 다르다.

>> 얕은 복사

class Book {
	private String title;
	private String author;

	public Book (String title, String author) {
		this.title = title;
		this.author = author;
	}
}

class Test {
	public static void main(String args[]) {
		Book library = new Book("태백산맥1","조정래");
		Book copy = library;  // 얕은 복사

		System.out.println(library.getTitle()+library.getAuthor());   // 태백산맥1조정래
		System.out.println(copy.getTitle()+copy.getAuthor());      // 태백산맥1조정래

		copy.setTitle("22");

       		 System.out.println(library.getTitle()+library.getAuthor());   // 22조정래
		System.out.println(copy.getTitle()+copy.getAuthor());      // 22조정래
	}
}

얕은 복사는 데이터를 복사하고 객체는 참조하지 않기 때문에 두 개의 값이 바뀌었다.
한 개의 Book 인스턴스를 가리키게 되므로 완전한 복제라고 볼 수 없다.
코드를 짜면 참조값이 아닌 실제 값을 복사해야 하는 경우가 있다.
그럴 경우 깊은 복사를 해야한다.

 

>> 깊은 복사

class Book {
	private String title;
	private String author;

	public Book (String title, String author) {
		this.title = title;
		this.author = author;
	}
	public Book deppCopy() {
		Object obj = null;
		
		try {
			obj=super.clone();
		} catch (Exception e) {}

		Book b= (Book) obj;
		b = new Book(this.title,this.author);
			
		return b;
	}
}

class Test {
	public static void main(String args[]) {
		Book library = new Book("태백산맥1","조정래");
		Book copy = library.deppCopy();  // 깊은 복사

		copy.setTitle("22");

        	System.out.println(library.getTitle()+library.getAuthor());   // 태백산맥1조정래
		System.out.println(copy.getTitle()+copy.getAuthor());      // 22조정래
	}
}

새로운 Book 인스턴스를 참조하여 새로운 값이 출력이 되었다. 
원본이 참조하고 있는 객체까지 복사한 것이다.

 

 

List 인터페이스

1. Vector
- Java 1 버전대에서 나와서 호환성이 떨어지지만 동기화를 보장해주어 공유 자원이나 복수 사용자가 존재할 때 유용하다. 단일 스레드의 경우 자동으로 동기화를 보장하는 것이 오히려 성능이 저하가 된다.

2. ArrayList
- 인덱스를 가지고 있어서 검색에 용이하다. 중간에 데이터가 삽입/삭제할 경우 전부 한 칸씩 당기거나 밀리기 때문에 삽입/삭제가 빈번한 데이터인 경우에는 부적합하다. Vector와 달리 동기화를 보장하지 않는다.

3. LinkedList
- ArrayList의 단점을 극복하기 위해 나온 것으로, 노드들이 줄줄이 연결되어 있다. 삽입/삭제할 때는 중간에 해당 노드의 주소지만 바꿔주면 되므로 삽입/삭제가 빈번한 데이터에 적합하다. 하지만 검색할 경우 처음부터 끝까지 줄줄이 이동해야 돼서 적합하지 않다.

4. Vector vs ArrayList 차이
- Vector는 한 번에 하나의 스레드만 접근 가능하지만 ArrayList는 동시에 여러 스레드가 작업할 수 있다.
- Vector는 동기화 되어있기 때문에 한 번에 하나의 스레드만 접근할 수 있어서 Thread Safe(스레드 안전) 하지만,
ArrayList는 동기화 되지 않기 때문에 명시적으로 동기화를 해줘야 한다.
- ArrayList는 동기화 되지 않기 때문에 Vector 보다 빠르다.
- 크기 증가에서 차이가 있다. Vector는 100% 증가, ArrayLisr는 50% 증가한다.

큰 관점에서 바라볼 때 동기화 차이, 크기 증가 차이, 호환성 차이가 있다.
단일 스레드 -> Vector 자동으로 동기화 돼서 부적합, ArrayList 수동 동기화 해야돼서 적합
멀티 스레드 -> Vector 자동으로 동기화 돼서 적합, ArrayList 수동 동기화 해야돼서 부적합

크기 차이 관점으로 보면 Vector < ArrayList 효율적이다.

Vector 클래스는 컬렉션 프레임워크 이전에 나온 클래스이고, 
현재 소스 코드 호환 때문에 남아 있는 클래스 이기 때문에 ArrayList를 사용하는 게 좋다.

5. LinkedList vs ArrayList 차이
- 순차적으로 추가/삭제하는 경우 ArrayList 더 빠르다.
- 중간 데이터를 추가/삭제하는 경우에는 LinkedList 더 빠르다.
>> 각 요소 간의 연결만 변경해주면 되기 때문에 처리속도가 상당히 빠르다.
>> 반면에 ArrayList는 각 요소들을 재배치하여 추가할 공간을 확보하거나 빈 공간을 채워야 하기 때문에 처리속도가 느리다.
- 낮은 접근성
>> LinkedList는 불연속적으로 위치한 각 요소들이 서로 연결된 것이라 데이터를 차례대로 따라가야만 원하는 값을 얻을 수 있다. 그래서 저장한 데이터가 많아질수록 접근 시간이 길어진다는 단점이 있다.
>> 반면에 ArrayList는 배열의 각 요소들이 연속적으로 메모리상에 존재하기 때문에 원하는 요소의 주소를 곧바로 읽을 수 있다.

데이터가 많고 빠른 접근성이 필요하다. -> ArrayList
데이터가 적은데 추가/삭제가 많다. -> LinkedList

 

 

Map 인터페이스

1. HashMap
- HashMap은 Map interface를 implements 한 클래스로서 중복을 허용하지 않는다.
- HashMap은 O(1)과 같은 get과 put 같은 기본 연산에 대해 일정한 시간 성능을 나타낸다.
널(null) 허용한다. 
- 내부 hash 값에 따라서 키순서가 정해지므로 특정 규칙없이 출력된다.
- 검색과 데이터 처리하는 경우에 적합하다. 

2. Hashtable
- HashMap의 구버전이다. 
널(null) 허용하지 않는다.
- 멀티 스레드 안전하다. 

3. TreeMap
- 널(null) 허용하지 않는다.
- TreeMap은 get 및 put 메서드에 대한 log(n) 시간 보장 비용을 제공한다.
- HashMap과 다른 점은 SortedMap을 implements 하였으므로, key 값들에 대한 정렬이 이루어진다는 점이다. 
- 정렬 상태로 출력된다. 
범위 검색이나 정렬이 필요한 경우에 사용하는 것이 좋다.

 

 

Iterator란

Iterator는 이런 집합체로부터 정보를 얻어낸다고 볼 수 있다. 집합체를 다룰 때는 개별적인 클래스에 대해 데이터를 읽는 방법을 알아야 하기 때문에 각 컬렉션에 접근이 힘들어진다.를 쓰게 되면 어떤 컬랙션이라도 동일한 방식으로 접근이 가능하여 그 안에 있는 항목들에 접근할 수 있는 방법을 제공한다.(다형성)

Iterator 메소드에는 hasNext(), next(), remove()가 있다.

 

람다식란

JDK1.8부터 추가된 람다식으로, 객체지향언어인 동시에 함수형 언어가 되었다. 
메서드를 람다식으로 표현하면 메서드의 이름과 반환값이 없어지므로, 람다식을 '익명 함수'이라고도 한다.

int max(int a, int b) {
    return a > b ? a : b;
}
람다식으로 하면
(a, b) -> a > b ? a : b

ArrayList<Integer> list = new ArrayList<>();
for(int i = 0; i < 10; i++) {
    list.add(i);
}
for(int i=0; i<10; i++) {
  System.out.println(i);
}
람다식으로 하면
ArrayList<Integer> list = new ArrayList<>();
for(int i = 0; i < 10; i++) {
   list.add(i);
}
list.forEach( i -> System.out.println(i); }    // 간결하게 바뀜

 

스트림이란

Collection이나 Array에 데이터를 담고 원하는 결과를 얻기 위해 for문과 Iterato를 이용해서 코드를 작성하면 코드가 길고 알아보기 어렵다. 또한 같은 기능의 메소드들이 중복해서 사용되고 있다. 예를 들어 Collection.sort( ) , Arrays.sort( ) 사용한다. 이러한 문제점을 해결하기 위해 만든 것이 스트림이다. 스트림은 데이터 소스를 추상화하고 데이터를 다루는데 자주 사용되는 메서드들을 정의하였다.

Arrays.sort(strArr);
Collection.sort(strList);

for(String str : strArr)
    System.out.println(str);
for(String str : strList)
    System.out.println(str);

스트림으로 하게 되면
strStream.soted( ).foreach(System.out::println);
strStream2.soted( ).foreach(System.out::println);

스트림을 사용한 코드가 간결하고 이해하기 쉬우며 재사용이 높다는 것을 알 수 있다.