通道阻塞的几种场景

func main() {
    // 示例1。
    ch1 := make(chan int, 1)
    ch1 <- 1
    //ch1 <- 2 // 通道已满,因此这里会造成阻塞。


    // 示例2。
    ch2 := make(chan int, 1)
    //elem, ok := <-ch2 // 通道已空,因此这里会造成阻塞。
    //_, _ = elem, ok
    ch2 <- 1


    // 示例3。
    var ch3 chan int
    //ch3 <- 1 // 通道的值为nil,因此这里会造成永久的阻塞!
    //<-ch3 // 通道的值为nil,因此这里会造成永久的阻塞!
    _ = ch3

    //示例4
    ch4 := make(chan int)
    ch4 <= 3
	//因为 make(chan int) 初始化的是不带缓冲的通道。非缓冲通道只有在收发双方都就绪的情况下才能传递元素值,否则就阻塞。
}```

### Java NIO 的定义及其适用场景 #### 一、Java NIO 的定义 Java NIO(New Input/Output)是一种新的输入输出模型,旨在提供更高效的方式处理文件和网络通信。它由以下几个核心部分组成[^1]: - **Channels**: 数据传输的载体,替代了传统 IO 流的概念。常见的 Channel 实现有 `FileChannel` 和各种 Socket 通道。 - **Buffers**: 缓冲区用于临时存储数据,在读取或入过程中充当中介角色。NIO 提供多种 Buffer 类型,如 `ByteBuffer`, `CharBuffer` 等。 - **Selectors**: 多路复用器,允许单线程管理多个 Channel,非常适合高并发环境下的事件驱动编程。 相比传统的 BIO(Blocking IO),NIO 更加灵活且效率更高,因为它支持非阻塞模式以及多路复用技术[^2]。 #### 二、Java NIO 的特点 1. **非阻塞性** 在同步非阻塞 IO 模式下,程序发起一个 IO 请求后可以立即返回继续执行其他任务,而无需等待当前操作完成。然而需要注意的是,这种方式可能带来额外开销——如果频繁轮询检查状态变化,则可能导致 CPU 资源浪费[^3]。 2. **缓冲机制优化** 使用直接内存作为缓冲区能够减少数据拷贝次数从而提升性能;尽管如此,由于绕过了 JVM 堆空间管理机制,可能会给 GC 增添压力因此建议仅在必要场合才启用此特性[^4]。 #### 三、适用场景分析 基于以上特性描述可知,Java NIO 特别适合应用于如下几种典型情境之中: | 场景 | 描述 | |-------------------------|--------------------------------------------------------------------------------------| | **高并发服务器开发** | 利用 Selector 可以让少量线程甚至单一主线程同时监听大量连接请求,这对于构建高性能 Web Server 或者即时通讯服务非常有用。 | | **大数据量快速传输** | 如果涉及到海量数据连续不断地被发送或者接收(例如视频流媒体播放),那么通过自定义分配大容量 Direct ByteBuffer 就能显著改善吞吐率表现。 | | **随机访问本地磁盘文件** | FileChannel 支持定位到任意偏移处进行读动作而不必顺序遍历整个文档内容,这使得某些特定业务逻辑得以简化实现比如数据库引擎内部页表结构维护工作等等 。| 综上所述,当面临需要兼顾低延迟响应时间与大规模客户端接入能力挑战的时候,采用基于 Java NIO 架构设计往往成为首选方案之一。 ```java // 示例代码片段展示了如何利用 Selectors 来监控多个 Sockets 上发生的事件. import java.nio.channels.*; import java.util.Set; import java.io.IOException; public class MultiplexingExample { public static void main(String[] args) throws IOException { Selector selector = Selector.open(); // 注册第一个 socket channel... SocketChannel sc1 = SocketChannel.open(new InetSocketAddress("localhost", 8080)); sc1.configureBlocking(false); SelectionKey key1 = sc1.register(selector, SelectionKey.OP_READ); // ...第二个socket channel同样注册上去 while(true){ int readyChannels = selector.select(); if(readyChannels == 0) continue ; Set<SelectionKey> selectedKeys = selector.selectedKeys(); Iterator<SelectionKey> iter = selectedKeys.iterator(); while(iter.hasNext()){ SelectionKey key = iter.next(); if(key.isReadable()) { /* handle read */ } else if(key.isWritable()){ /* handle write*/} iter.remove(); } } } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

code-Study

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值