Чтение онлайн

ЖАНРЫ

О чём не пишут в книгах по Delphi

Григорьев Антон Борисович

Шрифт:

ServerForm.RemoveConnection(Connection);

Exit;

 end;

 // Уменьшаем число оставшихся к чтению байтов

 // на размер полученных данных

 Dec(Connection.BytesLeft, cdTransferred);

 if Connection.BytesLeft < 0 then

// Страховка от "тупой" ошибки

ServerForm.AddMessageToLog('Клиент ' + Connection.ClientAddr +

' -
внутренняя ошибка программы: получено больше байтов, ' +

'чем ожидалось');

ServerForm.RemoveConnection(Connection);

 end

 else if Connection.BytesLeft = 0 then

 begin

// Длина строки прочитана целиком

if Connection.MsgSize <= 0 then

begin

ServerForm.AddMessageToLog('Клиент ' + Connection.ClientAddr +

' — получена неверная длина строки ' +

IntToStr(Conneсtion.MsgSizе));

ServerForm.RemoveConnection(Connection);

Exit;

end;

// Делаем строку нужной длины

SetLength(Connection.Msg, Connection.MsgSize);

// Данные пока не прочитаны, поэтому смещение - ноль,

// осталось прочитать полную длину.

Connection.Offset := 0;

Connection.BytesLeft := Connection.MsgSize;

// Заносим размер буфера и указатель на него в Buf.

// Данные будут складываться в строку,

// на которую ссылается Connection.Msg.

Buf.Len := Connection.MsgSize;

Buf.Buf := Pointer(Connection.Msg);

// Вызываем WSARecv для чтения самой строки

Flags := 0;

if WSARecv(Connect ion.ClientSocket, @Buf, 1, NumBytes, Flags,

@Connection.Overlapped, ReadMsgCompleted) = SOCKET_ERROR then

begin

if WSAGetLastError <> WSA_IO_PENDING then

begin

ServerForm.AddMessageToLog('Клиент ' + Connection.ClientAddr +

' - ошибка при чтении строки: ' + GetErrorString(dwError));

ServerForm.RemoveConnection(Connection);

end;

end;

 end

 else

 begin

// Connection.BytesLeft < 0 - длина строки

// прочитана не до конца.

// Увеличиваем смещение на число прочитанных байтов

Inc(Connection.Offset, cdTransferred);

// Формируем буфер для чтения оставшейся части длины

Buf.Len := Connection.BytesLeft;

Buf.Buf := PChar(@Connection.MsgSize) + Connection.Offset;

//
вызываем WSARecv для чтения оставшейся части длины строки

Flags := 0;

if WSARecv(Connection.ClientSocket, @Buf, 1, NumBytes, Flags,

@Connection.Overlapped, ReadMsgCompleted) = SOCKET_ERROR then

begin

if WSAGetLastError <> WSA_IO_PENDING then

begin

ServerForm.AddMessageToLog('Клиент ' + Connection.ClientAddr +

' - ошибка при чтении длины строки: ' +

GetErrorString(dwError));

ServerForm.RemoveConnection(Connection);

end;

end;

 end;

end;

// Функция ReadMsgCompleted используется в качестве функции завершения

// для перекрытого чтения строки.

// Она во многом аналогична функции ReadLenCompleted

procedure ReadMsgCompleted(dwError: DWORD; cdTransferred: DWORD; lpOverlapped: PWSAOverlapped; dwFlags: DWORD); stdcall;

var

 Connection: PConnection;

 Buf: TWSABuf;

 NumBytes, Flags: DWORD;

begin

 Connection := ServerForm.GetConnectionByOverlapped(lpOverlapped);

 if Connection = nil then

 begin

ServerForm.AddMessageToLog(

'Внутренняя ошибка программы - не найдено соединение');

Exit;

 end;

 if dwError <> 0 then

 begin

ServerForm.AddMessageToLog('Клиент ' + Connection.ClientAddr +

' ошибка при чтении строки: ' + GetErrorString(dwError));

ServerForm.RemoveConnection(Connection);

Exit;

 end;

 Dec(Connection.BytesLeft, cdTransferred);

 if Connection.BytesLeft < 0 then

 begin

ServerForm.AddMessageToLog('Клиент ' + Connection.ClientAddr +

' - внутренняя ошибка программы: получено больше байтов, ' +

'чем ожидалось');

ServerForm.RemoveConnection(Connection);

 end

 else if Connection.BytesLeft = 0 then

 begin

// Строка получена целиком. Выводим ее на экран.

Поделиться с друзьями: