SOCKET网络编程快速上手(二)——细节问题(1)

三、细节问题一个也不能少

Socket编程说简单也简单,程序很容易就能跑起来,说麻烦还真是麻烦,程序动不动就出问题。记得刚开始写网络代码的时候,那真是令人抓狂的经历,问题一个套一个,一会服务器起不来了,一会数据接收异常了,到最后自己都对那些系统调用都不放心了,怎么会要考虑那么多东西?起初,我是一万个怀疑,是不是自己人品出问题了,怎么别人没遇到,全给自己赶上了。后来,拿着《UNIX网络编程》随便看看,那书怎么会这么了解我的?细节!细节!细节!那些问题都被别人明明写出来了,自己又SX了。没办法,细节不注意,有苦说不出啊。

不过也不能怪自己不爱学习啊,说实话那书实在太厚了,下面只记录一些自己遇到的、知道原因和解决方法的细节问题,还有很多后面慢慢学习吧!问题中使用的示例程序多少有点编造的意思,旨在说明问题。现实当中肯定是会发生的,概率也不能说低,我也没那么多闲时去统计具体数据。

  1. 端口复用

此处描述的内容可能和端口复用的真实概念不符合,但我习惯用这种描述方法,理解下面内容即可。

在调试网络程序的时候,TCP服务器经常起不来,总是在bind时出错。经验告诉我,此时换一个绑定端口往往就能起效,或者等个几分钟,服务器也能正常启动了。因为当时跑的都是些小的测试代码,换个端口还是很方便的。可这种问题要是放到一个系统或者一个服务器上,怎能接受?

当时,自己摸索了几天都没什么结果,最后还是师傅威武,告诉我还有个端口复用的东西。现在整理,已经没了当时的环境,自己构造了这一问题。

客户端程序为:

 1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <unistd.h>
4 #include <string.h>
5 #include <sys/types.h>
6 #include <sys/socket.h>
7 #include <netinet/in.h>
8 #include <netdb.h>
9
10 #define PORT 1234
11 #define MAXDATASIZE 1000
12
13 int main(int argc, char *argv[])
14 {
15 int sockfd, num;
16 char buf[MAXDATASIZE + 1] = {0};
17 struct sockaddr_in server;
18
19 if (argc != 2)
20 {
21 printf("Usage:%s <IP Address>\n", argv[0]);
22 exit(1);
23 }
24
25 if ((sockfd=socket(AF_INET, SOCK_STREAM, 0)) == -1)
26 {
27 printf("socket()error\n");
28 exit(1);
29 }
30 bzero(&server, sizeof(server));
31 server.sin_family = AF_INET;
32 server.sin_port = htons(PORT);
33 server.sin_addr.s_addr = inet_addr(argv[1]);
34 if (connect(sockfd, (struct sockaddr *)&server, \
35 sizeof(server)) == -1)
36 {
37 printf("connect()error\n");
38 exit(1);
39 }
40
41 while (1)
42 {
43 memset(buf, 0, sizeof(buf));
44 if ((num = recv(sockfd, buf, MAXDATASIZE,0)) == -1)
45 {
46 printf("recv() error\n");
47 exit(1);
48 }
49 buf[num - 1]='\0';
50 printf("Server Message: %s\n",buf);
51 }
52
53 close(sockfd);
54
55 return 0;
56 }
 1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <unistd.h>
4 #include <string.h>
5 #include <sys/types.h>
6 #include <sys/socket.h>
7 #include <netinet/in.h>
8 #include <netdb.h>
9
10 #define PORT 1234
11 #define MAXDATASIZE 1000
12
13 int main(int argc, char *argv[])
14 {
15 int sockfd, num;
16 char buf[MAXDATASIZE + 1] = {0};
17 struct sockaddr_in server;
18
19 if (argc != 2)
20 {
21 printf("Usage:%s <IP Address>\n", argv[0]);
22 exit(1);
23 }
24
25 if ((sockfd=socket(AF_INET, SOCK_STREAM, 0)) == -1)
26 {
27 printf("socket()error\n");
28 exit(1);
29 }
30 bzero(&server, sizeof(server));
31 server.sin_family = AF_INET;
32 server.sin_port = htons(PORT);
33 server.sin_addr.s_addr = inet_addr(argv[1]);
34 if (connect(sockfd, (struct sockaddr *)&server, \
35 sizeof(server)) == -1)
36 {
37 printf("connect()error\n");
38 exit(1);
39 }
40
41 while (1)
42 {
43 memset(buf, 0, sizeof(buf));
44 if ((num = recv(sockfd, buf, MAXDATASIZE,0)) == -1)
45 {
46 printf("recv() error\n");
47 exit(1);
48 }
49 buf[num - 1]='\0';
50 printf("Server Message: %s\n",buf);
51 }
52
53 close(sockfd);
54
55 return 0;
56 }

服务器程序为:

 1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <string.h>
4 #include <unistd.h>
5 #include <sys/types.h>
6 #include <sys/socket.h>
7 #include <netinet/in.h>
8 #include <arpa/inet.h>
9 #include <signal.h>
10
11 #define PORT 1234
12 #define BACKLOG 5
13 #define MAXDATASIZE 1000
14
15 int main()
16 {
17 int listenfd, connectfd;
18 struct sockaddr_in server;
19 struct sockaddr_in client;
20 socklen_t addrlen;
21 char szbuf[MAXDATASIZE] = {0};
22 int iCount = 0;
23
24 if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
25 {
26 perror("Creating socket failed.");
27 exit(1);
28 }
29
30 int opt = SO_REUSEADDR;
31 // setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
32
33 bzero(&server, sizeof(server));
34 server.sin_family = AF_INET;
35 server.sin_port = htons(PORT);
36 server.sin_addr.s_addr = htonl(INADDR_ANY);
37 if (bind(listenfd, (struct sockaddr *)&server, sizeof(server)) == -1)
38 {
39 perror("Bind()error.");
40 exit(1);
41 }
42 if (listen(listenfd, BACKLOG) == -1)
43 {
44 perror("listen()error\n");
45 exit(1);
46 }
47
48 addrlen = sizeof(client);
49 if ((connectfd = accept(listenfd, (struct sockaddr*)&client, &addrlen)) == -1)
50 {
51 perror("accept()error\n");
52 exit(1);
53 }
54 printf("You got a connection from cient's ip is %s, prot is %d\n", inet_ntoa(client.sin_addr), htons(client.sin_port));
55
56 memset(szbuf, 'a', sizeof(szbuf));
57 while (iCount < 100)
58 {
59 send(connectfd, szbuf, sizeof(szbuf), 0);
60 iCount++;
61 }
62
63 printf("send over!\n");
64 sleep(10);
65
66 close(connectfd);
67 close(listenfd);
68
69 return 0;
70 }
 1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <string.h>
4 #include <unistd.h>
5 #include <sys/types.h>
6 #include <sys/socket.h>
7 #include <netinet/in.h>
8 #include <arpa/inet.h>
9 #include <signal.h>
10
11 #define PORT 1234
12 #define BACKLOG 5
13 #define MAXDATASIZE 1000
14
15 int main()
16 {
17 int listenfd, connectfd;
18 struct sockaddr_in server;
19 struct sockaddr_in client;
20 socklen_t addrlen;
21 char szbuf[MAXDATASIZE] = {0};
22 int iCount = 0;
23
24 if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
25 {
26 perror("Creating socket failed.");
27 exit(1);
28 }
29
30 int opt = SO_REUSEADDR;
31 // setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
32
33 bzero(&server, sizeof(server));
34 server.sin_family = AF_INET;
35 server.sin_port = htons(PORT);
36 server.sin_addr.s_addr = htonl(INADDR_ANY);
37 if (bind(listenfd, (struct sockaddr *)&server, sizeof(server)) == -1)
38 {
39 perror("Bind()error.");
40 exit(1);
41 }
42 if (listen(listenfd, BACKLOG) == -1)
43 {
44 perror("listen()error\n");
45 exit(1);
46 }
47
48 addrlen = sizeof(client);
49 if ((connectfd = accept(listenfd, (struct sockaddr*)&client, &addrlen)) == -1)
50 {
51 perror("accept()error\n");
52 exit(1);
53 }
54 printf("You got a connection from cient's ip is %s, prot is %d\n", inet_ntoa(client.sin_addr), htons(client.sin_port));
55
56 memset(szbuf, 'a', sizeof(szbuf));
57 while (iCount < 100)
58 {
59 send(connectfd, szbuf, sizeof(szbuf), 0);
60 iCount++;
61 }
62
63 printf("send over!\n");
64 sleep(10);
65
66 close(connectfd);
67 close(listenfd);
68
69 return 0;
70 }

客户端和服务器在两台电脑上运行,电脑通过交换机相连,当服务器将数据发送完成之后,sleep一段时间再关闭链接。根据程序,sleep的时候,手动断电关闭交换机,sleep完成,服务器关闭链路,进程退出。然后再重启服务器,服务器就起不来了,打印:

使用netstat  –an查看端口使用情况:

1234端口处于FIN_WAIT2状态,端口处于占用状态,所以bind返回失败了。解决办法:

代码中增加下面语句即可

int opt = SO_REUSEADDR;

setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));

上述语句在TCP服务器代码中是必不可少的。

 
 

SOCKET网络编程细节问题1的更多相关文章

  1. SOCKET网络编程细节问题(4)

    SOCKET网络编程快速上手(二)——细节问题(4) 5.慢系统调用及EINTR 还记得前面readn和writen函数么?里面有个EINTR,现在就来谈谈这个,这个很重要. Linux世界有个叫信号 ...

  2. SOCKET网络编程细节问题3

    SOCKET网络编程快速上手(二)——细节问题(3) 3.SIGPIPE问题 人怕牺牲,我们写的程序也一样,人有死不瞑目,程序又何尝不是?程序跑着跑着,突然就崩掉了.好一点的牺牲前告诉你些打印,差点的 ...

  3. SOCKET网络编程细节问题(2)

    SOCKET网络编程快速上手(二)——细节问题(2) 2.TCP数据包接收问题 对初学者来说,很多都会认为:客户端与服务器最终的打印数据接收或者发送条数都该是一致的,1000条发送打印,1000条接收 ...

  4. SOCKET网络编程5

    SOCKET网络编程快速上手(二)——细节问题(5)(完结篇) 6.Connect的使用方式 前面提到,connect发生EINTR错误时,是不能重新启动的.那怎么办呢,是关闭套接字还是直接退出进程呢 ...

  5. Py西游攻关之Socket网络编程

    新闻 管理   Py西游攻关之Socket网络编程   知识预览 计算机网络 回到顶部 网络通信要素: A:IP地址   (1) 用来标识网络上一台独立的主机 (2) IP地址 = 网络地址 + 主机 ...

  6. Python面向对象进阶和socket网络编程-day08

    写在前面 上课第八天,打卡: 为什么坚持?想一想当初: 一.面向对象进阶 - 1.反射补充 - 通过字符串去操作一个对象的属性,称之为反射: - 示例1: class Chinese: def __i ...

  7. Python面向对象进阶和socket网络编程

    写在前面 为什么坚持?想一想当初: 一.面向对象进阶 - 1.反射补充 - 通过字符串去操作一个对象的属性,称之为反射: - 示例1: class Chinese: def __init__(self ...

  8. Linux Socket 网络编程

    Linux下的网络编程指的是socket套接字编程,入门比较简单.在学校里学过一些皮毛,平时就是自学玩,没有见识过真正的socket编程大程序,比较遗憾.总感觉每次看的时候都有收获,但是每次看完了之后 ...

  9. Python Socket 网络编程

    Socket 是进程间通信的一种方式,它与其他进程间通信的一个主要不同是:它能实现不同主机间的进程间通信,我们网络上各种各样的服务大多都是基于 Socket 来完成通信的,例如我们每天浏览网页.QQ ...

随机推荐

  1. python_小爬虫

    import urllib.request as request import urllib.parse as parse import string print(""" ...

  2. oracle_修改Oracle数据库字符集 AL32UTF8;

    修改数据库字符集 以支持维文等  utf8 停掉库 进入装载模式 ALTER SYSTEM ENABLE RESTRICTED SESSION; ALTER SYSTEM SET JOB_QUEUE_ ...

  3. TRIZ系列-创新原理-28-替代机械系统原理

    替代机械系统原理的详细描写叙述例如以下:1)用光.声.热.嗅觉系统替代机械系统:2)用电.磁或电磁场来与物体交互作用:3)用移动场替代精巧场,用随时间变化的场替代固定场,用结构化的场替代随机场:4)使 ...

  4. JS中5秒中跳转到其他页面

    原文:JS中5秒中跳转到其他页面 <head> <meta http-equiv="Content-Type" content="text/html; ...

  5. TreeView的绑定

    近期遇到了TreeView的数据库绑定问题,确实是弄了我好几天,特别是多级节点的分步绑定,最開始不分步,发现所有载入页面都卡爆了,真心让人头疼.所以放出来,给须要的朋友看看,以免大家走冤枉路. 1.仅 ...

  6. VBOX安装Centos设置分辨率为1366x768[已解决]

    最近想了解下GTK+,但是对于直接在系统上搭建环境有点心里阴影,怕又把桌面玩挂,所以打算在虚拟机中先试试 本来打算使用Fedora的,怕gnome太吃资源所以下了个xfce的,不过貌似有BUG,无法安 ...

  7. js关闭当前页面不弹出提示的方法

    js关闭当前页面不弹出提示的方法 js关闭当前页面不弹出提示的方法 "window.opener=null;window.open('','_self','');window.close() ...

  8. JavaScript语言基础知识8

    这篇文章是对前面学习的知识进行总结: 1.JavaScript支持多种数据类型,如数值类型.字符串类型.布尔类型等. 2.在JavaScript中,字符串是用引號括起来的字符系列,转义字符能够用来表示 ...

  9. 异步陷阱之IO

    异步陷阱之IO篇 很多教程和资料都强调流畅的用户体验需要异步来辅助,核心思想就是保证用户前端的交互永远有最高的优先级,让一切费时的逻辑通通放到后台,等到诸事完备,通知一下前端给个提示或者继续下一步.随 ...

  10. Oracle wrap 和 unwrap( 加密与解密) 说明

    一. Wrap 说明 官网的说明如下: A PL/SQL Source Text Wrapping http://download.oracle.com/docs/cd/E11882_01/appde ...