server.c:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/types.h>
#include <unistd.h>
#define BUFLEN 10
int main(int argc, char **argv)
{
int sockfd;
int newfd;
struct sockaddr_in s_addr;
struct sockaddr_in c_addr;
char buf[BUFLEN];
socklen_t len;
/*任何完整的库必须定义socklen_t和int相同的尺寸大小*/
unsigned int listnum;
unsigned int port;
/*建立socket*/
if((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("socket");
/* perror(s) 用来将上一个函数发生错误的原因输出到标准设备(stderr)。参数 s 所指的字符串会先打印出,后面再加上错误原因字符串。此错误原因依照全局变量errno(这里的说法不准确,errno是一个宏,该宏返回左值) 的值来决定要输出的字符串。
在库函数中有个errno变量,每个errno值对应着以字符串表示的错误类型。当你调用"某些"函数出错时,该函数已经重新设置了errno的值。perror函数只是将你输入的一些信息和现在的errno所对应的错误一起输出。*/
exit(errno);
}
else
printf("socket create success!\n");
/*设置服务器端口*/
if(argv[2])
port = atoi(argv[2]);
else
port = 4567;
/*设置侦听队列长度*/
if(argv[3])
listnum = atoi(argv[3]);
else
listnum = 3;
/*设置服务器ip*/
bzero(&s_addr, sizeof(s_addr));
s_addr.sin_family = AF_INET;
s_addr.sin_port = htons(port);
if(argv[1])
s_addr.sin_addr.s_addr = inet_addr(argv[1]);
else
s_addr.sin_addr.s_addr = INADDR_ANY;
/*把地址和端口帮定到套接字上*/
if((bind(sockfd, (struct sockaddr*) &s_addr, sizeof(struct sockaddr))) == -1)
{
perror("bind");
exit(errno);
}
else
printf("bind success!\n");
/*侦听本地端口*/
if(listen(sockfd, listnum) == -1)
{
perror("listen");
exit(errno);
}
else
printf("the server is listening!\n");
while(1)
{
printf("*****************聊天开始***************\n");
len = sizeof(struct sockaddr);
if((newfd = accept(sockfd, (struct sockaddr*)&c_addr, &len)) == -1)
{
perror("accept");
exit(errno);
}
else
printf("正在与您聊天的客户端是:%s: %d\n", inet_ntoa(c_addr.sin_addr), ntohs(c_addr.sin_port));
while(1)
{
_retry:
/******发送消息*******/
bzero(buf, BUFLEN);
printf("请输入发送给对方的消息:");
/*fgets函数:从流中读取BUFLEN-1个字符*/
fgets(buf, BUFLEN, stdin);
/*打印发送的消息*/
//fputs(buf,stdout);
if(!strncasecmp(buf, "quit", 4))
/* 函数说明:strncasecmp()用来比较参数s1和s2字符串前n个字符,比较时会自动忽略大小写的差异.若参数s1和s2字符串相同,则返回0; 若s1大于s2,则返回大于0的值;若s1小于s2,则返回小于0的值。*/
{
printf("server 请求终止聊天!\n");
break;
}
/*如果输入的字符串只有"\n",即回车,那么请重新输入*/
if(!strncmp(buf, "\n", 1))
{
printf("输入的字符只有回车,这个是不正确的!!!\n");
goto _retry;
}
/*如果buf中含有'\n',那么要用strlen(buf)-1,去掉'\n'*/
if(strchr(buf, '\n'))
/* extern char *strchr(const char *s,char c);功能:查找字符串s中首次出现字符c的位置.说明:返回首次出现c的位置的指针,返回的地址是被查找字符串指针开始的第一个与Val相同字符的指针,如果s中不存在c则返回NULL。
返回值:成功则返回要查找字符第一次出现的位置,失败返回NULL */
len = send(newfd, buf, strlen(buf)-1, 0);
/*1 #include <sys/socket.h>
2 ssize_t recv(int sockfd, void *buff, size_t nbytes, int flags);
3 ssize_t send(int sockfd, const void *buff, size_t nbytes, int flags);
flags参数值为0或:
flags 说明 recv send
MSG_DONTROUTE 绕过路由表查找 •
MSG_DONTWAIT 仅本操作非阻塞 • •
MSG_OOB 发送或接收带外数据 • •
MSG_PEEK 窥看外来消息 •
MSG_WAITALL 等待所有数据 •
1. send解析
sockfd:指定发送端套接字描述符。
buff: 存放要发送数据的缓冲区
nbytes: 实际要改善的数据的字节数
flags: 一般设置为0
1) send先比较发送数据的长度nbytes和套接字sockfd的发送缓冲区的长度,如果nbytes > 套接字sockfd的发送缓冲区的长度, 该函数返回SOCKET_ERROR;
2) 如果nbtyes <= 套接字sockfd的发送缓冲区的长度,那么send先检查协议是否正在发送sockfd的发送缓冲区中的数据,如果是就等待协议把数据发送完,如果协议还没有开始发送sockfd的发送缓冲区中的数据或者sockfd的发送缓冲区中没有数据,那么send就比较sockfd的发送缓冲区的剩余空间和nbytes
3) 如果 nbytes > 套接字sockfd的发送缓冲区剩余空间的长度,send就一起等待协议把套接字sockfd的发送缓冲区中的数据发送完
4) 如果 nbytes < 套接字sockfd的发送缓冲区剩余空间大小,send就仅仅把buf中的数据copy到剩余空间里(注意并不是send把套接字sockfd的发送缓冲区中的数据传到连接的另一端的,而是协议传送的,send仅仅是把buf中的数据copy到套接字sockfd的发送缓冲区的剩余空间里)。
5) 如果send函数copy成功,就返回实际copy的字节数,如果send在copy数据时出现错误,那么send就返回SOCKET_ERROR; 如果在等待协议传送数据时网络断开,send函数也返回SOCKET_ERROR。
6) send函数把buff中的数据成功copy到sockfd的改善缓冲区的剩余空间后它就返回了,但是此时这些数据并不一定马上被传到连接的另一端。如果协议在后续的传送过程中出现网络错误的话,那么下一个socket函数就会返回SOCKET_ERROR。(每一个除send的socket函数在执行的最开始总要先等待套接字的发送缓冲区中的数据被协议传递完毕才能继续,如果在等待时出现网络错误那么该socket函数就返回SOCKET_ERROR)
7) 在unix系统下,如果send在等待协议传送数据时网络断开,调用send的进程会接收到一个SIGPIPE信号,进程对该信号的处理是进程终止。
2.recv函数
sockfd: 接收端套接字描述符
buff: 用来存放recv函数接收到的数据的缓冲区
nbytes: 指明buff的长度
flags: 一般置为0
1) recv先等待s的发送缓冲区的数据被协议传送完毕,如果协议在传送sock的发送缓冲区中的数据时出现网络错误,那么recv函数返回SOCKET_ERROR
2) 如果套接字sockfd的发送缓冲区中没有数据或者数据被协议成功发送完毕后,recv先检查套接字sockfd的接收缓冲区,如果sockfd的接收缓冲区中没有数据或者协议正在接收数据,那么recv就一起等待,直到把数据接收完毕。当协议把数据接收完毕,recv函数就把s的接收缓冲区中的数据copy到buff中(注意协议接收到的数据可能大于buff的长度,所以在这种情况下要调用几次recv函数才能把sockfd的接收缓冲区中的数据copy完。recv函数仅仅是copy数据,真正的接收数据是协议来完成的)
3) recv函数返回其实际copy的字节数,如果recv在copy时出错,那么它返回SOCKET_ERROR。如果recv函数在等待协议接收数据时网络中断了,那么它返回0。
4) 在unix系统下,如果recv函数在等待协议接收数据时网络断开了,那么调用 recv的进程会接收到一个SIGPIPE信号,进程对该信号的默认处理是进程终止。 */
/*如果buf中没有'\n',则用buf的真正长度strlen(buf)*/else
len = send(newfd, buf, strlen(buf), 0);
if(len > 0)
printf("消息发送成功,本次共发送的字节数是:%d\n",len);
else{
printf("消息发送失败!\n");
break;
}
/******接收消息*******/
bzero(buf, BUFLEN);
len = recv(newfd, buf, BUFLEN, 0);
if(len > 0)
printf("客户端发来的信息是:%s,共有字节数是: %d\n",buf,len);
else{
if(len < 0 )
printf("接受消息失败!\n");
else
printf("客户端退出了,聊天终止!\n");
break;
}
}
/*关闭聊天的套接字*/
close(newfd);
/*是否退出服务器*/
printf("服务器是否退出程序:y->是;n->否? ");
bzero(buf, BUFLEN);
fgets(buf, BUFLEN, stdin);
if(!strncasecmp(buf, "y", 1)){
printf("server 退出!\n");
break;
}
}
/*关闭服务器的套接字*/
close(sockfd);
return 0;
}
client.c:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/types.h>
#include <unistd.h>
#define BUFLEN 10
int main(int argc, char **argv)
{
int sockfd;
struct sockaddr_in s_addr;
socklen_t len;
unsigned int port;
char buf[BUFLEN];
/*建立socket*/
if((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("socket");
exit(errno);
}
else
printf("socket create success!\n");
/*设置服务器端口*/
if(argv[2])
port = atoi(argv[2]);
else
port = 4567;
/*设置服务器ip*/
printf ("hello world.\n");
bzero(&s_addr, sizeof(s_addr));
s_addr.sin_family = AF_INET;
s_addr.sin_port = htons(port);
if (0==inet_aton(argv[1], &s_addr.sin_addr))
{
perror(argv[1]);
exit(errno);
}
/*开始连接服务器*/
if(connect(sockfd, (struct sockaddr*)(&s_addr), sizeof(struct sockaddr)) == -1)
{
perror("connect");
exit(errno);
}
else
printf("conncet success!\n");
while(1)
{
/******接收消息*******/
bzero(buf, BUFLEN);
len = recv(sockfd, buf, BUFLEN, 0);
if(len > 0)
printf("服务器发来的消息是:%s,共有字节数是: %d\n", buf, len);
else{
if(len < 0 )
printf("接受消息失败!\n");
else
printf("服务器退出了,聊天终止!\n");
break;
}
_retry:
/******发送消息*******/
bzero(buf, BUFLEN);
printf("请输入发送给对方的消息:");
/*fgets函数:从流中读取BUFLEN-1个字符*/
fgets(buf, BUFLEN, stdin);
/*打印发送的消息*/
//fputs(buf,stdout);
if(!strncasecmp(buf, "quit", 4))
{
printf("client 请求终止聊天!\n");
break;
}
/*如果输入的字符串只有"\n",即回车,那么请重新输入*/
if(!strncmp(buf, "\n", 1))
{
printf("输入的字符只有回车,这个是不正确的!!!\n");
goto _retry;
}
/*如果buf中含有'\n',那么要用strlen(buf)-1,去掉'\n'*/
if(strchr(buf,'\n'))
len = send(sockfd,buf,strlen(buf)-1,0);
/*如果buf中没有'\n',则用buf的真正长度strlen(buf)*/
else
len = send(sockfd, buf, strlen(buf), 0);
if(len > 0)
printf("消息发送成功,本次共发送的字节数是:%d\n",len);
else
{
printf("消息发送失败!\n");
break;
}
}
/*关闭连接*/
close(sockfd);
return 0;
}