int socket_fd, connect_fd;
struct sockaddr_in servaddr;
初始化:
socket_fd = sock(AF_INET, SOCK_STREAM, 0);
bzero(&serveraddr, sizeof(serveraddr));
servaddr.sin_family = AF_INET; /设置IPv4通信/
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//IP地址设置成INADDR_ANY,让系统自动获取本机的IP地址。
servaddr.sin_port = htons(SERV_PORT);//设置服务器端口为SERV_PORT
bind():
bind(socket_fd, (struct sockaddr*)&serveraddr, sizeof(serveraddr));
listen():
listen(socket_fd, 10); //10指的是连接队列的最大长度(第二个参数就是未完成队列的大小)
accept():
int accept(int sockfd, struct sockaddr* addr, socklen_t* len)
accept默认会阻塞进程,直到有一个客户连接建立后返回,它返回的是一个新可用的套接字,这个套接字是连接套接字。此时我们需要区分两种套接字,一种套接字正如accept的参数sockfd,它是监听套接字,在调用listen函数之后,一个套接字会从主动连接的套接字变身为一个监听套接字;而accept返回是一个连接套接字,它代表着一个网络已经存在的点点连接。自然要问的是:为什么要有两种套接字?原因很简单,如果使用一个描述字的话,那么它的功能太多,使得使用很不直观,同时在内核确实产生了一个这样的新的描述字。如果accept成功返回,则服务器与客户已经正确建立连接了,此时服务器通过accept返回的套接字来完成与客户的通信。
参数sockfd就是上面解释中的监听套接字
参数addr,这是一个结果参数,它用来接受一个返回值,这返回值指定客户端的地址,当然这个地址是通过某个地址结构来描述的,用户应该知道这一个什么样的地址结 构。如果对客户的地址不感兴趣,那么可以把这个值设置为NULL。
如果你给accept函数第2和第3个参数赋值后,就要记得“用之”,否则就会报错。
connect_fd = accept(socket_fd, (struct sockaddr*)NULL, NULL))
connect_fd = accept(socket_fd, (struct sockaddr *)&clientaddr, &addrlen);//返回的是客户端和服务端专用通道的socket描述符
read()/write()/send()/recv:
Ssize_t write(int fd,const void *buf,size_t nbytes);
Write函数将buf中的nbytes字节内容写入到文件描述符中,成功返回写的字节数,失败返回-1.
Ssize_t read(int fd,void *buf,size_t nbyte)
Read函数是负责从fd中读取内容,当读取成功时,read返回实际读取到的字节数,如果返回值是0,表示已经读取到文件的结束了,小于0表示是读取错误。
int recv(int sockfd,void *buf,int len,int flags)
int send(int sockfd,void *buf,int len,int flags)
ecv和send函数提供了和read和write差不多的功能.但是他们提供 了第四个参数来控制读写操作.前面的三个参数和read,write相同,第四个参数能够是0或是以下的组合.
| MSG_DONTROUTE | 不查找路由表 |
| MSG_OOB | 接受或发送带外数据 |
| MSG_PEEK | 查看数据,并不从系统缓冲区移走数据|
| MSG_WAITALL | 等待任何数据 |
|————————————————————–|
MSG_DONTROUTE:是send函数使用的标志.这个标志告诉IP协议.目的主机在本地网络上面,没有必要查找路由表.这个标志一般用网络诊断和路由程式里面.
MSG_OOB:表示能够接收和发送带外的数据.关于带外数据我们以后会解释的.
MSG_PEEK:是recv函数的使用标志,表示只是从系统缓冲区中读取内容,而不清楚系统缓冲区的内容.这样下次读的时候,仍然是相同的内容.一般在有多个进程读写数据时能够使用这个标志.
MSG_WAITALL是recv函数的使用标志,表示等到任何的信息到达时才返回.使用这个标志的时候recv回一直阻塞,直到指定的条件满足,或是发生了错误.1)当读到了指定的字节时,函数正常返回.返回值等于len 2)当读到了文档的结尾时,函数正常返回.返回值小于len 3)当操作发生错误时,返回-1,且配置错误为相应的错误号(errno)
假如flags为0,则和read,write相同的操作.更有其他的几个选项,但是我们实际上用的很少,能够查看Linux Programmer’s Manual得到周详解释.
len= recv(connect_fd, buf, MAXLINE, 0);
buf[len] = ‘\0’;
close():
close(connect_fd);
close(socket_fd);
struct sockaddr_in servaddr;
int client_sockfd;
初始化:
client_sockfd=socket(AF_INET,SOCK_STREAM,0)
bzero(&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET;
//char ipstr[] = “192.168.6.254”;
inet_pton(AF_INET, ipstr, &serveraddr.sin_addr.s_addr);
//#define SERVER_PORT 8000
serveraddr.sin_port = htons(SERVER_PORT);
connect():
connect(confd, (struct sockaddr *)&serveraddr, sizeof(serveraddr));
向服务器端发送数据:
send(client_sockfd, sendbuf, strlen(sendbuf), 0)
接受服务器端传过来的数据:
rec_len = recv(client_sockfd,receivebuf, MAXLINE,0);
关闭:
close(client_sockfd);
服务器:
/*********************************************************************************
* Copyright: (C) 2017 zoulei
* All rights reserved.
*
* Filename: server.c
* Description: This file
*
* Version: 1.0.0(2017年06月18日)
* Author: zoulei
* ChangeLog: 1, Release initial version on "2017年06月18日 18时13分42秒"
*
********************************************************************************/
#include
#include
#include
#include
#include
#include
#include
#define SERV_PORT 9998
#define MAXLINE 4096
int main(int argc, char** argv)
{
int socket_fd, connect_fd;
struct sockaddr_in servaddr; /* 服务器端网络地址结构体 */
char buf[MAXLINE],sendbuf[MAXLINE];
int len;
/*创建服务器端套接字--IPv4协议,面向连接通信,TCP协议*/
if( (socket_fd = socket(AF_INET, SOCK_STREAM, 0)) == -1 ){
printf("create socket error: %s(errno: %d)\n",strerror(errno),errno);
exit(0);
}
/*初始化*/
memset(&servaddr, 0, sizeof(servaddr));/*数据初始化-清零 */
servaddr.sin_family = AF_INET; /*设置IPv4通信*/
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//IP地址设置成INADDR_ANY,让系统自动获取本机的IP地址。
servaddr.sin_port = htons(SERV_PORT);//设置服务器端口为SERV_PORT
/*将本地地址绑定到所创建的套接字上*/
if( bind(socket_fd, (struct sockaddr*)&servaddr, sizeof(servaddr)) <0)
{
printf("bind socket error: %s(errno: %d)\n",strerror(errno),errno);
exit(0);
}
/*开始监听是否有客户端连接*/
if( listen(socket_fd, 10) <0)
{
printf("listen socket error: %s(errno: %d)\n",strerror(errno),errno);
exit(0);
}
printf("waiting for client's connection......\n");
/*阻塞直到有客户端连接,不然多浪费CPU资源*/
if((connect_fd = accept(socket_fd, (struct sockaddr*)NULL, NULL)) <0)
{
printf("accept socket error: %s(errno: %d)",strerror(errno),errno);
exit(1);
}
/*接受客户端传过来的数据*/
while((len= recv(connect_fd, buf, MAXLINE, 0))>0)
{
buf[len] = '\0';
printf("receive message from client: %s\n", buf);
/*向客户端发送回应数据*/
printf("send message to client: \n");
fgets(sendbuf, 4096, stdin);
if( send(connect_fd, sendbuf, strlen(sendbuf), 0) < 0)
{
printf("send messaeg error: %s(errno: %d)\n", strerror(errno), errno);
exit(0);
}
}
close(connect_fd);
close(socket_fd);
}
客户端:
/*********************************************************************************
* Copyright: (C) 2017 zoulei
* All rights reserved.
*
* Filename: client.c
* Description: This file
*
* Version: 1.0.0(2017年06月12日)
* Author: zoulei
* ChangeLog: 1, Release initial version on "2017年06月12日 20时06分47秒"
*
********************************************************************************/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define MAXLINE 1024
#define SERV_PORT 9998
int main(int argc, char *argv[])
{
char sendbuf[MAXLINE],receivebuf[MAXLINE];
struct sockaddr_in servaddr;
int client_sockfd;
int rec_len;
/* 判断命令端输入的参数是否正确 */
if( argc != 2)
{
printf("usage: ./client \n");
exit(0);
}
/* 创建客户端套接字--IPv4协议,面向连接通信,TCP协议*/
if((client_sockfd=socket(AF_INET,SOCK_STREAM,0))<0)
{
perror("socket");
exit(0);
}
/* 初始化 */
memset(&servaddr,0,sizeof(servaddr)); /* 数据初始化-清零 */
servaddr.sin_family = AF_INET; /* 设置IPv4通信 */
servaddr.sin_port = htons(SERV_PORT);/* 设置服务器端口号 */
/* IP地址转换函数,将点分十进制转换为二进制 */
if( inet_pton(AF_INET, argv[1], &servaddr.sin_addr) <= 0)
{
printf("inet_pton error for %s\n",argv[1]);
exit(0);
}
/* 将套接字绑定到服务器的网络地址上*/
if( connect(client_sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr))<0)
{
perror("connected failed");
exit(0);
}
/* 循环发送接收数据,send发送数据,recv接收数据 */
while(1)
{
printf("send msg to server: \n");
fgets(sendbuf, 1024, stdin);
/* 向服务器端发送数据 */
if( send(client_sockfd, sendbuf, strlen(sendbuf), 0) < 0)
{
printf("send msg error: %s(errno: %d)\n", strerror(errno), errno);
exit(0);
}
/* 接受服务器端传过来的数据 */
if((rec_len = recv(client_sockfd,receivebuf, MAXLINE,0)) == -1)
{
perror("recv error");
exit(1);
}
receivebuf[rec_len]='\0';
printf("Response from server: %s\n",receivebuf);
}
/* 关闭套接字 */
close(client_sockfd);
return 0;
}