1.什么什么是socket
是套接字可以实现两个不同主机之间的通讯。
官方:就是对网络中不同主机上的应用进程之间进行双向通信的端点的抽象。一个套接字就是网络上进程通信的一端,提供了应用层进程利用网络协议交换数据的机制。从所处的地位来讲,套接字上联应用进程,下联网络协议栈,是应用程序通过网络协议进行通信的接口,是应用程序与网络协议根进行交互的接口。
通过上面的图形,由于底一层的需要向高一层的提供服务,我们大致的理解应用程序需要传输层的tcp和网络层的ip协议提供服务,但是我们这章要分析的socket它是在tcpip协议的那一部分呢,就好比,我们的通讯线路已经有明确的规定,我们的驿站要设计在哪个地方一样。
2.相关API
1.socket
其中domain
这个参数一般国内选择iPV4这个因特网域,type
会对应着因特网域选着TCP协议
2.bind
这个函数用于配置socket的一些信息,最为复杂和重要int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
- sockfd是socket的返回值描述符
- const struct sockaddr *add结构体
- 第二个结构体的大小
*const struct sockaddr *add**详细介绍,就是以上那个结构体,上面是老版本下面是新版本
struct sockaddr_in {
__kernel_sa_family_t sin_family;//这个参数就socket的domin
__be16 sin_port; //端口号
struct in_addr sin_addr;
unsigned char __pad[__SOCK_SIZE__ - sizeof(short int) -
sizeof(unsigned short int) - sizeof(struct in_addr)];
};
字节序转换 大端字节序:起始地址存放高序字节。小端字节序:起始地址存放低序字节。
上面那个结构体的端口号就需要,用到字节序的转换,转换成为网络看得懂的字节。
saa.sin_port =htons(8848);
地址转换
转换地址用这个API,如果用第一个API上面结构体中的结构体就不用配置
inet_aton("127.0.0.1 ",&saa.sin_addr);
3.listen
第一个参数是socket的标识符,第二个参数是连接数。
4.accept
返回值是客户端的标识符
3.编程示例
编程心得,调用函数尽量去判断一下它们的返回值
出现bug是 inet_aton("127.0.0.1 ",&saa.sin_addr);
配置出错,应该配置为inet_aton("192.168.0.108 ",(struct in_addr *)&saa.sin_addr.s_addr);bind才不会出错
服务端
#include<stdio.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
int main(int arge,char **argv)
{
int s_fd;
struct sockaddr_in caa;
memset((struct sockaddr*)&caa,0,sizeof(struct sockaddr_in));
int len=sizeof(struct sockaddr_in);
int c_fd;
char *p=NULL;
p=(char *)malloc(30*sizeof(char));
char * str;
str=(char *)malloc(30*sizeof(char));
int n_read;
int i=10;
//1.socket
s_fd=socket(AF_INET,SOCK_STREAM,0);
if(s_fd==-1)
{
perror("socket");
exit(-1);
}
//2.bind
struct sockaddr_in saa;
memset((struct sockaddr_in*)&saa,0,sizeof(struct sockaddr_in));
saa.sin_family =AF_INET;
saa.sin_port =htons(atoi(argv[2]));
//saa.sin_addr.s_addr=inet_addr();
inet_aton(argv[1],(struct in_addr *)&saa.sin_addr.s_addr);
int ret=0;
ret =bind(s_fd,(struct sockaddr *)&saa,sizeof(struct sockaddr_in));
if(ret<0)
{
perror("bind");
exit(-1);
}
//3.listen
int rmt;
rmt=listen(s_fd,10);
if(rmt<0)
{
perror("listen");
exit(-1);
}
//4.accept
while(1){
c_fd=accept(s_fd,(struct sockaddr *)&caa,&len);
if(c_fd==-1)
{
perror("accept");
exit(-1);
}
printf("listen suceesfull\n");
//5.read
if(fork()==0){
while(1){
n_read= read(c_fd,p,30);
if(n_read==-1)
{
perror("read");
}
printf("zhis is srever. formation from client %s\n",p);
}
}
// 6.write
if(fork()==0)
{
while(1){
printf("input\n");
scanf("%s",str);
write(c_fd,str,strlen(str));
memset(str,0,sizeof(len));
}
}
i--;
if(i==0)
{
close(s_fd);
}
}
return 0;
}
客户端
#include<stdio.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
int main()
{
int c_ad;
//1.socket
c_ad=socket(AF_INET,SOCK_STREAM,0);
struct sockaddr_in saa;
memset((struct sockaddr_in*)&saa,0,128);
saa.sin_family =AF_INET;
saa.sin_port =htons(8868);
inet_aton("192.168.58.128 ",(struct in_addr *)&saa.sin_addr.s_addr);
//2.connect
if(connect(c_ad,(struct sockaddr *)&saa,sizeof(struct sockaddr_in))==-1)
{
perror("connect");
exit(-1);
};
//3.write
char *len;
len=(char *)malloc(30*sizeof(char));
char buf[128];
int n_read;
while(1){
if(fork()==0){
while(1)
{
printf("input\n");
scanf("%s",len);
write(c_ad,len,sizeof(len));
memset(len,0,sizeof(len));
}
}
//4.read
n_read= read(c_ad,buf,128);
if(n_read==-1)
{
perror("read");
}
printf("i am client %s.formation is server %s\n",inet_ntoa(saa.sin_addr),buf);
}
//5.write
// write(c_ad,"i am client",15);
return 0;
}