我是刚接触网络的人,正在尝试用C创建一个简单的客户端,服务器套接字程序,其中的参数确定该程序应作为客户端还是服务器运行。我通过使用简单的if语句(如果给出了一个标志,以服务器身份运行,否则以客户端身份运行)来做到这一点,但是我不确定如何进行测试。我在一个终端(例如,在localhost和端口号3000上)中以参数作为服务器运行我的代码,并打开另一个终端,并以参数作为客户机(也在localhost和同一端口上)运行代码。
预期结果是看到客户端提示用户输入一条消息(如果连接成功),然后将该消息发送到服务器,服务器将打印出该消息,但是,我在客户端上没有得到提示输入一条消息。
(我从许多在线网站中获得了有关服务器和客户端行为的代码,但它们将client.c和server.c分开了,而我想将两者合并为一个.c程序)
下面是我的代码,该错误由
触发if (connect(sockfd,(struct sockaddr *)&serv_addr,sizeof(serv_addr)) < 0)
error("ERROR connecting");
代码的客户部分中的。
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
#include <string.h>
void error(char *msg)
{
perror(msg);
exit(1);
}
int main(int argc, char *argv[])
{
int sockfd, newsockfd, portno, clilen;
char buffer[256];
struct sockaddr_in serv_addr, cli_addr;
int n, i, server = 0; // 1 = server, 0 = client
// check if server or client
for (i = 0; i<argc; i++) {
if (strcmp(argv[i], "-l") == 0)
server = 1;
}
// client
if (server == 0) {
struct hostent *server;
if (argc < 3) {
fprintf(stderr,"usage %s hostname port\n", argv[0]);
exit(0);
}
portno = atoi(argv[2]);
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0)
error("ERROR opening socket");
server = gethostbyname(argv[1]);
if (server == NULL) {
fprintf(stderr,"FOUR*** ERROR, no such host\n");
exit(0);
}
bzero((char *) &serv_addr, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
bcopy((char *)server->h_addr,
(char *)&serv_addr.sin_addr.s_addr,
server->h_length);
serv_addr.sin_port = htons(portno);
if (connect(sockfd,(struct sockaddr *)&serv_addr,sizeof(serv_addr)) < 0)
error("ERROR connecting");
printf("Please enter the message: ");
bzero(buffer,256);
fgets(buffer,255,stdin);
n = write(sockfd,buffer,strlen(buffer));
if (n < 0)
error("ERROR writing to socket");
bzero(buffer,256);
n = read(sockfd,buffer,255);
if (n < 0)
error("ERROR reading from socket");
printf("%s\n",buffer);
return 0;
}
// server
if (server == 1) {
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0)
error("ERROR opening socket");
bzero((char *) &serv_addr, sizeof(serv_addr));
portno = atoi(argv[1]);
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(portno);
if (bind(sockfd, (struct sockaddr *) &serv_addr,
sizeof(serv_addr)) < 0)
error("ERROR on binding");
listen(sockfd,5);
clilen = sizeof(cli_addr);
newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, &clilen);
if (newsockfd < 0)
error("ERROR on accept");
bzero(buffer,256);
n = read(newsockfd,buffer,255);
if (n < 0) error("ERROR reading from socket");
printf("Here is the message: %s\n",buffer);
n = write(newsockfd,"I got your message",18);
if (n < 0) error("ERROR writing to socket");
return 0;
}
}
这是确切的错误输出:
在一个终端窗口中,我首先将该程序作为服务器运行:
$ ./socketz -l localhost 2003
然后在另一个终端窗口中,我以客户端身份运行该程序:
$ ./socketz localhost 2003
ERROR connecting: Connection refused
答案 0 :(得分:1)
无法连接的原因是因为您的服务器进程未在端口2003上侦听。特别是在此行上:
serv_addr.sin_port = htons(portno);
portno
的值为零,这导致serv_addr.sin_port
的值也为零,accept()
解释为意味着它应该只选择一个可用的TCP端口进行绑定
问题的根源在这里:
portno = atoi(argv[1]);
...该行根据您在运行程序时输入的第一个参数为portno
分配了一个值,但是您输入了以下内容:
./a.out -l localhost 2003
因此第一个参数是“ -l”,它是一个非数字,因此将导致atoi()
返回0。我想您打算使用的是portno = atoi(argv[3]);
。