了解及處理 SignalR 的連線存留期事件
警告
本檔不適用於最新版本的 SignalR。 看看 ASP.NET Core SignalR。
本文概述您可以處理的 SignalR 連線、重新連線和中斷連線事件,以及您可以設定的逾時和保留設定。
本文假設您已經瞭解 SignalR 和連線存留期事件。 如需 SignalR 簡介,請參閱 SignalR 簡介。 如需連線存留期事件的清單,請參閱下列資源:
本主題中使用的軟體版本
- Visual Studio 2017
- .NET 4.5
- SignalR 第 2 版
本主題的舊版
如需舊版 SignalR 的相關信息,請參閱 SignalR 舊版。
問題和批注
請留下您喜歡本教學課程的意見反應,以及我們可以在頁面底部的批注中改善的內容。 如果您有與教學課程沒有直接相關的問題,您可以將問題張貼至 ASP.NET SignalR 論壇 或 StackOverflow.com。
概觀
本文包含下列章節:
API 參考主題的連結是 API 的 .NET 4.5 版本。 如果您使用 .NET 4,請參閱 API 的 .NET 4 版本主題。
線上留期術語和案例
OnReconnected
SignalR Hub 中的事件處理程式可以直接在指定的客戶端之後OnConnected
執行,而不是在 之後OnDisconnected
執行。 您可以在沒有中斷連線的情況下進行重新連線的原因是有數種方式可在 SignalR 中使用「連線」一詞。
SignalR 連線、傳輸連線和實體連線
本文將區分 SignalR 連線、 傳輸連線和 實體連線:
- SignalR 連線 是指客戶端與伺服器 URL 之間的邏輯關聯性,由 SignalR API 維護,並以連線標識碼唯一識別。 SignalR 會維護此關聯性的相關數據,並用來建立傳輸連線。 當用戶端呼叫
Stop
方法或達到逾時限制時,關聯性會結束,且SignalR嘗試重新建立遺失的傳輸連線時,會處置數據。 - 傳輸連線 是指客戶端與伺服器之間的邏輯關聯性,由四個傳輸 API 之一所維護:WebSockets、伺服器傳送的事件、永遠框架或長時間輪詢。 SignalR 會使用傳輸 API 來建立傳輸連線,而傳輸 API 取決於實體網路連線的存在來建立傳輸連線。 當 SignalR 終止或傳輸 API 偵測到實體連線中斷時,傳輸連線就會結束。
- 實體連線 是指實體網路連結 -- 電線、無線訊號、路由器等 -- 有助於用戶端電腦與伺服器計算機之間的通訊。 實體連接必須存在才能建立傳輸連線,而且必須建立傳輸連線,才能建立 SignalR 連線。 不過,中斷實體連線並不總是會立即結束傳輸連線或 SignalR 連線,如本主題稍後所述。
在下圖中,SignalR 連線是由中樞 API 和 PersistentConnection API SignalR 層表示,傳輸連接是由傳輸層表示,而實體連線則以伺服器與客戶端之間的線條表示。
當您在 SignalR 用戶端中呼叫 Start
方法時,您會為 SignalR 用戶端程式代碼提供它所需的所有資訊,以建立與伺服器的實體連線。 SignalR 用戶端程式代碼會使用這項資訊來提出 HTTP 要求,並建立使用四種傳輸方法之一的實體連線。 如果傳輸連線失敗或伺服器失敗,則 SignalR 連線不會立即消失,因為用戶端仍有自動重新建立與相同 SignalR URL 的新傳輸連線所需的資訊。 在此案例中,不會涉及使用者應用程式的介入,而且當 SignalR 用戶端程式代碼建立新的傳輸連線時,它不會啟動新的 SignalR 連線。 SignalR 連線的持續性反映在您呼叫 Start
方法時所建立的連接標識碼不會變更。
當 OnReconnected
傳輸連線在遺失之後自動重新建立時,中樞上的事件處理程式就會執行。 OnDisconnected
事件處理程式會在 SignalR 連線的結尾執行。 SignalR 連線可透過下列任何方式結束:
- 如果用戶端呼叫
Stop
方法,則會將停止訊息傳送至伺服器,而且客戶端和伺服器都會立即結束 SignalR 連線。 - 用戶端與伺服器之間的連線中斷之後,客戶端會嘗試重新連線,而伺服器會等候用戶端重新連線。 如果嘗試重新連線失敗,且中斷連線逾時期間結束,用戶端和伺服器都會結束 SignalR 連線。 用戶端會停止嘗試重新連線,而伺服器會處置其 SignalR 連線的表示法。
- 如果用戶端停止執行,而沒有機會呼叫
Stop
方法,伺服器會等候用戶端重新連線,然後在中斷聯機逾時期間之後結束 SignalR 連線。 - 如果伺服器停止執行,用戶端會嘗試重新連線(重新建立傳輸連線),然後在中斷連線逾時期間之後結束 SignalR 連線。
當沒有任何連線問題,而且使用者應用程式會呼叫 Stop
方法來結束SignalR 連線時,SignalR 連線和傳輸聯機會同時開始和結束。 下列各節將詳細說明其他案例。
傳輸中斷連線案例
實體連線可能很慢,或連線中斷。 視中斷長度等因素而定,可能會卸除傳輸連線。 SignalR 接著會嘗試重新建立傳輸連線。 有時候傳輸連線 API 會偵測中斷並卸除傳輸連線,SignalR 會立即找出連線遺失。 在其他案例中,傳輸連線 API 和 SignalR 都不會立即察覺到連線已遺失。 針對長輪詢以外的所有傳輸,SignalR 用戶端會使用稱為 keepalive 的 函式來檢查傳輸 API 無法偵測到的連線遺失。 如需長輪詢連線的相關信息,請參閱 本主題稍後的逾時和保留設定 。
當連線處於非使用中狀態時,伺服器會定期將保留封包傳送至用戶端。 截至本文撰寫的日期,默認頻率為每 10 秒。 藉由接聽這些封包,用戶端可以判斷是否有連線問題。 如果預期未收到保留封包,用戶端會在短時間內假設有連線問題,例如緩慢或中斷。 如果 keepalive 在較長的時間之後仍未收到,用戶端會假設連線已卸除,並開始嘗試重新連線。
下圖說明當傳輸 API 無法立即辨識實體連線發生問題時,在一般案例中引發的客戶端和伺服器事件。 此圖表適用於下列情況:
- 傳輸是 WebSockets、永遠框架或伺服器傳送的事件。
- 實體網路連線有不同期間的中斷。
- 傳輸 API 不會察覺中斷,因此 SignalR 依賴保留功能來偵測它們。
如果客戶端進入重新連線模式,但在中斷連線逾時限制內無法建立傳輸連線,伺服器就會終止 SignalR 連線。 發生這種情況時,伺服器會執行中樞 OnDisconnected
的 方法,並將中斷連線訊息排入佇列,以在用戶端管理稍後連線時傳送至用戶端。 如果客戶端接著重新連線,則會接收 disconnect 命令並呼叫 Stop
方法。 在此案例中, OnReconnected
用戶端重新連線時不會執行,也不會 OnDisconnected
在用戶端呼叫 Stop
時執行。 下圖說明此案例。
用戶端上可能會引發的 SignalR 連線存留期事件如下:
ConnectionSlow
client 事件。自收到最後一則訊息或 Keepalive Ping 之後,保留逾時期間的預設比例時引發。 預設的 Keepalive 逾時警告期間為 2/3 的 Keepalive 逾時。 保留逾時為 20 秒,因此警告會在 13 秒左右發生。
根據預設,伺服器每隔 10 秒會傳送 keepalive Ping,而用戶端會每隔 2 秒檢查一次 keepalive ping (保留逾時值與 keepalive 逾時警告值之間的差異三分之一)。
如果傳輸 API 意識到中斷連線,SignalR 可能會在持續逾時警告期間通過之前收到中斷連線的通知。 在此情況下,
ConnectionSlow
不會引發事件,SignalR 會直接移至Reconnecting
事件。Reconnecting
client 事件。當 (a) 傳輸 API 偵測到連線遺失,或 (b) 自收到最後一則訊息或 Keepalive Ping 之後,已超過保留逾時期間時引發。 SignalR 用戶端程式代碼會開始嘗試重新連線。 如果您希望應用程式在傳輸連線遺失時採取一些動作,您可以處理此事件。 預設的保留逾時期間目前為20秒。
如果您的用戶端程式代碼嘗試在 SignalR 處於重新連線模式時呼叫 Hub 方法,SignalR 會嘗試傳送命令。 大部分情況下,這類嘗試將會失敗,但在某些情況下可能會成功。 對於伺服器傳送的事件,永遠框架和長輪詢傳輸,SignalR 會使用兩個通道,一個是用戶端用來傳送訊息,另一個是用來接收訊息。 用於接收的通道是永久開啟的通道,也就是實體連線中斷時關閉的通道。 用於傳送的通道仍可供使用,因此如果還原實體連線能力,在重新建立接收通道之前,用戶端到伺服器的方法呼叫可能會成功。 在 SignalR 重新開啟用於接收的通道之前,不會收到傳回值。
Reconnected
client 事件。重新建立傳輸連線時引發。 中
OnReconnected
樞中的事件處理程式會執行。Closed
client 事件 (disconnected
JavaScript 中的事件)。當 SignalR 用戶端程式代碼在失去傳輸連線之後嘗試重新連線時,中斷連線逾時期間到期時引發。 默認的中斷聯機逾時為30秒。 (當連接因為呼叫 方法而結束
Stop
時,也會引發這個事件。
傳輸 API 未偵測到的傳輸連線中斷,且不會將伺服器保留 Ping 的接收延遲時間超過 keepalive 逾時警告期間,可能不會引發任何連線存留期事件。
某些網路環境會刻意關閉閑置連線,而保留封包的另一個功能是讓這些網路知道 SignalR 連線正在使用中,以協助防止此情況。 在極端情況下,保留 Ping 的默認頻率可能不足以防止關閉的連線。 在此情況下,您可以將 Keepalive Ping 設定為更頻繁地傳送。 如需詳細資訊,請參閱 本主題稍後的逾時和保留設定 。
注意
重要事項:不保證這裡所述的事件順序。 SignalR 會根據此配置,嘗試以可預測的方式引發連線存留期事件,但網路事件有許多變化,以及許多基礎通訊架構,例如傳輸 API 處理它們的方式。 例如, Reconnected
當用戶端重新連線時,可能不會引發 事件,或 OnConnected
當嘗試建立連線失敗時,伺服器上的處理程式可能會執行。 本主題僅描述某些典型情況通常會產生的效果。
用戶端中斷連線案例
在瀏覽器用戶端中,維護 SignalR 連線的 SignalR 用戶端程式代碼會在網頁的 JavaScript 內容中執行。 這就是為什麼當您從某個頁面巡覽到另一個頁面時,SignalR 連線必須結束,因此,如果您從多個瀏覽器視窗或索引卷標連線,則有多個連線標識碼。 當使用者關閉瀏覽器視窗或索引標籤,或巡覽至新的頁面或重新整理頁面時,SignalR 聯機會立即結束,因為 SignalR 用戶端程式代碼會為您處理該瀏覽器事件並呼叫 Stop
方法。 在這些案例中,或在應用程式呼叫 Stop
方法時的任何用戶端平臺中, OnDisconnected
事件處理程式會在伺服器上立即執行,而用戶端會 Closed
引發事件(事件在 JavaScript 中命名 disconnected
)。
如果用戶端應用程式或正在當機或進入睡眠狀態的電腦(例如,當使用者關閉膝上型計算機時),則伺服器不會得知所發生的情況。 就伺服器所知,用戶端遺失可能是因為連線中斷而用戶端可能嘗試重新連線。 因此,在這些情況下,伺服器會等候讓客戶端有機會重新連線,而且 OnDisconnected
在中斷連線逾時期限到期之前不會執行(預設為約 30 秒)。 下圖說明此案例。
伺服器中斷連線案例
當伺服器離線時,它重新啟動、失敗、應用程式網域回收等-- 結果可能類似於遺失的連線,或傳輸 API 和 SignalR 可能立即知道伺服器已消失,而 SignalR 可能會開始嘗試重新連線,而不會引發 ConnectionSlow
事件。 如果客戶端進入重新連線模式,而且如果伺服器復原或重新啟動或新伺服器在中斷連線逾時期限到期之前上線,用戶端將會重新連線到已還原或新的伺服器。 在此情況下,SignalR 聯機會繼續在用戶端上,並 Reconnected
引發 事件。 在第一部伺服器上, OnDisconnected
永遠不會執行,而且在新伺服器上, OnReconnected
雖然從未針對該伺服器上的該客戶端執行過, 但 OnConnected
會執行。 (如果用戶端在重新啟動或應用程式網域回收后重新連線到同一部伺服器,則效果相同,因為當伺服器重新啟動時,它沒有先前連線活動的記憶體。下圖假設傳輸 API 會立即察覺遺失的連線,因此 ConnectionSlow
不會引發 事件。
逾時和保留設定
預設 ConnectionTimeout
、 DisconnectTimeout
和 KeepAlive
值適用於大部分案例,但如果您的環境有特殊需求,則可以變更。 例如,如果您的網路環境關閉閑置 5 秒的連線,您可能必須降低保留值。
ConnectionTimeout
此設定代表在關閉傳輸連接並開啟新連線之前等待回應的時間量。 預設值為 110 秒。
此設定僅適用於停用 Keepalive 功能時,通常只適用於長時間輪詢傳輸。 下圖說明此設定對長時間輪詢傳輸連線的影響。
DisconnectTimeout
此設定代表在引發 Disconnected
事件之前遺失傳輸連線之後等候的時間量。 預設值為 30 秒。 當您設定 DisconnectTimeout
時, KeepAlive
會自動設定為 值的 1/3 DisconnectTimeout
。
KeepAlive
此設定代表在透過閑置連線傳送保留封包之前要等候的時間量。 預設值為 10 秒。 這個值不能超過值 1/3 DisconnectTimeout
。
如果您要同時設定 DisconnectTimeout
和 KeepAlive
,請在 之後DisconnectTimeout
設定 KeepAlive
。 否則,當自動設定KeepAlive
為逾時值的 1/3 時DisconnectTimeout
,將會覆寫您的KeepAlive
設定。
如果您想要停用 keepalive 功能,請將 設定 KeepAlive
為 null。 長時間輪詢傳輸會自動停用 Keepalive 功能。
如何變更逾時和保留設定
若要變更這些設定的預設值,請在 Global.asax 檔案中Application_Start
設定預設值,如下列範例所示。 範例程式代碼中顯示的值與預設值相同。
protected void Application_Start(object sender, EventArgs e)
{
// Make long polling connections wait a maximum of 110 seconds for a
// response. When that time expires, trigger a timeout command and
// make the client reconnect.
GlobalHost.Configuration.ConnectionTimeout = TimeSpan.FromSeconds(110);
// Wait a maximum of 30 seconds after a transport connection is lost
// before raising the Disconnected event to terminate the SignalR connection.
GlobalHost.Configuration.DisconnectTimeout = TimeSpan.FromSeconds(30);
// For transports other than long polling, send a keepalive packet every
// 10 seconds.
// This value must be no more than 1/3 of the DisconnectTimeout value.
GlobalHost.Configuration.KeepAlive = TimeSpan.FromSeconds(10);
RouteTable.Routes.MapHubs();
}
如何通知用戶中斷連線
在某些應用程式中,您可能會想要在發生連線問題時向使用者顯示訊息。 您有數個選項可用來執行這項操作的方式和時機。 下列程式代碼範例適用於使用產生的 Proxy 的 JavaScript 用戶端。
處理事件,
connectionSlow
在 SignalR 知道連線問題後立即顯示訊息,再進入重新連線模式。$.connection.hub.connectionSlow(function() { notifyUserOfConnectionProblem(); // Your function to notify user. });
處理事件,
reconnecting
以在 SignalR 知道中斷連線並進入重新連線模式時顯示訊息。$.connection.hub.reconnecting(function() { notifyUserOfTryingToReconnect(); // Your function to notify user. });
處理事件,
disconnected
以在嘗試重新連線時顯示訊息已逾時。在此案例中,重新建立與伺服器的連線的唯一方式,是呼叫Start
方法重新啟動 SignalR 連線,這會建立新的連線標識碼。 下列程式代碼範例會使用 旗標,以確保只有在重新連線逾時之後才會發出通知,而不是在呼叫Stop
方法所造成之 SignalR 連線的一般端之後發出通知。var tryingToReconnect = false; $.connection.hub.reconnecting(function() { tryingToReconnect = true; }); $.connection.hub.reconnected(function() { tryingToReconnect = false; }); $.connection.hub.disconnected(function() { if(tryingToReconnect) { notifyUserOfDisconnect(); // Your function to notify user. } });
如何持續重新連線
在某些應用程式中,您可能會想要在連線遺失且嘗試重新連線已逾時之後自動重新建立連線。若要這樣做,您可以從事件處理程式呼叫 Start
方法 Closed
(disconnected
JavaScript 用戶端上的事件處理程式)。 您可能想要等候一段時間后再呼叫 Start
,以避免在伺服器或實體連線無法使用時太頻繁地執行此動作。 下列程式代碼範例適用於使用產生的 Proxy 的 JavaScript 用戶端。
$.connection.hub.disconnected(function() {
setTimeout(function() {
$.connection.hub.start();
}, 5000); // Restart connection after 5 seconds.
});
在行動用戶端中要注意的潛在問題是,當伺服器或實體連線無法使用時,連續重新連線嘗試可能會導致不必要的電池耗盡。
如何在伺服器程式代碼中中斷用戶端連線
SignalR 第 2 版沒有用於中斷用戶端連線的內建伺服器 API。 未來有一個 計劃新增這項功能。 在目前的 SignalR 版本中,將用戶端與伺服器中斷連線的最簡單方式是在用戶端上實作 disconnect 方法,並從伺服器呼叫該方法。 下列程式代碼範例示範使用產生的 Proxy 之 JavaScript 用戶端的中斷聯機方法。
var myHubProxy = $.connection.myHub
myHubProxy.client.stopClient = function() {
$.connection.hub.stop();
};
警告
安全性 - 無法中斷用戶端連線的方法,或建議的內建 API 都無法解決執行惡意代碼的駭客用戶端案例,因為用戶端可以重新連線或遭入侵的程式碼可能會移除 stopClient
方法或變更其用途。 實作具狀態阻斷服務 (DOS) 保護的適當位置不在架構或伺服器層中,而是在前端基礎結構中。
偵測中斷連線的原因
SignalR 2.1 會將多載新增至伺服器 OnDisconnect
事件,指出用戶端是否刻意中斷連線,而不是逾時。如果客戶端明確關閉連接,則 StopCalled
參數為 true。 在 JavaScript 中,如果伺服器錯誤導致用戶端中斷連線,錯誤資訊將會以的形式 $.connection.hub.lastError
傳遞至用戶端。
C# 伺服器程式代碼: stopCalled
參數
public override System.Threading.Tasks.Task OnDisconnected(bool stopCalled)
{
if (stopCalled)
{
Console.WriteLine(String.Format("Client {0} explicitly closed the connection.", Context.ConnectionId));
}
else
{
Console.WriteLine(String.Format("Client {0} timed out .", Context.ConnectionId));
}
return base.OnDisconnected(stopCalled);
}
JavaScript 用戶端程式代碼:在 lastError
事件中 disconnect
存取。
$.connection.hub.disconnected(function () {
if ($.connection.hub.lastError)
{ alert("Disconnected. Reason: " + $.connection.hub.lastError.message); }
});