Android面试题之Java网络通信基础知识

简介: Socket是应用与TCP/IP通信的接口,封装了底层细节。网络通信涉及连接、读写数据。BIO是同步阻塞,NIO支持多路复用(如Selector),AIO在某些平台提供异步非阻塞服务。BIO示例中,服务端用固定线程池处理客户端请求,客户端发起连接并读写数据。NIO的关键是Selector监控多个通道的事件,减少线程消耗。书中推荐《Java网络编程》和《UNIX网络编程》。关注公众号AntDream了解更多。

本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点

网络通信基础

socket

socket是应用层与TCP/IP协议族通信的中间软件抽象,操作系统把传输层一下的内容都包装了,应用层只需要用socket即可完成网络请求

Tcp是基于流;UDP是基于DatagramPacket数据报;socket可以利用DatagramPacket进行UDP通信

网络请求3部分
  • 连接(客户端和服务端)
  • 读网络数据
  • 写网络数据
BIO、NIO、AIO
  • BIO:阻塞的IO
  • NIO(IO多路复用):一个线程同时管理多个和客户端的连接,一般应用线程数量限制为1024个(系统限制,总共65535,要除去系统使用)
  • AIO(异步IO):目前只有Windows上完全实现

BIO网络编程

服务端
private static ExecutorService executorService = Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors()*2);
    public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = new ServerSocket();
        serverSocket.bind(new InetSocketAddress(10001));
        System.out.println("start server ....");
        try {
            while (true){
                executorService.execute(new ServerTask(serverSocket.accept()));
            }

        }finally {
            serverSocket.close();
        }
    }

    private static class ServerTask implements Runnable{
        private Socket socket;

        public ServerTask(Socket socket) {
            this.socket = socket;
        }

        @Override
        public void run() {
            //这样会自动关闭流
            try (ObjectInputStream inputStream = new ObjectInputStream(socket.getInputStream());
                 ObjectOutputStream outputStream = new ObjectOutputStream(socket.getOutputStream());) {
                String input = inputStream.readUTF();
                System.out.println("accept input :" +input);
                outputStream.writeUTF("hello " + input);
                outputStream.flush();

            } catch (IOException e) {
                e.printStackTrace();
            }finally {
                if (socket != null) {
                    try {
                        socket.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    }
客户端
public static void main(String[] args) throws IOException {
        Socket socket = new Socket();
        InetSocketAddress inetSocketAddress = new InetSocketAddress("127.0.0.1", 10001);
        ObjectOutputStream outputStream = null;
        ObjectInputStream inputStream = null;

        try {
            socket.connect(inetSocketAddress);
            outputStream = new ObjectOutputStream(socket.getOutputStream());
            inputStream = new ObjectInputStream(socket.getInputStream());
            outputStream.writeUTF("hello");
            outputStream.flush();

            System.out.println(inputStream.readUTF());

        } finally {
            if (socket != null) {
                socket.close();
            }
            if (outputStream != null) {
                outputStream.close();
            }
            if (inputStream != null) {
                inputStream.close();
            }
        }
    }

BIO网络编程

Selector、ServerSocketChannel、SocketChannel、Buffer、应用程序

服务端一个线程里面,会有一个Selector,用于接收客户端发来的事件,比如连接、有数产生事件等

ServerSocketChannel:用于处理连接事件,会在Selector里注册OP_ACCEPT事件监听,Selector收到客户端连接事件后会通知ServerSocketChannel;

SocketChannel:用于处理数据读的事件,一个客户端会对应一个SocketChannel;同样它会注册数据读的事件到Selector,有数据发送来了以后,Selector就会通知对应的SocketChannel;

Buffer:用于SocketChannel与服务端应用程序直接的数据操作,有数据时,SocketChannel会写到缓冲区Buffer中,然后应用程序从Buffer中读数据,处理完又写回到Buffer,然后SocketChannel从Buffer中读

有新的客户端连接进来后,会创建新的SocketChannel,进行数据的读写通信

Buffer
Buffer是内存中的一块区域,是一个字节数组,有3个重要变量:Position、limit、capacity;模式的话有读和写2种模式,不能同时进行读和写
  • position:指示当前数据的位置,写模式下,会随着写入数据而移动;当调用flip方法切换到读的模式时,position和limit会发生变化,position会指向0,limit会变成position的值,也就是之前最后写入数据的position
  • capacity:Buffer的容量
Buffer的内存分配:可以在堆上分配,也可以在直接内存上分配
  • 堆上分配:分配速度会快一点,网络通信慢一点,一般业务处理方面多一点

    ByteBuffer buffer = ByteBuffer.allocate(20000);
    
  • 直接内存分配:分配速度会慢一点,网络通信会快一点,一般直接读写网络数据用这个

    ByteBuffer buffer = ByteBuffer.allocateDirect(20000);
    
  • 把byte数组转换为Buffer

    ByteBuffer.wrap(bytes)
    
Buffer的写
  • Channel向Buffer写:用Buffer的read方法,表示从网络读数据写到Buffer
  • 应用程序向Buffer写:用Buffer提供的各种put方法
Buffer的读
  • 从Buffer读数据写到Channel:用Buffer的write方法,表示从Buffer读数据写到channel
  • Buffer从应用程序读:用Buffer的get方法

NIO网络编程

注意点
  • ServerSocketChannel和SocketChannel都要设置为非阻塞,configBolcking(false)
  • selector和ServerSocketChannel初始化的方式都是调用open函数
  • 调用selector.select(1000)表示开启事件监听,每个1秒唤醒一次,但是有事件过来会立刻唤醒处理
  • ServerSocketChannel和SocketChannel通过register函数来向selector注册事件监听
  • register函数可以同时支持多个事件的监听,用位操作符连接:OP_CONNNECT | OP_READ
  • register函数还可以同时注册Buffer,作为第3个参数,这样可以在其他地方,通过SelectionKey的attachment拿到
  • 所有的事件都在SelectionKey里面,包括OP_CONNNECT,OP_READ,OP_WRITE,OP_ACCEPT
  • Buffer在读模式切换到写模式的时候,需要调用flip方法进行模式切换
OP_WRITE和OP_READ事件

我们一般很少使用OP_WRITE事件,因为操作系统在建立socket连接的时候,会为socket创建2个缓冲区(Buffer),一个发送Buffer,一个接收Buffer;当客户端向服务端发送数据的时候,服务端接收缓存接收到数据,就会触发OP_READ事件;这2个缓冲区由操作系统内核管理。

而发送缓存里面只要还有空间可写,哪怕一个字节,就会触发OP_WRITE事件

OP_WRITE使用方式
  • 用SelectionKey的isWritable判断是否有数据
  • Buffer用SelectionKey的attachment来获取
  • 根据Buffer的hasRemaining来判断是否还有数据可写
  • 数据都写完的话要取消写事件的监听,用SelectionKey的interestOps方法
OKhttp中的心跳包

websocket实现,websocket本身定义了跟心跳有关的2个数据帧,实现以后,服务端会自动解析和应答

书籍
  • 《Java网络编程》:网络编程、UDP用socket实现等
  • 《UNIX网络编程》:socket原理等
  • 《Netty实战》:Netty相关的网络开发

欢迎关注我的公众号AntDream查看更多精彩文章!

目录
相关文章
|
12月前
|
存储 监控 安全
单位网络监控软件:Java 技术驱动的高效网络监管体系构建
在数字化办公时代,构建基于Java技术的单位网络监控软件至关重要。该软件能精准监管单位网络活动,保障信息安全,提升工作效率。通过网络流量监测、访问控制及连接状态监控等模块,实现高效网络监管,确保网络稳定、安全、高效运行。
267 11
|
4月前
|
JSON 移动开发 网络协议
Java网络编程:Socket通信与HTTP客户端
本文全面讲解Java网络编程,涵盖TCP与UDP协议区别、Socket编程、HTTP客户端开发及实战案例,助你掌握实时通信、文件传输、聊天应用等场景,附性能优化与面试高频问题解析。
|
2月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
174 1
|
2月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
198 1
|
2月前
|
机器学习/深度学习 分布式计算 Java
Java与图神经网络:构建企业级知识图谱与智能推理系统
图神经网络(GNN)作为处理非欧几里得数据的前沿技术,正成为企业知识管理和智能推理的核心引擎。本文深入探讨如何在Java生态中构建基于GNN的知识图谱系统,涵盖从图数据建模、GNN模型集成、分布式图计算到实时推理的全流程。通过具体的代码实现和架构设计,展示如何将先进的图神经网络技术融入传统Java企业应用,为构建下一代智能决策系统提供完整解决方案。
328 0
|
Java 程序员
JAVA程序员的进阶之路:掌握URL与URLConnection,轻松玩转网络资源!
在Java编程中,网络资源的获取与处理至关重要。本文介绍了如何使用URL与URLConnection高效、准确地获取网络资源。首先,通过`java.net.URL`类定位网络资源;其次,利用`URLConnection`类实现资源的读取与写入。文章还提供了最佳实践,包括异常处理、连接池、超时设置和请求头与响应头的合理配置,帮助Java程序员提升技能,应对复杂网络编程场景。
254 9
|
人工智能 Java 物联网
JAVA网络编程的未来:URL与URLConnection的无限可能,你准备好了吗?
随着技术的发展和互联网的普及,JAVA网络编程迎来新的机遇。本文通过案例分析,探讨URL与URLConnection在智能API调用和实时数据流处理中的关键作用,展望其未来趋势和潜力。
175 7
|
9月前
|
存储 网络协议 安全
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
363 23
|
10月前
|
安全 网络协议 Java
Java网络编程封装
Java网络编程封装原理旨在隐藏底层通信细节,提供简洁、安全的高层接口。通过简化开发、提高安全性和增强可维护性,封装使开发者能更高效地进行网络应用开发。常见的封装层次包括套接字层(如Socket和ServerSocket类),以及更高层次的HTTP请求封装(如RestTemplate)。示例代码展示了如何使用RestTemplate简化HTTP请求的发送与处理,确保代码清晰易维护。
|
10月前
|
缓存 网络协议 Java
JAVA网络IO之NIO/BIO
本文介绍了Java网络编程的基础与历史演进,重点阐述了IO和Socket的概念。Java的IO分为设备和接口两部分,通过流、字节、字符等方式实现与外部的交互。
307 0