1.
#include<stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include<string.h>
#include <sys/stat.h>
#include <fcntl.h>
#define ERR_MSG(msg) do{\
fprintf(stderr,"line : %d",__LINE__);\
perror(msg);\
return -1;\
}while(0)
int main(int argc, const char *argv[])
{
int cfd=socket( AF_INET , SOCK_DGRAM ,0);
if(-1==cfd){
ERR_MSG("socket errno");
}
struct sockaddr_in clientaddr;
clientaddr.sin_family=AF_INET ;
clientaddr.sin_port=htons(7777);
clientaddr.sin_addr.s_addr=inet_addr("192.168.17.165");
if(-1==bind(cfd,(struct sockaddr*)&clientaddr,sizeof(clientaddr))){
ERR_MSG("bind errno");
}
printf("绑定客户端自身成功\n");
// 填服务器的地址信息
struct sockaddr_in serveraddr;
serveraddr.sin_family=AF_INET ;
serveraddr.sin_port=htons(69);
serveraddr.sin_addr.s_addr=inet_addr("192.168.8.104");
//发送
char filename[32]="";
int blocknum=0; // 用于校验已经接到的数据包块编号
struct sockaddr_in temp;
socklen_t temp_len=sizeof(temp);
int fd=0;
RE_NAME:
printf("请输入要下载的文件名>>>");
scanf("%s",filename);
// 组装请求包
char buf[600]="";
/* // 组装方式1 读请求===>一个字节一个字节的赋值===>网络字节序
buf[0]=0;
buf[1]=1;
*/
/* // 组装方式2 通过改变指针的操作空间来赋网络字节序的值
*(short *)buf=htons(1);
*/
// 组装方式3 使用格式字符串去赋值
int nbytes=sprintf(buf,"%c%c%s%c%s%c",0,1,filename,0,"octet",0);
// 发送请求包
sendto(cfd,buf,nbytes,0,(struct sockaddr*)&serveraddr,sizeof(serveraddr));
// 接受数据有可能是数据包有可能出错
// 需要保存服务器的网络信息结构体因为临时端口号在里面
// 定义变量存放临时端口
short code=0; // 用来保存操作码
short num=0; // 用来保存 要么是块编号 要么是差错码
char txt[512]=""; // 保存要么是文件内容 要么是差错信息
while(1){
// 接收数据
nbytes=recvfrom(cfd,buf,600,0,(struct sockaddr*)&temp,&temp_len);
// 解析操作码
code=ntohs(*(short*)buf);
// 解析快编号或者是差错码
num=ntohs(*(short*)(buf+2));
// 解析文件内容或者差错信息
strncpy(txt,buf+4,nbytes-4);
if(code==5){
printf("errrnocode=[%d] errnotxt=[%s]\n",code,txt);
goto RE_NAME;
}else if(3==code&&blocknum+1=num){
blocknum++; // 用于校验下次接到的数据包
if(1==num){ // 只有第一个数据包需要打开文件后面直接使用即可
if(-1==(fd=open(filename,O_WRONLY|O_CREAT|O_TRUNC,0777))){
ERR_MSG("open errno");
}
}
//将文件内容写入文件
if(-1==write(fd,txt,nbytes-4)){
ERR_MSG("write errno");
}
// 组装ACK
*(short*)buf=htons(4);
*(short*)(buf+2)=htons(num);
// 发送ACK
if(1-==sendto(cfd,buf,4,0,(struct sockaddr*)&temp,temp_len)){
ERR_MSG("sendto errno"):
}
if(nbytes-4<512){
break;
}
}
close(fd);
close(cfd);
// printf("code=[%d] num=[%d] txt=[%s]\n",code,num,txt);
}
return 0;
}
2.
#include<stdio.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<netinet/ip.h>
#include<arpa/inet.h>
#include<string.h>
#include<unistd.h>
#define ERR_MSG(msg) do{\
fprintf(stderr,"line:%d",__LINE__);\
perror(msg);\
}while(0)
int main(int argc, const char *argv[])
{
// 创建套接字
int cfd=socket(AF_INET,SOCK_DGRAM,0);
if(cfd<0){
ERR_MSG("socket");
return -1;
}
printf("套接字创建成功\n");
// 绑定发送端自身的IP和端口
// 填充发送端的IP和端口
struct sockaddr_in cin;
cin.sin_family=AF_INET;
cin.sin_port=htons(7777);
cin.sin_addr.s_addr=inet_addr("192.168.8.102");
if(bind(cfd,(struct sockaddr*)&cin,sizeof(cin))<0){
ERR_MSG("bind");
return -1;
}
printf("绑定成功\n");
// 设置允许广播
int set_broadcast=1;
if(setsockopt(cfd,SOL_SOCKET,SO_BROADCAST,&set_broadcast,sizeof(set_broadcast))<0){
ERR_MSG("set_broadcast");
return -1;
}
// 填充接收端的IP和端口
struct sockaddr_in sin;
sin.sin_family=AF_INET;
sin.sin_port=htons(6666);
sin.sin_addr.s_addr=inet_addr("192.168.8.255");
char buf[128]="";
ssize_t res=0;
// 接收数据
while(1){
bzero(buf,sizeof(buf));
printf("请输入>>>");
fgets(buf,sizeof(buf),stdin);
buf[strlen(buf)-1]=0;
if(sendto(cfd,buf,sizeof(buf),0,(struct sockaddr*)&sin,sizeof(sin))<0){
ERR_MSG("sendto");
return -1;
}
printf("发送成功\n");
}
// 关闭文件描述符号
close(cfd);
return 0;
}
3.
#include<stdio.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<netinet/ip.h>
#include<arpa/inet.h>
#include<string.h>
#include<unistd.h>
#define ERR_MSG(msg) do{\
fprintf(stderr,"line:%d",__LINE__);\
perror(msg);\
}while(0)
int main(int argc, const char *argv[])
{
// 创建套接字
int sfd=socket(AF_INET,SOCK_DGRAM,0);
if(sfd<0){
ERR_MSG("socket");
return -1;
}
printf("套接字创建成功\n");
// 填充接受端IP和端口
struct sockaddr_in sin;
sin.sin_family=AF_INET;
sin.sin_port=htons(6666);
sin.sin_addr.s_addr=inet_addr("192.168.8.255");
if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))<0){
ERR_MSG("bind");
return -1;
}
printf("绑定成功\n");
struct sockaddr_in cin; // 存储数据包是从哪里发送过来的
socklen_t addrlen = sizeof(cin);
char buf[128]="";
ssize_t res=0;
while(1){
bzero(buf,sizeof(buf));
res=recvfrom(sfd,buf,sizeof(buf),0,(struct sockaddr*)&cin,&addrlen);
if(res<0){
ERR_MSG("recvfrom");
return -1;
}
printf("[%s:%d]:%s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),buf);
}
// 关闭文件描述符
close(sfd);
return 0;
}