日誌系統

来源:https://www.cnblogs.com/viaiu/archive/2018/08/05/9382088.html
-Advertisement-
Play Games

上一篇說了一下《解決問題的一般套路》,裡面講到了日誌系統的重要性,日誌重要嗎?監控重要嗎?of course!日誌就是要能找到用戶做了什麼請求那個機器。 ...


       上一篇說了一下《解決問題的一般套路》,裡面講到了日誌系統的重要性,日誌重要嗎?監控重要嗎?of course!日誌就是要能找到用戶做了什麼請求那個機器。

       上下游介面請求,請求參數和入參是否正確,我們可以統一寫一個面向切麵方法去列印日誌,不用每一處去寫,切入點大家自己按照規則定義,AOP是Spring提供的關鍵特性之一。AOP即面向切麵編程,是OOP編程的有效補充。使用AOP技術,可以將一些系統性相關的編程工作,獨立提取出來,獨立實現,然後通過切麵切入進系統。從而避免了在業務邏輯的代碼中混入很多的系統相關的邏輯——比如許可權管理,事物管理,日誌記錄等等。這些系統性的編程工作都可以獨立編碼實現,然後通過AOP技術切入進系統即可。從而達到了 將不同的關註點分離出來的效果。

@Aspect
@Component
public class ControllerLogAspect {
    private Logger logger = LoggerFactory.getLogger(getClass());

    ThreadLocal<Long> startTime = new ThreadLocal<>();

    @Pointcut("execution(* com.xxx.mobile.web.controller..*.*(..))")
    public void controllerLog() {

    }

    @Before("controllerLog()")
    public void doBefore(JoinPoint joinPoint) throws Throwable {
        startTime.set(System.currentTimeMillis());
        if (joinPoint == null) {
            return;
        }
        Signature signature = joinPoint.getSignature();
        // 接收到請求,記錄請求內容
        RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();
        ServletRequestAttributes attributes = null;
        HttpServletRequest request = null;
        String requestUrl = "";
        String httpMethod = "";
        String declaringTypeName = "";
        String actionName = "";
        String ip = "";
        if (requestAttributes != null) {
            attributes = (ServletRequestAttributes) requestAttributes;
        }
        if (attributes != null) {
            request = attributes.getRequest();
        }
        if (request != null) {
            requestUrl = request.getRequestURL().toString();
            httpMethod = request.getMethod();
            ip = IpUtils.getIpAddr(request);
        }
        if (signature != null) {
            declaringTypeName = joinPoint.getSignature().getDeclaringTypeName();
            actionName = joinPoint.getSignature().getName();
        }
        // 記錄下請求內容
        logger.debug("URL:[{}] HTTP_METHOD:[{}] CLASS_METHOD:[{}.{}] ip:[{}] ARGS:{}",
                requestUrl, httpMethod, declaringTypeName, actionName, ip, Arrays.toString(joinPoint.getArgs()));
    }

    @AfterReturning(returning = "ret", pointcut = "controllerLog()")
    public void doAfterReturning(Object ret) throws Throwable {
        // 處理完請求,返回內容
        logger.debug("Execute Time:{}ms \n{}", (System.currentTimeMillis() - startTime.get()), ret);
    }

      下麵將使用@ExceptionHandler處理全局異常,將異常信息更加人性化的輸出給用戶。當然我們記錄日誌還是會用log4j。至於log4j的用法大家可以百度。

@ControllerAdvice
public class MobileWebExceptionHandler {

    private static Logger logger = LoggerFactory.getLogger(MobileWebExceptionHandler.class);

    @ExceptionHandler(value = Exception.class)
    @ResponseBody
    public Object exceptionHandler(HttpServletRequest request, Exception e) throws Exception {
        String message = String.format("Url:[%s]-%s", request.getRequestURL().toString(), e.getMessage());
        logger.error(message, e);
        return MobileWebResponse.error(CODE_INVALID_PARAMETER, e.getMessage());
    }

  通過@ControllerAdvice。我們可以將對於控制器的全局配置放置在同一個位置,註解了@ControllerAdvice的類的方法可以使用@ExceptionHandler@InitBinder@ModelAttribute註解到方法上,這對所有註解了@RequestMapping的控制器內的方法有:

@ExceptionHandler:用於全局處理控制器裡面的異常。
@InitBinder:用來設置WebDataBinderWebDataBinder用來自動綁定前臺請求參數到Model中。
@ModelAttribute@ModelAttribute本來的作用是綁定鍵值對到Model里,此處是讓全局的@RequestMapping都能獲得在此處設置的鍵值對。

      什麼時候該列印什麼樣的日誌級別,這個也很重要,一般情況下我們列印德日誌級別info,warn,error居多,日誌級別有:

ALL:最低等級的,用於打開所有日誌記錄。
TRACE: designates finer­grained informational events than the DEBUG.Since:1.2.12,很低的日誌級別,一般不會使用。
DEBUG: 指出細粒度信息事件對調試應用程式是非常有幫助的,主要用於開發過程中列印一些運行信息。
INFO: 消息在粗粒度級別上突出強調應用程式的運行過程。列印一些你感興趣的或者重要的信息,這個可以用於生產環境中輸出程式運行的一些重要信息,但是不能濫用,避免列印過多的日誌。
WARN: 表明會出現潛在錯誤的情形,有些信息不是錯誤信息,但是也要給程式員的一些提示。
ERROR: 指出雖然發生錯誤事件,但仍然不影響系統的繼續運行。列印錯誤和異常信息,如果不想輸出太多的日誌,可以使用這個級別。
FATAL: 指出每個嚴重的錯誤事件將會導致應用程式的退出。這個級別比較高了。重大錯誤,這種級別你可以直接停止程式了。
OFF: 最高等級的,用於關閉所有日誌記錄 

     SpringBootAdmin顯示日誌監控級別,我們可以根據自己的需求控制列印什麼樣的日誌:

       我們在列印日誌一般日誌頭會有時間,應用名,spanId,traceId,代碼行數,堆棧信息等,如下:

2018-08-05 11:52:58.470 WARN [xxx-web,e1ec017e8247b79e,e1ec017e8247b79e,true] 10652 --- [qtp1033348658-177] o.s.web.servlet.PageNotFound [1147 ] : No mapping found for HTTP request with URI [/flyway] in DispatcherServlet with name 'dispatcherServlet'

       如果那個app報錯了,錯誤日誌怎樣讓大家看到,會選擇用RabbitMq+ELK(Elasticsearch , Logstash, Kibana), 這篇ELK原理與介紹(https://www.cnblogs.com/aresxin/p/8035137.html),這位小哥哥說的還不錯。Kafka可以被redis和RabbitMq 所替換。最終錯誤日誌會顯示在kibana上,如下圖,除了時時監控錯誤的個數,還可以DSL語言查詢某個時間段發生的錯誤日誌,幫助我們分析問題。歡迎指正!

 


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

-Advertisement-
Play Games
更多相關文章
  • CSS的基礎概念 CSS的作用就是對網頁中元素進行更精細的美化 在網頁中加入CSS的三種方法 比如:<p style=”今天天氣一般</p> <style type=”text/css”></style>開始與結束標簽之間的樣式表。 然後:在html網頁中使用: 或 將外部樣式表引入到當前網頁中來。 ...
  • 1. 一個對象以“{”(左括弧)開始,“}”(右括弧)結束。每個“名稱”後跟一個“:”(冒號);“"名稱/值" 對”之間運用 “,”(逗號)分隔。 名稱用引號括起來;值如果是字元串則必須用括弧,數值型則不須要。例如:var json = { "name": "姓名", "sex": "25" };2 ...
  • HTML基礎 一、 <font></font>字體 通常在網頁中我們儘量使用宋體,黑體,微軟雅黑(win7以上)這些字體。因為這些字體通常是操作系統自帶的。如果我們使用了電腦上不存在的字體,那麼電腦將使用預設字體,一般是宋體。 二、 color屬性(十六進位數值表示123456789abcdef ...
  • 首先是是在main.js文件中把國際化引入進來 1 import Vue from 'vue' 2 import App from './App' 3 import router from './router' 4 import VueI18n from 'vue-i18n' 5 6 Vue.use ...
  • 什麼是ES6? ECMAScript 6.0 (簡稱ES6)是繼ECMAScript 5.1 之後 JavaScript 語言的下一代標準,發佈在2015年6月。他的目標,是使得 JavaScript 語言可以用來編寫誒複雜的大型應用程式,成為企業級開發語言 前提條件: 熟練掌握ES5 的基本語法、 ...
  • 使用html5音頻背景 直到現在,仍然不存在一項旨在網頁上播放音頻的標準。今天,大多數音頻是通過插件(比如 Flash)來播放的。然而,並非所有瀏覽器都擁有同樣的插件。HTML5 規定了一種通過 audio 元素來包含音頻的標準方法,audio 元素能夠播放聲音文件或者音頻流。 音頻格式 當前,au ...
  • 前面講了創建一個對象實例的方法單例模式Singleton Parttern, 創造多個產品的工廠模式(簡單工廠模式 Simple Factory Pattern, 工廠方法模式 FactoryMothed Parttern,抽象工廠模式 Abstract Factory Method),以及創建複雜 ...
  • 最近感悟挺多,看了很多源碼,發現很多東西自己都不懂,程式員這條路真是慢慢其修遠兮啊,我愛寫一些感悟。在公司上班最重要的就是工作態度,(就是職業道德,工匠精神),工作習慣(有一套自己的工作方式方法),工作經驗(程式員絕對是長時間熬出來的),編程思想(把自己當成電腦才能理解電腦),業務邏輯(人的思維 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...