cannotloadm3u8的解决 assign requested address怎么解决

用户名:yt
文章数:22
访问量:2647
注册日期:
阅读量:5863
阅读量:12276
阅读量:339641
阅读量:1042062
51CTO推荐博文
tomcat启动报错:Cannot&assign&requested&address:jvm_bind经查端口并未被占用,发现最后原因是hosts文件改过,加了一行&ip& localhost,导致启动时绑定不存在的ip失败了,把这行去掉就可以了。本文出自 “” 博客,请务必保留此出处
了这篇文章
类别:┆阅读(0)┆评论(0)tomcat java.net.BindException: Cannot assign requested address - 钟悍 - 博客园
posts - 113, comments - 12, trackbacks - 0, articles - 1
问题: tomcat 启动失败。log如下。
INFO: Server startup in 15913 msNOV 4, :03 AM org.apache.catalina.core.StandardServer awaitSEVERE: StandardServer.await: create[8005]:&java.net.BindException: Cannot assign requested address&&&&&&& at java.net.PlainSocketImpl.socketBind(Native Method)&&&&&&& at java.net.AbstractPlainSocketImpl.bind(AbstractPlainSocketImpl.java:353)&&&&&&& at java.net.ServerSocket.bind(ServerSocket.java:336)&&&&&&& at java.net.ServerSocket.&init&(ServerSocket.java:202)&&&&&&& at org.apache.catalina.core.StandardServer.await(StandardServer.java:406)&&&&&&& at org.apache.catalina.startup.Catalina.await(Catalina.java:676)&&&&&&& at org.apache.catalina.startup.Catalina.start(Catalina.java:628)&&&&&&& at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
之前其他人以为是端口冲突,使用netstat 查看,发现不存在8005端口在使用。之后还以为是tomcat连续2次启动。
去搜索日志,发现日志里面就只有一个8005,所以确定不是连续启动2次。
之后发现原来这个端口是绑定在localhost上的, 去查看hosts文件,果真发现没有127.0.0.1 localhost这一行。
加上这一行后启动成功。Socket connect error 99(Cannot assign requested address) - 推酷
Socket connect error 99(Cannot assign requested address)
转载请注明转自:
, 本文固定链接:
这是最近使用libcurl写http服务的压力测试的时候遇到的一个问题,其直接表象是客户端在发送http请求时失败,最终原因是客户端的TIME_WAIT状态的socket进程过多,导致端口被占满。下面看整个分析过程:
(1) 首先看产生错误的源码:
/* get it! */
res = curl_easy_perform(curl_handle);
long http_code = 0;
curl_easy_getinfo(curl_handle, CURLINFO_RESPONSE_CODE, &http_code);
/* cleanup curl stuff */
curl_easy_cleanup(curl_handle);
if (res != CURLE_OK || http_code != 200) {
cout && uri && &, res = & && res && &, http_code = & && http_code &&
return (res == CURLE_OK && http_code == 200);
错误日志如下:
http://10.237.92.30:8746/thumbnail/jpeg/l820/AppStore/b262b95f-95b8-4e0e-b4e0-edc3b76e3c81,
res = 7, http_code = 0
http://10.237.92.30:8746/thumbnail/jpeg/l820/AppStore/a4c3-40ff-af27-4efcd1a58e71,
res = 7, http_code = 0
http://10.237.92.30:8746/thumbnail/jpeg/l820/AppStore/abab08ff-75e1-40da-a113-86,
res = 7, http_code = 0
查看curllib的错误代码,如下,错误代码为CURLE_COULDNT_CONNECT
CURLE_OK = 0,
CURLE_UNSUPPORTED_PROTOCOL,
CURLE_FAILED_INIT,
CURLE_URL_MALFORMAT,
CURLE_NOT_BUILT_IN,
/* 4 - [was obsoleted in August 2007 for
7.17.0, reused in April 2011 for 7.21.5] */
CURLE_COULDNT_RESOLVE_PROXY,
CURLE_COULDNT_RESOLVE_HOST,
CURLE_COULDNT_CONNECT,
CURLE_FTP_WEIRD_SERVER_REPLY,
CURLE_REMOTE_ACCESS_DENIED,
/* 9 a service was denied by the server
(2) 分析curl_easy_perform返回错误的原因
最直接的办法采用gdb跟踪客户端的运行情况,发现客户端在connect的时候返回错误,在源文件curl-7.28.1/lib/connect.c的singleipconnect函数中,于是加入日志在connect之后打印errno,代码如下:
if(!isconnected && (conn-&socktype == SOCK_STREAM)) {
rc = connect(sockfd, &addr.sa_addr, addr.addrlen);
if(-1 == rc) {
error = SOCKERRNO;
printf(&connect failed with errno = %d&, errno);
conn-&connecttime = Curl_tvnow();
if(conn-&num_addr & 1)
Curl_expire(data, conn-&timeoutms_per_addr);
再次运行测试程序,得到如下输出:
connect failed with errno = 99 http://127.0.0.1:8902/thumbnail/jpeg/l820/AppStore/f8913ca1-
ae5f-4fcc-abc5-cbe9ada1a67d, ret_code: 0, res: 7
connect failed with errno = 99 http://127.0.0.1:8902/thumbnail/jpeg/l820/AppStore/-
057e-402d-b347-61c5a5136cd9, ret_code: 0, res: 7
connect failed with errno = 99 http://127.0.0.1:8902/thumbnail/jpeg/l820/AppStore/c19bad67-
6b7d-4dc6-a17a-f74ea525c32a, ret_code: 0, res: 7
connect failed with errno = 99 http://127.0.0.1:8902/thumbnail/jpeg/l820/AppStore/5d778568-
d873-46a7-9651-ad8ac3810bf4, ret_code: 0, res: 7
&可以看到errno = 99,在内核的include/asm-generic/errno.h文件中可以查看errno = 99的解释为” Cannot assign requested address”。
#define EAFNOSUPPORT
/* Address family not supported by protocol */
#define EADDRINUSE
/* Address already in use */
#define EADDRNOTAVAIL
/* Cannot assign requested address */
#define ENETDOWN
100 /* Network is down */
(3) errno = 99的原因;
至于connect系统调用为什么返回失败,就只能看系统调用的实现了。
a) connect系统调用
connect系统调用在net/socket.c中实现,Sys_connect系统调用的调用栈如下:
Sys_connect---&
sock-&ops-&connect
// inet_stream_connect
sk-&sk_prot-&connect
// tcp_v4_connect
tcp_v4_connect的作用主要是完成TCP连接三次握手中的第一个握手,即向服务端发送SYNC = 1和一个32位的序号的连接请求包。要发送SYNC请求包,按照TCP/IP协议,就必须有源IP地址和端口,源IP地址的选择和路由相关,需要查询路由表,在ip_route_connect中实现,源端口的选择在__inet_hash_connect中实现,而且如果找不到一个可用的端口,这个函数会返回-EADDRNOTAVAIL,因此基本上可以确定是这个函数返回错误导致connect失败;
b) __inet_hash_connect
这个函数的主要作用是选择一个可用的端口,其主要的实现步骤如下:
i. 调用inet_get_local_port_range(&low, &high);获取可用的端口链表;
调用read_seqbegin(&sysctl_local_ports.lock);得到顺序锁;
得到可用端口的low和high:
*low = sysctl_local_ports.range[0];
*high = sysctl_local_ports.range[1];
ii. 对于每一个端口,进行下面的步骤:
在inet_hashinfo *hinfo中查找这个端口inet_hashinfo用于保存已经使用的端口信息,每个使用的端口在这个hash表中有一个entry;
对端口做hash得到链表头(使用链表解决hash冲突)
遍历链表中的每一个entry:
a) 判断是否与这个要使用的端口相同,如果相同转到步骤b,如果不相同则遍历下一个entry
b) 找到这个端口,调用check_established(__inet_check_established)判断这个端口是否可以重用(TIME_WAIT状态下的端口并且net.ipv4.tcp_tw_recycle = 1是端口可以重用)
如果在链表中没有找到这个端口,表示端口没有被使用,调用inet_bind_bucket_create在hash表中插入一个entry;
iii. 如果到最后都没有找到一个可用的端口就返回EADDRNOTAVAIL;
从这个函数的实现可以看出,主要是由于可用的端口被占满了,所以找不到一个可用的端口,导致连接失败。运行netstat可以发现确实存在很多TIME_WAIT状态的socket,这些socket将可用端口占满了。
[root@test miuistorage-dev]# netstat -n | awk '/^tcp/ {++state[$NF]} END {for(key in state)
print key,&\t&,state[key]}'
ESTABLISHED
(4) 解决办法:
要解决端口被TIME_WAIT状态的socket占满的问题,可以有以下的解决办法:
a) 修改可用端口范围
查看当前的端口范围:
root@guojun8-desktop:/linux-2.6.34# sysctl net.ipv4.ip_local_port_range
net.ipv4.ip_local_port_range = 32768
修改端口范围:
root@guojun8-desktop:linux-2.6.34# sysctl net.ipv4.ip_local_port_range=&32768
net.ipv4.ip_local_port_range = 32768
这种办法可能不能解决根本问题,因为如果使用短连接,即使增加可用端口还是会被占满的。&
b) 设置net.ipv4.tcp_tw_recycle = 1
这个参数表示系统的TIME-WAIT sockets是否可以快速回收
root@guojun8-desktop:linux-2.6.34# sysctl net.ipv4.tcp_tw_recycle=1
net.ipv4.tcp_tw_recycle = 1
c) 设置net.ipv4.tcp_tw_recycle = 1
这个参数表示是否可以重用TIME_WAIT状态的端口;
root@guojun8-desktop:linux-2.6.34# [root@test thumbnail]# sysctl net.ipv4.tcp_tw_reuse=1
net.ipv4.tcp_tw_reuse = 1
(5) 更深入的探讨:sysctl做了什么
可以用strace跟踪一下sysctl的系统调用:
root@guojun8-desktop:linux-2.6.34# strace sysctl net.ipv4.tcp_tw_recycle=1
execve(&/sbin/sysctl&, [&sysctl&, &net.ipv4.tcp_tw_recycle=1&], [/* 20 vars */]) = 0
= 0x952f000
open(&/proc/sys/net/ipv4/tcp_tw_recycle&, O_WRONLY|O_CREAT|O_TRUNC, 0666) = 3
fstat64(3, {st_mode=S_IFREG|0644, st_size=0, ...}) = 0
mmap2(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0xb788e000
write(3, &1\n&, 2)
munmap(0xb788e000, 4096)
fstat64(1, {st_mode=S_IFCHR|0620, st_rdev=makedev(136, 8), ...}) = 0
mmap2(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0xb788e000
write(1, &net.ipv4.tcp_tw_recycle = 1\n&, 28net.ipv4.tcp_tw_recycle = 1
exit_group(0)
可以看到这个程序打开/proc/sys/net/ipv4/tcp_tw_recycle并向文件中写入1,但是这个设置时怎样其作用的呢?在内核中对/proc/sys目录下的文件的i_fop做了特殊的处理,在proc_sys_make_inode 中设置:inode-&i_fop = &proc_sys_file_operationsproc_sys_file_operations的定义如下:
static const struct file_operations proc_sys_file_operations = {
= proc_sys_read,
= proc_sys_write,
proc_sys_write中会修改对应的文件,并且修改内存中的内容,不同的文件有不同的proc_handler,如tcp_tw_recycle对应的处理函数是proc_dointvec,这个函数会修改下面的变量:
tcp_death_row.sysctl_tw_recycle
这个变量在内核中表示TIME_WIAT状态的socket是否可以被快速回收。&
已发表评论数()
请填写推刊名
描述不能大于100个字符!
权限设置: 公开
仅自己可见
正文不准确
标题不准确
排版有问题
主题不准确
没有分页内容
图片无法显示
视频无法显示
与原文不一致Cannot assign requested address 标签_51CTO技术博客
2072897 篇
搜索推荐文章
搜索原创文章
搜索近期文章
&&&&异常问题:java.io.IOException:Failedtobindtoserversocket:tcp://localhost:61616dueto:java.net.BindException:Cannotassignrequestedaddress:JVM_Bind原因:1.端口号被占用。2.IP地址找不到。解决办法:1.关闭InternetConnectionSharing(ICS)服务或其它。2.修改IP地址或插上..
&&页数 ( 1/172742 )
Copyright&
版权所有 未经许可 请勿转载

我要回帖

更多关于 assign 的文章

 

随机推荐