App下載

Java 中的一個糟糕的默認(rèn)超時(shí)設(shè)置 Infinity

宇宙一級潛在鴿王 2021-09-24 11:19:58 瀏覽數(shù) (2312)
反饋

許多庫包裝了一些外部通信。無論是類 REST 的 API、消息隊(duì)列、數(shù)據(jù)庫、郵件服務(wù)器還是其他東西。因此,您必須有一些超時(shí)時(shí)間——用于連接、讀取、寫入或空閑。遺憾的是,許多庫的默認(rèn)超時(shí)設(shè)置為“0”或“-1”,這意味著無窮大。

這是一個非常無用甚至有害的默認(rèn)設(shè)置。沒有一個實(shí)際用例讓您希望永遠(yuǎn)等待資源。并且有很多情況會發(fā)生這種情況,例如另一端卡住了。在過去的 3 個月里,我有 2 個庫的默認(rèn)超時(shí)為“無窮大”,最終導(dǎo)致生產(chǎn)問題,因?yàn)槲覀兺浟苏_配置它們。有時(shí)您甚至看不到問題,直到線程池耗盡。

永遠(yuǎn)不要將“Inifinty”作為默認(rèn)超時(shí)。因?yàn)?,您的庫會?dǎo)致許多生產(chǎn)問題。還要注意,有時(shí)底層的 HTTP 客戶端(或 Socket)沒有合理的默認(rèn)值——在包裝它時(shí)修復(fù)它仍然是你的工作。

您應(yīng)該提供什么默認(rèn)值?合理的。5秒可以嗎?您可能說您不想對用戶強(qiáng)加任意超時(shí)。在這種情況下,我有一個更好的建議:

明確要求構(gòu)建“客戶端”超時(shí)(因?yàn)檫@些庫通常是某些外部系統(tǒng)的客戶端)。例如?Client.create(url, credentials, timeout)?。如果沒有提供超時(shí),則失敗。這使得客戶端的用戶積極考慮他們的用例什么是好的超時(shí) - 不強(qiáng)加任何東西,最重要的是 - 不會冒生產(chǎn)中連接卡住的風(fēng)險(xiǎn)。此外,您仍然可以向他們提供“默認(rèn)”選項(xiàng),但仍讓他們明確選擇它。例如:

Client client = ClientBuilder.create(url)
                   .withCredentials(credentials)
                   .withTimeouts(Timeouts.connect(1000).read(1000))
                   .build();
// OR
Client client = ClientBuilder.create(url)
                   .withCredentials(credentials)
                   .withDefaultTimeouts()
                   .build();

上面的構(gòu)建器應(yīng)該要求設(shè)置“timeout”,如果兩個方法都沒有被調(diào)用,則應(yīng)該失敗。即使你不提供這些選項(xiàng),至少有一個指定超時(shí)的好方法——一些庫需要反射來設(shè)置其底層客戶端的超時(shí)。

我相信這是那些看起來很小但在現(xiàn)實(shí)世界中會導(dǎo)致很多問題的問題之一。它可以(并且應(yīng)該)由庫/客戶端設(shè)計(jì)者解決。

但由于情況并非總是如此,我們必須確保每次使用 3rd 方庫時(shí)都配置超時(shí)。


0 人點(diǎn)贊