วิธีเขียนบันทึกในไฟล์ข้อความเมื่อใช้ java.util.logging.Logger


147

ฉันมีสถานการณ์ที่ฉันต้องการเขียนบันทึกทั้งหมดที่ฉันสร้างขึ้นเป็นไฟล์ข้อความ

เรากำลังใช้ java.util.logging.Logger API เพื่อสร้างบันทึก

ฉันเหนื่อย:

private static Logger logger = Logger.getLogger(className.class.getName());
FileHandler fh;   
fh = new FileHandler("C:/className.log");   
logger.addHandler(fh); 

แต่ยังได้รับบันทึกของฉันบนคอนโซลเท่านั้น ....



2
จำนวนคำตอบแนะนำให้ใช้ FileHandler เหมือนเดิมที่คุณพยายามทำ สิ่งหนึ่งที่จะตระหนักถึง (บทเรียนที่เจ็บปวดได้เรียนรู้): FileHandler มีการทำข้อมูลให้ตรงกัน ซึ่งหมายความว่าในแอพพลิเคชั่นที่มีมัลติเธรดสูงสิ่งที่คุณต้องมีคือการหยุดชะงักที่อาจเกิดขึ้นได้คือการส่งวัตถุที่จะเข้าสู่ระบบซึ่งเมธอด toString () เรียกวิธีการซิงโครไนซ์ ระวัง FileHandler
Tim Boudreau

คำตอบ:


238

ลองตัวอย่างนี้ มันใช้งานได้สำหรับฉัน

public static void main(String[] args) {  

    Logger logger = Logger.getLogger("MyLog");  
    FileHandler fh;  

    try {  

        // This block configure the logger with handler and formatter  
        fh = new FileHandler("C:/temp/test/MyLogFile.log");  
        logger.addHandler(fh);
        SimpleFormatter formatter = new SimpleFormatter();  
        fh.setFormatter(formatter);  

        // the following statement is used to log any messages  
        logger.info("My first log");  

    } catch (SecurityException e) {  
        e.printStackTrace();  
    } catch (IOException e) {  
        e.printStackTrace();  
    }  

    logger.info("Hi How r u?");  

}

สร้างเอาต์พุตที่ MyLogFile.log

Apr 2, 2013 9:57:08 AM testing.MyLogger main  
INFO: My first log  
Apr 2, 2013 9:57:08 AM testing.MyLogger main  
INFO: Hi How r u?

แก้ไข:

ในการลบตัวจัดการคอนโซลให้ใช้

logger.setUseParentHandlers(false);

เนื่องจาก ConsoleHandler ถูกลงทะเบียนด้วย parent logger ซึ่ง loggers ทั้งหมดได้รับมา


1
มันใช้งานได้สำหรับฉัน ... แต่ฉันได้รับบันทึกที่คอนโซลด้วยวิธีลบออกจากที่นั่น
Pankaj

คุณช่วยแนะนำฉันว่าจะทำอย่างไรถ้าฉันต้องการเก็บบันทึกทั้งหมด ที่จริงแล้วจากวิธีการนี้ไฟล์ข้อความของฉันจะถูกแทนที่หากฉันเรียกใช้แอปพลิเคชันของฉันเป็นครั้งที่สอง?
Pankaj

1
วิธีการทำ ... ฉัน google มัน แต่ฉันพบรหัสที่ทำให้เกิดความสับสนมากมาย ... คุณช่วยกรุณา ..
Pankaj

7
@bluemunch คุณสามารถใช้ตัวสร้างทางเลือกของFileHandler(path, true)เพื่อให้บันทึกต่อท้ายไปยังไฟล์บันทึกที่มีอยู่
Sri Harsha Chilakapati

1
@ ใช่ใช่ สำหรับกรณีนี้ฉันมักจะสร้างวิธีการอรรถประโยชน์ logger สร้าง
Sri Harsha Chilakapati

16

ประการแรกคุณกำหนด logger ของคุณจากที่ใดและ class \ method พยายามเรียกมันว่าอะไร มีตัวอย่างการทำงานอบสดใหม่:

public class LoggingTester {
    private final Logger logger = Logger.getLogger(LoggingTester.class
            .getName());
    private FileHandler fh = null;

    public LoggingTester() {
        //just to make our log file nicer :)
        SimpleDateFormat format = new SimpleDateFormat("M-d_HHmmss");
        try {
            fh = new FileHandler("C:/temp/test/MyLogFile_"
                + format.format(Calendar.getInstance().getTime()) + ".log");
        } catch (Exception e) {
            e.printStackTrace();
        }

        fh.setFormatter(new SimpleFormatter());
        logger.addHandler(fh);
    }

    public void doLogging() {
        logger.info("info msg");
        logger.severe("error message");
        logger.fine("fine message"); //won't show because to high level of logging
    }
}   

ในรหัสของคุณคุณลืมกำหนด formatter ถ้าคุณต้องการง่าย ๆ คุณสามารถทำได้ตามที่ฉันกล่าวข้างต้น แต่มีตัวเลือกอื่นคุณสามารถจัดรูปแบบด้วยตัวเองมีตัวอย่าง (เพียงแค่แทรกแทนบรรทัดนี้ fh .setFormatter (SimpleFormatter ใหม่ ()) รหัสต่อไปนี้):

fh.setFormatter(new Formatter() {
            @Override
            public String format(LogRecord record) {
                SimpleDateFormat logTime = new SimpleDateFormat("MM-dd-yyyy HH:mm:ss");
                Calendar cal = new GregorianCalendar();
                cal.setTimeInMillis(record.getMillis());
                return record.getLevel()
                        + logTime.format(cal.getTime())
                        + " || "
                        + record.getSourceClassName().substring(
                                record.getSourceClassName().lastIndexOf(".")+1,
                                record.getSourceClassName().length())
                        + "."
                        + record.getSourceMethodName()
                        + "() : "
                        + record.getMessage() + "\n";
            }
        });

หรือการดัดแปลงอื่น ๆ ตามที่คุณต้องการ หวังว่ามันจะช่วย


9

ตำแหน่งของไฟล์บันทึกสามารถควบคุมได้ผ่านไฟล์ logging.properties และสามารถส่งผ่านเป็นพารามิเตอร์ JVM เช่น:java -Djava.util.logging.config.file=/scratch/user/config/logging.properties

รายละเอียด: https://docs.oracle.com/cd/E23549_01/doc.1111/e14568/handler.htm

การกำหนดค่าตัวจัดการไฟล์

ในการส่งบันทึกไปยังไฟล์ให้เพิ่ม FileHandler ไปยังคุณสมบัติตัวจัดการในไฟล์ logging.properties นี่จะเป็นการเปิดใช้การบันทึกไฟล์ทั่วโลก

handlers= java.util.logging.FileHandler กำหนดค่าตัวจัดการโดยการตั้งค่าคุณสมบัติต่อไปนี้:

java.util.logging.FileHandler.pattern=<home directory>/logs/oaam.log
java.util.logging.FileHandler.limit=50000
java.util.logging.FileHandler.count=1
java.util.logging.FileHandler.formatter=java.util.logging.SimpleFormatter

java.util.logging.FileHandler.patternระบุตำแหน่งและรูปแบบของไฟล์เอาต์พุต การตั้งค่าเริ่มต้นคือโฮมไดเร็กตอรี่ของคุณ

java.util.logging.FileHandler.limitระบุจำนวนไบต์สูงสุดที่ตัวบันทึกเขียนไปยังไฟล์ใดไฟล์หนึ่ง

java.util.logging.FileHandler.countระบุจำนวนไฟล์เอาต์พุตที่จะวนรอบ

java.util.logging.FileHandler.formatterระบุคลาส java.util.logging formatter ที่คลาสตัวจัดการไฟล์ใช้เพื่อจัดรูปแบบข้อความบันทึก SimpleFormatter เขียนบทสรุปของบันทึกข้อมูลสรุปแบบสั้น "ที่มนุษย์อ่านได้"


เพื่อสั่งให้จาวาใช้ไฟล์คอนฟิกูเรชันนี้แทน $ JDK_HOME / jre / lib / logging.properties:

java -Djava.util.logging.config.file=/scratch/user/config/logging.properties

คำตอบที่ยอดเยี่ยมจากสิ่งที่ฉันเห็นคือระดับโลกเดียว ฉันตัดสินใจที่จะไปกับการเปลี่ยนแปลง logging.properties ใน JDK ตัวเอง (แม้ว่าใน Java 11 พวกเขาอยู่ในไดเรกทอรี conf จากไดเรกทอรีการติดตั้ง Java) น่าสังเกตว่าตำแหน่งเริ่มต้นของไฟล์บันทึกดังกล่าวจะเป็น user.home / javaX.log (โดยที่ user.home คือ por por ของระบบและ X เป็นหมายเลขถัดไปตามลำดับ - สำหรับอันแรกมันจะเป็น 0)
บรรทัด


5

บางทีนี่อาจเป็นสิ่งที่คุณต้องการ ...

import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.IOException;
import java.util.logging.FileHandler;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JOptionPane;
import javax.swing.JPanel;

/**
 * LogToFile class
 * This class is intended to be use with the default logging class of java
 * It save the log in an XML file  and display a friendly message to the user
 * @author Ibrabel <ibrabel@gmail.com>
 */
public class LogToFile {

    protected static final Logger logger=Logger.getLogger("MYLOG");
    /**
     * log Method 
     * enable to log all exceptions to a file and display user message on demand
     * @param ex
     * @param level
     * @param msg 
     */
    public static void log(Exception ex, String level, String msg){

        FileHandler fh = null;
        try {
            fh = new FileHandler("log.xml",true);
            logger.addHandler(fh);
            switch (level) {
                case "severe":
                    logger.log(Level.SEVERE, msg, ex);
                    if(!msg.equals(""))
                        JOptionPane.showMessageDialog(null,msg,
                            "Error", JOptionPane.ERROR_MESSAGE);
                    break;
                case "warning":
                    logger.log(Level.WARNING, msg, ex);
                    if(!msg.equals(""))
                        JOptionPane.showMessageDialog(null,msg,
                            "Warning", JOptionPane.WARNING_MESSAGE);
                    break;
                case "info":
                    logger.log(Level.INFO, msg, ex);
                    if(!msg.equals(""))
                        JOptionPane.showMessageDialog(null,msg,
                            "Info", JOptionPane.INFORMATION_MESSAGE);
                    break;
                case "config":
                    logger.log(Level.CONFIG, msg, ex);
                    break;
                case "fine":
                    logger.log(Level.FINE, msg, ex);
                    break;
                case "finer":
                    logger.log(Level.FINER, msg, ex);
                    break;
                case "finest":
                    logger.log(Level.FINEST, msg, ex);
                    break;
                default:
                    logger.log(Level.CONFIG, msg, ex);
                    break;
            }
        } catch (IOException | SecurityException ex1) {
            logger.log(Level.SEVERE, null, ex1);
        } finally{
            if(fh!=null)fh.close();
        }
    }

    public static void main(String[] args) {

        /*
            Create simple frame for the example
        */
        JFrame myFrame = new JFrame();
        myFrame.setTitle("LogToFileExample");
        myFrame.setSize(300, 100);
        myFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        myFrame.setLocationRelativeTo(null);
        JPanel pan = new JPanel();
        JButton severe = new JButton("severe");
        pan.add(severe);
        JButton warning = new JButton("warning");
        pan.add(warning);
        JButton info = new JButton("info");
        pan.add(info);

        /*
            Create an exception on click to use the LogToFile class
        */
        severe.addActionListener(new ActionListener(){

            @Override
            public void actionPerformed(ActionEvent ae) {
                int j = 20, i = 0;
                try {
                    System.out.println(j/i);
                } catch (ArithmeticException ex) {
                    log(ex,"severe","You can't divide anything by zero");
                }

            }

        });

        warning.addActionListener(new ActionListener(){

            @Override
            public void actionPerformed(ActionEvent ae) {
                int j = 20, i = 0;
                try {
                    System.out.println(j/i);
                } catch (ArithmeticException ex) {
                    log(ex,"warning","You can't divide anything by zero");
                }

            }

        });

        info.addActionListener(new ActionListener(){

            @Override
            public void actionPerformed(ActionEvent ae) {
                int j = 20, i = 0;
                try {
                    System.out.println(j/i);
                } catch (ArithmeticException ex) {
                    log(ex,"info","You can't divide anything by zero");
                }

            }

        });

        /*
            Add the JPanel to the JFrame and set the JFrame visible
        */
        myFrame.setContentPane(pan);
        myFrame.setVisible(true);
    }
}

4
import java.io.IOException;
import org.apache.log4j.Appender;
import org.apache.log4j.FileAppender;
import org.apache.log4j.Logger;
import org.apache.log4j.SimpleLayout;

/**
 * @author Kiran
 * 
 */
public class MyLogger {

    public MyLogger() {
    }

    public static void main(String[] args) {
        Logger logger = Logger.getLogger("MyLog");
        Appender fh = null;
        try {
            fh = new FileAppender(new SimpleLayout(), "MyLogFile.log");
            logger.addAppender(fh);
            fh.setLayout(new SimpleLayout());
            logger.info("My first log");
        } catch (SecurityException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        logger.info("Hi How r u?");
    }
}

1
ขอบคุณสิ่งนี้ช่วยฉันออก แต่ Log4j 2 ออกมาแล้วและฉันต้องตกปลารอบ ๆ ไซต์เพื่อหารุ่น 1.2
SoluableNonagon

3
int SIZE = "<intialize-here>"
int ROTATIONCOUNT = "<intialize-here>"

Handler handler = new FileHandler("test.log", SIZE, LOG_ROTATIONCOUNT);
logger.addHandler(handler);     // for your code.. 

// you can also set logging levels
Logger.getLogger(this.getClass().getName()).log(Level.[...]).addHandler(handler);

1

หวังว่าผู้คนจะพบว่าสิ่งนี้มีประโยชน์

public static void writeLog(String info) {
    String filename = "activity.log";
    String FILENAME = "C:\\testing\\" + filename;
    BufferedWriter bw = null;
    FileWriter fw = null;
    try {
        fw = new FileWriter(FILENAME, true);
        bw = new BufferedWriter(fw);
        bw.write(info);
        bw.write("\n");
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        try {
            if (bw != null)
                bw.close();
            if (fw != null)
                fw.close();
        } catch (IOException ex) {
            ex.printStackTrace();
        }
    }
}

1
อืมหายไปครึ่งชั่วโมงและปัญญาของฉันพยายามที่จะทำให้ log4j เขียนไปยังไฟล์ freakin ' เครื่องมือส่วนใหญ่มีวิธีที่ซับซ้อนเกินไปสำหรับปัญหาที่พวกเขากำลังแก้ไข
Mihai Raulea

0

นี่คือคลาสการบันทึกของฉันตามคำตอบที่ยอมรับ :

import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.logging.*;

public class ErrorLogger
{
    private Logger logger;

    public ErrorLogger()
    {
        logger = Logger.getAnonymousLogger();

        configure();
    }

    private void configure()
    {
        try
        {
            String logsDirectoryFolder = "logs";
            Files.createDirectories(Paths.get(logsDirectoryFolder));
            FileHandler fileHandler = new FileHandler(logsDirectoryFolder + File.separator + getCurrentTimeString() + ".log");
            logger.addHandler(fileHandler);
            SimpleFormatter formatter = new SimpleFormatter();
            fileHandler.setFormatter(formatter);
        } catch (IOException exception)
        {
            exception.printStackTrace();
        }

        addCloseHandlersShutdownHook();
    }

    private void addCloseHandlersShutdownHook()
    {
        Runtime.getRuntime().addShutdownHook(new Thread(() ->
        {
            // Close all handlers to get rid of empty .LCK files
            for (Handler handler : logger.getHandlers())
            {
                handler.close();
            }
        }));
    }

    private String getCurrentTimeString()
    {
        DateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss");
        return dateFormat.format(new Date());
    }

    public void log(Exception exception)
    {
        logger.log(Level.SEVERE, "", exception);
    }
}

0

นี่คือตัวอย่างของวิธีการเขียนทับการกำหนดค่า Logger จากรหัส ไม่ต้องการไฟล์กำหนดค่าภายนอก ..

FileLoggerTest.java:

import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.LogManager;
import java.util.logging.Logger;

public class FileLoggerTest {

    public static void main(String[] args) {

        try {
            String h = MyLogHandler.class.getCanonicalName();
            StringBuilder sb = new StringBuilder();
            sb.append(".level=ALL\n");
            sb.append("handlers=").append(h).append('\n');
            LogManager.getLogManager().readConfiguration(new ByteArrayInputStream(sb.toString().getBytes("UTF-8")));
        } catch (IOException | SecurityException ex) {
            // Do something about it
        }

        Logger.getGlobal().severe("Global SEVERE log entry");
        Logger.getLogger(FileLoggerTest.class.getName()).log(Level.SEVERE, "This is a SEVERE log entry");
        Logger.getLogger("SomeName").log(Level.WARNING, "This is a WARNING log entry");
        Logger.getLogger("AnotherName").log(Level.INFO, "This is an INFO log entry");
        Logger.getLogger("SameName").log(Level.CONFIG, "This is an CONFIG log entry");
        Logger.getLogger("SameName").log(Level.FINE, "This is an FINE log entry");
        Logger.getLogger("SameName").log(Level.FINEST, "This is an FINEST log entry");
        Logger.getLogger("SameName").log(Level.FINER, "This is an FINER log entry");
        Logger.getLogger("SameName").log(Level.ALL, "This is an ALL log entry");

    }
}

MyLogHandler.java

import java.io.IOException;
import java.util.logging.FileHandler;
import java.util.logging.Level;
import java.util.logging.LogRecord;
import java.util.logging.SimpleFormatter;

public final class MyLogHandler extends FileHandler {

    public MyLogHandler() throws IOException, SecurityException {
        super("/tmp/path-to-log.log");
        setFormatter(new SimpleFormatter());
        setLevel(Level.ALL);
    }

    @Override
    public void publish(LogRecord record) {
        System.out.println("Some additional logic");
        super.publish(record);
    }

}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.