socket编程遇到一个颇为诡异的问题

HW_Coder0501 2016-09-29 10:46:57
先贴上程序

/* tcp_server.c */

#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>

#define portnumber 3333

int main(int argc, char *argv[])
{
int sockfd,new_fd;
struct sockaddr_in server_addr;
struct sockaddr_in client_addr;
int sin_size;
int nbytes;
char buffer[1024];


/* 服务器端开始建立sockfd描述符 */
if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1) // AF_INET:IPV4
{
fprintf(stderr,"Socket error:%s\n\a",strerror(errno));
exit(1);
}

/* 服务器端填充 sockaddr结构 */
bzero(&server_addr,sizeof(struct sockaddr_in)); // 初始化,置0
server_addr.sin_family=AF_INET; // Internet
server_addr.sin_port=htons(portnumber); // (将本机器上的short数据转化为网络上的short数据)端口号
server_addr.sin_addr.s_addr=htonl(INADDR_ANY); // (将本机器上的long数据转化为网络上的long数据)和任何主机通信
//server_addr.sin_addr.s_addr=inet_addr("192.168.1.1"); //用于绑定到一个固定IP


/* 捆绑sockfd描述符到IP地址 */
if(bind(sockfd,(struct sockaddr *)(&server_addr),sizeof(struct sockaddr))==-1)
{
fprintf(stderr,"Bind error:%s\n\a",strerror(errno));
exit(1);
}

/* 设置允许连接的最大客户端数 */
if(listen(sockfd,5)==-1)
{
fprintf(stderr,"Listen error:%s\n\a",strerror(errno));
exit(1);
}

while(1)
{
/* 服务器阻塞,直到客户程序建立连接 */
sin_size=sizeof(struct sockaddr_in);
if((new_fd=accept(sockfd,(struct sockaddr *)(&client_addr),&sin_size))==-1)
{ //addrlen:输入参数,配合addr一起使用,指针
fprintf(stderr,"Accept error:%s\n\a",strerror(errno));
exit(1);
}
fprintf(stderr,"Server get connection from %s\n",inet_ntoa(client_addr.sin_addr)); // 将网络地址转换成.字符串

if((nbytes=read(new_fd,buffer,1024))==-1) //成功返回读取的字节数;
{
fprintf(stderr,"Read Error:%s\n",strerror(errno));
exit(1);
}
buffer[nbytes]='\0'; //nbytes是字节数,故可作为最后一个元素
printf("Server received %s\n",buffer);

/* 这个通讯已经结束 */
close(new_fd);
/* 循环下一个 */
}

/* 结束通讯 */
close(sockfd);
exit(0);
}

以上为socket编程中的服务器端的函数。

客户端程序:
/* tcp_client.c */

#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>

#define portnumber 3333

int main(int argc, char *argv[])
{
int sockfd;
char buffer[1024];
struct sockaddr_in server_addr; //用于记录网络地址
struct hostent *host;
char strhost[16]; //添加代码
/* 使用hostname查询host 名字 */
if(argc!=1) //改动代码:原为 if(argc!=2)
{
fprintf(stderr,"Usage:%s hostname \a\n",argv[0]);
exit(1);
}
scanf("%s",strhost); //添加代码
if((host=gethostbyname(strhost))==NULL) //改动部分:原代码为if((host=gethostbyname(argv[1]))==NULL)
{
fprintf(stderr,"Gethostname error\n");
exit(1);
}

/* 客户程序开始建立 sockfd描述符 */
if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1) // AF_INET:指定使用何种的地址类型
{
fprintf(stderr,"Socket Error:%s\a\n",strerror(errno));
exit(1);
}

/* 客户程序填充服务端的资料 */
bzero(&server_addr,sizeof(server_addr)); // 初始化,置0
server_addr.sin_family=AF_INET; // IPV4
server_addr.sin_port=htons(portnumber); // 端口号
server_addr.sin_addr=*((struct in_addr *)host->h_addr); // IP地址

/* 客户程序发起连接请求 */
if(connect(sockfd,(struct sockaddr *)(&server_addr),sizeof(struct sockaddr))==-1)
{
fprintf(stderr,"Connect Error:%s\a\n",strerror(errno));
exit(1);
}

/* 连接成功了 */
printf("Please input char:\n");

/* 发送数据 */
fgets(buffer,1024,stdin);
write(sockfd,buffer,strlen(buffer));
/* 结束通讯 */
close(sockfd);
printf("确认连接成功\n"); //自己添加的用来测试的代码
exit(0);
}


这是一个客户端发送消息,服务器端接收的程序,原程序可以正常实现设想的功能,但我改动其中的几处之后,客户端和服务器端虽然可以连接成功,但还没来得及发送消息,程序就退出了执行,以下是在Ubuntu下运行的结果:
/tcp_client2
127.0.0.1 //红色为终端输出
正在建立套接口...
套接字创建成功,正在连接服务器...
please input char:
确认连接成功


输出以上几行信息后,客户端程序直接退出运行,但原来的程序会一直阻塞在那里等待输入消息。。。我花了一晚上的时间也没看出自己改的代码哪里出了问题,请诸位大神帮忙看一,在下先谢过了!
...全文
502 12 打赏 收藏 转发到动态 举报
写回复
用AI写文章
12 条回复
切换为时间正序
请发表友善的回复…
发表回复
HW_Coder0501 2016-10-01
  • 打赏
  • 举报
回复
引用 8 楼 apple_v1 的回复:
[quote=引用 7 楼 hhhlizhao 的回复:] [quote=引用 5 楼 apple_v1 的回复:] 在scanf和fgets之间加个getchar,像这样: scanf("%s",strhostname); ... getchar(); ... fgets(buffer,1024,stdin); 因为scanf并没有处理换行符,而是把它退回到缓冲区中,此时fgets读到换行符并立即返回一个空的缓冲区,write只是把一个空的缓冲区写入到套接字中,所以服务器端显示的只是一个空的缓冲区。这里用getchar清掉缓冲区中的换行符以防止fgets返回一个空的缓冲区。这类问题一般很难发现,因为标准库的缓冲区对用户是不可见的。
又细细看了一下大神的解释,发现还有一点不明白的地方。fgets从stdin读入换行符之后应该是把它存到buffer缓冲区中去了吧,那为什么write只是把一个空的缓冲区写到了套接字,不应该有一个换行符输出么?[/quote] 你的理解是对的,之前我以为fgets不会存储换行符,是我理解错了,抱歉。[/quote] 没关系,我也不懂,这样的话是不是fgets存储了换行符,那缓冲区是不是会输出一个换行符?
apple_v1 2016-10-01
  • 打赏
  • 举报
回复
引用 11 楼 hhhlizhao 的回复:
[quote=引用 10 楼 apple_v1 的回复:] [quote=引用 9 楼 hhhlizhao 的回复:] [quote=引用 8 楼 apple_v1 的回复:] [quote=引用 7 楼 hhhlizhao 的回复:] [quote=引用 5 楼 apple_v1 的回复:] 在scanf和fgets之间加个getchar,像这样: scanf("%s",strhostname); ... getchar(); ... fgets(buffer,1024,stdin); 因为scanf并没有处理换行符,而是把它退回到缓冲区中,此时fgets读到换行符并立即返回一个空的缓冲区,write只是把一个空的缓冲区写入到套接字中,所以服务器端显示的只是一个空的缓冲区。这里用getchar清掉缓冲区中的换行符以防止fgets返回一个空的缓冲区。这类问题一般很难发现,因为标准库的缓冲区对用户是不可见的。
又细细看了一下大神的解释,发现还有一点不明白的地方。fgets从stdin读入换行符之后应该是把它存到buffer缓冲区中去了吧,那为什么write只是把一个空的缓冲区写到了套接字,不应该有一个换行符输出么?[/quote] 你的理解是对的,之前我以为fgets不会存储换行符,是我理解错了,抱歉。[/quote] 没关系,我也不懂,这样的话是不是fgets存储了换行符,那缓冲区是不是会输出一个换行符?[/quote] 对的[/quote] 那为什么我运行的时候什么都没有输出啊?[/quote] 服务器端是有一个换行符输出的。
HW_Coder0501 2016-10-01
  • 打赏
  • 举报
回复
引用 10 楼 apple_v1 的回复:
[quote=引用 9 楼 hhhlizhao 的回复:] [quote=引用 8 楼 apple_v1 的回复:] [quote=引用 7 楼 hhhlizhao 的回复:] [quote=引用 5 楼 apple_v1 的回复:] 在scanf和fgets之间加个getchar,像这样: scanf("%s",strhostname); ... getchar(); ... fgets(buffer,1024,stdin); 因为scanf并没有处理换行符,而是把它退回到缓冲区中,此时fgets读到换行符并立即返回一个空的缓冲区,write只是把一个空的缓冲区写入到套接字中,所以服务器端显示的只是一个空的缓冲区。这里用getchar清掉缓冲区中的换行符以防止fgets返回一个空的缓冲区。这类问题一般很难发现,因为标准库的缓冲区对用户是不可见的。
又细细看了一下大神的解释,发现还有一点不明白的地方。fgets从stdin读入换行符之后应该是把它存到buffer缓冲区中去了吧,那为什么write只是把一个空的缓冲区写到了套接字,不应该有一个换行符输出么?[/quote] 你的理解是对的,之前我以为fgets不会存储换行符,是我理解错了,抱歉。[/quote] 没关系,我也不懂,这样的话是不是fgets存储了换行符,那缓冲区是不是会输出一个换行符?[/quote] 对的[/quote] 那为什么我运行的时候什么都没有输出啊?
apple_v1 2016-10-01
  • 打赏
  • 举报
回复
引用 9 楼 hhhlizhao 的回复:
[quote=引用 8 楼 apple_v1 的回复:] [quote=引用 7 楼 hhhlizhao 的回复:] [quote=引用 5 楼 apple_v1 的回复:] 在scanf和fgets之间加个getchar,像这样: scanf("%s",strhostname); ... getchar(); ... fgets(buffer,1024,stdin); 因为scanf并没有处理换行符,而是把它退回到缓冲区中,此时fgets读到换行符并立即返回一个空的缓冲区,write只是把一个空的缓冲区写入到套接字中,所以服务器端显示的只是一个空的缓冲区。这里用getchar清掉缓冲区中的换行符以防止fgets返回一个空的缓冲区。这类问题一般很难发现,因为标准库的缓冲区对用户是不可见的。
又细细看了一下大神的解释,发现还有一点不明白的地方。fgets从stdin读入换行符之后应该是把它存到buffer缓冲区中去了吧,那为什么write只是把一个空的缓冲区写到了套接字,不应该有一个换行符输出么?[/quote] 你的理解是对的,之前我以为fgets不会存储换行符,是我理解错了,抱歉。[/quote] 没关系,我也不懂,这样的话是不是fgets存储了换行符,那缓冲区是不是会输出一个换行符?[/quote] 对的
renwotao2009 2016-09-30
  • 打赏
  • 举报
回复
客户端发送数据这块代码, /* 发送数据 */ fgets(buffer,1024,stdin); write(sockfd,buffer,strlen(buffer)); 检查下是否出错了??? 这里fgets时,你输入信息没 如果fgets成功,你的write写入成功没,你都要进行判断??write返回-1出错,你看下错误,如果返回值n小于写入数据的长度,你要循环写入剩余数据才行
振翅高飞 2016-09-30
  • 打赏
  • 举报
回复
客户端程序的这句exit(0); 代表直接正常退出。 write(sockfd,buffer,strlen(buffer)); /* 结束通讯 */ close(sockfd); printf("确认连接成功\n"); //自己添加的用来测试的代码 exit(0); }
HW_Coder0501 2016-09-30
  • 打赏
  • 举报
回复
引用 1 楼 xinzha 的回复:
char strhost[16]; 这个够长吗?粗略看了下你的所有改动,目前我只看到这处有点可疑。
我把char strhost[16];改成了char strhost[1024];还是不可以。。问题应该不是出在这。
apple_v1 2016-09-30
  • 打赏
  • 举报
回复
引用 7 楼 hhhlizhao 的回复:
[quote=引用 5 楼 apple_v1 的回复:] 在scanf和fgets之间加个getchar,像这样: scanf("%s",strhostname); ... getchar(); ... fgets(buffer,1024,stdin); 因为scanf并没有处理换行符,而是把它退回到缓冲区中,此时fgets读到换行符并立即返回一个空的缓冲区,write只是把一个空的缓冲区写入到套接字中,所以服务器端显示的只是一个空的缓冲区。这里用getchar清掉缓冲区中的换行符以防止fgets返回一个空的缓冲区。这类问题一般很难发现,因为标准库的缓冲区对用户是不可见的。
又细细看了一下大神的解释,发现还有一点不明白的地方。fgets从stdin读入换行符之后应该是把它存到buffer缓冲区中去了吧,那为什么write只是把一个空的缓冲区写到了套接字,不应该有一个换行符输出么?[/quote] 你的理解是对的,之前我以为fgets不会存储换行符,是我理解错了,抱歉。
HW_Coder0501 2016-09-30
  • 打赏
  • 举报
回复
引用 5 楼 apple_v1 的回复:
在scanf和fgets之间加个getchar,像这样: scanf("%s",strhostname); ... getchar(); ... fgets(buffer,1024,stdin); 因为scanf并没有处理换行符,而是把它退回到缓冲区中,此时fgets读到换行符并立即返回一个空的缓冲区,write只是把一个空的缓冲区写入到套接字中,所以服务器端显示的只是一个空的缓冲区。这里用getchar清掉缓冲区中的换行符以防止fgets返回一个空的缓冲区。这类问题一般很难发现,因为标准库的缓冲区对用户是不可见的。
又细细看了一下大神的解释,发现还有一点不明白的地方。fgets从stdin读入换行符之后应该是把它存到buffer缓冲区中去了吧,那为什么write只是把一个空的缓冲区写到了套接字,不应该有一个换行符输出么?
HW_Coder0501 2016-09-30
  • 打赏
  • 举报
回复
引用 5 楼 apple_v1 的回复:
在scanf和fgets之间加个getchar,像这样: scanf("%s",strhostname); ... getchar(); ... fgets(buffer,1024,stdin); 因为scanf并没有处理换行符,而是把它退回到缓冲区中,此时fgets读到换行符并立即返回一个空的缓冲区,write只是把一个空的缓冲区写入到套接字中,所以服务器端显示的只是一个空的缓冲区。这里用getchar清掉缓冲区中的换行符以防止fgets返回一个空的缓冲区。这类问题一般很难发现,因为标准库的缓冲区对用户是不可见的。
果然是大神,确实是这样的。。加入getchar();之后运行结果和原来一样。
apple_v1 2016-09-30
  • 打赏
  • 举报
回复
在scanf和fgets之间加个getchar,像这样: scanf("%s",strhostname); ... getchar(); ... fgets(buffer,1024,stdin); 因为scanf并没有处理换行符,而是把它退回到缓冲区中,此时fgets读到换行符并立即返回一个空的缓冲区,write只是把一个空的缓冲区写入到套接字中,所以服务器端显示的只是一个空的缓冲区。这里用getchar清掉缓冲区中的换行符以防止fgets返回一个空的缓冲区。这类问题一般很难发现,因为标准库的缓冲区对用户是不可见的。
猪头三小队长 2016-09-29
  • 打赏
  • 举报
回复
char strhost[16]; 这个够长吗?粗略看了下你的所有改动,目前我只看到这处有点可疑。

23,125

社区成员

发帖
与我相关
我的任务
社区描述
Linux/Unix社区 应用程序开发区
社区管理员
  • 应用程序开发区社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧