源碼分析SpringMvc日志打印被忽略輸出問題
1.寫在前面
在java的開發(fā)過程中,涉及到j(luò)ava web的開發(fā),基本上都是走spring這一套了。
我們之前一般來說,都會(huì)說mvc:Model(模型業(yè)務(wù))、View(視圖界面)、Controller(控制器)。這個(gè)學(xué)習(xí)java開發(fā)的,應(yīng)該都懂吧,這里就不多說了。
這里,我們先著重解析下Controller:
Controller控制器是指控制器接受用戶的輸入并調(diào)用模型和視圖去完成用戶的需求,控制器本身不輸出任何東西和做任何處理。它只是接收請(qǐng)求并決定調(diào)用哪個(gè)模型構(gòu)件去處理請(qǐng)求,然后再確定用哪個(gè)視圖來顯示返回的數(shù)據(jù)。
那作為控制層的框架,一般有springmvc、struts2,struts2一些老的框架可能會(huì)用到,新的一些框架,基本就很少見了,都走springmvc這套了。
springmvc,我們就用得多了,但是對(duì)于SpringMvc日志打印被忽略輸出的問題,有無人研究過呢?
嘿嘿,巧了,這個(gè)問題,哥們研究過了,那我們今天就來分析下這個(gè)問題嘍!??!
2.問題引出

DEBUG org.springframework.web.servlet.mvc.method.annotation.RequestResponseBodyMethodProcessor Read "application/json;charset=UTF-8" to [QueryData(pageNo=1, pageSize=10, sortField=, sortOrder=asc, params={Q_XM_S_EQ=llsydn, Q_SJH_S_EQ=135 (truncated)...]
可以看到控制臺(tái)打印的日志,出現(xiàn)了 (truncated)...
這里對(duì)應(yīng)的前端頁面的功能是這樣的:

很明顯,springmvc接收到前端傳進(jìn)來的參數(shù),然后會(huì)在控制臺(tái)打印相關(guān)的日志。但是這里的參數(shù),日志打印是不完整的。出現(xiàn)了 (truncated)...截取的問題。
看到這,有好奇心的小伙伴,估計(jì)都想探個(gè)究竟了吧?(別說你不想??)
好嘛好嘛,是我自己想探個(gè)究竟啦?。。?卑微)
那這里我們先從源碼中,找到被截取的地方嘍。
3.截取源碼分析
首先這里,要如何進(jìn)行源碼查看呢?這里我們可以從打印的日志入手......
- RequestResponseBodyMethodProcessor

進(jìn)入到RequestResponseBodyMethodProcessor類中,查看log打印日志的地方,并沒有發(fā)現(xiàn)打印日志的代碼。
那我們就將該類所有的方法都打上斷點(diǎn)(主要是該類不算很大),再發(fā)一個(gè)請(qǐng)求,斷點(diǎn)調(diào)試一下:

- 進(jìn)入到readWithMessageConverters,方法里面來了

經(jīng)過一步步執(zhí)行,執(zhí)行到
Object arg = readWithMessageConverters(inputMessage, parameter, paramType);
可以看到控制臺(tái),就打印了我們剛才的日志
那這里,我們就可以繼續(xù)往readWithMessageConverters,里面進(jìn)行查看(再發(fā)一個(gè)請(qǐng)求)

經(jīng)過一步步執(zhí)行,執(zhí)行到LogFormatUtils.traceDebug
可以看到控制臺(tái),就打印了我們剛才的日志
那這里,將其他的斷點(diǎn)去掉,再這里點(diǎn)一個(gè),然后再發(fā)一個(gè)請(qǐng)求進(jìn)來。


嘿嘿,看到這,估計(jì)大家都知道了吧。
當(dāng)參數(shù)字符超過100,這里就會(huì)截取。
看到這里,我們是通過源碼,一步步的找到最終的原因,那這里只是提供一個(gè)方法,思路。
大家伙可以自己動(dòng)手去調(diào)試一波,畢竟也不難,哈哈!
這里貼一下,最終的業(yè)務(wù)邏輯代碼,如下:
(limitLength && str.length() > 100 ? str.substring(0, 100) + " (truncated)..." : str);
可以看到,當(dāng)limitLength為true時(shí),str長(zhǎng)度大于100,就會(huì)截取前100個(gè)字符,后面再拼接 (truncated)...
那我們不禁會(huì)想,嘿,哥們想辦法把limitLength的值,改為false,不就永遠(yuǎn)都不會(huì)截取輸出日志了嗎?
嗯嗯,是這么一回事,那這個(gè)limitLength的值,如何改?怎么來的?
只能通過源碼分析一波了,好啦,這個(gè)任務(wù)就交給各位了,動(dòng)手去干活吧!
4.截取問題處理

public boolean isTraceEnabled(Marker marker) {
final FilterReply decision = callTurboFilters(marker, Level.TRACE);
if (decision == FilterReply.NEUTRAL) {
return effectiveLevelInt <= Level.TRACE_INT;
} else if (decision == FilterReply.DENY) {
return false;
} else if (decision == FilterReply.ACCEPT) {
return true;
} else {
throw new IllegalStateException("Unknown FilterReply value: " + decision);
}
}
通過查看這里的代碼,就是這里返回上面limitLength的值。
可以看到這里,我們的改變decision的值,讓它變成FilterReply.ACCEPT,這樣這里就能返回true 了。
callTurboFilters,如何改呢?這里我百度找到了一個(gè)方式:
Logback 使用TurboFilter實(shí)現(xiàn)日志級(jí)別等內(nèi)容的動(dòng)態(tài)修改操作。
- 定義TurboFilter
public class SpringMvcFilter extends TurboFilter {
@Override
public FilterReply decide(Marker marker, Logger logger, Level level, String format, Object[] params, Throwable throwable) {
if (logger.getName().equals("org.springframework.web.servlet.mvc.method.annotation.RequestResponseBodyMethodProcessor")) {
return FilterReply.ACCEPT; //返回accept
}
return FilterReply.NEUTRAL;
}
}
- logback的配置xml
<configuration>
<turboFilter class="com.llsydn.log.filter.SpringMvcFilter" />
......
</configuration>
看看最終效果:
DEBUG org.springframework.web.servlet.mvc.method.annotation.RequestResponseBodyMethodProcessor Read "application/json;charset=UTF-8" to [QueryData(pageNo=1, pageSize=10, sortField=, sortOrder=asc, params={Q_XM_S_EQ=llsydn, Q_SJH_S_EQ=135****4606]
可以看到,springmvc打印的日志,正常了,不再截取了。
好了,以上就是 SpringMvc日志打印被忽略輸出問題分析(源碼分析) 的分享了。
個(gè)人實(shí)操可能也不夠全面,班門弄斧了。
以上就是源碼分析SpringMvc日志打印被忽略輸出問題的詳細(xì)內(nèi)容,更多關(guān)于SpringMvc日志打印被忽略輸出的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java異步調(diào)用轉(zhuǎn)同步方法實(shí)例詳解
這篇文章主要介紹了Java異步調(diào)用轉(zhuǎn)同步方法實(shí)例詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06
SpringBoot中使用MQTT實(shí)現(xiàn)消息的訂閱和發(fā)布(示例代碼)
這篇文章主要介紹了SpringBoot中使用MQTT實(shí)現(xiàn)消息的訂閱和發(fā)布的相關(guān)知識(shí),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),感興趣的朋友跟隨小編一起看看吧2024-06-06
Spring?Boot整合Log4j2.xml的問題及解決方法
這篇文章主要介紹了Spring?Boot整合Log4j2.xml的問題,本文給大家分享解決方案,需要的朋友可以參考下2023-09-09
eclipse連接數(shù)據(jù)庫(kù)并實(shí)現(xiàn)用戶注冊(cè)登錄功能
這篇文章主要介紹了eclipse連接數(shù)據(jù)庫(kù)并實(shí)現(xiàn)用戶注冊(cè)登錄功能的相關(guān)資料,需要的朋友可以參考下2021-01-01
java新增關(guān)聯(lián)的三張表,每張表要求都插入集合,代碼實(shí)現(xiàn)方式
這篇文章主要介紹了java新增關(guān)聯(lián)的三張表,每張表要求都插入集合,代碼實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12
SQLSyntaxErrorException-ExecutorException報(bào)錯(cuò)解決分析
這篇文章主要為大家介紹了SQLSyntaxErrorException-ExecutorException報(bào)錯(cuò)解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08

