-
Notifications
You must be signed in to change notification settings - Fork 0
/
index.html
1101 lines (873 loc) · 132 KB
/
index.html
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
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Abeiv个人博客</title>
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
<meta property="og:type" content="website">
<meta property="og:title" content="Abeiv个人博客">
<meta property="og:url" content="http://yoursite.com/index.html">
<meta property="og:site_name" content="Abeiv个人博客">
<meta property="og:locale" content="zh-tw">
<meta name="twitter:card" content="summary">
<link rel="alternate" href="/atom.xml" title="Abeiv个人博客" type="application/atom+xml">
<link rel="icon" href="/1.jpg">
<link href="//fonts.googleapis.com/css?family=Source+Code+Pro" rel="stylesheet" type="text/css">
<link rel="stylesheet" href="/css/style.css">
<link rel="stylesheet" href="/css/highlight.css">
</head>
<meta name="baidu-site-verification" content="vsA6ZTqBVv" />
<body>
<div id="fullpage" class="mobile-nav-right">
<div id="wrapper" title="图片来自网络">
<header id="header">
<div id="nav-toggle" class="nav-toggle"></div>
<div class="head-box global-width">
<nav class="nav-box nav-right">
<a class="nav-item" href="/" title
>首页</a>
<a class="nav-item" href="/archives" title
>文章</a>
</nav>
</div>
</header>
<div id="middlecontent" title class="global-width sidebar-right">
<section id="main">
<article id="post-pikax" class="article global-container article-type-post" itemscope itemprop="blogPost">
<header class="article-header">
<h1 itemprop="name">
<a class="article-title" href="/2019/12/06/pikax/">pixiv批量下载工具</a>
</h1>
</header>
<div class="article-meta">
<a href="/2019/12/06/pikax/" class="article-date">
<time datetime="2019-12-06T09:49:54.041Z" itemprop="datePublished">2019-12-06</time>
</a>
<ul class="article-tag-list" itemprop="keywords"><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/pixiv/" rel="tag">pixiv</a></li><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/python/" rel="tag">python</a></li></ul>
</div>
<div class="article-inner">
<div class="article-content article-content-cloud" itemprop="articleBody">
<h2 id="Pikax"><a href="#Pikax" class="headerlink" title="Pikax"></a>Pikax</h2><p> 下载地址:</p>
<blockquote>
<p><a href="http://ambcc-shop.test.upcdn.net/blog/Pikax%200.1.9.exe" target="_blank" rel="noopener">点击下载</a><br><a href="http://ambcc-shop.test.upcdn.net/blog/Pikax%200.1.9.exe" target="_blank" rel="noopener">http://ambcc-shop.test.upcdn.net/blog/Pikax%200.1.9.exe</a> </p>
</blockquote>
</div>
</div>
</article>
<article id="post-httpclient400" class="article global-container article-type-post" itemscope itemprop="blogPost">
<header class="article-header">
<h1 itemprop="name">
<a class="article-title" href="/2019/12/06/httpclient400/">HttpClient 400</a>
</h1>
</header>
<div class="article-meta">
<a href="/2019/12/06/httpclient400/" class="article-date">
<time datetime="2019-12-06T09:40:56.863Z" itemprop="datePublished">2019-12-06</time>
</a>
<ul class="article-tag-list" itemprop="keywords"><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/400/" rel="tag">400</a></li><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/http/" rel="tag">http</a></li></ul>
</div>
<div class="article-inner">
<div class="article-content article-content-cloud" itemprop="articleBody">
<h2 id="HttpClient-发起网络请求-出现400"><a href="#HttpClient-发起网络请求-出现400" class="headerlink" title="HttpClient 发起网络请求 出现400"></a>HttpClient 发起网络请求 出现400</h2><p>解决:</p>
<blockquote>
<p>使用 URLEncoder.encode(“中文”, “utf-8”)</p>
</blockquote>
</div>
</div>
</article>
<article id="post-美团面试常见问题" class="article global-container article-type-post" itemscope itemprop="blogPost">
<header class="article-header">
<h1 itemprop="name">
<a class="article-title" href="/2019/11/27/%E7%BE%8E%E5%9B%A2%E9%9D%A2%E8%AF%95%E5%B8%B8%E8%A7%81%E9%97%AE%E9%A2%98/">美团面试常见问题总结</a>
</h1>
</header>
<div class="article-meta">
<a href="/2019/11/27/%E7%BE%8E%E5%9B%A2%E9%9D%A2%E8%AF%95%E5%B8%B8%E8%A7%81%E9%97%AE%E9%A2%98/" class="article-date">
<time datetime="2019-11-27T02:05:49.700Z" itemprop="datePublished">2019-11-27</time>
</a>
<ul class="article-tag-list" itemprop="keywords"><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/java/" rel="tag">java</a></li><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/%E9%9D%A2%E8%AF%95%E9%A2%98/" rel="tag">面试题</a></li></ul>
</div>
<div class="article-inner">
<div class="article-content article-content-cloud" itemprop="articleBody">
<ul>
<li><a href="#一-基础篇">一 基础篇</a><ul>
<li><a href="#1-systemoutprintln39输出什么">1. <code>System.out.println(3|9)</code>输出什么?</a></li>
<li><a href="#2-说一下转发forward和重定向redirect的区别">2. 说一下转发(Forward)和重定向(Redirect)的区别</a></li>
<li><a href="#3-在浏览器中输入url地址到显示主页的过程整个过程会使用哪些协议">3. 在浏览器中输入url地址到显示主页的过程,整个过程会使用哪些协议</a></li>
<li><a href="#4-tcp-三次握手和四次挥手">4. TCP 三次握手和四次挥手</a><ul>
<li><a href="#为什么要三次握手">为什么要三次握手</a></li>
<li><a href="#为什么要传回-syn">为什么要传回 SYN</a></li>
<li><a href="#传了-syn为啥还要传-ack">传了 SYN,为啥还要传 ACK</a></li>
<li><a href="#为什么要四次挥手">为什么要四次挥手</a></li>
</ul>
</li>
<li><a href="#5-ip地址与mac地址的区别">5. IP地址与MAC地址的区别</a></li>
<li><a href="#6-http请求响应报文格式">6. HTTP请求,响应报文格式</a></li>
<li><a href="#7-为什么要使用索引索引这么多优点为什么不对表中的每一个列创建一个索引呢索引是如何提高查询速度的说一下使用索引的注意事项mysql索引主要使用的两种数据结构什么是覆盖索引">7. 为什么要使用索引?索引这么多优点,为什么不对表中的每一个列创建一个索引呢?索引是如何提高查询速度的?说一下使用索引的注意事项?Mysql索引主要使用的两种数据结构?什么是覆盖索引?</a></li>
<li><a href="#8-进程与线程的区别是什么进程间的几种通信方式说一下线程间的几种通信方式知道不">8. 进程与线程的区别是什么?进程间的几种通信方式说一下?线程间的几种通信方式知道不?</a></li>
<li><a href="#9-为什么要用单例模式手写几种线程安全的单例模式">9. 为什么要用单例模式?手写几种线程安全的单例模式?</a></li>
<li><a href="#10-简单介绍一下bean知道spring的bean的作用域与生命周期吗">10. 简单介绍一下bean;知道Spring的bean的作用域与生命周期吗?</a></li>
<li><a href="#11-spring-中的事务传播行为了解吗transactiondefinition-接口中哪五个表示隔离级别的常量">11. Spring 中的事务传播行为了解吗?TransactionDefinition 接口中哪五个表示隔离级别的常量?</a><ul>
<li><a href="#事务传播行为">事务传播行为</a></li>
<li><a href="#隔离级别">隔离级别</a></li>
</ul>
</li>
<li><a href="#12-springmvc-原理了解吗">12. SpringMVC 原理了解吗?</a></li>
<li><a href="#13-spring-aop-ioc-实现原理">13. Spring AOP IOC 实现原理</a><ul>
<li><a href="#1-systemoutprintln39输出什么">1. <code>System.out.println(3|9)</code>输出什么?</a></li>
<li><a href="#2-说一下转发forward和重定向redirect的区别">2. 说一下转发(Forward)和重定向(Redirect)的区别</a></li>
<li><a href="#3-在浏览器中输入-url-地址到显示主页的过程整个过程会使用哪些协议">3. 在浏览器中输入 url 地址到显示主页的过程,整个过程会使用哪些协议</a></li>
<li><a href="#4-tcp-三次握手和四次挥手">4. TCP 三次握手和四次挥手</a><pre><code>- [为什么要三次握手](#为什么要三次握手)
- [为什么要传回 SYN](#为什么要传回-syn)
- [传了 SYN,为啥还要传 ACK](#传了-syn为啥还要传-ack)
- [为什么要四次挥手](#为什么要四次挥手)</code></pre></li>
<li><a href="#5-ip-地址与-mac-地址的区别">5. IP 地址与 MAC 地址的区别</a></li>
<li><a href="#6-http-请求响应报文格式">6. HTTP 请求,响应报文格式</a></li>
<li><a href="#7-为什么要使用索引索引这么多优点为什么不对表中的每一个列创建一个索引呢索引是如何提高查询速度的说一下使用索引的注意事项mysql-索引主要使用的两种数据结构什么是覆盖索引">7. 为什么要使用索引?索引这么多优点,为什么不对表中的每一个列创建一个索引呢?索引是如何提高查询速度的?说一下使用索引的注意事项?Mysql 索引主要使用的两种数据结构?什么是覆盖索引?</a></li>
<li><a href="#8-进程与线程的区别是什么进程间的几种通信方式说一下线程间的几种通信方式知道不">8. 进程与线程的区别是什么?进程间的几种通信方式说一下?线程间的几种通信方式知道不?</a></li>
<li><a href="#9-为什么要用单例模式手写几种线程安全的单例模式">9. 为什么要用单例模式?手写几种线程安全的单例模式?</a></li>
<li><a href="#10-简单介绍一下-bean知道-spring-的-bean-的作用域与生命周期吗">10. 简单介绍一下 bean;知道 Spring 的 bean 的作用域与生命周期吗?</a></li>
<li><a href="#11-spring-中的事务传播行为了解吗transactiondefinition-接口中哪五个表示隔离级别的常量">11. Spring 中的事务传播行为了解吗?TransactionDefinition 接口中哪五个表示隔离级别的常量?</a><pre><code>- [事务传播行为](#事务传播行为)
- [隔离级别](#隔离级别)</code></pre></li>
<li><a href="#12-springmvc-原理了解吗">12. SpringMVC 原理了解吗?</a></li>
<li><a href="#13-spring-aop-ioc-实现原理">13. Spring AOP IOC 实现原理</a></li>
</ul>
</li>
</ul>
</li>
<li><a href="#二-进阶篇">二 进阶篇</a><ul>
<li><a href="#1-消息队列mq的套路">1 消息队列MQ的套路</a><ul>
<li><a href="#11-介绍一下消息队列mq的应用场景使用消息队列的好处">1.1 介绍一下消息队列MQ的应用场景/使用消息队列的好处</a><ul>
<li><a href="#1通过异步处理提高系统性能">1)通过异步处理提高系统性能</a></li>
<li><a href="#2降低系统耦合性">2)降低系统耦合性</a></li>
</ul>
</li>
<li><a href="#12-那么使用消息队列会带来什么问题考虑过这些问题吗">1.2 那么使用消息队列会带来什么问题?考虑过这些问题吗?</a></li>
<li><a href="#13-介绍一下你知道哪几种消息队列该如何选择呢">1.3 介绍一下你知道哪几种消息队列,该如何选择呢?</a></li>
<li><a href="#14-关于消息队列其他一些常见的问题展望">1.4 关于消息队列其他一些常见的问题展望</a></li>
</ul>
</li>
<li><a href="#2-谈谈-innodb-和-myisam-两者的区别">2 谈谈 InnoDB 和 MyIsam 两者的区别</a><ul>
<li><a href="#21-两者的对比">2.1 两者的对比</a></li>
<li><a href="#22-关于两者的总结">2.2 关于两者的总结</a></li>
</ul>
</li>
<li><a href="#3-聊聊-java-中的集合吧">3 聊聊 Java 中的集合吧!</a><ul>
<li><a href="#31-arraylist-与-linkedlist-有什么不同注意加上从数据结构分析的内容">3.1 ArrayList 与 LinkedList 有什么不同?(注意加上从数据结构分析的内容)</a></li>
<li><a href="#32-hashmap的底层实现">3.2 HashMap的底层实现</a><ul>
<li><a href="#1jdk18之前">1)JDK1.8之前</a></li>
<li><a href="#2jdk18之后">2)JDK1.8之后</a></li>
</ul>
</li>
<li><a href="#33-既然谈到了红黑树你给我手绘一个出来吧然后简单讲一下自己对于红黑树的理解">3.3 既然谈到了红黑树,你给我手绘一个出来吧,然后简单讲一下自己对于红黑树的理解</a></li>
<li><a href="#34-红黑树这么优秀为何不直接使用红黑树得了">3.4 红黑树这么优秀,为何不直接使用红黑树得了?</a></li>
<li><a href="#35-hashmap-和-hashtable-的区别hashset-和-hashmap-区别">3.5 HashMap 和 Hashtable 的区别/HashSet 和 HashMap 区别</a></li>
<li><a href="#1-消息队列-mq-的套路">1 消息队列 MQ 的套路</a><ul>
<li><a href="#11-介绍一下消息队列-mq-的应用场景使用消息队列的好处">1.1 介绍一下消息队列 MQ 的应用场景/使用消息队列的好处</a><ul>
<li><a href="#1通过异步处理提高系统性能">1)通过异步处理提高系统性能</a></li>
<li><a href="#2降低系统耦合性">2)降低系统耦合性</a></li>
</ul>
</li>
<li><a href="#12-那么使用消息队列会带来什么问题考虑过这些问题吗">1.2 那么使用消息队列会带来什么问题?考虑过这些问题吗?</a></li>
<li><a href="#13-介绍一下你知道哪几种消息队列该如何选择呢">1.3 介绍一下你知道哪几种消息队列,该如何选择呢?</a></li>
<li><a href="#14-关于消息队列其他一些常见的问题展望">1.4 关于消息队列其他一些常见的问题展望</a></li>
</ul>
</li>
<li><a href="#2-谈谈-innodb-和-myisam-两者的区别">2 谈谈 InnoDB 和 MyIsam 两者的区别</a><ul>
<li><a href="#21-两者的对比">2.1 两者的对比</a></li>
<li><a href="#22-关于两者的总结">2.2 关于两者的总结</a></li>
</ul>
</li>
<li><a href="#3-聊聊-java-中的集合吧">3 聊聊 Java 中的集合吧!</a><ul>
<li><a href="#31-arraylist-与-linkedlist-有什么不同注意加上从数据结构分析的内容">3.1 Arraylist 与 LinkedList 有什么不同?(注意加上从数据结构分析的内容)</a></li>
<li><a href="#32-hashmap-的底层实现">3.2 HashMap 的底层实现</a><ul>
<li><a href="#1jdk18-之前">1)JDK1.8 之前</a></li>
<li><a href="#2jdk18-之后">2)JDK1.8 之后</a></li>
</ul>
</li>
<li><a href="#33-既然谈到了红黑树你给我手绘一个出来吧然后简单讲一下自己对于红黑树的理解">3.3 既然谈到了红黑树,你给我手绘一个出来吧,然后简单讲一下自己对于红黑树的理解</a></li>
<li><a href="#34-红黑树这么优秀为何不直接使用红黑树得了">3.4 红黑树这么优秀,为何不直接使用红黑树得了?</a></li>
<li><a href="#35-hashmap-和-hashtable-的区别hashset-和-hashmap-区别">3.5 HashMap 和 Hashtable 的区别/HashSet 和 HashMap 区别</a></li>
</ul>
</li>
</ul>
</li>
</ul>
</li>
<li><a href="#三-终结篇">三 终结篇</a><ul>
<li><a href="#1-object类有哪些方法">1. Object类有哪些方法?</a><ul>
<li><a href="#11-object类的常见方法总结">1.1 Object类的常见方法总结</a></li>
<li><a href="#12-hashcode与equals">1.2 hashCode与equals</a><ul>
<li><a href="#121-hashcode介绍">1.2.1 hashCode()介绍</a></li>
<li><a href="#122-为什么要有hashcode">1.2.2 为什么要有hashCode</a></li>
<li><a href="#123-hashcode与equals的相关规定">1.2.3 hashCode()与equals()的相关规定</a></li>
<li><a href="#124-为什么两个对象有相同的hashcode值它们也不一定是相等的">1.2.4 为什么两个对象有相同的hashcode值,它们也不一定是相等的?</a></li>
</ul>
</li>
<li><a href="#13-与equals">1.3 ==与equals</a></li>
</ul>
</li>
<li><a href="#2-concurrenthashmap-相关问题">2 ConcurrentHashMap 相关问题</a><ul>
<li><a href="#21-concurrenthashmap-和-hashtable-的区别">2.1 ConcurrentHashMap 和 Hashtable 的区别</a></li>
<li><a href="#22-concurrenthashmap线程安全的具体实现方式底层具体实现">2.2 ConcurrentHashMap线程安全的具体实现方式/底层具体实现</a><ul>
<li><a href="#jdk17上面有示意图">JDK1.7(上面有示意图)</a></li>
<li><a href="#jdk18上面有示意图">JDK1.8(上面有示意图)</a></li>
</ul>
</li>
</ul>
</li>
<li><a href="#3-谈谈-synchronized-和-reentrantlock-的区别">3 谈谈 synchronized 和 ReenTrantLock 的区别</a></li>
<li><a href="#4-线程池了解吗">4 线程池了解吗?</a><ul>
<li><a href="#41-为什么要用线程池">4.1 为什么要用线程池?</a></li>
<li><a href="#42-java-提供了哪几种线程池他们各自的使用场景是什么">4.2 Java 提供了哪几种线程池?他们各自的使用场景是什么?</a><ul>
<li><a href="#java-主要提供了下面4种线程池">Java 主要提供了下面4种线程池</a></li>
<li><a href="#各种线程池的适用场景介绍">各种线程池的适用场景介绍</a></li>
</ul>
</li>
<li><a href="#43-创建的线程池的方式">4.3 创建的线程池的方式</a></li>
</ul>
</li>
<li><a href="#5-nginx">5 Nginx</a><ul>
<li><a href="#51-简单介绍一下nginx">5.1 简单介绍一下Nginx</a><ul>
<li><a href="#反向代理">反向代理</a></li>
<li><a href="#负载均衡">负载均衡</a></li>
<li><a href="#动静分离">动静分离</a></li>
</ul>
</li>
<li><a href="#52-为什么要用-nginx">5.2 为什么要用 Nginx?</a></li>
<li><a href="#53-nginx-的四个主要组成部分了解吗">5.3 Nginx 的四个主要组成部分了解吗?</a></li>
</ul>
</li>
</ul>
</li>
</ul>
<!-- /MarkdownTOC -->
<p>这些问题是2018年去美团面试的同学被问到的一些常见的问题,希望对你有帮助!<br> - <a href="#1-object-类有哪些方法">1. Object 类有哪些方法?</a><br> - <a href="#11-object-类的常见方法总结">1.1 Object 类的常见方法总结</a><br> - <a href="#12-hashcode-与-equals">1.2 hashCode 与 equals</a><br> - <a href="#121-hashcode介绍">1.2.1 hashCode()介绍</a><br> - <a href="#122-为什么要有-hashcode">1.2.2 为什么要有 hashCode</a><br> - <a href="#123-hashcode与-equals的相关规定">1.2.3 hashCode()与 equals()的相关规定</a><br> - <a href="#124-为什么两个对象有相同的-hashcode-值它们也不一定是相等的">1.2.4 为什么两个对象有相同的 hashcode 值,它们也不一定是相等的?</a><br> - <a href="#13-与-equals">1.3 ==与 equals</a><br> - <a href="#2-concurrenthashmap-相关问题">2 ConcurrentHashMap 相关问题</a><br> - <a href="#21-concurrenthashmap-和-hashtable-的区别">2.1 ConcurrentHashMap 和 Hashtable 的区别</a><br> - <a href="#22-concurrenthashmap-线程安全的具体实现方式底层具体实现">2.2 ConcurrentHashMap 线程安全的具体实现方式/底层具体实现</a><br> - <a href="#jdk17上面有示意图">JDK1.7(上面有示意图)</a><br> - <a href="#jdk18上面有示意图">JDK1.8(上面有示意图)</a><br> - <a href="#3-谈谈-synchronized-和-reentrantlock-的区别">3 谈谈 synchronized 和 ReentrantLock 的区别</a><br> - <a href="#4-线程池了解吗">4 线程池了解吗?</a><br> - <a href="#41-为什么要用线程池">4.1 为什么要用线程池?</a><br> - <a href="#42-java-提供了哪几种线程池他们各自的使用场景是什么">4.2 Java 提供了哪几种线程池?他们各自的使用场景是什么?</a><br> - <a href="#java-主要提供了下面-4-种线程池">Java 主要提供了下面 4 种线程池</a><br> - <a href="#各种线程池的适用场景介绍">各种线程池的适用场景介绍</a><br> - <a href="#43-创建的线程池的方式">4.3 创建的线程池的方式</a><br> - <a href="#5-nginx">5 Nginx</a><br> - <a href="#51-简单介绍一下-nginx">5.1 简单介绍一下 Nginx</a><br> - <a href="#反向代理">反向代理</a><br> - <a href="#负载均衡">负载均衡</a><br> - <a href="#动静分离">动静分离</a><br> - <a href="#52-为什么要用-nginx">5.2 为什么要用 Nginx?</a><br> - <a href="#53-nginx-的四个主要组成部分了解吗">5.3 Nginx 的四个主要组成部分了解吗?</a></p>
<!-- /TOC -->
<p>这些问题是 2018 年去美团面试的同学被问到的一些常见的问题,希望对你有帮助!</p>
<h1 id="一-基础篇"><a href="#一-基础篇" class="headerlink" title="一 基础篇"></a>一 基础篇</h1><h2 id="1-System-out-println-3-9-输出什么"><a href="#1-System-out-println-3-9-输出什么" class="headerlink" title="1. System.out.println(3|9)输出什么?"></a>1. <code>System.out.println(3|9)</code>输出什么?</h2><p>正确答案:11。</p>
<p><strong>&和&&:</strong></p>
<p>共同点:两者都可做逻辑运算符。它们都表示运算符的两边都是true时,结果为true;</p>
<p>不同点: &也是位运算符。& 表示在运算时两边都会计算,然后再判断;&&表示先运算符号左边的东西,然后判断是否为true,是true就继续运算右边的然后判断并输出,是false就停下来直接输出不会再运行后面的东西。</p>
<p><strong>|和||:</strong></p>
<p>共同点:两者都可做逻辑运算符。它们都表示运算符的两边任意一边为true,结果为true,两边都不是true,结果就为false;</p>
<p>不同点:|也是位运算符。| 表示两边都会运算,然后再判断结果;|| 表示先运算符号左边的东西,然后判断是否为 true,是 true 就停下来直接输出不会再运行后面的东西,是 false 就继续运算右边的然后判断并输出。</p>
<p><strong>回到本题:</strong></p>
<p>3 | 9=0011(二进制) | 1001(二进制)=1011(二进制)=11(十进制)</p>
<h2 id="2-说一下转发-Forward-和重定向-Redirect-的区别"><a href="#2-说一下转发-Forward-和重定向-Redirect-的区别" class="headerlink" title="2. 说一下转发(Forward)和重定向(Redirect)的区别"></a>2. 说一下转发(Forward)和重定向(Redirect)的区别</h2><p><strong>回到本题:</strong></p>
<p><strong>转发是服务器行为,重定向是客户端行为。</strong></p>
<p><strong>转发(Forword)</strong> 通过RequestDispatcher对象的<code>forward(HttpServletRequest request,HttpServletResponse response)</code>方法实现的。<code>RequestDispatcher</code> 可以通过<code>HttpServletRequest</code> 的 <code>getRequestDispatcher()</code>方法获得。例如下面的代码就是跳转到 login_success.jsp 页面。</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="JAVA"><figure class="iseeu highlight /java"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">request.getRequestDispatcher(<span class="string">"login_success.jsp"</span>).forward(request, response);</span><br></pre></td></tr></table></figure></div>
<p><strong>重定向(Redirect)</strong> 是利用服务器返回的状态码来实现的。客户端浏览器请求服务器的时候,服务器会返回一个状态码。服务器通过HttpServletRequestResponse的setStatus(int status)方法设置状态码。如果服务器返回301或者302,则浏览器会到新的网址重新请求该资源。</p>
<ol>
<li><strong>从地址栏显示来说</strong>:forward是服务器请求资源,服务器直接访问目标地址的URL,把那个URL的响应内容读取过来,然后把这些内容再发给浏览器。浏览器根本不知道服务器发送的内容从哪里来的,所以它的地址栏还是原来的地址。redirect是服务端根据逻辑,发送一个状态码,告诉浏览器重新去请求那个地址。所以地址栏显示的是新的URL。</li>
<li><strong>从数据共享来说</strong>:forward:转发页面和转发到的页面可以共享request里面的数据。redirect:不能共享数据。</li>
<li><strong>从地址栏显示来说</strong>:forward 是服务器请求资源,服务器直接访问目标地址的 URL,把那个 URL 的响应内容读取过来,然后把这些内容再发给浏览器。浏览器根本不知道服务器发送的内容从哪里来的,所以它的地址栏还是原来的地址。redirect 是服务端根据逻辑,发送一个状态码,告诉浏览器重新去请求那个地址。所以地址栏显示的是新的 URL。</li>
<li><strong>从数据共享来说</strong>:forward:转发页面和转发到的页面可以共享 request 里面的数据。redirect:不能共享数据。</li>
<li><strong>从运用地方来说</strong>:forward:一般用于用户登陆的时候,根据角色转发到相应的模块。redirect:一般用于用户注销登陆时返回主页面和跳转到其它的网站等。</li>
<li><strong>从效率来说</strong>:forward:高。redirect:低。</li>
</ol>
<h2 id="3-在浏览器中输入-url-地址到显示主页的过程-整个过程会使用哪些协议"><a href="#3-在浏览器中输入-url-地址到显示主页的过程-整个过程会使用哪些协议" class="headerlink" title="3. 在浏览器中输入 url 地址到显示主页的过程,整个过程会使用哪些协议"></a>3. 在浏览器中输入 url 地址到显示主页的过程,整个过程会使用哪些协议</h2><p>图片来源:《图解HTTP》:</p>
<p><img src="https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-11/%E5%90%84%E7%A7%8D%E7%BD%91%E7%BB%9C%E8%AF%B7%E6%B1%82%E7%94%A8%E5%88%B0%E7%9A%84%E5%8D%8F%E8%AE%AE.jpg" alt="各种网络请求用到的协议"></p>
<p>总体来说分为以下几个过程:</p>
<ol>
<li>DNS解析</li>
<li>TCP连接</li>
<li>发送HTTP请求</li>
<li>服务器处理请求并返回HTTP报文</li>
<li>DNS 解析</li>
<li>TCP 连接</li>
<li>发送 HTTP 请求</li>
<li>服务器处理请求并返回 HTTP 报文</li>
<li>浏览器解析渲染页面</li>
<li>连接结束</li>
</ol>
<p>具体可以参考下面这篇文章:</p>
<ul>
<li><a href="https://segmentfault.com/a/1190000006879700" target="_blank" rel="noopener" title="https://segmentfault.com/a/1190000006879700">https://segmentfault.com/a/1190000006879700</a></li>
</ul>
<blockquote>
<p>修正 <a href="https://github.com/Snailclimb/JavaGuide/issues/568" target="_blank" rel="noopener" title="issue-568">issue-568</a>:上图中 IP 数据包在路由器之间使用的协议为 OPSF 协议错误,应该为 OSPF 协议 。</p>
<p>IP 数据包在路由器之间传播大致分为 IGP 和 BGP 协议,而 IGP 目前主流为 OSPF 协议,思科,华为和 H3C 等主流厂商都有各自实现并使用;BGP 协议为不同 AS(自治系统号)间路由传输,也分为 I-BGP 和 E-BGP,详细资料请查看《TCP/IP 卷一》</p>
</blockquote>
<h2 id="4-TCP-三次握手和四次挥手"><a href="#4-TCP-三次握手和四次挥手" class="headerlink" title="4. TCP 三次握手和四次挥手"></a>4. TCP 三次握手和四次挥手</h2><p>为了准确无误地把数据送达目标处,TCP协议采用了三次握手策略。</p>
<p><strong>漫画图解:</strong></p>
<p>图片来源:《图解HTTP》<br><img src="https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-11/tcp三次握手.jpg" style="zoom:50%;" /></p>
<p><strong>简单示意图:</strong><br>request.getRequestDispatcher(“login_success.jsp”).forward(request, response);<br>所以三次握手就能确认双发收发功能都正常,缺一不可。</p>
<h4 id="为什么要传回-SYN"><a href="#为什么要传回-SYN" class="headerlink" title="为什么要传回 SYN"></a>为什么要传回 SYN</h4><p>接收端传回发送端所发送的 SYN 是为了告诉发送端,我接收到的信息确实就是你所发送的信号了。</p>
<blockquote>
<p>SYN 是 TCP/IP 建立连接时使用的握手信号。在客户机和服务器之间建立正常的 TCP 网络连接时,客户机首先发出一个 SYN 消息,服务器使用 SYN-ACK 应答表示接收到了这个消息,最后客户机再以 ACK(Acknowledgement[汉译:确认字符 ,在数据通信传输中,接收站发给发送站的一种传输控制字符。它表示确认发来的数据已经接受无误。 ])消息响应。这样在客户机和服务器之间才能建立起可靠的TCP连接,数据才可以在客户机和服务器之间传递。<br>接收端传回发送端所发送的 SYN 是为了告诉发送端,我接收到的信息确实就是你所发送的信号了。</p>
</blockquote>
<h4 id="传了-SYN-为啥还要传-ACK"><a href="#传了-SYN-为啥还要传-ACK" class="headerlink" title="传了 SYN,为啥还要传 ACK"></a>传了 SYN,为啥还要传 ACK</h4><p>双方通信无误必须是两者互相发送信息都无误。传了 SYN,证明发送方(主动关闭方)到接收方(被动关闭方)的通道没有问题,但是接收方到发送方的通道还需要 ACK 信号来进行验证。</p>
<p>断开一个 TCP 连接则需要“四次挥手”:</p>
<ul>
<li>客户端-发送一个 FIN,用来关闭客户端到服务器的数据传送</li>
<li>服务器-收到这个 FIN,它发回一 个 ACK,确认序号为收到的序号加1 。和 SYN 一样,一个 FIN 将占用一个序号</li>
<li>服务器-关闭与客户端的连接,发送一个FIN给客户端</li>
<li>客户端-发回 ACK 报文确认,并将确认序号设置为收到序号加1</li>
</ul>
<h4 id="为什么要四次挥手"><a href="#为什么要四次挥手" class="headerlink" title="为什么要四次挥手"></a>为什么要四次挥手</h4><p>任何一方都可以在数据传送结束后发出连接释放的通知,待对方确认后进入半关闭状态。当另一方也没有数据再发送的时候,则发出连接释放通知,对方确认后就完全关闭了 TCP 连接。</p>
<p>举个例子:A 和 B 打电话,通话即将结束后,A 说“我没啥要说的了”,B 回答“我知道了”,但是 B 可能还会有要说的话,A 不能要求 B 跟着自己的节奏结束通话,于是 B 可能又巴拉巴拉说了一通,最后 B 说“我说完了”,A 回答“知道了”,这样通话才算结束。</p>
<h2 id="5-IP-地址与-MAC-地址的区别"><a href="#5-IP-地址与-MAC-地址的区别" class="headerlink" title="5. IP 地址与 MAC 地址的区别"></a>5. IP 地址与 MAC 地址的区别</h2><p>IP地址是指互联网协议地址(Internet Protocol Address)IP Address的缩写。IP地址是IP协议提供的一种统一的地址格式,它为互联网上的每一个网络和每一台主机分配一个逻辑地址,以此来屏蔽物理地址的差异。<br>参考:<a href="https://blog.csdn.net/guoweimelon/article/details/50858597" target="_blank" rel="noopener" title="https://blog.csdn.net/guoweimelon/article/details/50858597">https://blog.csdn.net/guoweimelon/article/details/50858597</a></p>
<p>IP 地址是指互联网协议地址(Internet Protocol Address)IP Address 的缩写。IP 地址是 IP 协议提供的一种统一的地址格式,它为互联网上的每一个网络和每一台主机分配一个逻辑地址,以此来屏蔽物理地址的差异。</p>
<p>MAC 地址又称为物理地址、硬件地址,用来定义网络设备的位置。网卡的物理地址通常是由网卡生产厂家写入网卡的,具有全球唯一性。MAC地址用于在网络中唯一标示一个网卡,一台电脑会有一或多个网卡,每个网卡都需要有一个唯一的MAC地址。</p>
<h2 id="6-HTTP请求-响应报文格式"><a href="#6-HTTP请求-响应报文格式" class="headerlink" title="6. HTTP请求,响应报文格式"></a>6. HTTP请求,响应报文格式</h2><p>HTTP响应报文主要由状态行、响应头部、响应正文3部分组成</p>
<p>详细内容可以参考:<a href="https://blog.csdn.net/a19881029/article/details/14002273" target="_blank" rel="noopener">https://blog.csdn.net/a19881029/article/details/14002273</a></p>
<h2 id="7-为什么要使用索引-索引这么多优点-为什么不对表中的每一个列创建一个索引呢-索引是如何提高查询速度的-说一下使用索引的注意事项-Mysql索引主要使用的两种数据结构-什么是覆盖索引"><a href="#7-为什么要使用索引-索引这么多优点-为什么不对表中的每一个列创建一个索引呢-索引是如何提高查询速度的-说一下使用索引的注意事项-Mysql索引主要使用的两种数据结构-什么是覆盖索引" class="headerlink" title="7. 为什么要使用索引?索引这么多优点,为什么不对表中的每一个列创建一个索引呢?索引是如何提高查询速度的?说一下使用索引的注意事项?Mysql索引主要使用的两种数据结构?什么是覆盖索引?"></a>7. 为什么要使用索引?索引这么多优点,为什么不对表中的每一个列创建一个索引呢?索引是如何提高查询速度的?说一下使用索引的注意事项?Mysql索引主要使用的两种数据结构?什么是覆盖索引?</h2><p><strong>为什么要使用索引?</strong></p>
<ol>
<li>通过创建唯一性索引,可以保证数据库表中每一行数据的唯一性。</li>
<li>可以大大加快 数据的检索速度(大大减少的检索的数据量), 这也是创建索引的最主要的原因。 </li>
<li>帮助服务器避免排序和临时表</li>
<li>将随机IO变为顺序IO</li>
<li>可以加速表和表之间的连接,特别是在实现数据的参考完整性方面特别有意义。</li>
</ol>
<p><strong>索引这么多优点,为什么不对表中的每一个列创建一个索引呢?</strong></p>
<ol>
<li>当对表中的数据进行增加、删除和修改的时候,索引也要动态的维护,这样就降低了数据的维护速度。 </li>
<li>索引需要占物理空间,除了数据表占数据空间之外,每一个索引还要占一定的物理空间,如果要建立聚簇索引,那么需要的空间就会更大。 </li>
<li>创建索引和维护索引要耗费时间,这种时间随着数据量的增加而增加。 </li>
</ol>
<p><strong>索引是如何提高查询速度的?</strong></p>
<p>HTTP响应报文主要由状态行、响应头部、响应正文3部分组成<br><strong>说一下使用索引的注意事项</strong></p>
<ol>
<li>避免 where 子句中对字段施加函数,这会造成无法命中索引。</li>
<li>在使用InnoDB时使用与业务无关的自增主键作为主键,即使用逻辑主键,而不要使用业务主键。</li>
<li>将打算加索引的列设置为 NOT NULL ,否则将导致引擎放弃使用索引而进行全表扫描</li>
<li>删除长期未使用的索引,不用的索引的存在会造成不必要的性能损耗 MySQL 5.7 可以通过查询 sys 库的 schema_unused_indexes 视图来查询哪些索引从未被使用</li>
<li>在使用 limit offset 查询缓慢时,可以借助索引来提高性能</li>
</ol>
<p><strong>Mysql索引主要使用的哪两种数据结构?</strong></p>
<ul>
<li>哈希索引:对于哈希索引来说,底层的数据结构就是哈希表,因此在绝大多数需求为单条记录查询的时候,可以选择哈希索引,查询性能最快;其余大部分场景,建议选择BTree索引。</li>
<li>BTree索引:Mysql的BTree索引使用的是B树中的B+Tree。但对于主要的两种存储引擎(MyISAM和InnoDB)的实现方式是不同的。</li>
</ul>
<p>更多关于索引的内容可以查看我的这篇文章:<a href="https://mp.weixin.qq.com/s?__biz=MzU4NDQ4MzU5OA==&mid=2247484486&idx=1&sn=215450f11e042bca8a58eac9f4a97686&chksm=fd985227caefdb3117b8375f150676f5824aa20d1ebfdbcfb93ff06e23e26efbafae6cf6b48e&token=1990180468&lang=zh_CN#rd" target="_blank" rel="noopener">【思维导图-索引篇】搞定数据库索引就是这么简单</a></p>
<p><strong>什么是覆盖索引?</strong></p>
<p>如果一个索引包含(或者说覆盖)所有需要查询的字段的值,我们就称<br>之为“覆盖索引”。我们知道在InnoDB存储引擎中,如果不是主键索引,叶子节点存储的是主键+列值。最终还是要“回表”,也就是要通过主键再查找一次,这样就会比较慢。覆盖索引就是把要查询出的列和索引是对应的,不做回表操作!</p>
<p>之为“覆盖索引”。我们知道在 InnoDB 存储引擎中,如果不是主键索引,叶子节点存储的是主键+列值。最终还是要“回表”,也就是要通过主键再查找一次,这样就会比较慢。覆盖索引就是把要查询出的列和索引是对应的,不做回表操作!</p>
<h2 id="8-进程与线程的区别是什么-进程间的几种通信方式说一下-线程间的几种通信方式知道不"><a href="#8-进程与线程的区别是什么-进程间的几种通信方式说一下-线程间的几种通信方式知道不" class="headerlink" title="8. 进程与线程的区别是什么?进程间的几种通信方式说一下?线程间的几种通信方式知道不?"></a>8. 进程与线程的区别是什么?进程间的几种通信方式说一下?线程间的几种通信方式知道不?</h2><p> <strong>进程与线程的区别是什么?</strong></p>
<p>线程与进程相似,但线程是一个比进程更小的执行单位。一个进程在其执行的过程中可以产生多个线程。与进程不同的是同类的多个线程共享同一块内存空间和一组系统资源,所以系统在产生一个线程,或是在各个线程之间作切换工作时,负担要比进程小得多,也正因为如此,线程也被称为轻量级进程。另外,也正是因为共享资源,所以线程中执行时一般都要进行同步和互斥。总的来说,进程和线程的主要差别在于它们是不同的操作系统资源管理方式。</p>
<p><strong>进程间的几种通信方式说一下?</strong></p>
<ol>
<li><strong>管道(pipe)</strong>:管道是一种半双工的通信方式,数据只能单向流动,而且只能在具有血缘关系的进程间使用。进程的血缘关系通常指父子进程关系。管道分为pipe(无名管道)和fifo(命名管道)两种,有名管道也是半双工的通信方式,但是它允许无亲缘关系进程间通信。</li>
<li><strong>信号量(semophore)</strong>:信号量是一个计数器,可以用来控制多个进程对共享资源的访问。它通常作为一种锁机制,防止某进程正在访问共享资源时,其他进程也访问该资源。因此,主要作为进程间以及同一进程内不同线程之间的同步手段。</li>
<li><strong>消息队列(message queue)</strong>:消息队列是由消息组成的链表,存放在内核中 并由消息队列标识符标识。消息队列克服了信号传递信息少,管道只能承载无格式字节流以及缓冲区大小受限等缺点。消息队列与管道通信相比,其优势是对每个消息指定特定的消息类型,接收的时候不需要按照队列次序,而是可以根据自定义条件接收特定类型的消息。</li>
<li><strong>信号(signal)</strong>:信号是一种比较复杂的通信方式,用于通知接收进程某一事件已经发生。</li>
<li><strong>共享内存(shared memory)</strong>:共享内存就是映射一段能被其他进程所访问的内存,这段共享内存由一个进程创建,但多个进程都可以访问,共享内存是最快的IPC方式,它是针对其他进程间的通信方式运行效率低而专门设计的。它往往与其他通信机制,如信号量配合使用,来实现进程间的同步和通信。</li>
<li><strong>套接字(socket)</strong>:socket,即套接字是一种通信机制,凭借这种机制,客户/服务器(即要进行通信的进程)系统的开发工作既可以在本地单机上进行,也可以跨网络进行。也就是说它可以让不在同一台计算机但通过网络连接计算机上的进程进行通信。也因为这样,套接字明确地将客户端和服务器区分开来。</li>
</ol>
<p><strong>线程间的几种通信方式知道不?</strong><br>public class Singleton {<br>只有通过显式调用 getInstance 方法时,才会显式装载 SingletonHolder 类,从而实例化 instance(只有第一次使用这个单例的实例的时候才加载,同时不会有线程安全问题)。</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="JAVA"><figure class="iseeu 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></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">Singleton</span> </span>{ </span><br><span class="line"> <span class="keyword">private</span> <span class="keyword">static</span> <span class="class"><span class="keyword">class</span> <span class="title">SingletonHolder</span> </span>{ </span><br><span class="line"> <span class="keyword">private</span> <span class="keyword">static</span> <span class="keyword">final</span> Singleton INSTANCE = <span class="keyword">new</span> Singleton(); </span><br><span class="line"> } </span><br><span class="line"> <span class="function"><span class="keyword">private</span> <span class="title">Singleton</span> <span class="params">()</span></span>{} </span><br><span class="line"> <span class="function"><span class="keyword">public</span> <span class="keyword">static</span> <span class="keyword">final</span> Singleton <span class="title">getInstance</span><span class="params">()</span> </span>{ </span><br><span class="line"> <span class="keyword">return</span> SingletonHolder.INSTANCE; </span><br><span class="line"> } </span><br><span class="line">}</span><br></pre></td></tr></table></figure></div>
<h2 id="10-简单介绍一下bean-知道Spring的bean的作用域与生命周期吗"><a href="#10-简单介绍一下bean-知道Spring的bean的作用域与生命周期吗" class="headerlink" title="10. 简单介绍一下bean;知道Spring的bean的作用域与生命周期吗?"></a>10. 简单介绍一下bean;知道Spring的bean的作用域与生命周期吗?</h2><p>在 Spring 中,那些组成应用程序的主体及由 Spring IOC 容器所管理的对象,被称之为 bean。简单地讲,bean 就是由 IOC 容器初始化、装配及管理的对象,除此之外,bean 就与应用程序中的其他对象没有什么区别了。而 bean 的定义以及 bean 相互间的依赖关系将通过配置元数据来描述。</p>
<p>Spring 中的 bean 默认都是单例的,这些单例 Bean 在多线程程序下如何保证线程安全呢? 例如对于 Web 应用来说,Web 容器对于每个用户请求都创建一个单独的 Sevlet 线程来处理请求,引入 Spring 框架之后,每个 Action 都是单例的,那么对于 Spring 托管的单例 Service Bean,如何保证其安全呢? Spring 的单例是基于 BeanFactory 也就是 Spring 容器的,单例 Bean 在此容器内只有一个,Java 的单例是基于 JVM,每个 JVM 内只有一个实例。</p>
<p><img src="https://user-gold-cdn.xitu.io/2018/11/10/166fd45773d5dd2e?w=563&h=299&f=webp&s=27930" alt="pring的bean的作用域"></p>
<p>Spring的bean的生命周期以及更多内容可以查看:<a href="https://mp.weixin.qq.com/s?__biz=MzU4NDQ4MzU5OA==&mid=2247484400&idx=2&sn=7201eb365102fce017f89cb3527fb0bc&chksm=fd985591caefdc872a2fac897288119f94c345e4e12150774f960bf5f816b79e4b9b46be3d7f&token=1990180468&lang=zh_CN#rd" target="_blank" rel="noopener">一文轻松搞懂Spring中bean的作用域与生命周期</a></p>
<h2 id="11-Spring-中的事务传播行为了解吗-TransactionDefinition-接口中哪五个表示隔离级别的常量"><a href="#11-Spring-中的事务传播行为了解吗-TransactionDefinition-接口中哪五个表示隔离级别的常量" class="headerlink" title="11. Spring 中的事务传播行为了解吗?TransactionDefinition 接口中哪五个表示隔离级别的常量?"></a>11. Spring 中的事务传播行为了解吗?TransactionDefinition 接口中哪五个表示隔离级别的常量?</h2><h4 id="事务传播行为"><a href="#事务传播行为" class="headerlink" title="事务传播行为"></a>事务传播行为</h4><p>事务传播行为(为了解决业务层方法之间互相调用的事务问题):<br>当事务方法被另一个事务方法调用时,必须指定事务应该如何传播。例如:方法可能继续在现有事务中运行,也可能开启一个新事务,并在自己的事务中运行。在TransactionDefinition定义中包括了如下几个表示传播行为的常量:<br>当事务方法被另一个事务方法调用时,必须指定事务应该如何传播。例如:方法可能继续在现有事务中运行,也可能开启一个新事务,并在自己的事务中运行。在 TransactionDefinition 定义中包括了如下几个表示传播行为的常量:</p>
<p><strong>支持当前事务的情况:</strong></p>
<p> Spring的bean的生命周期以及更多内容可以查看:[一文轻松搞懂</p>
<p><strong>其他情况:</strong></p>
<ul>
<li>TransactionDefinition.PROPAGATION_NESTED: 如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行;如果当前没有事务,则该取值等价于TransactionDefinition.PROPAGATION_REQUIRED。</li>
</ul>
<h4 id="隔离级别"><a href="#隔离级别" class="headerlink" title="隔离级别"></a>隔离级别</h4><p>TransactionDefinition 接口中定义了五个表示隔离级别的常量:</p>
<ul>
<li><strong>TransactionDefinition.ISOLATION_DEFAULT:</strong> 使用后端数据库默认的隔离级别,Mysql 默认采用的 REPEATABLE_READ隔离级别 Oracle 默认采用的 READ_COMMITTED隔离级别.</li>
<li><strong>TransactionDefinition.ISOLATION_READ_UNCOMMITTED:</strong> 最低的隔离级别,允许读取尚未提交的数据变更,可能会导致脏读、幻读或不可重复读</li>
<li><strong>TransactionDefinition.ISOLATION_READ_COMMITTED:</strong> 允许读取并发事务已经提交的数据,可以阻止脏读,但是幻读或不可重复读仍有可能发生</li>
<li><strong>TransactionDefinition.ISOLATION_REPEATABLE_READ:</strong> 对同一字段的多次读取结果都是一致的,除非数据是被本身事务自己所修改,可以阻止脏读和不可重复读,但幻读仍有可能发生。</li>
<li><strong>TransactionDefinition.ISOLATION_SERIALIZABLE:</strong> 最高的隔离级别,完全服从 ACID 的隔离级别。所有的事务依次逐个执行,这样事务之间就完全不可能产生干扰,也就是说,该级别可以防止脏读、不可重复读以及幻读。但是这将严重影响程序的性能。通常情况下也不会用到该级别。</li>
</ul>
<h2 id="12-SpringMVC-原理了解吗"><a href="#12-SpringMVC-原理了解吗" class="headerlink" title="12. SpringMVC 原理了解吗?"></a>12. SpringMVC 原理了解吗?</h2><p> TransactionDefinition 接口中定义了五个表示隔离级别的常量:</p>
<p>过了秋招挺长一段时间了,说实话我自己也忘了如何简要概括 Spring AOP IOC 实现原理,就在网上找了一个较为简洁的答案,下面分享给各位。</p>
<p><strong>IOC:</strong> 控制反转也叫依赖注入。IOC利用java反射机制,AOP利用代理模式。IOC 概念看似很抽象,但是很容易理解。说简单点就是将对象交给容器管理,你只需要在spring配置文件中配置对应的bean以及设置相关的属性,让spring容器来生成类的实例对象以及管理对象。在spring容器启动的时候,spring会把你在配置文件中配置的bean都初始化好,然后在你需要调用的时候,就把它已经初始化好的那些bean分配给你需要调用这些bean的类。</p>
<p><strong>AOP:</strong> 面向切面编程。(Aspect-Oriented Programming) 。AOP 可以说是对 OOP 的补充和完善。OOP 引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。实现 AOP 的技术,主要分为两大类:一是采用动态代理技术,利用截取消息的方式,对该消息进行装饰,以取代原有对象行为的执行;二是采用静态织入的方式,引入特定的语法创建“方面”,从而使得编译器可以在编译期间织入有关“方面”的代码,属于静态代理。</p>
<h1 id="二-进阶篇"><a href="#二-进阶篇" class="headerlink" title="二 进阶篇"></a>二 进阶篇</h1><h2 id="1-消息队列MQ的套路"><a href="#1-消息队列MQ的套路" class="headerlink" title="1 消息队列MQ的套路"></a>1 消息队列MQ的套路</h2><p>消息队列/消息中间件应该是Java程序员必备的一个技能了,如果你之前没接触过消息队列的话,建议先去百度一下某某消息队列入门,然后花2个小时就差不多可以学会任何一种消息队列的使用了。如果说仅仅学会使用是万万不够的,在实际生产环境还要考虑消息丢失等等情况。关于消息队列面试相关的问题,推荐大家也可以看一下视频《Java工程师面试突击第1季-中华石杉老师》,如果大家没有资源的话,可以在我的公众号“Java面试通关手册”后台回复关键字“1”即可!<br>消息队列/消息中间件应该是 Java 程序员必备的一个技能了,如果你之前没接触过消息队列的话,建议先去百度一下某某消息队列入门,然后花 2 个小时就差不多可以学会任何一种消息队列的使用了。如果说仅仅学会使用是万万不够的,在实际生产环境还要考虑消息丢失等等情况。关于消息队列面试相关的问题,推荐大家也可以看一下视频《Java 工程师面试突击第 1 季-中华石杉老师》,如果大家没有资源的话,可以在我的公众号“Java 面试通关手册”后台回复关键字“1”即可!</p>
<h3 id="1-1-介绍一下消息队列MQ的应用场景-使用消息队列的好处"><a href="#1-1-介绍一下消息队列MQ的应用场景-使用消息队列的好处" class="headerlink" title="1.1 介绍一下消息队列MQ的应用场景/使用消息队列的好处"></a>1.1 介绍一下消息队列MQ的应用场景/使用消息队列的好处</h3><p>面试官一般会先问你这个问题,预热一下,看你知道消息队列不,一般在第一面的时候面试官可能只会问消息队列 MQ 的应用场景/使用消息队列的好处、使用消息队列会带来什么问题、消息队列的技术选型这几个问题,不会太深究下去,在后面的第二轮/第三轮技术面试中可能会深入问一下。</p>
<p><strong>《大型网站技术架构》第四章和第七章均有提到消息队列对应用性能及扩展性的提升。</strong></p>
<h4 id="1-通过异步处理提高系统性能"><a href="#1-通过异步处理提高系统性能" class="headerlink" title="1)通过异步处理提高系统性能"></a>1)通过异步处理提高系统性能</h4><p><img src="https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-11/%E9%80%9A%E8%BF%87%E5%BC%82%E6%AD%A5%E5%A4%84%E7%90%86%E6%8F%90%E9%AB%98%E7%B3%BB%E7%BB%9F%E6%80%A7%E8%83%BD.jpg" alt="通过异步处理提高系统性能"><br>如上图,<strong>在不使用消息队列服务器的时候,用户的请求数据直接写入数据库,在高并发的情况下数据库压力剧增,使得响应速度变慢。但是在使用消息队列之后,用户的请求数据发送给消息队列之后立即 返回,再由消息队列的消费者进程从消息队列中获取数据,异步写入数据库。由于消息队列服务器处理速度快于数据库(消息队列也比数据库有更好的伸缩性),因此响应速度得到大幅改善。</strong></p>
<p> TransactionDefinition 接口中定义了五个表示隔离级别的常量:<br>因为<strong>用户请求数据写入消息队列之后就立即返回给用户了,但是请求数据在后续的业务校验、写数据库等操作中可能失败</strong>。因此使用消息队列进行异步处理之后,需要<strong>适当修改业务流程进行配合</strong>,比如<strong>用户在提交订单之后,订单数据写入消息队列,不能立即返回用户订单提交成功,需要在消息队列的订单消费者进程真正处理完该订单之后,甚至出库后,再通过电子邮件或短信通知用户订单成功</strong>,以免交易纠纷。这就类似我们平时手机订火车票和电影票。</p>
<h4 id="2-降低系统耦合性"><a href="#2-降低系统耦合性" class="headerlink" title="2)降低系统耦合性"></a>2)降低系统耦合性</h4><p>我们知道模块分布式部署以后聚合方式通常有两种:1.<strong>分布式消息队列</strong>和 2.<strong>分布式服务</strong>。</p>
<blockquote>
<p><strong>先来简单说一下分布式服务:</strong><br>目前使用比较多的用来构建<strong>SOA(Service Oriented Architecture面向服务体系结构)</strong>的<strong>分布式服务框架</strong>是阿里巴巴开源的<strong>Dubbo</strong>。如果想深入了解Dubbo的可以看我写的关于Dubbo的这一篇文章:<strong>《高性能优秀的服务框架-dubbo介绍》</strong>:<a href="https://juejin.im/post/5acadeb1f265da2375072f9c" target="_blank" rel="noopener">https://juejin.im/post/5acadeb1f265da2375072f9c</a><br>目前使用比较多的用来构建<strong>SOA(Service Oriented Architecture 面向服务体系结构)</strong>的<strong>分布式服务框架</strong>是阿里巴巴开源的<strong>Dubbo</strong>。如果想深入了解 Dubbo 的可以看我写的关于 Dubbo 的这一篇文章:<strong>《高性能优秀的服务框架-dubbo 介绍》</strong>:<a href="https://juejin.im/post/5acadeb1f265da2375072f9c" target="_blank" rel="noopener" title="https://juejin.im/post/5acadeb1f265da2375072f9c">https://juejin.im/post/5acadeb1f265da2375072f9c</a></p>
</blockquote>
<blockquote>
<p><strong>再来谈我们的分布式消息队列:</strong><br> TransactionDefinition 接口中定义了五个表示隔离级别的常量:</p>
</blockquote>
<p>消息接受者对消息进行过滤、处理、包装后,构造成一个新的消息类型,将消息继续发送出去,等待其他消息接受者订阅该消息。因此基于事件(消息对象)驱动的业务架构可以是一系列流程。</p>
<p><strong>另外为了避免消息队列服务器宕机造成消息丢失,会将成功发送到消息队列的消息存储在消息生产者服务器上,等消息真正被消费者服务器处理后才删除消息。在消息队列服务器宕机后,生产者服务器会选择分布式消息队列服务器集群中的其他服务器发布消息。</strong></p>
<p><strong>备注:</strong> 不要认为消息队列只能利用发布-订阅模式工作,只不过在解耦这个特定业务环境下是使用发布-订阅模式的,<strong>比如在我们的 ActiveMQ 消息队列中还有点对点工作模式</strong>,具体的会在后面的文章给大家详细介绍,这一篇文章主要还是让大家对消息队列有一个更透彻的了解。</p>
<blockquote>
<p>这个问题一般会在上一个问题问完之后,紧接着被问到。“使用消息队列会带来什么问题?”这个问题要引起重视,一般我们都会考虑使用消息队列会带来的好处而忽略它带来的问题!</p>
</blockquote>
<h3 id="1-2-那么使用消息队列会带来什么问题-考虑过这些问题吗"><a href="#1-2-那么使用消息队列会带来什么问题-考虑过这些问题吗" class="headerlink" title="1.2 那么使用消息队列会带来什么问题?考虑过这些问题吗?"></a>1.2 那么使用消息队列会带来什么问题?考虑过这些问题吗?</h3><ul>
<li><strong>系统可用性降低:</strong> 系统可用性在某种程度上降低,为什么这样说呢?在加入 MQ 之前,你不用考虑消息丢失或者说 MQ 挂掉等等的情况,但是,引入 MQ 之后你就需要去考虑了!</li>
<li><strong>系统复杂性提高:</strong> 加入 MQ 之后,你需要保证消息没有被重复消费、处理消息丢失的情况、保证消息传递的顺序性等等问题!</li>
<li><strong>一致性问题:</strong> 我上面讲了消息队列可以实现异步,消息队列带来的异步确实可以提高系统响应速度。但是,万一消息的真正消费者并没有正确消费消息怎么办?这样就会导致数据不一致的情况了!</li>
</ul>
<blockquote>
<p>了解下面这个问题是为了我们更好的进行技术选型!该部分摘自:《Java工程师面试突击第1季-中华石杉老师》,如果大家没有资源的话,可以在我的公众号“Java面试通关手册”后台回复关键字“1”即可!<br>了解下面这个问题是为了我们更好的进行技术选型!该部分摘自:《Java 工程师面试突击第 1 季-中华石杉老师》,如果大家没有资源的话,可以在我的公众号“Java 面试通关手册”后台回复关键字“1”即可!</p>
</blockquote>
<h3 id="1-3-介绍一下你知道哪几种消息队列-该如何选择呢"><a href="#1-3-介绍一下你知道哪几种消息队列-该如何选择呢" class="headerlink" title="1.3 介绍一下你知道哪几种消息队列,该如何选择呢?"></a>1.3 介绍一下你知道哪几种消息队列,该如何选择呢?</h3><table>
<thead>
<tr>
<th align="left">特性</th>
<th align="right">ActiveMQ</th>
<th align="right">RabbitMQ</th>
<th align="right">RocketMQ</th>
<th align="right">Kafka</th>
</tr>
</thead>
<tbody><tr>
<td align="left">单机吞吐量</td>
<td align="right">万级,吞吐量比 RocketMQ 和 Kafka 要低了一个数量级</td>
<td align="right">万级,吞吐量比 RocketMQ 和 Kafka 要低了一个数量级</td>
<td align="right">10 万级,RocketMQ 也是可以支撑高吞吐的一种 MQ</td>
<td align="right">10 万级别,这是 kafka 最大的优点,就是吞吐量高。一般配合大数据类的系统来进行实时数据计算、日志采集等场景</td>
</tr>
<tr>
<td align="left">topic 数量对吞吐量的影响</td>
<td align="right"></td>
<td align="right"></td>
<td align="right">topic 可以达到几百,几千个的级别,吞吐量会有较小幅度的下降这是 RocketMQ 的一大优势,在同等机器下,可以支撑大量的 topic</td>
<td align="right">topic 从几十个到几百个的时候,吞吐量会大幅度下降。所以在同等机器下,kafka 尽量保证 topic 数量不要过多。如果要支撑大规模 topic,需要增加更多的机器资源</td>
</tr>
<tr>
<td align="left">可用性</td>
<td align="right">高,基于主从架构实现高可用性</td>
<td align="right">高,基于主从架构实现高可用性</td>
<td align="right">非常高,分布式架构</td>
<td align="right">非常高,kafka 是分布式的,一个数据多个副本,少数机器宕机,不会丢失数据,不会导致不可用</td>
</tr>
<tr>
<td align="left">消息可靠性</td>
<td align="right">有较低的概率丢失数据</td>
<td align="right"></td>
<td align="right">经过参数优化配置,可以做到 0 丢失</td>
<td align="right">经过参数优化配置,消息可以做到 0 丢失</td>
</tr>
<tr>
<td align="left">时效性</td>
<td align="right">ms 级</td>
<td align="right">微秒级,这是 rabbitmq 的一大特点,延迟是最低的</td>
<td align="right">ms 级</td>
<td align="right">延迟在 ms 级以内</td>
</tr>
<tr>
<td align="left">功能支持</td>
<td align="right">MQ 领域的功能极其完备</td>
<td align="right">基于 erlang 开发,所以并发能力很强,性能极其好,延时很低</td>
<td align="right">MQ 功能较为完善,还是分布式的,扩展性好</td>
<td align="right">功能较为简单,主要支持简单的 MQ 功能,在大数据领域的实时计算以及日志采集被大规模使用,是事实上的标准</td>
</tr>
<tr>
<td align="left">优劣势总结</td>
<td align="right">非常成熟,功能强大,在业内大量的公司以及项目中都有应用。偶尔会有较低概率丢失消息,而且现在社区以及国内应用都越来越少,官方社区现在对 ActiveMQ 5.x 维护越来越少,几个月才发布一个版本而且确实主要是基于解耦和异步来用的,较少在大规模吞吐的场景中使用</td>
<td align="right">erlang 语言开发,性能极其好,延时很低;吞吐量到万级,MQ 功能比较完备而且开源提供的管理界面非常棒,用起来很好用。社区相对比较活跃,几乎每个月都发布几个版本分在国内一些互联网公司近几年用 rabbitmq 也比较多一些但是问题也是显而易见的,RabbitMQ 确实吞吐量会低一些,这是因为他做的实现机制比较重。而且 erlang 开发,国内有几个公司有实力做 erlang 源码级别的研究和定制?如果说你没这个实力的话,确实偶尔会有一些问题,你很难去看懂源码,你公司对这个东西的掌控很弱,基本职能依赖于开源社区的快速维护和修复 bug。而且 rabbitmq 集群动态扩展会很麻烦,不过这个我觉得还好。其实主要是 erlang 语言本身带来的问题。很难读源码,很难定制和掌控。</td>
<td align="right">接口简单易用,而且毕竟在阿里大规模应用过,有阿里品牌保障。日处理消息上百亿之多,可以做到大规模吞吐,性能也非常好,分布式扩展也很方便,社区维护还可以,可靠性和可用性都是 ok 的,还可以支撑大规模的 topic 数量,支持复杂 MQ 业务场景。而且一个很大的优势在于,阿里出品都是 java 系的,我们可以自己阅读源码,定制自己公司的 MQ,可以掌控。社区活跃度相对较为一般,不过也还可以,文档相对来说简单一些,然后接口这块不是按照标准 JMS 规范走的有些系统要迁移需要修改大量代码。还有就是阿里出台的技术,你得做好这个技术万一被抛弃,社区黄掉的风险,那如果你们公司有技术实力我觉得用 RocketMQ 挺好的</td>
<td align="right">kafka 的特点其实很明显,就是仅仅提供较少的核心功能,但是提供超高的吞吐量,ms 级的延迟,极高的可用性以及可靠性,而且分布式可以任意扩展。同时 kafka 最好是支撑较少的 topic 数量即可,保证其超高吞吐量。而且 kafka 唯一的一点劣势是有可能消息重复消费,那么对数据准确性会造成极其轻微的影响,在大数据领域中以及日志采集中,这点轻微影响可以忽略这个特性天然适合大数据实时计算以及日志收集。</td>
</tr>
</tbody></table>
<h2 id="2-谈谈-InnoDB-和-MyIsam-两者的区别"><a href="#2-谈谈-InnoDB-和-MyIsam-两者的区别" class="headerlink" title="2 谈谈 InnoDB 和 MyIsam 两者的区别"></a>2 谈谈 InnoDB 和 MyIsam 两者的区别</h2><h3 id="2-1-两者的对比"><a href="#2-1-两者的对比" class="headerlink" title="2.1 两者的对比"></a>2.1 两者的对比</h3><ol>
<li><p><strong>count运算上的区别:</strong> 因为MyISAM缓存有表meta-data(行数等),因此在做COUNT(*)时对于一个结构很好的查询是不需要消耗多少资源的。而对于InnoDB来说,则没有这种缓存</p>
</li>
<li><p><strong>是否支持事务和崩溃后的安全恢复:</strong> MyISAM 强调的是性能,每次查询具有原子性,其执行速度比InnoDB类型更快,但是不提供事务支持。但是 InnoDB 提供事务支持,外部键等高级数据库功能。 具有事务(commit)、回滚(rollback)和崩溃修复能力(crash recovery capabilities)的事务安全(transaction-safe (ACID compliant))型表。</p>
</li>
<li><p><strong>是否支持外键:</strong> MyISAM不支持,而InnoDB支持。</p>
</li>
<li><p><strong>count 运算上的区别:</strong> 因为 MyISAM 缓存有表 meta-data(行数等),因此在做 COUNT(*)时对于一个结构很好的查询是不需要消耗多少资源的。而对于 InnoDB 来说,则没有这种缓存</p>
</li>
<li><p><strong>是否支持事务和崩溃后的安全恢复:</strong> MyISAM 强调的是性能,每次查询具有原子性,其执行速度比 InnoDB 类型更快,但是不提供事务支持。但是 InnoDB 提供事务支持,外部键等高级数据库功能。 具有事务(commit)、回滚(rollback)和崩溃修复能力(crash recovery capabilities)的事务安全(transaction-safe (ACID compliant))型表。</p>
</li>
<li><p><strong>是否支持外键:</strong> MyISAM 不支持,而 InnoDB 支持。</p>
</li>
</ol>
<h3 id="2-2-关于两者的总结"><a href="#2-2-关于两者的总结" class="headerlink" title="2.2 关于两者的总结"></a>2.2 关于两者的总结</h3><p>MyISAM更适合读密集的表,而InnoDB更适合写密集的表。 在数据库做主从分离的情况下,经常选择MyISAM作为主库的存储引擎。</p>
<p>一般来说,如果需要事务支持,并且有较高的并发读取频率(MyISAM的表锁的粒度太大,所以当该表写并发量较高时,要等待的查询就会很多了),InnoDB是不错的选择。如果你的数据量很大(MyISAM支持压缩特性可以减少磁盘的空间占用),而且不需要支持事务时,MyISAM是最好的选择。<br>MyISAM 更适合读密集的表,而 InnoDB 更适合写密集的表。 在数据库做主从分离的情况下,经常选择 MyISAM 作为主库的存储引擎。</p>
<p>一般来说,如果需要事务支持,并且有较高的并发读取频率(MyISAM 的表锁的粒度太大,所以当该表写并发量较高时,要等待的查询就会很多了),InnoDB 是不错的选择。如果你的数据量很大(MyISAM 支持压缩特性可以减少磁盘的空间占用),而且不需要支持事务时,MyISAM 是最好的选择。</p>
<h2 id="3-聊聊-Java-中的集合吧"><a href="#3-聊聊-Java-中的集合吧" class="headerlink" title="3 聊聊 Java 中的集合吧!"></a>3 聊聊 Java 中的集合吧!</h2><h3 id="3-1-Arraylist-与-LinkedList-有什么不同-注意加上从数据结构分析的内容"><a href="#3-1-Arraylist-与-LinkedList-有什么不同-注意加上从数据结构分析的内容" class="headerlink" title="3.1 Arraylist 与 LinkedList 有什么不同?(注意加上从数据结构分析的内容)"></a>3.1 Arraylist 与 LinkedList 有什么不同?(注意加上从数据结构分析的内容)</h3><ul>
<li><strong>1. 是否保证线程安全:</strong> ArrayList 和 LinkedList 都是不同步的,也就是不保证线程安全;</li>
<li><strong>2. 底层数据结构:</strong> Arraylist 底层使用的是Object数组;LinkedList 底层使用的是双向链表数据结构(注意双向链表和双向循环链表的区别:);</li>
<li><strong>3. 插入和删除是否受元素位置的影响:</strong> ① <strong>ArrayList 采用数组存储,所以插入和删除元素的时间复杂度受元素位置的影响。</strong> 比如:执行<code>add(E e)</code>方法的时候, ArrayList 会默认在将指定的元素追加到此列表的末尾,这种情况时间复杂度就是O(1)。但是如果要在指定位置 i 插入和删除元素的话(<code>add(int index, E element)</code>)时间复杂度就为 O(n-i)。因为在进行上述操作的时候集合中第 i 和第 i 个元素之后的(n-i)个元素都要执行向后位/向前移一位的操作。 ② <strong>LinkedList 采用链表存储,所以插入,删除元素时间复杂度不受元素位置的影响,都是近似 O(1) 而数组为近似 O(n) 。</strong></li>
<li><strong>4. 是否支持快速随机访问:</strong> LinkedList 不支持高效的随机元素访问,而 ArrayList 支持。快速随机访问就是通过元素的序号快速获取元素对象(对应于<code>get(int index)</code>方法)。</li>
<li><strong>5. 内存空间占用:</strong> ArrayList的空 间浪费主要体现在在list列表的结尾会预留一定的容量空间,而LinkedList的空间花费则体现在它的每一个元素都需要消耗比ArrayList更多的空间(因为要存放直接后继和直接前驱以及数据)。 </li>
<li><strong>2. 底层数据结构:</strong> Arraylist 底层使用的是 Object 数组;LinkedList 底层使用的是双向链表数据结构(注意双向链表和双向循环链表的区别:);</li>
<li><strong>3. 插入和删除是否受元素位置的影响:</strong> ① <strong>ArrayList 采用数组存储,所以插入和删除元素的时间复杂度受元素位置的影响。</strong> 比如:执行<code>add(E e)</code>方法的时候, ArrayList 会默认在将指定的元素追加到此列表的末尾,这种情况时间复杂度就是 O(1)。但是如果要在指定位置 i 插入和删除元素的话(<code>add(int index, E element)</code>)时间复杂度就为 O(n-i)。因为在进行上述操作的时候集合中第 i 和第 i 个元素之后的(n-i)个元素都要执行向后位/向前移一位的操作。 ② <strong>LinkedList 采用链表存储,所以插入,删除元素时间复杂度不受元素位置的影响,都是近似 O(1) 而数组为近似 O(n) 。</strong></li>
<li><strong>4. 是否支持快速随机访问:</strong> LinkedList 不支持高效的随机元素访问,而 ArrayList 支持。快速随机访问就是通过元素的序号快速获取元素对象(对应于<code>get(int index)</code>方法)。</li>
<li><strong>5. 内存空间占用:</strong> ArrayList 的空 间浪费主要体现在在 list 列表的结尾会预留一定的容量空间,而 LinkedList 的空间花费则体现在它的每一个元素都需要消耗比 ArrayList 更多的空间(因为要存放直接后继和直接前驱以及数据)。</li>
</ul>
<p><strong>补充内容:RandomAccess接口</strong></p>
<p>查看源码我们发现实际上 RandomAccess 接口中什么都没有定义。所以,在我看来 RandomAccess 接口不过是一个标识罢了。标识什么? 标识实现这个接口的类具有随机访问功能。</p>
<p>在 binarySearch() 方法中,它要判断传入的 list 是否RamdomAccess的实例,如果是,调用 indexedBinarySearch() 方法,如果不是,那么调用 iteratorBinarySearch() 方法</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="JAVA"><figure class="iseeu 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="keyword">public</span> <span class="keyword">static</span> <T></span><br><span class="line"><span class="keyword">public</span> <span class="class"><span class="keyword">interface</span> <span class="title">RandomAccess</span> </span>{</span><br><span class="line"> }</span><br></pre></td></tr></table></figure></div>
<p>ArraysList 实现了 RandomAccess 接口, 而 LinkedList 没有实现。为什么呢?我觉得还是和底层数据结构有关!ArraysList 底层是数组,而 LinkedList 底层是链表。数组天然支持随机访问,时间复杂度为 O(1) ,所以称为快速随机访问。链表需要遍历到特定位置才能访问特定位置的元素,时间复杂度为 O(n) ,所以不支持快速随机访问。,ArraysList 实现了 RandomAccess 接口,就表明了他具有快速随机访问功能。 RandomAccess 接口只是标识,并不是说 ArraysList 实现 RandomAccess 接口才具有快速随机访问功能的!</p>
<p><strong>下面再总结一下 list 的遍历方式选择:</strong></p>
<ul>
<li>实现了RandomAccess接口的list,优先选择普通for循环 ,其次foreach,</li>
<li>未实现RandomAccess接口的ist, 优先选择iterator遍历(foreach遍历底层也是通过iterator实现的),大size的数据,千万不要使用普通for循环</li>
</ul>
<blockquote>
<p>Java 中的集合这类问题几乎是面试必问的,问到这类问题的时候,HashMap 又是几乎必问的问题,所以大家一定要引起重视!</p>
</blockquote>
<h3 id="3-2-HashMap的底层实现"><a href="#3-2-HashMap的底层实现" class="headerlink" title="3.2 HashMap的底层实现"></a>3.2 HashMap的底层实现</h3><h4 id="1-JDK1-8-之前"><a href="#1-JDK1-8-之前" class="headerlink" title="1)JDK1.8 之前"></a>1)JDK1.8 之前</h4><p>JDK1.8 之前 HashMap 底层是 <strong>数组和链表</strong> 结合在一起使用也就是 <strong>链表散列</strong>。<strong>HashMap 通过 key 的 hashCode 经过扰动函数处理过后得到 hash 值,然后通过 <code>(n - 1) & hash</code> 判断当前元素存放的位置(这里的 n 指的时数组的长度),如果当前位置存在元素的话,就判断该元素与要存入的元素的 hash 值以及 key 是否相同,如果相同的话,直接覆盖,不相同就通过拉链法解决冲突。</strong></p>
<p><strong>所谓扰动函数指的就是 HashMap 的 hash 方法。使用 hash 方法也就是扰动函数是为了防止一些实现比较差的 hashCode() 方法 换句话说使用扰动函数之后可以减少碰撞。</strong></p>
<p><strong>JDK 1.8 HashMap 的 hash 方法源码:</strong></p>
<p>JDK 1.8 的 hash方法 相比于 JDK 1.7 hash 方法更加简化,但是原理不变。</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="JAVA"><figure class="iseeu 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"> <span class="function"><span class="keyword">static</span> <span class="keyword">final</span> <span class="keyword">int</span> <span class="title">hash</span><span class="params">(Object key)</span> </span>{</span><br><span class="line">JDK <span class="number">1.8</span> 的 hash方法 相比于 JDK <span class="number">1.7</span> hash 方法更加简化,但是原理</span><br><span class="line"> <span class="keyword">return</span> (key == <span class="keyword">null</span>) ? <span class="number">0</span> : (h = key.hashCode()) ^ (h >>> <span class="number">16</span>);</span><br><span class="line"> }</span><br></pre></td></tr></table></figure></div>
<p>对比一下 JDK1.7 的 HashMap 的 hash 方法源码.</p>
<p><img src="https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-6/jdk1.8%E4%B9%8B%E5%89%8D%E7%9A%84%E5%86%85%E9%83%A8%E7%BB%93%E6%9E%84-HashMap.jpg" alt="jdk1.8之前的内部结构-HashMap"></p>
<h4 id="2-JDK1-8之后"><a href="#2-JDK1-8之后" class="headerlink" title="2)JDK1.8之后"></a>2)JDK1.8之后</h4><p>相比于之前的版本, JDK1.8之后在解决哈希冲突时有了较大的变化,当链表长度大于阈值(默认为8)时,将链表转化为红黑树,以减少搜索时间。<br>相比于之前的版本, JDK1.8 之后在解决哈希冲突时有了较大的变化,当链表长度大于阈值(默认为 8)时,将链表转化为红黑树,以减少搜索时间。</p>
<p><img src="https://my-blog-to-use.oss-cn-beijing.aliyuncs.com/2019-6/JDK1.8%E4%B9%8B%E5%90%8E%E7%9A%84HashMap%E5%BA%95%E5%B1%82%E6%95%B0%E6%8D%AE%E7%BB%93%E6%9E%84.jpg" alt="jdk1.8之后的内部结构-HashMap"></p>
<p>TreeMap、TreeSet以及JDK1.8之后的HashMap底层都用到了红黑树。红黑树就是为了解决二叉查找树的缺陷,因为二叉查找树在某些情况下会退化成一个线性结构。<br>TreeMap、TreeSet 以及 JDK1.8 之后的 HashMap 底层都用到了红黑树。红黑树就是为了解决二叉查找树的缺陷,因为二叉查找树在某些情况下会退化成一个线性结构。</p>
<blockquote>
<p>问完 HashMap 的底层原理之后,面试官可能就会紧接着问你 HashMap 底层数据结构相关的问题!</p>
</blockquote>
<h3 id="3-3-既然谈到了红黑树-你给我手绘一个出来吧-然后简单讲一下自己对于红黑树的理解"><a href="#3-3-既然谈到了红黑树-你给我手绘一个出来吧-然后简单讲一下自己对于红黑树的理解" class="headerlink" title="3.3 既然谈到了红黑树,你给我手绘一个出来吧,然后简单讲一下自己对于红黑树的理解"></a>3.3 既然谈到了红黑树,你给我手绘一个出来吧,然后简单讲一下自己对于红黑树的理解</h3><p><img src="https://user-gold-cdn.xitu.io/2018/11/14/16711ac29c138cba?w=851&h=614&f=jpeg&s=34458" alt="红黑树"></p>
<p><strong>红黑树特点:</strong></p>
<ol>
<li>每个节点非红即黑;</li>
<li>根节点总是黑色的;</li>
<li>每个叶子节点都是黑色的空节点(NIL 节点);</li>
<li>如果节点是红色的,则它的子节点必须是黑色的(反之不一定);</li>
<li>从根节点到叶节点或空子节点的每条路径,必须包含相同数目的黑色节点(即相同的黑色高度)</li>
</ol>
<p><strong>红黑树的应用:</strong></p>
<p>TreeMap、TreeSet 以及 JDK1.8 之后的 HashMap 底层都用到了红黑树。</p>
<p><strong>为什么要用红黑树</strong></p>
<p>简单来说红黑树就是为了解决二叉查找树的缺陷,因为二叉查找树在某些情况下会退化成一个线性结构。</p>
<h3 id="3-4-红黑树这么优秀-为何不直接使用红黑树得了"><a href="#3-4-红黑树这么优秀-为何不直接使用红黑树得了" class="headerlink" title="3.4 红黑树这么优秀,为何不直接使用红黑树得了?"></a>3.4 红黑树这么优秀,为何不直接使用红黑树得了?</h3><p>说一下自己对于这个问题的看法:我们知道红黑树属于(自)平衡二叉树,但是为了保持“平衡”是需要付出代价的,红黑树在插入新数据后可能需要通过左旋,右旋、变色这些操作来保持平衡,这费事啊。你说说我们引入红黑树就是为了查找数据快,如果链表长度很短的话,根本不需要引入红黑树的,你引入之后还要付出代价维持它的平衡。但是链表过长就不一样了。至于为什么选 8 这个值呢?通过概率统计所得,这个值是综合查询成本和新增元素成本得出的最好的一个值。</p>
<h3 id="3-5-HashMap-和-Hashtable-的区别-HashSet-和-HashMap-区别"><a href="#3-5-HashMap-和-Hashtable-的区别-HashSet-和-HashMap-区别" class="headerlink" title="3.5 HashMap 和 Hashtable 的区别/HashSet 和 HashMap 区别"></a>3.5 HashMap 和 Hashtable 的区别/HashSet 和 HashMap 区别</h3><p><strong>HashMap 和 Hashtable 的区别</strong></p>
<ol>
<li><strong>线程是否安全:</strong> HashMap 是非线程安全的,Hashtable 是线程安全的;Hashtable 内部的方法基本都经过 <code>synchronized</code> 修饰。(如果你要保证线程安全的话就使用 ConcurrentHashMap 吧!);</li>
<li><strong>效率:</strong> 因为线程安全的问题,HashMap 要比 Hashtable 效率高一点。另外,Hashtable 基本被淘汰,不要在代码中使用它;</li>
<li><strong>对Null key 和Null value的支持:</strong> HashMap 中,null 可以作为键,这样的键只有一个,可以有一个或多个键所对应的值为 null。但是在 Hashtable 中 put 进的键值只要有一个 null,直接抛出 NullPointerException。</li>
<li><strong>初始容量大小和每次扩充容量大小的不同 :</strong> ①创建时如果不指定容量初始值,Hashtable 默认的初始大小为11,之后每次扩充,容量变为原来的2n+1。HashMap 默认的初始化大小为16。之后每次扩充,容量变为原来的2倍。②创建时如果给定了容量初始值,那么 Hashtable 会直接使用你给定的大小,而 HashMap 会将其扩充为2的幂次方大小(HashMap 中的<code>tableSizeFor()</code>方法保证,下面给出了源代码)。也就是说 HashMap 总是使用2的幂作为哈希表的大小,后面会介绍到为什么是2的幂次方。</li>
<li><strong>底层数据结构:</strong> JDK1.8 以后的 HashMap 在解决哈希冲突时有了较大的变化,当链表长度大于阈值(默认为8)时,将链表转化为红黑树,以减少搜索时间。Hashtable 没有这样的机制。</li>
</ol>
<h1 id="三-终结篇"><a href="#三-终结篇" class="headerlink" title="三 终结篇"></a>三 终结篇</h1><h2 id="1-Object类有哪些方法"><a href="#1-Object类有哪些方法" class="headerlink" title="1. Object类有哪些方法?"></a>1. Object类有哪些方法?</h2><p>这个问题,面试中经常出现。我觉得不论是出于应付面试还是说更好地掌握Java这门编程语言,大家都要掌握!</p>
<h3 id="1-1-Object类的常见方法总结"><a href="#1-1-Object类的常见方法总结" class="headerlink" title="1.1 Object类的常见方法总结"></a>1.1 Object类的常见方法总结</h3><p>Object类是一个特殊的类,是所有类的父类。它主要提供了以下11个方法:</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="JAVA"><figure class="iseeu highlight /java"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">protected</span> <span class="keyword">void</span> <span class="title">finalize</span><span class="params">()</span> <span class="keyword">throws</span> Throwable </span>{ }<span class="comment">//实例被垃圾回收器回收</span></span><br></pre></td></tr></table></figure></div>
<blockquote>
<p>问完上面这个问题之后,面试官很可能紧接着就会问你“hashCode与equals”相关的问题。</p>
</blockquote>
<h3 id="1-2-hashCode与equals"><a href="#1-2-hashCode与equals" class="headerlink" title="1.2 hashCode与equals"></a>1.2 hashCode与equals</h3><p>面试官可能会问你:“你重写过 hashcode 和 equals 么,为什么重写 equals 时必须重写 hashCode 方法?”</p>
<h4 id="1-2-1-hashCode-介绍"><a href="#1-2-1-hashCode-介绍" class="headerlink" title="1.2.1 hashCode()介绍"></a>1.2.1 hashCode()介绍</h4><p>hashCode() 的作用是获取哈希码,也称为散列码;它实际上是返回一个 int 整数。这个哈希码的作用是确定该对象在哈希表中的索引位置。hashCode() 定义在 JDK 的 Object.java 中,这就意味着 Java 中的任何类都包含有 hashCode() 函数。另外需要注意的是: Object 的 hashcode 方法是本地方法,也就是用 c 语言或 c++ 实现的,该方法通常用来将对象的 内存地址 转换为整数之后返回。</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="JAVA"><figure class="iseeu highlight /java"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="function"><span class="keyword">public</span> <span class="keyword">native</span> <span class="keyword">int</span> <span class="title">hashCode</span><span class="params">()</span></span>;</span><br></pre></td></tr></table></figure></div>
<p>散列表存储的是键值对(key-value),它的特点是:能根据“键”快速的检索出对应的“值”。这其中就利用到了散列码!(可以快速找到所需要的对象)</p>
<h4 id="1-2-2-为什么要有-hashCode"><a href="#1-2-2-为什么要有-hashCode" class="headerlink" title="1.2.2 为什么要有 hashCode"></a>1.2.2 为什么要有 hashCode</h4><p><strong>我们以“HashSet如何检查重复”为例子来说明为什么要有hashCode:</strong></p>
<p>当你把对象加入HashSet时,HashSet会先计算对象的hashcode值来判断对象加入的位置,同时也会与其他已经加入的对象的hashcode值作比较,如果没有相符的hashcode,HashSet会假设对象没有重复出现。但是如果发现有相同hashcode值的对象,这时会调用equals()方法来检查hashcode相等的对象是否真的相同。如果两者相同,HashSet就不会让其加入操作成功。如果不同的话,就会重新散列到其他位置。(摘自我的Java启蒙书《Head fist java》第二版)。这样我们就大大减少了equals的次数,相应就大大提高了执行速度。</p>
<h4 id="1-2-3-hashCode-与equals-的相关规定"><a href="#1-2-3-hashCode-与equals-的相关规定" class="headerlink" title="1.2.3 hashCode()与equals()的相关规定"></a>1.2.3 hashCode()与equals()的相关规定</h4><ol>
<li>如果两个对象相等,则hashcode一定也是相同的</li>
<li>两个对象相等,对两个对象分别调用equals方法都返回true</li>
<li>两个对象有相同的hashcode值,它们也不一定是相等的</li>
<li><strong>因此,equals方法被覆盖过,则hashCode方法也必须被覆盖</strong></li>
<li>hashCode()的默认行为是对堆上的对象产生独特值。如果没有重写hashCode(),则该class的两个对象无论如何都不会相等(即使这两个对象指向相同的数据)</li>
</ol>
<h4 id="1-2-4-为什么两个对象有相同的hashcode值-它们也不一定是相等的"><a href="#1-2-4-为什么两个对象有相同的hashcode值-它们也不一定是相等的" class="headerlink" title="1.2.4 为什么两个对象有相同的hashcode值,它们也不一定是相等的?"></a>1.2.4 为什么两个对象有相同的hashcode值,它们也不一定是相等的?</h4><p>在这里解释一位小伙伴的问题。以下内容摘自《Head Fisrt Java》。</p>
<p>因为hashCode() 所使用的杂凑算法也许刚好会让多个对象传回相同的杂凑值。越糟糕的杂凑算法越容易碰撞,但这也与数据值域分布的特性有关(所谓碰撞也就是指的是不同的对象得到相同的 hashCode)。 </p>
<p>我们刚刚也提到了 HashSet,如果 HashSet 在对比的时候,同样的 hashcode 有多个对象,它会使用 equals() 来判断是否真的相同。也就是说 hashcode 只是用来缩小查找成本。</p>
<blockquote>
<p>==与 equals 的对比也是比较常问的基础问题之一!</p>
</blockquote>
<h3 id="1-3-与equals"><a href="#1-3-与equals" class="headerlink" title="1.3 ==与equals"></a>1.3 ==与equals</h3><p><strong>==</strong> : 它的作用是判断两个对象的地址是不是相等。即,判断两个对象是不是同一个对象。(基本数据类型==比较的是值,引用数据类型==比较的是内存地址)</p>
<p><strong>equals()</strong> : 它的作用也是判断两个对象是否相等。但它一般有两种使用情况:</p>
<ul>
<li>情况1:类没有覆盖equals()方法。则通过equals()比较该类的两个对象时,等价于通过“==”比较这两个对象。</li>
<li>情况2:类覆盖了equals()方法。一般,我们都覆盖equals()方法来两个对象的内容相等;若它们的内容相等,则返回true(即,认为这两个对象相等)。</li>
</ul>
<h2 id="2-ConcurrentHashMap-相关问题"><a href="#2-ConcurrentHashMap-相关问题" class="headerlink" title="2 ConcurrentHashMap 相关问题"></a>2 ConcurrentHashMap 相关问题</h2><h3 id="2-1-ConcurrentHashMap-和-Hashtable-的区别"><a href="#2-1-ConcurrentHashMap-和-Hashtable-的区别" class="headerlink" title="2.1 ConcurrentHashMap 和 Hashtable 的区别"></a>2.1 ConcurrentHashMap 和 Hashtable 的区别</h3><p>ConcurrentHashMap 和 Hashtable 的区别主要体现在实现线程安全的方式上不同。</p>
<ul>
<li><strong>底层数据结构:</strong> JDK1.7的 ConcurrentHashMap 底层采用 <strong>分段的数组+链表</strong> 实现,JDK1.8 采用的数据结构跟HashMap1.8的结构一样,数组+链表/红黑二叉树。Hashtable 和 JDK1.8 之前的 HashMap 的底层数据结构类似都是采用 <strong>数组+链表</strong> 的形式,数组是 HashMap 的主体,链表则是主要为了解决哈希冲突而存在的;</li>
<li><strong>实现线程安全的方式(重要):</strong> ① <strong>在JDK1.7的时候,ConcurrentHashMap(分段锁)</strong> 对整个桶数组进行了分割分段(Segment),每一把锁只锁容器其中一部分数据,多线程访问容器里不同数据段的数据,就不会存在锁竞争,提高并发访问率。(默认分配16个Segment,比Hashtable效率提高16倍。) <strong>到了 JDK1.8 的时候已经摒弃了Segment的概念,而是直接用 Node 数组+链表+红黑树的数据结构来实现,并发控制使用 synchronized 和 CAS 来操作。(JDK1.6以后 对 synchronized锁做了很多优化)</strong> 整个看起来就像是优化过且线程安全的 HashMap,虽然在JDK1.8中还能看到 Segment 的数据结构,但是已经简化了属性,只是为了兼容旧版本;② <strong>Hashtable(同一把锁)</strong>:使用 synchronized 来保证线程安全,效率非常低下。当一个线程访问同步方法时,其他线程也访问同步方法,可能会进入阻塞或轮询状态,如使用 put 添加元素,另一个线程不能使用 put 添加元素,也不能使用 get,竞争会越来越激烈效率越低。</li>
<li><strong>底层数据结构:</strong> JDK1.7 的 ConcurrentHashMap 底层采用 <strong>分段的数组+链表</strong> 实现,JDK1.8 采用的数据结构跟 HashMap1.8 的结构一样,数组+链表/红黑二叉树。Hashtable 和 JDK1.8 之前的 HashMap 的底层数据结构类似都是采用 <strong>数组+链表</strong> 的形式,数组是 HashMap 的主体,链表则是主要为了解决哈希冲突而存在的;</li>
<li><strong>实现线程安全的方式(重要):</strong> ① <strong>在 JDK1.7 的时候,ConcurrentHashMap(分段锁)</strong> 对整个桶数组进行了分割分段(Segment),每一把锁只锁容器其中一部分数据,多线程访问容器里不同数据段的数据,就不会存在锁竞争,提高并发访问率。(默认分配 16 个 Segment,比 Hashtable 效率提高 16 倍。) <strong>到了 JDK1.8 的时候已经摒弃了 Segment 的概念,而是直接用 Node 数组+链表+红黑树的数据结构来实现,并发控制使用 synchronized 和 CAS 来操作。(JDK1.6 以后 对 synchronized 锁做了很多优化)</strong> 整个看起来就像是优化过且线程安全的 HashMap,虽然在 JDK1.8 中还能看到 Segment 的数据结构,但是已经简化了属性,只是为了兼容旧版本;② <strong>Hashtable(同一把锁)</strong>:使用 synchronized 来保证线程安全,效率非常低下。当一个线程访问同步方法时,其他线程也访问同步方法,可能会进入阻塞或轮询状态,如使用 put 添加元素,另一个线程不能使用 put 添加元素,也不能使用 get,竞争会越来越激烈效率越低。</li>
</ul>
<p><strong>两者的对比图:</strong> </p>
<p>图片来源:<a href="http://www.cnblogs.com/chengxiao/p/6842045.html" target="_blank" rel="noopener">http://www.cnblogs.com/chengxiao/p/6842045.html</a></p>
<p>Hashtable:<br><img src="http://my-blog-to-use.oss-cn-beijing.aliyuncs.com/18-8-22/50656681.jpg" alt=""></p>
<p>JDK1.7的ConcurrentHashMap:</p>
<p><img src="http://my-blog-to-use.oss-cn-beijing.aliyuncs.com/18-8-22/33120488.jpg" alt=""><br>JDK1.8 的 ConcurrentHashMap(TreeBin: 红黑二叉树节点<br>Node: 链表节点):<br><img src="http://my-blog-to-use.oss-cn-beijing.aliyuncs.com/18-8-22/97739220.jpg" alt=""></p>
<h3 id="2-2-ConcurrentHashMap线程安全的具体实现方式-底层具体实现"><a href="#2-2-ConcurrentHashMap线程安全的具体实现方式-底层具体实现" class="headerlink" title="2.2 ConcurrentHashMap线程安全的具体实现方式/底层具体实现"></a>2.2 ConcurrentHashMap线程安全的具体实现方式/底层具体实现</h3><h3 id="2-2-ConcurrentHashMap-线程安全的具体实现方式-底层具体实现"><a href="#2-2-ConcurrentHashMap-线程安全的具体实现方式-底层具体实现" class="headerlink" title="2.2 ConcurrentHashMap 线程安全的具体实现方式/底层具体实现"></a>2.2 ConcurrentHashMap 线程安全的具体实现方式/底层具体实现</h3><h4 id="JDK1-7-上面有示意图"><a href="#JDK1-7-上面有示意图" class="headerlink" title="JDK1.7(上面有示意图)"></a>JDK1.7(上面有示意图)</h4><p>一个 ConcurrentHashMap 里包含一个 Segment 数组。Segment 的结构和HashMap类似,是一种数组和链表结构,一个 Segment 包含一个 HashEntry 数组,每个 HashEntry 是一个链表结构的元素,每个 Segment 守护着一个HashEntry数组里的元素,当对 HashEntry 数组的数据进行修改时,必须首先获得对应的 Segment的锁。<br>一个 ConcurrentHashMap 里包含一个 Segment 数组。Segment 的结构和 HashMap 类似,是一种数组和链表结构,一个 Segment 包含一个 HashEntry 数组,每个 HashEntry 是一个链表结构的元素,每个 Segment 守护着一个 HashEntry 数组里的元素,当对 HashEntry 数组的数据进行修改时,必须首先获得对应的 Segment 的锁。</p>
<h4 id="JDK1-8-上面有示意图"><a href="#JDK1-8-上面有示意图" class="headerlink" title="JDK1.8(上面有示意图)"></a>JDK1.8(上面有示意图)</h4><p>ConcurrentHashMap取消了Segment分段锁,采用CAS和synchronized来保证并发安全。数据结构跟HashMap1.8的结构类似,数组+链表/红黑二叉树。<br>ConcurrentHashMap 取消了 Segment 分段锁,采用 CAS 和 synchronized 来保证并发安全。数据结构跟 HashMap1.8 的结构类似,数组+链表/红黑二叉树。</p>
<p>synchronized只锁定当前链表或红黑二叉树的首节点,这样只要hash不冲突,就不会产生并发,效率又提升N倍。<br>synchronized 只锁定当前链表或红黑二叉树的首节点,这样只要 hash 不冲突,就不会产生并发,效率又提升 N 倍。</p>
<h2 id="3-谈谈-synchronized-和-ReentrantLock-的区别"><a href="#3-谈谈-synchronized-和-ReentrantLock-的区别" class="headerlink" title="3 谈谈 synchronized 和 ReentrantLock 的区别"></a>3 谈谈 synchronized 和 ReentrantLock 的区别</h2><p><strong>① 两者都是可重入锁</strong></p>
<p>两者都是可重入锁。“可重入锁”概念是:自己可以再次获取自己的内部锁。比如一个线程获得了某个对象的锁,此时这个对象锁还没有释放,当其再次想要获取这个对象的锁的时候还是可以获取的,如果不可锁重入的话,就会造成死锁。同一个线程每次获取锁,锁的计数器都自增1,所以要等到锁的计数器下降为0时才能释放锁。<br>两者都是可重入锁。“可重入锁”概念是:自己可以再次获取自己的内部锁。比如一个线程获得了某个对象的锁,此时这个对象锁还没有释放,当其再次想要获取这个对象的锁的时候还是可以获取的,如果不可锁重入的话,就会造成死锁。同一个线程每次获取锁,锁的计数器都自增 1,所以要等到锁的计数器下降为 0 时才能释放锁。</p>
<p><strong>② synchronized 依赖于 JVM 而 ReentrantLock 依赖于 API</strong></p>
<p>synchronized 是依赖于 JVM 实现的,前面我们也讲到了 虚拟机团队在 JDK1.6 为 synchronized 关键字进行了很多优化,但是这些优化都是在虚拟机层面实现的,并没有直接暴露给我们。ReentrantLock 是 JDK 层面实现的(也就是 API 层面,需要 lock() 和 unlock() 方法配合 try/finally 语句块来完成),所以我们可以通过查看它的源代码,来看它是如何实现的。</p>
<p><strong>③ ReentrantLock 比 synchronized 增加了一些高级功能</strong></p>
<p>相比 synchronized,ReentrantLock 增加了一些高级功能。主要来说主要有三点:<strong>① 等待可中断;② 可实现公平锁;③ 可实现选择性通知(锁可以绑定多个条件)</strong></p>
<ul>
<li><strong>ReentrantLock提供了一种能够中断等待锁的线程的机制</strong>,通过 lock.lockInterruptibly() 来实现这个机制。也就是说正在等待的线程可以选择放弃等待,改为处理其他事情。</li>
<li><strong>ReentrantLock可以指定是公平锁还是非公平锁。而synchronized只能是非公平锁。所谓的公平锁就是先等待的线程先获得锁。</strong> ReentrantLock默认情况是非公平的,可以通过 ReentrantLock类的<code>ReentrantLock(boolean fair)</code>构造方法来制定是否是公平的。</li>
<li>synchronized关键字与wait()和notify/notifyAll()方法相结合可以实现等待/通知机制,ReentrantLock类当然也可以实现,但是需要借助于Condition接口与newCondition() 方法。Condition是JDK1.5之后才有的,它具有很好的灵活性,比如可以实现多路通知功能也就是在一个Lock对象中可以创建多个Condition实例(即对象监视器),<strong>线程对象可以注册在指定的Condition中,从而可以有选择性的进行线程通知,在调度线程上更加灵活。 在使用notify/notifyAll()方法进行通知时,被通知的线程是由 JVM 选择的,用ReentrantLock类结合Condition实例可以实现“选择性通知”</strong> ,这个功能非常重要,而且是Condition接口默认提供的。而synchronized关键字就相当于整个Lock对象中只有一个Condition实例,所有的线程都注册在它一个身上。如果执行notifyAll()方法的话就会通知所有处于等待状态的线程这样会造成很大的效率问题,而Condition实例的signalAll()方法 只会唤醒注册在该Condition实例中的所有等待线程。</li>
<li><strong>ReentrantLock 提供了一种能够中断等待锁的线程的机制</strong>,通过 lock.lockInterruptibly() 来实现这个机制。也就是说正在等待的线程可以选择放弃等待,改为处理其他事情。</li>
<li><strong>ReentrantLock 可以指定是公平锁还是非公平锁。而 synchronized 只能是非公平锁。所谓的公平锁就是先等待的线程先获得锁。</strong> ReentrantLock 默认情况是非公平的,可以通过 ReentrantLock 类的<code>ReentrantLock(boolean fair)</code>构造方法来制定是否是公平的。</li>
<li>synchronized 关键字与 wait()和 notify/notifyAll()方法相结合可以实现等待/通知机制,ReentrantLock 类当然也可以实现,但是需要借助于 Condition 接口与 newCondition() 方法。Condition 是 JDK1.5 之后才有的,它具有很好的灵活性,比如可以实现多路通知功能也就是在一个 Lock 对象中可以创建多个 Condition 实例(即对象监视器),<strong>线程对象可以注册在指定的 Condition 中,从而可以有选择性的进行线程通知,在调度线程上更加灵活。 在使用 notify/notifyAll()方法进行通知时,被通知的线程是由 JVM 选择的,用 ReentrantLock 类结合 Condition 实例可以实现“选择性通知”</strong> ,这个功能非常重要,而且是 Condition 接口默认提供的。而 synchronized 关键字就相当于整个 Lock 对象中只有一个 Condition 实例,所有的线程都注册在它一个身上。如果执行 notifyAll()方法的话就会通知所有处于等待状态的线程这样会造成很大的效率问题,而 Condition 实例的 signalAll()方法 只会唤醒注册在该 Condition 实例中的所有等待线程。</li>
</ul>
<p>如果你想使用上述功能,那么选择ReentrantLock是一个不错的选择。<br>如果你想使用上述功能,那么选择 ReentrantLock 是一个不错的选择。</p>
<p><strong>④ 两者的性能已经相差无几</strong></p>
<p>在 JDK1.6 之前,synchronized 的性能是比 ReentrantLock 差很多。具体表示为:synchronized 关键字吞吐量岁线程数的增加,下降得非常严重。而 ReentrantLock 基本保持一个比较稳定的水平。我觉得这也侧面反映了, synchronized 关键字还有非常大的优化余地。后续的技术发展也证明了这一点,我们上面也讲了在 JDK1.6 之后 JVM 团队对 synchronized 关键字做了很多优化。JDK1.6 之后,synchronized 和 ReentrantLock 的性能基本是持平了。所以网上那些说因为性能才选择 ReentrantLock 的文章都是错的!JDK1.6 之后,性能已经不是选择 synchronized 和 ReentrantLock 的影响因素了!而且虚拟机在未来的性能改进中会更偏向于原生的 synchronized,所以还是提倡在 synchronized 能满足你的需求的情况下,优先考虑使用 synchronized 关键字来进行同步!优化后的 synchronized 和 ReentrantLock 一样,在很多地方都是用到了 CAS 操作。</p>
<h2 id="4-线程池了解吗"><a href="#4-线程池了解吗" class="headerlink" title="4 线程池了解吗?"></a>4 线程池了解吗?</h2><h3 id="4-1-为什么要用线程池"><a href="#4-1-为什么要用线程池" class="headerlink" title="4.1 为什么要用线程池?"></a>4.1 为什么要用线程池?</h3><p>线程池提供了一种限制和管理资源(包括执行一个任务)。 每个线程池还维护一些基本统计信息,例如已完成任务的数量。</p>
<p>这里借用《Java 并发编程的艺术》提到的来说一下使用线程池的好处:</p>
<ul>
<li><strong>降低资源消耗。</strong> 通过重复利用已创建的线程降低线程创建和销毁造成的消耗。</li>
<li><strong>提高响应速度。</strong> 当任务到达时,任务可以不需要的等到线程创建就能立即执行。</li>
<li><strong>提高线程的可管理性。</strong> 线程是稀缺资源,如果无限制的创建,不仅会消耗系统资源,还会降低系统的稳定性,使用线程池可以进行统一的分配,调优和监控。</li>
</ul>
<h3 id="4-2-Java-提供了哪几种线程池-他们各自的使用场景是什么"><a href="#4-2-Java-提供了哪几种线程池-他们各自的使用场景是什么" class="headerlink" title="4.2 Java 提供了哪几种线程池?他们各自的使用场景是什么?"></a>4.2 Java 提供了哪几种线程池?他们各自的使用场景是什么?</h3><h4 id="Java-主要提供了下面-4-种线程池"><a href="#Java-主要提供了下面-4-种线程池" class="headerlink" title="Java 主要提供了下面 4 种线程池"></a>Java 主要提供了下面 4 种线程池</h4><ul>
<li><strong>FixedThreadPool:</strong> 该方法返回一个固定线程数量的线程池。该线程池中的线程数量始终不变。当有一个新的任务提交时,线程池中若有空闲线程,则立即执行。若没有,则新的任务会被暂存在一个任务队列中,待有线程空闲时,便处理在任务队列中的任务。</li>
<li><strong>SingleThreadExecutor:</strong> 方法返回一个只有一个线程的线程池。若多余一个任务被提交到该线程池,任务会被保存在一个任务队列中,待线程空闲,按先入先出的顺序执行队列中的任务。</li>
<li><strong>CachedThreadPool:</strong> 该方法返回一个可根据实际情况调整线程数量的线程池。线程池的线程数量不确定,但若有空闲线程可以复用,则会优先使用可复用的线程。若所有线程均在工作,又有新的任务提交,则会创建新的线程处理任务。所有线程在当前任务执行完毕后,将返回线程池进行复用。</li>
<li><strong>ScheduledThreadPoolExecutor:</strong> 主要用来在给定的延迟后运行任务,或者定期执行任务。ScheduledThreadPoolExecutor 又分为:ScheduledThreadPoolExecutor(包含多个线程)和 SingleThreadScheduledExecutor (只包含一个线程)两种。</li>
</ul>
<h4 id="各种线程池的适用场景介绍"><a href="#各种线程池的适用场景介绍" class="headerlink" title="各种线程池的适用场景介绍"></a>各种线程池的适用场景介绍</h4><p>synchronized 是依赖于 JVM 实现的,前面我们也讲到了 虚拟机团</p>
<p><strong>(1) 使用 Executors 创建</strong></p>
<p>我们上面刚刚提到了 Java 提供的几种线程池,通过 Executors 工具类我们可以很轻松的创建我们上面说的几种线程池。但是实际上我们一般都不是直接使用 Java 提供好的线程池,另外在《阿里巴巴 Java 开发手册》中强制线程池不允许使用 Executors 去创建,而是通过 ThreadPoolExecutor 构造函数 的方式,这样的处理方式让写的同学更加明确线程池的运行规则,规避资源耗尽的风险。</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="JAVA"><figure class="iseeu 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">Executors 返回线程池对象的弊端如下:</span><br><span class="line"> FixedThreadPool 和 SingleThreadExecutor : 允许请求的队列长度为 Inte</span><br><span class="line">CachedThreadPool 和 ScheduledThreadPool : 允许创建的线程数量为 Integer.MAX_VALUE ,可能会创建大量线程,从而导致OOM。</span><br></pre></td></tr></table></figure></div>
<p><strong>(2) ThreadPoolExecutor 的构造函数创建</strong></p>
<p>我们可以自己直接调用 ThreadPoolExecutor 的构造函数来自己创建线程池。在创建的同时,给 BlockQueue 指定容量就可以了。示例如下:</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="PLAIN"><figure class="iseeu 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">private static ExecutorService executor = new ThreadPoolExecutor(13, 13,</span><br><span class="line"> new ArrayBlockingQueue(13));</span><br></pre></td></tr></table></figure></div>
<p>这种情况下,一旦提交的线程数超过当前可用线程数时,就会抛出 java.util.concurrent.RejectedExecutionException,这是因为当前线程池使用的队列是有边界队列,队列已经满了便无法继续处理新的请求。但是异常(Exception)总比发生错误(Error)要好。</p>
<h2 id="5-Nginx"><a href="#5-Nginx" class="headerlink" title="5 Nginx"></a>5 Nginx</h2><h3 id="5-1-简单介绍一下-Nginx"><a href="#5-1-简单介绍一下-Nginx" class="headerlink" title="5.1 简单介绍一下 Nginx"></a>5.1 简单介绍一下 Nginx</h3><p>Nginx 是一款轻量级的 Web 服务器/反向代理服务器及电子邮件(IMAP/POP3)代理服务器。 Nginx 主要提供反向代理、负载均衡、动静分离(静态资源服务)等服务。下面我简单地介绍一下这些名词。</p>
<h4 id="反向代理"><a href="#反向代理" class="headerlink" title="反向代理"></a>反向代理</h4><p>谈到反向代理,就不得不提一下正向代理。无论是正向代理,还是反向代理,说到底,就是代理模式的衍生版本罢了</p>
<ul>
<li><strong>正向代理:</strong>某些情况下,代理我们用户去访问服务器,需要用户手动的设置代理服务器的 ip 和端口号。正向代理比较常见的一个例子就是 VPN 了。</li>
<li><strong>反向代理:</strong> 是用来代理服务器的,代理我们要访问的目标服务器。代理服务器接受请求,然后将请求转发给内部网络的服务器,并将从服务器上得到的结果返回给客户端,此时代理服务器对外就表现为一个服务器。</li>
</ul>
<p>通过下面两幅图,大家应该更好理解(图源:<a href="http://blog.720ui.com/2016/nginx_action_05_proxy/):" target="_blank" rel="noopener">http://blog.720ui.com/2016/nginx_action_05_proxy/):</a><br> Nginx是一款轻量级的Web 服务器/反向代理服务器及电子邮件(I</p>
<p>在高并发情况下需要使用,其原理就是将并发请求分摊到多个服务器执行,减轻每台服务器的压力,多台服务器(集群)共同完成工作任务,从而提高了数据的吞吐量。</p>
<p>Nginx 支持的 weight 轮询(默认)、ip_hash、fair、url_hash 这四种负载均衡调度算法,感兴趣的可以自行查阅。</p>
<p>负载均衡相比于反向代理更侧重的是将请求分担到多台服务器上去,所以谈论负载均衡只有在提供某服务的服务器大于两台时才有意义。</p>
<p>Nginx支持的weight轮询(默认)、ip_hash、fair、url_hash这四种负</p>
<h3 id="5-2-为什么要用-Nginx"><a href="#5-2-为什么要用-Nginx" class="headerlink" title="5.2 为什么要用 Nginx?"></a>5.2 为什么要用 Nginx?</h3><blockquote>
<p>这部分内容参考极客时间—<a href="https://time.geekbang.org/course/intro/138?code=AycjiiQk6uQRxnVJzBupFkrGkvZlmYELPRsZbWzaAHE=" target="_blank" rel="noopener" title="Nginx核心知识100讲的内容">Nginx 核心知识 100 讲的内容</a>。<br>如果面试官问你这个问题,就一定想看你知道 Nginx 服务器的一些优点吗。</p>
</blockquote>
<p>Nginx 有以下 5 个优点:</p>
<ol>
<li>高并发、高性能(这是其他 web 服务器不具有的)</li>
<li>可扩展性好(模块化设计,第三方插件生态圈丰富)</li>
<li>高可靠性(可以在服务器行持续不间断的运行数年)</li>
<li>热部署(这个功能对于 Nginx 来说特别重要,热部署指可以在不停止 Nginx 服务的情况下升级 Nginx)</li>
<li>BSD 许可证(意味着我们可以将源代码下载下来进行修改然后使用自己的版本)</li>
</ol>
<h3 id="5-3-Nginx-的四个主要组成部分了解吗"><a href="#5-3-Nginx-的四个主要组成部分了解吗" class="headerlink" title="5.3 Nginx 的四个主要组成部分了解吗?"></a>5.3 Nginx 的四个主要组成部分了解吗?</h3><blockquote>
<p>这部分内容参考极客时间—<a href="https://time.geekbang.org/course/intro/138?code=AycjiiQk6uQRxnVJzBupFkrGkvZlmYELPRsZbWzaAHE=" target="_blank" rel="noopener" title="Nginx核心知识100讲的内容">Nginx 核心知识 100 讲的内容</a>。</p>
</blockquote>
<ul>
<li>Nginx 二进制可执行文件:由各模块源码编译出一个文件</li>
<li>nginx.conf 配置文件:控制 Nginx 行为</li>
<li>acess.log 访问日志: 记录每一条 HTTP 请求信息</li>
<li>error.log 错误日志:定位问题</li>
</ul>
<p><a href="https://github.com/Snailclimb/JavaGuide/commit/6e286c68974f71ba008532b35291a0d287107e14#diff-82afb3f12840aa5dee16173b05f482d2" target="_blank" rel="noopener">原文链接</a></p>
</div>
</div>
</article>
<article id="post-使用flutter二维码的坑" class="article global-container article-type-post" itemscope itemprop="blogPost">
<header class="article-header">
<h1 itemprop="name">
<a class="article-title" href="/2019/11/26/%E4%BD%BF%E7%94%A8flutter%E4%BA%8C%E7%BB%B4%E7%A0%81%E7%9A%84%E5%9D%91/">使用flutter二维码的坑</a>
</h1>
</header>
<div class="article-meta">
<a href="/2019/11/26/%E4%BD%BF%E7%94%A8flutter%E4%BA%8C%E7%BB%B4%E7%A0%81%E7%9A%84%E5%9D%91/" class="article-date">
<time datetime="2019-11-26T08:50:03.065Z" itemprop="datePublished">2019-11-26</time>
</a>
<ul class="article-tag-list" itemprop="keywords"><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/barcode-scan/" rel="tag">barcode_scan</a></li><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/flutter/" rel="tag">flutter</a></li><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/%E6%8F%92%E4%BB%B6/" rel="tag">插件</a></li></ul>
</div>
<div class="article-inner">
<div class="article-content article-content-cloud" itemprop="articleBody">
<p>插件barcode_scan</p>
<p>项目要进行二维码扫码,然后看了一下,参考网上的资料,设置了但是怎么也运行不起来。查了半天发现是AndroidX版本冲突的问题。</p>
<p>参考解决方式:<br> (<a href="https://blog.csdn.net/ocean20/article/details/89215290" target="_blank" rel="noopener">https://blog.csdn.net/ocean20/article/details/89215290</a>)</p>
<p>在<code>android/gradle/wrapper/gradle-wrapper.properties</code>更改以<code>distributionUrl</code>这样开头的行:<br>至少是4.10.2</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="DART"><figure class="iseeu highlight /dart"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">distributionUrl=https\:<span class="comment">//services.gradle.org/distributions/gradle-4.10.2-all.zip</span></span><br></pre></td></tr></table></figure></div>
<p>在<code>android/build.gradle</code>,替换:</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="PLAIN"><figure class="iseeu 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">dependencies {</span><br><span class="line"> classpath 'com.android.tools.build:gradle:3.2.1'</span><br><span class="line">}</span><br></pre></td></tr></table></figure></div>
<p>替换为</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="PLAIN"><figure class="iseeu 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">dependencies {</span><br><span class="line"> classpath 'com.android.tools.build:gradle:3.3.0'</span><br><span class="line">}</span><br></pre></td></tr></table></figure></div>
<p>在<code>android/gradle.properties</code>,附加:</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="PLAIN"><figure class="iseeu 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">android.enableJetifier=true</span><br><span class="line">android.useAndroidX=true</span><br></pre></td></tr></table></figure></div>
<p>在<code>android/app/build.gradle</code>:</p>
<p>在<code>android</code> 块中,确保<code>compileSdkVersion</code>并且targetSdkVersion至少<code>28</code>。</p>
<p>将所有已弃用的库替换为AndroidX等效项。例如,如果您使用默认.gradle文件,请进行以下更改:</p>
<p>在 <code>android/app/build.gradle</code></p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="PLAIN"><figure class="iseeu highlight /plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"</span><br></pre></td></tr></table></figure></div>
<p>改为:</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="PLAIN"><figure class="iseeu highlight /plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"</span><br></pre></td></tr></table></figure></div>
<p>最后,在 <code>dependencies</code> 块中,替换</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="PLAIN"><figure class="iseeu 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">androidTestImplementation 'com.android.support.test:runner:1.0.2'</span><br><span class="line">androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'</span><br></pre></td></tr></table></figure></div>
<p>改为</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="PLAIN"><figure class="iseeu 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">androidTestImplementation 'androidx.test:runner:1.1.1'</span><br><span class="line">androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.1'</span><br></pre></td></tr></table></figure></div>
<p>除此之外,还要检查 <code>kotlin_version</code> 的版本不能低于<code>1.3.0</code><br>在 <code>android/build.gradle</code> 修改 <code>kotlin_version</code> 为 <code>1.3.0</code></p>
<p>ok,项目跑起来 完毕</p>
</div>
</div>
</article>
<article id="post-android虚拟机快捷开启" class="article global-container article-type-post" itemscope itemprop="blogPost">
<header class="article-header">
<h1 itemprop="name">
<a class="article-title" href="/2019/11/26/android%E8%99%9A%E6%8B%9F%E6%9C%BA%E5%BF%AB%E6%8D%B7%E5%BC%80%E5%90%AF/">Android avd 虚拟机快捷开启</a>
</h1>
</header>
<div class="article-meta">
<a href="/2019/11/26/android%E8%99%9A%E6%8B%9F%E6%9C%BA%E5%BF%AB%E6%8D%B7%E5%BC%80%E5%90%AF/" class="article-date">
<time datetime="2019-11-26T01:48:44.323Z" itemprop="datePublished">2019-11-26</time>
</a>
<ul class="article-tag-list" itemprop="keywords"><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/android/" rel="tag">android</a></li><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/avd/" rel="tag">avd</a></li><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/%E8%99%9A%E6%8B%9F%E6%9C%BA/" rel="tag">虚拟机</a></li></ul>
</div>
<div class="article-inner">
<div class="article-content article-content-cloud" itemprop="articleBody">
<p>新建一个 <code>bat</code> 文件</p>
<p>全局搜索 <code>emulator.exe</code>, 找到<code>emulator</code> 目录下的<code>emulator.exe</code> 复制路径 到 <code>bat</code> 文件中</p>
<p>例如: 我的路径是 <code>E:\android-sdk-windows\emulator</code></p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="PLAIN"><figure class="iseeu highlight /plain"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line">E:\android-sdk-windows\emulator\emulator.exe</span><br></pre></td></tr></table></figure></div>
<p>然后追加参数设置 :</p>
<ul>
<li>netdelay: 网络延迟</li>
<li>netspeed: 网速</li>
<li>avd: 要开启的虚拟机名称</li>
</ul>
<p>完整的命令:</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="BAT"><figure class="iseeu highlight /bat"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="function">E:\<span class="title">android</span>-<span class="title">sdk</span>-<span class="title">windows</span>\<span class="title">emulator</span>\<span class="title">emulator.exe</span> -<span class="title">netdelay</span> <span class="title">none</span> -<span class="title">netspeed</span> <span class="title">full</span> -<span class="title">avd</span> <span class="title">Pixel_2_API_29</span></span></span><br></pre></td></tr></table></figure></div>
<p>^[注意: 如果虚拟机名称有空格要用下划线代替]</p>
</div>
</div>
</article>
<article id="post-filename" class="article global-container article-type-post" itemscope itemprop="blogPost">
<header class="article-header">
<h1 itemprop="name">
<a class="article-title" href="/2019/11/25/filename/">Flutter_html插件使用</a>
</h1>
</header>
<div class="article-meta">
<a href="/2019/11/25/filename/" class="article-date">
<time datetime="2019-11-25T08:52:48.103Z" itemprop="datePublished">2019-11-25</time>
</a>
<ul class="article-tag-list" itemprop="keywords"><li class="article-tag-list-item"><a class="article-tag-list-link" href="/tags/flutter-html/" rel="tag">flutter,html</a></li></ul>
</div>
<div class="article-inner">
<div class="article-content article-content-cloud" itemprop="articleBody">
<p><a href="https://pub.dev/packages/flutter_html" target="_blank" rel="noopener">Flutter_html插件地址</a><br>添加依赖项<br> 打开 <code>pubspec.yml</code> 添加 :</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="DART"><figure class="iseeu highlight /dart"><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">dependencies:</span><br><span class="line"> flutter_html: ^<span class="number">0.11</span><span class="number">.0</span></span><br></pre></td></tr></table></figure></div>
<p>然后执行 <code>flutter pub git</code> 获取依赖项</p>
<p>再 <code>dart</code> 文件中引入包</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="DART"><figure class="iseeu highlight /dart"><table><tr><td class="gutter"><pre><span class="line">1</span><br></pre></td><td class="code"><pre><span class="line"><span class="keyword">import</span> <span class="string">'package:flutter_html/flutter_html.dart'</span>;</span><br></pre></td></tr></table></figure></div>
<p>目前支持的 <code>html标签</code> </p>
<blockquote>
<p>a, abbr, acronym, address, article, aside, b, bdi, bdo, big, blockquote, body, br, caption, cite, code, data, dd, del, dfn, div, dl, dt, em, figcaption, figure, footer, h1, h2, h3, h4, h5, h6, header, hr, i, img, ins, kbd, li, main, mark, nav, noscript, ol, p, pre, q, rp, rt, ruby, s, samp, section, small, span, strike, strong, sub, sup, table, tbody, td, template, tfoot, th, thead, time, tr, tt, u, ul, var</p>
</blockquote>
<p>使用示例</p>
<div class="highlight-box"autocomplete="off" autocorrect="off" autocapitalize="off" spellcheck="false" contenteditable="true"data-rel="DART"><figure class="iseeu highlight /dart"><table><tr><td class="gutter"><pre><span class="line">1</span><br><span class="line">2</span><br><span class="line">3</span><br><span class="line">4</span><br><span class="line">5</span><br><span class="line">6</span><br><span class="line">7</span><br><span class="line">8</span><br><span class="line">9</span><br><span class="line">10</span><br><span class="line">11</span><br><span class="line">12</span><br><span class="line">13</span><br><span class="line">14</span><br><span class="line">15</span><br><span class="line">16</span><br><span class="line">17</span><br><span class="line">18</span><br><span class="line">19</span><br><span class="line">20</span><br><span class="line">21</span><br><span class="line">22</span><br><span class="line">23</span><br><span class="line">24</span><br><span class="line">25</span><br><span class="line">26</span><br><span class="line">27</span><br><span class="line">28</span><br></pre></td><td class="code"><pre><span class="line">Html(</span><br><span class="line"> data: <span class="string">"""</span></span><br><span class="line"><span class="string"> <div></span></span><br><span class="line"><span class="string"> <h1>Demo Page</h1></span></span><br><span class="line"><span class="string"> <p>This is a fantastic nonexistent product that you should buy!</p></span></span><br><span class="line"><span class="string"> <h2>Pricing</h2></span></span><br><span class="line"><span class="string"> <p>Lorem ipsum <b>dolor</b> sit amet.</p></span></span><br><span class="line"><span class="string"> <h2>The Team</h2></span></span><br><span class="line"><span class="string"> <p>There isn't <i>really</i> a team...</p></span></span><br><span class="line"><span class="string"> <h2>Installation</h2></span></span><br><span class="line"><span class="string"> <p>You <u>cannot</u> install a nonexistent product!</p></span></span><br><span class="line"><span class="string"> <!--You can pretty much put any html in here!--></span></span><br><span class="line"><span class="string"> </div></span></span><br><span class="line"><span class="string"> """</span>,</span><br><span class="line"> <span class="comment">//设置样式:</span></span><br><span class="line"> padding: EdgeInsets.all(<span class="number">8.0</span>),</span><br><span class="line"> backgroundColor: Colors.white70,</span><br><span class="line"> defaultTextStyle: TextStyle(fontFamily: <span class="string">'serif'</span>),</span><br><span class="line"> linkStyle: <span class="keyword">const</span> TextStyle(</span><br><span class="line"> color: Colors.redAccent,</span><br><span class="line"> ),</span><br><span class="line"> onLinkTap: (url) {</span><br><span class="line"> <span class="comment">// 点击 html 中 a 标签链接触发</span></span><br><span class="line"> },</span><br><span class="line"> onImageTap: (src) {</span><br><span class="line"> <span class="comment">// 点击 图片 触发.</span></span><br><span class="line"> },</span><br><span class="line">)</span><br></pre></td></tr></table></figure></div>
</div>
</div>
</article>
<article id="post-hello-world" class="article global-container article-type-post" itemscope itemprop="blogPost">
<header class="article-header">
<h1 itemprop="name">
<a class="article-title" href="/2019/11/25/hello-world/">Hello World</a>
</h1>
</header>
<div class="article-meta">
<a href="/2019/11/25/hello-world/" class="article-date">
<time datetime="2019-11-25T02:06:45.000Z" itemprop="datePublished">2019-11-25</time>
</a>
</div>
<div class="article-inner">
<div class="article-content article-content-cloud" itemprop="articleBody">
<h2 id="Abeiv"><a href="#Abeiv" class="headerlink" title="Abeiv"></a>Abeiv</h2><h3 id="菜就是憨"><a href="#菜就是憨" class="headerlink" title="菜就是憨"></a>菜就是憨</h3>
</div>
</div>
</article>
</section>
<aside id="sidebar">
<div class="widget-box">
<div class="avatar-box">
<img class="avatar" src="/1.jpg" title="图片来自网络"></img>
<h3 class="avatar-name">
Abeiv
</h3>
<p class="avatar-slogan">
一个渣渣,月牙姐姐天下第一可爱。
</p>
</div>
</div>
<div class="widget-box">
<h3 class="widget-title">Tags</h3>
<div class="widget">
<ul class="tag-list" itemprop="keywords"><li class="tag-list-item"><a class="tag-list-link" href="/tags/400/" rel="tag">400</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/android/" rel="tag">android</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/avd/" rel="tag">avd</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/barcode-scan/" rel="tag">barcode_scan</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/flutter/" rel="tag">flutter</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/flutter-html/" rel="tag">flutter,html</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/http/" rel="tag">http</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/java/" rel="tag">java</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/pixiv/" rel="tag">pixiv</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/python/" rel="tag">python</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/%E6%8F%92%E4%BB%B6/" rel="tag">插件</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/%E8%99%9A%E6%8B%9F%E6%9C%BA/" rel="tag">虚拟机</a></li><li class="tag-list-item"><a class="tag-list-link" href="/tags/%E9%9D%A2%E8%AF%95%E9%A2%98/" rel="tag">面试题</a></li></ul>
</div>
</div>