java网络编程的心得_java网络编程总结

OSI网络七层模型

为了不同计算机厂商生产的电脑能够通信,以便在更大的范围内建立计算机网络,有必要建立一个国际范围内的网络体系结构标准,也就有了OSI网络七层模型

10091cec53c9937fc6a5bcce36081125.png

TCP是一个重要的传输层协议,提供面向连接、可靠、有序的字节流传输服务,在传输数据前必须先建立tcp连接,传输报文如下:

9ef2c2ed7e58a202b62157a4837b74e2.png

tcp三次握手和四次挥手

95ca965b2b81a11dd8773b8a2e9cf518.png

UDP协议提供无连接、不可靠的数据包尽力传输的服务

a25fd3658da3624aad2a40cc8f638837.png

BIO阻塞式网络编程

bio网络编程基于socket和io,先看一个简单的client程序和server端程序

packagecom.example.test.BIO;import java.io.*;importjava.net.Socket;importjava.util.Scanner;/***@authorhehang on 2019-05-24

* @descriptionasd*/

public classBioClient {public static void main(String[] args) throwsIOException {

Socket socket= new Socket("localhost",8080);

OutputStream outputStream=socket.getOutputStream();

BufferedWriter bufferedWriter= new BufferedWriter(new OutputStreamWriter(outputStream,"utf-8"));

Scanner scanner= newScanner(System.in);

System.out.println("请输入:");

String msg=scanner.nextLine();

bufferedWriter.write(msg);

scanner.close();

socket.close();

}

}

packagecom.example.test.BIO;importjava.io.BufferedReader;importjava.io.IOException;importjava.io.InputStream;importjava.io.InputStreamReader;importjava.net.ServerSocket;importjava.net.Socket;/***@authorhehang on 2019-05-24

* @descriptionasd*/

public classBioServer {public static void main(String[] args) throwsException {

ServerSocket serverSocket= new ServerSocket(8080);

System.out.println("server启动");while(!serverSocket.isClosed()){

Socket socket=serverSocket.accept();try{

InputStream in=socket.getInputStream();

BufferedReader bufferedReader= new BufferedReader(newInputStreamReader(in));

String msg= null;while ((msg = bufferedReader.readLine()) != null) {if (msg.length() == 0) {break;

}

System.out.println(msg);

}

System.out.println("收到消息,来自:" +socket.toString());

}catch(IOException e) {

e.printStackTrace();

}finally{

socket.close();

}

}

serverSocket.close();

}

}

由于serverSocket.accept()及bufferedReader.readLine()都是阻塞的方法,因此server端是一个单线程的,不能满足多个client同时请求,改造server端,使其支持多线程

packagecom.example.test.BIO;importjava.io.BufferedReader;importjava.io.IOException;importjava.io.InputStream;importjava.io.InputStreamReader;importjava.net.ServerSocket;importjava.net.Socket;importjava.util.concurrent.ExecutorService;importjava.util.concurrent.Executors;/***@authorhehang on 2019-05-24

* @descriptionasd*/

public classBioServer2 {private static ExecutorService threadPool =Executors.newCachedThreadPool();public static void main(String[] args) throwsException {

ServerSocket serverSocket= new ServerSocket(8080);

System.out.println("server启动");while(!serverSocket.isClosed()){

Socket socket=serverSocket.accept();

System.out.println("收到新的请求" +socket.toString());

threadPool.execute(()->{try{

InputStream in=socket.getInputStream();

String msg= null;while ((msg = bufferedReader.readLine()) != null) {if (msg.length() == 0) {break;

}

System.out.println(msg);

}

System.out.println("收到消息,来自:" +socket.toString());

}catch(IOException e) {

e.printStackTrace();

}finally{try{

socket.close();

}catch(IOException e) {

e.printStackTrace();

}

}

});

}

serverSocket.close();

}

}

此时sever端的并发数就受限于线程池的大小,那么浏览器和我们server端如何交互呢,此时就需要对http协议有基础的认识

http请求数据包解析

897922bdd0c8bcef04ac005cc2be5cc5.png

http响应数据包解析

87df66f56f5fdf3356d0988e0320872b.png

http响应码状态

1d8bc390c7b733ed77e98f6e6c399a14.png

对现有的server端程序改造,使其支持http请求

packagecom.example.test.BIO;import java.io.*;importjava.net.ServerSocket;importjava.net.Socket;importjava.util.concurrent.ExecutorService;importjava.util.concurrent.Executors;/***@authorhehang on 2019-05-24

* @descriptionasd*/

public classBioServer3 {private static ExecutorService threadPool =Executors.newCachedThreadPool();public static void main(String[] args) throwsException {

ServerSocket serverSocket= new ServerSocket(8080);

System.out.println("server启动");while(!serverSocket.isClosed()){

Socket socket=serverSocket.accept();

System.out.println("收到新的请求" +socket.toString());

threadPool.execute(()->{try{

InputStream in=socket.getInputStream();

BufferedReader bufferedReader= new BufferedReader(new InputStreamReader(in,"utf-8"));

String msg= null;while ((msg = bufferedReader.readLine()) != null) {if (msg.length() == 0) {break;

}

System.out.println(msg);

}

System.out.println("收到消息,来自:" +socket.toString());

OutputStream outputStream=socket.getOutputStream();

outputStream.write("HTTP/1.1 200 OK\r\n".getBytes());

outputStream.write("Content-Length: 11\r\n\r\n".getBytes());

outputStream.write("Hello World".getBytes());

outputStream.flush();

bufferedReader.close();

outputStream.close();

}catch(IOException e) {

e.printStackTrace();

}finally{try{

socket.close();

}catch(IOException e) {

e.printStackTrace();

}

}

});

}

serverSocket.close();

}

}

改造client端,使其能够接受server端的响应

packagecom.example.test.BIO;import java.io.*;importjava.net.Socket;importjava.util.Scanner;/***@authorhehang on 2019-05-24

* @descriptionasd*/

public classBioClient {public static void main(String[] args) throwsIOException {

Socket socket= new Socket("localhost",8080);

OutputStream outputStream=socket.getOutputStream();

BufferedWriter bufferedWriter= new BufferedWriter(new OutputStreamWriter(outputStream,"utf-8"));

Scanner scanner= newScanner(System.in);

System.out.println("请输入:");

String msg=scanner.nextLine();

bufferedWriter.write(msg);

bufferedWriter.flush();

socket.shutdownOutput();

InputStream inputStream=socket.getInputStream();

BufferedReader bufferedReader= new BufferedReader(newInputStreamReader(inputStream));

String returnMsg= null;while((returnMsg = bufferedReader.readLine())!=null){if(returnMsg.length()==0){break;

}

System.out.println(returnMsg);

}

scanner.close();

socket.close();

}

}

阻塞式编程相关概念

1825f0b4a1699eca129355d08ec162b7.png

NIO网络编程

从java1.4开始,出现了新的JAVA IO非阻塞API,NIO有三个核心概念:Buffer缓冲区、Channel通道和Selector选择器

Buffer缓冲区

859d2969cc40ff6dfffcc4c4ebc60293.png

4dd9ef657607d39d1fcb6fc6ab376d51.png

e841817f57ce4729d183920ffbbce345.png

demo如下

packagecom.example.test.NIO;importjava.nio.ByteBuffer;/***@authorhehang on 2019-05-24

* @descriptionsdf*/

public classBufferDemo {public static voidmain(String[] args) {

ByteBuffer byteBuffer= ByteBuffer.allocate(4);

System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",

byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));

byteBuffer.put((byte) 1);

byteBuffer.put((byte) 2);

byteBuffer.put((byte) 3);

System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",

byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));

System.out.println("开始读取");//切换为读模式

byteBuffer.flip();byte a =byteBuffer.get();

System.out.println(a);byte c =byteBuffer.get();

System.out.println(c);

System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",

byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));//清除已读取的数据,转为写模式

byteBuffer.compact();

System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",

byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));

byteBuffer.put((byte) 3);

byteBuffer.put((byte) 4);

byteBuffer.put((byte) 5);

System.out.println(String.format("最终的情况,capacity容量:%s, position位置:%s, limit限制:%s", byteBuffer.capacity(),

byteBuffer.position(), byteBuffer.limit()));

}

}

packagecom.example.test.NIO;importjava.nio.ByteBuffer;/***@authorhehang on 2019-05-24

* @descriptionasd*/

public classDerictBufferDemo {public static voidmain(String[] args) {

ByteBuffer byteBuffer= ByteBuffer.allocateDirect(4);

System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",

byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));

byteBuffer.put((byte) 1);

byteBuffer.put((byte) 2);

byteBuffer.put((byte) 3);

System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",

byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));

System.out.println("开始读取");//切换为读模式

byteBuffer.flip();byte a =byteBuffer.get();

System.out.println(a);byte c =byteBuffer.get();

System.out.println(c);

System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",

byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));//清除已读取的数据,转为写模式

byteBuffer.compact();

System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",

byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));

byteBuffer.put((byte) 3);

byteBuffer.put((byte) 4);

byteBuffer.put((byte) 5);

System.out.println(String.format("最终的情况,capacity容量:%s, position位置:%s, limit限制:%s", byteBuffer.capacity(),

byteBuffer.position(), byteBuffer.limit()));

}

}

Channel通道

60cd79ff2da55c9928071b0506dbe896.png

利用nio构建简单的client程序和server端程序

packagecom.example.test.NIO;importjava.io.IOException;importjava.net.InetSocketAddress;importjava.nio.ByteBuffer;importjava.nio.channels.SocketChannel;importjava.util.Scanner;public classNioClient {public static void main(String[] args) throwsIOException {

SocketChannel socketChannel=SocketChannel.open();

socketChannel.configureBlocking(false);

socketChannel.connect(new InetSocketAddress("127.0.0.1",8080));while(!socketChannel.finishConnect()){//没连接上则一直连接

Thread.yield();

}

Scanner scanner= newScanner(System.in);

System.out.println("请输入:");

String msg=scanner.nextLine();

ByteBuffer byteBuffer=ByteBuffer.wrap(msg.getBytes());while(byteBuffer.hasRemaining()){

socketChannel.write(byteBuffer);

}

System.out.println("收到服务端的响应:");

ByteBuffer returnBuffer= ByteBuffer.allocate(1024);while(socketChannel.isOpen()&& socketChannel.read(returnBuffer)!=-1){if(returnBuffer.position()>0){break;

}

}

returnBuffer.flip();

System.out.println(returnBuffer.limit());byte[] content = new byte[returnBuffer.limit()];

returnBuffer.get(content);

System.out.println(newString(content));

scanner.close();

socketChannel.close();

}

}

packagecom.example.test.NIO;importjava.io.IOException;importjava.net.InetSocketAddress;importjava.nio.ByteBuffer;importjava.nio.channels.ServerSocketChannel;importjava.nio.channels.SocketChannel;public classNioServer {public static void main(String[] args) throwsIOException {//创建网络服务

ServerSocketChannel serverSocketChannel =ServerSocketChannel.open();

serverSocketChannel.configureBlocking(false);//设置为非阻塞

serverSocketChannel.socket().bind(new InetSocketAddress(8080));//绑定端口

System.out.println("启动成功");while(true){

SocketChannel socketChannel= serverSocketChannel.accept();//获取tcp连接通道

if(socketChannel!=null){

System.out.println("收到新连接" +socketChannel.getRemoteAddress());

socketChannel.configureBlocking(false);//设置为非阻塞

ByteBuffer requestByte = ByteBuffer.allocate(1024);while(socketChannel.isOpen()&& socketChannel.read(requestByte)!=-1){//长连接情况下,需要手动判断数据是否读取完毕,此处做一个简单判断,position>0表示读完

if(requestByte.position()>0){break;

}

}if(requestByte.position()==0){continue;//没数据则不继续后续处理

}

requestByte.flip();byte[] content = new byte[requestByte.limit()];

requestByte.get(content);

System.out.println(newString(content));

System.out.println("收到数据,来自" +socketChannel.getRemoteAddress());

String response= "HTTP/1.1 200 OK \r\n" + "Content_Length: 11 \r\n\r\n" + "Hello Word";

ByteBuffer returnByteBuffer=ByteBuffer.wrap(response.getBytes());while(returnByteBuffer.hasRemaining()){

socketChannel.write(returnByteBuffer);

}

}

}

}

}

此时发现由于循环读取ByteBuffer中的数据,server只能同时接受一个请求,因此需要对server进行改造

packagecom.example.test.NIO;importjava.io.IOException;importjava.net.InetSocketAddress;importjava.nio.ByteBuffer;importjava.nio.channels.ServerSocketChannel;importjava.nio.channels.SocketChannel;importjava.util.ArrayList;importjava.util.Iterator;/***@authorhehang on 2019-05-24

* @descriptionasd*/

public classNioServer1 {private static ArrayList socketChannels = new ArrayList<>();public static void main(String[] args) throwsIOException {//创建网络服务

ServerSocketChannel serverSocketChannel =ServerSocketChannel.open();

serverSocketChannel.configureBlocking(false);//设置为非阻塞

serverSocketChannel.socket().bind(new InetSocketAddress(8080));//绑定端口

System.out.println("启动成功");while(true){

SocketChannel socketChannel= serverSocketChannel.accept();//获取tcp连接通道

if(socketChannel!=null) {

System.out.println("收到新连接" +socketChannel.getRemoteAddress());

socketChannel.configureBlocking(false);//设置为非阻塞

socketChannels.add(socketChannel);

}else{//没有新连接的时候,就去处理现有连接的数据

Iterator iterator =socketChannels.iterator();while(iterator.hasNext()){

SocketChannel socketChannel1=iterator.next();

ByteBuffer requestByte= ByteBuffer.allocate(1024);if(socketChannel1.read(requestByte)==0){continue;

}while(socketChannel1.isOpen()&& socketChannel1.read(requestByte)!=-1){//长连接情况下,需要手动判断数据是否读取完毕,此处做一个简单判断,position>0表示读完

if(requestByte.position()>0){break;

}

}

requestByte.flip();byte[] content = new byte[requestByte.limit()];

requestByte.get(content);

System.out.println(newString(content));

System.out.println("收到数据,来自" +socketChannel1.getRemoteAddress());

String response= "HTTP/1.1 200 OK \r\n" + "Content_Length: 11 \r\n\r\n" + "Hello Word";

ByteBuffer returnByteBuffer=ByteBuffer.wrap(response.getBytes());while(returnByteBuffer.hasRemaining()){

socketChannel1.write(returnByteBuffer);

}

iterator.remove();

}

}

}

}

}

利用ByteBuffer和Channel已经可以进行NIO编程,但是这样却很不方便,JDK为我们提供一个新的API:Selector

7707e5b90e68256eff46b91713c7caef.png

改造server端程序如下

packagecom.example.test.NIO;importjava.io.IOException;importjava.net.InetSocketAddress;importjava.nio.ByteBuffer;importjava.nio.channels.SelectionKey;importjava.nio.channels.Selector;importjava.nio.channels.ServerSocketChannel;importjava.nio.channels.SocketChannel;importjava.util.Iterator;importjava.util.Set;public classNioServer2 {public static void main(String[] args) throwsIOException {//创建网络服务

ServerSocketChannel serverSocketChannel =ServerSocketChannel.open();

serverSocketChannel.configureBlocking(false);//设置为非阻塞//构建选择器,将serverSocketChannel注册上去,并且selector对serverSocketChannel上面的accept事件感兴趣

Selector selector =Selector.open();

SelectionKey selectionKey= serverSocketChannel.register(selector,0,serverSocketChannel);

selectionKey.interestOps(SelectionKey.OP_ACCEPT);

serverSocketChannel.socket().bind(new InetSocketAddress(8080));//绑定端口

System.out.println("启动成功");while(true){

selector.select();//该方法会阻塞,直到有事件通知才会返回//获取事件

Set selectionKeys =selector.selectedKeys();

Iterator iter =selectionKeys.iterator();while(iter.hasNext()){//System.out.println("-------------");

SelectionKey key =iter.next();

iter.remove();if(key.isAcceptable()){

ServerSocketChannel server=(ServerSocketChannel) key.attachment();//将拿到的客户端连接通道,注册到selector上面

SocketChannel clientSocketChannel = server.accept(); //mainReactor 轮询accept

clientSocketChannel.configureBlocking(false);

clientSocketChannel.register(selector, SelectionKey.OP_READ, clientSocketChannel);

System.out.println("收到新连接 : " +clientSocketChannel.getRemoteAddress());

}if(key.isReadable()){

SocketChannel socketChannel=(SocketChannel) key.attachment();try{

ByteBuffer requestBuffer= ByteBuffer.allocate(1024);while (socketChannel.isOpen() && socketChannel.read(requestBuffer) != -1) {//长连接情况下,需要手动判断数据有没有读取结束 (此处做一个简单的判断: 超过0字节就认为请求结束了)

if (requestBuffer.position() > 0) break;

}if(requestBuffer.position() == 0) continue; //如果没数据了, 则不继续后面的处理

requestBuffer.flip();byte[] content = new byte[requestBuffer.limit()];

requestBuffer.get(content);

System.out.println(newString(content));

System.out.println("收到数据,来自:" +socketChannel.getRemoteAddress());//TODO 业务操作 数据库 接口调用等等//响应结果 200

String response = "HTTP/1.1 200 OK\r\n" +

"Content-Length: 11\r\n\r\n" +

"Hello World";

ByteBuffer buffer=ByteBuffer.wrap(response.getBytes());while(buffer.hasRemaining()) {

socketChannel.write(buffer);

}

}catch(IOException e) {//e.printStackTrace();

key.cancel(); //取消事件订阅

}

}

}

selector.selectNow();

}

}

}

在上面的server端程序中,一个selector监听所有事件,一个线程处理所有请求事件,难以利用现代服务器多核特性,会成为性能瓶颈!,因此实际开发中要有多线程的运用,对此,JDK作者给出了NIO与多线程的结合使用:《Scalable IO in java》

864af869623de4f1b7d085c584151f83.png

下面是多Reactor服务端实现

packagecom.example.test.NIO;importjava.io.IOException;importjava.net.InetSocketAddress;importjava.nio.ByteBuffer;import java.nio.channels.*;importjava.util.Iterator;importjava.util.Random;importjava.util.Set;importjava.util.concurrent.ExecutorService;importjava.util.concurrent.Executors;importjava.util.concurrent.FutureTask;importjava.util.concurrent.LinkedBlockingQueue;importjava.util.concurrent.atomic.AtomicInteger;/***@authorhehang on 2020-02-02

* @description*/

public classNioServer3 {/**处理业务操作的线程*/

private static ExecutorService workPool =Executors.newCachedThreadPool();/*** 封装了selector.select()等事件轮询的代码*/

abstract class ReactorThread extendsThread {

Selector selector;

LinkedBlockingQueue taskQueue = new LinkedBlockingQueue<>();/*** Selector监听到有事件后,调用这个方法*/

public abstract void handler(SelectableChannel channel) throwsException;private ReactorThread() throwsIOException {

selector=Selector.open();

}volatile boolean running = false;

@Overridepublic voidrun() {//轮询Selector事件

while(running) {try{//执行队列中的任务

Runnable task;while ((task = taskQueue.poll()) != null) {

task.run();

}

selector.select(1000);//获取查询结果

Set selected =selector.selectedKeys();//遍历查询结果

Iterator iter =selected.iterator();while(iter.hasNext()) {//被封装的查询结果

SelectionKey key =iter.next();

iter.remove();int readyOps =key.readyOps();//关注 Read 和 Accept两个事件

if ((readyOps & (SelectionKey.OP_READ | SelectionKey.OP_ACCEPT)) != 0 || readyOps == 0) {try{

SelectableChannel channel=(SelectableChannel) key.attachment();

channel.configureBlocking(false);

handler(channel);if (!channel.isOpen()) {

key.cancel();//如果关闭了,就取消这个KEY的订阅

}

}catch(Exception ex) {

key.cancel();//如果有异常,就取消这个KEY的订阅

}

}

}

selector.selectNow();

}catch(IOException e) {

e.printStackTrace();

}

}

}private SelectionKey register(SelectableChannel channel) throwsException {//为什么register要以任务提交的形式,让reactor线程去处理?//因为线程在执行channel注册到selector的过程中,会和调用selector.select()方法的线程争用同一把锁//而select()方法实在eventLoop中通过while循环调用的,争抢的可能性很高,为了让register能更快的执行,就放到同一个线程来处理

FutureTask futureTask = new FutureTask<>(() -> channel.register(selector, 0, channel));

taskQueue.add(futureTask);returnfutureTask.get();

}private voiddoStart() {if (!running) {

running= true;

start();

}

}

}privateServerSocketChannel serverSocketChannel;//1、创建多个线程 - accept处理reactor线程 (accept线程)

private ReactorThread[] mainReactorThreads = new ReactorThread[1];//2、创建多个线程 - io处理reactor线程 (I/O线程)

private ReactorThread[] subReactorThreads = new ReactorThread[8];/*** 初始化线程组*/

private void newGroup() throwsIOException {//创建IO线程,负责处理客户端连接以后socketChannel的IO读写

for (int i = 0; i < subReactorThreads.length; i++) {

subReactorThreads[i]= newReactorThread() {

@Overridepublic void handler(SelectableChannel channel) throwsIOException {//work线程只负责处理IO处理,不处理accept事件

SocketChannel ch =(SocketChannel) channel;

ByteBuffer requestBuffer= ByteBuffer.allocate(1024);while (ch.isOpen() && ch.read(requestBuffer) != -1) {//长连接情况下,需要手动判断数据有没有读取结束 (此处做一个简单的判断: 超过0字节就认为请求结束了)

if (requestBuffer.position() > 0) break;

}if (requestBuffer.position() == 0) return; //如果没数据了, 则不继续后面的处理

requestBuffer.flip();byte[] content = new byte[requestBuffer.limit()];

requestBuffer.get(content);

System.out.println(newString(content));

System.out.println(Thread.currentThread().getName()+ "收到数据,来自:" +ch.getRemoteAddress());//TODO 业务操作 数据库、接口...

workPool.submit(() ->{

});//响应结果 200

String response = "HTTP/1.1 200 OK\r\n" +

"Content-Length: 11\r\n\r\n" +

"Hello World";

ByteBuffer buffer=ByteBuffer.wrap(response.getBytes());while(buffer.hasRemaining()) {

ch.write(buffer);

}

}

};

}//创建mainReactor线程, 只负责处理serverSocketChannel

for (int i = 0; i < mainReactorThreads.length; i++) {

mainReactorThreads[i]= newReactorThread() {

AtomicInteger incr= new AtomicInteger(0);

@Overridepublic void handler(SelectableChannel channel) throwsException {//只做请求分发,不做具体的数据读取

ServerSocketChannel ch =(ServerSocketChannel) channel;

SocketChannel socketChannel=ch.accept();

socketChannel.configureBlocking(false);//收到连接建立的通知之后,分发给I/O线程继续去读取数据

int index = incr.getAndIncrement() %subReactorThreads.length;

ReactorThread workEventLoop=subReactorThreads[index];

workEventLoop.doStart();

SelectionKey selectionKey=workEventLoop.register(socketChannel);

selectionKey.interestOps(SelectionKey.OP_READ);

System.out.println(Thread.currentThread().getName()+ "收到新连接 : " +socketChannel.getRemoteAddress());

}

};

}

}/*** 初始化channel,并且绑定一个eventLoop线程

*

*@throwsIOException IO异常*/

private void initAndRegister() throwsException {//1、 创建ServerSocketChannel

serverSocketChannel =ServerSocketChannel.open();

serverSocketChannel.configureBlocking(false);//2、 将serverSocketChannel注册到selector

int index = newRandom().nextInt(mainReactorThreads.length);

mainReactorThreads[index].doStart();

SelectionKey selectionKey=mainReactorThreads[index].register(serverSocketChannel);

selectionKey.interestOps(SelectionKey.OP_ACCEPT);

}/*** 绑定端口

*

*@throwsIOException IO异常*/

private void bind() throwsIOException {//1、 正式绑定端口,对外服务

serverSocketChannel.bind(new InetSocketAddress(8080));

System.out.println("启动完成,端口8080");

}public static void main(String[] args) throwsException {

NioServer3 nioServer3= newNioServer3();

nioServer3.newGroup();//1、 创建main和sub两组线程

nioServer3.initAndRegister(); //2、 创建serverSocketChannel,注册到mainReactor线程上的selector上

nioServer3.bind(); //3、 为serverSocketChannel绑定端口

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值