转自:http://blog.csdn.net/pingd/article/details/16341467
SERVER端
- #include <iostream>
- #include "udt.h"
- #include <io.h>
- #pragma comment(lib,"ws2_32.lib")
- using namespace std;
- #define MAXLEN 4096
- int main(int argc,char *argv[])
- {
- if ((argc != 4))
- {
- cout<<"Use: appserver.exe server_port client_ip client_port"<<endl;
- return 0;
- }
- //startup
- //这里是对UDT的启动记性初始化操作
- if (UDT::ERROR == UDT::startup())
- {
- cout<<"startup: "<<UDT::getlasterror().getErrorMessage()<<endl;
- }else{
- cout<<"startup suc..."<<endl;
- }
- //socket
- //像声明一个普通的socket一样声明一个UDTSOCKET
- UDTSOCKET serv = UDT::socket(AF_INET, SOCK_DGRAM, 0);
- if (UDT::ERROR == serv)
- {
- cout<<"socket: "<<UDT::getlasterror().getErrorMessage()<<endl;
- }else{
- cout<<"client suc..."<<endl;
- }
- //声明udp socket,这里是udp的哈,不是udt
- int sersocket = socket(AF_INET,SOCK_DGRAM,0);
- if (SOCKET_ERROR == sersocket)
- {
- cout<<"udp socket error!"<<endl;
- }else{
- cout<<"clientsocket suc..."<<endl;
- }
- //为了能够在局域网中直接进行处理,先默认设置两个
- sockaddr_in my_addr,client_addr;
- my_addr.sin_family = AF_INET;
- my_addr.sin_port = htons(atoi(argv[1]));
- my_addr.sin_addr.s_addr = INADDR_ANY;
- memset(&(my_addr.sin_zero), '\0', 8);
- bind(sersocket,(struct sockaddr*)&my_addr,sizeof(my_addr));
- client_addr.sin_family = AF_INET;
- client_addr.sin_port = htons(atoi(argv[3]));
- client_addr.sin_addr.s_addr = inet_addr(argv[2]);
- //client_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
- memset(&(client_addr.sin_zero), '\0', 8);
- int mss = 1052;//最大传输单位
- //设置收发缓冲区大小 接收限时 和地址重用
- if( !( UDT::ERROR != (UDT::setsockopt(serv, 0, UDT_SNDBUF, new int(32000), sizeof(int)))
- && UDT::ERROR != (UDT::setsockopt(serv, 0, UDP_RCVBUF, new int(32000), sizeof(int)))
- && UDT::ERROR != (UDT::setsockopt(serv,0,UDT_REUSEADDR,new int(1),sizeof(int)))
- && UDT::ERROR != (UDT::setsockopt(serv, 0, UDT_RENDEZVOUS, new bool(true), sizeof(bool))))
- && UDT::ERROR != (UDT::setsockopt(serv, 0, UDT_MSS, &mss, sizeof(int)) ))
- {
- cout<<"udt socket: "<<UDT::getlasterror().getErrorMessage()<<endl;
- UDT::close(serv);
- return 0;
- }
- //这里是直接将udp的接口绑定在udt的接口之上,如果不这样做的话是没法使用UDT中的SOCK_DGRAM的
- if (UDT::ERROR == UDT::bind2(serv,sersocket))
- {
- cout<<"udt bind2:"<<UDT::getlasterror().getErrorMessage()<<endl;
- return 0;
- }else{
- cout<<"bind2 suc"<<endl;
- }
- //这里也是关键部分,与client端对应的connect操作,就是相互之间的打洞处理
- if (UDT::ERROR == UDT::connect(serv, (sockaddr*)&client_addr, sizeof(client_addr)))
- {
- cout << "connect: " << UDT::getlasterror().getErrorMessage();
- UDT::close(serv);
- return 0;
- }else{
- cout<<"connetc suc"<<endl;
- }
- //这里已经可以正常接收了,接收从client发过来的filename,目的是用于本地的文件创建
- char filename[100];
- if (UDT::ERROR == UDT::recvmsg(serv, filename, 100))
- {
- cout << "recv:" << UDT::getlasterror().getErrorMessage() << endl;
- return 0;
- }
- cout <<"filename: "<< filename <<endl;
- //使用FILE进行文件操作,关于文件的相关操作这里不详述了,实在不懂的可以留言
- FILE *fp;
- char localfile[100];
- memset(localfile,0,sizeof(localfile));
- strcpy(localfile,"d:\\");
- strcat(localfile,filename);
- if((fp = fopen(localfile,"w+"))==NULL)
- {
- cout<<filename<<" open failure!"<<endl;
- return 0;
- }
- fclose(fp);
- fp = fopen(localfile,"wb");
- char data[MAXLEN];
- int len=0,package=0,filelen=0;
- UDT::TRACEINFO trace;
- UDT::perfmon(serv,&trace);
- while(1)
- {
- //前面部分打开文件后,这里就是循环接收文件并保存
- memset(data,0,sizeof(data));
- len = UDT::recvmsg(serv, data, MAXLEN);
- filelen += len;
- //cout<<"filelen = "<<filelen<<endl;
- if (strncmp("quit",data,4)==0)
- {
- cout<<data<<endl;
- fclose(fp);
- break;
- }else
- {
- package ++;//record recv all the packages
- }
- fwrite(data,len,1,fp);
- }
- fclose(fp);
- fp = fopen(localfile,"rb");
- fseek(fp,0,SEEK_END);//move to the end
- filelen = ftell(fp)
- fseek(fp,0,SEEK_SET);
- cout<<"filesize = "<<filelen<<endl;
- UDT::perfmon(serv,&trace);
- cout << "speed = " << trace.mbpsRecvRate << "Mbits/sec" << endl;
- cout<<"recv all the packages: "<<package<<endl;
- fclose(fp);
- UDT::close(serv);
- UDT::cleanup();
- return 1;
- }
上面一篇文章中提出了服务端的,其实这里没有严格意义的服务端和客户端之分,因为我在代码中是基于UDP的,不存在服务端与客户端,两个都是对等的,只是我这里进行一下简单的区分而已。在这里,客户端所进行的主要操作就是首先给服务端发送文件名,然后将文件直接通过fopen(),fread()等一系列的操作将文件发送过去。
与客户端相对应的源码在另一篇博文中:http://blog.csdn.net/pingd/article/details/16341467
CLIENT端
- #include <iostream>
- #include "udt.h"
- #include <io.h>
- #pragma comment(lib,"ws2_32.lib")
- using namespace std;
- #define MAXLEN 4096
- int main(int argc,char *argv[])
- {
- if ((argc != 5))
- {
- cout<<"Usage: appclient.exe client_port server_ip server_port local_filename"<<endl;
- return 0;
- }
- //startup
- if (UDT::ERROR == UDT::startup())
- {
- cout<<"startup: "<<UDT::getlasterror().getErrorMessage()<<endl;
- }else{
- cout<<"startup suc..."<<endl;
- }
- //Initialize the UDT library
- UDTSOCKET client = UDT::socket(AF_INET, SOCK_DGRAM, 0);
- if (UDT::ERROR == client)
- {
- cout<<"socket: "<<UDT::getlasterror().getErrorMessage()<<endl;
- }else{
- cout<<"client suc..."<<endl;
- }
- //声明udp socket
- int clientsocket = socket(AF_INET,SOCK_DGRAM,0);
- if (SOCKET_ERROR == clientsocket)
- {
- cout<<"udp socket error!"<<endl;
- }else{
- cout<<"clientsocket suc..."<<endl;
- }
- sockaddr_in serv_addr,my_addr;
- serv_addr.sin_family = AF_INET;
- serv_addr.sin_port = htons(atoi(argv[3]));
- serv_addr.sin_addr.s_addr = inet_addr(argv[2]);
- memset(&(serv_addr.sin_zero), '\0', 8);
- my_addr.sin_family = AF_INET;
- my_addr.sin_port = htons(atoi(argv[1]));
- my_addr.sin_addr.s_addr = INADDR_ANY;
- memset(&(my_addr.sin_zero), '\0', 8);
- bind(clientsocket,(struct sockaddr*)&my_addr,sizeof(my_addr));
- int mss = 1052;//最大传输单位
- //设置收发缓冲区大小 接收限时 和地址重用
- if( !( UDT::ERROR != (UDT::setsockopt(client, 0, UDT_SNDBUF, new int(32000), sizeof(int)))
- && UDT::ERROR != (UDT::setsockopt(client, 0, UDP_RCVBUF, new int(32000), sizeof(int)))
- && UDT::ERROR != (UDT::setsockopt(client,0,UDT_REUSEADDR,new int(1),sizeof(int)))
- && UDT::ERROR != (UDT::setsockopt(client, 0, UDT_RENDEZVOUS, new bool(true), sizeof(bool))))
- && UDT::ERROR != (UDT::setsockopt(client, 0, UDT_MSS, &mss, sizeof(int))))
- {
- cout<<"udt socket: "<<UDT::getlasterror().getErrorMessage()<<endl;
- UDT::close(client);
- return 0;
- }
- if (UDT::ERROR == UDT::bind2(client,clientsocket))
- {
- cout<<"udt bind2:"<<UDT::getlasterror().getErrorMessage()<<endl;
- return 0;
- }else{
- cout<<"bind2 suc"<<endl;
- }
- // connect to the server, implict bind
- if (UDT::ERROR == UDT::connect(client, (sockaddr*)&serv_addr, sizeof(serv_addr)))
- {
- cout << "connect: " << UDT::getlasterror().getErrorMessage();
- UDT::close(client);
- return 0;
- }else{
- cout<<"connect suc"<<endl;
- }
- char* hello = argv[4];
- if (UDT::ERROR == UDT::sendmsg(client, hello, strlen(hello) + 1,-1,true))
- {
- cout << "send: " << UDT::getlasterror().getErrorMessage();
- return 0;
- }
- FILE *fp;
- fp = fopen(hello,"rb");
- fseek(fp,0,SEEK_END);
- int filesize = ftell(fp);
- //rewind(fp);//移动到头部
- fseek(fp,0,SEEK_SET);
- static int filepos = 0;//记录文件偏移量
- cout<<"filesize = "<<filesize<<endl;
- char data[MAXLEN+1];
- int len=0,package=0;
- UDT::TRACEINFO trace;
- UDT::perfmon(client,&trace);
- while(1)
- {
- memset(data,0,sizeof(data));
- fread(data,MAXLEN,1,fp);
- if(filesize>=MAXLEN)
- {
- len = UDT::sendmsg(client,data,MAXLEN,-1,true);
- if (len<0)
- {
- cout<<"send failure!!\n"<<endl;
- break;
- }
- filesize -= MAXLEN;
- package ++;//record send all the packages
- }else
- {
- len = UDT::sendmsg(client,data,filesize,-1,true);
- if (len<0)
- {
- cout<<"send failure!!\n"<<endl;
- break;
- }
- package ++;//record recv all the packages
- char *quit = "quit";
- if (UDT::ERROR == UDT::sendmsg(client,quit,strlen(quit)+1,-1,true))
- {
- cout<<"send quit error"<<endl;
- fclose(fp);
- break;
- }
- break;
- }
- }
- UDT::perfmon(client,&trace);
- cout << "speed = " << trace.mbpsSendRate << "Mbits/sec" << endl;
- cout<<"send all the packages: "<<package<<endl;
- fclose(fp);
- UDT::close(client);
- UDT::close(clientsocket);
- UDT::cleanup();
- return 1;
- }