5.TCP 实例
在server实例中,我们搞清了socket() bind() listen() accept() close() 这几个socket
server编程里非常重要的函数的功能意义,还有sockfd connfd 两个socket描述符的作用
现在我先分析一下C&S的连接流程:
双方都配置好以后,Client的connect(),引起TCP的三路握手过程。
Client的connet()和Server的accept()均返回,Client将阻塞在fgets()的用户键入上,
Server调用fork()后,子进程调用strcho(),阻塞在read()上
Server主进程的accept()阻塞
正常的终止连接的过程Clinet应该要发送一个EOF给server后,进行4次挥手
5.8 POSIX信号处理
信号有3种相关处置
signal函数建立POSIX的信号处置
1.提供一个handler函数,只要有特定信号发生就调用
2.把信号处置为SIG_IGN来忽略它
多进程并发,回收子进程资源
在server实例中,我们搞清了socket() bind() listen() accept() close() 这几个socket
server编程里非常重要的函数的功能意义,还有sockfd connfd 两个socket描述符的作用
现在我先分析一下C&S的连接流程:
双方都配置好以后,Client的connect(),引起TCP的三路握手过程。
Client的connet()和Server的accept()均返回,Client将阻塞在fgets()的用户键入上,
Server调用fork()后,子进程调用strcho(),阻塞在read()上
Server主进程的accept()阻塞
正常的终止连接的过程Clinet应该要发送一个EOF给server后,进行4次挥手
5.8 POSIX信号处理
信号有3种相关处置
signal函数建立POSIX的信号处置
1.提供一个handler函数,只要有特定信号发生就调用
2.把信号处置为SIG_IGN来忽略它
3.SIG_DFL启用默认处置
预处理:
typedef void Sigfunc(int);
Sigfunc *signal(int signo, Sigfunc *func);
//第二个参数的返回值是指向信号处理函数的指针
/* include signal */
typedef void Sigfunc(int);
Sigfunc * signal(int signo, Sigfunc *func)
{
struct sigaction act, oact;
act.sa_handler = func;
sigemptyset(&act.sa_mask);
act.sa_flags = 0;
if (signo == SIGALRM) {
#ifdef SA_INTERRUPT
act.sa_flags |= SA_INTERRUPT; /* SunOS 4.x */
#endif
} else {
#ifdef SA_RESTART
act.sa_flags |= SA_RESTART; /* SVR4, 44BSD */
#endif
}
if (sigaction(signo, &act, &oact) < 0)
return(SIG_ERR);
return(oact.sa_handler);
}
/* end signal */
waitpid()函数处理僵死的进程
Sigfunc * Signal(int signo, Sigfunc *func) /* for our signal() function */
{
Sigfunc *sigfunc;
if ( (sigfunc = signal(signo, func)) == SIG_ERR)
err_sys("signal error");
return(sigfunc);
}
void sig_chld(int signo)
{
pid_t pid;
int stat;
while( (pid = waitpid(-1, &stat, WNOHANG)) > 0)
printf("child %d terminated\n", pid);
return;
}
多进程并发,回收子进程资源
int main(int argc, char **argv)
{
int listenfd, connfd;
pid_t childpid;
socklen_t clilen;
struct sockaddr_in cliaddr, servaddr;
listenfd = Socket(AF_INET, SOCK_STREAM, 0);
bzero(&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = INADDR_ANY;
servaddr.sin_port = htons(SERVER_PORT);
Bind(listenfd, (struct sockaddr *) &servaddr, sizeof(servaddr));
Listen(listenfd, LISTENQ);
Signal(SIGCHLD, sig_chld);
for ( ; ; ) {
clilen = sizeof(cliaddr);
if ( (connfd = accept(listenfd, (struct sockaddr *) &cliaddr, &clilen)) < 0) {
if (errno == EINTR)
continue;
else
err_sys("accept error");
}
if ( (childpid = Fork() == 0)) {
Close(listenfd);
str_echo(connfd);
exit(0);
}
Close(connfd);
}
}