Java的accept()

在Java中,accept() 方法通常与网络编程中的套接字(Sockets)相关联,尤其是在服务器端的 ServerSocket 类中。accept() 方法用于等待并接受到来的连接请求。当一个客户端尝试与服务器建立TCP连接时,如果服务器端的 ServerSocket 正在监听指定的端口,那么 accept() 方法就会建立连接,并返回一个新的 Socket 对象来与客户端进行通信。

下面是 accept() 方法的详细解释:

方法签名

在Java的 ServerSocket 类中,accept() 方法的签名如下:

public Socket accept() throws IOException

这个方法在成功建立连接时返回一个新的 Socket 对象,如果在等待连接时出现任何I/O错误,将抛出 IOException

工作原理

  1. 监听端口
    在调用 accept() 之前,必须先创建一个 ServerSocket 实例,并指定一个端口号用于监听。这个端口号就是客户端用来与服务器建立连接的端口。

  2. 等待连接
    当调用 accept() 方法时,程序会阻塞(暂停执行),直到一个连接请求到达。这意味着如果没有客户端尝试连接,程序就会在这里等待。

  3. 连接建立
    一旦有客户端尝试连接到服务器的指定端口,accept() 方法就会接受这个连接请求,并建立一个新的TCP连接。

  4. 返回新的Socket
    在连接建立之后,accept() 方法会创建并返回一个 Socket 实例,该实例代表了与客户端的连接。通过这个新的 Socket 对象,服务器可以与客户端进行通信,如读取输入和写入输出。

示例代码

下面是一个简单的服务器程序示例,展示了如何使用 accept() 方法:

import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;

public class SimpleServer {
    public static void main(String[] args) {
        int port = 1234; // 服务器监听的端口号
        ServerSocket serverSocket = null;

        try {
            serverSocket = new ServerSocket(port);
            System.out.println("Server is listening on port " + port);

            while (true) {
                // 等待客户端连接
                Socket clientSocket = serverSocket.accept();
                System.out.println("A client connected.");

                // 在这里可以开启一个新的线程来处理与客户端的交互,以便可以同时处理多个客户端
                // ...

                // 这里仅仅是关闭连接作为示例
                clientSocket.close();
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (serverSocket != null) {
                try {
                    serverSocket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

在上述代码中,服务器监听端口 1234,循环调用 accept() 方法等待客户端的连接。每当有客户端连接时,就会打印一条消息,并立即关闭与客户端的连接。在实际的应用中,服务器会为每个连接创建一个新的线程或者提交给线程池处理,以实现并发处理多个客户端请求。

 

以下是几个不同上下文中 accept() 方法的示例,包括网络编程的 ServerSocket 使用,以及函数式接口 Consumer 和 BiConsumer 的使用。

例子 1:使用 ServerSocket 接受客户端连接

这个例子扩展了之前的服务器示例,通过多线程允许服务器同时处理多个客户端连接。

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

public class MultiThreadedServer {
    public static void main(String[] args) {
        int port = 1234;
        ServerSocket serverSocket = null;

        try {
            serverSocket = new ServerSocket(port);
            System.out.println("Server is listening on port " + port);

            while (true) {
                Socket clientSocket = serverSocket.accept();
                System.out.println("A client connected.");

                // 为每个客户端创建一个新的线程
                new Thread(new ClientHandler(clientSocket)).start();
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (serverSocket != null) {
                try {
                    serverSocket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    private static class ClientHandler implements Runnable {
        private final Socket clientSocket;

        public ClientHandler(Socket socket) {
            this.clientSocket = socket;
        }

        public void run() {
            try (BufferedReader in = new BufferedReader(
                 new InputStreamReader(clientSocket.getInputStream()));
                 PrintWriter out = new PrintWriter(
                 clientSocket.getOutputStream(), true)) {

                String inputLine;
                while ((inputLine = in.readLine()) != null) {
                    System.out.println("Received: " + inputLine);
                    out.println(inputLine);
                }
            } catch (IOException e) {
                e.printStackTrace();
            } finally {
                try {
                    clientSocket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

在这个例子中,每当 accept() 方法接受一个新的客户端连接,它就会启动一个新的线程(ClientHandler 类的一个实例),允许服务器并行处理多个客户端。

例子 2:使用 Consumer 接口的 accept() 方法

Consumer 是一个函数式接口,它有一个 accept() 方法用来执行某项操作,接受一个输入参数,返回无结果。

import java.util.function.Consumer;

public class ConsumerExample {
    public static void main(String[] args) {
        Consumer<String> printConsumer = System.out::println;
        printConsumer.accept("Hello, World!");  // 输出 "Hello, World!"
    }
}

在这个例子中,我们创建了一个 Consumer 来打印传递给它的字符串。

例子 3:使用 BiConsumer 接口的 accept() 方法

BiConsumer 同样是一个函数式接口,与 Consumer 类似,但它接受两个参数而不是一个。

import java.util.function.BiConsumer;

public class BiConsumerExample {
    public static void main(String[] args) {
        BiConsumer<Integer, Integer> sumConsumer = (a, b) -> System.out.println("Sum is: " + (a + b));
        sumConsumer.accept(10, 20);  // 输出 "Sum is: 30"
    }
}

在这个例子中,BiConsumer 接受两个整数参数并打印它们的和。

这些例子展示了 accept() 方法在不同上下文中的使用方式。在网络编程中,它用于接受连接;在函数式编程中,它用于接受参数并执行操作。

  • 6
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
"br" 是一种较新的 HTTP 压缩算法,也称为 Brotli 压缩算法。在 Java 中,可以使用第三方库 okhttp 来解析 "br" 编码的响应体。 以下是一个示例代码: ```java import okhttp3.OkHttpClient; import okhttp3.Request; import okhttp3.Response; import okhttp3.ResponseBody; import okhttp3.Headers; import okhttp3.MediaType; import okhttp3.internal.http.HttpHeaders; import okhttp3.internal.http.RealResponseBody; import okhttp3.internal.http1.Http1Codec; import java.io.IOException; import java.util.ArrayList; public class BrotliResponseDecoder { public static void main(String[] args) throws IOException { OkHttpClient client = new OkHttpClient(); Request request = new Request.Builder() .url("https://example.com") .addHeader("Accept-Encoding", "br") .build(); Response response = client.newCall(request).execute(); ResponseBody responseBody = response.body(); Headers headers = response.headers(); MediaType contentType = responseBody.contentType(); if (HttpHeaders.hasBody(response)) { ArrayList<ResponseBody> intermediates = new ArrayList<>(); ResponseBody intermediateBody = new RealResponseBody(contentType.toString(), -1L, intermediates); Http1Codec http1Codec = new Http1Codec(null, null, null, null, -1L, false); intermediates.add(responseBody); responseBody = http1Codec.openResponseBody(intermediateBody); } String decodedResponse = responseBody.string(); System.out.println(decodedResponse); } } ``` 在这个示例代码中,我们通过 OkHttpClient 发送了一个带有 "Accept-Encoding: br" 请求头部的 GET 请求。然后,我们可以通过调用 Response 对象的 body() 方法获取响应体 ResponseBody 对象,并通过 headers() 方法获取响应头 Headers 对象。接着,我们判断响应体是否存在,如果存在则调用 openResponseBody() 方法对响应体进行解压缩,最后通过 string() 方法获取解压后的响应体字符串。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值