Non-Blocking Socket
๋ธ๋กํน ๋ฐฉ์์ Client๊ฐ ์ฐ๊ฒฐ ์์ฒญ์ ํ ๋๋ง๋ค accept()์์ ๋ธ๋กํน์ด ๋ฉ๋๋ค.
์ด์ ๋ฐ๋๋ก Non-blocking ๋ฐฉ์์ connect() accept() read() write()์์ ๋ธ๋กํน์ด ๋์ง ์์ต๋๋ค.
ํ์ง๋ง Blocking์ด ๋์ง ์์ผ๋ฏ๋ก ๋ฌดํ์ read,write๋ฅผ ์คํํ์ฌ CPU ์ฌ์ฉ๋ฅ ์ ์ฆ๊ฐ ์ํฌ์๋ ์๊ธฐ ๋๋ฌธ์, event Listener์ธ Selector๋ฅผ ์ฌ์ฉํ์ฌ ํน์ ์ด๋ฒคํธ๋ฅผ ๊ฐ์งํฉ๋๋ค.
Selector๋ ์ผ์ข
์ ๋ฉํฐ ํ๋ ์ ์
๋๋ค.
์ฑ๋์ด Selector์ ๋ฑ๋ก ์์ฒญ์ ํ ๋, ์์
์ ํค๋ก ์ค์ ํ ๋ค ๊ด์ฌ ํค์
(interest key)์ ์ ์ฅ ์ํต๋๋ค
๊ด์ฌ ํค์
์ ๋ฑ๋ก๋ ํค ์ค ์์
์ค๋น๊ฐ ๋ ํค๋ key Set(selected keySet)์ ์ ์ฅํ ๋ค, ํ๋์ฉ ๊บผ๋ด์ด ์ฑ๋์์
์ ์ฒ๋ฆฌํ๊ฒ ๋ฉ๋๋ค.
๋๋ธ๋กํน ๋ฐฉ์์ ์ค๋ ๋ํ์ ์ด์ฉํ์ฌ ์ ์์์ ์ค๋ ๋๋ก๋ ๋ง์ ์์
์ ์ฒ๋ฆฌํ ์ ์์ต๋๋ค.
Create Selector
์ฒซ ๋จ๊ณ๋ก Channel์ ์์ฑํด ์ค๋๋ค.
server) ServerSocetChannel
ServerSocketChannel channel = ServerSocketChannel.open();
channel.configureBlocking(false);
channel.bind(new InetSocketAddress("localhost", 8080));
client) SocketChannel
SocketChannel channel = SocketChannel.open();
channel.configureBlocking(false);
๋๋ฒ์งธ ๋จ๊ณ๊ณ๋ก register()๋ก Selector์ ๋ฑ๋กํฉ๋๋ค.
Selector selector = Selector.open();
SelectionKey selectionKey = open.register(selector, {OPTION});
ServerSocketChannel์ ์ฐ๊ฒฐ ์๋ฝ
SocketChannel์ ์๋ฒ ์ฐ๊ฒฐ
SocketChannel์ ๋ฐ์ดํฐ ์ฝ๊ธฐ
SocketChannel์ ๋ฐ์ดํฐ ์ฐ๊ธฐ
๋์ผํ ์์ผ ์ฑ๋์ ํ๊ฐ์ง์ ์์
์ ํ(option)๋ง ์ฒ๋ฆฌํ ์ ์์ต๋๋ค.
์ฝ๊ธฐ๋ฅผ ๋ฑ๋กํ ๋ค ์ฐ๊ธฐ๋ก ๋ณ๊ฒฝํ๊ธฐ ์ํด์ ๊ธฐ์กด selectionKey๋ฅผ ์์ ํด์ผ ํฉ๋๋ค
SelectionKey selectionKey = client.register(clientSelector, SelectionKey.OP_CONNECT);
selectionKey.interestOps(SelectionKey.OP_READ);
selectionKey.interestOps(SelectionKey.OP_WRITE);
์ธ๋ฒ์งธ ๋จ๊ณ๋ก Select๋ฅผ ํธ์ถํฉ๋๋ค.
๋ฑ๋ก๋ SelectionKey์์ ํ๋ ์ด์์ ์ฑ๋์ ์์
์ค๋น ์๋ต์ ๊ธฐ๋ค๋ฆฝ๋๋ค.
select()๋ blocking์
๋๋ค
clientSelector.select(); // ์๋ต ๋๊ธฐ
clientSelector.select(1000L); // 1์ด ๋๊ธฐ
clientSelector.selectNow(); // ์ฆ์ ํธ์ถ ์์ผ๋ฉด 0
select()๋์ฑ๋์ ์ค๋น ์๋ฃ ์๋ต, Selector์ wakeUp(), Thread Interrupt์ ์ํด ๋ฆฌํด๋ฉ๋๋ค.
Selector clientSelector = Selector.open();
SelectionKey selectionKey = client.register(clientSelector, SelectionKey.OP_CONNECT);
selectionKey.interestOps(SelectionKey.OP_WRITE);
clientSelector.wakeup();
Channel Process
ํค์
์์ SelectionKey๋ค์ ๋ฐ์์ ์ ํ๋ณ๋ก ์์
์ ์ฒ๋ฆฌํฉ๋๋ค.
public class SelectorWorker extends Thread {
private final Selector selector;
public SelectorWorker(Selector selector) {
this.selector = selector;
}
@Override
public void run() {
while (true) {
if (isSelectActive()) continue;
findOption();
}
}
private boolean isSelectActive(){
try {
if (selector.select() == 0) {
return true;
}
} catch (IOException e) {
throw new RuntimeException(e);
}
return false;
}
private void findOption() {
selector.selectedKeys().forEach(selectionKey -> {
switch (selectionKey.readyOps()) {
case SelectionKey.OP_ACCEPT -> System.out.println("OPTION = OP_ACCEPT");
case SelectionKey.OP_CONNECT -> System.out.println("OPTION = OP_CONNECT");
case SelectionKey.OP_READ -> System.out.println("OPTION = OP_READ");
case SelectionKey.OP_WRITE -> System.out.println("OPTION = OP_WRITE");
default -> System.out.println("OPTION = UNKNOWN");
}
});
}
}
์ฑ๋ ๊ฐ์ฒด๋ selectionKey์์ ๊ฐ์ ธ์ฌ ์ ์์ต๋๋ค.
ServerSocketChannel channel = (ServerSocketChannel) selectionKey.channel();
์ฑ๋ ๊ฐ์ฒด ์ธ์ ๋ค๋ฅธ ๊ฐ์ฒด๋ selectionKey์ ์ฃผ์
ํ ์ ์๋๋ฐ, ๊ฐ์ฒด๋ฅผ ๋ฑ๋กํ ๋ค selectionKey์์ ๊บผ๋ด์ ์ฌ์ฉํ๋ฉด ๋ฉ๋๋ค.
SelectionKey selectionKey = client.register(clientSelector, SelectionKey.OP_CONNECT);
selectionKey.attach(new Attached());
selector.selectedKeys().forEach(selectionKey -> {
Attached attached = (Attached) selectionKey.attachment();
});