Хтось згадав про можливість KeepAlive TCP Socket. Тут це красиво описано:
http://tldp.org/HOWTO/TCP-Keepalive-HOWTO/overview.html
Я використовую його таким чином: після підключення сокета я викликаю цю функцію, яка встановлює keepAlive. keepAliveTime
Параметр визначає час очікування в мілісекундах, при відсутності активності до першого підтримки активності пакета не передаються. keepAliveInterval
Параметр визначає інтервал, в мілісекундах, між тим, коли послідовні Keep-Alive пакети надсилаються , якщо підтвердження не отримано.
void SetKeepAlive(bool on, uint keepAliveTime, uint keepAliveInterval)
{
int size = Marshal.SizeOf(new uint());
var inOptionValues = new byte[size * 3];
BitConverter.GetBytes((uint)(on ? 1 : 0)).CopyTo(inOptionValues, 0);
BitConverter.GetBytes((uint)keepAliveTime).CopyTo(inOptionValues, size);
BitConverter.GetBytes((uint)keepAliveInterval).CopyTo(inOptionValues, size * 2);
socket.IOControl(IOControlCode.KeepAliveValues, inOptionValues, null);
}
Я також використовую асинхронне читання:
socket.BeginReceive(packet.dataBuffer, 0, 128,
SocketFlags.None, new AsyncCallback(OnDataReceived), packet);
А при зворотному виклику тут виявляється тайм-аут SocketException
, який збільшується, коли сокет не отримує сигнал ACK після збереження пакета.
public void OnDataReceived(IAsyncResult asyn)
{
try
{
SocketPacket theSockId = (SocketPacket)asyn.AsyncState;
int iRx = socket.EndReceive(asyn);
}
catch (SocketException ex)
{
SocketExceptionCaught(ex);
}
}
Таким чином, я можу безпечно виявити відключення між клієнтом TCP та сервером.