基础步骤
服务端
第一步:调用socket
函数创建套接字
第二步:调用bind
函数分配IP
地址和端口号
第三步:调用listen
函数转为可接听状态
第四步:调用accept
函数受理可接听状态
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<arpa/inet.h>
#include<sys/socket.h>
void error_handling(char *message);
int main(int argc,char *argv[]){
int serv_sock;
int clnt_sock;
struct sockaddr_in serv_addr;
struct sockaddr_in clnt_addr;
socklen_t clnt_addr_size;
char message[]="Hello World!";
if(argc!=2){
printf("Usage : %s <port>\n",argv[0]);
exit(1);
}
serv_sock=socket(PF_INET,SOCK_STREAM,0);//通过socket函数创建套接字
if(serv_sock == -1)
error_handling("server:socket() error");
printf("[server] socket ok!\n");
// printf("server:socket() is ok\n");
memset(&serv_addr,0,sizeof(serv_addr));
serv_addr.sin_family=AF_INET;
serv_addr.sin_addr.s_addr=htonl(INADDR_ANY);
serv_addr.sin_port=htons(atoi(argv[1]));
printf("%d\n",serv_addr.sin_port);
if(bind(serv_sock,(struct sockaddr*) &serv_addr,sizeof(serv_addr))==-1)
error_handling("server:bind() error");//利用bind函数分配IP地址和端口号
printf("[server] bind ok!\n");
if(listen(serv_sock,5)==-1)
error_handling("server:listen() error");//调用listen函数将套接字转为可接收状态
printf("[server] is listening ok !\n");
clnt_addr_size = sizeof(clnt_addr);
clnt_sock = accept(serv_sock,(struct sockaddr*)&clnt_addr,&clnt_addr_size);
printf("[server] accpet() ok");
if(clnt_sock==-1)
error_handling("server:accept() error");//调用accpet函数受理连接请求。如果在没有连接请求的情况下调用该函数,则不会返回,直到有连接请求位置。
write(clnt_sock,message,sizeof(message));//传输数据
close(clnt_sock);
close(serv_sock);
return 0 ;
}
void error_handling(char *message){
fputs(message,stderr);
fputs("(server)\n",stderr);
exit(1);
}
客户端
第一步:创建套接字
第二步:连接
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<arpa/inet.h>
#include<sys/socket.h>
void error_handling(char *message);
int main(int argc,char* argv[]){
int sock;
struct sockaddr_in serv_addr;
char message[30];
int str_len;
if(argc != 3){
printf("Usage : %s <IP><port>\n",argv[0]);
exit(1);
}
printf("[client] is trying creating socket.....`\n");
sock=socket(PF_INET, SOCK_STREAM, 0);
if(sock == -1)
error_handling("client's socket() error");
printf("[client]'s socket is ok\n");
memset(&serv_addr, 0, sizeof(serv_addr));
serv_addr.sin_family=AF_INET;
serv_addr.sin_addr.s_addr=inet_addr(argv[1]);
serv_addr.sin_port=htons(atoi(argv[2]));
printf("%d\n",serv_addr.sin_port);
printf("%d",atoi(argv[2]));
if(connect(sock, (struct sockaddr*)&serv_addr,sizeof(serv_addr))==-1)
error_handling("[client]:connect() error!");
str_len = read(sock,message,sizeof(message)-1);
if(str_len==-1)
error_handling("[client] read() error\n");
printf("Message from server : %s \n",message);
close(sock);
return 0;
}
void error_handling(char *message){
perror(message);
fputs(message,stderr);
fputc('\n',stderr);
exit(1);
}
细节:
perror
可以输出具体错误信息,很有用!!!输出的是系统调用的最近一次错误。
setsid
可以让程序后台运行
文件操作
socket
在linux
中是文件的一种。
文件描述符(文件句柄):
标准输入:0
标准输出:1
标准错误:2
open
函数,打开文件:
int open(const char *path,int flag);
成功时返回文件描述符,失败时返回-1
文件打开模式有:
close
函数,关闭文件:
close(fd)
,所以可以关闭套接字。
write
函数,输出数据:
ssize_t write(int fd,const void * buf,size_t nbytes);
ssize_t
有符号的int
。
read
函数,读取数据:
read(fd,buf,sizeof(buf));
文件描述符是按顺序编号的,从3
开始。