发布新日志

  • 如何使用Log4j?(转)

    2010-10-24 18:15:45

    1、 Log4j是什么?
    Log4j可以帮助调试(有时候debug是发挥不了作 用的)和分析,要下载和了解更详细的内容,还是访问其官方网站吧:http://jakarta.apache.org/log4j 

    2、Log4j的概念
       Log4j中有三个主要的组件,它们分别是 Logger、Appender和Layout,Log4j 允许开发人员定义多个Logger,每个Logger拥有自己的名字,Logger之间通过名字来表明隶属关系。有一个Logger称为Root,它永远存在,且不能通过名字检索或引用,可以通过Logger.getRootLogger()方法获得,其它Logger通过 Logger.getLogger(String name)方法。
       Appender则是用来指明将所有的log信息存放到什么地方,Log4j中支持多种appender,如 console、files、GUI components、NT Event Loggers等,一个Logger可以拥有多个Appender,也就是你既可以将Log信息输出到屏幕,同时存储到一个文件中。
       Layout的作用是控制Log信息的输出方式,也就是格式化输出的信息。
       Log4j中将要输出的Log信息定义了5种级别,依次为DEBUG、INFO、WARN、ERROR和FATAL,当输出时,只有级别高过配置中规定的级别的信息才能真正的输出,这样就很方便的来配置不同情况下要输出的内容,而不需要更改代码,这点实在是方便啊。

    3、Log4j的配置文件
    虽然可以不用配置文件,而在程序中实现配置,但这种方法在如今的系统开发中显然是不可取的,能采用配置文件的地方一定一定要用配置文件。Log4j支持两种格式的配置文件:XML格式和Java的property格式,本人更喜欢后者,首先看一个简单的例子吧,如下:

    log4j.rootLogger=debug, stdout, R
    log4j.appender.stdout=org.apache.log4j.ConsoleAppender
    log4j.appender.stdout.layout=org.apache.log4j.PatternLayout

    # Pattern to output the caller's file name and line number.
    log4j.appender.stdout.layout.ConversionPattern=%5p [%t] (%F:%L) - %m%n

    log4j.appender.R=org.apache.log4j.RollingFileAppender
    log4j.appender.R.File=example.log
    log4j.appender.R.MaxFileSize= 100KB

    # Keep one backup file
    log4j.appender.R.MaxBackupIndex=1

    log4j.appender.R.layout=org.apache.log4j.PatternLayout
    log4j.appender.R.layout.ConversionPattern=%p %t %c - %m%n        

    首先,是设置root,格式为 log4j.rootLogger=[level],appenderName, ...,其中level就是设置需要输出信息的级别,后面是appender的输出的目的地,appenderName就是指定日志信息输出到哪个地方。您可以同时指定多个输出目的地。 配置日志信息输出目的地Appender,其语法为
    log4j.appender.appenderName = fully.qualified.name.of.appender.class
    log4j.appender.appenderName.option1 = value1
    ...
    log4j.appender.appenderName.option = valueN 
    Log4j提供的appender有以下几种:
    org.apache.log4j.ConsoleAppender(控制台)
    org.apache.log4j.FileAppender(文件)
    org.apache.log4j.DailyRollingFileAppender(每天产生一个日志文件)
    org.apache.log4j.RollingFileAppender(文件大小到达指定尺寸的时候产生新文件)
    org.apache.log4j.WriterAppender(将日志信息以流格式发送到任意指定的地方)
    配置日志信息的格式(布局),其语法为:
    log4j.appender.appenderName.layout = fully.qualified.name.of.layout.class
    log4j.appender.appenderName.layout.option1 = value1
    ....
    log4j.appender.appenderName.layout.option = valueN 
    Log4j提供的layout有以下几种:
    org.apache.log4j.HTMLLayout(以HTML表格形式布局),
    org.apache.log4j.PatternLayout(可以灵活地指定布局模式),
    org.apache.log4j.SimpleLayout(包含日志信息的级别和信息字符串),
    org.apache.log4j.TTCCLayout(包含日志产生的时间、线程、类别等等信息)

    Log4J采用类似C语言中的printf函数的打印格式格式化日志信息,打印参数如下: %m 输出代码中指定的消息 
       %p 输出优先级,即DEBUG,INFO,WARN,ERROR,FATAL 
      %r 输出自应用启动到输出该log信息耗费的毫秒数 
      %c 输出所属的类目,通常就是所在类的全名 
      %t 输出产生该日志事件的线程名 
      %n 输出一个回车换行符,Windows平台为“\r\n”,Unix平台为“\n” 
      %d 输出日志时间点的日期或时间,默认格式为ISO8601,也可以在其后指定格式,比如:%d{yyy MMM dd HH:mm:ss,SSS},输出类似: 2002年10月18日 22:10:28,921 
      %l 输出日志事件的发生位置,包括类目名、发生的线程,以及在代码中的行数。举例:Testlog4.main(TestLog4.java:10)

    4、Log4j在程序中的使用 
    要在自己的程序中使用Log4j,首先需要将commons-logging.jar和logging-log4j-1.2.9.jar导入到构建路径中。然后再将log4j.properties放到src根目录下。这样就可以在程序中使用log4j了。在类中使用log4j, 首先声明一个静态变量 Logger logger=Logger.getLog("classname");现在就可以使用了,用法如下:logger.debug("debug message")或者logger.info("info message"),看下面一个小例子:

    import com.foo.Bar;
    import org.apache.log4j.Logger;
    import org.apache.log4j.PropertyConfigurator;
    public class MyApp {
        static Logger logger = Logger.getLogger(MyApp.class.getName());
        public static void main(String[] args) {
          // BasicConfigurator replaced with PropertyConfigurator.
          PropertyConfigurator.configure(args[0]);
          logger.info("Entering application.");
          Bar bar = new Bar();
          bar.doIt();
          logger.info("Exiting application.");
        }
    }

    log4j.rootLogger=DEBUG,CONSOLE,DATABASE,FILE
    log4j.addivity.org.apache=true

    # 应用于控制台
    log4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender
    log4j.appender.CONSOLE.Threshold=INFO
    log4j.appender.CONSOLE.Target=System.out
    log4j.appender.CONSOLE.Encoding=GBK
    log4j.appender.CONSOLE.layout=org.apache.log4j.PatternLayout
    log4j.appender.CONSOLE.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n

    # 用于数据库
    log4j.appender.DATABASE=org.apache.log4j.jdbc.JDBCAppender
    log4j.appender.DATABASE.URL=jdbc:oracle:thin:@192.168.0.103:1521:ORCL
    log4j.appender.DATABASE.driver=oracle.jdbc.driver.OracleDriver
    log4j.appender.DATABASE.user=Nation
    log4j.appender.DATABASE.password=1
    log4j.appender.CONSOLE.Threshold=WARN
    log4j.appender.DATABASE.sql=INSERT INTO LOG4J(stamp,thread, infolevel,class,messages) VALUES ('%d{yyyy-MM-dd HH:mm:ss}', '%t', '%p', '%l', '%m')
    # INSERT INTO LOG4J (Message) VALUES ('[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n')
    # 写入数据库中的表LOG4J的Message字段中,内容%d(日期)%c: 日志信息所在地(类名)%p: 日志信息级别%m: 产生的日志具体信息 %n: 输出日志信息换行
    log4j.appender.DATABASE.layout=org.apache.log4j.PatternLayout
    log4j.appender.DATABASE.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n

    # 每天新建日志
    log4j.appender.A1=org.apache.log4j.DailyRollingFileAppender
    log4j.appender.A1.File=C:/log4j/log
    log4j.appender.A1.Encoding=GBK
    log4j.appender.A1.Threshold=DEBUG
    log4j.appender.A1.DatePattern='.'yyyy-MM-dd
    log4j.appender.A1.layout=org.apache.log4j.PatternLayout
    log4j.appender.A1.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L : %m%n

    #应用于文件
    log4j.appender.FILE=org.apache.log4j.FileAppender
    log4j.appender.FILE.File=C:/log4j/file.log
    log4j.appender.FILE.Append=false
    log4j.appender.FILE.Encoding=GBK
    log4j.appender.FILE.layout=org.apache.log4j.PatternLayout
    log4j.appender.FILE.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n

    # 应用于文件回滚
    log4j.appender.ROLLING_FILE=org.apache.log4j.RollingFileAppender
    log4j.appender.ROLLING_FILE.Threshold=ERROR
    log4j.appender.ROLLING_FILE.File=rolling.log
    log4j.appender.ROLLING_FILE.Append=true
    log4j.appender.CONSOLE_FILE.Encoding=GBK
    log4j.appender.ROLLING_FILE.MaxFileSize=10KB
    log4j.appender.ROLLING_FILE.MaxBackupIndex=1
    log4j.appender.ROLLING_FILE.layout=org.apache.log4j.PatternLayout
    log4j.appender.ROLLING_FILE.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n

    #自定义Appender
    log4j.appender.im = net.cybercorlin.util.logger.appender.IMAppender
    log4j.appender.im.host = mail.cybercorlin.net
    log4j.appender.im.username = username
    log4j.appender.im.password = password
    log4j.appender.im.recipient = yyflyons@163.com
    log4j.appender.im.layout=org.apache.log4j.PatternLayout
    log4j.appender.im.layout.ConversionPattern =[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n

    #应用于socket
    log4j.appender.SOCKET=org.apache.log4j.RollingFileAppender
    log4j.appender.SOCKET.RemoteHost=localhost
    log4j.appender.SOCKET.Port=5001
    log4j.appender.SOCKET.LocationInfo=true
    # Set up for Log Facter 5
    log4j.appender.SOCKET.layout=org.apache.log4j.PatternLayout
    log4j.appender.SOCET.layout.ConversionPattern=[start]%d{DATE}[DATE]%n%p[PRIORITY]%n%x[NDC]%n%t[THREAD]%n%c[CATEGORY]%n%m[MESSAGE]%n%n
    # Log Factor 5 Appender
    log4j.appender.LF5_APPENDER=org.apache.log4j.lf5.LF5Appender
    log4j.appender.LF5_APPENDER.MaxNumberOfRecords=2000

    # 发送日志给邮件
    log4j.appender.MAIL=org.apache.log4j.net.SMTPAppender
    log4j.appender.MAIL.Threshold=FATAL
    log4j.appender.MAIL.BufferSize=10
    log4j.appender.MAIL.From=yyflyons@163.com
    log4j.appender.MAIL.SMTPHost=www.wusetu.com
    log4j.appender.MAIL.Subject=Log4J Message
    log4j.appender.MAIL.To=yyflyons@126.com
    log4j.appender.MAIL.layout=org.apache.log4j.PatternLayout
    log4j.appender.MAIL.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n

    在实际编程时,要使Log4j真正在系统中运行事先还要对配置文件进行定义。定义步骤就是对Logger、Appender及Layout的分别使用。Log4j支持两种配置文件格式,一种是XML格式的文件,一种是java properties(key=value)【Java特性文件(键=值)】。(这里只说明properties文件)

    1、配置根Logger

          其语法为: 
          log4j.rootLogger = [ level ] , appenderName1, appenderName2, … 
          level : 是日志记录的优先级,分为OFF、FATAL、ERROR、WARN、INFO、DEBUG、ALL或者您定义的级别。Log4j建议只使用四个级别,优先级从高到低分别是ERROR、WARN、INFO、DEBUG。通过在这里定义的级别,您可以控制到应用程序中相应级别的日志信息的开关。比如在这里定 义了INFO级别,则应用程序中所有DEBUG级别的日志信息将不被打印出来。appenderName:就是指定日志信息输出到哪个地方。您可以同时指定多个输出目的地。 
         例如:log4j.rootLogger=info,A1,B2,C3

    2、配置日志信息输出目的地

          其语法为: 
          log4j.appender.appenderName = fully.qualified.name.of.appender.class // 
          "fully.qualified.name.of.appender.class" 可以指定下面五个目的地中的一个:

              1.org.apache.log4j.ConsoleAppender(控制台) 
              2.org.apache.log4j.FileAppender(文件) 
              3.org.apache.log4j.DailyRollingFileAppender(每天产生一个日志文件) 
              4.org.apache.log4j.RollingFileAppender(文件大小到达指定尺寸的时候产生一个新的文件) 
              5.org.apache.log4j.WriterAppender(将日志信息以流格式发送到任意指定的地方)

                 1.ConsoleAppender选项 
                        Threshold=WARN:指定日志消息的输出最低层次。 
                        ImmediateFlush=true:默认值是true,意谓着所有的消息都会被立即输出。 
                        Target=System.err:默认情况下是:System.out,指定输出控制台 
                  2.FileAppender 选项 
                        Threshold=WARN:指定日志消息的输出最低层次。 
                        ImmediateFlush=true:默认值是true,意谓着所有的消息都会被立即输出。 
                        File=mylog.txt:指定消息输出到mylog.txt文件。 
                        Append=false:默认值是true,即将消息增加到指定文件中,false指将消息覆盖指定的文件内容。 
                 3.DailyRollingFileAppender 选项 
                        Threshold=WARN:指定日志消息的输出最低层次。 
                        ImmediateFlush=true:默认值是true,意谓着所有的消息都会被立即输出。 
                        File=mylog.txt:指定消息输出到mylog.txt文件。 
                        Append=false:默认值是true,即将消息增加到指定文件中,false指将消息覆盖指定的文件内容。 
                        DatePattern=''.''yyyy-ww:每周滚动一次文件,即每周产生一个新的文件。当然也可以指定按月、周、天、时和分。即对应的格式如下: 
                        1)''.''yyyy-MM: 每月 
                        2)''.''yyyy-ww: 每周 
                        3)''.''yyyy-MM-dd: 每天 
                        4)''.''yyyy-MM-dd-a: 每天两次 
                        5)''.''yyyy-MM-dd-HH: 每小时 
                        6)''.''yyyy-MM-dd-HH-mm: 每分钟 
                 4.RollingFileAppender 选项 
                        Threshold=WARN:指定日志消息的输出最低层次。 
                        ImmediateFlush=true:默认值是true,意谓着所有的消息都会被立即输出。 
                        File=mylog.txt:指定消息输出到mylog.txt文件。 
                        Append=false:默认值是true,即将消息增加到指定文件中,false指将消息覆盖指定的文件内容。 
                        MaxFileSize=100KB: 后缀可以是KB, MB 或者是 GB. 在日志文件到达该大小时,将会自动滚动,即将原来的内容移到mylog.log.1文件。 
                        MaxBackupIndex=2:指定可以产生的滚动文件的最大数。

    3、配置日志信息的格式

            其语法为: 
      1). log4j.appender.appenderName.layout = fully.qualified.name.of.layout.class 
                  "fully.qualified.name.of.layout.class" 可以指定下面4个格式中的一个: 
                   1.org.apache.log4j.HTMLLayout(以HTML表格形式布局), 
             2.org.apache.log4j.PatternLayout(可以灵活地指定布局模式), 
             3.org.apache.log4j.SimpleLayout(包含日志信息的级别和信息字符串), 
             4.org.apache.log4j.TTCCLayout(包含日志产生的时间、线程、类别等等信息) 
                       1.HTMLLayout 选项 
                          LocationInfo=true:默认值是false,输出java文件名称和行号 
                          Title=my app file: 默认值是 Log4J Log Messages. 
                       2.PatternLayout 选项 
                          ConversionPattern=%m%n :指定怎样格式化指定的消息。 
                       3.XMLLayout 选项 
                          LocationInfo=true:默认值是false,输出java文件和行号 
            2). log4j.appender.A1.layout.ConversionPattern=%-4r %-5p %d{yyyy-MM-dd HH:mm:ssS} %c %m%n 
               这里需要说明的就是日志信息格式中几个符号所代表的含义: 
               -X号: X信息输出时左对齐; 
                       %p: 输出日志信息优先级,即DEBUG,INFO,WARN,ERROR,FATAL, 
                       %d: 输出日志时间点的日期或时间,默认格式为ISO8601,也可以在其后指定格式,比如:%d{yyy MMM dd HH:mm:ss,SSS},输出类似:2002年10月18日 22:10:28,921 
                       %r: 输出自应用启动到输出该log信息耗费的毫秒数 
                       %c: 输出日志信息所属的类目,通常就是所在类的全名 
                       %t: 输出产生该日志事件的线程名 
                       %l: 输出日志事件的发生位置,相当于%C.%M(%F:%L)的组合,包括类目名、发生的线程,以及在代码中的行数。举例:Testlog4.main(TestLog4.java:10) 
                       %x: 输出和当前线程相关联的NDC(嵌套诊断环境),尤其用到像java servlets这样的多客户多线程的应用中。 
                       %%: 输出一个"%"字符 
                       %F: 输出日志消息产生时所在的文件名称 
                       %L: 输出代码中的行号 
                       %m: 输出代码中指定的消息,产生的日志具体信息 
                       %n: 输出一个回车换行符,Windows平台为"rn",Unix平台为"n"输出日志信息换行 
                可以在%与模式字符之间加上修饰符来控制其最小宽度、最大宽度、和文本的对齐方式。如: 
                         1)%20c:指定输出category的名称,最小的宽度是20,如果category的名称小于20的话,默认的情况下右对齐。 
                         2)%-20c:指定输出category的名称,最小的宽度是20,如果category的名称小于20的话,"-"号指定左对齐。 
                         3)%.30c:指定输出category的名称,最大的宽度是30,如果category的名称大于30的话,就会将左边多出的字符截掉,但小于30的话也不会有空格。 
                         4)%20.30c:如果category的名称小于20就补空格,并且右对齐,如果其名称长于30字符,就从左边交远销出的字符截掉。

  • JAVA入门学习

    2007-07-30 19:40:19

    JAVA的环境变量如何设置?

    如果是Win2000或者XP,使用鼠标右击“我的电脑”->属性->高级->环境变量
    系统变量->新建->变量名:JAVA_HOME 变量值:c:\j2sdk1.4.2
    系统变量->编辑->变量名:Path 在变量值的最前面加上:%JAVA_HOME%\bin;
    系统变量->新建->变量名:CLASSPATH 变量值:.;
    CLASSPATH前面的那个"."和上面的意义是一样的。

    Q:javac 不是有效的内部命令
    A:有两个原因:1 没有安装jdk,安装了jre,这时候是可以运行java命令,但是javac不能运行。2 安装了jdk但路径设置不对,也就是path这个环境变量设置不对,请参考如上设置方法,一定要在path的最前边加上jdk的bin目录的路径。例如,如果设置了JAVA_HOME,那么在path前加【%JAVA_HOME%/bin;】

    Q: JDK怎么使用?
    A: 按照提示把JDK安装成功,设置好环境变量,打开记事本,输入:

    1
    2
    3
    4
    5
     public class HelloWorld {
            public static void main(String[] args) {
                System.out.println("Hello ,world!");
            }
        }//注意大小写 !!!
    

    保存为HelloWorld.java ,注意保存时要选择"所有类型", 打开命令提示符(Dos),进入你保存文件的目录,
    javac HelloWorld.java (编译) 如果没错误,再
    java HelloWorld (运行)
    java.sun.com官方关于第一个Java程序编写的教程.
    Your First Cup of Java (for Microsoft Windows)

    Q: 运行javac HelloWorld.java 为何error:cannot read:HelloWorld.java 1 error
    A: 首先检查文件名的的拼写,大小写是否正确,再看目录是否正确。可以在运行该命令的窗口中dir HelloWorld.java,看有无该文件。

    Q: 为什么 java HelloWorld 不成功?
    经常会发现这样的错误:
    java.lang.NoClassDefFoundError: HelloWorld
    Exception in thread "main"
    A: 首先可能是你CLASSPATH没有设置正确,请按照上面的那个步骤进行设置。如果你路径设置正确了,请确认你的类名是否正确,JAVA中大小写是区分的,看看你的类名是否正确。

    Q: Applet 怎样运行?
    A: 主要有两种方法
    1, JDK中的applet浏览器。
    2, IE浏览器
    这两种方法都要将Applet嵌入到 HTML 文件中,如
    <applet code="MyApplet.class",width=200 height=150>
    </applet>
    保存在和源文件同目录下,文件名为xxx.html,如果用JDK中
    的applet浏览器看,就在命令提示符下运行
    appletviewer xxx.heml
    用IE看直接双击 html 文件即可

    Q: 为什么用IE运行Applet时,显示的就是一片空白
    A: IE本身并不支持Applet 的运行,要去 java.sun.com 下载一个插件(java plug_in)

    Q: NullPointerException的原因?
    A: NullPointerException发生的原因是操作了一个为null的变量,比如使用该为null变量的【.】操作.如
    1
    2
    File[] fileSet=new File[3];//只是初始化了fileSet数组,并没有初始化各个元素,各个元素现在仍为null.
    fileSet[0].exists();//该操作就会造成NullPointerException异常.
    


    Q: 查看Applet不能运行的错误原因?
    A:对于用浏览器来查看的方式,如果是ie的话,可以通过工具-》Sun java控制台来看看是什么错误。其他浏览器也有类似的sun java控制台。

    Q:为什么编译提示uses or overrides a deprecated API?(或java文件使用或覆盖了已过时的API)
    A:From jdk文档 java.langAnnotation Type Deprecated

    programmers are discouraged from using, typically because it is dangerous, or because a better alternative exists. Compilers warn when a deprecated program element is used or overridden in non-deprecated code。

    Q:为什么编译提示class YourClass(注:你的类名)is pubic, should be declared in a file named YourClass.java?
    A:有两种情况,只有一个解释。两种情况是:
    1 把两个public类放在同一个文件中,这样就会出现该问题。
    2 一个文件虽然只有一个公共类,但该文件的名字和public的类名不一致(注意大小写,必须和你的public的类的名字大小写完全一样)。

    一个解释也就很清楚了,public类所在的文件名不是“类名+.java”。改正的方法也很明白了,一个java文件中只有一个public类,并且该文件名的大小写和public的类的类名完全一致。


    Q:uses unchecked or unsafe operations.
    Note: Recompile with -Xlint:unchecked for details.

    A:和泛型有关。不过只是个Note,你可以不用管它。
    在1.5中,使用util中的类时,如List,希望你指定特定的类型,如List<String>,这样就只能add String类型的元素。如果没有参数化集合类,就会出现这样的提示,你可用 javac -Xlint:unchecked编译来试试。

    Q:Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 6
    A:数组越界,访问数组下标为6的元素时发生的越界异常。


    Q:编译提示:不是抽象的,并且未覆盖超类或接口中的抽象方法。例如如下代码:
    1
    2
    3
    4
    5
    6
    7
    8
    9
    public class check extends Applet implements ItemListener{
          public void init(){ 
          .......
          } 
          
          public void itemStateChange(ItemEvent e){
          ......
          }
    }
    

    编译提示:check不是抽象的,并且未覆盖java.awt.event.itemListener中的抽象方法:itemStateChanged(java.awt.event.ItemEvent)

    A: 编译提示中说得很明确。check实现了ItemListener,但却没有覆盖其中的抽象方法:itemStateChanged(java.awt.event.Item.Event)。检查上边的代码,其中的方法名错写成itemStateChange,最后一个单词是Changed。
    在java中,如果类继承了超类,或是实现了接口,而没有实现其中的抽象方法,该类必须被声明为abstract类,否则就会出现如上的提示。出现这种错误的原因有三:1 没有写这个方法。2 写这个方法了,但是方法的签名和需要的不一样,容易出现方法名写错。3 参数类型不一致,参数顺序不一致。其中方法名写错最容易出现。
    这种情况下,仔细对照ItemListener的API文档,看看自己写的方法和其中的抽象方法的方法名,参数类型,参数顺序是否相同。仔细检查应该可以发现问题,那时必将会心一笑,原来如此简单,:)。

Open Toolbar