最近接一個這樣的需求,為某個類的日志輸出到指定的文件。一般都是按日志級別輸出到對應的文件中。查閱相關資料和logback教程,寫出下面的demo供參考。
1.添加一個appender
<!-- jetty日志單獨輸出 -->
<appender name="demo—all" class="ch.qos.logback.core.rolling.RollingFileAppender"> <file>${rootpath}gateserver_demo-all.log</file> <filter class="ch.qos.logback.classic.filter.ThresholdFilter"> <level>DEBUG</level> </filter> <rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy" > <FileNamePattern>${rootpath}demo-all.%i.log</FileNamePattern> <minIndex>1</minIndex> <maxIndex>10</maxIndex> </rollingPolicy> <triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy"> <maxFileSize>800MB</maxFileSize> </triggeringPolicy> <encoder> <pattern>%date [%thread] %-5level %logger{80} - %msg%n</pattern> </encoder> </appender>
2 添加一個logger
<logger name="org.xx.demo" level="DEBUG" additivity="false">
<appender-ref ref="demo—all" />
</logger>
additivity必須設置為false,否則不光新的jetty日志文件會存儲jetty的日志,原業務日志(root)文件也會存儲.不繼承root。
這樣就OK了。
以下是logback的常用配置:
<?xml version="1.0" encoding="UTF-8"?> <!-- 從高到地低 OFF 、 FATAL 、 ERROR 、 WARN 、 INFO 、 DEBUG 、 TRACE 、 ALL --> <!-- 日志輸出規則 根據當前ROOT 級別,日志輸出時,級別高於root默認的級別時 會輸出 --> <!-- 以下 每個配置的 filter 是過濾掉輸出文件里面,會出現高級別文件,依然出現低級別的日志信息,通過filter 過濾只記錄本級別的日志--> <!-- 屬性描述 scan:性設置為true時,配置文件如果發生改變,將會被重新加載,默認值為true scanPeriod:設置監測配置文件是否有修改的時間間隔,如果沒有給出時間單位,默認單位是毫秒。當scan為true時,此屬性生效。默認的時間間隔為1分鍾。 debug:當此屬性設置為true時,將打印出logback內部日志信息,實時查看logback運行狀態。默認值為false。 --> <configuration scan="true" scanPeriod="60 seconds" debug="false"> <!-- 定義日志文件 輸入位置 --> <property name="log_dir" value="/logs/ev_cmdb" /> <!-- 日志最大的歷史 30天 --> <property name="maxHistory" value="30"/> <!-- ConsoleAppender 控制台輸出日志 --> <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender"> <!-- 對日志進行格式化 --> <encoder> <pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger -%msg%n</pattern> </encoder> </appender> <!-- ERROR級別日志 --> <!-- 滾動記錄文件,先將日志記錄到指定文件,當符合某個條件時,將日志記錄到其他文件 RollingFileAppender--> <appender name="ERROR" class="ch.qos.logback.core.rolling.RollingFileAppender"> <!-- 過濾器,只記錄WARN級別的日志 --> <filter class="ch.qos.logback.classic.filter.LevelFilter"> <level>ERROR</level> <onMatch>ACCEPT</onMatch> <onMismatch>DENY</onMismatch> </filter> <!-- 最常用的滾動策略,它根據時間來制定滾動策略.既負責滾動也負責出發滾動 --> <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> <!--日志輸出位置 可相對、和絕對路徑 --> <fileNamePattern>${log_dir}/%d{yyyy-MM-dd}/error-log.log</fileNamePattern> <!-- 可選節點,控制保留的歸檔文件的最大數量,超出數量就刪除舊文件假設設置每個月滾動,且<maxHistory>是6, 則只保存最近6個月的文件,刪除之前的舊文件。注意,刪除舊文件是,那些為了歸檔而創建的目錄也會被刪除--> <maxHistory>${maxHistory}</maxHistory> </rollingPolicy> <!-- 按照固定窗口模式生成日志文件,當文件大於20MB時,生成新的日志文件。窗口大小是1到3,當保存了3個歸檔文件后,將覆蓋最早的日志。 <rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy"> <fileNamePattern>${log_dir}/%d{yyyy-MM-dd}/.log.zip</fileNamePattern> <minIndex>1</minIndex> <maxIndex>3</maxIndex> </rollingPolicy> --> <!-- 查看當前活動文件的大小,如果超過指定大小會告知RollingFileAppender 觸發當前活動文件滾動 <triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy"> <maxFileSize>5MB</maxFileSize> </triggeringPolicy> --> <encoder> <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger - %msg%n</pattern> </encoder> </appender> <!-- WARN級別日志 appender --> <appender name="WARN" class="ch.qos.logback.core.rolling.RollingFileAppender"> <!-- 過濾器,只記錄WARN級別的日志 --> <filter class="ch.qos.logback.classic.filter.LevelFilter"> <level>WARN</level> <onMatch>ACCEPT</onMatch> <onMismatch>DENY</onMismatch> </filter> <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> <!-- 按天回滾 daily --> <fileNamePattern>${log_dir}/%d{yyyy-MM-dd}/warn-log.log </fileNamePattern> <!-- 日志最大的歷史 60天 --> <maxHistory>${maxHistory}</maxHistory> </rollingPolicy> <encoder> <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger - %msg%n</pattern> </encoder> </appender> <!-- INFO級別日志 appender --> <appender name="INFO" class="ch.qos.logback.core.rolling.RollingFileAppender"> <!-- 過濾器,只記錄INFO級別的日志 --> <filter class="ch.qos.logback.classic.filter.LevelFilter"> <level>INFO</level> <onMatch>ACCEPT</onMatch> <onMismatch>DENY</onMismatch> </filter> <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> <!-- 按天回滾 daily --> <fileNamePattern>${log_dir}/%d{yyyy-MM-dd}/info-log.log </fileNamePattern> <!-- 日志最大的歷史 60天 --> <maxHistory>${maxHistory}</maxHistory> </rollingPolicy> <encoder> <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger - %msg%n</pattern> </encoder> </appender> <!-- DEBUG級別日志 appender --> <appender name="DEBUG" class="ch.qos.logback.core.rolling.RollingFileAppender"> <!-- 過濾器,只記錄DEBUG級別的日志 --> <filter class="ch.qos.logback.classic.filter.LevelFilter"> <level>DEBUG</level> <onMatch>ACCEPT</onMatch> <onMismatch>DENY</onMismatch> </filter> <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> <!-- 按天回滾 daily --> <fileNamePattern>${log_dir}/%d{yyyy-MM-dd}/debug-log.log </fileNamePattern> <!-- 日志最大的歷史 60天 --> <maxHistory>${maxHistory}</maxHistory> </rollingPolicy> <encoder> <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger - %msg%n</pattern> </encoder> </appender> <!-- TRACE級別日志 appender --> <appender name="TRACE" class="ch.qos.logback.core.rolling.RollingFileAppender"> <!-- 過濾器,只記錄ERROR級別的日志 --> <filter class="ch.qos.logback.classic.filter.LevelFilter"> <level>TRACE</level> <onMatch>ACCEPT</onMatch> <onMismatch>DENY</onMismatch> </filter> <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> <!-- 按天回滾 daily --> <fileNamePattern>${log_dir}/%d{yyyy-MM-dd}/trace-log.log </fileNamePattern> <!-- 日志最大的歷史 60天 --> <maxHistory>${maxHistory}</maxHistory> </rollingPolicy> <encoder> <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger - %msg%n</pattern> </encoder> </appender> <logger name="java.sql.PreparedStatement" value="DEBUG" /> <logger name="java.sql.Connection" value="DEBUG" /> <logger name="java.sql.Statement" value="DEBUG" /> <logger name="com.ibatis" value="DEBUG" /> <logger name="com.ibatis.common.jdbc.SimpleDataSource" value="DEBUG" /> <logger name="com.ibatis.common.jdbc.ScriptRunner" level="DEBUG"/> <logger name="com.ibatis.sqlmap.engine.impl.SqlMapClientDelegate" value="DEBUG" /> <!-- root級別 DEBUG --> <root level="debug"> <!-- 控制台輸出 --> <appender-ref ref="STDOUT" /> <!-- 文件輸出 --> <appender-ref ref="ERROR" /> <appender-ref ref="INFO" /> <appender-ref ref="WARN" /> <appender-ref ref="DEBUG" /> <appender-ref ref="TRACE" /> </root> </configuration>
java代碼:
import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class LogUtils { /** * 錯誤輸入日志 */ public static final Logger log = LoggerFactory.getLogger(LogUtils.class); /** * 記錄一直 info信息 * * @param message */ public static void logInfo(String message) { StringBuilder s = new StringBuilder(); s.append((message)); log.info(s.toString()); } public static void logInfo(String message, Throwable e) { StringBuilder s = new StringBuilder(); s.append(("exception : -->>")); s.append((message)); log.info(s.toString(), e); } public static void logWarn(String message) { StringBuilder s = new StringBuilder(); s.append((message)); log.warn(s.toString()); } public static void logWarn(String message, Throwable e) { StringBuilder s = new StringBuilder(); s.append(("exception : -->>")); s.append((message)); log.warn(s.toString(), e); } public static void logDebug(String message) { StringBuilder s = new StringBuilder(); s.append((message)); log.debug(s.toString()); } public static void logDebug(String message, Throwable e) { StringBuilder s = new StringBuilder(); s.append(("exception : -->>")); s.append((message)); log.debug(s.toString(), e); } public static void logError(String message) { StringBuilder s = new StringBuilder(); s.append(message); log.error(s.toString()); } /** * 記錄日志錯誤信息 * * @param message * @param e */ public static void logError(String message, Throwable e) { StringBuilder s = new StringBuilder(); s.append(("exception : -->>")); s.append((message)); log.error(s.toString(), e); } }
測試:
import java.io.File; import java.io.IOException; import org.junit.Before; import org.junit.Test; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import ch.qos.logback.core.joran.spi.JoranException; import com.ghca.easyview.server.common.utils.LogBackConfigLoader; import com.ghca.easyview.server.common.utils.LogUtils; import com.ghca.easyview.server.config.ServerConfig; public class TestLog { @Before public void before() { File file = new File(System.getProperty("user.dir")); String path = file.getPath() + File.separator + "conf"; // 將主要配置文件路徑賦值給 ServerConfig.configPath屬性 ServerConfig.config_path = path; try { // 加載日志文件 LogBackConfigLoader.load(ServerConfig .loadLogBackConfPath("logback.xml")); } catch (IOException | JoranException e) { e.printStackTrace(); } } @Test public void test1() { LogUtils.logInfo("INFO ~"); LogUtils.logDebug("DEBUG ~"); LogUtils.logError("ERROR~"); LogUtils.logWarn("WARN ~"); } }