This is the 21st day of my participation in the November Gwen Challenge. Check out the event details: The last Gwen Challenge 2021
Public number Mac code split reading link
preface
Previously, SparkStreaming was used for development. Recently, I decided to learn Flink, so I downloaded Flink 1.11 from the official website. I planned to build a client and submit the program to YARN. Since Flink has not provided Hadoop dependencies since 1.7, many dependencies have to be downloaded yourself, resulting in various ClassNotFoundException.
Slf4j, LOG4j, Logback jar package a dozen, Baidu did not know for a long time what is the difference between each JAR, where to use, I plan to summarize their own.
Log Development history
Long Long Ago, just like when I first learned Java, I used system.out.println console printing to check whether the program output meets my expectations. This is a relatively primitive method, which cannot automatically distinguish the types of logs and is almost impossible to use in production systems.
The java.until.logging framework was provided in JDK1.4 to print logs, but JUL was too difficult to use, so log4j was manually used. Log4j stopped updating log4j since 1.2 because of security vulnerabilities and code structure issues that were difficult to maintain. And then he tried the log4J2 again, and then he tried the LogBack with better performance and more features.
Since then, this big guy has built the log world and created the most common logging frameworks: Log4j, log4j2, logBack.
SLF4J (Simple Logging Facade for Java)
Four logging frameworks have been mentioned so far. If we want to use them for logging, in addition to the necessary configuration files, we also need to get Logger in the code and print the log.
The code is as follows:
// To use log4j, you need log4j.jar
import org.apache.log4j.Logger;
Logger logger_log4j = Logger.getLogger(Test.class);
logger_log4j.info("Hello World!");
// To use log4j2, you need log4j-api.jar, log4j-core.jar
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
Logger logger_log4j2 = LogManager.getLogger(Test.class);
logger_log4j2.info("Hello World!");
Logback-classic. jar, logback-core.jar
import ch.qos.logback.classic.Logger;
import ch.qos.logback.classic.LoggerContext;
Logger logger_logback = new LoggerContext().getLogger(Test.class);
logger_logback.info("Hello World!");
// java.until.logging, jul for short
import java.util.logging.Logger;
Logger logger_jul = Logger.getLogger("java.Test");
Copy the code
Why use a facade system
As you can see from the above, using different logging frameworks means importing different JARS and using different code to get the Logger.
Suppose a project wants to upgrade from JUL to Logback during a lengthy upgrade process and needs to change the code to get a new Logger. If you use JUl in 100 classes, you have to change 100 places. What a chore!
The role of facade system
Apache Commons Logging appears.
Common-logging provides a logging entry called “facade logging”, meaning that it does not write logs, but provides a unified interface to determine the logging framework to use through the JAR, so that code is not modified when changing frameworks. Slf4j, a facade logging framework, was developed by the developers of Log4j because of the difficulty of common-logging. Slf4j is used to describe facade logging.
Facade logging, like facade patterns in design mode, does not provide services by itself. It provides a unified entrance to subsystems, encapsulates the complexity of subsystems, and facilitates client invocation. Slf4j is like a cainiao Courier station. It has no express service, but it provides express services such as SF express and ZTO express. As for whether you want to use SF express or Zto express, it totally depends on your idea.
The code for using SLf4J is as follows:
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
Logger logger = LoggerFactory.getLogger(Test.class);
logger.info("Hello World!")
Copy the code
This line of code is like when you want to send something in a logger, which one to use? A minute later… You decided to use SF (Logback. Jar), so you filled in THE EXPRESS order of SF (log4j.jar), but you found there was 10 yuan in wechat balance, so you could only use LOG4j, so you returned the order of SF (remove logback.jar), and filled in the express order of ZTO (log4j.jar). Then send the Courier (print the log).
So how does SLF4J decide which framework logs to use and which JAR packages to introduce?
As shown in the official slF4J diagram:
To use SLf4J, you need to import slf4J-api.jar.
To work with log4j, import log4j.jar and bridge package slf4J-log412.jar.
Jar, log4j-api.jar, log4j-core.jar, and bridge package log4j-slf4j-impl.
Jar and logback-core.jar. No bridge package is required.
What is a bridge packet and why does logBack not have one
Let’s see what SLf4J does starting with LoggerFactory.getLogger().
The flow chart is as follows:
The idea is to have the ClassLoader find StaticLoggerBinder from the classpath and use it to return the Log4J, Logback Logger and print the log.
The so-called bridge package implements the StaticLoggerBinder class, which connects SLF4J to the logging framework. Because Log4j and Log4J2 didn’t initially have a class called StaticLoggerBinder, they had to write a new JAR to implement StaticLoggerBinder without changing the program structure. With slF4J in logBack, StaticLoggerBinder is implemented in the JAR of logback itself, so there is no need for bridging packages.
StaticLoggerBinder implements the function of creating loggers using the underlying logging framework, and their respective StaticLoggerBinder provides loggers for SLF4J to print logs to the user.
Log4j and Log4J2 bridge packages and LogBack dependencies have the StaticLoggerBinder class.
Used to summarize
“Class path contains multiple SLF4J bindings.”
The above report information is encountered when using SLF4J. I have also had web services fail to start due to SLF4J issues. This is because the logback-classic, log4J-slf4j-impl, slf4j-log412, slf4j-JDK jars cannot exist at the same time. They both implement the StaticLoggerBinder class, which causes conflicts, and SLF4J cannot determine which logging framework to use.
conclusion
The above is combined with the official website and some of my own debugging code summary, I hope to help you understand SLF4J and its application, which is not enough also hope to point out, common progress, mutual encourage!!