Java?NIO實(shí)戰(zhàn)之多人聊天室
本文實(shí)例為大家分享了Java NIO實(shí)戰(zhàn)之多人聊天室的具體代碼,供大家參考,具體內(nèi)容如下
NIO服務(wù)端
public class NioServer {
/**
* 啟動(dòng)
*/
public void start() throws IOException {
/**
* 1. 創(chuàng)建Selector
*/
Selector selector = Selector.open();
/**
* 2. 通過ServerSocketChannel創(chuàng)建channel通道
*/
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
/**
* 3. 為channel通道綁定監(jiān)聽端口
*/
serverSocketChannel.bind(new InetSocketAddress(8000));
/**
* 4. **設(shè)置channel為非阻塞模式**
*/
serverSocketChannel.configureBlocking(false);
/**
* 5. 將channel注冊(cè)到selector上,監(jiān)聽連接事件
*/
serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);
System.out.println("服務(wù)器啟動(dòng)成功!");
/**
* 6. 循環(huán)等待新接入的連接
*/
for (;;) { // while(true) c for;;
/**
* TODO 獲取可用channel數(shù)量
*/
int readyChannels = selector.select();
/**
* TODO 為什么要這樣?。??
*/
if (readyChannels == 0) continue;
/**
* 獲取可用channel的集合
*/
Set<SelectionKey> selectionKeys = selector.selectedKeys();
Iterator iterator = selectionKeys.iterator();
while (iterator.hasNext()) {
/**
* selectionKey實(shí)例
*/
SelectionKey selectionKey = (SelectionKey) iterator.next();
/**
* **移除Set中的當(dāng)前selectionKey**
*/
iterator.remove();
/**
* 7. 根據(jù)就緒狀態(tài),調(diào)用對(duì)應(yīng)方法處理業(yè)務(wù)邏輯
*/
/**
* 如果是 接入事件
*/
if (selectionKey.isAcceptable()) {
acceptHandler(serverSocketChannel, selector);
}
/**
* 如果是 可讀事件
*/
if (selectionKey.isReadable()) {
readHandler(selectionKey, selector);
}
}
}
}
/**
* 接入事件處理器
*/
private void acceptHandler(ServerSocketChannel serverSocketChannel,
Selector selector)
throws IOException {
/**
* 如果要是接入事件,創(chuàng)建socketChannel
*/
SocketChannel socketChannel = serverSocketChannel.accept();
/**
* 將socketChannel設(shè)置為非阻塞工作模式
*/
socketChannel.configureBlocking(false);
/**
* 將channel注冊(cè)到selector上,監(jiān)聽 可讀事件
*/
socketChannel.register(selector, SelectionKey.OP_READ);
/**
* 回復(fù)客戶端提示信息
*/
socketChannel.write(Charset.forName("UTF-8")
.encode("你與聊天室里其他人都不是朋友關(guān)系,請(qǐng)注意隱私安全"));
}
/**
* 可讀事件處理器
*/
private void readHandler(SelectionKey selectionKey, Selector selector)
throws IOException {
/**
* 要從 selectionKey 中獲取到已經(jīng)就緒的channel
*/
SocketChannel socketChannel = (SocketChannel) selectionKey.channel();
/**
* 創(chuàng)建buffer
*/
ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
/**
* 循環(huán)讀取客戶端請(qǐng)求信息
*/
String request = "";
while (socketChannel.read(byteBuffer) > 0) {
/**
* 切換buffer為讀模式
*/
byteBuffer.flip();
/**
* 讀取buffer中的內(nèi)容
*/
request += Charset.forName("UTF-8").decode(byteBuffer);
}
/**
* 將channel再次注冊(cè)到selector上,監(jiān)聽他的可讀事件
*/
socketChannel.register(selector, SelectionKey.OP_READ);
/**
* 將客戶端發(fā)送的請(qǐng)求信息 廣播給其他客戶端
*/
if (request.length() > 0) {
// 廣播給其他客戶端
broadCast(selector, socketChannel, request);
}
}
/**
* 廣播給其他客戶端
*/
private void broadCast(Selector selector,
SocketChannel sourceChannel, String request) {
/**
* 獲取到所有已接入的客戶端channel
*/
Set<SelectionKey> selectionKeySet = selector.keys();
/**
* 循環(huán)向所有channel廣播信息
*/
selectionKeySet.forEach(selectionKey -> {
Channel targetChannel = selectionKey.channel();
// 剔除發(fā)消息的客戶端
if (targetChannel instanceof SocketChannel
&& targetChannel != sourceChannel) {
try {
// 將信息發(fā)送到targetChannel客戶端
((SocketChannel) targetChannel).write(
Charset.forName("UTF-8").encode(request));
} catch (IOException e) {
e.printStackTrace();
}
}
});
}
/**
* 主方法
* @param args
*/
public static void main(String[] args) throws IOException {
new NioServer().start();
}
}
NIO客戶端
public class NioClient {
/**
* 啟動(dòng)
*/
public void start(String nickname) throws IOException {
/**
* 連接服務(wù)器端
*/
SocketChannel socketChannel = SocketChannel.open(
new InetSocketAddress("127.0.0.1", 8000));
/**
* 接收服務(wù)器端響應(yīng)
*/
// 新開線程,專門負(fù)責(zé)來接收服務(wù)器端的響應(yīng)數(shù)據(jù)
// selector , socketChannel , 注冊(cè)
Selector selector = Selector.open();
socketChannel.configureBlocking(false);
socketChannel.register(selector, SelectionKey.OP_READ);
new Thread(new NioClientHandler(selector)).start();
/**
* 向服務(wù)器端發(fā)送數(shù)據(jù)
*/
Scanner scanner = new Scanner(System.in);
while (scanner.hasNextLine()) {
String request = scanner.nextLine();
if (request != null && request.length() > 0) {
socketChannel.write(
Charset.forName("UTF-8")
.encode(nickname + " : " + request));
}
}
}
public static void main(String[] args) throws IOException {
// new NioClient().start();
}
}
客戶端線程,處理服務(wù)器端響應(yīng)的的消息
public class NioClientHandler implements Runnable {
private Selector selector;
public NioClientHandler(Selector selector) {
this.selector = selector;
}
@Override
public void run() {
try {
for (;;) {
int readyChannels = selector.select();
if (readyChannels == 0) continue;
/**
* 獲取可用channel的集合
*/
Set<SelectionKey> selectionKeys = selector.selectedKeys();
Iterator iterator = selectionKeys.iterator();
while (iterator.hasNext()) {
/**
* selectionKey實(shí)例
*/
SelectionKey selectionKey = (SelectionKey) iterator.next();
/**
* **移除Set中的當(dāng)前selectionKey**
*/
iterator.remove();
/**
* 7. 根據(jù)就緒狀態(tài),調(diào)用對(duì)應(yīng)方法處理業(yè)務(wù)邏輯
*/
/**
* 如果是 可讀事件
*/
if (selectionKey.isReadable()) {
readHandler(selectionKey, selector);
}
}
}
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* 可讀事件處理器
*/
private void readHandler(SelectionKey selectionKey, Selector selector)
throws IOException {
/**
* 要從 selectionKey 中獲取到已經(jīng)就緒的channel
*/
SocketChannel socketChannel = (SocketChannel) selectionKey.channel();
/**
* 創(chuàng)建buffer
*/
ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
/**
* 循環(huán)讀取服務(wù)器端響應(yīng)信息
*/
String response = "";
while (socketChannel.read(byteBuffer) > 0) {
/**
* 切換buffer為讀模式
*/
byteBuffer.flip();
/**
* 讀取buffer中的內(nèi)容
*/
response += Charset.forName("UTF-8").decode(byteBuffer);
}
/**
* 將channel再次注冊(cè)到selector上,監(jiān)聽他的可讀事件
*/
socketChannel.register(selector, SelectionKey.OP_READ);
/**
* 將服務(wù)器端響應(yīng)信息打印到本地
*/
if (response.length() > 0) {
System.out.println(response);
}
}
}
我們定義三個(gè)客戶端,模擬三個(gè)用戶在聊天室發(fā)送消息
public class AClient {
public static void main(String[] args)
throws IOException {
new NioClient().start("AClient");
}
}
public class BClient {
public static void main(String[] args)
throws IOException {
new NioClient().start("BClient");
}
}
public class CClient {
public static void main(String[] args)
throws IOException {
new NioClient().start("CClient");
}
}
NIO 聊天室到此結(jié)束
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
面向切面的Spring通過切點(diǎn)來選擇連接點(diǎn)實(shí)例詳解
這篇文章主要為大家介紹了面向切面的Spring通過切點(diǎn)來選擇連接點(diǎn)實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10
SpringBoot+WebSocket+Netty實(shí)現(xiàn)消息推送的示例代碼
這篇文章主要介紹了SpringBoot+WebSocket+Netty實(shí)現(xiàn)消息推送的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04
Java實(shí)現(xiàn)讀取Jar文件屬性的方法詳解
這篇文章主要為大家詳細(xì)介紹了如何利用Java語言實(shí)現(xiàn)讀取Jar文件屬性的功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-08-08
Springcloud-nacos實(shí)現(xiàn)配置和注冊(cè)中心的方法
這篇文章主要介紹了Springcloud-nacos實(shí)現(xiàn)配置和注冊(cè)中心的方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07
spring的pointcut正則表達(dá)式的實(shí)現(xiàn)
本文主要介紹了spring的pointcut正則表達(dá)式的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
SpringBoot發(fā)送郵件功能 驗(yàn)證碼5分鐘過期
這篇文章主要為大家詳細(xì)介紹了SpringBoot發(fā)送郵件功能,驗(yàn)證碼5分鐘過期,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-03-03

