소켓 프로그래밍에서 프로그램 짜다 잘안되서 질문 드립니다.
글쓴이: lee3390 / 작성시간: 수, 2004/11/17 - 9:07오후
프로그램을 짜는도중 잘 안되서 질문을 드립니다.
소켓프로그램을 짜고있는데 클라이언트에서 서버로 쿼리문을 보내면
서버는 클라이언트에서 그 쿼리문을 받아서 sql서버와 통신후 그 쿼리결과를
다시 클라이언트에 보내주어서 클라이언트는 그 쿼리결과를 파일에 저장하는
것입니다.
그런데 쿼리를 주고난 후 받으면 다 받고나서 다운이 되어버립니다.
5개를 연속으로 쿼리를 보내야하는데 1개 받고나서 다운이 되어버리니 다음 쿼
리를 보낼수가 없네요 이거가지고 몇시간동안 고민하다 질문 올립니다.
아래 소스코드 있습니다.
server program code
#include<sys/socket.h> #include<sys/types.h> #include<netinet/in.h> #include <arpa/inet.h> #include <sys/stat.h> #include <sys/wait.h> #include <signal.h> #include <unistd.h> #include <string.h> #include <stdio.h> #include <errno.h> #include <netdb.h> #include <fcntl.h> #include <time.h> #include <mysql.h> #define PORT 7777 // 서버의 포트번호 #define BUFSIZE 512 // 한번에 전송 또는 수신 할 데이터의 양 #define MAX 5 // 최대 접속 대기 수 #define mys_host_name "localhost" #define mys_user_name "root" #define mys_password "estee" #define mys_db_name "trade" void sig_cld(int signo); int main(int argc, char *argv[]) { struct sockaddr_in saddr, caddr; // 소켓 어드레스 구조체 서버, 클라이언트의 IP, >포트정보를 갖는다. int sockfd, newsockfd; // 리스닝 소켓, 컨넥티드 소켓 char buf[BUFSIZE]; // 1024만큼의 buf를 생성 int count, cli_len; // count : 실질적인 데이터의 수 // char complete[10]="complete"; int pid,yes; int i; int c; MYSQL *db; MYSQL_RES *res_set; MYSQL_ROW row; db = mysql_init (NULL); if(db==NULL) { fprintf(stderr,"out of memory\n"); exit(1); } if (mysql_real_connect ( db,mys_host_name,mys_user_name,mys_password, mys_db_name,0,NULL,0) == NULL) { fprintf(stderr,"mysql connection failed : \nError %u (%s)\n", mysql_errno(db),mysql_error(db)); exit(1); } signal(SIGCHLD,sig_cld); bzero(buf,sizeof(buf)); memset(&saddr, 0, sizeof(saddr)); // 메모리 초기화 saddr.sin_family = AF_INET; // 인터넷 소켓 saddr.sin_addr.s_addr = htonl(INADDR_ANY); saddr.sin_port = htons(PORT); // 포트정보 수록 sockfd = socket(AF_INET, SOCK_STREAM, 0); // 소켓 생성, 인터넷 소켓, TCP, 0, sockfd = 소켓 번호 setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (void*)&yes, sizeof(int)); if( bind(sockfd, (struct sockaddr*) &saddr, sizeof(saddr)) == -1) { printf("bind Error \n"); exit(1); } listen(sockfd, MAX); while(1) { cli_len = sizeof(caddr); newsockfd = accept(sockfd,(struct sockaddr*) &caddr, &cli_len); if((pid=fork()) < 0) printf("fork error\n"); if(pid==0) { while((count = read(newsockfd, buf, BUFSIZE)) > 0) { c=1; mysql_query(db,buf); res_set=mysql_store_result(db); while((row=mysql_fetch_row(res_set))!=NULL) { bzero(buf,sizeof(buf)); for(i=0;i<mysql_num_fields(res_set);i++) { if(i>0) strcat(buf,","); if(row[i] == NULL) strcat(buf,"NULL"); else strcat(buf,row[i]); } strcat(buf,"\n"); write(newsockfd,buf,strlen(buf)); printf("%d %s\n",c,buf); c++; } bzero(buf,sizeof(buf)); printf("Send complete\n"); } close(newsockfd); exit(0); } }close(sockfd); mysql_close(db); } void sig_cld(int signo) { int stat; pid_t pid; while( (pid=waitpid(-1,&stat,WNOHANG)) > 0 ); }
client program code 입니다.
#include<stdio.h> #include<stdlib.h> #include<sys/socket.h> #include<sys/types.h> #include<netinet/in.h> #include<arpa/inet.h> #include<errno.h> #include<signal.h> #include<sys/wait.h> #include<fcntl.h> #include<string.h> #include<unistd.h> #define PORT 7777 #define BUFSIZE 512 int main() { struct sockaddr_in addr;// addr : 접속하고자 하는 서버의 IP와 Port int sockfd; // 소켓 번호 char buf[BUFSIZE]; int count=0,yes; int fd,aaa; char file[10][10] = {"data1.dat","data2.dat","data3.dat","data4.dat","data5.dat", "data6.dat","data7.dat","data8.dat","data9.dat","data10.dat"}; fd=open("receive.dat",O_CREAT|O_TRUNC|O_WRONLY,00644); memset(&addr, 0, sizeof(addr)); // 일단 0으로 다 채운다. addr.sin_family = AF_INET; addr.sin_addr.s_addr = inet_addr("128.134.54.178"); addr.sin_port = htons(PORT); sockfd = socket(AF_INET, SOCK_STREAM,0); setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (void*)&yes, sizeof(int)); connect(sockfd,(struct sockaddr*) &addr, sizeof(addr)); while(fgets(buf,BUFSIZE, stdin) > 0 ) { if(strncmp(buf, "quit",4) == 0) break; write(sockfd,buf,strlen(buf)); bzero(buf,sizeof(buf)); fd=open(file[count],O_CREAT|O_TRUNC|O_WRONLY,00644); while((aaa=read(sockfd, buf, BUFSIZE)) > 0) { // if ( !strcmp(buf,"complete") ) break; // printf("%d\n",aaa); write(fd,buf,aaa); printf("%s\n",buf); bzero(buf,sizeof(buf)); } close(fd); printf("Receive complete\n"); count++; }; close(sockfd); }
Forums:
자세히는 살펴보지 않았지만, 기본적으로 에러처리를 전혀 하지 않는 것이
자세히는 살펴보지 않았지만, 기본적으로 에러처리를 전혀 하지 않는 것이 문제가 될 듯 합니다.
예를 들어, socket(), connect() 함수 등등...
예외처리를 하고 다시 실행하면 문제가 뭔지 발견할 수 있을 듯 합니다.^^
아름다운 세상을 꿈꾸는 아름다운...잔인할 정도로 아름다운 녀석...그대이름은 빈(彬)!!!
[code:1]setsockopt(sockfd, SOL_SOCKE
setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (void*)&yes, sizeof(int));
음.. 저도 지금 소켓 공부하는데요.. 이 코드는 TIME_WAIT를 바로 죽이시는 옵션을 사용하신거죠? 이것 때문이 아닐까요??커피는 블랙이나 설탕만..
댓글 달기