|
|
*/pre code.hljs{display:block;overflow-x:auto;padding:1em}code.hljs{padding:3px 5px}.hljs{background:#f3f3f3;color:#444}.hljs-comment{color:#697070}.hljs-punctuation,.hljs-tag{color:#444a}.hljs-tag .hljs-attr,.hljs-tag .hljs-name{color:#444}.hljs-attribute,.hljs-doctag,.hljs-keyword,.hljs-meta .hljs-keyword,.hljs-name,.hljs-selector-tag{font-weight:700}.hljs-deletion,.hljs-number,.hljs-quote,.hljs-selector-class,.hljs-selector-id,.hljs-string,.hljs-template-tag,.hljs-type{color:#800}.hljs-section,.hljs-title{color:#800;font-weight:700}.hljs-link,.hljs-operator,.hljs-regexp,.hljs-selector-attr,.hljs-selector-pseudo,.hljs-symbol,.hljs-template-variable,.hljs-variable{color:#ab5656}.hljs-literal{color:#695}.hljs-addition,.hljs-built_in,.hljs-bullet,.hljs-code{color:#397300}.hljs-meta{color:#1f7199}.hljs-meta .hljs-string{color:#38a}.hljs-emphasis{font-style:italic}.hljs-strong{font-weight:700}code[class*=language-],pre[class*=language-]{color:#000;background:0 0;text-shadow:0 1px #fff;font-family:Consolas,Monaco,'Andale Mono','Ubuntu Mono',monospace;font-size:1em;text-align:left;white-space:pre;word-spacing:normal;word-break:normal;word-wrap:normal;line-height:1.5;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-hyphens:none;-moz-hyphens:none;-ms-hyphens:none;hyphens:none}code[class*=language-] ::-moz-selection,code[class*=language-]::-moz-selection,pre[class*=language-] ::-moz-selection,pre[class*=language-]::-moz-selection{text-shadow:none;background:#b3d4fc}code[class*=language-] ::selection,code[class*=language-]::selection,pre[class*=language-] ::selection,pre[class*=language-]::selection{text-shadow:none;background:#b3d4fc}@media print{code[class*=language-],pre[class*=language-]{text-shadow:none}}pre[class*=language-]{padding:1em;margin:.5em 0;overflow:auto}:not(pre)>code[class*=language-],pre[class*=language-]{background:#f5f2f0}:not(pre)>code[class*=language-]{padding:.1em;border-radius:.3em;white-space:normal}.token.cdata,.token.comment,.token.doctype,.token.prolog{color:#708090}.token.punctuation{color:#999}.token.namespace{opacity:.7}.token.boolean,.token.constant,.token.deleted,.token.number,.token.property,.token.symbol,.token.tag{color:#905}.token.attr-name,.token.builtin,.token.char,.token.inserted,.token.selector,.token.string{color:#690}.language-css .token.string,.style .token.string,.token.entity,.token.operator,.token.url{color:#9a6e3a;background:hsla(0,0%,100%,.5)}.token.atrule,.token.attr-value,.token.keyword{color:#07a}.token.class-name,.token.function{color:#dd4a68}.token.important,.token.regex,.token.variable{color:#e90}.token.bold,.token.important{font-weight:700}.token.italic{font-style:italic}.token.entity{cursor:help}pre[class*=language-].line-numbers{position:relative;padding-left:3.8em;counter-reset:linenumber}pre[class*=language-].line-numbers>code{position:relative;white-space:inherit}.line-numbers .line-numbers-rows{position:absolute;pointer-events:none;top:0;font-size:100%;left:-3.8em;width:3em;letter-spacing:-1px;border-right:1px solid #999;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.line-numbers-rows>span{display:block;counter-increment:linenumber}.line-numbers-rows>span:before{content:counter(linenumber);color:#999;display:block;padding-right:.8em;text-align:right}.code-container:hover .copy-btn,.highlight:hover .copy-btn{opacity:1}.code-container{position:relative}.copy-btn{color:#333;cursor:pointer;line-height:1.6;opacity:0;padding:2px 6px;position:absolute;transition:opacity .2s ease-in-out;background-color:#eee;background-image:linear-gradient(#fcfcfc,#eee);border:1px solid #d5d5d5;border-radius:3px;font-size:.8125em;right:4px;top:8px}code,figure.highlight,kbd,pre{background:var(--highlight-background);color:var(--highlight-foreground)}figure.highlight,pre{line-height:1.6;margin:0 auto 20px}figure.highlight figcaption,pre .caption,pre figcaption{background:var(--highlight-gutter-background);color:var(--highlight-foreground);display:flow-root;font-size:.875em;line-height:1.2;padding:.5em}figure.highlight figcaption a,pre .caption a,pre figcaption a{color:var(--highlight-foreground);float:right}figure.highlight figcaption a:hover,pre .caption a:hover,pre figcaption a:hover{border-bottom-color:var(--highlight-foreground)}code,pre{font-family:consolas,Menlo,monospace,'PingFang SC','Microsoft YaHei'}code{border-radius:3px;font-size:.875em;padding:2px 4px;overflow-wrap:break-word}kbd{border:2px solid #ccc;border-radius:.2em;box-shadow:.1em .1em .2em rgba(0,0,0,.1);font-family:inherit;padding:.1em .3em;white-space:nowrap}figure.highlight{overflow:auto;position:relative}figure.highlight pre{border:0;margin:0;padding:10px 0}figure.highlight table{border:0;margin:0;width:auto}figure.highlight td{border:0;padding:0}figure.highlight .gutter{-moz-user-select:none;-ms-user-select:none;-webkit-user-select:none;user-select:none}figure.highlight .gutter pre{background:var(--highlight-gutter-background);color:var(--highlight-gutter-foreground);padding-left:10px;padding-right:10px;text-align:right}figure.highlight .code pre{padding-left:10px;width:100%}figure.highlight .marked{background:rgba(0,0,0,.3)}pre .caption,pre figcaption{margin-bottom:10px}.gist table{width:auto}.gist table td{border:0}pre code{background:0 0;padding:0;text-shadow:none}.blockquote-center{border-left:0;margin:40px 0;padding:0;position:relative;text-align:center}.blockquote-center::after,.blockquote-center::before{left:0;line-height:1;opacity:.6;position:absolute;width:100%}.blockquote-center::before{border-top:1px solid #ccc;text-align:left;top:-20px;content:'\f10d';font-family:'Font Awesome 6 Free';font-weight:900}.blockquote-center::after{border-bottom:1px solid #ccc;bottom:-20px;text-align:right;content:'\f10e';font-family:'Font Awesome 6 Free';font-weight:900}.blockquote-center div,.blockquote-center p{text-align:center}.group-picture{margin-bottom:20px}.group-picture .group-picture-row{display:flex;gap:3px;margin-bottom:3px}.group-picture .group-picture-column{flex:1}.group-picture .group-picture-column img{height:100%;margin:0;object-fit:cover;width:100%}.post-body .label{color:#555;padding:0 2px}.post-body .label.default{background:#f0f0f0}.post-body .label.primary{background:#efe6f7}.post-body .label.info{background:#e5f2f8}.post-body .label.success{background:#e7f4e9}.post-body .label.warning{background:#fcf6e1}.post-body .label.danger{background:#fae8eb}.post-body .link-grid{display:grid;grid-gap:1.5rem;gap:1.5rem;grid-template-columns:1fr 1fr;margin-bottom:20px;padding:1rem}.post-body .link-grid .link-grid-container{border:solid #ddd;box-shadow:1rem 1rem .5rem rgba(0,0,0,.5);min-height:5rem;min-width:0;padding:.5rem;position:relative;transition:background .3s}.post-body .link-grid .link-grid-container:hover{animation:.5s next-shake;background:var(--card-bg-color)}.post-body .link-grid .link-grid-container:active{box-shadow:.5rem .5rem .25rem rgba(0,0,0,.5);transform:translate(.2rem,.2rem)}.post-body .link-grid .link-grid-container .link-grid-image{border:1px solid #ddd;border-radius:50%;box-sizing:border-box;height:5rem;padding:3px;position:absolute;width:5rem}.post-body .link-grid .link-grid-container p{margin:0 1rem 0 6rem}.post-body .link-grid .link-grid-container p:first-of-type{font-size:1.2em}.post-body .link-grid .link-grid-container p:last-of-type{font-size:.8em;line-height:1.3rem;opacity:.7}.post-body .link-grid .link-grid-container a{border:0;height:100%;left:0;position:absolute;top:0;width:100%}@keyframes next-shake{0%{transform:translate(1pt,1pt) rotate(0)}10%{transform:translate(-1pt,-2pt) rotate(-1deg)}20%{transform:translate(-3pt,0) rotate(1deg)}30%{transform:translate(3pt,2pt) rotate(0)}40%{transform:translate(1pt,-1pt) rotate(1deg)}50%{transform:translate(-1pt,2pt) rotate(-1deg)}60%{transform:translate(-3pt,1pt) rotate(0)}70%{transform:translate(3pt,1pt) rotate(-1deg)}80%{transform:translate(-1pt,-1pt) rotate(1deg)}90%{transform:translate(1pt,2pt) rotate(0)}100%{transform:translate(1pt,-2pt) rotate(-1deg)}}.post-body .note{border-radius:3px;margin-bottom:20px;padding:1em;position:relative;border:1px solid #eee;border-left-width:5px}.post-body .note summary{cursor:pointer;outline:0}.post-body .note summary p{display:inline}.post-body .note h2,.post-body .note h3,.post-body .note h4,.post-body .note h5,.post-body .note h6{border-bottom:initial;margin:0;padding-top:0}.post-body .note :first-child{margin-top:0}.post-body .note :last-child{margin-bottom:0}.post-body .note.default{border-left-color:#777}.post-body .note.default h2,.post-body .note.default h3,.post-body .note.default h4,.post-body .note.default h5,.post-body .note.default h6{color:#777}.post-body .note.primary{border-left-color:#6f42c1}.post-body .note.primary h2,.post-body .note.primary h3,.post-body .note.primary h4,.post-body .note.primary h5,.post-body .note.primary h6{color:#6f42c1}.post-body .note.info{border-left-color:#428bca}.post-body .note.info h2,.post-body .note.info h3,.post-body .note.info h4,.post-body .note.info h5,.post-body .note.info h6{color:#428bca}.post-body .note.success{border-left-color:#5cb85c}.post-body .note.success h2,.post-body .note.success h3,.post-body .note.success h4,.post-body .note.success h5,.post-body .note.success h6{color:#5cb85c}.post-body .note.warning{border-left-color:#f0ad4e}.post-body .note.warning h2,.post-body .note.warning h3,.post-body .note.warning h4,.post-body .note.warning h5,.post-body .note.warning h6{color:#f0ad4e}.post-body .note.danger{border-left-color:#d9534f}.post-body .note.danger h2,.post-body .note.danger h3,.post-body .note.danger h4,.post-body .note.danger h5,.post-body .note.danger h6{color:#d9534f}.post-body .tabs{margin-bottom:20px}.post-body .tabs,.tabs-comment{padding-top:10px}.post-body .tabs ul.nav-tabs,.tabs-comment ul.nav-tabs{background:var(--content-bg-color);display:flex;display:flex;flex-wrap:wrap;justify-content:center;margin:0;padding:0;position:-webkit-sticky;position:sticky;top:0;z-index:5}.post-body .tabs ul.nav-tabs li.tab,.tabs-comment ul.nav-tabs li.tab{border-bottom:1px solid #ddd;border-left:1px solid transparent;border-right:1px solid transparent;border-radius:0;border-top:3px solid transparent;flex-grow:1;list-style-type:none}@media (max-width:413px){.post-body .tabs ul.nav-tabs,.tabs-comment ul.nav-tabs{display:block;margin-bottom:5px}.post-body .tabs ul.nav-tabs li.tab,.tabs-comment ul.nav-tabs li.tab{border-bottom:1px solid transparent;border-left:3px solid transparent;border-right:1px solid transparent;border-top:1px solid transparent;border-radius:0}}.post-body .tabs ul.nav-tabs li.tab a,.tabs-comment ul.nav-tabs li.tab a{border-bottom:initial;display:block;line-height:1.8;padding:.25em .75em;text-align:center;transition:.2s ease-out}.post-body .tabs ul.nav-tabs li.tab a i[class^=fa],.tabs-comment ul.nav-tabs li.tab a i[class^=fa]{width:1.285714285714286em}.post-body .tabs ul.nav-tabs li.tab.active,.tabs-comment ul.nav-tabs li.tab.active{border-color:#fc6423 #ddd transparent}@media (max-width:413px){.post-body .tabs ul.nav-tabs li.tab.active,.tabs-comment ul.nav-tabs li.tab.active{border-color:#ddd #ddd #ddd #fc6423}}.post-body .tabs ul.nav-tabs li.tab.active a,.tabs-comment ul.nav-tabs li.tab.active a{cursor:default}.post-body .tabs .tab-content,.tabs-comment .tab-content{border:1px solid #ddd;border-radius:0;border-top-color:transparent}@media (max-width:413px){.post-body .tabs .tab-content,.tabs-comment .tab-content{border-radius:0;border-top-color:#ddd}}.post-body .tabs .tab-content .tab-pane,.tabs-comment .tab-content .tab-pane{padding:20px 20px 0}.post-body .tabs .tab-content .tab-pane:not(.active),.tabs-comment .tab-content .tab-pane:not(.active){display:none}.pagination .next,.pagination .page-number,.pagination .prev,.pagination .space{display:inline-block;margin:-1px 10px 0;padding:0 10px}.algolia-pagination .current .page-number,.pagination .page-number.current{background:#ccc;border-color:#ccc;color:var(--content-bg-color)}.pagination{border-top:1px solid #eee;margin:120px 0 0;text-align:center}.pagination .next,.pagination .page-number,.pagination .prev{border-bottom:0;border-top:1px solid #eee;transition:border-color .2s ease-in-out}.pagination .next:hover,.pagination .page-number:hover,.pagination .prev:hover{border-top-color:var(--link-hover-color)}@media (max-width:767px){.post-body .link-grid{grid-template-columns:1fr}.pagination .next,.pagination .page-number,.pagination .prev,.pagination .space{margin:0 5px}.pagination{border-top:0}.pagination .next,.pagination .page-number,.pagination .prev{border-bottom:1px solid #eee;border-top:0}.pagination .next:hover,.pagination .page-number:hover,.pagination .prev:hover{border-bottom-color:var(--link-hover-color)}.site-meta{text-align:center}}.pagination .space{margin:0;padding:0}.comments{margin-top:60px;overflow:hidden}.comment-button-group{display:flex;display:flex;flex-wrap:wrap;justify-content:center;justify-content:center;margin:1em 0}.comment-button-group .comment-button{margin:.1em .2em}.comment-button-group .comment-button.active{background:var(--btn-default-hover-bg);border-color:var(--btn-default-hover-border-color);color:var(--btn-default-hover-color)}.comment-position{display:none}.comment-position.active{display:block}.tabs-comment{margin-top:4em;padding-top:0}.tabs-comment .comments{margin-top:0;padding-top:0}.headband{background:var(--theme-color);height:3px}@media (max-width:991px){.headband{display:none}}.site-brand-container{display:flex;flex-shrink:0;padding:0 10px}.use-motion .column,.use-motion .site-brand-container .toggle{opacity:0}.site-meta{flex-grow:1;text-align:center}.custom-logo-image{margin-top:20px}@media (max-width:991px){.custom-logo-image{display:none}}.brand{border-bottom:0;color:var(--brand-color);display:inline-block;padding:0}.brand:hover{color:var(--brand-hover-color)}.site-title{font-family:Lato,'PingFang SC','Microsoft YaHei',sans-serif;font-size:1.375em;font-weight:400;line-height:1.5;margin:0}.site-subtitle{color:#ddd;font-size:.8125em;margin:10px 10px 0}.use-motion .custom-logo-image,.use-motion .site-subtitle,.use-motion .site-title{opacity:0;position:relative;top:-10px}.site-nav-right,.site-nav-toggle{display:none}.site-nav-right .toggle,.site-nav-toggle .toggle{color:var(--text-color);padding:10px;width:22px}.site-nav-right .toggle .toggle-line,.site-nav-toggle .toggle .toggle-line{background:var(--text-color);border-radius:1px}@media (max-width:767px){.site-nav-right,.site-nav-toggle{display:flex;flex-direction:column;justify-content:center}.site-nav{--scroll-height:0;height:0;overflow:hidden;transition:height .2s ease-in-out,visibility .2s ease-in-out;visibility:hidden}body:not(.site-nav-on) .site-nav .animated{animation:none}body.site-nav-on .site-nav{height:var(--scroll-height);visibility:unset}}.menu{margin:0;padding:1em 0;text-align:center}.menu-item{display:inline-block;list-style:none;margin:0 10px}@media (max-width:767px){.menu-item{display:block;margin-top:10px}.menu-item.menu-item-search{display:none}}.menu-item a{border-bottom:0;display:block;font-size:.8125em;transition:border-color .2s ease-in-out}.menu-item a.menu-item-active,.menu-item a:hover{background:var(--menu-item-bg-color)}.menu-item i[class^=fa]{margin-right:8px}.menu-item .badge{display:inline-block;font-weight:700;line-height:1;margin-left:.35em;margin-top:.35em;text-align:center;white-space:nowrap}.use-motion .menu-item{visibility:hidden}.github-corner :hover .octo-arm{animation:560ms ease-in-out octocat-wave}.github-corner svg{color:#fff;fill:var(--theme-color);position:absolute;right:0;top:0;z-index:5}@media (max-width:991px){.github-corner{display:none}.github-corner svg{color:var(--theme-color);fill:#fff}.github-corner .github-corner:hover .octo-arm{animation:none}.github-corner .github-corner .octo-arm{animation:560ms ease-in-out octocat-wave}}@keyframes octocat-wave{0%,100%{transform:rotate(0)}20%,60%{transform:rotate(-25deg)}40%,80%{transform:rotate(10deg)}}.sidebar-inner{color:#999;padding:18px 10px;text-align:center;display:flex;flex-direction:column;justify-content:center}.cc-license .cc-opacity{border-bottom:0;opacity:.7}.cc-license .cc-opacity:hover{opacity:.9}.cc-license img{display:inline-block}.site-author-image{border:1px solid #eee;max-width:120px;padding:2px}.site-author-name{color:var(--text-color);font-weight:600;margin:0}.site-description{color:#999;font-size:.8125em;margin-top:0}.links-of-author a{font-size:.8125em}.links-of-author i[class^=fa]{margin-right:2px}.sidebar .sidebar-button:not(:first-child){margin-top:15px}.sidebar .sidebar-button button{background:0 0;cursor:pointer;line-height:2;padding:0 15px;border-radius:4px}.sidebar .sidebar-button button i[class^=fa]{margin-right:5px}.links-of-blogroll{font-size:.8125em}.links-of-blogroll-title{font-size:.875em;font-weight:600}.links-of-blogroll-list{list-style:none;margin:0;padding:0}.sidebar-nav{font-size:.875em;height:0;margin:0;overflow:hidden;padding-left:0;pointer-events:none;transition:height .2s ease-in-out,visibility .2s ease-in-out;visibility:hidden}.sidebar-nav-active .sidebar-nav{height:calc(2em + 1px);pointer-events:unset;visibility:unset}.sidebar-nav li{border-bottom:1px solid transparent;color:var(--text-color);cursor:pointer;display:inline-block;transition:border-bottom-color .2s ease-in-out,color .2s ease-in-out}.sidebar-nav li.sidebar-nav-overview{margin-left:10px}.sidebar-nav li:hover{color:#fc6423}.sidebar-overview-active .sidebar-nav-overview,.sidebar-toc-active .sidebar-nav-toc{border-bottom-color:#fc6423;color:#fc6423;transition-delay:0.2s}.sidebar-overview-active .sidebar-nav-overview:hover,.sidebar-toc-active .sidebar-nav-toc:hover{color:#fc6423}.sidebar-panel-container{align-items:start;display:grid;flex:1;overflow-x:hidden;overflow-y:auto;padding-top:0;transition:padding-top .2s ease-in-out}.sidebar-nav-active .sidebar-panel-container{padding-top:20px}.sidebar-panel{animation:.2s ease-in-out deactivate-sidebar-panel;grid-area:1/1;height:0;opacity:0;overflow:hidden;pointer-events:none;transform:translateY(0);transition:.2s ease-in-out;transition-property:opacity,transform,visibility;visibility:hidden}.sidebar-nav-active .sidebar-panel,.sidebar-overview-active .sidebar-panel.post-toc-wrap{transform:translateY(-20px)}.sidebar-overview-active:not(.sidebar-nav-active) .sidebar-panel.post-toc-wrap{transition-delay:0s,0.2s,0s}.sidebar-overview-active .sidebar-panel.site-overview-wrap,.sidebar-toc-active .sidebar-panel.post-toc-wrap{animation-name:activate-sidebar-panel;height:auto;opacity:1;pointer-events:unset;transform:translateY(0);transition-delay:0.2s,0.2s,0s;visibility:unset}.sidebar-panel.site-overview-wrap{display:flex;flex-direction:column;justify-content:center;gap:10px;justify-content:flex-start}@keyframes deactivate-sidebar-panel{from{height:var(--inactive-panel-height,0)}to{height:var(--active-panel-height,0)}}@keyframes activate-sidebar-panel{from{height:var(--inactive-panel-height,auto)}to{height:var(--active-panel-height,auto)}}.sidebar-toggle{bottom:61px;height:16px;padding:5px;width:16px;background:#222;cursor:pointer;opacity:.6;position:fixed;z-index:30;right:30px}.sidebar-toggle:hover{opacity:.8}@media (max-width:991px){.sidebar-toggle{right:20px;opacity:.8}}.sidebar-toggle:hover .toggle-line{background:#fc6423}@media (any-hover:hover){body:not(.sidebar-active) .sidebar-toggle:hover :first-child{left:50%;top:2px;transform:rotate(45deg);width:50%}body:not(.sidebar-active) .sidebar-toggle:hover :last-child{left:50%;top:-2px;transform:rotate(-45deg);width:50%}}.sidebar-active .sidebar-toggle :nth-child(2){opacity:0}.sidebar-active .sidebar-toggle :first-child{top:6px;transform:rotate(45deg)}.sidebar-active .sidebar-toggle :last-child{top:-6px;transform:rotate(-45deg)}.post-toc{font-size:.875em}.post-toc ol{list-style:none;margin:0;padding:0 2px 0 10px;text-align:left}.post-toc ol>:last-child{margin-bottom:5px}.post-toc ol>ol{padding-left:0}.post-toc ol a{transition:.2s ease-in-out}.post-toc .nav-item{line-height:1.8;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.post-toc .nav .nav-child{--height:0;height:0;opacity:0;overflow:hidden;transition:.2s ease-in-out;visibility:hidden}.post-toc .nav .active>.nav-child{height:var(--height,auto);opacity:1;visibility:unset}.post-toc .nav .active>a{border-bottom-color:#fc6423;color:#fc6423}.post-toc .nav .active-current>a,.post-toc .nav .active-current>a:hover{color:#fc6423}.site-state{display:flex;flex-wrap:wrap;justify-content:center;line-height:1.4}.site-state-item a{border-bottom:0;display:block}.site-state-item-count{display:block;font-size:1em;font-weight:600}.site-state-item-name{color:#999;font-size:.8125em}.sidebar-post-related{font-size:.8125em;padding:18px 0 0}.popular-posts{margin:0;padding:1em 0;text-align:left}.popular-posts .popular-posts-item{display:block}.popular-posts .popular-posts-item .popular-posts-link{border-bottom:0;display:block;padding:5px 20px;transition:background .2s ease-in-out}.popular-posts .popular-posts-item .popular-posts-link:hover{background:var(--menu-item-bg-color)}.popular-posts .popular-posts-item .popular-posts-time{color:#999}.footer{color:#999;font-size:.875em;padding:20px 0;transition:left .2s ease-in-out,right .2s ease-in-out}.footer.footer-fixed{bottom:0;left:0;position:absolute;right:0}.footer-inner{box-sizing:border-box;text-align:center;display:flex;flex-direction:column;justify-content:center;margin:0 auto;width:calc(100% - 20px)}@media (max-width:767px){.menu-item .badge{float:right;margin-left:0}.footer-inner{width:auto}}@media (min-width:1200px){.footer-inner{width:1160px}}@media (min-width:1600px){.footer-inner{width:73%}}.use-motion .footer{opacity:0}.languages{display:inline-block;font-size:1.125em;position:relative}.languages .lang-select-label span{margin:0 .5em}.languages .lang-select{height:100%;left:0;opacity:0;position:absolute;top:0;width:100%}.with-love{color:red;display:inline-block;margin:0 5px}.beian img{display:inline-block;margin:0 3px;vertical-align:middle}.busuanzi-count #busuanzi_container_site_pv,.busuanzi-count #busuanzi_container_site_uv{display:none}@keyframes icon-animate{0%,100%{transform:scale(1)}10%,30%{transform:scale(.9)}20%,40%,50%,60%,70%,80%{transform:scale(1.1)}}@media (max-width:567px){.main-inner{padding:initial!important}.posts-expand .post-header{margin-bottom:10px!important}.post-block{margin-top:initial!important;padding:8px 18px!important}.post-body h1,.post-body h2,.post-body h3,.post-body h4,.post-body h5,.post-body h6{margin:20px 0 8px}.post-body .note h1,.post-body .note h2,.post-body .note h3,.post-body .note h4,.post-body .note h5,.post-body .note h6,.post-body .tabs .tab-content .tab-pane h1,.post-body .tabs .tab-content .tab-pane h2,.post-body .tabs .tab-content .tab-pane h3,.post-body .tabs .tab-content .tab-pane h4,.post-body .tabs .tab-content .tab-pane h5,.post-body .tabs .tab-content .tab-pane h6{margin:0 5px}.post-body>p{margin:0 0 10px}.post-body .note>p,.post-body .tabs .tab-content .tab-pane>p{padding:0 5px}.post-body img,.post-body video{margin-bottom:10px!important}.post-body .fancybox+figcaption,.post-body img+figcaption{margin:-5px auto 15px!important}.post-body .note{margin-bottom:10px!important;padding:10px!important}.post-body .tabs .tab-content .tab-pane{padding:10px 10px 0!important}.post-eof{margin:40px auto 20px!important}.pagination{margin-top:40px}}.back-to-top{font-size:12px;align-items:center;bottom:-100px;color:#fff;display:flex;height:26px;transition:bottom .2s ease-in-out;background:#222;cursor:pointer;opacity:.6;position:fixed;z-index:30;right:30px}.back-to-top span{margin-right:8px;display:none}.back-to-top .fa{text-align:center;width:26px}.back-to-top:hover{opacity:.8;color:#fc6423}.back-to-top.back-to-top-on{bottom:30px}.rtl.post-body a,.rtl.post-body h1,.rtl.post-body h2,.rtl.post-body h3,.rtl.post-body h4,.rtl.post-body h5,.rtl.post-body h6,.rtl.post-body li,.rtl.post-body ol,.rtl.post-body p,.rtl.post-body ul{direction:rtl;font-family:UKIJ Ekran}.rtl.post-title{font-family:UKIJ Ekran}.post-button{margin-top:40px;text-align:center}.use-motion .collection-header,.use-motion .comments,.use-motion .pagination,.use-motion .post-block,.use-motion .post-body,.use-motion .post-header{visibility:hidden}.posts-collapse .post-content{margin-bottom:35px;margin-left:35px;position:relative}@media (max-width:767px){.posts-collapse .post-content{margin-left:0;margin-right:0}}.posts-collapse .post-content .collection-title{font-size:1.125em;position:relative}.posts-collapse .post-content .collection-title::before{background:#999;border:1px solid #fff;margin-left:-6px;margin-top:-4px;position:absolute;top:50%;border-radius:50%;content:' ';height:10px;width:10px}.posts-collapse .post-content .collection-year{font-size:1.5em;font-weight:700;margin:60px 0;position:relative}.posts-collapse .post-content .collection-year::before{background:#bbb;margin-left:-4px;margin-top:-4px;position:absolute;top:50%;border-radius:50%;content:' ';height:8px;width:8px}.posts-collapse .post-content .collection-header{display:block;margin-left:20px}.posts-collapse .post-content .collection-header small{color:#bbb;margin-left:5px}.posts-collapse .post-content .post-header{border-bottom:1px dashed #ccc;margin:30px 2px 0;padding-left:15px;position:relative;transition:border .2s ease-in-out}.posts-collapse .post-content .post-header::before{background:#bbb;border:1px solid #fff;left:-6px;position:absolute;top:.75em;transition:background .2s ease-in-out;border-radius:50%;content:' ';height:6px;width:6px}.posts-collapse .post-content .post-header:hover{border-bottom-color:#666}.posts-collapse .post-content .post-header:hover::before{background:#222}.posts-collapse .post-content .post-meta-container{display:inline;font-size:.75em;margin-right:10px}.posts-collapse .post-content .post-title{display:inline}.posts-collapse .post-content .post-title a{border-bottom:0;color:var(--link-color)}.posts-collapse .post-content::before{background:#f5f5f5;content:' ';height:100%;margin-left:-2px;position:absolute;top:1.25em;width:4px}.post-body{font-family:Lato,'PingFang SC','Microsoft YaHei',sans-serif;overflow-wrap:break-word}@media (min-width:1200px){.post-body{font-size:1.125em}}@media (min-width:992px){.post-body{text-align:justify}}.post-body h1 .header-anchor,.post-body h1 .headerlink,.post-body h2 .header-anchor,.post-body h2 .headerlink,.post-body h3 .header-anchor,.post-body h3 .headerlink,.post-body h4 .header-anchor,.post-body h4 .headerlink,.post-body h5 .header-anchor,.post-body h5 .headerlink,.post-body h6 .header-anchor,.post-body h6 .headerlink{border-bottom-style:none;color:inherit;float:right;font-size:.875em;margin-left:10px;opacity:0}.post-body h1 .header-anchor::before,.post-body h1 .headerlink::before,.post-body h2 .header-anchor::before,.post-body h2 .headerlink::before,.post-body h3 .header-anchor::before,.post-body h3 .headerlink::before,.post-body h4 .header-anchor::before,.post-body h4 .headerlink::before,.post-body h5 .header-anchor::before,.post-body h5 .headerlink::before,.post-body h6 .header-anchor::before,.post-body h6 .headerlink::before{content:'\f0c1';font-family:'Font Awesome 6 Free';font-weight:900}.post-body h1:hover .header-anchor,.post-body h1:hover .headerlink,.post-body h2:hover .header-anchor,.post-body h2:hover .headerlink,.post-body h3:hover .header-anchor,.post-body h3:hover .headerlink,.post-body h4:hover .header-anchor,.post-body h4:hover .headerlink,.post-body h5:hover .header-anchor,.post-body h5:hover .headerlink,.post-body h6:hover .header-anchor,.post-body h6:hover .headerlink{opacity:.5}.post-body h1:hover .header-anchor:hover,.post-body h1:hover .headerlink:hover,.post-body h2:hover .header-anchor:hover,.post-body h2:hover .headerlink:hover,.post-body h3:hover .header-anchor:hover,.post-body h3:hover .headerlink:hover,.post-body h4:hover .header-anchor:hover,.post-body h4:hover .headerlink:hover,.post-body h5:hover .header-anchor:hover,.post-body h5:hover .headerlink:hover,.post-body h6:hover .header-anchor:hover,.post-body h6:hover .headerlink:hover{opacity:1}.post-body .exturl .fa{font-size:.875em;margin-left:4px}.post-body .fancybox+figcaption,.post-body img+figcaption{color:#999;font-size:.875em;font-weight:700;line-height:1;margin:-15px auto 15px;text-align:center}.post-body embed,.post-body iframe,.post-body img,.post-body video{margin-bottom:20px}.post-body .video-container{height:0;margin-bottom:20px;overflow:hidden;padding-top:75%;position:relative;width:100%}.post-body .video-container embed,.post-body .video-container iframe,.post-body .video-container object{height:100%;left:0;margin:0;position:absolute;top:0;width:100%}.post-gallery{display:flex;min-height:200px}.post-gallery .post-gallery-image{flex:1}.post-gallery .post-gallery-image:not(:first-child){clip-path:polygon(40px 0,100% 0,100% 100%,0 100%);margin-left:-20px}.post-gallery .post-gallery-image:not(:last-child){margin-right:-20px}.post-gallery .post-gallery-image img{height:100%;object-fit:cover;opacity:1;width:100%}.posts-expand .post-gallery{margin-bottom:60px}.posts-collapse .post-gallery{margin:15px 0}.posts-expand .post-header{font-size:1.125em;margin-bottom:60px;text-align:center}.posts-expand .post-title{font-size:1.5em;font-weight:400;margin:initial;overflow-wrap:break-word}.posts-expand .post-title-link{border-bottom:0;color:var(--link-color);display:inline-block;position:relative}.posts-expand .post-title-link::before{background:var(--link-color);bottom:0;content:'';height:2px;left:0;position:absolute;transform:scaleX(0);transition:transform .2s ease-in-out;width:100%}.posts-expand .post-title-link:hover::before{transform:scaleX(1)}.posts-expand .post-title-link .fa{font-size:.875em;margin-left:5px}.post-sticky-flag{display:inline-block;margin-right:8px;transform:rotate(30deg)}.posts-expand .post-meta-container{color:#999;font-family:Lato,'PingFang SC','Microsoft YaHei',sans-serif;font-size:.75em;margin-top:3px}.posts-expand .post-meta-container .post-description{font-size:.875em;margin-top:2px}.posts-expand .post-meta-container time{border-bottom:1px dashed #999}.post-meta{display:flex;flex-wrap:wrap;justify-content:center}:not(.post-meta-break)+.post-meta-item::before{content:'|';margin:0 .5em}.post-meta-item-icon{margin-right:3px}@media (max-width:991px){.back-to-top{right:20px;opacity:.8}.post-body{text-align:justify}.post-meta-item-text{display:none}}.post-meta-break{flex-basis:100%;height:0}#busuanzi_container_page_pv{display:none}.post-nav{border-top:1px solid #eee;display:flex;gap:30px;justify-content:space-between;margin-top:1em;padding:10px 5px 0}.post-nav-item{flex:1}.post-nav-item a{border-bottom:0;display:block;font-size:.875em;line-height:1.6}.post-nav-item a:active{top:2px}.post-nav-item .fa{font-size:.75em}.post-nav-item:first-child .fa{margin-right:5px}.post-nav-item:last-child{text-align:right}.post-nav-item:last-child .fa{margin-left:5px}.post-footer{display:flex;flex-direction:column;justify-content:center}.post-eof{background:#ccc;height:1px;margin:80px auto 60px;width:8%}.post-block:last-of-type .post-eof{display:none}.post-copyright ul{list-style:none;overflow:hidden;padding:.5em 1em;position:relative;background:var(--card-bg-color);border-left:3px solid #ff2a2a;margin:1em 0 0}.post-copyright ul::after{content:'\f25e';font-family:'Font Awesome 6 Brands';font-size:200px;opacity:.1;position:absolute;right:-50px;top:-150px}.post-tags{margin-top:40px;text-align:center}.post-tags a{display:inline-block;font-size:.8125em}.post-tags a:not(:last-child){margin-right:10px}.social-like{border-top:1px solid #eee;font-size:.875em;margin-top:1em;padding-top:1em;display:flex;flex-wrap:wrap;justify-content:center}.social-like a{border-bottom:none}.reward-container{margin:1em 0 0;padding:1em 0;text-align:center}.reward-container button{background:0 0;color:#fc6423;cursor:pointer;line-height:2;padding:0 15px;border:2px solid #fc6423;border-radius:2px;outline:0;transition:.2s ease-in-out;vertical-align:text-top}.reward-container button:hover{background:#fc6423;color:#fff}.post-reward{display:none;padding-top:20px}.post-reward.active{display:block}.post-reward div{display:inline-block}.post-reward div span{display:block}.post-reward img{display:inline-block;margin:.8em 2em 0;max-width:100%;width:180px}@keyframes next-roll{from{transform:rotateZ(30deg)}to{transform:rotateZ(-30deg)}}.category-all-page .category-all-title{text-align:center}.category-all-page .category-all{margin-top:20px}.category-all-page .category-list{list-style:none;margin:0;padding:0}.category-all-page .category-list-item{margin:5px 10px}.category-all-page .category-list-count{color:#bbb}.category-all-page .category-list-count::before{content:' ('}.category-all-page .category-list-count::after{content:') '}.category-all-page .category-list-child{padding-left:10px}.event-list hr{background:#222;margin:20px 0 45px}.event-list hr::after{background:#222;color:#fff;content:'NOW';display:inline-block;font-weight:700;padding:0 5px}.event-list .event{--event-background:#222;--event-foreground:#bbb;--event-title:#fff;background:var(--event-background);padding:15px}.event-list .event .event-summary{border-bottom:0;color:var(--event-title);margin:0;padding:0 0 0 35px;position:relative}.event-list .event .event-summary::before{animation:1s ease-in-out infinite alternate dot-flash;background:var(--event-title);left:0;margin-top:-6px;position:absolute;top:50%;border-radius:50%;content:' ';height:12px;width:12px}.event-list .event:nth-of-type(odd) .event-summary::before{animation-delay:.5s}.event-list .event:not(:last-child){margin-bottom:20px}.event-list .event .event-relative-time{color:var(--event-foreground);display:inline-block;font-size:12px;font-weight:400;padding-left:12px}.event-list .event .event-details{color:var(--event-foreground);display:block;line-height:18px;padding:6px 0 6px 35px}.event-list .event .event-details::before{color:var(--event-foreground);display:inline-block;margin-right:9px;width:14px;font-family:'Font Awesome 6 Free';font-weight:900}.event-list .event .event-details.event-location::before{content:'\f041'}.event-list .event .event-details.event-duration::before{content:'\f017'}.event-list .event .event-details.event-description::before{content:'\f024'}.event-list .event-past{--event-background:#f5f5f5;--event-foreground:#999;--event-title:#222}@keyframes dot-flash{from{opacity:1;transform:scale(1)}to{opacity:0;transform:scale(.8)}}ul.breadcrumb{font-size:.75em;list-style:none;margin:1em 0;padding:0 2em;text-align:center}ul.breadcrumb li{display:inline}ul.breadcrumb li:not(:first-child)::before{content:'/\00a0';font-weight:400;padding:.5em}ul.breadcrumb li:last-child{font-weight:700}.tag-cloud{text-align:center}.tag-cloud a{display:inline-block;margin:10px}.tag-cloud-0{border-bottom-color:#aaa;color:#aaa}.tag-cloud-1{border-bottom-color:#9a9a9a;color:#9a9a9a}.tag-cloud-2{border-bottom-color:#8b8b8b;color:#8b8b8b}.tag-cloud-3{border-bottom-color:#7c7c7c;color:#7c7c7c}.tag-cloud-4{border-bottom-color:#6c6c6c;color:#6c6c6c}.tag-cloud-5{border-bottom-color:#5d5d5d;color:#5d5d5d}.tag-cloud-6{border-bottom-color:#4e4e4e;color:#4e4e4e}.tag-cloud-7{border-bottom-color:#3e3e3e;color:#3e3e3e}.tag-cloud-8{border-bottom-color:#2f2f2f;color:#2f2f2f}.tag-cloud-9{border-bottom-color:#202020;color:#202020}.tag-cloud-10{border-bottom-color:#111;color:#111}.search-active{overflow:hidden}.search-pop-overlay{background:rgba(0,0,0,0);display:flex;height:100%;left:0;position:fixed;top:0;transition:visibility .4s,background .4s;visibility:hidden;width:100%;z-index:40}.search-active .search-pop-overlay{background:rgba(0,0,0,.3);visibility:visible}.search-popup{background:var(--card-bg-color);border-radius:5px;height:80%;margin:auto;transform:scale(0);transition:transform .4s;width:700px}.search-active .search-popup{transform:scale(1)}@media (max-width:767px){.search-popup{border-radius:0;height:100%;width:100%}}.search-popup .popup-btn-close,.search-popup .search-icon{color:#999;font-size:18px;padding:0 10px}.search-popup .popup-btn-close{cursor:pointer}.search-popup .popup-btn-close:hover .fa{color:#222}.search-popup .search-header{background:#eee;border-top-left-radius:5px;border-top-right-radius:5px;display:flex;padding:5px}.search-popup input.search-input{background:0 0;border:0;outline:0;width:100%}.search-popup input.search-input::-webkit-search-cancel-button{display:none}.search-popup .search-result-container{height:calc(100% - 55px);overflow:auto;padding:5px 25px}.search-popup .search-result-container hr{margin:5px 0 10px}.search-popup .search-result-container hr:first-child{display:none}.search-popup .search-result-list{margin:0 5px;padding:0}.search-popup a.search-result-title{font-weight:700}.search-popup p.search-result{border-bottom:1px dashed #ccc;padding:5px 0}.search-input-container{flex-grow:1}.search-input-container form{padding:2px}.search-stats{align-items:center;display:flex;justify-content:space-between}.search-stats img{height:1em;margin:0}.algolia-pagination{margin:40px 0;opacity:1;padding:0}.algolia-pagination .pagination-item{display:inline-block}.algolia-pagination .current .page-number{cursor:default}.algolia-pagination .disabled-item{visibility:hidden}.use-motion .animated{animation-fill-mode:none;visibility:inherit}.use-motion .sidebar .animated{animation-fill-mode:both}header.header{background:var(--content-bg-color);border-radius:initial;box-shadow:initial}.main{align-items:stretch;display:flex;justify-content:space-between;margin:0 auto;width:calc(100% - 20px)}@media (max-width:767px){.main{width:auto}}@media (min-width:1200px){.main{width:1160px}}@media (min-width:1600px){.main{width:73%}}@media (max-width:991px){header.header{border-radius:initial}.main{display:block;width:auto}}.main-inner{border-radius:initial;box-sizing:border-box;width:calc(100% - 252px)}.footer-inner{padding-left:252px}@media (max-width:991px){.main-inner{border-radius:initial;width:100%}.footer-inner{padding-left:0;padding-right:0;width:auto}}.column{width:240px}.site-brand-container{background:var(--theme-color)}.site-meta{padding:20px 0}.site-nav-right .toggle,.site-nav-toggle .toggle{color:#fff}.site-nav-right .toggle .toggle-line,.site-nav-toggle .toggle .toggle-line{background:#fff}@media (min-width:768px) and (max-width:991px){.site-nav-right,.site-nav-toggle{display:flex;flex-direction:column;justify-content:center}.site-nav{--scroll-height:0;height:0;overflow:hidden;transition:height .2s ease-in-out,visibility .2s ease-in-out;visibility:hidden}body:not(.site-nav-on) .site-nav .animated{animation:none}body.site-nav-on .site-nav{height:var(--scroll-height);visibility:unset}}.menu .menu-item{display:block;margin:0}.menu .menu-item a{padding:5px 20px;position:relative;text-align:left;transition-property:background-color}.menu .menu-item .badge{background:#ccc;border-radius:10px;color:var(--content-bg-color);float:right;padding:2px 5px;text-shadow:1px 1px 0 rgba(0,0,0,.1)}.main-menu .menu-item-active::after{background:#bbb;border-radius:50%;content:' ';height:6px;margin-top:-3px;position:absolute;right:15px;top:50%;width:6px}.sub-menu{margin:0;padding:6px 0}.sub-menu .menu-item{display:inline-block}.sub-menu .menu-item a{background:0 0;margin:5px 10px;padding:initial}.sub-menu .menu-item a:hover{background:0 0;color:#fc6423}.sub-menu .menu-item-active{border-bottom-color:#fc6423;color:#fc6423}.sub-menu .menu-item-active:hover{border-bottom-color:#fc6423}.sidebar{position:-webkit-sticky;position:sticky;top:12px}@media (max-width:991px){.column{width:auto}.site-nav-on .site-brand-container{box-shadow:0 0 16px rgba(0,0,0,.5)}.menu .menu-item.menu-item-search,.sidebar{display:none}}.sidebar-inner{background:var(--content-bg-color);border-radius:initial;box-shadow:initial;box-sizing:border-box;color:var(--text-color);margin-top:12px;max-height:calc(100vh - 24px);visibility:hidden}.site-state-item{padding:0 10px}.sidebar .sidebar-button{border-bottom:1px dotted #ccc;border-top:1px dotted #ccc}.sidebar .sidebar-button button{border:0;color:#fc6423;display:block;width:100%}.sidebar .sidebar-button button:hover{background:0 0;border:0;color:#e34603}.links-of-author{display:flex;flex-wrap:wrap;justify-content:center}.links-of-author-item{margin:5px 0 0;width:50%}.links-of-author-item a{box-sizing:border-box;max-width:100%;overflow:hidden;padding:0 5px;text-overflow:ellipsis;white-space:nowrap;border-bottom:0;border-radius:4px;display:block}.links-of-author-item a:hover{background:var(--body-bg-color)}.main-inner{background:var(--content-bg-color);box-shadow:initial;padding:40px}@media (max-width:991px){.main-inner{padding:20px}}.sub-menu{border-bottom:1px solid #ddd}.post-block:first-of-type{padding-top:40px}@media (max-width:767px){.pagination{margin-bottom:10px}}</style><link as=style href=https://fonts.googleapis.com/css?family=Lato:300,300italic,400,400italic,700,700italic&display=swap&subset=latin,latin-ext onload=this.rel='stylesheet' rel=preload><link crossorigin=anonymous href=https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.5.0/css/all.min.css integrity=sha256-yIDrPSXHZdOZhAqiBP7CKzIwMQmRCJ8UeB8Jo17YC4o= rel=stylesheet><link crossorigin=anonymous href=https://cdnjs.cloudflare.com/ajax/libs/animate.css/3.1.1/animate.min.css integrity=sha256-PR7ttpcvz8qrF57fur/yAx1qXMFJeJFiA6pSzWi0OIE= rel=stylesheet><link crossorigin=anonymous href=https://cdnjs.cloudflare.com/ajax/libs/fancyapps-ui/5.0.28/fancybox/fancybox.css integrity=sha256-6cQIC71/iBIYXFK+0RHAvwmjwWzkWd+r7v/BX3/vZDc= rel=stylesheet><script class=next-config data-name=main type=application/json>{"hostname":"nicksxs.me","root":"/","images":"/images","scheme":"Pisces","darkmode":false,"version":"8.19.1","exturl":false,"sidebar":{"position":"left","display":"post","padding":18,"offset":12},"copycode":{"enable":true,"style":"default"},"fold":{"enable":false,"height":500},"bookmark":{"enable":false,"color":"#222","save":"auto"},"mediumzoom":false,"lazyload":true,"pangu":false,"comments":{"style":"tabs","active":null,"storage":true,"lazyload":false,"nav":null},"stickytabs":false,"motion":{"enable":true,"async":false,"transition":{"menu_item":"fadeInDown","post_block":"fadeIn","post_header":"fadeInDown","post_body":"fadeInDown","coll_header":"fadeInLeft","sidebar":"fadeInUp"}},"prism":false,"i18n":{"placeholder":"搜索...","empty":"没有找到任何搜索结果:${query}","hits_time":"找到 ${hits} 个搜索结果(用时 ${time} 毫秒)","hits":"找到 ${hits} 个搜索结果"},"algolia":{"appID":"663Q99OQQD","apiKey":"e461a1371d12cec1491c1153b288a9ed","indexName":"nicksxs","hits":{"per_page":10}}}</script><script src=/js/config.js></script><meta content="learn from zero,技术博客,Nicksxs,史学森" name=description><meta content=website property=og:type><meta content="Nicksxs's Blog" property=og:title><meta content=https://nicksxs.me/page/22/index.html property=og:url><meta content="Nicksxs's Blog" property=og:site_name><meta content="learn from zero,技术博客,Nicksxs,史学森" property=og:description><meta content=zh_CN property=og:locale><meta content=Nicksxs property=article:author><meta content=Nicksxs,史学森,米方方,米方方的男朋友,森哥 property=article:tag><meta content=summary name=twitter:card><link href=https://nicksxs.me/page/22/ rel=canonical><script class=next-config data-name=page type=application/json>{"sidebar":"","isHome":true,"isPost":false,"lang":"zh-CN","comments":"","permalink":"","path":"page/22/index.html","title":""}</script><script class=next-config data-name=calendar type=application/json>""</script><title>Nicksxs's Blog - What hurts more, the pain of hard work or the pain of regret?</title><script async src=https://www.googletagmanager.com/gtag/js?id=UA-61358619-1></script><script class=next-config data-name=google_analytics type=application/json>{"tracking_id":"UA-61358619-1","only_pageview":false,"measure_protocol_api_secret":null}</script><script src=/js/third-party/analytics/google-analytics.js></script><script src=/js/third-party/analytics/baidu-analytics.js></script><script async src=https://hm.baidu.com/hm.js?20f33b3c0c0eff9b1522999c0015646d></script><noscript><link href=/css/noscript.css rel=stylesheet></noscript><link title="Nicksxs's Blog" href=/atom.xml rel=alternate type=application/atom+xml><body class=use-motion itemscope itemtype=http://schema.org/WebPage><div class=headband></div><main class=main><div class=column><header class=header itemscope itemtype=http://schema.org/WPHeader><div class=site-brand-container><div class=site-nav-toggle><div aria-label=切换导航栏 class=toggle role=button><span class=toggle-line></span><span class=toggle-line></span><span class=toggle-line></span></div></div><div class=site-meta><a class=brand href=/ rel=start><i class=logo-line></i><h1 class=site-title>Nicksxs's Blog</h1><i class=logo-line></i></a><p class=site-subtitle itemprop=description>What hurts more, the pain of hard work or the pain of regret?</div><div class=site-nav-right><div class="toggle popup-trigger" aria-label=搜索 role=button><i class="fa fa-search fa-fw fa-lg"></i></div></div></div><nav class=site-nav><ul class="main-menu menu"><li class="menu-item menu-item-home"><a href=/ rel=section><i class="fa fa-home fa-fw"></i>首页</a><li class="menu-item menu-item-about"><a href=/about/ rel=section><i class="fa fa-user fa-fw"></i>关于我</a><li class="menu-item menu-item-mirror"><a href=https://nicksxs.com/ rel=section target=_blank><i class="fa fa-user fa-fw"></i>国内镜像</a><li class="menu-item menu-item-tags"><a href=/tags/ rel=section><i class="fa fa-tags fa-fw"></i>标签</a><li class="menu-item menu-item-categories"><a href=/categories/ rel=section><i class="fa fa-th fa-fw"></i>分类</a><li class="menu-item menu-item-archives"><a href=/archives/ rel=section><i class="fa fa-archive fa-fw"></i>归档</a><li class="menu-item menu-item-top"><a href=/top/ rel=section><i class="fa fa-th fa-fw"></i>热度</a><li class="menu-item menu-item-sitemap"><a href=/sitemap.xml rel=section><i class="fa fa-sitemap fa-fw"></i>站点地图</a><li class="menu-item menu-item-commonweal"><a href=/404/ rel=section><i class="fa fa-heartbeat fa-fw"></i>公益 404</a><li class="menu-item menu-item-search"><a class=popup-trigger role=button><i class="fa fa-search fa-fw"></i>搜索</a></ul></nav><div class=search-pop-overlay><div class="popup search-popup"><div class=search-header><span class=search-icon><i class="fa fa-search"></i></span><div class=search-input-container></div><span class=popup-btn-close role=button><i class="fa fa-times-circle"></i></span></div><div class=search-result-container><div class=algolia-stats><hr></div><div class=algolia-hits></div><div class=algolia-pagination></div></div></div></div></header><aside class=sidebar><div class="sidebar-inner sidebar-overview-active"><ul class=sidebar-nav><li class=sidebar-nav-toc>文章目录<li class=sidebar-nav-overview>站点概览</ul><div class=sidebar-panel-container><div class="post-toc-wrap sidebar-panel"></div><div class="site-overview-wrap sidebar-panel"><div class="site-author animated" itemprop=author itemscope itemtype=http://schema.org/Person><img alt=Nicksxs class=site-author-image itemprop=image src=/uploads/avatar.jpg><p class=site-author-name itemprop=name>Nicksxs<div class=site-description itemprop=description>learn from zero,技术博客,Nicksxs,史学森</div></div><div class="site-state-wrap animated"><nav class=site-state><div class="site-state-item site-state-posts"><a href=/archives/><span class=site-state-item-count>341</span> <span class=site-state-item-name>日志</span></a></div><div class="site-state-item site-state-categories"><a href=/categories/><span class=site-state-item-count>175</span> <span class=site-state-item-name>分类</span></a></div><div class="site-state-item site-state-tags"><a href=/tags/><span class=site-state-item-count>307</span> <span class=site-state-item-name>标签</span></a></div></nav></div><div class="links-of-author animated"><span class=links-of-author-item><a rel="noopener me" title="GitHub → https://github.com/nicksxs" href=https://github.com/nicksxs target=_blank><i class="fab fa-github fa-fw"></i>GitHub</a> </span><span class=links-of-author-item><a rel="noopener me" title="E-Mail → mailto:nicksxs1202@gmail.com" href=mailto:nicksxs1202@gmail.com target=_blank><i class="fa fa-envelope fa-fw"></i>E-Mail</a></span></div><div class="cc-license animated" itemprop=license><a class=cc-opacity href=https://creativecommons.org/licenses/by-nc-sa/4.0/ rel=noopener target=_blank><img alt="Creative Commons" src=https://cdnjs.cloudflare.com/ajax/libs/creativecommons-vocabulary/2020.11.3/assets/license_badges/small/by_nc_sa.svg></a></div><div class=post-gallery itemscope itemtype=http://schema.org/ImageGallery style=height:120px;min-height:120px><div class=post-gallery-row><a class="post-gallery-img fancybox" href=https://url.cn/LLWrL7gx itemprop=url itemscope itemtype=http://schema.org/ImageObject rel=gallery_ target=_blank><img alt=腾讯云推广 itemprop=contentUrl src=https://img.nicksxs.com/blog/345X200.jpg style=padding:12px></a></div></div><script charset=utf-8 src=/js/tagcloud.js></script><script charset=utf-8 src=/js/tagcanvas.js></script><div class=widget-wrap><div class="widget tagcloud" id=myCanvasContainer><canvas height=250 id=resCanvas width=250><ul class=tag-list itemprop=keywords><li class=tag-list-item><a class=tag-list-link href=/tags/2019/ rel=tag>2019</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/2020/ rel=tag>2020</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/2021/ rel=tag>2021</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/2022/ rel=tag>2022</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/2023/ rel=tag>2023</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/2PC/ rel=tag>2PC</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/360-%E5%85%A8%E5%AE%B6%E6%A1%B6/ rel=tag>360 全家桶</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/3PC/ rel=tag>3PC</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/3Sum-Closest/ rel=tag>3Sum Closest</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/AOP/ rel=tag>AOP</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Adaptive/ rel=tag>Adaptive</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Apollo/ rel=tag>Apollo</a><span class=tag-list-count>4</span><li class=tag-list-item><a class=tag-list-link href=/tags/AutoConfiguration/ rel=tag>AutoConfiguration</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Binary-Tree/ rel=tag>Binary Tree</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/Broker/ rel=tag>Broker</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/C/ rel=tag>C</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/C/ rel=tag>C++</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/CachedThreadPool/ rel=tag>CachedThreadPool</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Comparator/ rel=tag>Comparator</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/DFS/ rel=tag>DFS</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/DP/ rel=tag>DP</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/DefaultMQPushConsumer/ rel=tag>DefaultMQPushConsumer</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Design-Patterns/ rel=tag>Design Patterns</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Disruptor/ rel=tag>Disruptor</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/Distributed-Lock/ rel=tag>Distributed Lock</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Docker/ rel=tag>Docker</a><span class=tag-list-count>6</span><li class=tag-list-item><a class=tag-list-link href=/tags/Dockerfile/ rel=tag>Dockerfile</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Druid/ rel=tag>Druid</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Dubbo/ rel=tag>Dubbo</a><span class=tag-list-count>6</span><li class=tag-list-item><a class=tag-list-link href=/tags/EagerThreadPool/ rel=tag>EagerThreadPool</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Evict/ rel=tag>Evict</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Filter/ rel=tag>Filter</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/First-Bad-Version/ rel=tag>First Bad Version</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/FixedThreadPool/ rel=tag>FixedThreadPool</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/G1/ rel=tag>G1</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/GC/ rel=tag>GC</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Garbage-First-Collector/ rel=tag>Garbage-First Collector</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Gogs/ rel=tag>Gogs</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Homebrew/ rel=tag>Homebrew</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Inorder-Traversal/ rel=tag>Inorder Traversal</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Interceptor/ rel=tag>Interceptor</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Intersection-of-Two-Arrays/ rel=tag>Intersection of Two Arrays</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/JMap/ rel=tag>JMap</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/JPS/ rel=tag>JPS</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/JStack/ rel=tag>JStack</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/JVM/ rel=tag>JVM</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Java/ rel=tag>Java</a><span class=tag-list-count>76</span><li class=tag-list-item><a class=tag-list-link href=/tags/LLM/ rel=tag>LLM</a><span class=tag-list-count>19</span><li class=tag-list-item><a class=tag-list-link href=/tags/Leetcode-42/ rel=tag>Leetcode 42</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/LimitedThreadPool/ rel=tag>LimitedThreadPool</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Linked-List/ rel=tag>Linked List</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Lowest-Common-Ancestor-of-a-Binary-Tree/ rel=tag>Lowest Common Ancestor of a Binary Tree</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/MQ/ rel=tag>MQ</a><span class=tag-list-count>9</span><li class=tag-list-item><a class=tag-list-link href=/tags/Mac/ rel=tag>Mac</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Maven/ rel=tag>Maven</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Median-of-Two-Sorted-Arrays/ rel=tag>Median of Two Sorted Arrays</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Mybatis/ rel=tag>Mybatis</a><span class=tag-list-count>13</span><li class=tag-list-item><a class=tag-list-link href=/tags/Mysql/ rel=tag>Mysql</a><span class=tag-list-count>13</span><li class=tag-list-item><a class=tag-list-link href=/tags/NameServer/ rel=tag>NameServer</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/PHP/ rel=tag>PHP</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Preorder-Traversal/ rel=tag>Preorder Traversal</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Print-FooBar-Alternately/ rel=tag>Print FooBar Alternately</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/RPC/ rel=tag>RPC</a><span class=tag-list-count>4</span><li class=tag-list-item><a class=tag-list-link href=/tags/Redis/ rel=tag>Redis</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Remove-Duplicates-from-Sorted-List/ rel=tag>Remove Duplicates from Sorted List</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/RocketMQ/ rel=tag>RocketMQ</a><span class=tag-list-count>9</span><li class=tag-list-item><a class=tag-list-link href=/tags/Rotate-Image/ rel=tag>Rotate Image</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Rust/ rel=tag>Rust</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/SPI/ rel=tag>SPI</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Servlet/ rel=tag>Servlet</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Sharding-Jdbc/ rel=tag>Sharding-Jdbc</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/Shift-2D-Grid/ rel=tag>Shift 2D Grid</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Singleton/ rel=tag>Singleton</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Spring/ rel=tag>Spring</a><span class=tag-list-count>7</span><li class=tag-list-item><a class=tag-list-link href=/tags/Spring-Event/ rel=tag>Spring Event</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/SpringBoot/ rel=tag>SpringBoot</a><span class=tag-list-count>22</span><li class=tag-list-item><a class=tag-list-link href=/tags/Sql%E6%B3%A8%E5%85%A5/ rel=tag>Sql注入</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Stream/ rel=tag>Stream</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Synchronized/ rel=tag>Synchronized</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/Thread-dump/ rel=tag>Thread dump</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/ThreadLocal/ rel=tag>ThreadLocal</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/ThreadPool/ rel=tag>ThreadPool</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Tomcat/ rel=tag>Tomcat</a><span class=tag-list-count>13</span><li class=tag-list-item><a class=tag-list-link href=/tags/Trapping-Rain-Water/ rel=tag>Trapping Rain Water</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/WeakReference/ rel=tag>WeakReference</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Web/ rel=tag>Web</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Webhook/ rel=tag>Webhook</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/Windows/ rel=tag>Windows</a><span class=tag-list-count>5</span><li class=tag-list-item><a class=tag-list-link href=/tags/WordPress/ rel=tag>WordPress</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/aqs/ rel=tag>aqs</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/await/ rel=tag>await</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/bloom-filter/ rel=tag>bloom filter</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/c/ rel=tag>c++</a><span class=tag-list-count>14</span><li class=tag-list-item><a class=tag-list-link href=/tags/cglib/ rel=tag>cglib</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/cgroup/ rel=tag>cgroup</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/cluster/ rel=tag>cluster</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/condition/ rel=tag>condition</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/dnsmasq/ rel=tag>dnsmasq</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/docker/ rel=tag>docker</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/dp/ rel=tag>dp</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/dubbo/ rel=tag>dubbo</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/echo/ rel=tag>echo</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/environment/ rel=tag>environment</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/gap-lock/ rel=tag>gap lock</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/gc/ rel=tag>gc</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/git/ rel=tag>git</a><span class=tag-list-count>4</span><li class=tag-list-item><a class=tag-list-link href=/tags/grep/ rel=tag>grep</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/hadoop/ rel=tag>hadoop</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/headscale/ rel=tag>headscale</a><span class=tag-list-count>6</span><li class=tag-list-item><a class=tag-list-link href=/tags/hexo/ rel=tag>hexo</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/http/ rel=tag>http</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/icu4c/ rel=tag>icu4c</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/im/ rel=tag>im</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/is-not-null/ rel=tag>is not null</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/is-null/ rel=tag>is null</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/j-u-c/ rel=tag>j.u.c</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/java/ rel=tag>java</a><span class=tag-list-count>54</span><li class=tag-list-item><a class=tag-list-link href=/tags/jvm/ rel=tag>jvm</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/leetcode/ rel=tag>leetcode</a><span class=tag-list-count>44</span><li class=tag-list-item><a class=tag-list-link href=/tags/leetcode-155/ rel=tag>leetcode 155</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/linked-list/ rel=tag>linked list</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/linux/ rel=tag>linux</a><span class=tag-list-count>7</span><li class=tag-list-item><a class=tag-list-link href=/tags/lock/ rel=tag>lock</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/mac/ rel=tag>mac</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/mfc/ rel=tag>mfc</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/min-stack/ rel=tag>min stack</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/mq/ rel=tag>mq</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/mvcc/ rel=tag>mvcc</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/mysql/ rel=tag>mysql</a><span class=tag-list-count>9</span><li class=tag-list-item><a class=tag-list-link href=/tags/namespace/ rel=tag>namespace</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/nas/ rel=tag>nas</a><span class=tag-list-count>6</span><li class=tag-list-item><a class=tag-list-link href=/tags/next-key-lock/ rel=tag>next-key lock</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/nginx/ rel=tag>nginx</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/nullsfirst/ rel=tag>nullsfirst</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/openresty/ rel=tag>openresty</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/openwrt/ rel=tag>openwrt</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/php/ rel=tag>php</a><span class=tag-list-count>6</span><li class=tag-list-item><a class=tag-list-link href=/tags/powershell/ rel=tag>powershell</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/procedure/ rel=tag>procedure</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/python/ rel=tag>python</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/read-view/ rel=tag>read view</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/redis/ rel=tag>redis</a><span class=tag-list-count>12</span><li class=tag-list-item><a class=tag-list-link href=/tags/scp/ rel=tag>scp</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/signal/ rel=tag>signal</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/sort/ rel=tag>sort</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/spark/ rel=tag>spark</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/ssh/ rel=tag>ssh</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/stack/ rel=tag>stack</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/stream/ rel=tag>stream</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/string/ rel=tag>string</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/swoole/ rel=tag>swoole</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/top/ rel=tag>top</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/trace/ rel=tag>trace</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/uname/ rel=tag>uname</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/unlock/ rel=tag>unlock</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/utf8/ rel=tag>utf8</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/utf8mb4/ rel=tag>utf8mb4</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/utf8mb4-0900-ai-ci/ rel=tag>utf8mb4_0900_ai_ci</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/utf8mb4-general-ci/ rel=tag>utf8mb4_general_ci</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/utf8mb4-unicode-ci/ rel=tag>utf8mb4_unicode_ci</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/value/ rel=tag>value</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/websocket/ rel=tag>websocket</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/windows/ rel=tag>windows</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/wsl/ rel=tag>wsl</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/zookeeper/ rel=tag>zookeeper</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/zsh/ rel=tag>zsh</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%B8%89%E9%98%B6%E6%AE%B5%E6%8F%90%E4%BA%A4/ rel=tag>三阶段提交</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%B8%8D%E5%8F%AF%E5%8F%98%E5%BC%95%E7%94%A8/ rel=tag>不可变引用</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%B8%9C%E4%BA%AC%E5%A5%A5%E8%BF%90%E4%BC%9A/ rel=tag>东京奥运会</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%B8%A4%E9%98%B6%E6%AE%B5%E6%8F%90%E4%BA%A4/ rel=tag>两阶段提交</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%B8%AD%E5%B1%B1%E8%B7%AF/ rel=tag>中山路</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%B8%AD%E5%BA%8F/ rel=tag>中序</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%B8%AD%E9%97%B4%E4%BB%B6/ rel=tag>中间件</a><span class=tag-list-count>4</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%B8%BE%E9%87%8D/ rel=tag>举重</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%B9%92%E4%B9%93%E7%90%83/ rel=tag>乒乓球</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%BA%8B%E5%8A%A1/ rel=tag>事务</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%BA%8C%E5%8F%89%E6%A0%91/ rel=tag>二叉树</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%BA%91%E6%9C%8D%E5%8A%A1%E5%99%A8/ rel=tag>云服务器</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%BA%92%E6%96%A5%E9%94%81/ rel=tag>互斥锁</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%BB%A3%E7%A0%81%E9%A2%98%E8%A7%A3/ rel=tag>代码题解</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E4%BF%AE%E7%94%B5%E8%84%91%E7%9A%84/ rel=tag>修电脑的</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%81%8F%E5%90%91%E9%94%81/ rel=tag>偏向锁</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%81%A5%E5%BA%B7%E7%A0%81/ rel=tag>健康码</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%85%AC%E4%BA%A4/ rel=tag>公交</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%85%AC%E4%BA%A4%E8%BD%A6/ rel=tag>公交车</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%86%85%E5%AD%98%E5%88%86%E5%B8%83/ rel=tag>内存分布</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%86%85%E5%AD%98%E6%B3%84%E6%BC%8F/ rel=tag>内存泄漏</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%87%86%E5%A4%87/ rel=tag>准备</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%87%8F%E8%82%A5/ rel=tag>减肥</a><span class=tag-list-count>7</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%88%86%E5%B8%83%E5%BC%8F%E4%BA%8B%E5%8A%A1/ rel=tag>分布式事务</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%88%86%E5%B8%83%E5%BC%8F%E9%94%81/ rel=tag>分布式锁</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%88%87%E7%89%87/ rel=tag>切片</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%88%9D%E5%A7%8B%E5%8C%96/ rel=tag>初始化</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%89%8A%E5%B3%B0%E5%A1%AB%E8%B0%B7/ rel=tag>削峰填谷</a><span class=tag-list-count>4</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%89%8D%E5%BA%8F/ rel=tag>前序</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%8A%A0%E5%A1%9E/ rel=tag>加塞</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%8A%A0%E8%BD%BD/ rel=tag>加载</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%8D%95%E4%BE%8B/ rel=tag>单例</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%8D%9A%E5%AE%A2%EF%BC%8C%E6%96%87%E7%AB%A0/ rel=tag>博客,文章</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%8E%A6%E9%97%A8/ rel=tag>厦门</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%8F%8C%E4%BA%B2%E5%A7%94%E6%B4%BE/ rel=tag>双亲委派</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%8F%91%E8%A1%8C%E7%89%88/ rel=tag>发行版</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%8F%A3%E7%BD%A9/ rel=tag>口罩</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%8F%AF%E5%8F%98%E5%BC%95%E7%94%A8/ rel=tag>可变引用</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%90%90%E6%A7%BD/ rel=tag>吐槽</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%9B%A4%E7%89%A9%E8%B5%84/ rel=tag>囤物资</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%9E%83%E5%9C%BE%E5%9B%9E%E6%94%B6/ rel=tag>垃圾回收</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%9F%BA%E7%A1%80%E8%AE%BE%E6%96%BD/ rel=tag>基础设施</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%A4%A7%E6%89%AB%E9%99%A4/ rel=tag>大扫除</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%AD%97%E7%AC%A6%E9%9B%86/ rel=tag>字符集</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%AE%89%E5%85%A8/ rel=tag>安全</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%AE%B9%E9%94%99%E6%9C%BA%E5%88%B6/ rel=tag>容错机制</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%AF%84%E7%94%9F%E8%99%AB/ rel=tag>寄生虫</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B0%84%E5%87%BB/ rel=tag>射击</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B0%8F%E6%8A%80%E5%B7%A7/ rel=tag>小技巧</a><span class=tag-list-count>10</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B1%80%E5%8F%A3%E8%A1%97/ rel=tag>局口街</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B7%A5%E5%85%B7/ rel=tag>工具</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B8%83%E9%9A%86%E8%BF%87%E6%BB%A4%E5%99%A8/ rel=tag>布隆过滤器</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B9%B2%E6%B4%BB/ rel=tag>干活</a><span class=tag-list-count>6</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B9%B4%E4%B8%AD%E6%80%BB%E7%BB%93/ rel=tag>年中总结</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B9%B4%E7%BB%88%E6%80%BB%E7%BB%93/ rel=tag>年终总结</a><span class=tag-list-count>4</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B9%B6%E5%8F%91/ rel=tag>并发</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B9%B8%E7%A6%8F%E4%BA%86%E5%90%97/ rel=tag>幸福了吗</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%B9%BB%E8%AF%BB/ rel=tag>幻读</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%BA%94%E7%94%A8/ rel=tag>应用</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%BC%80%E8%BD%A6/ rel=tag>开车</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%BC%B1%E5%BC%95%E7%94%A8/ rel=tag>弱引用</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E5%BD%B1%E8%AF%84/ rel=tag>影评</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%81%B6%E6%84%8F%E7%9B%97%E5%88%B7/ rel=tag>恶意盗刷</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%89%80%E6%9C%89%E6%9D%83/ rel=tag>所有权</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%89%93%E5%8D%A1/ rel=tag>打卡</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%89%B6%E6%A2%AF/ rel=tag>扶梯</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%8A%80%E5%B7%A7/ rel=tag>技巧</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%8A%80%E6%9C%AF/ rel=tag>技术</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%8A%98%E8%85%BE/ rel=tag>折腾</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%8B%96%E6%9B%B4/ rel=tag>拖更</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%8D%A2%E8%BD%A6%E7%89%8C/ rel=tag>换车牌</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%8E%92%E5%BA%8F/ rel=tag>排序</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%8E%A5%E9%9B%A8%E6%B0%B4/ rel=tag>接雨水</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%95%B0%E6%8D%AE%E6%BA%90%E5%8A%A8%E6%80%81%E5%88%87%E6%8D%A2/ rel=tag>数据源动态切换</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%95%B0%E6%8D%AE%E7%BB%93%E6%9E%84/ rel=tag>数据结构</a><span class=tag-list-count>11</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%96%B0%E8%AF%AD%E8%A8%80/ rel=tag>新语言</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%97%85%E6%B8%B8/ rel=tag>旅游</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%97%A5%E5%BF%97/ rel=tag>日志</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%9B%BE%E5%8E%9D%E5%9E%B5/ rel=tag>曾厝垵</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%9C%80%E5%B0%8F%E6%A0%88/ rel=tag>最小栈</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%9D%80%E4%BA%BA%E8%AF%9B%E5%BF%83/ rel=tag>杀人诛心</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%9D%AD%E5%B7%9E/ rel=tag>杭州</a><span class=tag-list-count>4</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%A0%87%E8%AE%B0%E6%95%B4%E7%90%86/ rel=tag>标记整理</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%A4%8D%E7%89%A9%E5%9B%AD/ rel=tag>植物园</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%B2%99%E8%8C%B6%E9%9D%A2/ rel=tag>沙茶面</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%B3%A8%E8%A7%A3/ rel=tag>注解</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%B5%B7%E8%9B%8E%E7%85%8E/ rel=tag>海蛎煎</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%B6%88%E6%81%AF%E9%98%9F%E5%88%97/ rel=tag>消息队列</a><span class=tag-list-count>9</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%B7%98%E6%B1%B0%E7%AD%96%E7%95%A5/ rel=tag>淘汰策略</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%B7%B1%E5%BA%A6%E5%AD%A6%E4%B9%A0/ rel=tag>深度学习</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%BA%90%E7%A0%81/ rel=tag>源码</a><span class=tag-list-count>11</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E6%BA%90%E7%A0%81%E8%A7%A3%E6%9E%90/ rel=tag>源码解析</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%94%9F%E6%B4%BB/ rel=tag>生活</a><span class=tag-list-count>42</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%94%B5%E7%93%B6%E8%BD%A6/ rel=tag>电瓶车</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%96%AB%E6%83%85/ rel=tag>疫情</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%9C%8B%E4%B9%A6/ rel=tag>看书</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%9C%8B%E5%89%A7/ rel=tag>看剧</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%9F%A9%E9%98%B5/ rel=tag>矩阵</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%AB%AF%E5%8F%A3%E8%BD%AC%E5%8F%91/ rel=tag>端口转发</a><span class=tag-list-count>3</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%AE%97%E6%B3%95/ rel=tag>算法</a><span class=tag-list-count>5</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%B1%BB%E5%8A%A0%E8%BD%BD/ rel=tag>类加载</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%B3%9F%E5%BF%83%E4%BA%8B/ rel=tag>糟心事</a><span class=tag-list-count>5</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%B4%A2%E5%BC%95/ rel=tag>索引</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%BA%BF%E7%A8%8B%E6%B1%A0/ rel=tag>线程池</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%BC%93%E5%AD%98/ rel=tag>缓存</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%BC%93%E5%AD%98%E5%87%BB%E7%A9%BF/ rel=tag>缓存击穿</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%BC%93%E5%AD%98%E7%A9%BF%E9%80%8F/ rel=tag>缓存穿透</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%BC%93%E5%AD%98%E9%9B%AA%E5%B4%A9/ rel=tag>缓存雪崩</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%BC%96%E7%A0%81/ rel=tag>编码</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E7%BE%8E%E5%9B%BD/ rel=tag>美国</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%80%81%E7%94%B5%E8%84%91/ rel=tag>老电脑</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%87%AA%E5%8A%A8%E8%A3%85%E9%85%8D/ rel=tag>自动装配</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%87%AA%E6%97%8B/ rel=tag>自旋</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%87%AA%E9%80%82%E5%BA%94%E6%8B%93%E5%B1%95/ rel=tag>自适应拓展</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%A3%85%E7%94%B5%E8%84%91/ rel=tag>装电脑</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%A7%84%E5%88%99/ rel=tag>规则</a><span class=tag-list-count>4</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%A7%A3%E6%9E%90/ rel=tag>解析</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F/ rel=tag>设计模式</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%AF%BB%E4%B9%A6/ rel=tag>读书</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%AF%BB%E5%90%8E%E6%84%9F/ rel=tag>读后感</a><span class=tag-list-count>4</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%B4%9F%E8%BD%BD%E5%9D%87%E8%A1%A1/ rel=tag>负载均衡</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%B6%B3%E7%90%83/ rel=tag>足球</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%B7%91%E6%AD%A5/ rel=tag>跑步</a><span class=tag-list-count>7</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%B7%AF%E6%94%BF%E8%A7%84%E5%88%92/ rel=tag>路政规划</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%B7%AF%E7%94%B1%E5%99%A8/ rel=tag>路由器</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%B7%B3%E6%B0%B4/ rel=tag>跳水</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%B8%A9%E8%B8%8F/ rel=tag>踩踏</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%BD%AC%E4%B9%89/ rel=tag>转义</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%BD%BB%E9%87%8F%E7%BA%A7%E9%94%81/ rel=tag>轻量级锁</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%BF%87%E6%9C%9F%E7%AD%96%E7%95%A5/ rel=tag>过期策略</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%BF%90%E5%8A%A8/ rel=tag>运动</a><span class=tag-list-count>9</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E8%BF%9C%E7%A8%8B%E5%8A%9E%E5%85%AC/ rel=tag>远程办公</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E9%80%92%E5%BD%92/ rel=tag>递归</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E9%87%8D%E9%87%8F%E7%BA%A7%E9%94%81/ rel=tag>重量级锁</a><span class=tag-list-count>2</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E9%93%BE%E6%8E%A5/ rel=tag>链接</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E9%A2%98%E8%A7%A3/ rel=tag>题解</a><span class=tag-list-count>28</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E9%A9%AC%E6%88%8F%E5%9B%A2/ rel=tag>马戏团</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E9%AA%8C%E8%AF%81/ rel=tag>验证</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E9%AA%91%E8%BD%A6/ rel=tag>骑车</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E9%AB%98%E9%80%9F/ rel=tag>高速</a><span class=tag-list-count>1</span><li class=tag-list-item><a class=tag-list-link href=/tags/%E9%BC%93%E6%B5%AA%E5%B1%BF/ rel=tag>鼓浪屿</a><span class=tag-list-count>1</span></ul></canvas></div></div></div></div></div><div class="sidebar-inner sidebar-blogroll"><div class="links-of-blogroll animated"><div class=links-of-blogroll-title><i class="fa fa-globe fa-fw"></i> 链接</div><ul class=links-of-blogroll-list><li class=links-of-blogroll-item><a href=https://covermusic.cn/ rel=noopener target=_blank title=https://covermusic.cn>69伙伴</a></ul></div></div></aside></div><div class="main-inner index posts-expand"><div class=post-block><article class=post-content itemscope itemtype=http://schema.org/Article><link href=https://nicksxs.me/2023/10/21/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%85%AD-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Lifecycle/ itemprop=mainEntityOfPage><span hidden itemprop=author itemscope itemtype=http://schema.org/Person><meta content=/uploads/avatar.jpg itemprop=image><meta content=Nicksxs itemprop=name></span><span hidden itemprop=publisher itemscope itemtype=http://schema.org/Organization><meta content="Nicksxs's Blog" itemprop=name><meta content="learn from zero,技术博客,Nicksxs,史学森" itemprop=description></span><span hidden itemprop=post itemscope itemtype=http://schema.org/CreativeWork><meta content="undefined | Nicksxs's Blog" itemprop=name><meta itemprop=description></span><header class=post-header><h2 itemprop="name headline" class=post-title><a class=post-title-link href=/2023/10/21/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%85%AD-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Lifecycle/ itemprop=url>Tomcat 系列篇六-介绍下 Lifecycle</a></h2><div class=post-meta-container><div class=post-meta><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-calendar"></i> </span><span class=post-meta-item-text>发表于</span> <time itemprop="dateCreated datePublished" title="创建时间:2023-10-21 20:30:32" datetime=2023-10-21T20:30:32+08:00>2023-10-21</time> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-calendar-check"></i> </span><span class=post-meta-item-text>更新于</span> <time title="修改时间:2023-11-05 19:09:18" datetime=2023-11-05T19:09:18+08:00 itemprop=dateModified>2023-11-05</time> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-folder"></i> </span><span class=post-meta-item-text>分类于</span> <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/ itemprop=url rel=index><span itemprop=name>Java</span></a> </span>, <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/SpringBoot/ itemprop=url rel=index><span itemprop=name>SpringBoot</span></a> </span>, <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/SpringBoot/Tomcat/ itemprop=url rel=index><span itemprop=name>Tomcat</span></a> </span></span><span class="post-meta-item leancloud_visitors" data-flag-title="Tomcat 系列篇六-介绍下 Lifecycle" id=/2023/10/21/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%85%AD-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Lifecycle/ title=阅读次数><span class=post-meta-item-icon><i class="far fa-eye"></i> </span><span class=post-meta-item-text>阅读次数:</span> <span class=leancloud-visitors-count></span> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-comment"></i> </span><span class=post-meta-item-text>Disqus:</span> <a href=/2023/10/21/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%85%AD-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Lifecycle/#disqus_thread itemprop=discussionUrl title=disqus><span class="post-comments-count disqus-comment-count" data-disqus-identifier=2023/10/21/Tomcat-系列篇六-介绍下-Lifecycle/ itemprop=commentCount></span></a></span></div></div></header><div class=post-body itemprop=articleBody><p>Tomcat 中的很多组件都是继承了LifecycleBase这个抽象类的,包括之前讲过的 connector,server,service,context,host 这些组件都是,我们先来看下接口<br>主体是分为两部分,第一部分是定义状态时间,如 before_init 和 after_init,跟事件监听器的几个方法,添加查询和移除,第二部分是生命周期的相关方法,初始化开始结束与销毁等方法<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br><span class=line>41</span><br><span class=line>42</span><br><span class=line>43</span><br><span class=line>44</span><br><span class=line>45</span><br><span class=line>46</span><br><span class=line>47</span><br><span class=line>48</span><br><span class=line>49</span><br><span class=line>50</span><br><span class=line>51</span><br><span class=line>52</span><br><span class=line>53</span><br><span class=line>54</span><br><span class=line>55</span><br><span class=line>56</span><br><span class=line>57</span><br><span class=line>58</span><br></pre><td class=code><pre><span class=line><span class=keyword>public</span> <span class=keyword>interface</span> <span class="title class_">Lifecycle</span> {</span><br><span class=line></span><br><span class=line></span><br><span class=line> </span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>BEFORE_INIT_EVENT</span> <span class=operator>=</span> <span class=string>"before_init"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>AFTER_INIT_EVENT</span> <span class=operator>=</span> <span class=string>"after_init"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>START_EVENT</span> <span class=operator>=</span> <span class=string>"start"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>BEFORE_START_EVENT</span> <span class=operator>=</span> <span class=string>"before_start"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>AFTER_START_EVENT</span> <span class=operator>=</span> <span class=string>"after_start"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>STOP_EVENT</span> <span class=operator>=</span> <span class=string>"stop"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>BEFORE_STOP_EVENT</span> <span class=operator>=</span> <span class=string>"before_stop"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>AFTER_STOP_EVENT</span> <span class=operator>=</span> <span class=string>"after_stop"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>AFTER_DESTROY_EVENT</span> <span class=operator>=</span> <span class=string>"after_destroy"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>BEFORE_DESTROY_EVENT</span> <span class=operator>=</span> <span class=string>"before_destroy"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>PERIODIC_EVENT</span> <span class=operator>=</span> <span class=string>"periodic"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>CONFIGURE_START_EVENT</span> <span class=operator>=</span> <span class=string>"configure_start"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>final</span> <span class=type>String</span> <span class=variable>CONFIGURE_STOP_EVENT</span> <span class=operator>=</span> <span class=string>"configure_stop"</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">addLifecycleListener</span><span class=params>(LifecycleListener listener)</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> LifecycleListener[] findLifecycleListeners();</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">removeLifecycleListener</span><span class=params>(LifecycleListener listener)</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">init</span><span class=params>()</span> <span class=keyword>throws</span> LifecycleException;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">start</span><span class=params>()</span> <span class=keyword>throws</span> LifecycleException;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">stop</span><span class=params>()</span> <span class=keyword>throws</span> LifecycleException;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">destroy</span><span class=params>()</span> <span class=keyword>throws</span> LifecycleException;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> LifecycleState <span class="title function_">getState</span><span class=params>()</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> String <span class="title function_">getStateName</span><span class=params>()</span>;</span><br><span class=line></span><br><span class=line></span><br><span class=line> <span class=comment>/**</span></span><br><span class=line><span class=comment> * Marker interface used to indicate that the instance should only be used</span></span><br><span class=line><span class=comment> * once. Calling {<span class=doctag>@link</span> #stop()} on an instance that supports this interface</span></span><br><span class=line><span class=comment> * will automatically call {<span class=doctag>@link</span> #destroy()} after {<span class=doctag>@link</span> #stop()}</span></span><br><span class=line><span class=comment> * completes.</span></span><br><span class=line><span class=comment> */</span></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>interface</span> <span class="title class_">SingleUse</span> {</span><br><span class=line> }</span><br><span class=line>}</span><br></pre></table></figure><p>对应的就是 Lifecycle 的状态,可以在 LifecycleState 中看到,这个枚举有两个字段组成,available 表示该状态下这个对应的组件状态是否可用,另一个就是对应的事件,比如 INITIALIZING 这个 state 对应的 BEFORE_INIT_EVENT<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br><span class=line>41</span><br><span class=line>42</span><br><span class=line>43</span><br></pre><td class=code><pre><span class=line><span class=keyword>public</span> <span class=keyword>enum</span> <span class="title class_">LifecycleState</span> {</span><br><span class=line> NEW(<span class=literal>false</span>, <span class=literal>null</span>),</span><br><span class=line> INITIALIZING(<span class=literal>false</span>, Lifecycle.BEFORE_INIT_EVENT),</span><br><span class=line> INITIALIZED(<span class=literal>false</span>, Lifecycle.AFTER_INIT_EVENT),</span><br><span class=line> STARTING_PREP(<span class=literal>false</span>, Lifecycle.BEFORE_START_EVENT),</span><br><span class=line> STARTING(<span class=literal>true</span>, Lifecycle.START_EVENT),</span><br><span class=line> STARTED(<span class=literal>true</span>, Lifecycle.AFTER_START_EVENT),</span><br><span class=line> STOPPING_PREP(<span class=literal>true</span>, Lifecycle.BEFORE_STOP_EVENT),</span><br><span class=line> STOPPING(<span class=literal>false</span>, Lifecycle.STOP_EVENT),</span><br><span class=line> STOPPED(<span class=literal>false</span>, Lifecycle.AFTER_STOP_EVENT),</span><br><span class=line> DESTROYING(<span class=literal>false</span>, Lifecycle.BEFORE_DESTROY_EVENT),</span><br><span class=line> DESTROYED(<span class=literal>false</span>, Lifecycle.AFTER_DESTROY_EVENT),</span><br><span class=line> FAILED(<span class=literal>false</span>, <span class=literal>null</span>);</span><br><span class=line></span><br><span class=line> <span class=keyword>private</span> <span class=keyword>final</span> <span class=type>boolean</span> available;</span><br><span class=line> <span class=keyword>private</span> <span class=keyword>final</span> String lifecycleEvent;</span><br><span class=line></span><br><span class=line> <span class=keyword>private</span> <span class="title function_">LifecycleState</span><span class=params>(<span class=type>boolean</span> available, String lifecycleEvent)</span> {</span><br><span class=line> <span class=built_in>this</span>.available = available;</span><br><span class=line> <span class=built_in>this</span>.lifecycleEvent = lifecycleEvent;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>/**</span></span><br><span class=line><span class=comment> * May the public methods other than property getters/setters and lifecycle</span></span><br><span class=line><span class=comment> * methods be called for a component in this state? It returns</span></span><br><span class=line><span class=comment> * <code>true</code> for any component in any of the following states:</span></span><br><span class=line><span class=comment> * <ul></span></span><br><span class=line><span class=comment> * <li>{<span class=doctag>@link</span> #STARTING}</li></span></span><br><span class=line><span class=comment> * <li>{<span class=doctag>@link</span> #STARTED}</li></span></span><br><span class=line><span class=comment> * <li>{<span class=doctag>@link</span> #STOPPING_PREP}</li></span></span><br><span class=line><span class=comment> * </ul></span></span><br><span class=line><span class=comment> *</span></span><br><span class=line><span class=comment> * <span class=doctag>@return</span> <code>true</code> if the component is available for use,</span></span><br><span class=line><span class=comment> * otherwise <code>false</code></span></span><br><span class=line><span class=comment> */</span></span><br><span class=line> <span class=keyword>public</span> <span class=type>boolean</span> <span class="title function_">isAvailable</span><span class=params>()</span> {</span><br><span class=line> <span class=keyword>return</span> available;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> String <span class="title function_">getLifecycleEvent</span><span class=params>()</span> {</span><br><span class=line> <span class=keyword>return</span> lifecycleEvent;</span><br><span class=line> }</span><br><span class=line>}</span><br></pre></table></figure><p>然后可以看看 Lifecycle 的状态流转示意,下面是从 Lifecycle 的代码注释里拷出来的<blockquote><p>Common interface for component life cycle methods. Catalina components may implement this interface (as well as the appropriate interface(s) for the functionality they support) in order to provide a consistent mechanism to start and stop the component. The valid state transitions for components that support Lifecycle are:</blockquote><figure class="highlight shell"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br></pre><td class=code><pre><span class=line> start()</span><br><span class=line> -----------------------------</span><br><span class=line> | |</span><br><span class=line> | init() |</span><br><span class=line>NEW -»-- INITIALIZING |</span><br><span class=line>| | | | ------------------«-----------------------</span><br><span class=line>| | |auto | | |</span><br><span class=line>| | \|/ start() \|/ \|/ auto auto stop() |</span><br><span class=line>| | INITIALIZED --»-- STARTING_PREP --»- STARTING --»- STARTED --»--- |</span><br><span class=line>| | | | |</span><br><span class=line>| |destroy()| | |</span><br><span class=line>| --»-----«-- ------------------------«-------------------------------- ^</span><br><span class=line>| | | |</span><br><span class=line>| | \|/ auto auto start() |</span><br><span class=line>| | STOPPING_PREP ----»---- STOPPING ------»----- STOPPED -----»-----</span><br><span class=line>| \|/ ^ | ^</span><br><span class=line>| | stop() | | |</span><br><span class=line>| | -------------------------- | |</span><br><span class=line>| | | | |</span><br><span class=line>| | | destroy() destroy() | |</span><br><span class=line>| | FAILED ----»------ DESTROYING ---«----------------- |</span><br><span class=line>| | ^ | |</span><br><span class=line>| | destroy() | |auto |</span><br><span class=line>| --------»----------------- \|/ |</span><br><span class=line>| DESTROYED |</span><br><span class=line>| |</span><br><span class=line>| stop() |</span><br><span class=line>----»-----------------------------»------------------------------</span><br></pre></table></figure><p>Any state can transition to FAILED.<p>Calling start() while a component is in states STARTING_PREP, STARTING or<br>STARTED has no effect.<p>Calling start() while a component is in state NEW will cause init() to be<br>called immediately after the start() method is entered.<p>Calling stop() while a component is in states STOPPING_PREP, STOPPING or<br>STOPPED has no effect.<p>Calling stop() while a component is in state NEW transitions the component<br>to STOPPED. This is typically encountered when a component fails to start and<br>does not start all its sub-components. When the component is stopped, it will<br>try to stop all sub-components - even those it didn’t start.<p>Attempting any other transition will throw LifecycleException.<p>The LifecycleEvents fired during state changes are defined in the methods that trigger the changed. No LifecycleEvents are fired if the attempted transition is not valid.<br>好的项目就是会把这样的示意图画得很好,把状态流转都画得很明确,而不一定要什么好看的作图工具,用字符就可以<br>而在 LifecycleBase类中是把更细节的实现了,<br>init方法就是先判断了状态<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>public</span> <span class=keyword>final</span> <span class=keyword>synchronized</span> <span class=keyword>void</span> <span class="title function_">init</span><span class=params>()</span> <span class=keyword>throws</span> LifecycleException {</span><br><span class=line> <span class=keyword>if</span> (!state.equals(LifecycleState.NEW)) {</span><br><span class=line> <span class=comment>// 只有状态是 new 才可以执行初始化 init</span></span><br><span class=line> invalidTransition(Lifecycle.BEFORE_INIT_EVENT);</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> setStateInternal(LifecycleState.INITIALIZING, <span class=literal>null</span>, <span class=literal>false</span>);</span><br><span class=line> initInternal();</span><br><span class=line> setStateInternal(LifecycleState.INITIALIZED, <span class=literal>null</span>, <span class=literal>false</span>);</span><br><span class=line> } <span class=keyword>catch</span> (Throwable t) {</span><br><span class=line> handleSubClassException(t, <span class=string>"lifecycleBase.initFail"</span>, toString());</span><br><span class=line> }</span><br><span class=line>}</span><br></pre></table></figure><p>在判断状态后就先设置了初始化中这个状态,这里设置的时候是 check 是 false 的,所以直接到设置状态,并且触发事件<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br></pre><td class=code><pre><span class=line><span class=keyword>private</span> <span class=keyword>synchronized</span> <span class=keyword>void</span> <span class="title function_">setStateInternal</span><span class=params>(LifecycleState state, Object data, <span class=type>boolean</span> check)</span></span><br><span class=line> <span class=keyword>throws</span> LifecycleException {</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (log.isDebugEnabled()) {</span><br><span class=line> log.debug(sm.getString(<span class=string>"lifecycleBase.setState"</span>, <span class=built_in>this</span>, state));</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (check) {</span><br><span class=line> <span class=comment>// Must have been triggered by one of the abstract methods (assume</span></span><br><span class=line> <span class=comment>// code in this class is correct)</span></span><br><span class=line> <span class=comment>// null is never a valid state</span></span><br><span class=line> <span class=keyword>if</span> (state == <span class=literal>null</span>) {</span><br><span class=line> invalidTransition(<span class=string>"null"</span>);</span><br><span class=line> <span class=comment>// Unreachable code - here to stop eclipse complaining about</span></span><br><span class=line> <span class=comment>// a possible NPE further down the method</span></span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Any method can transition to failed</span></span><br><span class=line> <span class=comment>// startInternal() permits STARTING_PREP to STARTING</span></span><br><span class=line> <span class=comment>// stopInternal() permits STOPPING_PREP to STOPPING and FAILED to</span></span><br><span class=line> <span class=comment>// STOPPING</span></span><br><span class=line> <span class=keyword>if</span> (!(state == LifecycleState.FAILED ||</span><br><span class=line> (<span class=built_in>this</span>.state == LifecycleState.STARTING_PREP &&</span><br><span class=line> state == LifecycleState.STARTING) ||</span><br><span class=line> (<span class=built_in>this</span>.state == LifecycleState.STOPPING_PREP &&</span><br><span class=line> state == LifecycleState.STOPPING) ||</span><br><span class=line> (<span class=built_in>this</span>.state == LifecycleState.FAILED &&</span><br><span class=line> state == LifecycleState.STOPPING))) {</span><br><span class=line> <span class=comment>// No other transition permitted</span></span><br><span class=line> invalidTransition(state.name());</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=built_in>this</span>.state = state;</span><br><span class=line> <span class=type>String</span> <span class=variable>lifecycleEvent</span> <span class=operator>=</span> state.getLifecycleEvent();</span><br><span class=line> <span class=keyword>if</span> (lifecycleEvent != <span class=literal>null</span>) {</span><br><span class=line> fireLifecycleEvent(lifecycleEvent, data);</span><br><span class=line> }</span><br><span class=line> }</span><br></pre></table></figure><p>触发事件调用了 fireLifecycleEvent<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br></pre><td class=code><pre><span class=line><span class=keyword>protected</span> <span class=keyword>void</span> <span class="title function_">fireLifecycleEvent</span><span class=params>(String type, Object data)</span> {</span><br><span class=line> <span class=type>LifecycleEvent</span> <span class=variable>event</span> <span class=operator>=</span> <span class=keyword>new</span> <span class="title class_">LifecycleEvent</span>(<span class=built_in>this</span>, type, data);</span><br><span class=line> <span class=keyword>for</span> (LifecycleListener listener : lifecycleListeners) {</span><br><span class=line> listener.lifecycleEvent(event);</span><br><span class=line> }</span><br><span class=line>}</span><br></pre></table></figure><p>会调用所有的 listener 将事件发送出去<br>而回到上面 init 方法,第二步的 initInternal 就是 Tomcat 的 Lifecycle 中非常核心的一点了<br>因为上面说到很多组件都是继承了 LifecycleBase 的,实际外部被调用的其实都是 LifecycleBase 的<br>init,start 等方法,内部是由这个抽象基类去设置前后的状态,以及调用 initInternal ,也就是具体<br>实现类的实现方法,譬如 connector 的 initInternal 方法,就是真正处理初始化逻辑的<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br><span class=line>41</span><br><span class=line>42</span><br><span class=line>43</span><br><span class=line>44</span><br><span class=line>45</span><br><span class=line>46</span><br><span class=line>47</span><br><span class=line>48</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>protected</span> <span class=keyword>void</span> <span class="title function_">initInternal</span><span class=params>()</span> <span class=keyword>throws</span> LifecycleException {</span><br><span class=line></span><br><span class=line> <span class=built_in>super</span>.initInternal();</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (protocolHandler == <span class=literal>null</span>) {</span><br><span class=line> <span class=keyword>throw</span> <span class=keyword>new</span> <span class="title class_">LifecycleException</span>(</span><br><span class=line> sm.getString(<span class=string>"coyoteConnector.protocolHandlerInstantiationFailed"</span>));</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Initialize adapter</span></span><br><span class=line> adapter = <span class=keyword>new</span> <span class="title class_">CoyoteAdapter</span>(<span class=built_in>this</span>);</span><br><span class=line> protocolHandler.setAdapter(adapter);</span><br><span class=line> <span class=keyword>if</span> (service != <span class=literal>null</span>) {</span><br><span class=line> protocolHandler.setUtilityExecutor(service.getServer().getUtilityExecutor());</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Make sure parseBodyMethodsSet has a default</span></span><br><span class=line> <span class=keyword>if</span> (<span class=literal>null</span> == parseBodyMethodsSet) {</span><br><span class=line> setParseBodyMethods(getParseBodyMethods());</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (protocolHandler.isAprRequired() && !AprStatus.isInstanceCreated()) {</span><br><span class=line> <span class=keyword>throw</span> <span class=keyword>new</span> <span class="title class_">LifecycleException</span>(sm.getString(<span class=string>"coyoteConnector.protocolHandlerNoAprListener"</span>,</span><br><span class=line> getProtocolHandlerClassName()));</span><br><span class=line> }</span><br><span class=line> <span class=keyword>if</span> (protocolHandler.isAprRequired() && !AprStatus.isAprAvailable()) {</span><br><span class=line> <span class=keyword>throw</span> <span class=keyword>new</span> <span class="title class_">LifecycleException</span>(sm.getString(<span class=string>"coyoteConnector.protocolHandlerNoAprLibrary"</span>,</span><br><span class=line> getProtocolHandlerClassName()));</span><br><span class=line> }</span><br><span class=line> <span class=keyword>if</span> (AprStatus.isAprAvailable() && AprStatus.getUseOpenSSL() &&</span><br><span class=line> protocolHandler <span class=keyword>instanceof</span> AbstractHttp11JsseProtocol) {</span><br><span class=line> AbstractHttp11JsseProtocol<?> jsseProtocolHandler =</span><br><span class=line> (AbstractHttp11JsseProtocol<?>) protocolHandler;</span><br><span class=line> <span class=keyword>if</span> (jsseProtocolHandler.isSSLEnabled() &&</span><br><span class=line> jsseProtocolHandler.getSslImplementationName() == <span class=literal>null</span>) {</span><br><span class=line> <span class=comment>// OpenSSL is compatible with the JSSE configuration, so use it if APR is available</span></span><br><span class=line> jsseProtocolHandler.setSslImplementationName(OpenSSLImplementation.class.getName());</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> protocolHandler.init();</span><br><span class=line> } <span class=keyword>catch</span> (Exception e) {</span><br><span class=line> <span class=keyword>throw</span> <span class=keyword>new</span> <span class="title class_">LifecycleException</span>(</span><br><span class=line> sm.getString(<span class=string>"coyoteConnector.protocolHandlerInitializationFailed"</span>), e);</span><br><span class=line> }</span><br><span class=line>}</span><br></pre></table></figure><p>start 方法也是类似的,但是更复杂,第一步也是状态判断,是不是已经被启动过了,如果还是 new 状态的话就先调用 init,如果已经 failed 启动失败了就直接调用 stop 结束了,如果是非已初始化和非已停止的状态则是错误的状态转变<br>如果是正常情况,那就是先设置成 STARTING_PREP,然后调用实现类的 startInternal,如果启动失败了就调用 stop,此时理论上应该是启动中 STARTING 状态,否则就是异常状态转变,剩下的就是状态启动完的状态,<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br><span class=line>41</span><br><span class=line>42</span><br><span class=line>43</span><br><span class=line>44</span><br><span class=line>45</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>public</span> <span class=keyword>final</span> <span class=keyword>synchronized</span> <span class=keyword>void</span> <span class="title function_">start</span><span class=params>()</span> <span class=keyword>throws</span> LifecycleException {</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (LifecycleState.STARTING_PREP.equals(state) || LifecycleState.STARTING.equals(state) ||</span><br><span class=line> LifecycleState.STARTED.equals(state)) {</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (log.isDebugEnabled()) {</span><br><span class=line> <span class=type>Exception</span> <span class=variable>e</span> <span class=operator>=</span> <span class=keyword>new</span> <span class="title class_">LifecycleException</span>();</span><br><span class=line> log.debug(sm.getString(<span class=string>"lifecycleBase.alreadyStarted"</span>, toString()), e);</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (log.isInfoEnabled()) {</span><br><span class=line> log.info(sm.getString(<span class=string>"lifecycleBase.alreadyStarted"</span>, toString()));</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (state.equals(LifecycleState.NEW)) {</span><br><span class=line> init();</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (state.equals(LifecycleState.FAILED)) {</span><br><span class=line> stop();</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (!state.equals(LifecycleState.INITIALIZED) &&</span><br><span class=line> !state.equals(LifecycleState.STOPPED)) {</span><br><span class=line> invalidTransition(Lifecycle.BEFORE_START_EVENT);</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> setStateInternal(LifecycleState.STARTING_PREP, <span class=literal>null</span>, <span class=literal>false</span>);</span><br><span class=line> startInternal();</span><br><span class=line> <span class=keyword>if</span> (state.equals(LifecycleState.FAILED)) {</span><br><span class=line> <span class=comment>// This is a 'controlled' failure. The component put itself into the</span></span><br><span class=line> <span class=comment>// FAILED state so call stop() to complete the clean-up.</span></span><br><span class=line> stop();</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (!state.equals(LifecycleState.STARTING)) {</span><br><span class=line> <span class=comment>// Shouldn't be necessary but acts as a check that sub-classes are</span></span><br><span class=line> <span class=comment>// doing what they are supposed to.</span></span><br><span class=line> invalidTransition(Lifecycle.AFTER_START_EVENT);</span><br><span class=line> } <span class=keyword>else</span> {</span><br><span class=line> setStateInternal(LifecycleState.STARTED, <span class=literal>null</span>, <span class=literal>false</span>);</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (Throwable t) {</span><br><span class=line> <span class=comment>// This is an 'uncontrolled' failure so put the component into the</span></span><br><span class=line> <span class=comment>// FAILED state and throw an exception.</span></span><br><span class=line> handleSubClassException(t, <span class=string>"lifecycleBase.startFail"</span>, toString());</span><br><span class=line> }</span><br><span class=line>}</span><br></pre></table></figure><p>stop 跟 destroy 也类似的</div><footer class=post-footer><div class=post-eof></div></footer></article></div><div class=post-block><article class=post-content itemscope itemtype=http://schema.org/Article><link href=https://nicksxs.me/2023/10/15/%E8%81%8A%E8%81%8A%E5%AF%B9-FunctionalInterface-%E6%B3%A8%E8%A7%A3%E7%9A%84%E4%B8%80%E4%BA%9B%E7%90%86%E8%A7%A3/ itemprop=mainEntityOfPage><span hidden itemprop=author itemscope itemtype=http://schema.org/Person><meta content=/uploads/avatar.jpg itemprop=image><meta content=Nicksxs itemprop=name></span><span hidden itemprop=publisher itemscope itemtype=http://schema.org/Organization><meta content="Nicksxs's Blog" itemprop=name><meta content="learn from zero,技术博客,Nicksxs,史学森" itemprop=description></span><span hidden itemprop=post itemscope itemtype=http://schema.org/CreativeWork><meta content="undefined | Nicksxs's Blog" itemprop=name><meta itemprop=description></span><header class=post-header><h2 itemprop="name headline" class=post-title><a class=post-title-link href=/2023/10/15/%E8%81%8A%E8%81%8A%E5%AF%B9-FunctionalInterface-%E6%B3%A8%E8%A7%A3%E7%9A%84%E4%B8%80%E4%BA%9B%E7%90%86%E8%A7%A3/ itemprop=url>聊聊对 FunctionalInterface 注解的一些理解</a></h2><div class=post-meta-container><div class=post-meta><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-calendar"></i> </span><span class=post-meta-item-text>发表于</span> <time itemprop="dateCreated datePublished" title="创建时间:2023-10-15 20:13:42 / 修改时间:20:14:55" datetime=2023-10-15T20:13:42+08:00>2023-10-15</time> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-folder"></i> </span><span class=post-meta-item-text>分类于</span> <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/java/ itemprop=url rel=index><span itemprop=name>java</span></a> </span></span><span class="post-meta-item leancloud_visitors" data-flag-title="聊聊对 FunctionalInterface 注解的一些理解" id=/2023/10/15/%E8%81%8A%E8%81%8A%E5%AF%B9-FunctionalInterface-%E6%B3%A8%E8%A7%A3%E7%9A%84%E4%B8%80%E4%BA%9B%E7%90%86%E8%A7%A3/ title=阅读次数><span class=post-meta-item-icon><i class="far fa-eye"></i> </span><span class=post-meta-item-text>阅读次数:</span> <span class=leancloud-visitors-count></span> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-comment"></i> </span><span class=post-meta-item-text>Disqus:</span> <a href=/2023/10/15/%E8%81%8A%E8%81%8A%E5%AF%B9-FunctionalInterface-%E6%B3%A8%E8%A7%A3%E7%9A%84%E4%B8%80%E4%BA%9B%E7%90%86%E8%A7%A3/#disqus_thread itemprop=discussionUrl title=disqus><span class="post-comments-count disqus-comment-count" data-disqus-identifier=2023/10/15/聊聊对-FunctionalInterface-注解的一些理解/ itemprop=commentCount></span></a></span></div></div></header><div class=post-body itemprop=articleBody><p>在看 Tomcat 代码过程中碰到了一个小问题,可以用代码来举例讲一下<br>先定义一个简单的接口<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br></pre><td class=code><pre><span class=line><span class=meta>@FunctionalInterface</span></span><br><span class=line><span class=keyword>public</span> <span class=keyword>interface</span> <span class="title class_">FunctionalInterfaceTest</span> {</span><br><span class=line></span><br><span class=line> String <span class="title function_">getInfo</span><span class=params>()</span>;</span><br><span class=line>}</span><br></pre></table></figure><p>functionalInterface本质还是个接口,所以可以用类实现接口的方式<br>比如像这样<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br></pre><td class=code><pre><span class=line><span class=keyword>public</span> <span class=keyword>class</span> <span class="title class_">FunctionalInterfaceImpl</span> <span class=keyword>implements</span> <span class="title class_">FunctionalInterfaceTest</span>{</span><br><span class=line> <span class=meta>@Override</span></span><br><span class=line> <span class=keyword>public</span> String <span class="title function_">getInfo</span><span class=params>()</span> {</span><br><span class=line> <span class=keyword>return</span> <span class=string>"info"</span>;</span><br><span class=line> }</span><br><span class=line>}</span><br></pre></table></figure><p>但是有下面另外两种方式也是可以实现接口的<br>第一种是 lambda,<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br></pre><td class=code><pre><span class=line><span class=type>FunctionalInterfaceTest</span> <span class=variable>interfaceTest</span> <span class=operator>=</span> () -> <span class=string>"aaa"</span>;</span><br><span class=line>System.out.println(interfaceTest.getInfo());</span><br></pre></table></figure><p>这个实现方式就是直接把 lambda 表达式作为接口实现,但是如果这样就要注意如果实在代码中间的话,debug 还是会跳到这个 lambda 的实现,有时候会混乱,因为只是跳转到 lambda 表达式,而不是所在的方法<br>第二种是这样<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br></pre><td class=code><pre><span class=line><span class=keyword>public</span> <span class=keyword>static</span> <span class=keyword>void</span> <span class="title function_">main</span><span class=params>(String[] args)</span> {</span><br><span class=line> <span class=type>FunctionalInterfaceTest</span> <span class=variable>interfaceTest1</span> <span class=operator>=</span> demo.getSelfInfo();</span><br><span class=line> System.out.println(interfaceTest.getInfo());</span><br><span class=line>}</span><br><span class=line></span><br><span class=line><span class=keyword>public</span> FunctionalInterfaceTest <span class="title function_">getSelfInfo</span><span class=params>()</span> {</span><br><span class=line> <span class=keyword>return</span> <span class=built_in>this</span>::info;</span><br><span class=line>}</span><br><span class=line></span><br><span class=line><span class=keyword>public</span> String <span class="title function_">info</span><span class=params>()</span> {</span><br><span class=line> <span class=keyword>return</span> <span class=string>"ccc"</span>;</span><br><span class=line>}</span><br></pre></table></figure><p>前面 <code>getSelfInfo</code> 的返回值就是类里的一个方法,可以用来作为 FunctionalInterface 的实现,这里跟上面有个比较奇怪的,或者说对于常规的接口理解,一般来说实现的方法签名必须和接口的方法签名一致,但是对于函数接口,私域就突破了这个限制,不管是lambda 这种匿名函数或者后面这种方法引用,都没有用 <code>getInfo</code> 这个方法名,这也是一个比较最近的一个理解。</div><footer class=post-footer><div class=post-eof></div></footer></article></div><div class=post-block><article class=post-content itemscope itemtype=http://schema.org/Article><link href=https://nicksxs.me/2023/10/07/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%9B%9B-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Service-%E5%90%AF%E5%8A%A8%E8%BF%87%E7%A8%8B/ itemprop=mainEntityOfPage><span hidden itemprop=author itemscope itemtype=http://schema.org/Person><meta content=/uploads/avatar.jpg itemprop=image><meta content=Nicksxs itemprop=name></span><span hidden itemprop=publisher itemscope itemtype=http://schema.org/Organization><meta content="Nicksxs's Blog" itemprop=name><meta content="learn from zero,技术博客,Nicksxs,史学森" itemprop=description></span><span hidden itemprop=post itemscope itemtype=http://schema.org/CreativeWork><meta content="undefined | Nicksxs's Blog" itemprop=name><meta itemprop=description></span><header class=post-header><h2 itemprop="name headline" class=post-title><a class=post-title-link href=/2023/10/07/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%9B%9B-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Service-%E5%90%AF%E5%8A%A8%E8%BF%87%E7%A8%8B/ itemprop=url>Tomcat 系列篇五-介绍下 Service 启动过程</a></h2><div class=post-meta-container><div class=post-meta><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-calendar"></i> </span><span class=post-meta-item-text>发表于</span> <time itemprop="dateCreated datePublished" title="创建时间:2023-10-07 16:03:56 / 修改时间:16:04:46" datetime=2023-10-07T16:03:56+08:00>2023-10-07</time> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-folder"></i> </span><span class=post-meta-item-text>分类于</span> <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/ itemprop=url rel=index><span itemprop=name>Java</span></a> </span>, <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/SpringBoot/ itemprop=url rel=index><span itemprop=name>SpringBoot</span></a> </span>, <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/SpringBoot/Tomcat/ itemprop=url rel=index><span itemprop=name>Tomcat</span></a> </span></span><span class="post-meta-item leancloud_visitors" data-flag-title="Tomcat 系列篇五-介绍下 Service 启动过程" id=/2023/10/07/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%9B%9B-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Service-%E5%90%AF%E5%8A%A8%E8%BF%87%E7%A8%8B/ title=阅读次数><span class=post-meta-item-icon><i class="far fa-eye"></i> </span><span class=post-meta-item-text>阅读次数:</span> <span class=leancloud-visitors-count></span> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-comment"></i> </span><span class=post-meta-item-text>Disqus:</span> <a href=/2023/10/07/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%9B%9B-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Service-%E5%90%AF%E5%8A%A8%E8%BF%87%E7%A8%8B/#disqus_thread itemprop=discussionUrl title=disqus><span class="post-comments-count disqus-comment-count" data-disqus-identifier=2023/10/07/Tomcat-系列篇四-介绍下-Service-启动过程/ itemprop=commentCount></span></a></span></div></div></header><div class=post-body itemprop=articleBody><p>这里开始介绍下 Service 的启动过程,Tomcat 的启动过程中<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br></pre><td class=code><pre><span class=line><span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">start</span><span class=params>()</span> <span class=keyword>throws</span> LifecycleException {</span><br><span class=line> getServer();</span><br><span class=line> server.start();</span><br><span class=line>}</span><br></pre></table></figure><p>getServer之前讲到过<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br></pre><td class=code><pre><span class=line><span class=keyword>public</span> Server <span class="title function_">getServer</span><span class=params>()</span> {</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (server != <span class=literal>null</span>) {</span><br><span class=line> <span class=keyword>return</span> server;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> System.setProperty(<span class=string>"catalina.useNaming"</span>, <span class=string>"false"</span>);</span><br><span class=line></span><br><span class=line> server = <span class=keyword>new</span> <span class="title class_">StandardServer</span>();</span><br><span class=line></span><br><span class=line> initBaseDir();</span><br><span class=line></span><br><span class=line> <span class=comment>// Set configuration source</span></span><br><span class=line> ConfigFileLoader.setSource(<span class=keyword>new</span> <span class="title class_">CatalinaBaseConfigurationSource</span>(<span class=keyword>new</span> <span class="title class_">File</span>(basedir), <span class=literal>null</span>));</span><br><span class=line></span><br><span class=line> server.setPort( -<span class=number>1</span> );</span><br><span class=line></span><br><span class=line> <span class=type>Service</span> <span class=variable>service</span> <span class=operator>=</span> <span class=keyword>new</span> <span class="title class_">StandardService</span>();</span><br><span class=line> service.setName(<span class=string>"Tomcat"</span>);</span><br><span class=line> server.addService(service);</span><br><span class=line> <span class=keyword>return</span> server;</span><br><span class=line>}</span><br></pre></table></figure><p>会 new 一个StandardService,添加到 server 里,然后进行启动<br>而这个外面的 server.start 其实调用的是 <code>org.apache.catalina.Lifecycle#start</code>,<br>里面是一个 Lifecycle 的接口,这个接口被很多 Tomcat 的组件实现,其实都共用了 Lifecycle 的机制<br>然后 Lifecycle 里面会根据状态,调用实际的实现层的 startInternal 方法<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line> <span class=keyword>protected</span> <span class=keyword>void</span> <span class="title function_">startInternal</span><span class=params>()</span> <span class=keyword>throws</span> LifecycleException {</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span>(log.isInfoEnabled()) {</span><br><span class=line> log.info(sm.getString(<span class=string>"standardService.start.name"</span>, <span class=built_in>this</span>.name));</span><br><span class=line> }</span><br><span class=line> setState(LifecycleState.STARTING);</span><br><span class=line></span><br><span class=line> <span class=comment>// Start our defined Container first</span></span><br><span class=line> <span class=keyword>if</span> (engine != <span class=literal>null</span>) {</span><br><span class=line> <span class=keyword>synchronized</span> (engine) {</span><br><span class=line> engine.start();</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>synchronized</span> (executors) {</span><br><span class=line> <span class=keyword>for</span> (Executor executor: executors) {</span><br><span class=line> executor.start();</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> mapperListener.start();</span><br><span class=line></span><br><span class=line> <span class=comment>// Start our defined Connectors second</span></span><br><span class=line> <span class=keyword>synchronized</span> (connectorsLock) {</span><br><span class=line> <span class=keyword>for</span> (Connector connector: connectors) {</span><br><span class=line> <span class=comment>// If it has already failed, don't try and start it</span></span><br><span class=line> <span class=keyword>if</span> (connector.getState() != LifecycleState.FAILED) {</span><br><span class=line> connector.start();</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line> }</span><br></pre></table></figure><p>这里会先设置状态,这个状态也有点东西,可以后面再讲,因为里面有个状态事件的触发<br>然后是启动 Engine,也就是 container,在这里就是 StandardEngine,先不深入去讲里面做了啥,后面讲 StandardEngine 的启动过程会讲,然后是启动线程池了,对于 StandardService 是没有线程池要启动的,或者是 springboot 集成的这个 Tomcat 中不需要,接着就是 mapperListener 的启动,这里其实是给 container 这种添加 listener,用来监听事件,做处理<br>然后就是启动 connector,connector 的启动就是之前说的,里面会启动 protocolHandler,这里还是一样的通过 Lifecycle 的接口,再通过 Lifecycle 的模板方法调用实际的 connector 实现 startInternal 方法,这也是 Tomcat 的一大特点,关于 Tomcat 也是个很大的课题,后面可能还会调整下组织结构,对新同学更友好一点。<br>值得注意的还有两个<br>第一个是添加 connector,先是锁一下,设置 connector 的 Service,然后 connectors 是个数组,这里进行了重新申请一个数组,然后进行拷贝,再把新添加的放到数组最后,<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">addConnector</span><span class=params>(Connector connector)</span> {</span><br><span class=line></span><br><span class=line> <span class=keyword>synchronized</span> (connectorsLock) {</span><br><span class=line> connector.setService(<span class=built_in>this</span>);</span><br><span class=line> Connector results[] = <span class=keyword>new</span> <span class="title class_">Connector</span>[connectors.length + <span class=number>1</span>];</span><br><span class=line> System.arraycopy(connectors, <span class=number>0</span>, results, <span class=number>0</span>, connectors.length);</span><br><span class=line> results[connectors.length] = connector;</span><br><span class=line> connectors = results;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> <span class=keyword>if</span> (getState().isAvailable()) {</span><br><span class=line> connector.start();</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (LifecycleException e) {</span><br><span class=line> <span class=keyword>throw</span> <span class=keyword>new</span> <span class="title class_">IllegalArgumentException</span>(</span><br><span class=line> sm.getString(<span class=string>"standardService.connector.startFailed"</span>, connector), e);</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Report this property change to interested listeners</span></span><br><span class=line> support.firePropertyChange(<span class=string>"connector"</span>, <span class=literal>null</span>, connector);</span><br><span class=line>}</span><br></pre></table></figure><p>然后是判断当前的状态是否可用,如果可用就启动 connector, 最后触发下 connector 的变更事件<br>还有一个是设置Engine,<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br><span class=line>41</span><br><span class=line>42</span><br></pre><td class=code><pre><span class=line></span><br><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">setContainer</span><span class=params>(Engine engine)</span> {</span><br><span class=line> <span class=type>Engine</span> <span class=variable>oldEngine</span> <span class=operator>=</span> <span class=built_in>this</span>.engine;</span><br><span class=line> <span class=keyword>if</span> (oldEngine != <span class=literal>null</span>) {</span><br><span class=line> oldEngine.setService(<span class=literal>null</span>);</span><br><span class=line> }</span><br><span class=line> <span class=built_in>this</span>.engine = engine;</span><br><span class=line> <span class=keyword>if</span> (<span class=built_in>this</span>.engine != <span class=literal>null</span>) {</span><br><span class=line> <span class=built_in>this</span>.engine.setService(<span class=built_in>this</span>);</span><br><span class=line> }</span><br><span class=line> <span class=keyword>if</span> (getState().isAvailable()) {</span><br><span class=line> <span class=keyword>if</span> (<span class=built_in>this</span>.engine != <span class=literal>null</span>) {</span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> <span class=built_in>this</span>.engine.start();</span><br><span class=line> } <span class=keyword>catch</span> (LifecycleException e) {</span><br><span class=line> log.error(sm.getString(<span class=string>"standardService.engine.startFailed"</span>), e);</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line> <span class=comment>// Restart MapperListener to pick up new engine.</span></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> mapperListener.stop();</span><br><span class=line> } <span class=keyword>catch</span> (LifecycleException e) {</span><br><span class=line> log.error(sm.getString(<span class=string>"standardService.mapperListener.stopFailed"</span>), e);</span><br><span class=line> }</span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> mapperListener.start();</span><br><span class=line> } <span class=keyword>catch</span> (LifecycleException e) {</span><br><span class=line> log.error(sm.getString(<span class=string>"standardService.mapperListener.startFailed"</span>), e);</span><br><span class=line> }</span><br><span class=line> <span class=keyword>if</span> (oldEngine != <span class=literal>null</span>) {</span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> oldEngine.stop();</span><br><span class=line> } <span class=keyword>catch</span> (LifecycleException e) {</span><br><span class=line> log.error(sm.getString(<span class=string>"standardService.engine.stopFailed"</span>), e);</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Report this property change to interested listeners</span></span><br><span class=line> support.firePropertyChange(<span class=string>"container"</span>, oldEngine, <span class=built_in>this</span>.engine);</span><br><span class=line>}</span><br></pre></table></figure><p>前面两步是把第一个的 engine 的 service 设置为 null,然后第二个 engine 也就是新的这个的 service 设置成当前的 service,然后也是判断状态,启动 engine,接着是重启 mapperListener,先关闭再启动,最后是关闭老的 engine,最后的最后就是触发 container 变更事件。</div><footer class=post-footer><div class=post-eof></div></footer></article></div><div class=post-block><article class=post-content itemscope itemtype=http://schema.org/Article><link href=https://nicksxs.me/2023/10/01/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%9B%9B-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Valve-%E6%9E%B6%E6%9E%84/ itemprop=mainEntityOfPage><span hidden itemprop=author itemscope itemtype=http://schema.org/Person><meta content=/uploads/avatar.jpg itemprop=image><meta content=Nicksxs itemprop=name></span><span hidden itemprop=publisher itemscope itemtype=http://schema.org/Organization><meta content="Nicksxs's Blog" itemprop=name><meta content="learn from zero,技术博客,Nicksxs,史学森" itemprop=description></span><span hidden itemprop=post itemscope itemtype=http://schema.org/CreativeWork><meta content="undefined | Nicksxs's Blog" itemprop=name><meta itemprop=description></span><header class=post-header><h2 itemprop="name headline" class=post-title><a class=post-title-link href=/2023/10/01/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%9B%9B-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Valve-%E6%9E%B6%E6%9E%84/ itemprop=url>Tomcat 系列篇四-介绍下 Valve 架构</a></h2><div class=post-meta-container><div class=post-meta><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-calendar"></i> </span><span class=post-meta-item-text>发表于</span> <time itemprop="dateCreated datePublished" title="创建时间:2023-10-01 19:54:06" datetime=2023-10-01T19:54:06+08:00>2023-10-01</time> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-calendar-check"></i> </span><span class=post-meta-item-text>更新于</span> <time title="修改时间:2025-10-12 20:06:06" datetime=2025-10-12T20:06:06+08:00 itemprop=dateModified>2025-10-12</time> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-folder"></i> </span><span class=post-meta-item-text>分类于</span> <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/ itemprop=url rel=index><span itemprop=name>Java</span></a> </span>, <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/SpringBoot/ itemprop=url rel=index><span itemprop=name>SpringBoot</span></a> </span>, <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/SpringBoot/Tomcat/ itemprop=url rel=index><span itemprop=name>Tomcat</span></a> </span></span><span class="post-meta-item leancloud_visitors" data-flag-title="Tomcat 系列篇四-介绍下 Valve 架构" id=/2023/10/01/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%9B%9B-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Valve-%E6%9E%B6%E6%9E%84/ title=阅读次数><span class=post-meta-item-icon><i class="far fa-eye"></i> </span><span class=post-meta-item-text>阅读次数:</span> <span class=leancloud-visitors-count></span> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-comment"></i> </span><span class=post-meta-item-text>Disqus:</span> <a href=/2023/10/01/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E5%9B%9B-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Valve-%E6%9E%B6%E6%9E%84/#disqus_thread itemprop=discussionUrl title=disqus><span class="post-comments-count disqus-comment-count" data-disqus-identifier=2023/10/01/Tomcat-系列篇四-介绍下-Valve-架构/ itemprop=commentCount></span></a></span></div></div></header><div class=post-body itemprop=articleBody><p>valve 是 Tomcat 架构中比较重要的一个组成部分,<br>之前说到<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br></pre><td class=code><pre><span class=line>connector.getService().getContainer().getPipeline().getFirst().invoke(</span><br><span class=line> request, response);</span><br></pre></table></figure><p>这段代码是通过 CoyoteAdapter 将请求处理往 container 传,这里就有个 pipeline 机制,这个 pipeline 可以看一下接口<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br></pre><td class=code><pre><span class=line><span class=keyword>public</span> <span class=keyword>interface</span> <span class="title class_">Pipeline</span> <span class=keyword>extends</span> <span class="title class_">Contained</span> {</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> Valve <span class="title function_">getBasic</span><span class=params>()</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">setBasic</span><span class=params>(Valve valve)</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">addValve</span><span class=params>(Valve valve)</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> Valve[] getValves();</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">removeValve</span><span class=params>(Valve valve)</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> Valve <span class="title function_">getFirst</span><span class=params>()</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=type>boolean</span> <span class="title function_">isAsyncSupported</span><span class=params>()</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">findNonAsyncValves</span><span class=params>(Set<String> result)</span>;</span><br><span class=line>}</span><br></pre></table></figure><p>这里可以往 pipeline 里添加 valve,然后看下 valve 的接口<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br></pre><td class=code><pre><span class=line><span class=keyword>public</span> <span class=keyword>interface</span> <span class="title class_">Valve</span> {</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> Valve <span class="title function_">getNext</span><span class=params>()</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">setNext</span><span class=params>(Valve valve)</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">backgroundProcess</span><span class=params>()</span>;</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">invoke</span><span class=params>(Request request, Response response)</span></span><br><span class=line> <span class=keyword>throws</span> IOException, ServletException;</span><br><span class=line></span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class=type>boolean</span> <span class="title function_">isAsyncSupported</span><span class=params>()</span>;</span><br><span class=line>}</span><br></pre></table></figure><p>这里主要看的就是 getNext 跟 setNext,就变成了一个有序的 pipeline,然后就是 invoke 方法,其实 pipeline 是由两部分组成,valve 在其中起到了前后衔接的重要作用,而且可以再 invoke 中进一步串联调用<br><img data-src=https://img.nicksxs.me/blog/Ttnmpu.png><br>图中我们可以看到,对于 container 这个 pipeline,是没设置 first 的,只有 basic,basic 就是个兜底的 valve,在 StandardPipeline 中的 getFirst 实现<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>public</span> Valve <span class="title function_">getFirst</span><span class=params>()</span> {</span><br><span class=line> <span class=keyword>if</span> (first != <span class=literal>null</span>) {</span><br><span class=line> <span class=keyword>return</span> first;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>return</span> basic;</span><br><span class=line>}</span><br></pre></table></figure><p>取不到 first 就降级到 basic,也就是这里的 <code>StandardEngineValve</code>,<br>它的 invoke 我们来看下<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>public</span> <span class=keyword>final</span> <span class=keyword>void</span> <span class="title function_">invoke</span><span class=params>(Request request, Response response)</span></span><br><span class=line> <span class=keyword>throws</span> IOException, ServletException {</span><br><span class=line></span><br><span class=line> <span class=comment>// Select the Host to be used for this Request</span></span><br><span class=line> <span class=type>Host</span> <span class=variable>host</span> <span class=operator>=</span> request.getHost();</span><br><span class=line> <span class=keyword>if</span> (host == <span class=literal>null</span>) {</span><br><span class=line> <span class=comment>// HTTP 0.9 or HTTP 1.0 request without a host when no default host</span></span><br><span class=line> <span class=comment>// is defined.</span></span><br><span class=line> <span class=comment>// Don't overwrite an existing error</span></span><br><span class=line> <span class=keyword>if</span> (!response.isError()) {</span><br><span class=line> response.sendError(<span class=number>404</span>);</span><br><span class=line> }</span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line> <span class=keyword>if</span> (request.isAsyncSupported()) {</span><br><span class=line> request.setAsyncSupported(host.getPipeline().isAsyncSupported());</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Ask this Host to process this request</span></span><br><span class=line> host.getPipeline().getFirst().invoke(request, response);</span><br><span class=line>}</span><br></pre></table></figure><p>比较简单,就是调用 host 中的 pipeline 里的第一个 valve 来处理<br><img data-src=https://img.nicksxs.me/blog/TOBldQ.png><br>第一个是 org.apache.catalina.valves.ErrorReportValve,<br>这里处理的其实是先调用了 next<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br><span class=line>41</span><br><span class=line>42</span><br><span class=line>43</span><br><span class=line>44</span><br><span class=line>45</span><br><span class=line>46</span><br><span class=line>47</span><br><span class=line>48</span><br><span class=line>49</span><br><span class=line>50</span><br><span class=line>51</span><br><span class=line>52</span><br><span class=line>53</span><br><span class=line>54</span><br><span class=line>55</span><br><span class=line>56</span><br><span class=line>57</span><br><span class=line>58</span><br><span class=line>59</span><br><span class=line>60</span><br><span class=line>61</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>void</span> <span class="title function_">invoke</span><span class=params>(Request request, Response response)</span> <span class=keyword>throws</span> IOException, ServletException {</span><br><span class=line></span><br><span class=line> <span class=comment>// Perform the request</span></span><br><span class=line> getNext().invoke(request, response);</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (response.isCommitted()) {</span><br><span class=line> <span class=keyword>if</span> (response.setErrorReported()) {</span><br><span class=line> <span class=comment>// Error wasn't previously reported but we can't write an error</span></span><br><span class=line> <span class=comment>// page because the response has already been committed.</span></span><br><span class=line></span><br><span class=line> <span class=comment>// See if IO is allowed</span></span><br><span class=line> <span class=type>AtomicBoolean</span> <span class=variable>ioAllowed</span> <span class=operator>=</span> <span class=keyword>new</span> <span class="title class_">AtomicBoolean</span>(<span class=literal>true</span>);</span><br><span class=line> response.getCoyoteResponse().action(ActionCode.IS_IO_ALLOWED, ioAllowed);</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (ioAllowed.get()) {</span><br><span class=line> <span class=comment>// I/O is currently still allowed. Flush any data that is</span></span><br><span class=line> <span class=comment>// still to be written to the client.</span></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> response.flushBuffer();</span><br><span class=line> } <span class=keyword>catch</span> (Throwable t) {</span><br><span class=line> ExceptionUtils.handleThrowable(t);</span><br><span class=line> }</span><br><span class=line> <span class=comment>// Now close immediately to signal to the client that</span></span><br><span class=line> <span class=comment>// something went wrong</span></span><br><span class=line> response.getCoyoteResponse().action(ActionCode.CLOSE_NOW,</span><br><span class=line> request.getAttribute(RequestDispatcher.ERROR_EXCEPTION));</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=type>Throwable</span> <span class=variable>throwable</span> <span class=operator>=</span> (Throwable) request.getAttribute(RequestDispatcher.ERROR_EXCEPTION);</span><br><span class=line></span><br><span class=line> <span class=comment>// If an async request is in progress and is not going to end once this</span></span><br><span class=line> <span class=comment>// container thread finishes, do not process any error page here.</span></span><br><span class=line> <span class=keyword>if</span> (request.isAsync() && !request.isAsyncCompleting()) {</span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (throwable != <span class=literal>null</span> && !response.isError()) {</span><br><span class=line> <span class=comment>// Make sure that the necessary methods have been called on the</span></span><br><span class=line> <span class=comment>// response. (It is possible a component may just have set the</span></span><br><span class=line> <span class=comment>// Throwable. Tomcat won't do that but other components might.)</span></span><br><span class=line> <span class=comment>// These are safe to call at this point as we know that the response</span></span><br><span class=line> <span class=comment>// has not been committed.</span></span><br><span class=line> response.reset();</span><br><span class=line> response.sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// One way or another, response.sendError() will have been called before</span></span><br><span class=line> <span class=comment>// execution reaches this point and suspended the response. Need to</span></span><br><span class=line> <span class=comment>// reverse that so this valve can write to the response.</span></span><br><span class=line> response.setSuspended(<span class=literal>false</span>);</span><br><span class=line></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> report(request, response, throwable);</span><br><span class=line> } <span class=keyword>catch</span> (Throwable tt) {</span><br><span class=line> ExceptionUtils.handleThrowable(tt);</span><br><span class=line> }</span><br><span class=line> }</span><br></pre></table></figure><p>虽然是放在 first,实际是先调用 next 的 invoke,也就是 <code>org.apache.catalina.core.StandardHostValve</code> 的invoke 方法<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br><span class=line>41</span><br><span class=line>42</span><br><span class=line>43</span><br><span class=line>44</span><br><span class=line>45</span><br><span class=line>46</span><br><span class=line>47</span><br><span class=line>48</span><br><span class=line>49</span><br><span class=line>50</span><br><span class=line>51</span><br><span class=line>52</span><br><span class=line>53</span><br><span class=line>54</span><br><span class=line>55</span><br><span class=line>56</span><br><span class=line>57</span><br><span class=line>58</span><br><span class=line>59</span><br><span class=line>60</span><br><span class=line>61</span><br><span class=line>62</span><br><span class=line>63</span><br><span class=line>64</span><br><span class=line>65</span><br><span class=line>66</span><br><span class=line>67</span><br><span class=line>68</span><br><span class=line>69</span><br><span class=line>70</span><br><span class=line>71</span><br><span class=line>72</span><br><span class=line>73</span><br><span class=line>74</span><br><span class=line>75</span><br><span class=line>76</span><br><span class=line>77</span><br><span class=line>78</span><br><span class=line>79</span><br><span class=line>80</span><br><span class=line>81</span><br><span class=line>82</span><br><span class=line>83</span><br><span class=line>84</span><br><span class=line>85</span><br><span class=line>86</span><br><span class=line>87</span><br><span class=line>88</span><br><span class=line>89</span><br><span class=line>90</span><br><span class=line>91</span><br><span class=line>92</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>final</span> <span class=keyword>void</span> <span class="title function_">invoke</span><span class=params>(Request request, Response response)</span></span><br><span class=line> <span class=keyword>throws</span> IOException, ServletException {</span><br><span class=line></span><br><span class=line> <span class=comment>// Select the Context to be used for this Request</span></span><br><span class=line> <span class=type>Context</span> <span class=variable>context</span> <span class=operator>=</span> request.getContext();</span><br><span class=line> <span class=keyword>if</span> (context == <span class=literal>null</span>) {</span><br><span class=line> <span class=comment>// Don't overwrite an existing error</span></span><br><span class=line> <span class=keyword>if</span> (!response.isError()) {</span><br><span class=line> response.sendError(<span class=number>404</span>);</span><br><span class=line> }</span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (request.isAsyncSupported()) {</span><br><span class=line> request.setAsyncSupported(context.getPipeline().isAsyncSupported());</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=type>boolean</span> <span class=variable>asyncAtStart</span> <span class=operator>=</span> request.isAsync();</span><br><span class=line></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> context.bind(Globals.IS_SECURITY_ENABLED, MY_CLASSLOADER);</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (!asyncAtStart && !context.fireRequestInitEvent(request.getRequest())) {</span><br><span class=line> <span class=comment>// Don't fire listeners during async processing (the listener</span></span><br><span class=line> <span class=comment>// fired for the request that called startAsync()).</span></span><br><span class=line> <span class=comment>// If a request init listener throws an exception, the request</span></span><br><span class=line> <span class=comment>// is aborted.</span></span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Ask this Context to process this request. Requests that are</span></span><br><span class=line> <span class=comment>// already in error must have been routed here to check for</span></span><br><span class=line> <span class=comment>// application defined error pages so DO NOT forward them to the the</span></span><br><span class=line> <span class=comment>// application for processing.</span></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> <span class=keyword>if</span> (!response.isErrorReportRequired()) {</span><br><span class=line> <span class=comment>// 交给 context 去处理请求了</span></span><br><span class=line> context.getPipeline().getFirst().invoke(request, response);</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (Throwable t) {</span><br><span class=line> ExceptionUtils.handleThrowable(t);</span><br><span class=line> container.getLogger().error(<span class=string>"Exception Processing "</span> + request.getRequestURI(), t);</span><br><span class=line> <span class=comment>// If a new error occurred while trying to report a previous</span></span><br><span class=line> <span class=comment>// error allow the original error to be reported.</span></span><br><span class=line> <span class=keyword>if</span> (!response.isErrorReportRequired()) {</span><br><span class=line> request.setAttribute(RequestDispatcher.ERROR_EXCEPTION, t);</span><br><span class=line> throwable(request, response, t);</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Now that the request/response pair is back under container</span></span><br><span class=line> <span class=comment>// control lift the suspension so that the error handling can</span></span><br><span class=line> <span class=comment>// complete and/or the container can flush any remaining data</span></span><br><span class=line> response.setSuspended(<span class=literal>false</span>);</span><br><span class=line></span><br><span class=line> <span class=type>Throwable</span> <span class=variable>t</span> <span class=operator>=</span> (Throwable) request.getAttribute(RequestDispatcher.ERROR_EXCEPTION);</span><br><span class=line></span><br><span class=line> <span class=comment>// Protect against NPEs if the context was destroyed during a</span></span><br><span class=line> <span class=comment>// long running request.</span></span><br><span class=line> <span class=keyword>if</span> (!context.getState().isAvailable()) {</span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Look for (and render if found) an application level error page</span></span><br><span class=line> <span class=keyword>if</span> (response.isErrorReportRequired()) {</span><br><span class=line> <span class=comment>// If an error has occurred that prevents further I/O, don't waste time</span></span><br><span class=line> <span class=comment>// producing an error report that will never be read</span></span><br><span class=line> <span class=type>AtomicBoolean</span> <span class=variable>result</span> <span class=operator>=</span> <span class=keyword>new</span> <span class="title class_">AtomicBoolean</span>(<span class=literal>false</span>);</span><br><span class=line> response.getCoyoteResponse().action(ActionCode.IS_IO_ALLOWED, result);</span><br><span class=line> <span class=keyword>if</span> (result.get()) {</span><br><span class=line> <span class=keyword>if</span> (t != <span class=literal>null</span>) {</span><br><span class=line> throwable(request, response, t);</span><br><span class=line> } <span class=keyword>else</span> {</span><br><span class=line> status(request, response);</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (!request.isAsync() && !asyncAtStart) {</span><br><span class=line> context.fireRequestDestroyEvent(request.getRequest());</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>finally</span> {</span><br><span class=line> <span class=comment>// Access a session (if present) to update last accessed time, based</span></span><br><span class=line> <span class=comment>// on a strict interpretation of the specification</span></span><br><span class=line> <span class=keyword>if</span> (ACCESS_SESSION) {</span><br><span class=line> request.getSession(<span class=literal>false</span>);</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> context.unbind(Globals.IS_SECURITY_ENABLED, MY_CLASSLOADER);</span><br><span class=line> }</span><br><span class=line> }</span><br></pre></table></figure><p>这里的往下就是调用 context 的 pipeline 去处理请求了, StandardContext 的 pipeline 里的 first 是<br>org.apache.catalina.authenticator.NonLoginAuthenticator,处理认证相关的,然后 basic 就是<br>org.apache.catalina.core.StandardContextValve, 这里的来看下<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>final</span> <span class=keyword>void</span> <span class="title function_">invoke</span><span class=params>(Request request, Response response)</span></span><br><span class=line> <span class=keyword>throws</span> IOException, ServletException {</span><br><span class=line></span><br><span class=line> <span class=comment>// Disallow any direct access to resources under WEB-INF or META-INF</span></span><br><span class=line> <span class=type>MessageBytes</span> <span class=variable>requestPathMB</span> <span class=operator>=</span> request.getRequestPathMB();</span><br><span class=line> <span class=keyword>if</span> ((requestPathMB.startsWithIgnoreCase(<span class=string>"/META-INF/"</span>, <span class=number>0</span>))</span><br><span class=line> || (requestPathMB.equalsIgnoreCase(<span class=string>"/META-INF"</span>))</span><br><span class=line> || (requestPathMB.startsWithIgnoreCase(<span class=string>"/WEB-INF/"</span>, <span class=number>0</span>))</span><br><span class=line> || (requestPathMB.equalsIgnoreCase(<span class=string>"/WEB-INF"</span>))) {</span><br><span class=line> response.sendError(HttpServletResponse.SC_NOT_FOUND);</span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Select the Wrapper to be used for this Request</span></span><br><span class=line> <span class=type>Wrapper</span> <span class=variable>wrapper</span> <span class=operator>=</span> request.getWrapper();</span><br><span class=line> <span class=keyword>if</span> (wrapper == <span class=literal>null</span> || wrapper.isUnavailable()) {</span><br><span class=line> response.sendError(HttpServletResponse.SC_NOT_FOUND);</span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Acknowledge the request</span></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> response.sendAcknowledgement(ContinueResponseTiming.IMMEDIATELY);</span><br><span class=line> } <span class=keyword>catch</span> (IOException ioe) {</span><br><span class=line> container.getLogger().error(sm.getString(</span><br><span class=line> <span class=string>"standardContextValve.acknowledgeException"</span>), ioe);</span><br><span class=line> request.setAttribute(RequestDispatcher.ERROR_EXCEPTION, ioe);</span><br><span class=line> response.sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);</span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>if</span> (request.isAsyncSupported()) {</span><br><span class=line> request.setAsyncSupported(wrapper.getPipeline().isAsyncSupported());</span><br><span class=line> }</span><br><span class=line> wrapper.getPipeline().getFirst().invoke(request, response);</span><br><span class=line> }</span><br></pre></table></figure><p>会调用 wrapper 的 pipeline 去处理请求,这里也只有一个<br><code>org.apache.catalina.core.StandardWrapperValve</code><br>这部分的逻辑比较长,因为要串联后面的 filter 流程<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br><span class=line>41</span><br><span class=line>42</span><br><span class=line>43</span><br><span class=line>44</span><br><span class=line>45</span><br><span class=line>46</span><br><span class=line>47</span><br><span class=line>48</span><br><span class=line>49</span><br><span class=line>50</span><br><span class=line>51</span><br><span class=line>52</span><br><span class=line>53</span><br><span class=line>54</span><br><span class=line>55</span><br><span class=line>56</span><br><span class=line>57</span><br><span class=line>58</span><br><span class=line>59</span><br><span class=line>60</span><br><span class=line>61</span><br><span class=line>62</span><br><span class=line>63</span><br><span class=line>64</span><br><span class=line>65</span><br><span class=line>66</span><br><span class=line>67</span><br><span class=line>68</span><br><span class=line>69</span><br><span class=line>70</span><br><span class=line>71</span><br><span class=line>72</span><br><span class=line>73</span><br><span class=line>74</span><br><span class=line>75</span><br><span class=line>76</span><br><span class=line>77</span><br><span class=line>78</span><br><span class=line>79</span><br><span class=line>80</span><br><span class=line>81</span><br><span class=line>82</span><br><span class=line>83</span><br><span class=line>84</span><br><span class=line>85</span><br><span class=line>86</span><br><span class=line>87</span><br><span class=line>88</span><br><span class=line>89</span><br><span class=line>90</span><br><span class=line>91</span><br><span class=line>92</span><br><span class=line>93</span><br><span class=line>94</span><br><span class=line>95</span><br><span class=line>96</span><br><span class=line>97</span><br><span class=line>98</span><br><span class=line>99</span><br><span class=line>100</span><br><span class=line>101</span><br><span class=line>102</span><br><span class=line>103</span><br><span class=line>104</span><br><span class=line>105</span><br><span class=line>106</span><br><span class=line>107</span><br><span class=line>108</span><br><span class=line>109</span><br><span class=line>110</span><br><span class=line>111</span><br><span class=line>112</span><br><span class=line>113</span><br><span class=line>114</span><br><span class=line>115</span><br><span class=line>116</span><br><span class=line>117</span><br><span class=line>118</span><br><span class=line>119</span><br><span class=line>120</span><br><span class=line>121</span><br><span class=line>122</span><br><span class=line>123</span><br><span class=line>124</span><br><span class=line>125</span><br><span class=line>126</span><br><span class=line>127</span><br><span class=line>128</span><br><span class=line>129</span><br><span class=line>130</span><br><span class=line>131</span><br><span class=line>132</span><br><span class=line>133</span><br><span class=line>134</span><br><span class=line>135</span><br><span class=line>136</span><br><span class=line>137</span><br><span class=line>138</span><br><span class=line>139</span><br><span class=line>140</span><br><span class=line>141</span><br><span class=line>142</span><br><span class=line>143</span><br><span class=line>144</span><br><span class=line>145</span><br><span class=line>146</span><br><span class=line>147</span><br><span class=line>148</span><br><span class=line>149</span><br><span class=line>150</span><br><span class=line>151</span><br><span class=line>152</span><br><span class=line>153</span><br><span class=line>154</span><br><span class=line>155</span><br><span class=line>156</span><br><span class=line>157</span><br><span class=line>158</span><br><span class=line>159</span><br><span class=line>160</span><br><span class=line>161</span><br><span class=line>162</span><br><span class=line>163</span><br><span class=line>164</span><br><span class=line>165</span><br><span class=line>166</span><br><span class=line>167</span><br><span class=line>168</span><br><span class=line>169</span><br><span class=line>170</span><br><span class=line>171</span><br><span class=line>172</span><br><span class=line>173</span><br><span class=line>174</span><br><span class=line>175</span><br><span class=line>176</span><br><span class=line>177</span><br><span class=line>178</span><br><span class=line>179</span><br><span class=line>180</span><br><span class=line>181</span><br><span class=line>182</span><br><span class=line>183</span><br><span class=line>184</span><br><span class=line>185</span><br><span class=line>186</span><br><span class=line>187</span><br><span class=line>188</span><br><span class=line>189</span><br><span class=line>190</span><br><span class=line>191</span><br><span class=line>192</span><br><span class=line>193</span><br><span class=line>194</span><br><span class=line>195</span><br><span class=line>196</span><br><span class=line>197</span><br><span class=line>198</span><br><span class=line>199</span><br><span class=line>200</span><br><span class=line>201</span><br><span class=line>202</span><br><span class=line>203</span><br><span class=line>204</span><br><span class=line>205</span><br><span class=line>206</span><br><span class=line>207</span><br><span class=line>208</span><br><span class=line>209</span><br><span class=line>210</span><br><span class=line>211</span><br><span class=line>212</span><br><span class=line>213</span><br><span class=line>214</span><br><span class=line>215</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line> <span class=keyword>public</span> <span class=keyword>final</span> <span class=keyword>void</span> <span class="title function_">invoke</span><span class=params>(Request request, Response response)</span></span><br><span class=line> <span class=keyword>throws</span> IOException, ServletException {</span><br><span class=line></span><br><span class=line> <span class=comment>// Initialize local variables we may need</span></span><br><span class=line> <span class=type>boolean</span> <span class=variable>unavailable</span> <span class=operator>=</span> <span class=literal>false</span>;</span><br><span class=line> <span class=type>Throwable</span> <span class=variable>throwable</span> <span class=operator>=</span> <span class=literal>null</span>;</span><br><span class=line> <span class=comment>// This should be a Request attribute...</span></span><br><span class=line> <span class=type>long</span> t1=System.currentTimeMillis();</span><br><span class=line> requestCount.incrementAndGet();</span><br><span class=line> <span class=type>StandardWrapper</span> <span class=variable>wrapper</span> <span class=operator>=</span> (StandardWrapper) getContainer();</span><br><span class=line> <span class=type>Servlet</span> <span class=variable>servlet</span> <span class=operator>=</span> <span class=literal>null</span>;</span><br><span class=line> <span class=type>Context</span> <span class=variable>context</span> <span class=operator>=</span> (Context) wrapper.getParent();</span><br><span class=line></span><br><span class=line> <span class=comment>// Check for the application being marked unavailable</span></span><br><span class=line> <span class=keyword>if</span> (!context.getState().isAvailable()) {</span><br><span class=line> response.sendError(HttpServletResponse.SC_SERVICE_UNAVAILABLE,</span><br><span class=line> sm.getString(<span class=string>"standardContext.isUnavailable"</span>));</span><br><span class=line> unavailable = <span class=literal>true</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Check for the servlet being marked unavailable</span></span><br><span class=line> <span class=keyword>if</span> (!unavailable && wrapper.isUnavailable()) {</span><br><span class=line> container.getLogger().info(sm.getString(<span class=string>"standardWrapper.isUnavailable"</span>,</span><br><span class=line> wrapper.getName()));</span><br><span class=line> <span class=type>long</span> <span class=variable>available</span> <span class=operator>=</span> wrapper.getAvailable();</span><br><span class=line> <span class=keyword>if</span> ((available > <span class=number>0L</span>) && (available < Long.MAX_VALUE)) {</span><br><span class=line> response.setDateHeader(<span class=string>"Retry-After"</span>, available);</span><br><span class=line> response.sendError(HttpServletResponse.SC_SERVICE_UNAVAILABLE,</span><br><span class=line> sm.getString(<span class=string>"standardWrapper.isUnavailable"</span>,</span><br><span class=line> wrapper.getName()));</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (available == Long.MAX_VALUE) {</span><br><span class=line> response.sendError(HttpServletResponse.SC_NOT_FOUND,</span><br><span class=line> sm.getString(<span class=string>"standardWrapper.notFound"</span>,</span><br><span class=line> wrapper.getName()));</span><br><span class=line> }</span><br><span class=line> unavailable = <span class=literal>true</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Allocate a servlet instance to process this request</span></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> <span class=keyword>if</span> (!unavailable) {</span><br><span class=line> servlet = wrapper.allocate();</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (UnavailableException e) {</span><br><span class=line> container.getLogger().error(</span><br><span class=line> sm.getString(<span class=string>"standardWrapper.allocateException"</span>,</span><br><span class=line> wrapper.getName()), e);</span><br><span class=line> <span class=type>long</span> <span class=variable>available</span> <span class=operator>=</span> wrapper.getAvailable();</span><br><span class=line> <span class=keyword>if</span> ((available > <span class=number>0L</span>) && (available < Long.MAX_VALUE)) {</span><br><span class=line> response.setDateHeader(<span class=string>"Retry-After"</span>, available);</span><br><span class=line> response.sendError(HttpServletResponse.SC_SERVICE_UNAVAILABLE,</span><br><span class=line> sm.getString(<span class=string>"standardWrapper.isUnavailable"</span>,</span><br><span class=line> wrapper.getName()));</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (available == Long.MAX_VALUE) {</span><br><span class=line> response.sendError(HttpServletResponse.SC_NOT_FOUND,</span><br><span class=line> sm.getString(<span class=string>"standardWrapper.notFound"</span>,</span><br><span class=line> wrapper.getName()));</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (ServletException e) {</span><br><span class=line> container.getLogger().error(sm.getString(<span class=string>"standardWrapper.allocateException"</span>,</span><br><span class=line> wrapper.getName()), StandardWrapper.getRootCause(e));</span><br><span class=line> throwable = e;</span><br><span class=line> exception(request, response, e);</span><br><span class=line> } <span class=keyword>catch</span> (Throwable e) {</span><br><span class=line> ExceptionUtils.handleThrowable(e);</span><br><span class=line> container.getLogger().error(sm.getString(<span class=string>"standardWrapper.allocateException"</span>,</span><br><span class=line> wrapper.getName()), e);</span><br><span class=line> throwable = e;</span><br><span class=line> exception(request, response, e);</span><br><span class=line> servlet = <span class=literal>null</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=type>MessageBytes</span> <span class=variable>requestPathMB</span> <span class=operator>=</span> request.getRequestPathMB();</span><br><span class=line> <span class=type>DispatcherType</span> <span class=variable>dispatcherType</span> <span class=operator>=</span> DispatcherType.REQUEST;</span><br><span class=line> <span class=keyword>if</span> (request.getDispatcherType()==DispatcherType.ASYNC) {</span><br><span class=line> dispatcherType = DispatcherType.ASYNC;</span><br><span class=line> }</span><br><span class=line> request.setAttribute(Globals.DISPATCHER_TYPE_ATTR,dispatcherType);</span><br><span class=line> request.setAttribute(Globals.DISPATCHER_REQUEST_PATH_ATTR,</span><br><span class=line> requestPathMB);</span><br><span class=line> <span class=comment>// Create the filter chain for this request</span></span><br><span class=line> <span class=type>ApplicationFilterChain</span> <span class=variable>filterChain</span> <span class=operator>=</span></span><br><span class=line> ApplicationFilterFactory.createFilterChain(request, wrapper, servlet);</span><br><span class=line></span><br><span class=line> <span class=comment>// Call the filter chain for this request</span></span><br><span class=line> <span class=comment>// <span class=doctag>NOTE:</span> This also calls the servlet's service() method</span></span><br><span class=line> <span class=type>Container</span> <span class=variable>container</span> <span class=operator>=</span> <span class=built_in>this</span>.container;</span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> <span class=keyword>if</span> ((servlet != <span class=literal>null</span>) && (filterChain != <span class=literal>null</span>)) {</span><br><span class=line> <span class=comment>// Swallow output if needed</span></span><br><span class=line> <span class=keyword>if</span> (context.getSwallowOutput()) {</span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> SystemLogHandler.startCapture();</span><br><span class=line> <span class=keyword>if</span> (request.isAsyncDispatching()) {</span><br><span class=line> request.getAsyncContextInternal().doInternalDispatch();</span><br><span class=line> } <span class=keyword>else</span> {</span><br><span class=line> filterChain.doFilter(request.getRequest(),</span><br><span class=line> response.getResponse());</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>finally</span> {</span><br><span class=line> <span class=type>String</span> <span class=variable>log</span> <span class=operator>=</span> SystemLogHandler.stopCapture();</span><br><span class=line> <span class=keyword>if</span> (log != <span class=literal>null</span> && log.length() > <span class=number>0</span>) {</span><br><span class=line> context.getLogger().info(log);</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>else</span> {</span><br><span class=line> <span class=keyword>if</span> (request.isAsyncDispatching()) {</span><br><span class=line> request.getAsyncContextInternal().doInternalDispatch();</span><br><span class=line> } <span class=keyword>else</span> {</span><br><span class=line> filterChain.doFilter</span><br><span class=line> (request.getRequest(), response.getResponse());</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (ClientAbortException | CloseNowException e) {</span><br><span class=line> <span class=keyword>if</span> (container.getLogger().isDebugEnabled()) {</span><br><span class=line> container.getLogger().debug(sm.getString(</span><br><span class=line> <span class=string>"standardWrapper.serviceException"</span>, wrapper.getName(),</span><br><span class=line> context.getName()), e);</span><br><span class=line> }</span><br><span class=line> throwable = e;</span><br><span class=line> exception(request, response, e);</span><br><span class=line> } <span class=keyword>catch</span> (IOException e) {</span><br><span class=line> container.getLogger().error(sm.getString(</span><br><span class=line> <span class=string>"standardWrapper.serviceException"</span>, wrapper.getName(),</span><br><span class=line> context.getName()), e);</span><br><span class=line> throwable = e;</span><br><span class=line> exception(request, response, e);</span><br><span class=line> } <span class=keyword>catch</span> (UnavailableException e) {</span><br><span class=line> container.getLogger().error(sm.getString(</span><br><span class=line> <span class=string>"standardWrapper.serviceException"</span>, wrapper.getName(),</span><br><span class=line> context.getName()), e);</span><br><span class=line> <span class=comment>// throwable = e;</span></span><br><span class=line> <span class=comment>// exception(request, response, e);</span></span><br><span class=line> wrapper.unavailable(e);</span><br><span class=line> <span class=type>long</span> <span class=variable>available</span> <span class=operator>=</span> wrapper.getAvailable();</span><br><span class=line> <span class=keyword>if</span> ((available > <span class=number>0L</span>) && (available < Long.MAX_VALUE)) {</span><br><span class=line> response.setDateHeader(<span class=string>"Retry-After"</span>, available);</span><br><span class=line> response.sendError(HttpServletResponse.SC_SERVICE_UNAVAILABLE,</span><br><span class=line> sm.getString(<span class=string>"standardWrapper.isUnavailable"</span>,</span><br><span class=line> wrapper.getName()));</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (available == Long.MAX_VALUE) {</span><br><span class=line> response.sendError(HttpServletResponse.SC_NOT_FOUND,</span><br><span class=line> sm.getString(<span class=string>"standardWrapper.notFound"</span>,</span><br><span class=line> wrapper.getName()));</span><br><span class=line> }</span><br><span class=line> <span class=comment>// Do not save exception in 'throwable', because we</span></span><br><span class=line> <span class=comment>// do not want to do exception(request, response, e) processing</span></span><br><span class=line> } <span class=keyword>catch</span> (ServletException e) {</span><br><span class=line> <span class=type>Throwable</span> <span class=variable>rootCause</span> <span class=operator>=</span> StandardWrapper.getRootCause(e);</span><br><span class=line> <span class=keyword>if</span> (!(rootCause <span class=keyword>instanceof</span> ClientAbortException)) {</span><br><span class=line> container.getLogger().error(sm.getString(</span><br><span class=line> <span class=string>"standardWrapper.serviceExceptionRoot"</span>,</span><br><span class=line> wrapper.getName(), context.getName(), e.getMessage()),</span><br><span class=line> rootCause);</span><br><span class=line> }</span><br><span class=line> throwable = e;</span><br><span class=line> exception(request, response, e);</span><br><span class=line> } <span class=keyword>catch</span> (Throwable e) {</span><br><span class=line> ExceptionUtils.handleThrowable(e);</span><br><span class=line> container.getLogger().error(sm.getString(</span><br><span class=line> <span class=string>"standardWrapper.serviceException"</span>, wrapper.getName(),</span><br><span class=line> context.getName()), e);</span><br><span class=line> throwable = e;</span><br><span class=line> exception(request, response, e);</span><br><span class=line> } <span class=keyword>finally</span> {</span><br><span class=line> <span class=comment>// Release the filter chain (if any) for this request</span></span><br><span class=line> <span class=keyword>if</span> (filterChain != <span class=literal>null</span>) {</span><br><span class=line> filterChain.release();</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// Deallocate the allocated servlet instance</span></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> <span class=keyword>if</span> (servlet != <span class=literal>null</span>) {</span><br><span class=line> wrapper.deallocate(servlet);</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (Throwable e) {</span><br><span class=line> ExceptionUtils.handleThrowable(e);</span><br><span class=line> container.getLogger().error(sm.getString(<span class=string>"standardWrapper.deallocateException"</span>,</span><br><span class=line> wrapper.getName()), e);</span><br><span class=line> <span class=keyword>if</span> (throwable == <span class=literal>null</span>) {</span><br><span class=line> throwable = e;</span><br><span class=line> exception(request, response, e);</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=comment>// If this servlet has been marked permanently unavailable,</span></span><br><span class=line> <span class=comment>// unload it and release this instance</span></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> <span class=keyword>if</span> ((servlet != <span class=literal>null</span>) &&</span><br><span class=line> (wrapper.getAvailable() == Long.MAX_VALUE)) {</span><br><span class=line> wrapper.unload();</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (Throwable e) {</span><br><span class=line> ExceptionUtils.handleThrowable(e);</span><br><span class=line> container.getLogger().error(sm.getString(<span class=string>"standardWrapper.unloadException"</span>,</span><br><span class=line> wrapper.getName()), e);</span><br><span class=line> <span class=keyword>if</span> (throwable == <span class=literal>null</span>) {</span><br><span class=line> exception(request, response, e);</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line> <span class=type>long</span> t2=System.currentTimeMillis();</span><br><span class=line></span><br><span class=line> <span class=type>long</span> time=t2-t1;</span><br><span class=line> processingTime += time;</span><br><span class=line> <span class=keyword>if</span>( time > maxTime) {</span><br><span class=line> maxTime=time;</span><br><span class=line> }</span><br><span class=line> <span class=keyword>if</span>( time < minTime) {</span><br><span class=line> minTime=time;</span><br><span class=line> }</span><br><span class=line> }</span><br><span class=line> }</span><br></pre></table></figure><p>这里就会创建 ApplicationFilterChain 然后进行<br><code>filterChain.doFilter(request.getRequest(), response.getResponse());</code><br>doFilter 处理</div><footer class=post-footer><div class=post-eof></div></footer></article></div><div class=post-block><article class=post-content itemscope itemtype=http://schema.org/Article><link href=https://nicksxs.me/2023/09/24/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E4%B8%89-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Coyote/ itemprop=mainEntityOfPage><span hidden itemprop=author itemscope itemtype=http://schema.org/Person><meta content=/uploads/avatar.jpg itemprop=image><meta content=Nicksxs itemprop=name></span><span hidden itemprop=publisher itemscope itemtype=http://schema.org/Organization><meta content="Nicksxs's Blog" itemprop=name><meta content="learn from zero,技术博客,Nicksxs,史学森" itemprop=description></span><span hidden itemprop=post itemscope itemtype=http://schema.org/CreativeWork><meta content="undefined | Nicksxs's Blog" itemprop=name><meta itemprop=description></span><header class=post-header><h2 itemprop="name headline" class=post-title><a class=post-title-link href=/2023/09/24/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E4%B8%89-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Coyote/ itemprop=url>Tomcat 系列篇三-介绍下 Coyote</a></h2><div class=post-meta-container><div class=post-meta><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-calendar"></i> </span><span class=post-meta-item-text>发表于</span> <time itemprop="dateCreated datePublished" title="创建时间:2023-09-24 21:36:43" datetime=2023-09-24T21:36:43+08:00>2023-09-24</time> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-folder"></i> </span><span class=post-meta-item-text>分类于</span> <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/ itemprop=url rel=index><span itemprop=name>Java</span></a> </span>, <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/SpringBoot/ itemprop=url rel=index><span itemprop=name>SpringBoot</span></a> </span>, <span itemprop=about itemscope itemtype=http://schema.org/Thing><a href=/categories/Java/SpringBoot/Tomcat/ itemprop=url rel=index><span itemprop=name>Tomcat</span></a> </span></span><span class="post-meta-item leancloud_visitors" data-flag-title="Tomcat 系列篇三-介绍下 Coyote" id=/2023/09/24/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E4%B8%89-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Coyote/ title=阅读次数><span class=post-meta-item-icon><i class="far fa-eye"></i> </span><span class=post-meta-item-text>阅读次数:</span> <span class=leancloud-visitors-count></span> </span><span class=post-meta-item><span class=post-meta-item-icon><i class="far fa-comment"></i> </span><span class=post-meta-item-text>Disqus:</span> <a href=/2023/09/24/Tomcat-%E7%B3%BB%E5%88%97%E7%AF%87%E4%B8%89-%E4%BB%8B%E7%BB%8D%E4%B8%8B-Coyote/#disqus_thread itemprop=discussionUrl title=disqus><span class="post-comments-count disqus-comment-count" data-disqus-identifier=2023/09/24/Tomcat-系列篇三-介绍下-Coyote/ itemprop=commentCount></span></a></span></div></div></header><div class=post-body itemprop=articleBody><p>前面介绍了 connector,这里边还有个很重要的概念是 Coyote,真正将前面的 connector 跟后面的 container 做了连接,<br><code>org.apache.tomcat.util.net.AbstractEndpoint#createSocketProcessor</code> 从这里开始<br>然后会调用到 <code>org.apache.tomcat.util.net.NioEndpoint#createSocketProcessor</code><figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>protected</span> SocketProcessorBase<NioChannel> <span class="title function_">createSocketProcessor</span><span class=params>(</span></span><br><span class=line><span class=params> SocketWrapperBase<NioChannel> socketWrapper, SocketEvent event)</span> {</span><br><span class=line> <span class=keyword>return</span> <span class=keyword>new</span> <span class="title class_">SocketProcessor</span>(socketWrapper, event);</span><br><span class=line>}</span><br></pre></table></figure><p>里面实际的是 new 了<br><code>org.apache.tomcat.util.net.NioEndpoint.SocketProcessor#SocketProcessor</code><figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br></pre><td class=code><pre><span class=line><span class=keyword>protected</span> <span class=keyword>class</span> <span class="title class_">SocketProcessor</span> <span class=keyword>extends</span> <span class="title class_">SocketProcessorBase</span><NioChannel> {</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class="title function_">SocketProcessor</span><span class=params>(SocketWrapperBase<NioChannel> socketWrapper, SocketEvent event)</span> {</span><br><span class=line> <span class=built_in>super</span>(socketWrapper, event);</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=meta>@Override</span></span><br><span class=line> <span class=keyword>protected</span> <span class=keyword>void</span> <span class="title function_">doRun</span><span class=params>()</span> {</span><br><span class=line> <span class=comment>/*</span></span><br><span class=line><span class=comment> * Do not cache and re-use the value of socketWrapper.getSocket() in</span></span><br><span class=line><span class=comment> * this method. If the socket closes the value will be updated to</span></span><br><span class=line><span class=comment> * CLOSED_NIO_CHANNEL and the previous value potentially re-used for</span></span><br><span class=line><span class=comment> * a new connection. That can result in a stale cached value which</span></span><br><span class=line><span class=comment> * in turn can result in unintentionally closing currently active</span></span><br><span class=line><span class=comment> * connections.</span></span><br><span class=line><span class=comment> */</span></span><br><span class=line> <span class=type>Poller</span> <span class=variable>poller</span> <span class=operator>=</span> NioEndpoint.<span class=built_in>this</span>.poller;</span><br><span class=line> <span class=keyword>if</span> (poller == <span class=literal>null</span>) {</span><br><span class=line> socketWrapper.close();</span><br></pre></table></figure><p>然后是 <code>org.apache.tomcat.util.net.NioEndpoint.SocketProcessor#doRun</code> 这里开始运行<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br><span class=line>28</span><br><span class=line>29</span><br><span class=line>30</span><br><span class=line>31</span><br><span class=line>32</span><br><span class=line>33</span><br><span class=line>34</span><br><span class=line>35</span><br><span class=line>36</span><br><span class=line>37</span><br><span class=line>38</span><br><span class=line>39</span><br><span class=line>40</span><br><span class=line>41</span><br><span class=line>42</span><br><span class=line>43</span><br><span class=line>44</span><br><span class=line>45</span><br><span class=line>46</span><br><span class=line>47</span><br><span class=line>48</span><br><span class=line>49</span><br><span class=line>50</span><br><span class=line>51</span><br><span class=line>52</span><br><span class=line>53</span><br><span class=line>54</span><br><span class=line>55</span><br><span class=line>56</span><br><span class=line>57</span><br></pre><td class=code><pre><span class=line><span class=keyword>protected</span> <span class=keyword>void</span> <span class="title function_">doRun</span><span class=params>()</span> {</span><br><span class=line> <span class=comment>/*</span></span><br><span class=line><span class=comment> * Do not cache and re-use the value of socketWrapper.getSocket() in</span></span><br><span class=line><span class=comment> * this method. If the socket closes the value will be updated to</span></span><br><span class=line><span class=comment> * CLOSED_NIO_CHANNEL and the previous value potentially re-used for</span></span><br><span class=line><span class=comment> * a new connection. That can result in a stale cached value which</span></span><br><span class=line><span class=comment> * in turn can result in unintentionally closing currently active</span></span><br><span class=line><span class=comment> * connections.</span></span><br><span class=line><span class=comment> */</span></span><br><span class=line> <span class=type>Poller</span> <span class=variable>poller</span> <span class=operator>=</span> NioEndpoint.<span class=built_in>this</span>.poller;</span><br><span class=line> <span class=keyword>if</span> (poller == <span class=literal>null</span>) {</span><br><span class=line> socketWrapper.close();</span><br><span class=line> <span class=keyword>return</span>;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> <span class=type>int</span> <span class=variable>handshake</span> <span class=operator>=</span> -<span class=number>1</span>;</span><br><span class=line> <span class=keyword>try</span> {</span><br><span class=line> <span class=keyword>if</span> (socketWrapper.getSocket().isHandshakeComplete()) {</span><br><span class=line> <span class=comment>// No TLS handshaking required. Let the handler</span></span><br><span class=line> <span class=comment>// process this socket / event combination.</span></span><br><span class=line> handshake = <span class=number>0</span>;</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (event == SocketEvent.STOP || event == SocketEvent.DISCONNECT ||</span><br><span class=line> event == SocketEvent.ERROR) {</span><br><span class=line> <span class=comment>// Unable to complete the TLS handshake. Treat it as</span></span><br><span class=line> <span class=comment>// if the handshake failed.</span></span><br><span class=line> handshake = -<span class=number>1</span>;</span><br><span class=line> } <span class=keyword>else</span> {</span><br><span class=line> handshake = socketWrapper.getSocket().handshake(event == SocketEvent.OPEN_READ, event == SocketEvent.OPEN_WRITE);</span><br><span class=line> <span class=comment>// The handshake process reads/writes from/to the</span></span><br><span class=line> <span class=comment>// socket. status may therefore be OPEN_WRITE once</span></span><br><span class=line> <span class=comment>// the handshake completes. However, the handshake</span></span><br><span class=line> <span class=comment>// happens when the socket is opened so the status</span></span><br><span class=line> <span class=comment>// must always be OPEN_READ after it completes. It</span></span><br><span class=line> <span class=comment>// is OK to always set this as it is only used if</span></span><br><span class=line> <span class=comment>// the handshake completes.</span></span><br><span class=line> event = SocketEvent.OPEN_READ;</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (IOException x) {</span><br><span class=line> handshake = -<span class=number>1</span>;</span><br><span class=line> <span class=keyword>if</span> (log.isDebugEnabled()) {</span><br><span class=line> log.debug(<span class=string>"Error during SSL handshake"</span>,x);</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>catch</span> (CancelledKeyException ckx) {</span><br><span class=line> handshake = -<span class=number>1</span>;</span><br><span class=line> }</span><br><span class=line> <span class=keyword>if</span> (handshake == <span class=number>0</span>) {</span><br><span class=line> <span class=type>SocketState</span> <span class=variable>state</span> <span class=operator>=</span> SocketState.OPEN;</span><br><span class=line> <span class=comment>// Process the request from this socket</span></span><br><span class=line> <span class=keyword>if</span> (event == <span class=literal>null</span>) {</span><br><span class=line> state = getHandler().process(socketWrapper, SocketEvent.OPEN_READ);</span><br><span class=line> } <span class=keyword>else</span> {</span><br><span class=line> state = getHandler().process(socketWrapper, event);</span><br><span class=line> }</span><br><span class=line> <span class=keyword>if</span> (state == SocketState.CLOSED) {</span><br><span class=line> poller.cancelledKey(getSelectionKey(), socketWrapper);</span><br><span class=line> }</span><br></pre></table></figure><p><code>org.apache.coyote.AbstractProtocol.ConnectionHandler#process</code> 这个 <code>getHandler</code> 是哪个呢<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br></pre><td class=code><pre><span class=line><span class=keyword>public</span> <span class="title function_">AbstractHttp11Protocol</span><span class=params>(AbstractEndpoint<S,?> endpoint)</span> {</span><br><span class=line> <span class=built_in>super</span>(endpoint);</span><br><span class=line> setConnectionTimeout(Constants.DEFAULT_CONNECTION_TIMEOUT);</span><br><span class=line> ConnectionHandler<S> cHandler = <span class=keyword>new</span> <span class="title class_">ConnectionHandler</span><>(<span class=built_in>this</span>);</span><br><span class=line> setHandler(cHandler);</span><br><span class=line> getEndpoint().setHandler(cHandler);</span><br><span class=line>}</span><br></pre></table></figure><p>上面补充下这个 Handler,帮助后面的理解,而这个 connectionHandler 则是实现了 <code>AbstractEndpoint.Handler</code><figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br></pre><td class=code><pre><span class=line><span class=keyword>protected</span> <span class=keyword>static</span> <span class=keyword>class</span> <span class="title class_">ConnectionHandler</span><S> <span class=keyword>implements</span> <span class="title class_">AbstractEndpoint</span>.Handler<S> {</span><br><span class=line></span><br><span class=line> <span class=keyword>private</span> <span class=keyword>final</span> AbstractProtocol<S> proto;</span><br><span class=line> <span class=keyword>private</span> <span class=keyword>final</span> <span class=type>RequestGroupInfo</span> <span class=variable>global</span> <span class=operator>=</span> <span class=keyword>new</span> <span class="title class_">RequestGroupInfo</span>();</span><br><span class=line> <span class=keyword>private</span> <span class=keyword>final</span> <span class=type>AtomicLong</span> <span class=variable>registerCount</span> <span class=operator>=</span> <span class=keyword>new</span> <span class="title class_">AtomicLong</span>(<span class=number>0</span>);</span><br><span class=line> <span class=keyword>private</span> <span class=keyword>final</span> <span class=type>RecycledProcessors</span> <span class=variable>recycledProcessors</span> <span class=operator>=</span> <span class=keyword>new</span> <span class="title class_">RecycledProcessors</span>(<span class=built_in>this</span>);</span><br><span class=line></span><br><span class=line> <span class=keyword>public</span> <span class="title function_">ConnectionHandler</span><span class=params>(AbstractProtocol<S> proto)</span> {</span><br><span class=line> <span class=built_in>this</span>.proto = proto;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=keyword>protected</span> AbstractProtocol<S> <span class="title function_">getProtocol</span><span class=params>()</span> {</span><br><span class=line> <span class=keyword>return</span> proto;</span><br><span class=line> }</span><br></pre></table></figure><p>然后会继续寻找真实的 Processer<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br><span class=line>27</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>public</span> SocketState <span class="title function_">process</span><span class=params>(SocketWrapperBase<S> wrapper, SocketEvent status)</span> {</span><br><span class=line> <span class=keyword>if</span> (getLog().isDebugEnabled()) {</span><br><span class=line> getLog().debug(sm.getString(<span class=string>"abstractConnectionHandler.process"</span>,</span><br><span class=line> wrapper.getSocket(), status));</span><br><span class=line> }</span><br><span class=line> <span class=keyword>if</span> (wrapper == <span class=literal>null</span>) {</span><br><span class=line> <span class=comment>// Nothing to do. Socket has been closed.</span></span><br><span class=line> <span class=keyword>return</span> SocketState.CLOSED;</span><br><span class=line> }</span><br><span class=line></span><br><span class=line> <span class=type>S</span> <span class=variable>socket</span> <span class=operator>=</span> wrapper.getSocket();</span><br><span class=line></span><br><span class=line> <span class=type>Processor</span> <span class=variable>processor</span> <span class=operator>=</span> (Processor) wrapper.getCurrentProcessor();</span><br><span class=line> <span class=keyword>if</span> (getLog().isDebugEnabled()) {</span><br><span class=line> getLog().debug(sm.getString(<span class=string>"abstractConnectionHandler.connectionsGet"</span>,</span><br><span class=line> processor, socket));</span><br><span class=line> }</span><br><span class=line> <span class=comment>// 省略代码</span></span><br><span class=line> <span class=comment>// 直到这里创建 processor</span></span><br><span class=line> <span class=keyword>if</span> (processor == <span class=literal>null</span>) {</span><br><span class=line> processor = getProtocol().createProcessor();</span><br><span class=line> register(processor);</span><br><span class=line> <span class=keyword>if</span> (getLog().isDebugEnabled()) {</span><br><span class=line> getLog().debug(sm.getString(<span class=string>"abstractConnectionHandler.processorCreate"</span>, processor));</span><br><span class=line> }</span><br><span class=line> }</span><br></pre></table></figure><p>也就是 <code>org.apache.coyote.http11.AbstractHttp11Protocol#createProcessor</code><figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>protected</span> Processor <span class="title function_">createProcessor</span><span class=params>()</span> {</span><br><span class=line> <span class=type>Http11Processor</span> <span class=variable>processor</span> <span class=operator>=</span> <span class=keyword>new</span> <span class="title class_">Http11Processor</span>(<span class=built_in>this</span>, adapter);</span><br><span class=line> <span class=keyword>return</span> processor;</span><br><span class=line>}</span><br></pre></table></figure><p>再往后就是调用 process 方法了,然后它是 Http11Processor 的抽象父类<br><code>org.apache.coyote.AbstractProcessorLight</code><br>会调用 <code>org.apache.coyote.AbstractProcessorLight#process</code> 来处理前面说的 socket<br>接着会跑到这<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br><span class=line>5</span><br><span class=line>6</span><br><span class=line>7</span><br><span class=line>8</span><br><span class=line>9</span><br><span class=line>10</span><br><span class=line>11</span><br><span class=line>12</span><br><span class=line>13</span><br><span class=line>14</span><br><span class=line>15</span><br><span class=line>16</span><br><span class=line>17</span><br><span class=line>18</span><br><span class=line>19</span><br><span class=line>20</span><br><span class=line>21</span><br><span class=line>22</span><br><span class=line>23</span><br><span class=line>24</span><br><span class=line>25</span><br><span class=line>26</span><br></pre><td class=code><pre><span class=line><span class=meta>@Override</span></span><br><span class=line><span class=keyword>public</span> SocketState <span class="title function_">process</span><span class=params>(SocketWrapperBase<?> socketWrapper, SocketEvent status)</span></span><br><span class=line> <span class=keyword>throws</span> IOException {</span><br><span class=line></span><br><span class=line> <span class=type>SocketState</span> <span class=variable>state</span> <span class=operator>=</span> SocketState.CLOSED;</span><br><span class=line> Iterator<DispatchType> dispatches = <span class=literal>null</span>;</span><br><span class=line> <span class=keyword>do</span> {</span><br><span class=line> <span class=keyword>if</span> (dispatches != <span class=literal>null</span>) {</span><br><span class=line> <span class=type>DispatchType</span> <span class=variable>nextDispatch</span> <span class=operator>=</span> dispatches.next();</span><br><span class=line> <span class=keyword>if</span> (getLog().isDebugEnabled()) {</span><br><span class=line> getLog().debug(<span class=string>"Processing dispatch type: ["</span> + nextDispatch + <span class=string>"]"</span>);</span><br><span class=line> }</span><br><span class=line> state = dispatch(nextDispatch.getSocketStatus());</span><br><span class=line> <span class=keyword>if</span> (!dispatches.hasNext()) {</span><br><span class=line> state = checkForPipelinedData(state, socketWrapper);</span><br><span class=line> }</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (status == SocketEvent.DISCONNECT) {</span><br><span class=line> <span class=comment>// Do nothing here, just wait for it to get recycled</span></span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (isAsync() || isUpgrade() || state == SocketState.ASYNC_END) {</span><br><span class=line> state = dispatch(status);</span><br><span class=line> state = checkForPipelinedData(state, socketWrapper);</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (status == SocketEvent.OPEN_WRITE) {</span><br><span class=line> <span class=comment>// Extra write event likely after async, ignore</span></span><br><span class=line> state = SocketState.LONG;</span><br><span class=line> } <span class=keyword>else</span> <span class=keyword>if</span> (status == SocketEvent.OPEN_READ) {</span><br><span class=line> state = service(socketWrapper);</span><br></pre></table></figure><p>接下去就是<br><code>org.apache.coyote.http11.Http11Processor#service</code><br>再就是调用coyote的 service 方法也就是 <code>org.apache.catalina.connector.CoyoteAdapter#service</code><br>这里就会调用到<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br><span class=line>3</span><br><span class=line>4</span><br></pre><td class=code><pre><span class=line></span><br><span class=line>connector.getService().getContainer().getPipeline().getFirst().invoke(</span><br><span class=line> request, response);</span><br><span class=line> </span><br></pre></table></figure><p>然后进行 valve 串的执行到 <code>org.apache.catalina.core.StandardWrapperValve#invoke</code> 中<br>会调用<figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br><span class=line>2</span><br></pre><td class=code><pre><span class=line>filterChain.doFilter</span><br><span class=line> (request.getRequest(), response.getResponse());</span><br></pre></table></figure><p>就会执行 filter 链<br>最后到<br><code>org.apache.catalina.core.ApplicationFilterChain#internalDoFilter</code><figure class="highlight java"><table><tr><td class=gutter><pre><span class=line>1</span><br></pre><td class=code><pre><span class=line>servlet.service(request, response);</span><br></pre></table></figure><p>就到了 DispatcherServlet 处理的流程, 这样就和之前介绍 DispatcherServlet开始的请求处理接上了。</div><footer class=post-footer><div class=post-eof></div></footer></article></div><nav class=pagination><a class="extend prev" aria-label=上一页 href=/page/21/ rel=prev title=上一页><i class="fa fa-angle-left"></i></a><a class=page-number href=/>1</a><span class=space>…</span><a class=page-number href=/page/21/>21</a><span class="page-number current">22</span><a class=page-number href=/page/23/>23</a><span class=space>…</span><a class=page-number href=/page/69/>69</a><a class="extend next" aria-label=下一页 href=/page/23/ rel=next title=下一页><i class="fa fa-angle-right"></i></a></nav></div></main><footer class=footer><div class=footer-inner><div class=beian><a href=https://beian.miit.gov.cn/ rel=noopener target=_blank>浙ICP备16002580号-2 </a><img alt src=/uploads/beian.png><a href=https://beian.mps.gov.cn/#/query/webSearch?code=33010602011094 rel=noopener target=_blank>浙公网安备 33010602011094号</a></div><div class=copyright>© <span itemprop=copyrightYear>2025</span><span class=with-love><i class="fa fa-heart"></i> </span><span class=author itemprop=copyrightHolder>Nicksxs</span></div><div class=busuanzi-count><span class=post-meta-item id=busuanzi_container_site_uv><span class=post-meta-item-icon><i class="fa fa-user"></i> </span><span class=site-uv title=总访客量><span id=busuanzi_value_site_uv></span> </span></span><span class=post-meta-item id=busuanzi_container_site_pv><span class=post-meta-item-icon><i class="fa fa-eye"></i> </span><span class=site-pv title=总访问量><span id=busuanzi_value_site_pv></span></span></span></div></div></footer><div aria-label=返回顶部 class=back-to-top role=button><i class="fa fa-arrow-up fa-lg"></i><span>0%</span></div><a aria-label="在 GitHub 上关注我" title="在 GitHub 上关注我" class=github-corner href=https://github.com/nicksxs rel=noopener target=_blank><svg viewbox="0 0 250 250" aria-hidden=true height=80 width=80><path d="M0,0 L115,115 L130,115 L142,142 L250,250 L250,0 Z"></path><path d="M128.3,109.0 C113.8,99.7 119.0,89.6 119.0,89.6 C122.0,82.7 120.5,78.6 120.5,78.6 C119.2,72.0 123.4,76.3 123.4,76.3 C127.3,80.9 125.5,87.3 125.5,87.3 C122.9,97.6 130.6,101.9 134.4,103.2" style="transform-origin:130px 106px" class=octo-arm fill=currentColor></path><path d="M115.0,115.0 C114.9,115.1 118.7,116.5 119.8,115.4 L133.7,101.6 C136.9,99.2 139.9,98.4 142.2,98.6 C133.8,88.0 127.5,74.4 143.8,58.0 C148.5,53.4 154.0,51.2 159.7,51.0 C160.3,49.4 163.2,43.6 171.4,40.1 C171.4,40.1 176.1,42.5 178.8,56.2 C183.1,58.6 187.2,61.8 190.9,65.4 C194.5,69.0 197.7,73.2 200.1,77.6 C213.8,80.2 216.3,84.9 216.3,84.9 C212.7,93.1 206.9,96.0 205.4,96.6 C205.1,102.4 203.0,107.8 198.3,112.5 C181.9,128.9 168.3,122.5 157.7,114.1 C157.9,116.9 156.7,120.9 152.7,124.9 L141.0,136.5 C139.8,137.7 141.6,141.9 141.8,141.8 Z" class=octo-body fill=currentColor></path></svg></a><noscript><div class=noscript-warning>Theme NexT works best with JavaScript enabled</div></noscript><script crossorigin=anonymous integrity=sha256-XL2inqUJaslATFnHdJOi9GfQ60on8Wx1C2H8DYiN1xY= src=https://cdnjs.cloudflare.com/ajax/libs/animejs/3.2.1/anime.min.js></script><script crossorigin=anonymous integrity=sha256-ytMJGN3toR+a84u7g7NuHm91VIR06Q41kMWDr2pq7Zo= src=https://cdnjs.cloudflare.com/ajax/libs/fancyapps-ui/5.0.28/fancybox/fancybox.umd.js></script><script crossorigin=anonymous integrity=sha256-mOFREFhqmHeQbXpK2lp4nA3qooVgACfh88fpJftLBbc= src=https://cdnjs.cloudflare.com/ajax/libs/lozad.js/1.16.0/lozad.min.js></script><script src=/js/comments.js></script><script src=/js/utils.js></script><script src=/js/motion.js></script><script src=/js/next-boot.js></script><script crossorigin=anonymous integrity=sha256-DABVk+hYj0mdUzo+7ViJC6cwLahQIejFvC+my2M/wfM= src=https://cdnjs.cloudflare.com/ajax/libs/algoliasearch/4.20.0/algoliasearch-lite.umd.js></script><script crossorigin=anonymous integrity=sha256-9242vN47QUX50UG5Gf5XDO1YREWCEJRyXHofh5fsl24= src=https://cdnjs.cloudflare.com/ajax/libs/instantsearch.js/4.60.0/instantsearch.production.min.js></script><script src=/js/third-party/search/algolia-search.js></script><script src=/js/third-party/fancybox.js></script><script async src=https://busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js></script><script class=next-config data-name=leancloud_visitors type=application/json>{"enable":true,"app_id":"ysza182Vghlqjdt7QiwGLLJy-gzGzoHsz","app_key":"s9GDqbn7gnGGkusf66YRVccw","server_url":"https://leancloud.cn","security":true}</script><script src=/js/third-party/statistics/lean-analytics.js></script><script crossorigin=anonymous integrity=sha256-yvJQOINiH9fWemHn0vCA5lsHWJaHs6/ZmO+1Ft04SvM= src=https://cdnjs.cloudflare.com/ajax/libs/quicklink/2.3.0/quicklink.umd.js></script><script class=next-config data-name=quicklink type=application/json>{"enable":false,"home":false,"archive":false,"delay":true,"timeout":3000,"priority":true,"url":"https://nicksxs.me/page/22/"}</script><script src=/js/third-party/quicklink.js></script><script class=next-config data-name=disqus type=application/json>{"enable":true,"shortname":"nicksxs","count":true,"i18n":{"disqus":"disqus"}}</script><script src=/js/third-party/comments/disqus.js></script>
|