這下對阿裡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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...