java.io 패키지 (try-with-resources, InputStream, OutputStream, Reader, Writer, Serialization)
참조문서 : https://docs.oracle.com/javase/8/docs/api/java/io/package-frame.html
io 부분에서 JDK 1.4 이전 방식을 클래식 I/O 라고 하고, 이후 방식을 NIO(Non-blocking I/O)라고 한다. 이제는 클래식 IO를 사용한다 해도 내부적으론 NIO를 활용하기 때문에 성능차이는 크게 없다.
1. java에서 I/O란?
I/O를 스트림(Stream)이라고 부르기도 한다. 접두어로 in, read 가 붙으면 입력관련이고, out, write 가 붙으면 출력관련이다. 데이터를 읽고 쓰는 대상으로는 디스크, 사용자 입출력, 네트워크 등이 있다.
읽기 전용쓰기 전용
1 Byte 단위 처리 | InputStream | OutputStream |
2 Byte 단위 처리 | Reader | Writer |
2. try-with-resources 를 이용한 close 자동 처리
참조문서 : https://docs.oracle.com/javase/tutorial/essential/exceptions/tryResourceClose.html
스트림 이용에서 마지막으로 close() 처리는 필수이다.(물론 스트림뿐만 아니라 DB커넥션 등에서도..) finally 구문에서 close()를 반복적으로 호출했어야 했는데, JDK 1.7부터는 try-with-resources 로 편리하게 자원을 이용끝과 동시에 해제할 수 있다. java.lang.AutoCloseable의 구현체는 try-with-resources 구조의 resource 에 해당된다.
public static void main(String[] args) {
String str = "숨겨도 트윙클 어쩌나 눈에 확 띄잖아";
byte[] arr = str.getBytes();
try (OutputStream out = new FileOutputStream("C:\\sjh836\\test.txt")) {
out.write(arr);
} catch (Exception e) {
e.printStackTrace();
}
}
1.7 이전이였다면 try 옆에 (...) 대신에 finally로 out.close() 했어야 한다. 2개 이상이면 (...) 안에서 ; 로 구별하면 된다.
3. *InputStream 계열
InputStream는 추상클래스이다. 위 사진처럼 데이터 읽는 위치가 다양하기 때문에 많은 구현체들이 있다. InputStream을 얻는 방법으로는 구현체이거나 InputStream을 반환받으면 된다.
대표적인 구현 클래스
- FileInputStream : 파일에서 데이터를 읽음
- ZipInputStream : 압축파일 포맷(zip)에서 데이터를 읽음
- DataInputStream, BufferedInputStream : 다양한 스타일의 데이터를 읽음
주요 메소드
- read() : 1바이트를 읽어들인다. N번 호출은 N바이트를 읽는다. 반환값은 int이다. 음수가 나오는 경우는 더 읽을 데이터가 없을 때이고, 이외에는 +255를 더해서 양수가 나오게 한다.
- read(byte[]) : 한번에 byte[] 크기만큼 데이터를 읽는다.
4. *OutputStream 계열
OutputStream도 추상클래스이다.
대표적인 구현 클래스
- DataOutputStream : 객체 생성시 OutputStream을 넣어줘야한다.(확장 개념) UTF-8로 문자열 쓴다. 앞의 2Byte가 뒷 데이터가 어떻게 구성되었는지 알려준다.
주요 메소드
- write() : 1바이트를 쓴다.
- write(byte[]) : byte[] 크기만큼 쓴다.
- write(byte[], int offset, int length) : 데이터를 offset부터 length까지 byte[] 크기만큼 쓴다.
- flush() : 버퍼가 가득안찼더라도 버퍼의 내용을 바로 보내버리고 비운다.
5. *Reader 계열
Reader 는 추상클래스이다.
대표적인 구현 클래스
- BufferedReader : Reader를 생성자로 받는다. readLine()를 통해 문자열을 편하게 읽을 수 있다.
주요 메소드
- read() : 문자하나를 읽는다. 유니코드 값을 반환한다. 더 읽을 거 없으면 -1 반환
- read(char[] cbuf) : 문자 입력 스트림에서 문자를 하나씩 읽어 char[]에 저장하고 읽은 수 만큼 반환
- read(char[] cbuf, int offset, int length)
- skip(long n) : n만큼 문자를 건너뜀
Scanner와 BufferedReader의 차이
- 정리 잘된 문서 : http://mygumi.tistory.com/43
요약하면..
BufferReaderScanner
JDK 지원 | 1.1 | 1.5 |
버퍼크기(char) | 8192 | 1024 |
예외발생 | 터트림 | 숨김 |
동기화(멀티스레드 안전) | O | X |
이외에는 scanner가 읽어다가 요리하는데는 useDelimiter() 메소드처럼 더 잘 지원해준다. 참고로 scanner 나오기 전에는 InputStreamReader 클래스같은 변환해주는 것들을 많이 썼다고 한다.
6. *Writer 계열
Writer 도 추상클래스이다.
주요 메소드
- write(int c) : 문자 출력 스트림으로 c값을 char로 변환하여 쓴다.
- write(char[] cbuf) : 문자 배열을 쓴다.
- write(char[] cbuf, int offset, int length) : 문자 배열을 offset부터 length까지 쓴다.
- write(String str) : 문자열을 쓴다
- flush()
7. RandomAccessFile
파일관련해서 읽고 쓰는 작업이 RandomAccessFile 클래스 하나로 가능하다.(readxxx와 writexxx 가 공존) seek() 를 통해 마음대로 읽고 쓰는 위치를 조정할 수 있다.
이외에 io.File에서 os별 경로구분자, 파일 및 디렉터리 여부 등을 지원해준다.
8. 객체 직렬화
객체 자체를 읽거나 쓸 수도 있다. 이 때 ObjectInputStream 이나 ObjectOutputStream 을 사용한다. 이때 객체는 String 변수1, int 변수2, boolean 변수3 일 수도 있고, 변수3, 변수1, 변수2 일 수도 있겠다. 이런 잡다한 정보들은 개발자들이 신경쓰지 않도록 JVM 단에서 처리해준다. 이렇게 하면 객체의 정보의 순서없이 저장할 수 있다. 다만 오라클 jvm이나 ibm jvm 간에는 다르게 동작할 수 있다.(JDK 문서에서 이런거 까지 안정해줬으므로..)
대신 이렇게 객체를 주고받을 때는 같은 소스(ex VO.java)가 있어야한다. 이를 극복하기 위해 RMI(Remote Method Invocation) 등이 있다고 한다.
Serializable 인터페이스
메소드가 없는 인터페이스이다. 이 인터페이스를 구현하면 객체가 직렬화가 가능하다는 걸 나타내준다. 상수 serialVersionUID 는 나중에 클래스가 수정될 때 버전 정보를 가지고 확인하기 위한 수단이다.
출처: https://sjh836.tistory.com/120?category=679845 [빨간색코딩]
'JAVA > Java IO' 카테고리의 다른 글
Tomcat에서 발생하는 ClientAbortException (0) | 2021.05.11 |
---|---|
[JAVA] Externalizable 인터페이스 (0) | 2019.08.07 |
[JAVA] Serializable 과 transient (0) | 2019.08.07 |
[JAVA] serialVersionUID 이란? Warning 해결하기 (0) | 2019.08.07 |
[JAVA] 객체 직렬화 ObjectInputStream / ObjectOutputStream (0) | 2019.08.07 |
[JAVA] I/O Char 와 Encoding (0) | 2019.08.07 |
[JAVA] I/O 입출력 클래스의 종류 (0) | 2019.08.07 |
[JAVA] 문자 Stream : BufferedReader / BufferedWriter (파일 복사 예제) (0) | 2019.08.07 |