這下對阿裡java這幾條規範有更深理解了

来源:https://www.cnblogs.com/jtea/archive/2023/12/04/17874300.html
-Advertisement-
Play Games

背景 阿裡java開發規範是阿裡巴巴總結多年來的最佳編程實踐,其中每一條規範都經過仔細打磨或踩坑而來,目的是為社區提供一份最佳編程規範,提升代碼質量,減少bug。 這基本也是java業界都認可的開發規範,我們團隊也是以此規範為基礎,在結合實際情況,補充完善。最近在團隊遇到的幾個問題,加深了我對這份開 ...


背景

阿裡java開發規範是阿裡巴巴總結多年來的最佳編程實踐,其中每一條規範都經過仔細打磨或踩坑而來,目的是為社區提供一份最佳編程規範,提升代碼質量,減少bug。
這基本也是java業界都認可的開發規範,我們團隊也是以此規範為基礎,在結合實際情況,補充完善。最近在團隊遇到的幾個問題,加深了我對這份開發規範中幾個點的理解,下麵就一一道來。

日誌規約

這條規範說明瞭,在異常發送記錄日誌時,要記錄案發現場信息和異常堆棧信息,不處理要往上throws,切勿吃掉異常。
堆棧信息比較好理解,就是把整個方法調用鏈列印出來,方便定位具體是哪個方法出錯。而案發現場信息我認為至少要能說明:“誰發生了什麼錯誤”。
例如,哪個uid下單報錯了,哪個訂單支付失敗了,原因是什麼。否則滿屏列印:“user error”,看到你都無從下手。

在我們這次出現的問題就是有一個feign,調用外部介面報錯了,降級列印了不規範日誌,導致排查問題花了很多時間。偽代碼如下:

	@Slf4j
	@Component
	class MyClientFallbackFactory implements FallbackFactory<MyClient> {		

		@Override
		public MyClient create(Throwable cause) {
			return new MyClient() {
				@Override
				public Result<DataInfoVo> findDataInfo(Long id) {
					log.error("findDataInfo error");
					return Result.error(SYS_ERROR);
				}
			};
		}
	}

發版後錯誤日誌開始告警,打開kibana看到了滿屏了:“findDataInfo error”,然後開始一頓盲查。
因為這個介面本次並沒有修改,所以猜測是目標服務出問題,上伺服器curl介面,發現調用是正常的。
接著猜測是不是熔斷器有問題,熔斷後沒有恢復,但重啟服務後,還是繼續報錯。開始各種排查,arthas跟蹤,最後實在沒辦法了,還是老老實實把異常列印出來,走發版流程。

log.error("{} findDataInfo error", id, cause);

有了異常堆棧信息就很清晰了,原來是返回參數反序列失敗了,介面提供方新增一個不相容的參數導致反序列失敗。(這點在下一個規範還會提到)
可見日誌列印不清晰給排查問題帶來多大的麻煩,記住:日誌一定要列印關鍵信息,異常要列印堆棧。

二方庫依賴

上面提到的返回參數反序列化失敗就是枚舉造成的,原因是這個介面返回新增一個枚舉值,這個枚舉值原本返回給前端使用的,沒想到還有其它服務也調用了它,最終在反序列化時就報錯了,找不到“xxx”枚舉值。
比如如下介面,你提交一個不認得的黑色BLACK,就會報反序列錯誤:

	enum Color {
		GREEN, RED
	}

	@Data
	class Test {
		private Color color;
	}

	@PostMapping(value = "/post/info")
	public void info(@NotNull Test test) {

	}

	curl --location 'localhost/post/info' \
	--header 'Content-Type: application/json' \
	--data '{
    	"testEnum": "BLACK"
	}'

關於這一點我們看下作者孤盡對它的闡述:

這就是我們出問題的場景,提供方新增了一個枚舉值,而使用方沒有升級,導致錯誤。可能有的同學說那通知使用方升級不就可以了?是的,但這出現了依賴問題,如果使用方有成百上千個,你會非常頭痛。

那又為什麼說不要使用枚舉作為返回值,而可以作為輸入參數呢?
我的理解是:作為枚舉的提供者,不得隨意新增/修改內容,或者說修改前要同步到所有枚舉使用者,讓大家知道,否則使用者就可能因為不認識這個枚舉而報錯,這是不可接受的。
但反過來,枚舉提供者是可以將它作為輸入參數的,如果調用者傳了一個不存在的值就會報錯,這是合理的,因為提供者並沒有說支持這個值,調用者正常就不應該傳遞這個值,所以這種報錯是合理的。

ORM映射

以下是規範里的說明:
1)增加查詢分析器解析成本。
2)增減欄位容易與 resultMap 配置不一致。
3)無用欄位增加網路消耗,尤其是 text 類型的欄位。

這都很好理解,就不過多說明。
在我們開發中,有的同學為了方便,還是使用了select *,一直以來也風平浪靜,運行得好好的,直到有一天對該表加了個欄位,代碼沒更新,報錯了~,你沒看錯,代碼沒動,加個欄位程式就報錯了。
報錯信息如下:

數組越界!問題可以在本地穩定復現,先把程式跑起來,執行 select * 的sql,再add column給表新增一個欄位,再次執行相同的sql,報錯。

具體原因是我們程式使用了sharding-jdbc做分表(5.1.2版本),它會在服務啟動時,載入欄位信息緩存,在查詢後做欄位匹配,出錯就在匹配時。
具體代碼位置在:com.mysql.cj.protocol.a.MergingColumnDefinitionFactory#createFromFields

這個緩存是跟資料庫鏈接相關的,只有鏈接失效時,才會重新載入。主要有兩個參數和它相關:
spring.shardingsphere.datasource.master.idle-timeout 預設10min
spring.shardingsphere.datasource.master.max-lifetime 預設30min

預設緩存時間都比較長,你只能趕緊重啟服務解決,而如果服務數量非常多,又是一個生產事故。
我在sharding sphere github搜了一圈,沒有好的處理方案,相關鏈接如:
https://github.com/apache/shardingsphere/issues/21728
https://github.com/apache/shardingsphere/issues/22824

大體意思是如果真想這麼做,資料庫ddl需要通過sharding proxy,它會負責刷新客戶端的緩存,但我們使用的是sharding jdbc模式,那隻能老老實實遵循規範,不要select * 了。如果select具體欄位,那新增的欄位也不會被select出來,和緩存的就能對應上。
那麼以後面試除了上面規範說到的,把這一點親身經歷也擺出來,應該可以加分吧。

總結

每條開發規範都有其背後的含義,都是經驗總結和踩坑教訓,對於團隊的開發規範我們都要仔細閱讀,嚴格遵守。可以看到上面每個小問題都可能導致不小的生產事故,保持敬畏之心,大概就是這個意思了吧。

更多分享,歡迎關註我的github:https://github.com/jmilktea/jtea


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 常用操作文件目錄的函數 1. CreateDirectory 創建文件夾 原型: BOOL CreateDirectory( LPCTSTR lpPathName, LPSECURITY_ATTRIBUTES lpSecurityAttributes ); 參數說明: lpPathName 要創建的 ...
  • developer-roadmap —— 提供最全的開發者技術路線指南。前端開發、後端開發、全棧開發、DevOps、Android 開發、AI、大數據、游戲開發等方向都有詳盡的學習路線思維導圖。 ...
  • 原文:juejin.cn/post/7291564831710445622 JDK線上程的Stop方法時明確不得強行銷毀一個線程,要優雅的退出線程。 何謂優雅退出線程,即業務將進行中請求正確被處理,取消待執行請求,執行資源回收,最終Thread Runable run 方法return 結束執行。 ...
  • 在業務邏輯中有時候會遇到兩層for迴圈的情況,觸發某些條件時,需要直接退出兩層for迴圈 而python官方是沒有 goto 語句的那麼我們可以這樣實現 第一種定義變數flag,根據flag的值做退出 flag=True for i in range(10): for j in range(10): ...
  • 當你使用一個不存在的鍵(key)去訪問一個Python字典(dict)時,會觸發一個KeyError異常。這是Python提供的一種機制,用於指示你正在嘗試訪問一個字典中不存在的鍵。 以下是一個簡單的示例,演示了當使用一個不存在的鍵去訪問字典時會發生的情況: my_dict = {"apple": ...
  • 在開始講解之前,我想給大家介紹一個很有用的第三方包,它就是gradio。如果你想與他人共用你的機器學習模型、API或數據科學工作流的最佳方式之一,可以創建一個互動式應用,讓用戶或同事可以在瀏覽器中試用你的演示。而gradio正是可以幫助你在Python中構建這樣的演示,並且只需要幾行代碼即可完成! ...
  • 將Word轉換為HTML能將文檔內容發佈在網頁上,這樣,用戶就可以通過瀏覽器直接查看或閱讀文檔而無需安裝特定的軟體。Word轉HTML對於線上發佈信息、創建線上文檔庫以及構建互動式網頁應用程式都非常有用。以下是使用Python將Word轉換為HTML網頁的攻略,包含兩個示例。 使用Python 將W ...
  • 機器學習的第一步是準備數據,好的數據能幫助我們加深對機器學習演算法的理解。 不管是在學習還是實際工作中,準備數據永遠是一個枯燥乏味的步驟。scikit-learn庫顯然看到了這個痛點,才在它的數據載入子模塊中為我們準備了直接可用的數據集。 在它的數據載入子模塊中,提供了6種直接可用來學習演算法的經典數據 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...