You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

1537 lines
173 KiB

<!DOCTYPE html>
<html lang="zh-Hans">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=2">
<meta name="theme-color" content="#222">
<meta name="generator" content="Hexo 4.2.0">
<link rel="apple-touch-icon" sizes="180x180" href="/images/apple-touch-icon-next.png">
<link rel="icon" type="image/png" sizes="32x32" href="/images/favicon-32x32-next.png">
<link rel="icon" type="image/png" sizes="16x16" href="/images/favicon-16x16-next.png">
<link rel="mask-icon" href="/images/logo.svg" color="#222">
<meta name="google-site-verification" content="2X6S9P7CAjXjVvw8YyQR8pCu-B0oEu7O9quLgxXuWyA">
<meta name="baidu-site-verification" content="dV8JGNzi0c">
<meta name="chinaz-site-verification" content="500A176AA29CFB31">
<link rel="stylesheet" href="/css/main.css">
<link rel="stylesheet" href="//fonts.loli.net/css?family=Lato:300,300italic,400,400italic,700,700italic|M+ 2p:300,300italic,400,400italic,700,700italic&display=swap&subset=latin,latin-ext">
<link rel="stylesheet" href="/lib/font-awesome/css/font-awesome.min.css">
<link rel="stylesheet" href="//cdn.jsdelivr.net/gh/fancyapps/fancybox@3/dist/jquery.fancybox.min.css">
<link rel="stylesheet" href="/lib/pace/pace-theme-minimal.min.css">
<script src="/lib/pace/pace.min.js"></script>
<script id="hexo-configurations">
var NexT = window.NexT || {};
var CONFIG = {
hostname: new URL('https://nicksxs.me').hostname,
root: '/',
scheme: 'Pisces',
version: '7.7.0',
exturl: false,
sidebar: {"position":"left","display":"post","padding":18,"offset":12,"onmobile":false},
copycode: {"enable":true,"show_result":false,"style":null},
back2top: {"enable":true,"sidebar":false,"scrollpercent":false},
bookmark: {"enable":false,"color":"#222","save":"auto"},
fancybox: true,
mediumzoom: false,
lazyload: true,
pangu: false,
comments: {"style":"tabs","active":null,"storage":true,"lazyload":false,"nav":null},
algolia: {
appID: '',
apiKey: '',
indexName: '',
hits: {"per_page":10},
labels: {"input_placeholder":"Search for Posts","hits_empty":"We didn't find any results for the search: ${query}","hits_stats":"${hits} results found in ${time} ms"}
},
localsearch: {"enable":false,"trigger":"auto","top_n_per_article":1,"unescape":false,"preload":false},
path: '',
motion: {"enable":true,"async":false,"transition":{"post_block":"fadeIn","post_header":"slideDownIn","post_body":"slideDownIn","coll_header":"slideLeftIn","sidebar":"slideUpIn"}}
};
</script>
<meta name="description" content="learn from zero,技术博客,Nicksxs,史学森">
<meta property="og:type" content="website">
<meta property="og:title" content="Nicksxs&#39;s Blog">
<meta property="og:url" content="https://nicksxs.me/page/2/index.html">
<meta property="og:site_name" content="Nicksxs&#39;s Blog">
<meta property="og:description" content="learn from zero,技术博客,Nicksxs,史学森">
<meta property="article:author" content="Nicksxs">
<meta property="article:tag" content="Nicksxs">
<meta property="article:tag" content="史学森">
<meta property="article:tag" content="米方方">
<meta property="article:tag" content="米方方的男朋友">
<meta property="article:tag" content="森哥">
<meta name="twitter:card" content="summary">
<link rel="canonical" href="https://nicksxs.me/page/2/">
<script id="page-configurations">
// https://hexo.io/docs/variables.html
CONFIG.page = {
sidebar: "",
isHome: true,
isPost: false
};
</script>
<title>Nicksxs's Blog</title>
<script async src="https://www.googletagmanager.com/gtag/js?id=UA-61358619-1"></script>
<script>
if (CONFIG.hostname === location.hostname) {
window.dataLayer = window.dataLayer || [];
function gtag(){dataLayer.push(arguments);}
gtag('js', new Date());
gtag('config', 'UA-61358619-1');
}
</script>
<script>
var _hmt = _hmt || [];
(function() {
var hm = document.createElement("script");
hm.src = "https://hm.baidu.com/hm.js?20f33b3c0c0eff9b1522999c0015646d";
var s = document.getElementsByTagName("script")[0];
s.parentNode.insertBefore(hm, s);
})();
</script>
<noscript>
<style>
.use-motion .brand,
.use-motion .menu-item,
.sidebar-inner,
.use-motion .post-block,
.use-motion .pagination,
.use-motion .comments,
.use-motion .post-header,
.use-motion .post-body,
.use-motion .collection-header { opacity: initial; }
.use-motion .site-title,
.use-motion .site-subtitle {
opacity: initial;
top: initial;
}
.use-motion .logo-line-before i { left: initial; }
.use-motion .logo-line-after i { right: initial; }
</style>
</noscript>
<link rel="alternate" href="/atom.xml" title="Nicksxs's Blog" type="application/atom+xml">
</head>
<body itemscope itemtype="http://schema.org/WebPage">
<div class="container use-motion">
<div class="headband"></div>
<header class="header" itemscope itemtype="http://schema.org/WPHeader">
<div class="header-inner"><div class="site-brand-container">
<div class="site-meta">
<div>
<a href="/" class="brand" rel="start">
<span class="logo-line-before"><i></i></span>
<span class="site-title">Nicksxs's Blog</span>
<span class="logo-line-after"><i></i></span>
</a>
</div>
<p class="site-subtitle">What hurts more, the pain of hard work or the pain of regret?</p>
</div>
<div class="site-nav-toggle">
<div class="toggle" aria-label="Toggle navigation bar">
<span class="toggle-line toggle-line-first"></span>
<span class="toggle-line toggle-line-middle"></span>
<span class="toggle-line toggle-line-last"></span>
</div>
</div>
</div>
<nav class="site-nav">
<ul id="menu" class="menu">
<li class="menu-item menu-item-home">
<a href="/" rel="section"><i class="fa fa-fw fa-home"></i>Home</a>
</li>
<li class="menu-item menu-item-about">
<a href="/about/" rel="section"><i class="fa fa-fw fa-about"></i>About</a>
</li>
<li class="menu-item menu-item-tags">
<a href="/tags/" rel="section"><i class="fa fa-fw fa-tags"></i>Tags</a>
</li>
<li class="menu-item menu-item-categories">
<a href="/categories/" rel="section"><i class="fa fa-fw fa-categories"></i>Categories</a>
</li>
<li class="menu-item menu-item-archives">
<a href="/archives/" rel="section"><i class="fa fa-fw fa-archives"></i>Archives</a>
</li>
<li class="menu-item menu-item-sitemap">
<a href="/sitemap.xml" rel="section"><i class="fa fa-fw fa-sitemap"></i>Sitemap</a>
</li>
<li class="menu-item menu-item-commonweal">
<a href="/404/" rel="section"><i class="fa fa-fw fa-heartbeat"></i>Commonweal 404</a>
</li>
</ul>
</nav>
</div>
</header>
<div class="back-to-top">
<i class="fa fa-arrow-up"></i>
<span>0%</span>
</div>
<a href="https://github.com/nicksxs" class="github-corner" title="Follow me on GitHub" aria-label="Follow me on GitHub" rel="noopener" target="_blank"><svg width="80" height="80" viewBox="0 0 250 250" aria-hidden="true"><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" fill="currentColor" style="transform-origin: 130px 106px;" class="octo-arm"></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" fill="currentColor" class="octo-body"></path></svg></a>
<main class="main">
<div class="main-inner">
<div class="content-wrap">
<div class="content">
<div class="posts-expand">
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2019/12/18/1Q84%E8%AF%BB%E5%90%8E%E6%84%9F/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2019/12/18/1Q84%E8%AF%BB%E5%90%8E%E6%84%9F/" class="post-title-link" itemprop="url">村上春树《1Q84》读后感</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2019-12-18 00:33:09" itemprop="dateCreated datePublished" datetime="2019-12-18T00:33:09+08:00">2019-12-18</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2019-12-19 00:08:42" itemprop="dateModified" datetime="2019-12-19T00:08:42+08:00">2019-12-19</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</span>
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/%E7%94%9F%E6%B4%BB/" itemprop="url" rel="index">
<span itemprop="name">生活</span>
</a>
</span>
,
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/%E8%AF%BB%E5%90%8E%E6%84%9F/" itemprop="url" rel="index">
<span itemprop="name">读后感</span>
</a>
</span>
,
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/%E8%AF%BB%E5%90%8E%E6%84%9F/%E6%9D%91%E4%B8%8A%E6%98%A5%E6%A0%91/" itemprop="url" rel="index">
<span itemprop="name">村上春树</span>
</a>
</span>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2019/12/18/1Q84%E8%AF%BB%E5%90%8E%E6%84%9F/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2019/12/18/1Q84读后感/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<p>看完了村上春树的《1Q84》,这应该是第五本看的他的书了,继 跑步,挪威的森林,刺杀骑士团长,海边的卡夫卡之后,不是其中最长的,好像是海边的卡夫卡还是刺杀骑士团长比较长一点,都是在微信读书上看的,比较方便,最开始在上面看的是高晓松的《鱼羊野史》,不知道为啥取这个名字,但是还是满吸引我的,不过由于去年的种种,没有很多心思把它看完,而且本身的组织形式就是比较松散的,看到哪算哪,其实一些野史部分是我比较喜欢,有些谈到人物的就不太有兴趣,而且类似于大祥哥吃的东西,反正都是哇,怎么这么好吃,嗯,太爱(niu)你(bi)了,高晓松就是这个人是我最喜欢的 xxx 家,我也没去细究过他有没有说重复过,反正是不太爱,后来因为这书还一度对战争史有了浓厚的兴趣,然而事实告诉我,大部头的战争史,其实正史我是真的啃不下去,我可能只对其中 10%的内容感兴趣,不过终于也在今年把它看完了,好像高晓松的晓说也最终季了,貌似其中讲朝鲜战争的还被和谐了,看样子是说出了一些故事(truth)。</p>
<p>本来只是想把 《1Q84》的读后感写下,现在觉得还是把这篇当成我今年的读书总结吧,不过先从《1Q84》说起。</p>
<p>严格来讲,这不是很书面化的读后感,可能我想写的也只是像聊天一样的说下我读过的书,包括的技术博客其实也是类似的,以后或许会转变,但是目前水平如此吧,写多了可能会变好,也可能不会。</p>
<p>开始正文吧,这书有点类似于海边的卡夫卡,一开始是通过两条故事线,穿插着叙述,一条是青豆的,不算是个职业杀手的女杀手,要去解决一个经常家暴的斯文败类,穿着描述得比较性感吧,杀人方式是通过比较长的细针,从脖子后面一个精巧的位置插入,可以造成是未知原因死亡的假象,可能会推断成心梗之类的,这里有个前置的细节,就是青豆是乘坐一辆很高级的出租车,内饰什么的都非常有质感,有点不像一辆出租车,然后车里放了一首比较小众的歌,雅纳切克的《小交响曲》,但是青豆知道它,这跟后面的情节也有些许关系,这是女主人公青豆的出场;相应的男主的出场印象不是太深刻,男主叫天吾,是个不知名的作家,跟一个叫小松的编辑有比较好的关系,虽然天吾还没有拿到比较有分量的奖项,但是小松很看好他,也让他帮忙审校一个新作家奖的投稿文章,虽然天吾自身还没获得过这个奖,天吾还有个正式工作,是当数学老师,天吾在学生时代是个数学天才,但后面有对文学产生了兴趣,文学还不足以养活自己,靠着教课还是能保持温饱;</p>
<p>接下来是正式故事的起点了,就是小松收到了一部小说投稿,名叫《空气蛹》,是个叫深绘里的女孩子投的稿,小松对他赋予了很高的评价,这里好像记岔了,好像是天吾对这部小说很有好感,但是小松比较怀疑,然后小松看了之后也有了浓厚的兴趣,这里就是开端了,小松想让天吾来重写润色这部《空气蛹》,因为故事本身很有分量,但是描写手法叙事方式等都很拙劣,而天吾正好擅长这个,小松对天吾的评价是,描写技巧无可挑剔,就是故事主体的火花还没际遇迸发,需要一个导火索,这个就可以类比我们程序员,很多比较初中级的程序员主要擅长在原来的代码上修修改改或者给他分配个小功能,比较高级的程序员就需要能做一些项目的架构设计,核心的技术方案设计,以前我也觉得写文档这个比较无聊,但是当一个项目真的比较庞大,复杂的时候,整体和核心部分的架构设计和方案还是需要有文档沉淀的,不然别人不知道没法接受,自己过段时间也会忘记。</p>
<p>对于小松的这个建议,他的初衷是想搅一搅这个死气沉沉套路颇深的文坛,因为本身《空气蛹》这部小说的内容很吸引人,小松想通过天吾的润色补充让这部小说冲击新人奖,有种恶作剧的意图,天吾对此表示很多担心和顾虑,小松的这个建议其实也是一种文学作假,有两方面的担心,一方面是原作者深绘里是否同意如此操作,一方面是外界如果发现了这个事实会有什么样的后果,但是小松表示不用担心,前一步由小松牵线,让天吾跟原作者深绘里当面沟通这个代写是否被允许,结果当然是被允许了,这里有了对深绘里的初步描写,按我的理解是比较仙的感觉,然后语言沟通有些吃力,或者说有她自己的特色,当面沟通时貌似是让深绘里回去再考虑下,然后后面再由天吾去深绘里寄宿的戎野老师家沟通具体的细节。</p>
<p>2019年12月18日23:37:19 更新<br>去到戎野老师家之后,天吾知道了关于深绘里的一些事情,深绘里的父亲与戎野老师应该是老友,深绘里的父亲在当初成立了一个叫”先驱”的公社,一个独立运行的社会组织,以运营农场作为物资来源,追求更为松散的共同体,即不过分激进地公有制,进行松散的共同生活,承认私有财产,简而言之就是这样一个能稳定存活下来的独立社会组织,但是随着稳定运行,内部的激进派和稳健派开始出现分歧,不可磨合,后来两派就分裂了,深绘里的父亲,深田保留在了稳健派,但是此时其实深田保内心是矛盾的,以为一开始其实是他倡导的独立革命才组织起了这群人,然而现在他又认清了现实社会已经不太相信能通过革命来独立的可能性,后来激进派便开始越加封闭,而且进行军事训练和思想教育,而后这个先驱的激进派别便有了新的名字”黎明”,深绘里也是在此时从先驱逃离来投靠戎野老师<br>暂时先写到这,未完待续~</p>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2019/12/10/Redis-Part-1/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2019/12/10/Redis-Part-1/" class="post-title-link" itemprop="url">Redis_分布式锁</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2019-12-10 23:26:26" itemprop="dateCreated datePublished" datetime="2019-12-10T23:26:26+08:00">2019-12-10</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2020-01-12 21:08:27" itemprop="dateModified" datetime="2020-01-12T21:08:27+08:00">2020-01-12</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</span>
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/Redis/" itemprop="url" rel="index">
<span itemprop="name">Redis</span>
</a>
</span>
,
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/Redis/Distributed-Lock/" itemprop="url" rel="index">
<span itemprop="name">Distributed Lock</span>
</a>
</span>
,
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/C/" itemprop="url" rel="index">
<span itemprop="name">C</span>
</a>
</span>
,
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/C/Redis/" itemprop="url" rel="index">
<span itemprop="name">Redis</span>
</a>
</span>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2019/12/10/Redis-Part-1/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2019/12/10/Redis-Part-1/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<p>今天看了一下 <code>redis</code> 分布式锁 <code>redlock</code> 的实现,简单记录下,</p>
<h2 id="加锁"><a href="#加锁" class="headerlink" title="加锁"></a>加锁</h2><p>原先我对 <code>redis</code> 锁的概念就是加锁使用 <code>setnx</code>,解锁使用 <code>lua</code> 脚本,但是 <code>setnx</code> 具体是啥,<code>lua</code> 脚本是啥不是很清楚<br>首先简单思考下这个问题,首先为啥不是先 <code>get</code> 一下 <code>key</code> 存不存在,然后再 <code>set</code> 一个 <code>key value</code>,因为加锁这个操作我们是要保证两点,一个是不能中途被打断,也就是说要原子性,如果是先 <code>get</code> 一下 <code>key</code>,如果不存在再 <code>set</code> 值的话,那就不是原子操作了;第二个是可不可以直接 <code>set</code> 值呢,显然不行,锁要保证唯一性,有且只能有一个线程或者其他应用单位获得该锁,正好 <code>setnx</code> 给了我们这种原子命令</p>
<p>然后是 <code>setnx</code> 的键和值分别是啥,键比较容易想到是要锁住的资源,比如 <code>user_id</code>, 这里有个我自己之前比较容易陷进去的误区,但是这个误区后<br>面再说,这里其实是把<code>user_id</code> 作为要锁住的资源,在我获得锁的时候别的线程不允许操作,以此保证业务的正确性,不会被多个线程同时修改,确定了键,再来看看值是啥,其实原先我认为值是啥都没关系,我只要锁住了,光键就够我用了,但是考虑下多个线程的问题,如果我这个线程加了锁,然后我因为 <code>gc</code> 停顿等原因卡死了,这个时候<code>redis</code> 的锁或者说就是 <code>redis</code> 的缓存已经过期了,这时候另一个线程获得锁成功,然后我这个线程又活过来了,然后我就仍然认为我拿着锁,我去对数据进行修改或者释放锁,是不是就出现问题了,所以是不是我们还需要一个东西来区分这个锁是哪个线程加的,所以我们可以将值设置成为一个线程独有识别的值,至少在相对长的一段时间内不会重复。</p>
<p>上面其实还有两个问题,一个是当 <code>gc</code> 超时时,我这个线程如何知道我手里的锁已经过期了,一种方法是我在加好锁之后就维护一个超时时间,这里其实还有个问题,不过跟第二个问题相关,就一起说了,就是设置超时时间,有些对于不是锁的 <code>redis</code> 缓存操作可以是先设置好值,然后在设置过期时间,那么这就又有上面说到的不是原子性的问题,那么就需要在同一条指令里把超时时间也设置了,幸好 <code>redis</code> 提供了这种支持</p>
<figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">SET resource_name my_random_value NX PX 30000</span><br></pre></td></tr></table></figure>
<p>这里借鉴一下解释下,<code>resource_name</code>就是 key,代表要锁住的东西,<code>my_random_value</code>就是识别我这个线程的,<code>NX</code>代表只有在不存在的时候才设置,然后<code>PX 30000</code>表示超时时间是 30秒自动过期</p>
<p>PS:记录下我原先有的一个误区,是不是要用 key 来区分加锁的线程,这样只有一个用处,就是自身线程可以识别是否是自己加的锁,但是最大的问题是别的线程不知道,其实这个用户的出发点是我在担心前面提过的一个问题,就是当 gc 停顿后,我要去判断当前的这个锁是否是我加的,还有就是当释放锁的时候,如果保证不会错误释放了其他线程加的锁,但是这样附带很多其他问题,最大的就是其他线程怎么知道能不能加这个锁。</p>
<h2 id="解锁"><a href="#解锁" class="headerlink" title="解锁"></a>解锁</h2><p>当线程在锁过期之前就处理完了业务逻辑,那就可以提前释放这个锁,那么提前释放要怎么操作,直接<code>del key</code>显然是不行的,因为这样就是我前面想用线程随机值加资源名作为锁的初衷,我不能去释放别的线程加的锁,那么我要怎么办呢,先 <code>get</code> 一下看是不是我的?那又变成非原子的操作了,幸好redis 也考虑到了这个问题,给了<code>lua</code> 脚本来操作这种</p>
<figure class="highlight lua"><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><td class="code"><pre><span class="line"><span class="keyword">if</span> redis.call(<span class="string">"get"</span>,KEYS[<span class="number">1</span>]) == ARGV[<span class="number">1</span>] <span class="keyword">then</span></span><br><span class="line"> <span class="keyword">return</span> redis.call(<span class="string">"del"</span>,KEYS[<span class="number">1</span>])</span><br><span class="line"><span class="keyword">else</span></span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span></span><br><span class="line"><span class="keyword">end</span></span><br></pre></td></tr></table></figure>
<p>这里的<code>KEYS[1]</code>就是前面加锁的<code>resource_name</code>,<code>ARGV[1]</code>就是线程的随机值<code>my_random_value</code></p>
<h2 id="多节点"><a href="#多节点" class="headerlink" title="多节点"></a>多节点</h2><p>前面说的其实是单节点 <code>redis</code> 作为分布式锁的情况,那么当我们的 <code>redis</code> 有多节点的情况呢,如果多节点下处于加锁或者解锁或者锁有效情况下<br><code>redis</code> 的某个节点宕掉了怎么办,这里就有一些需要思考的地方,是否单独搞一个单节点的 <code>redis</code>作为分布式锁专用的,但是如果这个单节点的挂了呢,还有就是成本问题,所以我们需要一个多节点的分布式锁方案<br>这里就引出了开头说到的<code>redlock</code>,这个可是 <code>redis</code>的作者写的, 他的加锁过程是分以下几步去做这个事情</p>
<ul>
<li>获取当前时间(毫秒数)。</li>
<li>按顺序依次向N个Redis节点执行获取锁的操作。这个获取操作跟前面基于单Redis节点的获取锁的过程相同,包含随机字符串my_random_value,也包含过期时间(比如PX 30000,即锁的有效时间)。为了保证在某个Redis节点不可用的时候算法能够继续运行,这个获取锁的操作还有一个超时时间(time out),它要远小于锁的有效时间(几十毫秒量级)。客户端在向某个Redis节点获取锁失败以后,应该立即尝试下一个Redis节点。这里的失败,应该包含任何类型的失败,比如该Redis节点不可用,或者该Redis节点上的锁已经被其它客户端持有(注:Redlock原文中这里只提到了Redis节点不可用的情况,但也应该包含其它的失败情况)。</li>
<li>计算整个获取锁的过程总共消耗了多长时间,计算方法是用当前时间减去第1步记录的时间。如果客户端从大多数Redis节点(&gt;= N/2+1)成功获取到了锁,并且获取锁总共消耗的时间没有超过锁的有效时间(lock validity time),那么这时客户端才认为最终获取锁成功;否则,认为最终获取锁失败。</li>
<li>如果最终获取锁成功了,那么这个锁的有效时间应该重新计算,它等于最初的锁的有效时间减去第3步计算出来的获取锁消耗的时间。</li>
<li>如果最终获取锁失败了(可能由于获取到锁的Redis节点个数少于N/2+1,或者整个获取锁的过程消耗的时间超过了锁的最初有效时间),那么客户端应该立即向所有Redis节点发起释放锁的操作(即前面介绍的Redis Lua脚本)。<br>释放锁的过程比较简单:客户端向所有Redis节点发起释放锁的操作,不管这些节点当时在获取锁的时候成功与否。这里为什么要向所有的节点发送释放锁的操作呢,这里是因为有部分的节点的失败原因可能是加锁时阻塞,加锁成功的结果没有及时返回,所以为了防止这种情况还是需要向所有发起这个释放锁的操作。<br>初步记录就先到这。</li>
</ul>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2019/12/07/JVM-G1-Part-1/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2019/12/07/JVM-G1-Part-1/" class="post-title-link" itemprop="url">JVM源码分析之G1垃圾收集器分析一</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2019-12-07 00:54:19" itemprop="dateCreated datePublished" datetime="2019-12-07T00:54:19+08:00">2019-12-07</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2019-12-20 00:01:42" itemprop="dateModified" datetime="2019-12-20T00:01:42+08:00">2019-12-20</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</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/C/" itemprop="url" rel="index">
<span itemprop="name">C++</span>
</a>
</span>
,
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/Java/JVM/" itemprop="url" rel="index">
<span itemprop="name">JVM</span>
</a>
</span>
,
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/Java/GC/" itemprop="url" rel="index">
<span itemprop="name">GC</span>
</a>
</span>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2019/12/07/JVM-G1-Part-1/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2019/12/07/JVM-G1-Part-1/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<p>对 Java 的 gc 实现比较感兴趣,原先一般都是看周志明的书,但其实并没有讲具体的 gc 源码,而是把整个思路和流程讲解了一下<br>特别是 G1 的具体实现<br>一般对 G1 的理解其实就是把原先整块的新生代老年代分成了以 region 为单位的小块内存,简而言之,就是原先对新生代老年代的收集会涉及到整个代的堆内存空间,而G1 把它变成了更细致的小块内存<br>这带来了一个很明显的好处和一个很明显的坏处,好处是内存收集可以更灵活,耗时会变短,但整个收集的处理复杂度就变高了<br>目前看了一点点关于 G1 收集的预期时间相关的代码</p>
<figure class="highlight c++"><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></pre></td><td class="code"><pre><span class="line">HeapWord* G1CollectedHeap::do_collection_pause(<span class="keyword">size_t</span> word_size,</span><br><span class="line"> uint gc_count_before,</span><br><span class="line"> <span class="keyword">bool</span>* succeeded,</span><br><span class="line"> GCCause::Cause gc_cause) &#123;</span><br><span class="line"> assert_heap_not_locked_and_not_at_safepoint();</span><br><span class="line"> <span class="function">VM_G1CollectForAllocation <span class="title">op</span><span class="params">(word_size,</span></span></span><br><span class="line"><span class="function"><span class="params"> gc_count_before,</span></span></span><br><span class="line"><span class="function"><span class="params"> gc_cause,</span></span></span><br><span class="line"><span class="function"><span class="params"> <span class="literal">false</span>, <span class="comment">/* should_initiate_conc_mark */</span></span></span></span><br><span class="line"><span class="function"><span class="params"> g1_policy()-&gt;max_pause_time_ms())</span></span>;</span><br><span class="line"> VMThread::execute(&amp;op);</span><br><span class="line"></span><br><span class="line"> HeapWord* result = op.result();</span><br><span class="line"> <span class="keyword">bool</span> ret_succeeded = op.prologue_succeeded() &amp;&amp; op.pause_succeeded();</span><br><span class="line"> assert(result == <span class="literal">NULL</span> || ret_succeeded,</span><br><span class="line"> <span class="string">"the result should be NULL if the VM did not succeed"</span>);</span><br><span class="line"> *succeeded = ret_succeeded;</span><br><span class="line"></span><br><span class="line"> assert_heap_not_locked();</span><br><span class="line"> <span class="keyword">return</span> result;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>这里就是收集时需要停顿的,其中<code>VMThread::execute(&amp;op);</code>是具体执行的,真正执行的是<code>VM_G1CollectForAllocation::doit</code>方法</p>
<figure class="highlight c++"><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">void</span> VM_G1CollectForAllocation::doit() &#123;</span><br><span class="line"> G1CollectedHeap* g1h = G1CollectedHeap::heap();</span><br><span class="line"> assert(!_should_initiate_conc_mark || g1h-&gt;should_do_concurrent_full_gc(_gc_cause),</span><br><span class="line"> <span class="string">"only a GC locker, a System.gc(), stats update, whitebox, or a hum allocation induced GC should start a cycle"</span>);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (_word_size &gt; <span class="number">0</span>) &#123;</span><br><span class="line"> <span class="comment">// An allocation has been requested. So, try to do that first.</span></span><br><span class="line"> _result = g1h-&gt;attempt_allocation_at_safepoint(_word_size,</span><br><span class="line"> <span class="literal">false</span> <span class="comment">/* expect_null_cur_alloc_region */</span>);</span><br><span class="line"> <span class="keyword">if</span> (_result != <span class="literal">NULL</span>) &#123;</span><br><span class="line"> <span class="comment">// If we can successfully allocate before we actually do the</span></span><br><span class="line"> <span class="comment">// pause then we will consider this pause successful.</span></span><br><span class="line"> _pause_succeeded = <span class="literal">true</span>;</span><br><span class="line"> <span class="keyword">return</span>;</span><br><span class="line"> &#125;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function">GCCauseSetter <span class="title">x</span><span class="params">(g1h, _gc_cause)</span></span>;</span><br><span class="line"> <span class="keyword">if</span> (_should_initiate_conc_mark) &#123;</span><br><span class="line"> <span class="comment">// It's safer to read old_marking_cycles_completed() here, given</span></span><br><span class="line"> <span class="comment">// that noone else will be updating it concurrently. Since we'll</span></span><br><span class="line"> <span class="comment">// only need it if we're initiating a marking cycle, no point in</span></span><br><span class="line"> <span class="comment">// setting it earlier.</span></span><br><span class="line"> _old_marking_cycles_completed_before = g1h-&gt;old_marking_cycles_completed();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// At this point we are supposed to start a concurrent cycle. We</span></span><br><span class="line"> <span class="comment">// will do so if one is not already in progress.</span></span><br><span class="line"> <span class="keyword">bool</span> res = g1h-&gt;g1_policy()-&gt;force_initial_mark_if_outside_cycle(_gc_cause);</span><br><span class="line"></span><br><span class="line"> <span class="comment">// The above routine returns true if we were able to force the</span></span><br><span class="line"> <span class="comment">// next GC pause to be an initial mark; it returns false if a</span></span><br><span class="line"> <span class="comment">// marking cycle is already in progress.</span></span><br><span class="line"> <span class="comment">//</span></span><br><span class="line"> <span class="comment">// If a marking cycle is already in progress just return and skip the</span></span><br><span class="line"> <span class="comment">// pause below - if the reason for requesting this initial mark pause</span></span><br><span class="line"> <span class="comment">// was due to a System.gc() then the requesting thread should block in</span></span><br><span class="line"> <span class="comment">// doit_epilogue() until the marking cycle is complete.</span></span><br><span class="line"> <span class="comment">//</span></span><br><span class="line"> <span class="comment">// If this initial mark pause was requested as part of a humongous</span></span><br><span class="line"> <span class="comment">// allocation then we know that the marking cycle must just have</span></span><br><span class="line"> <span class="comment">// been started by another thread (possibly also allocating a humongous</span></span><br><span class="line"> <span class="comment">// object) as there was no active marking cycle when the requesting</span></span><br><span class="line"> <span class="comment">// thread checked before calling collect() in</span></span><br><span class="line"> <span class="comment">// attempt_allocation_humongous(). Retrying the GC, in this case,</span></span><br><span class="line"> <span class="comment">// will cause the requesting thread to spin inside collect() until the</span></span><br><span class="line"> <span class="comment">// just started marking cycle is complete - which may be a while. So</span></span><br><span class="line"> <span class="comment">// we do NOT retry the GC.</span></span><br><span class="line"> <span class="keyword">if</span> (!res) &#123;</span><br><span class="line"> assert(_word_size == <span class="number">0</span>, <span class="string">"Concurrent Full GC/Humongous Object IM shouldn't be allocating"</span>);</span><br><span class="line"> <span class="keyword">if</span> (_gc_cause != GCCause::_g1_humongous_allocation) &#123;</span><br><span class="line"> _should_retry_gc = <span class="literal">true</span>;</span><br><span class="line"> &#125;</span><br><span class="line"> <span class="keyword">return</span>;</span><br><span class="line"> &#125;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Try a partial collection of some kind.</span></span><br><span class="line"> _pause_succeeded = g1h-&gt;do_collection_pause_at_safepoint(_target_pause_time_ms);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (_pause_succeeded) &#123;</span><br><span class="line"> <span class="keyword">if</span> (_word_size &gt; <span class="number">0</span>) &#123;</span><br><span class="line"> <span class="comment">// An allocation had been requested. Do it, eventually trying a stronger</span></span><br><span class="line"> <span class="comment">// kind of GC.</span></span><br><span class="line"> _result = g1h-&gt;satisfy_failed_allocation(_word_size, &amp;_pause_succeeded);</span><br><span class="line"> &#125; <span class="keyword">else</span> &#123;</span><br><span class="line"> <span class="keyword">bool</span> should_upgrade_to_full = !g1h-&gt;should_do_concurrent_full_gc(_gc_cause) &amp;&amp;</span><br><span class="line"> !g1h-&gt;has_regions_left_for_allocation();</span><br><span class="line"> <span class="keyword">if</span> (should_upgrade_to_full) &#123;</span><br><span class="line"> <span class="comment">// There has been a request to perform a GC to free some space. We have no</span></span><br><span class="line"> <span class="comment">// information on how much memory has been asked for. In case there are</span></span><br><span class="line"> <span class="comment">// absolutely no regions left to allocate into, do a maximally compacting full GC.</span></span><br><span class="line"> log_info(gc, ergo)(<span class="string">"Attempting maximally compacting collection"</span>);</span><br><span class="line"> _pause_succeeded = g1h-&gt;do_full_collection(<span class="literal">false</span>, <span class="comment">/* explicit gc */</span></span><br><span class="line"> <span class="literal">true</span> <span class="comment">/* clear_all_soft_refs */</span>);</span><br><span class="line"> &#125;</span><br><span class="line"> &#125;</span><br><span class="line"> guarantee(_pause_succeeded, <span class="string">"Elevated collections during the safepoint must always succeed."</span>);</span><br><span class="line"> &#125; <span class="keyword">else</span> &#123;</span><br><span class="line"> assert(_result == <span class="literal">NULL</span>, <span class="string">"invariant"</span>);</span><br><span class="line"> <span class="comment">// The only reason for the pause to not be successful is that, the GC locker is</span></span><br><span class="line"> <span class="comment">// active (or has become active since the prologue was executed). In this case</span></span><br><span class="line"> <span class="comment">// we should retry the pause after waiting for the GC locker to become inactive.</span></span><br><span class="line"> _should_retry_gc = <span class="literal">true</span>;</span><br><span class="line"> &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>这里可以看到核心的是<code>G1CollectedHeap::do_collection_pause_at_safepoint</code>这个方法,它带上了目标暂停时间的值</p>
<figure class="highlight c++"><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><span class="line">216</span><br><span class="line">217</span><br><span class="line">218</span><br><span class="line">219</span><br><span class="line">220</span><br><span class="line">221</span><br><span class="line">222</span><br><span class="line">223</span><br><span class="line">224</span><br><span class="line">225</span><br><span class="line">226</span><br><span class="line">227</span><br><span class="line">228</span><br><span class="line">229</span><br><span class="line">230</span><br><span class="line">231</span><br><span class="line">232</span><br><span class="line">233</span><br><span class="line">234</span><br><span class="line">235</span><br><span class="line">236</span><br><span class="line">237</span><br><span class="line">238</span><br><span class="line">239</span><br><span class="line">240</span><br><span class="line">241</span><br><span class="line">242</span><br><span class="line">243</span><br><span class="line">244</span><br><span class="line">245</span><br><span class="line">246</span><br><span class="line">247</span><br><span class="line">248</span><br><span class="line">249</span><br><span class="line">250</span><br><span class="line">251</span><br><span class="line">252</span><br><span class="line">253</span><br><span class="line">254</span><br><span class="line">255</span><br><span class="line">256</span><br><span class="line">257</span><br><span class="line">258</span><br><span class="line">259</span><br><span class="line">260</span><br><span class="line">261</span><br><span class="line">262</span><br><span class="line">263</span><br><span class="line">264</span><br><span class="line">265</span><br><span class="line">266</span><br><span class="line">267</span><br><span class="line">268</span><br><span class="line">269</span><br><span class="line">270</span><br><span class="line">271</span><br><span class="line">272</span><br><span class="line">273</span><br><span class="line">274</span><br><span class="line">275</span><br><span class="line">276</span><br><span class="line">277</span><br><span class="line">278</span><br><span class="line">279</span><br><span class="line">280</span><br><span class="line">281</span><br><span class="line">282</span><br><span class="line">283</span><br><span class="line">284</span><br><span class="line">285</span><br><span class="line">286</span><br><span class="line">287</span><br><span class="line">288</span><br><span class="line">289</span><br><span class="line">290</span><br><span class="line">291</span><br><span class="line">292</span><br><span class="line">293</span><br><span class="line">294</span><br><span class="line">295</span><br><span class="line">296</span><br><span class="line">297</span><br><span class="line">298</span><br><span class="line">299</span><br><span class="line">300</span><br><span class="line">301</span><br><span class="line">302</span><br><span class="line">303</span><br><span class="line">304</span><br><span class="line">305</span><br><span class="line">306</span><br><span class="line">307</span><br><span class="line">308</span><br><span class="line">309</span><br><span class="line">310</span><br><span class="line">311</span><br><span class="line">312</span><br><span class="line">313</span><br><span class="line">314</span><br><span class="line">315</span><br><span class="line">316</span><br><span class="line">317</span><br><span class="line">318</span><br><span class="line">319</span><br><span class="line">320</span><br><span class="line">321</span><br><span class="line">322</span><br><span class="line">323</span><br><span class="line">324</span><br><span class="line">325</span><br><span class="line">326</span><br><span class="line">327</span><br><span class="line">328</span><br><span class="line">329</span><br><span class="line">330</span><br></pre></td><td class="code"><pre><span class="line">G1CollectedHeap::do_collection_pause_at_safepoint(<span class="keyword">double</span> target_pause_time_ms) &#123;</span><br><span class="line"> assert_at_safepoint_on_vm_thread();</span><br><span class="line"> guarantee(!is_gc_active(), <span class="string">"collection is not reentrant"</span>);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (GCLocker::check_active_before_gc()) &#123;</span><br><span class="line"> <span class="keyword">return</span> <span class="literal">false</span>;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> _gc_timer_stw-&gt;register_gc_start();</span><br><span class="line"></span><br><span class="line"> GCIdMark gc_id_mark;</span><br><span class="line"> _gc_tracer_stw-&gt;report_gc_start(gc_cause(), _gc_timer_stw-&gt;gc_start());</span><br><span class="line"></span><br><span class="line"> <span class="function">SvcGCMarker <span class="title">sgcm</span><span class="params">(SvcGCMarker::MINOR)</span></span>;</span><br><span class="line"> ResourceMark rm;</span><br><span class="line"></span><br><span class="line"> g1_policy()-&gt;note_gc_start();</span><br><span class="line"></span><br><span class="line"> wait_for_root_region_scanning();</span><br><span class="line"></span><br><span class="line"> print_heap_before_gc();</span><br><span class="line"> print_heap_regions();</span><br><span class="line"> trace_heap_before_gc(_gc_tracer_stw);</span><br><span class="line"></span><br><span class="line"> _verifier-&gt;verify_region_sets_optional();</span><br><span class="line"> _verifier-&gt;verify_dirty_young_regions();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// We should not be doing initial mark unless the conc mark thread is running</span></span><br><span class="line"> <span class="keyword">if</span> (!_cm_thread-&gt;should_terminate()) &#123;</span><br><span class="line"> <span class="comment">// This call will decide whether this pause is an initial-mark</span></span><br><span class="line"> <span class="comment">// pause. If it is, in_initial_mark_gc() will return true</span></span><br><span class="line"> <span class="comment">// for the duration of this pause.</span></span><br><span class="line"> g1_policy()-&gt;decide_on_conc_mark_initiation();</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// We do not allow initial-mark to be piggy-backed on a mixed GC.</span></span><br><span class="line"> assert(!collector_state()-&gt;in_initial_mark_gc() ||</span><br><span class="line"> collector_state()-&gt;in_young_only_phase(), <span class="string">"sanity"</span>);</span><br><span class="line"></span><br><span class="line"> <span class="comment">// We also do not allow mixed GCs during marking.</span></span><br><span class="line"> assert(!collector_state()-&gt;mark_or_rebuild_in_progress() || collector_state()-&gt;in_young_only_phase(), <span class="string">"sanity"</span>);</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Record whether this pause is an initial mark. When the current</span></span><br><span class="line"> <span class="comment">// thread has completed its logging output and it's safe to signal</span></span><br><span class="line"> <span class="comment">// the CM thread, the flag's value in the policy has been reset.</span></span><br><span class="line"> <span class="keyword">bool</span> should_start_conc_mark = collector_state()-&gt;in_initial_mark_gc();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Inner scope for scope based logging, timers, and stats collection</span></span><br><span class="line"> &#123;</span><br><span class="line"> EvacuationInfo evacuation_info;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (collector_state()-&gt;in_initial_mark_gc()) &#123;</span><br><span class="line"> <span class="comment">// We are about to start a marking cycle, so we increment the</span></span><br><span class="line"> <span class="comment">// full collection counter.</span></span><br><span class="line"> increment_old_marking_cycles_started();</span><br><span class="line"> _cm-&gt;gc_tracer_cm()-&gt;set_gc_cause(gc_cause());</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> _gc_tracer_stw-&gt;report_yc_type(collector_state()-&gt;yc_type());</span><br><span class="line"></span><br><span class="line"> GCTraceCPUTime tcpu;</span><br><span class="line"></span><br><span class="line"> G1HeapVerifier::G1VerifyType verify_type;</span><br><span class="line"> FormatBuffer&lt;&gt; gc_string(<span class="string">"Pause Young "</span>);</span><br><span class="line"> <span class="keyword">if</span> (collector_state()-&gt;in_initial_mark_gc()) &#123;</span><br><span class="line"> gc_string.append(<span class="string">"(Concurrent Start)"</span>);</span><br><span class="line"> verify_type = G1HeapVerifier::G1VerifyConcurrentStart;</span><br><span class="line"> &#125; <span class="keyword">else</span> <span class="keyword">if</span> (collector_state()-&gt;in_young_only_phase()) &#123;</span><br><span class="line"> <span class="keyword">if</span> (collector_state()-&gt;in_young_gc_before_mixed()) &#123;</span><br><span class="line"> gc_string.append(<span class="string">"(Prepare Mixed)"</span>);</span><br><span class="line"> &#125; <span class="keyword">else</span> &#123;</span><br><span class="line"> gc_string.append(<span class="string">"(Normal)"</span>);</span><br><span class="line"> &#125;</span><br><span class="line"> verify_type = G1HeapVerifier::G1VerifyYoungNormal;</span><br><span class="line"> &#125; <span class="keyword">else</span> &#123;</span><br><span class="line"> gc_string.append(<span class="string">"(Mixed)"</span>);</span><br><span class="line"> verify_type = G1HeapVerifier::G1VerifyMixed;</span><br><span class="line"> &#125;</span><br><span class="line"> GCTraceTime(Info, gc) tm(gc_string, <span class="literal">NULL</span>, gc_cause(), <span class="literal">true</span>);</span><br><span class="line"></span><br><span class="line"> uint active_workers = AdaptiveSizePolicy::calc_active_workers(workers()-&gt;total_workers(),</span><br><span class="line"> workers()-&gt;active_workers(),</span><br><span class="line"> Threads::number_of_non_daemon_threads());</span><br><span class="line"> active_workers = workers()-&gt;update_active_workers(active_workers);</span><br><span class="line"> log_info(gc,task)(<span class="string">"Using %u workers of %u for evacuation"</span>, active_workers, workers()-&gt;total_workers());</span><br><span class="line"></span><br><span class="line"> <span class="function">TraceCollectorStats <span class="title">tcs</span><span class="params">(g1mm()-&gt;incremental_collection_counters())</span></span>;</span><br><span class="line"> <span class="function">TraceMemoryManagerStats <span class="title">tms</span><span class="params">(&amp;_memory_manager, gc_cause(),</span></span></span><br><span class="line"><span class="function"><span class="params"> collector_state()-&gt;yc_type() == Mixed <span class="comment">/* allMemoryPoolsAffected */</span>)</span></span>;</span><br><span class="line"></span><br><span class="line"> <span class="function">G1HeapTransition <span class="title">heap_transition</span><span class="params">(<span class="keyword">this</span>)</span></span>;</span><br><span class="line"> <span class="keyword">size_t</span> heap_used_bytes_before_gc = used();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Don't dynamically change the number of GC threads this early. A value of</span></span><br><span class="line"> <span class="comment">// 0 is used to indicate serial work. When parallel work is done,</span></span><br><span class="line"> <span class="comment">// it will be set.</span></span><br><span class="line"></span><br><span class="line"> &#123; <span class="comment">// Call to jvmpi::post_class_unload_events must occur outside of active GC</span></span><br><span class="line"> IsGCActiveMark x;</span><br><span class="line"></span><br><span class="line"> gc_prologue(<span class="literal">false</span>);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (VerifyRememberedSets) &#123;</span><br><span class="line"> log_info(gc, verify)(<span class="string">"[Verifying RemSets before GC]"</span>);</span><br><span class="line"> VerifyRegionRemSetClosure v_cl;</span><br><span class="line"> heap_region_iterate(&amp;v_cl);</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> _verifier-&gt;verify_before_gc(verify_type);</span><br><span class="line"></span><br><span class="line"> _verifier-&gt;check_bitmaps(<span class="string">"GC Start"</span>);</span><br><span class="line"></span><br><span class="line"><span class="meta">#<span class="meta-keyword">if</span> COMPILER2_OR_JVMCI</span></span><br><span class="line"> DerivedPointerTable::<span class="built_in">clear</span>();</span><br><span class="line"><span class="meta">#<span class="meta-keyword">endif</span></span></span><br><span class="line"></span><br><span class="line"> <span class="comment">// Please see comment in g1CollectedHeap.hpp and</span></span><br><span class="line"> <span class="comment">// G1CollectedHeap::ref_processing_init() to see how</span></span><br><span class="line"> <span class="comment">// reference processing currently works in G1.</span></span><br><span class="line"></span><br><span class="line"> <span class="comment">// Enable discovery in the STW reference processor</span></span><br><span class="line"> _ref_processor_stw-&gt;enable_discovery();</span><br><span class="line"></span><br><span class="line"> &#123;</span><br><span class="line"> <span class="comment">// We want to temporarily turn off discovery by the</span></span><br><span class="line"> <span class="comment">// CM ref processor, if necessary, and turn it back on</span></span><br><span class="line"> <span class="comment">// on again later if we do. Using a scoped</span></span><br><span class="line"> <span class="comment">// NoRefDiscovery object will do this.</span></span><br><span class="line"> <span class="function">NoRefDiscovery <span class="title">no_cm_discovery</span><span class="params">(_ref_processor_cm)</span></span>;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Forget the current alloc region (we might even choose it to be part</span></span><br><span class="line"> <span class="comment">// of the collection set!).</span></span><br><span class="line"> _allocator-&gt;release_mutator_alloc_region();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// This timing is only used by the ergonomics to handle our pause target.</span></span><br><span class="line"> <span class="comment">// It is unclear why this should not include the full pause. We will</span></span><br><span class="line"> <span class="comment">// investigate this in CR 7178365.</span></span><br><span class="line"> <span class="comment">//</span></span><br><span class="line"> <span class="comment">// Preserving the old comment here if that helps the investigation:</span></span><br><span class="line"> <span class="comment">//</span></span><br><span class="line"> <span class="comment">// The elapsed time induced by the start time below deliberately elides</span></span><br><span class="line"> <span class="comment">// the possible verification above.</span></span><br><span class="line"> <span class="keyword">double</span> sample_start_time_sec = os::elapsedTime();</span><br><span class="line"></span><br><span class="line"> g1_policy()-&gt;record_collection_pause_start(sample_start_time_sec);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (collector_state()-&gt;in_initial_mark_gc()) &#123;</span><br><span class="line"> concurrent_mark()-&gt;pre_initial_mark();</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> g1_policy()-&gt;finalize_collection_set(target_pause_time_ms, &amp;_survivor);</span><br><span class="line"></span><br><span class="line"> evacuation_info.set_collectionset_regions(collection_set()-&gt;region_length());</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Make sure the remembered sets are up to date. This needs to be</span></span><br><span class="line"> <span class="comment">// done before register_humongous_regions_with_cset(), because the</span></span><br><span class="line"> <span class="comment">// remembered sets are used there to choose eager reclaim candidates.</span></span><br><span class="line"> <span class="comment">// If the remembered sets are not up to date we might miss some</span></span><br><span class="line"> <span class="comment">// entries that need to be handled.</span></span><br><span class="line"> g1_rem_set()-&gt;cleanupHRRS();</span><br><span class="line"></span><br><span class="line"> register_humongous_regions_with_cset();</span><br><span class="line"></span><br><span class="line"> assert(_verifier-&gt;check_cset_fast_test(), <span class="string">"Inconsistency in the InCSetState table."</span>);</span><br><span class="line"></span><br><span class="line"> <span class="comment">// We call this after finalize_cset() to</span></span><br><span class="line"> <span class="comment">// ensure that the CSet has been finalized.</span></span><br><span class="line"> _cm-&gt;verify_no_cset_oops();</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (_hr_printer.is_active()) &#123;</span><br><span class="line"> <span class="function">G1PrintCollectionSetClosure <span class="title">cl</span><span class="params">(&amp;_hr_printer)</span></span>;</span><br><span class="line"> _collection_set.iterate(&amp;cl);</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Initialize the GC alloc regions.</span></span><br><span class="line"> _allocator-&gt;init_gc_alloc_regions(evacuation_info);</span><br><span class="line"></span><br><span class="line"> <span class="function">G1ParScanThreadStateSet <span class="title">per_thread_states</span><span class="params">(<span class="keyword">this</span>, workers()-&gt;active_workers(), collection_set()-&gt;young_region_length())</span></span>;</span><br><span class="line"> pre_evacuate_collection_set();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Actually do the work...</span></span><br><span class="line"> evacuate_collection_set(&amp;per_thread_states);</span><br><span class="line"></span><br><span class="line"> post_evacuate_collection_set(evacuation_info, &amp;per_thread_states);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">const</span> <span class="keyword">size_t</span>* surviving_young_words = per_thread_states.surviving_young_words();</span><br><span class="line"> free_collection_set(&amp;_collection_set, evacuation_info, surviving_young_words);</span><br><span class="line"></span><br><span class="line"> eagerly_reclaim_humongous_regions();</span><br><span class="line"></span><br><span class="line"> record_obj_copy_mem_stats();</span><br><span class="line"> _survivor_evac_stats.adjust_desired_plab_sz();</span><br><span class="line"> _old_evac_stats.adjust_desired_plab_sz();</span><br><span class="line"></span><br><span class="line"> <span class="keyword">double</span> start = os::elapsedTime();</span><br><span class="line"> start_new_collection_set();</span><br><span class="line"> g1_policy()-&gt;phase_times()-&gt;record_start_new_cset_time_ms((os::elapsedTime() - start) * <span class="number">1000.0</span>);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (evacuation_failed()) &#123;</span><br><span class="line"> set_used(recalculate_used());</span><br><span class="line"> <span class="keyword">if</span> (_archive_allocator != <span class="literal">NULL</span>) &#123;</span><br><span class="line"> _archive_allocator-&gt;clear_used();</span><br><span class="line"> &#125;</span><br><span class="line"> <span class="keyword">for</span> (uint i = <span class="number">0</span>; i &lt; ParallelGCThreads; i++) &#123;</span><br><span class="line"> <span class="keyword">if</span> (_evacuation_failed_info_array[i].has_failed()) &#123;</span><br><span class="line"> _gc_tracer_stw-&gt;report_evacuation_failed(_evacuation_failed_info_array[i]);</span><br><span class="line"> &#125;</span><br><span class="line"> &#125;</span><br><span class="line"> &#125; <span class="keyword">else</span> &#123;</span><br><span class="line"> <span class="comment">// The "used" of the the collection set have already been subtracted</span></span><br><span class="line"> <span class="comment">// when they were freed. Add in the bytes evacuated.</span></span><br><span class="line"> increase_used(g1_policy()-&gt;bytes_copied_during_gc());</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (collector_state()-&gt;in_initial_mark_gc()) &#123;</span><br><span class="line"> <span class="comment">// We have to do this before we notify the CM threads that</span></span><br><span class="line"> <span class="comment">// they can start working to make sure that all the</span></span><br><span class="line"> <span class="comment">// appropriate initialization is done on the CM object.</span></span><br><span class="line"> concurrent_mark()-&gt;post_initial_mark();</span><br><span class="line"> <span class="comment">// Note that we don't actually trigger the CM thread at</span></span><br><span class="line"> <span class="comment">// this point. We do that later when we're sure that</span></span><br><span class="line"> <span class="comment">// the current thread has completed its logging output.</span></span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> allocate_dummy_regions();</span><br><span class="line"></span><br><span class="line"> _allocator-&gt;init_mutator_alloc_region();</span><br><span class="line"></span><br><span class="line"> &#123;</span><br><span class="line"> <span class="keyword">size_t</span> expand_bytes = _heap_sizing_policy-&gt;expansion_amount();</span><br><span class="line"> <span class="keyword">if</span> (expand_bytes &gt; <span class="number">0</span>) &#123;</span><br><span class="line"> <span class="keyword">size_t</span> bytes_before = capacity();</span><br><span class="line"> <span class="comment">// No need for an ergo logging here,</span></span><br><span class="line"> <span class="comment">// expansion_amount() does this when it returns a value &gt; 0.</span></span><br><span class="line"> <span class="keyword">double</span> expand_ms;</span><br><span class="line"> <span class="keyword">if</span> (!expand(expand_bytes, _workers, &amp;expand_ms)) &#123;</span><br><span class="line"> <span class="comment">// We failed to expand the heap. Cannot do anything about it.</span></span><br><span class="line"> &#125;</span><br><span class="line"> g1_policy()-&gt;phase_times()-&gt;record_expand_heap_time(expand_ms);</span><br><span class="line"> &#125;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// We redo the verification but now wrt to the new CSet which</span></span><br><span class="line"> <span class="comment">// has just got initialized after the previous CSet was freed.</span></span><br><span class="line"> _cm-&gt;verify_no_cset_oops();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// This timing is only used by the ergonomics to handle our pause target.</span></span><br><span class="line"> <span class="comment">// It is unclear why this should not include the full pause. We will</span></span><br><span class="line"> <span class="comment">// investigate this in CR 7178365.</span></span><br><span class="line"> <span class="keyword">double</span> sample_end_time_sec = os::elapsedTime();</span><br><span class="line"> <span class="keyword">double</span> pause_time_ms = (sample_end_time_sec - sample_start_time_sec) * MILLIUNITS;</span><br><span class="line"> <span class="keyword">size_t</span> total_cards_scanned = g1_policy()-&gt;phase_times()-&gt;sum_thread_work_items(G1GCPhaseTimes::ScanRS, G1GCPhaseTimes::ScanRSScannedCards);</span><br><span class="line"> g1_policy()-&gt;record_collection_pause_end(pause_time_ms, total_cards_scanned, heap_used_bytes_before_gc);</span><br><span class="line"></span><br><span class="line"> evacuation_info.set_collectionset_used_before(collection_set()-&gt;bytes_used_before());</span><br><span class="line"> evacuation_info.set_bytes_copied(g1_policy()-&gt;bytes_copied_during_gc());</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (VerifyRememberedSets) &#123;</span><br><span class="line"> log_info(gc, verify)(<span class="string">"[Verifying RemSets after GC]"</span>);</span><br><span class="line"> VerifyRegionRemSetClosure v_cl;</span><br><span class="line"> heap_region_iterate(&amp;v_cl);</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> _verifier-&gt;verify_after_gc(verify_type);</span><br><span class="line"> _verifier-&gt;check_bitmaps(<span class="string">"GC End"</span>);</span><br><span class="line"></span><br><span class="line"> assert(!_ref_processor_stw-&gt;discovery_enabled(), <span class="string">"Postcondition"</span>);</span><br><span class="line"> _ref_processor_stw-&gt;verify_no_references_recorded();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// CM reference discovery will be re-enabled if necessary.</span></span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"><span class="meta">#<span class="meta-keyword">ifdef</span> TRACESPINNING</span></span><br><span class="line"> ParallelTaskTerminator::print_termination_counts();</span><br><span class="line"><span class="meta">#<span class="meta-keyword">endif</span></span></span><br><span class="line"></span><br><span class="line"> gc_epilogue(<span class="literal">false</span>);</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Print the remainder of the GC log output.</span></span><br><span class="line"> <span class="keyword">if</span> (evacuation_failed()) &#123;</span><br><span class="line"> log_info(gc)(<span class="string">"To-space exhausted"</span>);</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> g1_policy()-&gt;print_phases();</span><br><span class="line"> heap_transition.<span class="built_in">print</span>();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// It is not yet to safe to tell the concurrent mark to</span></span><br><span class="line"> <span class="comment">// start as we have some optional output below. We don't want the</span></span><br><span class="line"> <span class="comment">// output from the concurrent mark thread interfering with this</span></span><br><span class="line"> <span class="comment">// logging output either.</span></span><br><span class="line"></span><br><span class="line"> _hrm.verify_optional();</span><br><span class="line"> _verifier-&gt;verify_region_sets_optional();</span><br><span class="line"></span><br><span class="line"> TASKQUEUE_STATS_ONLY(print_taskqueue_stats());</span><br><span class="line"> TASKQUEUE_STATS_ONLY(reset_taskqueue_stats());</span><br><span class="line"></span><br><span class="line"> print_heap_after_gc();</span><br><span class="line"> print_heap_regions();</span><br><span class="line"> trace_heap_after_gc(_gc_tracer_stw);</span><br><span class="line"></span><br><span class="line"> <span class="comment">// We must call G1MonitoringSupport::update_sizes() in the same scoping level</span></span><br><span class="line"> <span class="comment">// as an active TraceMemoryManagerStats object (i.e. before the destructor for the</span></span><br><span class="line"> <span class="comment">// TraceMemoryManagerStats is called) so that the G1 memory pools are updated</span></span><br><span class="line"> <span class="comment">// before any GC notifications are raised.</span></span><br><span class="line"> g1mm()-&gt;update_sizes();</span><br><span class="line"></span><br><span class="line"> _gc_tracer_stw-&gt;report_evacuation_info(&amp;evacuation_info);</span><br><span class="line"> _gc_tracer_stw-&gt;report_tenuring_threshold(_g1_policy-&gt;tenuring_threshold());</span><br><span class="line"> _gc_timer_stw-&gt;register_gc_end();</span><br><span class="line"> _gc_tracer_stw-&gt;report_gc_end(_gc_timer_stw-&gt;gc_end(), _gc_timer_stw-&gt;time_partitions());</span><br><span class="line"> &#125;</span><br><span class="line"> <span class="comment">// It should now be safe to tell the concurrent mark thread to start</span></span><br><span class="line"> <span class="comment">// without its logging output interfering with the logging output</span></span><br><span class="line"> <span class="comment">// that came from the pause.</span></span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (should_start_conc_mark) &#123;</span><br><span class="line"> <span class="comment">// CAUTION: after the doConcurrentMark() call below,</span></span><br><span class="line"> <span class="comment">// the concurrent marking thread(s) could be running</span></span><br><span class="line"> <span class="comment">// concurrently with us. Make sure that anything after</span></span><br><span class="line"> <span class="comment">// this point does not assume that we are the only GC thread</span></span><br><span class="line"> <span class="comment">// running. Note: of course, the actual marking work will</span></span><br><span class="line"> <span class="comment">// not start until the safepoint itself is released in</span></span><br><span class="line"> <span class="comment">// SuspendibleThreadSet::desynchronize().</span></span><br><span class="line"> do_concurrent_mark();</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">return</span> <span class="literal">true</span>;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>往下走就是这一步<code>G1Policy::finalize_collection_set</code>,去处理新生代和老年代</p>
<figure class="highlight c++"><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><td class="code"><pre><span class="line"><span class="keyword">void</span> G1Policy::finalize_collection_set(<span class="keyword">double</span> target_pause_time_ms, G1SurvivorRegions* survivor) &#123;</span><br><span class="line"> <span class="keyword">double</span> time_remaining_ms = _collection_set-&gt;finalize_young_part(target_pause_time_ms, survivor);</span><br><span class="line"> _collection_set-&gt;finalize_old_part(time_remaining_ms);</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>这里分别调用了两个方法,可以看到剩余时间是往下传的,来看一下具体的方法</p>
<figure class="highlight c++"><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><td class="code"><pre><span class="line"><span class="keyword">double</span> G1CollectionSet::finalize_young_part(<span class="keyword">double</span> target_pause_time_ms, G1SurvivorRegions* survivors) &#123;</span><br><span class="line"> <span class="keyword">double</span> young_start_time_sec = os::elapsedTime();</span><br><span class="line"></span><br><span class="line"> finalize_incremental_building();</span><br><span class="line"></span><br><span class="line"> guarantee(target_pause_time_ms &gt; <span class="number">0.0</span>,</span><br><span class="line"> <span class="string">"target_pause_time_ms = %1.6lf should be positive"</span>, target_pause_time_ms);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">size_t</span> pending_cards = _policy-&gt;pending_cards();</span><br><span class="line"> <span class="keyword">double</span> base_time_ms = _policy-&gt;predict_base_elapsed_time_ms(pending_cards);</span><br><span class="line"> <span class="keyword">double</span> time_remaining_ms = MAX2(target_pause_time_ms - base_time_ms, <span class="number">0.0</span>);</span><br><span class="line"></span><br><span class="line"> log_trace(gc, ergo, cset)(<span class="string">"Start choosing CSet. pending cards: "</span> SIZE_FORMAT <span class="string">" predicted base time: %1.2fms remaining time: %1.2fms target pause time: %1.2fms"</span>,</span><br><span class="line"> pending_cards, base_time_ms, time_remaining_ms, target_pause_time_ms);</span><br><span class="line"></span><br><span class="line"> <span class="comment">// The young list is laid with the survivor regions from the previous</span></span><br><span class="line"> <span class="comment">// pause are appended to the RHS of the young list, i.e.</span></span><br><span class="line"> <span class="comment">// [Newly Young Regions ++ Survivors from last pause].</span></span><br><span class="line"></span><br><span class="line"> uint survivor_region_length = survivors-&gt;length();</span><br><span class="line"> uint eden_region_length = _g1h-&gt;eden_regions_count();</span><br><span class="line"> init_region_lengths(eden_region_length, survivor_region_length);</span><br><span class="line"></span><br><span class="line"> verify_young_cset_indices();</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Clear the fields that point to the survivor list - they are all young now.</span></span><br><span class="line"> survivors-&gt;convert_to_eden();</span><br><span class="line"></span><br><span class="line"> _bytes_used_before = _inc_bytes_used_before;</span><br><span class="line"> time_remaining_ms = MAX2(time_remaining_ms - _inc_predicted_elapsed_time_ms, <span class="number">0.0</span>);</span><br><span class="line"></span><br><span class="line"> log_trace(gc, ergo, cset)(<span class="string">"Add young regions to CSet. eden: %u regions, survivors: %u regions, predicted young region time: %1.2fms, target pause time: %1.2fms"</span>,</span><br><span class="line"> eden_region_length, survivor_region_length, _inc_predicted_elapsed_time_ms, target_pause_time_ms);</span><br><span class="line"></span><br><span class="line"> <span class="comment">// The number of recorded young regions is the incremental</span></span><br><span class="line"> <span class="comment">// collection set's current size</span></span><br><span class="line"> set_recorded_rs_lengths(_inc_recorded_rs_lengths);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">double</span> young_end_time_sec = os::elapsedTime();</span><br><span class="line"> phase_times()-&gt;record_young_cset_choice_time_ms((young_end_time_sec - young_start_time_sec) * <span class="number">1000.0</span>);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">return</span> time_remaining_ms;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>下面是老年代的部分</p>
<figure class="highlight c++"><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">void</span> G1CollectionSet::finalize_old_part(<span class="keyword">double</span> time_remaining_ms) &#123;</span><br><span class="line"> <span class="keyword">double</span> non_young_start_time_sec = os::elapsedTime();</span><br><span class="line"> <span class="keyword">double</span> predicted_old_time_ms = <span class="number">0.0</span>;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (collector_state()-&gt;in_mixed_phase()) &#123;</span><br><span class="line"> cset_chooser()-&gt;verify();</span><br><span class="line"> <span class="keyword">const</span> uint min_old_cset_length = _policy-&gt;calc_min_old_cset_length();</span><br><span class="line"> <span class="keyword">const</span> uint max_old_cset_length = _policy-&gt;calc_max_old_cset_length();</span><br><span class="line"></span><br><span class="line"> uint expensive_region_num = <span class="number">0</span>;</span><br><span class="line"> <span class="keyword">bool</span> check_time_remaining = _policy-&gt;adaptive_young_list_length();</span><br><span class="line"></span><br><span class="line"> HeapRegion* hr = cset_chooser()-&gt;<span class="built_in">peek</span>();</span><br><span class="line"> <span class="keyword">while</span> (hr != <span class="literal">NULL</span>) &#123;</span><br><span class="line"> <span class="keyword">if</span> (old_region_length() &gt;= max_old_cset_length) &#123;</span><br><span class="line"> <span class="comment">// Added maximum number of old regions to the CSet.</span></span><br><span class="line"> log_debug(gc, ergo, cset)(<span class="string">"Finish adding old regions to CSet (old CSet region num reached max). old %u regions, max %u regions"</span>,</span><br><span class="line"> old_region_length(), max_old_cset_length);</span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// Stop adding regions if the remaining reclaimable space is</span></span><br><span class="line"> <span class="comment">// not above G1HeapWastePercent.</span></span><br><span class="line"> <span class="keyword">size_t</span> reclaimable_bytes = cset_chooser()-&gt;remaining_reclaimable_bytes();</span><br><span class="line"> <span class="keyword">double</span> reclaimable_percent = _policy-&gt;reclaimable_bytes_percent(reclaimable_bytes);</span><br><span class="line"> <span class="keyword">double</span> threshold = (<span class="keyword">double</span>) G1HeapWastePercent;</span><br><span class="line"> <span class="keyword">if</span> (reclaimable_percent &lt;= threshold) &#123;</span><br><span class="line"> <span class="comment">// We've added enough old regions that the amount of uncollected</span></span><br><span class="line"> <span class="comment">// reclaimable space is at or below the waste threshold. Stop</span></span><br><span class="line"> <span class="comment">// adding old regions to the CSet.</span></span><br><span class="line"> log_debug(gc, ergo, cset)(<span class="string">"Finish adding old regions to CSet (reclaimable percentage not over threshold). "</span></span><br><span class="line"> <span class="string">"old %u regions, max %u regions, reclaimable: "</span> SIZE_FORMAT <span class="string">"B (%1.2f%%) threshold: "</span> UINTX_FORMAT <span class="string">"%%"</span>,</span><br><span class="line"> old_region_length(), max_old_cset_length, reclaimable_bytes, reclaimable_percent, G1HeapWastePercent);</span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">double</span> predicted_time_ms = predict_region_elapsed_time_ms(hr);</span><br><span class="line"> <span class="keyword">if</span> (check_time_remaining) &#123;</span><br><span class="line"> <span class="keyword">if</span> (predicted_time_ms &gt; time_remaining_ms) &#123;</span><br><span class="line"> <span class="comment">// Too expensive for the current CSet.</span></span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (old_region_length() &gt;= min_old_cset_length) &#123;</span><br><span class="line"> <span class="comment">// We have added the minimum number of old regions to the CSet,</span></span><br><span class="line"> <span class="comment">// we are done with this CSet.</span></span><br><span class="line"> log_debug(gc, ergo, cset)(<span class="string">"Finish adding old regions to CSet (predicted time is too high). "</span></span><br><span class="line"> <span class="string">"predicted time: %1.2fms, remaining time: %1.2fms old %u regions, min %u regions"</span>,</span><br><span class="line"> predicted_time_ms, time_remaining_ms, old_region_length(), min_old_cset_length);</span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// We'll add it anyway given that we haven't reached the</span></span><br><span class="line"> <span class="comment">// minimum number of old regions.</span></span><br><span class="line"> expensive_region_num += <span class="number">1</span>;</span><br><span class="line"> &#125;</span><br><span class="line"> &#125; <span class="keyword">else</span> &#123;</span><br><span class="line"> <span class="keyword">if</span> (old_region_length() &gt;= min_old_cset_length) &#123;</span><br><span class="line"> <span class="comment">// In the non-auto-tuning case, we'll finish adding regions</span></span><br><span class="line"> <span class="comment">// to the CSet if we reach the minimum.</span></span><br><span class="line"></span><br><span class="line"> log_debug(gc, ergo, cset)(<span class="string">"Finish adding old regions to CSet (old CSet region num reached min). old %u regions, min %u regions"</span>,</span><br><span class="line"> old_region_length(), min_old_cset_length);</span><br><span class="line"> <span class="keyword">break</span>;</span><br><span class="line"> &#125;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// We will add this region to the CSet.</span></span><br><span class="line"> time_remaining_ms = MAX2(time_remaining_ms - predicted_time_ms, <span class="number">0.0</span>);</span><br><span class="line"> predicted_old_time_ms += predicted_time_ms;</span><br><span class="line"> cset_chooser()-&gt;pop(); <span class="comment">// already have region via peek()</span></span><br><span class="line"> _g1h-&gt;old_set_remove(hr);</span><br><span class="line"> add_old_region(hr);</span><br><span class="line"></span><br><span class="line"> hr = cset_chooser()-&gt;<span class="built_in">peek</span>();</span><br><span class="line"> &#125;</span><br><span class="line"> <span class="keyword">if</span> (hr == <span class="literal">NULL</span>) &#123;</span><br><span class="line"> log_debug(gc, ergo, cset)(<span class="string">"Finish adding old regions to CSet (candidate old regions not available)"</span>);</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="keyword">if</span> (expensive_region_num &gt; <span class="number">0</span>) &#123;</span><br><span class="line"> <span class="comment">// We print the information once here at the end, predicated on</span></span><br><span class="line"> <span class="comment">// whether we added any apparently expensive regions or not, to</span></span><br><span class="line"> <span class="comment">// avoid generating output per region.</span></span><br><span class="line"> log_debug(gc, ergo, cset)(<span class="string">"Added expensive regions to CSet (old CSet region num not reached min)."</span></span><br><span class="line"> <span class="string">"old: %u regions, expensive: %u regions, min: %u regions, remaining time: %1.2fms"</span>,</span><br><span class="line"> old_region_length(), expensive_region_num, min_old_cset_length, time_remaining_ms);</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> cset_chooser()-&gt;verify();</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> stop_incremental_building();</span><br><span class="line"></span><br><span class="line"> log_debug(gc, ergo, cset)(<span class="string">"Finish choosing CSet. old: %u regions, predicted old region time: %1.2fms, time remaining: %1.2f"</span>,</span><br><span class="line"> old_region_length(), predicted_old_time_ms, time_remaining_ms);</span><br><span class="line"></span><br><span class="line"> <span class="keyword">double</span> non_young_end_time_sec = os::elapsedTime();</span><br><span class="line"> phase_times()-&gt;record_non_young_cset_choice_time_ms((non_young_end_time_sec - non_young_start_time_sec) * <span class="number">1000.0</span>);</span><br><span class="line"></span><br><span class="line"> QuickSort::sort(_collection_set_regions, _collection_set_cur_length, compare_region_idx, <span class="literal">true</span>);</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>上面第三行是个判断,当前是否是 mixed 回收阶段,如果不是的话其实是没有老年代什么事的,所以可以看到代码基本是从这个 if 判断<br><code>if (collector_state()-&gt;in_mixed_phase()) {</code>开始往下走的<br>先写到这,偏向于做笔记用,有错轻拍</p>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2019/09/23/AbstractQueuedSynchronizer/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2019/09/23/AbstractQueuedSynchronizer/" class="post-title-link" itemprop="url">AbstractQueuedSynchronizer</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2019-09-23 23:40:05" itemprop="dateCreated datePublished" datetime="2019-09-23T23:40:05+08:00">2019-09-23</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2020-01-12 21:08:26" itemprop="dateModified" datetime="2020-01-12T21:08:26+08:00">2020-01-12</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</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">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2019/09/23/AbstractQueuedSynchronizer/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2019/09/23/AbstractQueuedSynchronizer/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<p>最近看了大神的 AQS 的文章,之前总是断断续续地看一点,每次都知难而退,下次看又从头开始,昨天总算硬着头皮看完了第一部分<br>首先 AQS 只要有这些属性</p>
<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><td class="code"><pre><span class="line"><span class="comment">// 头结点,你直接把它当做 当前持有锁的线程 可能是最好理解的</span></span><br><span class="line"><span class="keyword">private</span> <span class="keyword">transient</span> <span class="keyword">volatile</span> Node head;</span><br><span class="line"></span><br><span class="line"><span class="comment">// 阻塞的尾节点,每个新的节点进来,都插入到最后,也就形成了一个链表</span></span><br><span class="line"><span class="keyword">private</span> <span class="keyword">transient</span> <span class="keyword">volatile</span> Node tail;</span><br><span class="line"></span><br><span class="line"><span class="comment">// 这个是最重要的,代表当前锁的状态,0代表没有被占用,大于 0 代表有线程持有当前锁</span></span><br><span class="line"><span class="comment">// 这个值可以大于 1,是因为锁可以重入,每次重入都加上 1</span></span><br><span class="line"><span class="keyword">private</span> <span class="keyword">volatile</span> <span class="keyword">int</span> state;</span><br><span class="line"></span><br><span class="line"><span class="comment">// 代表当前持有独占锁的线程,举个最重要的使用例子,因为锁可以重入</span></span><br><span class="line"><span class="comment">// reentrantLock.lock()可以嵌套调用多次,所以每次用这个来判断当前线程是否已经拥有了锁</span></span><br><span class="line"><span class="comment">// if (currentThread == getExclusiveOwnerThread()) &#123;state++&#125;</span></span><br><span class="line"><span class="keyword">private</span> <span class="keyword">transient</span> Thread exclusiveOwnerThread; <span class="comment">//继承自AbstractOwnableSynchronizer</span></span><br></pre></td></tr></table></figure>
<p>大概了解了 aqs 底层的双向等待队列,<br>结构是这样的<br><img data-src="https://tva1.sinaimg.cn/large/006tNbRwly1g9mxu0ndt1j319o08w0t7.jpg" alt=""><br>每个 node 里面主要是的代码结构也比较简单</p>
<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><td class="code"><pre><span class="line"><span class="keyword">static</span> <span class="keyword">final</span> <span class="class"><span class="keyword">class</span> <span class="title">Node</span> </span>&#123;</span><br><span class="line"> <span class="comment">// 标识节点当前在共享模式下</span></span><br><span class="line"> <span class="keyword">static</span> <span class="keyword">final</span> Node SHARED = <span class="keyword">new</span> Node();</span><br><span class="line"> <span class="comment">// 标识节点当前在独占模式下</span></span><br><span class="line"> <span class="keyword">static</span> <span class="keyword">final</span> Node EXCLUSIVE = <span class="keyword">null</span>;</span><br><span class="line"></span><br><span class="line"> <span class="comment">// ======== 下面的几个int常量是给waitStatus用的 ===========</span></span><br><span class="line"> <span class="comment">/** waitStatus value to indicate thread has cancelled */</span></span><br><span class="line"> <span class="comment">// 代码此线程取消了争抢这个锁</span></span><br><span class="line"> <span class="keyword">static</span> <span class="keyword">final</span> <span class="keyword">int</span> CANCELLED = <span class="number">1</span>;</span><br><span class="line"> <span class="comment">/** waitStatus value to indicate successor's thread needs unparking */</span></span><br><span class="line"> <span class="comment">// 官方的描述是,其表示当前node的后继节点对应的线程需要被唤醒</span></span><br><span class="line"> <span class="keyword">static</span> <span class="keyword">final</span> <span class="keyword">int</span> SIGNAL = -<span class="number">1</span>;</span><br><span class="line"> <span class="comment">/** waitStatus value to indicate thread is waiting on condition */</span></span><br><span class="line"> <span class="comment">// 本文不分析condition,所以略过吧,下一篇文章会介绍这个</span></span><br><span class="line"> <span class="keyword">static</span> <span class="keyword">final</span> <span class="keyword">int</span> CONDITION = -<span class="number">2</span>;</span><br><span class="line"> <span class="comment">/**</span></span><br><span class="line"><span class="comment"> * waitStatus value to indicate the next acquireShared should</span></span><br><span class="line"><span class="comment"> * unconditionally propagate</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"> <span class="comment">// 同样的不分析,略过吧</span></span><br><span class="line"> <span class="keyword">static</span> <span class="keyword">final</span> <span class="keyword">int</span> PROPAGATE = -<span class="number">3</span>;</span><br><span class="line"> <span class="comment">// =====================================================</span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"> <span class="comment">// 取值为上面的1、-1、-2、-3,或者0(以后会讲到)</span></span><br><span class="line"> <span class="comment">// 这么理解,暂时只需要知道如果这个值 大于0 代表此线程取消了等待,</span></span><br><span class="line"> <span class="comment">// ps: 半天抢不到锁,不抢了,ReentrantLock是可以指定timeouot的。。。</span></span><br><span class="line"> <span class="keyword">volatile</span> <span class="keyword">int</span> waitStatus;</span><br><span class="line"> <span class="comment">// 前驱节点的引用</span></span><br><span class="line"> <span class="keyword">volatile</span> Node prev;</span><br><span class="line"> <span class="comment">// 后继节点的引用</span></span><br><span class="line"> <span class="keyword">volatile</span> Node next;</span><br><span class="line"> <span class="comment">// 这个就是线程本尊</span></span><br><span class="line"> <span class="keyword">volatile</span> Thread thread;</span><br><span class="line"></span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>其实可以主要关注这个 <code>waitStatus</code> 因为这个是后面的节点给前面的节点设置的,等于-1 的时候代表后面有节点等待,需要去唤醒,<br>这里使用了一个变种的 CLH 队列实现,CLH 队列相关内容可以查看这篇 <a href="https://coderbee.net/index.php/concurrent/20131115/577" target="_blank" rel="noopener">自旋锁、排队自旋锁、MCS锁、CLH锁</a></p>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2019/06/18/openresty/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2019/06/18/openresty/" class="post-title-link" itemprop="url">openresty</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2019-06-18 19:03:05" itemprop="dateCreated datePublished" datetime="2019-06-18T19:03:05+08:00">2019-06-18</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2020-01-12 21:08:27" itemprop="dateModified" datetime="2020-01-12T21:08:27+08:00">2020-01-12</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</span>
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/nginx/" itemprop="url" rel="index">
<span itemprop="name">nginx</span>
</a>
</span>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2019/06/18/openresty/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2019/06/18/openresty/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<p>目前公司要对一些新的产品功能做灰度测试,因为在后端业务代码层面添加判断比较麻烦,所以想在nginx上做点手脚,就想到了openresty<br>前后也踩了不少坑,这边先写一点</p>
<p>首先是日志<br><code>error_log logs/error.log debug;</code><br>需要nginx开启日志的debug才能看到日志</p>
<p>使用 <code>lua_code_cache off</code>即可, 另外注意只有使用 <code>content_by_lua_file</code> 才会生效</p>
<figure class="highlight lua"><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><td class="code"><pre><span class="line">http &#123;</span><br><span class="line"> lua_code_cache off;</span><br><span class="line">&#125;</span><br><span class="line"></span><br><span class="line">location ~* /(\d+-.*)/api/orgunits/load_all(.*) &#123;</span><br><span class="line"> default_type <span class="string">'application/json;charset=utf-8'</span>;</span><br><span class="line"> content_by_lua_file /data/projects/xxx/current/lua/controller/load_data.lua;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>使用lua给nginx请求response头添加内容可以用这个</p>
<figure class="highlight lua"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">ngx.header[<span class="string">'response'</span>] = <span class="string">'header'</span></span><br></pre></td></tr></table></figure>
<p><a href="http://cyukang.com/2017/05/22/try-on-openresty.html" target="_blank" rel="noopener">使用总结</a></p>
<p>后续:</p>
<ol>
<li>一开始在本地环境的时候使用content_by_lua_file只关注了头,后来发到测试环境发现请求内容都没代理转发到后端服务上<br>网上查了下发现content_by_lua_file是将请求的所有内容包括response都用这里面的lua脚本生成了,content这个词就表示是请求内容<br>后来改成了access_by_lua_file就正常了,只是要去获取请求内容和修改响应头,并不是要完整的接管请求</li>
</ol>
<ol start="2">
<li><p>后来又碰到了一个坑是nginx有个client_body_buffer_size的配置参数,nginx在32位和64位系统里有8K和16K两个默认值,当请求内容大于这两个值的时候,会把请求内容放到临时文件里,这个时候openresty里的ngx.req.get_post_args()就会报“failed to get post args: requesty body in temp file not supported”这个错误,将client_body_buffer_size这个参数配置调大一点就好了</p>
</li>
<li><p>还有就是lua的异常捕获,网上看一般是用pcall和xpcall来进行保护调用,因为问题主要出在cjson的decode,这里有两个解决方案,一个就是将cjson.decode使用pcall封装,</p>
<figure class="highlight lua"><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">local</span> decode = <span class="built_in">require</span>(<span class="string">"cjson"</span>).decode</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">function</span> <span class="title">json_decode</span><span class="params">( str )</span></span></span><br><span class="line"> <span class="keyword">local</span> ok, t = <span class="built_in">pcall</span>(decode, str)</span><br><span class="line"> <span class="keyword">if</span> <span class="keyword">not</span> ok <span class="keyword">then</span></span><br><span class="line"> <span class="keyword">return</span> <span class="literal">nil</span></span><br><span class="line"> <span class="keyword">end</span></span><br><span class="line"></span><br><span class="line"> <span class="keyword">return</span> t</span><br><span class="line"><span class="keyword">end</span></span><br></pre></td></tr></table></figure>
<p> 这个是使用了pcall,称为保护调用,会在内部错误后返回两个参数,第一个是false,第二个是错误信息<br> 还有一种是使用cjson.safe包</p>
<figure class="highlight lua"><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><td class="code"><pre><span class="line"><span class="keyword">local</span> json = <span class="built_in">require</span>(<span class="string">"cjson.safe"</span>)</span><br><span class="line"><span class="keyword">local</span> str = <span class="string">[[ &#123;"key:"value"&#125; ]]</span></span><br><span class="line"></span><br><span class="line"><span class="keyword">local</span> t = json.decode(str)</span><br><span class="line"><span class="keyword">if</span> t <span class="keyword">then</span></span><br><span class="line"> ngx.say(<span class="string">" --&gt; "</span>, <span class="built_in">type</span>(t))</span><br><span class="line"><span class="keyword">end</span></span><br></pre></td></tr></table></figure>
<p> cjson.safe包会在解析失败的时候返回nil</p>
</li>
<li><p>还有一个是redis链接时如果host使用的是域名的话会提示“failed to connect: no resolver defined to resolve “redis.xxxxxx.com””,这里需要使用nginx的resolver指令,<br><code>resolver 8.8.8.8 valid=3600s;</code></p>
</li>
<li><p>还有一点补充下<br>就是业务在使用redis的时候使用了db的特性,所以在lua访问redis的时候也需要执行db,这里lua的redis库也支持了这个特性,可以使用instance:select(config:get(‘db’))来切换db</p>
</li>
<li><p>性能优化tips<br><a href="https://juejin.im/entry/5b0e8fcef265da09210072a3#openresty" target="_blank" rel="noopener">建议是尽量少使用阶段钩子,例如content_by_lua_file,*_by_lua</a></p>
</li>
<li><p>发现一个不错的openresty站点<br><a href="http://orhub.org/" target="_blank" rel="noopener">地址</a></p>
</li>
</ol>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2017/05/09/ambari-summary/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2017/05/09/ambari-summary/" class="post-title-link" itemprop="url">ambari-summary</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2017-05-09 23:53:05" itemprop="dateCreated datePublished" datetime="2017-05-09T23:53:05+08:00">2017-05-09</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2020-01-12 21:08:27" itemprop="dateModified" datetime="2020-01-12T21:08:27+08:00">2020-01-12</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</span>
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/data-analysis/" itemprop="url" rel="index">
<span itemprop="name">data analysis</span>
</a>
</span>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2017/05/09/ambari-summary/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2017/05/09/ambari-summary/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<h2 id="初识ambari"><a href="#初识ambari" class="headerlink" title="初识ambari"></a>初识ambari</h2><p><a href="http://ambari.apache.org/" target="_blank" rel="noopener">ambari</a>是一个大数据平台的管理工具,包含了<code>hadoop</code>, <code>yarn</code>, <code>hive</code>, <code>hbase</code>, <code>spark</code>等大数据的基础架构和工具,简化了数据平台的搭建,之前只是在同事搭建好平台后的一些使用,这次有机会从头开始用<code>ambari</code>来搭建一个测试的数据平台,过程中也踩到不少坑,简单记录下。</p>
<h2 id="简单过程"><a href="#简单过程" class="headerlink" title="简单过程"></a>简单过程</h2><ul>
<li>第一个坑<br>在刚开始是按照官网的指南,用maven构建,因为GFW的原因,导致反复失败等待,也就是这个<a href="https://cwiki.apache.org/confluence/display/AMBARI/Installation+Guide+for+Ambari+2.5.0" target="_blank" rel="noopener">guide</a>,因为对maven不熟悉导致有些按图索骥,浪费了很多时间,之后才知道可以直接加repo用yum安装,然而用yum安装马上就出现了第二个坑。</li>
<li>第二个坑<br>因为在线的repo还是因为网络原因很慢很慢,用proxychains勉强把ambari-server本身安装好了,<a href="http://public-repo-1.hortonworks.com/ambari/centos7/2.x/updates/2.5.0.3/ambari.repo" target="_blank" rel="noopener">ambari.repo</a>将这个放进<code>/etc/yum.repos.d/</code>路径下,然后<code>yum update &amp;&amp; yum install ambari-server</code>安装即可,如果有条件就用proxychains走下代理。</li>
<li>第三步<br>安装好ambari-server后先执行<code>ambari-server setup</code>做一些初始化设置,其中包含了JDK路径的设置,数据库设置,设置好就OK了,然后执行<code>ambari-server start</code>启动服务,这里有个小插曲,因为<code>ambari-server</code>涉及到这么多服务,所以管理控制监控之类的模块是必不可少的,这部分可以在<code>ambari-server</code>的web ui界面安装,也可以命令行提前安装,这部分被称为<code>HDF Management Pack</code>,运行```ambari-server install-mpack \</li>
<li>-mpack=<a href="http://public-repo-1.hortonworks.com/HDF/centos7/2.x/updates/2.1.4.0/tars/hdf_ambari_mp/hdf-ambari-mpack-2.1.4.0-5.tar.gz" target="_blank" rel="noopener">http://public-repo-1.hortonworks.com/HDF/centos7/2.x/updates/2.1.4.0/tars/hdf_ambari_mp/hdf-ambari-mpack-2.1.4.0-5.tar.gz</a> \</li>
<li>-purge \</li>
<li>-verbose```<br>安装,当然这个压缩包可以下载之后指到本地路径安装,然后就可以重启<code>ambari-server</code></li>
</ul>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2017/04/25/rabbitmq-tips/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2017/04/25/rabbitmq-tips/" class="post-title-link" itemprop="url">rabbitmq-tips</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2017-04-25 21:46:55" itemprop="dateCreated datePublished" datetime="2017-04-25T21:46:55+08:00">2017-04-25</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2020-01-12 21:08:26" itemprop="dateModified" datetime="2020-01-12T21:08:26+08:00">2020-01-12</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</span>
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/php/" itemprop="url" rel="index">
<span itemprop="name">php</span>
</a>
</span>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2017/04/25/rabbitmq-tips/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2017/04/25/rabbitmq-tips/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<h2 id="rabbitmq-介绍"><a href="#rabbitmq-介绍" class="headerlink" title="rabbitmq 介绍"></a>rabbitmq 介绍</h2><p>接触了一下rabbitmq,原来在选型的时候是在rabbitmq跟kafka之间做选择,网上搜了一下之后发现kafka的优势在于吞吐量,而rabbitmq相对注重可靠性,因为应用在im上,需要保证消息不能丢失所以就暂时选定rabbitmq,<br>Message Queue的需求由来已久,80年代最早在金融交易中,高盛等公司采用Teknekron公司的产品,当时的Message queuing软件叫做:the information bus(TIB)。 TIB被电信和通讯公司采用,路透社收购了Teknekron公司。之后,IBM开发了MQSeries,微软开发了Microsoft Message Queue(MSMQ)。这些商业MQ供应商的问题是厂商锁定,价格高昂。2001年,Java Message queuing试图解决锁定和交互性的问题,但对应用来说反而更加麻烦了。<br>RabbitMQ采用Erlang语言开发。Erlang语言由Ericson设计,专门为开发concurrent和distribution系统的一种语言,在电信领域使用广泛。OTP(Open Telecom Platform)作为Erlang语言的一部分,包含了很多基于Erlang开发的中间件/库/工具,如mnesia/SASL,极大方便了Erlang应用的开发。OTP就类似于Python语言中众多的module,用户借助这些module可以很方便的开发应用。<br>于是2004年,摩根大通和iMatrix开始着手<a href="https://en.wikipedia.org/wiki/Advanced_Message_Queuing_Protocol" target="_blank" rel="noopener">Advanced Message Queuing Protocol (AMQP)</a>开放标准的开发。2006年,AMQP规范发布。2007年,Rabbit技术公司基于AMQP标准开发的RabbitMQ 1.0 发布。所有主要的编程语言均有与代理接口通讯的客户端库。</p>
<h2 id="简单的使用经验"><a href="#简单的使用经验" class="headerlink" title="简单的使用经验"></a>简单的使用经验</h2><h3 id="通俗的理解"><a href="#通俗的理解" class="headerlink" title="通俗的理解"></a>通俗的理解</h3><p>这里介绍下其中的一些概念,connection表示和队列服务器的连接,一般情况下是tcp连接, channel表示通道,可以在一个连接上建立多个通道,这里主要是节省了tcp连接握手的成本,exchange可以理解成一个路由器,将消息推送给对应的队列queue,其实是像一个订阅的模式。</p>
<h3 id="集群经验"><a href="#集群经验" class="headerlink" title="集群经验"></a>集群经验</h3><p><code>rabbitmqctl stop</code>这个是关闭rabbitmq,在搭建集群时候先关闭服务,然后使用<code>rabbitmq-server -detached</code>静默启动,这时候使用<code>rabbitmqctl cluster_status</code>查看集群状态,因为还没将节点加入集群,所以只能看到类似</p>
<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></pre></td><td class="code"><pre><span class="line">Cluster status of node rabbit@rabbit1 ...</span><br><span class="line">[&#123;nodes,[&#123;disc,[rabbit@rabbit1,rabbit@rabbit2,rabbit@rabbit3]&#125;]&#125;,</span><br><span class="line"> &#123;running_nodes,[rabbit@rabbit2,rabbit@rabbit1]&#125;]</span><br><span class="line">...done.</span><br></pre></td></tr></table></figure>
<p>然后就可以把当前节点加入集群,</p>
<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></pre></td><td class="code"><pre><span class="line"><span class="meta">rabbit2$</span><span class="bash"> rabbitmqctl stop_app <span class="comment">#这个stop_app与stop的区别是前者停的是rabbitmq应用,保留erlang节点,</span></span></span><br><span class="line"> #后者是停止了rabbitmq和erlang节点</span><br><span class="line">Stopping node rabbit@rabbit2 ...done.</span><br><span class="line"><span class="meta">rabbit2$</span><span class="bash"> rabbitmqctl join_cluster rabbit@rabbit1 <span class="comment">#这里可以用--ram指定将当前节点作为内存节点加入集群</span></span></span><br><span class="line">Clustering node rabbit@rabbit2 with [rabbit@rabbit1] ...done.</span><br><span class="line"><span class="meta">rabbit2$</span><span class="bash"> rabbitmqctl start_app</span></span><br><span class="line">Starting node rabbit@rabbit2 ...done.</span><br></pre></td></tr></table></figure>
<p>其他可以参考<a href="http://www.rabbitmq.com/clustering.html" target="_blank" rel="noopener">官方文档</a></p>
<h2 id="一些坑"><a href="#一些坑" class="headerlink" title="一些坑"></a>一些坑</h2><h3 id="消息丢失"><a href="#消息丢失" class="headerlink" title="消息丢失"></a>消息丢失</h3><p>这里碰到过一个坑,对于使用exchange来做消息路由的,会有一个情况,就是在routing_key没被订阅的时候,会将该条找不到路由对应的queue的消息丢掉<code>What happens if we break our contract and send a message with one or four words, like &quot;orange&quot; or &quot;quick.orange.male.rabbit&quot;? Well, these messages won&#39;t match any bindings and will be lost.</code><a href="http://www.rabbitmq.com/tutorials/tutorial-five-python.html" target="_blank" rel="noopener">对应链接</a>,而当使用空的exchange时,会保留消息,当出现消费者的时候就可以将收到之前生产者所推送的消息<a href="http://www.rabbitmq.com/tutorials/tutorial-two-python.html" target="_blank" rel="noopener">对应链接</a>,这里就是用了空的exchange。</p>
<h3 id="集群搭建"><a href="#集群搭建" class="headerlink" title="集群搭建"></a>集群搭建</h3><p>集群搭建的时候有个erlang vm生成的random cookie,这个是用来做集群之间认证的,相同的cookie才能连接,但是如果通过vim打开复制后在其他几点新建文件写入会多一个换行,导致集群建立是报错,所以这里最好使用scp等传输命令直接传输cookie文件,同时要注意下cookie的文件权限。<br>另外在集群搭建的时候如果更改过hostname,那么要把rabbitmq的数据库删除,否则启动后会马上挂掉</p>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2017/03/28/spark-little-tips/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2017/03/28/spark-little-tips/" class="post-title-link" itemprop="url">spark-little-tips</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2017-03-28 19:20:56" itemprop="dateCreated datePublished" datetime="2017-03-28T19:20:56+08:00">2017-03-28</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2020-01-12 21:08:26" itemprop="dateModified" datetime="2020-01-12T21:08:26+08:00">2020-01-12</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</span>
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/data-analysis/" itemprop="url" rel="index">
<span itemprop="name">data analysis</span>
</a>
</span>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2017/03/28/spark-little-tips/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2017/03/28/spark-little-tips/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<h2 id="spark-的一些粗浅使用经验"><a href="#spark-的一些粗浅使用经验" class="headerlink" title="spark 的一些粗浅使用经验"></a>spark 的一些粗浅使用经验</h2><p>工作中学习使用了一下Spark做数据分析,主要是用spark的python接口,首先是<code>pyspark.SparkContext(appName=xxx)</code>,这是初始化一个Spark应用实例或者说会话,不能重复,<br>返回的实例句柄就可以调用<code>textFile(path)</code>读取文本文件,这里的文本文件可以是HDFS上的文本文件,也可以普通文本文件,但是需要在Spark的所有集群上都存在,否则会<br>读取失败,<code>parallelize</code>则可以将python生成的集合数据读取后转换成rdd(A Resilient Distributed Dataset (RDD),一种spark下的基本抽象数据集),基于这个RDD就可以做<br>数据的流式计算,例如<code>map reduce</code>,在Spark中可以非常方便地实现 </p>
<h3 id="简单的mapreduce-word-count示例"><a href="#简单的mapreduce-word-count示例" class="headerlink" title="简单的mapreduce word count示例"></a>简单的mapreduce word count示例</h3><figure class="highlight python"><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><td class="code"><pre><span class="line">textFile = sc.parallelize([(<span class="number">1</span>,<span class="number">1</span>), (<span class="number">2</span>,<span class="number">1</span>), (<span class="number">3</span>,<span class="number">1</span>), (<span class="number">4</span>,<span class="number">1</span>), (<span class="number">5</span>,<span class="number">1</span>),(<span class="number">1</span>,<span class="number">1</span>), (<span class="number">2</span>,<span class="number">1</span>), (<span class="number">3</span>,<span class="number">1</span>), (<span class="number">4</span>,<span class="number">1</span>), (<span class="number">5</span>,<span class="number">1</span>)])</span><br><span class="line">data = textFile.reduceByKey(<span class="keyword">lambda</span> x, y: x + y).collect()</span><br><span class="line"><span class="keyword">for</span> _ <span class="keyword">in</span> data:</span><br><span class="line"> print(_)</span><br></pre></td></tr></table></figure>
<h3 id="结果"><a href="#结果" class="headerlink" title="结果"></a>结果</h3><figure class="highlight plain"><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><td class="code"><pre><span class="line">(3, 2)</span><br><span class="line">(1, 2)</span><br><span class="line">(4, 2)</span><br><span class="line">(2, 2)</span><br><span class="line">(5, 2)</span><br></pre></td></tr></table></figure>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2016/11/10/php-abstract-class-and-interface/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2016/11/10/php-abstract-class-and-interface/" class="post-title-link" itemprop="url">php-abstract-class-and-interface</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2016-11-10 10:42:35" itemprop="dateCreated datePublished" datetime="2016-11-10T10:42:35+08:00">2016-11-10</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2020-01-12 21:08:27" itemprop="dateModified" datetime="2020-01-12T21:08:27+08:00">2020-01-12</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</span>
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/php/" itemprop="url" rel="index">
<span itemprop="name">php</span>
</a>
</span>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2016/11/10/php-abstract-class-and-interface/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2016/11/10/php-abstract-class-and-interface/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<h2 id="PHP抽象类和接口"><a href="#PHP抽象类和接口" class="headerlink" title="PHP抽象类和接口"></a>PHP抽象类和接口</h2><ul>
<li>抽象类与接口</li>
<li>抽象类内可以包含非抽象函数,即可实现函数</li>
<li>抽象类内必须包含至少一个抽象方法,抽象类和接口均不能实例化</li>
<li>抽象类可以设置访问级别,接口默认都是public</li>
<li>类可以实现多个接口但不能继承多个抽象类</li>
<li>类必须实现抽象类和接口里的抽象方法,不一定要实现抽象类的非抽象方法</li>
<li>接口内不能定义变量,但是可以定义常量</li>
</ul>
<h2 id="示例代码"><a href="#示例代码" class="headerlink" title="示例代码"></a>示例代码</h2><figure class="highlight plain"><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><td class="code"><pre><span class="line">&lt;?php</span><br><span class="line">interface int1&#123;</span><br><span class="line"> const INTER1 &#x3D; 111;</span><br><span class="line"> function inter1();</span><br><span class="line">&#125;</span><br><span class="line">interface int2&#123;</span><br><span class="line"> const INTER1 &#x3D; 222;</span><br><span class="line"> function inter2();</span><br><span class="line">&#125;</span><br><span class="line">abstract class abst1&#123;</span><br><span class="line"> public function abstr1()&#123;</span><br><span class="line"> echo 1111;</span><br><span class="line"> &#125;</span><br><span class="line"> abstract function abstra1()&#123;</span><br><span class="line"> echo &#39;ahahahha&#39;;</span><br><span class="line"> &#125;</span><br><span class="line">&#125;</span><br><span class="line">abstract class abst2&#123;</span><br><span class="line"> public function abstr2()&#123;</span><br><span class="line"> echo 1111;</span><br><span class="line"> &#125;</span><br><span class="line"> abstract function abstra2();</span><br><span class="line">&#125;</span><br><span class="line">class normal1 extends abst1&#123;</span><br><span class="line"> protected function abstr2()&#123;</span><br><span class="line"> echo 222;</span><br><span class="line"> &#125;</span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<h2 id="result"><a href="#result" class="headerlink" title="result"></a>result</h2><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br></pre></td><td class="code"><pre><span class="line">PHP Fatal error: Abstract function abst1::abstra1() cannot contain body in new.php on line 17</span><br><span class="line"></span><br><span class="line">Fatal error: Abstract function abst1::abstra1() cannot contain body in php on line 17</span><br></pre></td></tr></table></figure>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
<article itemscope itemtype="http://schema.org/Article" class="post-block home" lang="zh-Hans">
<link itemprop="mainEntityOfPage" href="https://nicksxs.me/2016/10/12/summary-ranges-228/">
<span hidden itemprop="author" itemscope itemtype="http://schema.org/Person">
<meta itemprop="image" content="/uploads/avatar.jpg">
<meta itemprop="name" content="Nicksxs">
<meta itemprop="description" content="learn from zero,技术博客,Nicksxs,史学森">
</span>
<span hidden itemprop="publisher" itemscope itemtype="http://schema.org/Organization">
<meta itemprop="name" content="Nicksxs's Blog">
</span>
<header class="post-header">
<h1 class="post-title" itemprop="name headline">
<a href="/2016/10/12/summary-ranges-228/" class="post-title-link" itemprop="url">summary-ranges-228</a>
</h1>
<div class="post-meta">
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-o"></i>
</span>
<span class="post-meta-item-text">Posted on</span>
<time title="Created: 2016-10-12 23:25:17" itemprop="dateCreated datePublished" datetime="2016-10-12T23:25:17+08:00">2016-10-12</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-calendar-check-o"></i>
</span>
<span class="post-meta-item-text">Edited on</span>
<time title="Modified: 2020-01-12 21:08:26" itemprop="dateModified" datetime="2020-01-12T21:08:26+08:00">2020-01-12</time>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-folder-o"></i>
</span>
<span class="post-meta-item-text">In</span>
<span itemprop="about" itemscope itemtype="http://schema.org/Thing">
<a href="/categories/leetcode/" itemprop="url" rel="index">
<span itemprop="name">leetcode</span>
</a>
</span>
</span>
<span class="post-meta-item">
<span class="post-meta-item-icon">
<i class="fa fa-comment-o"></i>
</span>
<span class="post-meta-item-text">Disqus: </span>
<a title="disqus" href="/2016/10/12/summary-ranges-228/#comments" itemprop="discussionUrl">
<span class="post-comments-count disqus-comment-count" data-disqus-identifier="2016/10/12/summary-ranges-228/" itemprop="commentCount"></span>
</a>
</span>
</div>
</header>
<div class="post-body" itemprop="articleBody">
<h4 id="problem"><a href="#problem" class="headerlink" title="problem"></a>problem</h4><p>Given a sorted integer array without duplicates, return the summary of its ranges.</p>
<p>For example, given <code>[0,1,2,4,5,7]</code>, return <code>[&quot;0-&gt;2&quot;,&quot;4-&gt;5&quot;,&quot;7&quot;]</code>.</p>
<h4 id="题解"><a href="#题解" class="headerlink" title="题解"></a>题解</h4><p>每一个区间的起点<code>nums[i]</code>加上<code>j</code>是否等于<code>nums[i+j]</code><br><a href="http://www.cnblogs.com/grandyang/p/4603555.html" target="_blank" rel="noopener">参考</a></p>
<h4 id="Code"><a href="#Code" class="headerlink" title="Code"></a>Code</h4><figure class="highlight c++"><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><td class="code"><pre><span class="line"><span class="class"><span class="keyword">class</span> <span class="title">Solution</span> &#123;</span></span><br><span class="line"><span class="keyword">public</span>:</span><br><span class="line"> <span class="built_in">vector</span>&lt;<span class="built_in">string</span>&gt; summaryRanges(<span class="built_in">vector</span>&lt;<span class="keyword">int</span>&gt;&amp; nums) &#123;</span><br><span class="line"> <span class="keyword">int</span> i = <span class="number">0</span>, j = <span class="number">1</span>, n;</span><br><span class="line"> <span class="built_in">vector</span>&lt;<span class="built_in">string</span>&gt; res;</span><br><span class="line"> n = nums.<span class="built_in">size</span>();</span><br><span class="line"> <span class="keyword">while</span>(i &lt; n)&#123;</span><br><span class="line"> j = <span class="number">1</span>;</span><br><span class="line"> <span class="keyword">while</span>(j &lt; n &amp;&amp; nums[i+j] - nums[i] == j) j++;</span><br><span class="line"> res.push_back(j &lt;= <span class="number">1</span> ? to_string(nums[i]) : to_string(nums[i]) + <span class="string">"-&gt;"</span> + to_string(nums[i + j - <span class="number">1</span>]));</span><br><span class="line"> i += j;</span><br><span class="line"> &#125;</span><br><span class="line"> <span class="keyword">return</span> res;</span><br><span class="line"> &#125;</span><br><span class="line">&#125;;</span><br></pre></td></tr></table></figure>
</div>
<footer class="post-footer">
<div class="post-eof"></div>
</footer>
</article>
</div>
<nav class="pagination">
<a class="extend prev" rel="prev" href="/"><i class="fa fa-angle-left" aria-label="Previous page"></i></a><a class="page-number" href="/">1</a><span class="page-number current">2</span><a class="page-number" href="/page/3/">3</a><a class="page-number" href="/page/4/">4</a><a class="extend next" rel="next" href="/page/3/"><i class="fa fa-angle-right" aria-label="Next page"></i></a>
</nav>
</div>
<script>
window.addEventListener('tabs:register', () => {
let activeClass = CONFIG.comments.activeClass;
if (CONFIG.comments.storage) {
activeClass = localStorage.getItem('comments_active') || activeClass;
}
if (activeClass) {
let activeTab = document.querySelector(`a[href="#comment-${activeClass}"]`);
if (activeTab) {
activeTab.click();
}
}
});
if (CONFIG.comments.storage) {
window.addEventListener('tabs:click', event => {
if (!event.target.matches('.tabs-comment .tab-content .tab-pane')) return;
let commentClass = event.target.classList[1];
localStorage.setItem('comments_active', commentClass);
});
}
</script>
</div>
<div class="toggle sidebar-toggle">
<span class="toggle-line toggle-line-first"></span>
<span class="toggle-line toggle-line-middle"></span>
<span class="toggle-line toggle-line-last"></span>
</div>
<aside class="sidebar">
<div class="sidebar-inner">
<ul class="sidebar-nav motion-element">
<li class="sidebar-nav-toc">
Table of Contents
</li>
<li class="sidebar-nav-overview">
Overview
</li>
</ul>
<!--noindex-->
<div class="post-toc-wrap sidebar-panel">
</div>
<!--/noindex-->
<div class="site-overview-wrap sidebar-panel">
<div class="site-author motion-element" itemprop="author" itemscope itemtype="http://schema.org/Person">
<img class="site-author-image" itemprop="image" alt="Nicksxs"
src="/uploads/avatar.jpg">
<p class="site-author-name" itemprop="name">Nicksxs</p>
<div class="site-description" itemprop="description">learn from zero,技术博客,Nicksxs,史学森</div>
</div>
<div class="site-state-wrap motion-element">
<nav class="site-state">
<div class="site-state-item site-state-posts">
<a href="/archives/">
<span class="site-state-item-count">37</span>
<span class="site-state-item-name">posts</span>
</a>
</div>
<div class="site-state-item site-state-categories">
<a href="/categories/">
<span class="site-state-item-count">24</span>
<span class="site-state-item-name">categories</span></a>
</div>
<div class="site-state-item site-state-tags">
<a href="/tags/">
<span class="site-state-item-count">41</span>
<span class="site-state-item-name">tags</span></a>
</div>
</nav>
</div>
<div class="links-of-author motion-element">
<span class="links-of-author-item">
<a href="https://github.com/nicksxs" title="GitHub → https:&#x2F;&#x2F;github.com&#x2F;nicksxs" rel="noopener" target="_blank"><i class="fa fa-fw fa-github"></i>GitHub</a>
</span>
<span class="links-of-author-item">
<a href="mailto:nicksxs1202@gmail.com" title="E-Mail → mailto:nicksxs1202@gmail.com" rel="noopener" target="_blank"><i class="fa fa-fw fa-envelope"></i>E-Mail</a>
</span>
<span class="links-of-author-item">
<a href="/atom.xml" title="RSS → &#x2F;atom.xml"><i class="fa fa-fw fa-rss"></i>RSS</a>
</span>
</div>
<div class="cc-license motion-element" itemprop="license">
<a href="https://creativecommons.org/licenses/by-nc-sa/4.0/" class="cc-opacity" rel="noopener" target="_blank"><img src="/images/cc-by-nc-sa.svg" alt="Creative Commons"></a>
</div>
<script type="text/javascript" charset="utf-8" src="/js/tagcloud.js"></script>
<script type="text/javascript" charset="utf-8" src="/js/tagcanvas.js"></script>
<div class="widget-wrap">
<div id="myCanvasContainer" class="widget tagcloud">
<canvas width="250" height="250" id="resCanvas" style="width=100%">
<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><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><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><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><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><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><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><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><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><li class="tag-list-item"><a class="tag-list-link" href="/tags/Java/" rel="tag">Java</a><span class="tag-list-count">3</span></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/Maven/" rel="tag">Maven</a><span class="tag-list-count">1</span></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/Redis/" rel="tag">Redis</a><span class="tag-list-count">1</span></li><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><li class="tag-list-item"><a class="tag-list-link" href="/tags/aqs/" rel="tag">aqs</a><span class="tag-list-count">1</span></li><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><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><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><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><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><li class="tag-list-item"><a class="tag-list-link" href="/tags/java/" rel="tag">java</a><span class="tag-list-count">1</span></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/leetcode/" rel="tag">leetcode</a><span class="tag-list-count">13</span></li><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><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><li class="tag-list-item"><a class="tag-list-link" href="/tags/mysql/" rel="tag">mysql</a><span class="tag-list-count">1</span></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/nginx/" rel="tag">nginx</a><span class="tag-list-count">1</span></li><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><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><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><li class="tag-list-item"><a class="tag-list-link" href="/tags/redis/" rel="tag">redis</a><span class="tag-list-count">6</span></li><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><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><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><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><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><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><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">1</span></li><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">6</span></li><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">6</span></li><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">1</span></li><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><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">1</span></li></ul>
</canvas>
</div>
</div>
<div class="links-of-blogroll motion-element">
<div class="links-of-blogroll-title">
<i class="fa fa-fw fa-link"></i>
Links
</div>
<ul class="links-of-blogroll-list">
<li class="links-of-blogroll-item">
<a href="http://www.zipperary.com/" title="http:&#x2F;&#x2F;www.zipperary.com&#x2F;" rel="noopener" target="_blank">Zippera's blog</a>
</li>
<li class="links-of-blogroll-item">
<a href="http://x20a.xyz/" title="http:&#x2F;&#x2F;x20a.xyz&#x2F;" rel="noopener" target="_blank">Freedom</a>
</li>
<li class="links-of-blogroll-item">
<a href="http://samuel.group/" title="http:&#x2F;&#x2F;samuel.group&#x2F;" rel="noopener" target="_blank">sfwtt</a>
</li>
<li class="links-of-blogroll-item">
<a href="http://blog.leanote.com/buru" title="http:&#x2F;&#x2F;blog.leanote.com&#x2F;buru" rel="noopener" target="_blank">bruce</a>
</li>
<li class="links-of-blogroll-item">
<a href="http://m2shad0w.com/" title="http:&#x2F;&#x2F;m2shad0w.com&#x2F;" rel="noopener" target="_blank">m2shad0w</a>
</li>
<li class="links-of-blogroll-item">
<a href="http://www.programcat.com/" title="http:&#x2F;&#x2F;www.programcat.com" rel="noopener" target="_blank">程序喵的厨房</a>
</li>
</ul>
</div>
</div>
</div>
</aside>
<div id="sidebar-dimmer"></div>
</div>
</main>
<footer class="footer">
<div class="footer-inner">
<div class="copyright">
&copy;
<span itemprop="copyrightYear">2020</span>
<span class="with-love">
<i class="fa fa-user"></i>
</span>
<span class="author" itemprop="copyrightHolder">Nicksxs</span>
</div>
</div>
</footer>
</div>
<script src="/lib/anime.min.js"></script>
<script src="//cdn.jsdelivr.net/npm/jquery@3/dist/jquery.min.js"></script>
<script src="//cdn.jsdelivr.net/gh/fancyapps/fancybox@3/dist/jquery.fancybox.min.js"></script>
<script src="//cdn.jsdelivr.net/npm/lozad@1/dist/lozad.min.js"></script>
<script src="/lib/velocity/velocity.min.js"></script>
<script src="/lib/velocity/velocity.ui.min.js"></script>
<script src="/js/utils.js"></script>
<script src="/js/motion.js"></script>
<script src="/js/schemes/pisces.js"></script>
<script src="/js/next-boot.js"></script>
<script>
(function(){
var bp = document.createElement('script');
var curProtocol = window.location.protocol.split(':')[0];
bp.src = (curProtocol === 'https') ? 'https://zz.bdstatic.com/linksubmit/push.js' : 'http://push.zhanzhang.baidu.com/push.js';
var s = document.getElementsByTagName("script")[0];
s.parentNode.insertBefore(bp, s);
})();
</script>
<script>
function loadCount() {
var d = document, s = d.createElement('script');
s.src = 'https://nicksxs.disqus.com/count.js';
s.id = 'dsq-count-scr';
(d.head || d.body).appendChild(s);
}
// defer loading until the whole page loading is completed
window.addEventListener('load', loadCount, false);
</script>
</body>
</html>