我一直在试验http://www.linuxhowtos.org/C_C++/socket.htm的服务器代码。
常规服务器代码工作正常,我已将其略微更改为端口号中的硬编码而不是从命令行读取,但我想要一个能够容纳多个连接的服务器。它将等待来自3个不同客户端程序的短字符串定期进入。
我编译了下面但是我收到错误“ERROR,没有提供端口”,这没有任何意义,因为我肯定包含了端口号。
我正在使用带有CentOS的双核迷你处理器。它需要从套接字中读取少量数据,这些数据将定期传递。我知道forking可能会使cpu过载,而且多线程对于这种类型的机器来说也可能是内存密集型的。
是否有其他替代方案可以处理多个套接字连接,值得我研究一下?数据将分别每5秒钟,每隔几小时,每天两次。
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
void read_socket(int); /* function prototype */
void error(const char *msg)
{
perror(msg);
exit(1);
}
int main(int argc, char *argv[]){
int store_limit=10; //user input
int store_limit_secs=store_limit * 24 * 60 * 60; //user input
int sockfd, newsockfd, portno, pid;
socklen_t clilen;
struct sockaddr_in serv_addr, cli_addr;
if (argc < 2) {
fprintf(stderr,"ERROR, no port provided\n");
exit(1);
}
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0)
error("ERROR opening socket");
bzero((char *) &serv_addr, sizeof(serv_addr));
portno = 15000;
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);
while (1) {
newsockfd = accept(sockfd,
(struct sockaddr *) &cli_addr, &clilen);
if (newsockfd < 0)
error("ERROR on accept");
pid = fork();
if (pid < 0)
error("ERROR on fork");
if (pid == 0) {
close(sockfd);
read_socket(newsockfd);
exit(0);
}
else close(newsockfd);
} /* end of while */
close(sockfd);
return 0; /* we never get here */
}
void read_socket (int sock)
{
int n;
char buffer[256];
bzero(buffer,256);
n = read(sock,buffer,255);
if (n < 0) error("ERROR reading from socket");
printf("Here is the message: %s\n",buffer);
n = write(sock,"I got your message",18);
if (n < 0) error("ERROR writing to socket");
}
答案 0 :(得分:3)
如果没有命令行参数,argc
将为1
,因此该块将执行,导致程序退出:
if (argc < 2) {
fprintf(stderr,"ERROR, no port provided\n");
exit(1);
}
由于您对portno
的值进行了硬编码,因此可以删除此块。