当前位置: 首页 > news >正文

logback之配置文件使用详解

目录

(一)配置文件的加载

(二)使用介绍

1、configuration:配置文件的跟元素

2、contextName:设置日志上下文名称

3、contextListener:设置上下文监听事件

4、property/variable/substitutionProperty:设置变量

5、conversionRule:自定义关键字的转换规则

6、statusListener:配置日志的状态监听器

7、appender:日志事件的输出组件

8、logger:日志记录实例

9、include:内嵌配置文件

(一)配置文件的加载

(1)设置系统变量:logback.configurationFile指定配置文件路径

启动时加入启动参数-Dlogback.configurationFile=/path/log/logback.xml

(2)查找resource下的logback-test.xml(这个通常是单元测试时使用,配置在test/resources下)

(3)查找resource下的logback.xml

(二)使用介绍

先给出一份比较详细的配置文件(注意此配置文件并不一定合理,因为很多标签属性是为了展示其用法而加,实际的配置文件需要根据项目所需设置),然后再对其中标签元素一一介绍

<configuration scan="true" scanPeriod="1 minute" debug="true"><!--<turboFilter class=""/>--><!-- Appenders --><!--<contextName>mallLogSystem</contextName>--><!--<contextListener class=""/>--><property name="logPath" value="logs" scope="SYSTEM"/><property name="appName" value="mallSystem"/><!--<property resource="log.properties"/>--><appender name="CONSOLE" class="ch.qos.logback.core.ConsoleAppender"><encoder><pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger{36} - %msg%n</pattern></encoder></appender><appender name="ASYNC" class="ch.qos.logback.classic.AsyncAppender"><!-- 设置是否在启动时触发之前丢弃的事件 --><discardingThreshold>0</discardingThreshold><!-- 设置这个appender的neveQueueSize --><queueSize>512</queueSize><!-- 添加真实的appender --><appender-ref ref="ROLLING"/></appender><appender name="ROLLING" class="ch.qos.logback.core.rolling.RollingFileAppender"><file>${logPath}/${appName}.log</file><rollingPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy"><fileNamePattern>${logPath}/${appName}.%d{yyyy-MM-dd}.%i.log</fileNamePattern><maxHistory>30</maxHistory> <!-- 保留最近30天的日志文件 --><MaxFileSize>10MB</MaxFileSize></rollingPolicy><encoder><pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger{36} - %msg%n</pattern></encoder></appender><appender name="impossible" class="ch.qos.logback.core.rolling.RollingFileAppender"><file>${logPath}/${impossibleName}.log</file><rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy"><fileNamePattern>${logPath}/${impossibleName}.%i.log.zip</fileNamePattern><minIndex>1</minIndex><maxIndex>3</maxIndex></rollingPolicy><triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy"><maxFileSize>16kb</maxFileSize></triggeringPolicy><encoder><pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger{50} %msg%n</pattern></encoder><filter class="com.mall.common.log.LevelRangeFilter"><minLevel>WARN</minLevel><maxLevel>ERROR</maxLevel><onMatch>ACCEPT</onMatch><onMismatch>DENY</onMismatch></filter></appender><logger name="impossible" additivity="false"><appender-ref ref="impossible"/></logger><!-- Root Logger --><root level="Info"><appender-ref ref="CONSOLE"/><appender-ref ref="ASYNC"/></root>
</configuration>

1、configuration:配置文件的跟元素

        属性介绍

        scan:true表示会添加一个定时任务检查配置文件是否改变,改变则重新加载,和scanPeriod配合使用。默认为false.

        scanPeriod:扫描周期,支持Duration字符串转换,300 millisecond,1 minute,60 second, 1 hour,1 day。空格将数量和单位分割开。

         debug:true,表示开启日志初始化时的内部日志打印,可以看到日志初始化的一些输出日志。默认是关闭的,日志初始化有问题可以将此项打开,方便看初始化过程。

       packagingData: 如果设置为true,那么设置context中对应的属性为true。此属性控制有异常抛出时,异常堆栈是否包含jar包以及对应版本号,默认为false

2、contextName:设置日志上下文名称

3、contextListener:设置上下文监听事件

        可以监听上下文生命周期的事件,属性class设置监听器的类。监听事件包含:onStart、onReset、onStop、onLevelChange

4、property/variable/substitutionProperty:设置变量

        这3个标签是同一个(历史原因造成),可以定义key-value属性值

        scope:作用域,有3种范围:LOCAL, CONTEXT, SYSTEM。变量的查找顺序是本地,然后是CONTEXT,然后才是SYSTEM。默认是LOCAL。注意SYSTEM实际会设置到系统变量中,所以需要谨慎。

        name:key值

        value:key对应的value。

        file:支持批量配置文件导入,值为文件绝对路径

        resource:配置的是资源文件,相对于resources路径,如:<property resource="log.properties"/>

        name和value是成对出现的,并且和file、resource互斥,即一个property只配置3者其中之一即可

5、conversionRule:自定义关键字的转换规则

        只需要配置class属性,对应的是转换器类。详情可参见《logback之自定义pattern使用的转换器》

6、statusListener:配置日志的状态监听器

        也只需要配置监听器类,只需要实现addStatusEvent即可,然后结合StatusPrinter,可以打印状态事件,在排错方面很有用。

7、appender:日志事件的输出组件

        name:定义一个名称

        class:对应的appender类

        通常有4种appender:ConsoleAppender,FileAppender,RollingFileAppender,AsyncAppender。除了AsyncAppender需要引用实际的appender以外,其余appender都需要定义encoder,以设置日志输出格式:

        <encoder><pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger{36} - %msg%n</pattern></encoder>

        pattern详细介绍,可参加《logback之pattern详解以及源码分析》同时还可以添加过滤器,如下:

        <filter class="ch.qos.logback.classic.filter.LevelFilter"><Level>INFO</Level><onMatch>ACCEPT</onMatch><onMismatch>DENY</onMismatch></filter>

        过滤器可自定义,详情可参见另一篇《logback之自定义过滤器》

       (1) ConsoleAppender:输出到控制台,示例如下:

    <appender name="CONSOLE" class="ch.qos.logback.core.ConsoleAppender"><encoder><pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger{36} - %msg%n</pattern></encoder></appender>

       (2)FileAppender:输出到固定的日志文件

    <appender name="fileAppender" class="ch.qos.logback.core.FileAppender"><file>${logPath}/${appName}_fix.log</file><append>true</append><prudent>false</prudent><BufferSize>1024</BufferSize><encoder><pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger{36} - %msg%n</pattern></encoder></appender>

        file:设置文件名称,支持变量绑定

        append:是否将日志追加到文件现有内容的末尾,默认为true。

        prudent:是否prudent模式,默认为false。此项开启需谨慎,用于多个进程间的共享文件写入,会加文件锁,所以可能会有性能影响。(如果文件每秒只有30条日志,性能影响不大,但如果每秒超过100以上,那么就可能会有性能问题了)

        BufferSize:缓存区大小。FileAppender用BufferedOutputStream实际输出日志,所以可以设置缓存区大小,默认8096。

       (3)RollingFileAppender:也是输出到日志文件,继承FileAppender,但可以配置滚动策略实现日志滚动,这个在实际生产中用的比较大。如下示例是每天滚动一次文件,最多保留30天的日志。同时还按文件大小滚动,如果文件超过10M也会滚动

    <appender name="ROLLING" class="ch.qos.logback.core.rolling.RollingFileAppender"><file>${logPath}/${appName}.log</file><rollingPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy"><fileNamePattern>${logPath}/${appName}.%d{yyyy-MM-dd}.%i.log</fileNamePattern><maxHistory>30</maxHistory> <!-- 保留最近30天的日志文件 --><MaxFileSize>10MB</MaxFileSize></rollingPolicy><encoder><pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger{36} - %msg%n</pattern></encoder></appender>

        继承自FileAppender,所以FileAppender可以设置的属性,它都可以设置。另外增加了RollingPolicy和TriggeringPolicy。设置滚动策略以及滚动触发的策略。SizeAndTimeBasedRollingPolicy继承自TimeBasedRollingPolicy,既是RollingPolicy也是TriggeringPolicy。所以不用额外设置TriggeringPolicy。

        滚动策略和触发策略还可以有如下搭配:

          按文件大小滚动:最多3个滚动文件,超过3个,会把旧的删掉。触发条件按文件大小,下面配置的是16kb。

        <rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy"><fileNamePattern>${logPath}/${impossibleName}.%i.log.zip</fileNamePattern><minIndex>1</minIndex><maxIndex>3</maxIndex></rollingPolicy><triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy"><maxFileSize>16kb</maxFileSize></triggeringPolicy>

        按日期滚动:每天滚动一次,滚动的频率是根据所配置${yyyy-MM-dd}文件日期格式定的,如果配置按月,那么就是${yyyy-MM}

    <appender name="ROLLING" class="ch.qos.logback.core.rolling.RollingFileAppender"><file>logs/logback.log</file><rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"><fileNamePattern>logs/logback.%d{yyyy-MM-dd}.log</fileNamePattern><maxHistory>30</maxHistory> <!-- 保留最近30天的日志文件 --></rollingPolicy><encoder><pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger{36} - %msg%n</pattern></encoder></appender>

       (4)AsyncAppender:异步日志输出,需要appender-ref指定实际的appender组件。

    <appender name="ASYNC" class="ch.qos.logback.classic.AsyncAppender"><!-- 如果队列的80%已满,则会丢弃TRACT、DEBUG、INFO级别的日志 --><discardingThreshold>20</discardingThreshold><!-- 更改默认的队列的深度,该值会影响性能.默认值为256 --><queueSize>512</queueSize><!-- 队列满了不阻塞调用者--><neverBlock>true</neverBlock><!-- 添加真实的appender,最多只能添加一个 --><appender-ref ref="ROLLING"/></appender>

        AsyncAppender是为了不阻塞主线程而使用的,日志不会直接输出,而是放到阻塞队列中。它会开启一个单独的后台线程,从ArrayBlockingQueue中日志获取再调用实际的appender输出日志。

        discardingThreshold:设置队列容量剩余多少时直接丢弃info(包含info)级别以下的日志的阈值。日志容量达到阈值,说明此时日志负载是比较重的,这时保证重要级别的日志输出。默认为-1,表示不限制。

        queueSize:队列的容量

        neverBlock:设置日志事件添加到队列中的方式,如果是neverBlock那么使用offer添加日志事件,那么在日志容量满的情况下是不会阻塞的;但如果是false,那么使用put添加就会阻塞主线程进入等待了。

8、logger:日志记录实例

        标签属性:

        name:logger的名称,有层级关系。如com.example.ClassInfo,那么它父级logger是com.example-》com-》root。所有的logger的根logger都是root。

        level:定义日志级别,如果没有定义则继承父级logger的日志级别。

        additivity:定义日志是否传递到父级,如果为true的话,那么除了会输出到当前logger的appender以外,还会传递给父级的appender。那么一条日志记录会输出到多个日志文件中。默认为true。

        内嵌标签:

                appender-ref:引用的appender

    <logger name="impossible" additivity="false"><appender-ref ref="impossible"/><appender-ref ref="fileAppender"/></logger>

9、include:内嵌配置文件

        标签属性:

        file:定义内嵌文件的绝对路径

        url:url形式引入内嵌文件

        resource:相对于resources路径引入内嵌文件

        optional:是否输出解析内嵌配置文件过程中的状态日志,默认是false。

        file、url、resource只能配置其中一个。内嵌文件必须以<included>开头,以<included>结尾。

        如定义一个logProperty.xml:

<included><property name="logPath" value="logs"/><property name="appName" value="mallSystem"/>
</included>

        然后在主配置文件中引入:

        <include resources="logProperty.xml"/>

        

相关文章:

logback之配置文件使用详解

目录 &#xff08;一&#xff09;配置文件的加载 &#xff08;二&#xff09;使用介绍 1、configuration&#xff1a;配置文件的跟元素 2、contextName&#xff1a;设置日志上下文名称 3、contextListener&#xff1a;设置上下文监听事件 4、property/variable/substituti…...

壁纸样机神器,这个工具适合专业设计师用吗?

壁纸样机神器在一定程度上适合专业设计师使用&#xff0c;但是否适合具体取决于设计师的需求和使用场景&#xff1a; 适合专业设计师的方面 快速实现设计想法&#xff1a;专业设计师在创作过程中&#xff0c;有时需要快速将设计想法变为可视化的效果图&#xff0c;以便进行初…...

MySQL秘籍之索引与查询优化实战指南

MySQL秘籍之索引与查询优化实战指南 目录 MySQL秘籍之索引与查询优化实战指南相关阅读索引相关EXPLAIN 版本 1. 初级篇1.1 【练体术】基础1.1.1 库操作1.1.1 表操作创建一个表增加表字段 1.1.2 增删改插入一条数据删除一条数据更新一条数据库 1.1.3 查询查询所有数据条件查询&a…...

【AI日记】25.01.03 kaggle 比赛 3-2 未来的命运

【AI论文解读】【AI知识点】【AI小项目】【AI战略思考】【AI日记】 工作 参加&#xff1a;kaggle 比赛 Forecasting Sticker Sales时间&#xff1a;8 小时 读书 书名&#xff1a;秦制两千年时间&#xff1a;1.5 小时评估&#xff1a;读完&#xff0c;非常不错&#xff0c;很…...

Linux(Centos 7.6)命令详解:ls

1.命令作用 列出目录内容(list directory contents) 2.命令语法 Usage: ls [OPTION]... [FILE]... 3.参数详解 OPTION: -l&#xff0c;long list 使用长列表格式-a&#xff0c;all 不忽略.开头的条目&#xff08;打印所有条目&#xff0c;包括.开头的隐藏条目&#xff09…...

【Unity3D】UGUI Canvas画布渲染流程

参考文档&#xff1a;画布 - Unity 手册 Canvas组件&#xff1a;画布组件是进行 UI 布局和渲染的抽象空间。所有 UI 元素都必须是附加了画布组件的游戏对象的子对象。 参数&#xff1a; Render Mode 渲染模式&#xff1a;Screen Space - Overlay、Screen Spa…...

minikube安装k8s

一、安装k8s版本 export REGISTRY_MIRRORhttps://registry.cn-hangzhou.aliyuncs.com curl -sSL https://kuboard.cn/install-script/v1.30.x/install_kubelet.sh | sh -s 1.30.0 二、安装docker及minikube useradd docker passwd docker 密码也设置为docker #创建docker组…...

Docker图形化界面工具Portainer最佳实践

前言 安装Portainer 实践-基于Portainer安装redis-sentinel部署 Spring Boot集成Redis Sentinel 前言 本篇文章笔者推荐一个笔者最常用的docker图形化管理工具——Portainer。 安装Portainer 编写docker-compose文件 Portainer部署的步骤比较简单&#xff0c;我们还是以…...

借助 FinClip 跨端技术探索鸿蒙原生应用开发之旅

在当今数字化浪潮汹涌澎湃的时代&#xff0c;移动应用开发领域正经历着深刻的变革与创新。鸿蒙操作系统的崛起&#xff0c;以其独特的分布式架构和强大的性能表现&#xff0c;吸引了众多开发者的目光。而FinClip 跨端技术的出现&#xff0c;为开发者涉足鸿蒙原生应用开发提供了…...

【网络】ARP表、MAC表、路由表

ARP表 网络设备存储IP-MAC映射关系的表项&#xff0c;便于快速查找和转发数据包 ARP协议工作原理 ARP&#xff08;Address Resolution Protocol&#xff09;&#xff0c;地址解析协议&#xff0c;能够将网络层的IP地址解析为数据链路层的MAC地址。 1.主机在自己的ARP缓冲区中建…...

Linux驱动开发学习准备(Linux内核源码添加到工程-Workspace)

Linux内核源码添加到VsCode工程 下载Linux-4.9.88源码&#xff1a; 没有处理同名文件的压缩包&#xff1a; https://pan.baidu.com/s/1yjIBXmxG9pwP0aOhW8VAVQ?pwde9cv 已把同名文件中以大写命名的文件加上_2后缀的压缩包&#xff1a; https://pan.baidu.com/s/1RIRRUllYFn2…...

25.1.3

java数组&#xff1a; dataType[] arrayRefVar //推荐写法 //int[] mylist //或 dataType arrayRefVar[] //int mylist[]创建数组对象&#xff1a; arrayRefVar new dataType[arraySize]; dataType[] arrayRefVar new dataType[arraySize];for-each循环&#xff1a; jav…...

Leecode刷题C语言之我的日程安排表②

执行结果:通过 执行用时和内存消耗如下&#xff1a; typedef struct {int start;int end; }BOOKING;#define MAX_BOOK_NUM (1000) typedef struct MyCalendar_ {BOOKING book[MAX_BOOK_NUM];int bnum;BOOKING *sorted[MAX_BOOK_NUM];int num;int conflict[MAX_BOOK_NUM];int c…...

十二、Vue 路由

文章目录 一、简介二、安装与基本配置安装 Vue Router创建路由实例在应用中使用路由实例三、路由组件与视图路由组件的定义与使用四、动态路由动态路由参数的定义与获取动态路由的应用场景五、嵌套路由嵌套路由的概念与配置嵌套路由的应用场景六、路由导航<router - link>…...

smell---Paddle-DI

跨模态文档智能大模型–Ernie-Layout 目标&#xff1a;提取文档中无结构或半结构化的知识 github项目地址 Paddle NLP ERNIE-Layout基于Transformer Encode架构&#xff0c;并提出以下trick&#xff1a; 1、OCR工具提取信息 借助OCR工具提取图片中的文字及文字对应的坐标信息…...

PCL点云库入门——PCL库点云特征之点云法向量(NormalEstimation)及其可视化

1、PCL点云库中点云特征综述 1.1、点云特征综述 点云特征描述在三维数据处理领域扮演着至关重要的角色&#xff0c;它直接决定了后续的识别、分类以及重建等关键任务的执行效果。在众多的特征描述方法中&#xff0c;我们可以看到基于几何形状的特征、基于统计信息的特征以及…...

25.Java JUC 引入(进程与线程、线程的状态、并发与并行、管程、用户线程与守护线程)

一、JUC 简介 JUC 是 java.util.concurrent 工具包的简称&#xff0c;这是一个处理线程的工具包&#xff0c;从 JDK1.5 开始出现 二、进程与线程 1、基本介绍 &#xff08;1&#xff09;进程 进程是计算机中的程序关于某数据集合上的一次运行活动&#xff0c;是系统进行资源…...

Linux 异步 I/O 框架 io_uring:基本原理、程序示例与性能压测

大家觉得有意义和帮助记得关注和点赞&#xff01;&#xff01;&#xff01; io_uring 是 2019 年 Linux 5.1 内核首次引入的高性能 异步 I/O 框架&#xff0c;能显著加速 I/O 密集型应用的性能。 但如果你的应用已经在使用 传统 Linux AIO 了&#xff0c;并且使用方式恰当&…...

Uniapp中使用`wxml-to-canvas`开发DOM生成图片功能

Uniapp中使用wxml-to-canvas开发DOM生成图片功能 在移动端开发中&#xff0c;生成图片是一个常见需求&#xff0c;例如用于分享海报、生成动态二维码等。在Uniapp框架中&#xff0c;我们可以通过wxml-to-canvas插件轻松实现将DOM转化为图片的功能。本文将详细介绍如何在Uniapp…...

Linux之ARM(MX6U)裸机篇----5.仿stm32的LED驱动实验

一&#xff0c;启动文件 .global _start .global _bss_start /* 类似宏定义把__bss_start定义为_bss_start */ _bss_start:.word __bss_start.global _bss_end _bss_end:.word __bss_end_start:#设置处理器进入SVC模式mrs r0, cpsr /* 读取cpsr到r0 */bic r0, r0, …...

【根据当天日期输出明天的日期(需对闰年做判定)。】2022-5-15

缘由根据当天日期输出明天的日期(需对闰年做判定)。日期类型结构体如下&#xff1a; struct data{ int year; int month; int day;};-编程语言-CSDN问答 struct mdata{ int year; int month; int day; }mdata; int 天数(int year, int month) {switch (month){case 1: case 3:…...

椭圆曲线密码学(ECC)

一、ECC算法概述 椭圆曲线密码学&#xff08;Elliptic Curve Cryptography&#xff09;是基于椭圆曲线数学理论的公钥密码系统&#xff0c;由Neal Koblitz和Victor Miller在1985年独立提出。相比RSA&#xff0c;ECC在相同安全强度下密钥更短&#xff08;256位ECC ≈ 3072位RSA…...

关于nvm与node.js

1 安装nvm 安装过程中手动修改 nvm的安装路径&#xff0c; 以及修改 通过nvm安装node后正在使用的node的存放目录【这句话可能难以理解&#xff0c;但接着往下看你就了然了】 2 修改nvm中settings.txt文件配置 nvm安装成功后&#xff0c;通常在该文件中会出现以下配置&…...

1688商品列表API与其他数据源的对接思路

将1688商品列表API与其他数据源对接时&#xff0c;需结合业务场景设计数据流转链路&#xff0c;重点关注数据格式兼容性、接口调用频率控制及数据一致性维护。以下是具体对接思路及关键技术点&#xff1a; 一、核心对接场景与目标 商品数据同步 场景&#xff1a;将1688商品信息…...

macOS多出来了:Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用

文章目录 问题现象问题原因解决办法 问题现象 macOS启动台&#xff08;Launchpad&#xff09;多出来了&#xff1a;Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用。 问题原因 很明显&#xff0c;都是Google家的办公全家桶。这些应用并不是通过独立安装的…...

屋顶变身“发电站” ,中天合创屋面分布式光伏发电项目顺利并网!

5月28日&#xff0c;中天合创屋面分布式光伏发电项目顺利并网发电&#xff0c;该项目位于内蒙古自治区鄂尔多斯市乌审旗&#xff0c;项目利用中天合创聚乙烯、聚丙烯仓库屋面作为场地建设光伏电站&#xff0c;总装机容量为9.96MWp。 项目投运后&#xff0c;每年可节约标煤3670…...

Mac软件卸载指南,简单易懂!

刚和Adobe分手&#xff0c;它却总在Library里给你写"回忆录"&#xff1f;卸载的Final Cut Pro像电子幽灵般阴魂不散&#xff1f;总是会有残留文件&#xff0c;别慌&#xff01;这份Mac软件卸载指南&#xff0c;将用最硬核的方式教你"数字分手术"&#xff0…...

高防服务器能够抵御哪些网络攻击呢?

高防服务器作为一种有着高度防御能力的服务器&#xff0c;可以帮助网站应对分布式拒绝服务攻击&#xff0c;有效识别和清理一些恶意的网络流量&#xff0c;为用户提供安全且稳定的网络环境&#xff0c;那么&#xff0c;高防服务器一般都可以抵御哪些网络攻击呢&#xff1f;下面…...

高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数

高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数 在软件开发中,单例模式(Singleton Pattern)是一种常见的设计模式,确保一个类仅有一个实例,并提供一个全局访问点。在多线程环境下,实现单例模式时需要注意线程安全问题,以防止多个线程同时创建实例,导致…...

C++:多态机制详解

目录 一. 多态的概念 1.静态多态&#xff08;编译时多态&#xff09; 二.动态多态的定义及实现 1.多态的构成条件 2.虚函数 3.虚函数的重写/覆盖 4.虚函数重写的一些其他问题 1&#xff09;.协变 2&#xff09;.析构函数的重写 5.override 和 final关键字 1&#…...