基于IOCP的高速文件传输代码
//服务端:
const
//transmit用的参数
TF_USE_KERNEL_APC =
$20
;
//命令类型
CMD_CapScreen =
2000
;
CMD_CapVideo =
2001
;
CMD_CapAudio =
2002
;
CMD_GetSystemInfo =
2003
;
CMD_TransmitFiles =
2004
;
//通用数据传输包体封装
type
//每个完整数据的头描述
TPacketHeader =
packed
record
PacketCMD :
Word
;
//包类型
DataLength :
Word
;
//包体长度
IsCompressed :
Boolean
;
//包体是否为压缩数据
end
;
TBytes =
array
[
0..65535
]
of
Byte
;
TPacketBody =
packed
record
Data : TBytes;
end
;
//完整的数据包
TPacketInfo =
packed
record
Header : TPacketHeader;
Body : TPacketBody;
end
;
//文件发送包
TFileSendPacket =
packed
record
FileName :
array
[
0..127
]
of
Char
;
FileSize :
LongWord
;
StartWritePositon :
LongWord
;
hFile : THandle;
end
;
function
TServerClientSocket
.
TransFile(FileName:
string
;StartWritePositon:
LongWord
):
Boolean
;
var
hFile : THandle;
NumberOfByteSend :
LongWord
;
Block:PBlock;
PacketInfo: TPacketInfo;
FileSendPacket : TFileSendPacket;
AFileName :
string
[
128
];
TransmitFileBuffers : TTransmitFileBuffers;
begin
if
not
FileExists(FileName)
then
begin
Result :=
False
;
Exit;
end
;
hFile := CreateFile(
PChar
(FileName), GENERIC_READ, FILE_SHARE_READ,
nil
, OPEN_EXISTING,
0
,
0
);
//如果文件打开错误,则退出
if
hFile = INVALID_HANDLE_VALUE
then
begin
Result :=
False
;
Closehandle(hFile);
Exit;
end
;
//得到需要传输的字节数
NumberOfByteSend := windows
.
GetFileSize(hFile,
nil
) - StartWritePositon;
if
NumberOfByteSend <=
0
then
begin
Closehandle(hFile);
Exit;
end
;
Block := AllocBlock;
Block
.
Data
.
Event := seFileSend;
Block
.
Data
.
Overlapped
.
Offset := StartWritePositon;
AFileName := ExtractFileName(FileName);
FillChar(PacketInfo,SizeOf(TPacketInfo),
0
);
FillChar(FileSendPacket,SizeOf(TFileSendPacket),
0
);
Move(AFileName[
1
],FileSendPacket
.
FileName,length(AFileName));
FileSendPacket
.
FileSize := NumberOfByteSend;
FileSendPacket
.
StartWritePositon := StartWritePositon;
FileSendPacket
.
hFile := hFile;
PacketInfo
.
Header
.
PacketCMD := CMD_TransmitFiles;
PacketInfo
.
Header
.
DataLength := Sizeof(TFileSendPacket);
PacketInfo
.
Header
.
IsCompressed :=
False
;
Move(FileSendPacket,PacketInfo
.
Body
.
Data,SizeOf(TFileSendPacket));
Move(PacketInfo,Block^.Data
.
Buffer,SizeOf(TPacketHeader) + SizeOf(TFileSendPacket));
//传输文件前发送的包
TransmitFileBuffers
.
Head := @Block^.Data
.
Buffer[
0
];
TransmitFileBuffers
.
HeadLength := SizeOf(TPacketHeader) + SizeOf(TFileSendPacket);
//传输文件完毕后发送的包
TransmitFileBuffers
.
Tail :=
nil
;
TransmitFileBuffers
.
TailLength :=
0
;
LogMsg(&#
39
;开始发送文件:&#
39
; + FileName + &#
39
; Size=&#
39
; + IntToStr(NumberOfByteSend));
//发送命令,并将文件名、继传点、需要传输的大小传递过去
if
not
TransmitFile(SocketHandle, hFile, NumberOfByteSend, MAX_BUFSIZE,
@Block^.Data
.
Overlapped, @TransmitFileBuffers, TF_USE_KERNEL_APC)
then
begin
if
GetLastError <> ERROR_IO_PENDING
then
begin
Result :=
False
;
Exit;
end
;
end
;
Result :=
True
;
end
;
//如果发送完毕,可以接收到重叠IO的返回结果
case
Block^.Data
.
Event
of
seFileSend:
begin
Block
.
IsUse :=
False
;
Move(Block
.
Data
.
Buffer,PacketInfo,SizeOf(TPacketHeader) + SizeOf(TFileSendPacket));
if
PacketInfo
.
Header
.
PacketCMD = CMD_TransmitFiles
then
begin
FillChar(FileSendPacket,SizeOf(TFileSendPacket),
0
);
Move(PacketInfo
.
body
.
data,FileSendPacket,SizeOf(TFileSendPacket));
Closehandle(FileSendPacket
.
hFile);
//发送完毕,关闭文件句柄
end
;
LogMsg(&#
39
;文件:&#
39
; + StrPas(FileSendPacket
.
FileName) + &#
39
; 发送完毕!&#
39
;);
if
not
PrepareRecv()
then
Result := RESPONSE_FAIL;
end
;
seRead: 。。。。。。。
//客户端:
procedure
TrecvThread
.
Execute;
var
PacketInfo : TPacketInfo;
str:
string
;
FileSendPacket:TFileSendPacket;
FileStream:TFileStream;
FileName :
string
;
RecBuf:
array
[
0..1023
]
of
Char
;
RemainByts,RecvedBytes:
Integer
;
begin
while
(
not
self
.
Terminated )
DO
begin
cs
.
CheckForDisconnect(
False
);
if
cs
.
ClosedGracefully
then
begin
Fmm
.
Lines
.
Add(&#
39
;链路断开!&#
39
;);
self
.
Terminate;
end
;
cs
.
ReadBuffer(PacketInfo
.
Header,SizeOf(TPacketHeader));
cs
.
ReadBuffer(PacketInfo
.
Body
.
Data, PacketInfo
.
Header
.
DataLength);
if
PacketInfo
.
Header
.
PacketCMD = CMD_TransmitFiles
then
begin
Move(PacketInfo
.
Body
.
Data,FileSendPacket,PacketInfo
.
Header
.
DataLength);
FileName := StrPas(FileSendPacket
.
FileName);
try
FileStream := TFileStream
.
Create(&#
39
;C:\&#
39
;+ FileName, fmCreate
or
fmOpenWrite);
Fmm
.
Lines
.
Add(&#
39
;接收:&#
39
; + FileName + &#
39
; Size=&#
39
; + IntToStr(FileSendPacket
.
FileSize));
RecvedBytes :=
0
;
while
(RecvedBytes < FileSendPacket
.
FileSize)
do
begin
if
FileSendPacket
.
FileSize <=
1024
then
begin
cs
.
ReadBuffer(RecBuf,FileSendPacket
.
FileSize);
RecvedBytes := FileSendPacket
.
FileSize;
FileStream
.
WriteBuffer(RecBuf,RecvedBytes);
Break;
end
else
begin
cs
.
ReadBuffer(RecBuf,
1024
);
RecvedBytes := RecvedBytes +
1024
;
FileStream
.
WriteBuffer(RecBuf,
1024
);
RemainByts := FileSendPacket
.
FileSize - RecvedBytes;
if
RemainByts <=
1024
then
begin
cs
.
ReadBuffer(RecBuf,RemainByts);
RecvedBytes := RecvedBytes + RemainByts;
FileStream
.
WriteBuffer(RecBuf,RemainByts);
Break;
end
;
end
;
end
;
finally
FileStream
.
Free;
end
;
end
;
end
;
end
;
http://www.delphi6.com/thread-554.htm
基于IOCP的高速文件传输代码的更多相关文章
- Tftp文件传输服务器(基于UDP协议)
一个简单的UDP服务端与客户端 服务端: from socket import * #创建套接字 udp_server = socket(AF_INET,SOCK_DGRAM) msg_server ...
- WCF大文件传输服务
由于项目需要,自己写一个基于WCF的大文件传输服务雏形.觉得有一定的参考价值,因此放在网上分享. 目前版本为v1.1特点如下: 1.文件传输端口为18650 2.上传和下载文件 3.支持获取文件传输状 ...
- vsftpd-基于ftp协议的文件传输服务器软件
第一部分:在Linux上部署vsftpd服务 1. vsftpd简介 1.1 vsftpd是什么? ftp(File Transfer Protocol)文件传输协议.(实现不同操作系统之间文件的传输 ...
- QT从入门到入土(九)——TCP/IP网络通信(以及文件传输)
引言 TCP/IP通信(即SOCKET通信)是通过网线将服务器Server端和客户机Client端进行连接,在遵循ISO/OSI模型的四层层级构架的基础上通过TCP/IP协议建立的通讯.控制器可以设置 ...
- Python之路(第三十二篇) 网络编程:udp套接字、简单文件传输
一.UDP套接字 服务端 # udp是无链接的,先启动哪一端都不会报错 # udp没有链接,与tcp相比没有链接循环,只有通讯循环 server = socket.socket(socket.AF_I ...
- Python实现终端FTP文件传输
实现终端FTP文件传输 代码结构: .├── client.py├── readme.txt└── server.py 运行截图: readme.txt tftp文件服务器 项目功能: * 客户端有简 ...
- Python自带HTTP文件传输服务
一行命令搭建一个基于python的http文件传输服务 由于今天朋友想要一个文件,而我恰好有,因为这个文件比较大,网速不是很给力,所以想到了python自己有这么一个功能,这样不仅不需要下载其他软件, ...
- Java基于Socket文件传输示例(转)
最近需要进行网络传输大文件,于是对基于socket的文件传输作了一个初步的了解.在一位网友提供的程序基础上,俺进行了一些加工,采用了缓冲输入/输出流来包装输出流,再采用数据输入/输出输出流进行包装,加 ...
- Java基于Socket文件传输示例
http://www.blogjava.net/sterning/archive/2007/10/13/152508.html 最近需要进行网络传输大文件,于是对基于socket的文件传输作了一个初步 ...
随机推荐
- C#生成、解析xml文件以及处理报错原因
转载自:http://blog.csdn.net/lilinoscar/article/details/21027319 简单的介绍一下生成XML文件以及解析,因为有些数据不一定放到数据库,减少链接数 ...
- wait()、notify()、notifyAll()与线程通信方式总结
1.通过wait().notify().notifyAll()进行线程通信 线程通信的目标是使线程间能够互相发送信号.另一方面,线程通信使线程能够等待其他线程的信号.例如,线程B可以等待线程A的一个信 ...
- poj1007——DNA Sorting
Description One measure of ``unsortedness'' in a sequence is the number of pairs of entries that are ...
- 【t080】遗址
Time Limit: 1 second Memory Limit: 128 MB [问题描述] 很久很久以前有一座寺庙,从上往下看寺庙的形状正好是一个正方形,在4个角上竖立着圆柱搭建而成.现在圆柱都 ...
- [Angular] Difference between ngAfterViewInit and ngAfterContentInit
Content is what is passed as children. View is the template of the current component. The view is in ...
- Mochiweb的设计分析
http://blog.csdn.net/dp0304/article/details/6994435 Web服务器的基本工作大致分3步: 接收HTTP请求: 处理HTTP请求,生成响应内容: 发送响 ...
- Watchdog问题实例分析
1.日志获取 Watchdog相关的问题甚至需要以下所有的日志: logcat 通过adb logcat命令输出Android的一些当前运行日志,可以通过logcat的 -b 参数指定要输出的日志缓冲 ...
- Eclipse Che开发Spring Web应用(入门) (二)
在上篇博客中我们介绍了如何安装Eclipse Che这种浏览器SDK之后,收到了许多开发者的提问,为了方便初学者开发Java web应用,笔者又一步步实践了spring web开发(demo)过程,欢 ...
- 【序列操作IV】树状数组套线段树/树套树
题目描述 给出序列 a1,a2,…,an(0≤ai≤109),有关序列的两种操作. 1. ai(1≤i≤n)变成 x(0≤x≤109). 2. 求 al,al+1,…,ar(1≤l≤r≤n)第 k(1 ...
- scala 主从构造器
package cn.scala_base.oop.scalaclass /** * 构造器分为两种,一种是主构造器,另一种是从构造器,所有的从构造器必须在其方法体 * 的第一行调用主构造器 * * ...