修改Hyper-V动态端口范围
1. 问题
1.1. 现象
因为使用Windows 10中的WSL 2功能需要启用Hyper-V,其结果是导致了IDEA在启动的时候提示端口被占用。
错误信息如下:
Internal error. Please refer to http://jb.gg/ide/critical-startup-errors
java.net.BindException: Address already in use: bind
at java.base/sun.nio.ch.Net.bind0(Native Method)
at java.base/sun.nio.ch.Net.bind(Net.java:461)
at java.base/sun.nio.ch.Net.bind(Net.java:453)
at java.base/sun.nio.ch.ServerSocketChannelImpl.bind(ServerSocketChannelImpl.java:227)
at io.netty.channel.socket.nio.NioServerSocketChannel.doBind(NioServerSocketChannel.java:132)
at io.netty.channel.AbstractChannel$AbstractUnsafe.bind(AbstractChannel.java:551)
at io.netty.channel.DefaultChannelPipeline$HeadContext.bind(DefaultChannelPipeline.java:1345)
at io.netty.channel.AbstractChannelHandlerContext.invokeBind(AbstractChannelHandlerContext.java:503)
at io.netty.channel.AbstractChannelHandlerContext.bind(AbstractChannelHandlerContext.java:488)
at io.netty.channel.DefaultChannelPipeline.bind(DefaultChannelPipeline.java:984)
at io.netty.channel.AbstractChannel.bind(AbstractChannel.java:247)
at io.netty.bootstrap.AbstractBootstrap$2.run(AbstractBootstrap.java:355)
at io.netty.util.concurrent.AbstractEventExecutor.safeExecute(AbstractEventExecutor.java:163)
at io.netty.util.concurrent.SingleThreadEventExecutor.runAllTasks(SingleThreadEventExecutor.java:416)
at io.netty.channel.nio.NioEventLoop.run(NioEventLoop.java:515)
at io.netty.util.concurrent.SingleThreadEventExecutor$5.run(SingleThreadEventExecutor.java:918)
at io.netty.util.internal.ThreadExecutorMap$2.run(ThreadExecutorMap.java:74)
at java.base/java.lang.Thread.run(Thread.java:834)
1.2. 查看原因
首先查看一下我们系统默认的端口占用范围;
netsh int ipv4 show dynamicport tcp
结果如下:
Microsoft Windows [版本 10.0.19041.388]
(c) 2020 Microsoft Corporation. 保留所有权利。C:\Users\Test>netsh int ipv4 show dynamicport tcp
协议 tcp 动态端口范围
启动端口 : 1024
端口数 : 13977
我们可以看到Windows系统默认的 TCP
动态端口范围为:1024~13977。当我们开启Hyper-V
后,系统默认会分配给一些保留端口供Hyper-V
使用:
netsh interface ipv4 show excludedportrange protocol=tcp
结果如下:
C:\Users\Test>netsh interface ipv4 show excludedportrange protocol=tcp
协议 tcp 端口排除范围
开始端口 结束端口
1026 1125
1226 1325
1326 1425
1426 1525
1526 1625
2180 2279
... ...
1.3. 问题结论
IDEA需要在端口6942~6991间找到一个可用端口并绑定(bind),往后面看可以看到端口这个端口范围在排除范围内
这样就导致了IDEA需要使用的端口是被占用,这样你当然就不能运行了。
2. 处理方案
2.1. 方法一:重置端口
使用管理员身份运行cmd,重置端口,然后重启
netsh winsock reset
这样你的tcp端口排除范围可能刚好不包含1099端口,这样你当然就可以用你的IDEA运行Tomcat应用了。但是你啥时候会出现就不得而知了。
2.2. 方法二:修改动态端口范围(推荐)
以下步骤需要使用管理员权限操作
2.2.1. 关闭Hyper-V
# 关闭Hyper-V
dism.exe /Online /Disable-Feature:Microsoft-Hyper-V
2.2.2. 修改动态端口范围
使用管理员身份运行cmd
# 设置动态端口TCP范围
netsh int ipv4 set dynamicport tcp start=51555 num=13980
# 设置动态端口UDP范围
netsh int ipv4 set dynamicport udp start=51555 num=13980
然后检查修改结果
# 查看动态端口范围
netsh int ipv4 show dynamicport tcp
输出结果如下:
协议 tcp 动态端口范围
启动端口 : 51555
端口数 : 13980
2.2.3. 开启Hyper-V
# 启用Hyper-V
dism.exe /Online /Enable-Feature:Microsoft-Hyper-V /All
输出结果如下:
部署映像服务和管理工具
版本: 10.0.18362.1映像版本: 10.0.18363.752
启用一个或多个功能
[100.0%]
操作成功完成。
重新启动 Windows 以完成该操作。
是否立即重新启动计算机? (Y/N)
3. 常用操作
# 禁用Hyper-V
dism.exe /Online /Disable-Feature:Microsoft-Hyper-V
# 启动Hyper-V
dism.exe /Online /Enable-Feature:Microsoft-Hyper-V /All
# 显示动态端口范围
netsh int ipv4 show dynamicport tcp
# 显示例外端口范围
netsh interface ipv4 show excludedportrange protocol=tcp
# 设置动态端口TCP范围
netsh int ipv4 set dynamicport tcp start=51555 num=13980
# 设置动态端口UDP范围
netsh int ipv4 set dynamicport udp start=51555 num=13980
# 添加例外端口
netsh int ipv4 add excludedportrange protocol=tcp startport=50051 numberofports=1
参考
IDEA启动报错-java.net.BindException: Address already in use: bind