电子产业一站式赋能平台

PCB联盟网

搜索
查看: 69|回复: 0
收起左侧

一文带你深入理解分布式基础

[复制链接]

864

主题

864

帖子

8156

积分

高级会员

Rank: 5Rank: 5

积分
8156
发表于 2023-12-19 08:30:00 | 显示全部楼层 |阅读模式
点击上方“C语言与CPP编程”,选择“关注/置顶/星标公众号  o& _7 S7 N7 |: d8 l6 c
干货福利,第一时间送达!/ f4 \( n' S  J- F

/ E( ?$ ^7 O+ e8 m" o1 V

cbuskmyfhvi64067888443.png

cbuskmyfhvi64067888443.png

1 P# M1 _- B7 T& g" ^最近有小伙伴说没有收到当天的文章推送,这是因为微信改了推送机制,有一部分小伙伴刷不到当天的文章,一些比较实用的知识和信息,错过了就是错过了,建议大家加个星标??,就能第一时间收到推送。
: F& u  y8 n  v0 W1 M

fgk44m5jgxu64067888543.png

fgk44m5jgxu64067888543.png

6 I" O6 o6 Q. Y8 e" ~大家好,我是飞宇。
7 ^5 D- z) ]5 J0 a
! m- L$ T: R0 S4 I; A7 j( U互联网最重要任务莫过于海量数据处理,这里是大厂不得不解决问题(研究技术),即大规模分布式系统(单机性能有限),分布式是互联网的核心技术,希望大家可以掌握。  {# m% I+ G2 o! {6 ^
* b; v& J. D4 o; k9 s

tbz2ff5shgd64067888644.png

tbz2ff5shgd64067888644.png

( j, W, w$ l3 ~  |5 @
% d' ?2 J; O) T: Y什么是 CAP?
- _; O& V. ^0 R: _' a1 ?. ^$ U) Q1 I; i8 A+ ?: F  u6 `# [
如果你之前没有听说过 CAP 理论的话,看到这三个字母第一反应或许是“帽子”吧。那么,在分布式领域中,CAP 这顶“帽子”到底是什么呢?我们先来看看这三个字母分别指的是什么吧。接下来,我结合电商的例子,带你理解 CAP 的含义。
6 K* s% s) q: L3 L7 d$ F) [% L
2 [: b: z3 F8 B9 S. N0 q假设某电商,在北京、杭州、上海三个城市建立了仓库,同时建立了对应的服务器{A, B, C}用于存储商品信息。比如,某电吹风在北京仓库有 20 个,在杭州仓库有 10 个,在上海仓库有 30 个。那么,CAP 这三个字母在这个例子中分别代表什么呢?
; S  n4 ^% w7 F/ b" M/ W) t' I7 N9 _, i# S, @
首先,我们来看一下 C。C 代表 Consistency,一致性,是指所有节点在同一时刻的数据是相同的,即更新操作执行结束并响应用户完成后,所有节点存储的数据会保持相同。
+ |6 M: q% M1 o, F6 j
% t3 R4 ?# c- t4 k$ F' b) v在电商系统中,A、B、C 中存储的该电吹风的数量应该是 20+10+30=60。假设,现在有一个北京用户买走一个电吹风,服务器 A 会更新数据为 60-1=59,与此同时要求 B 和 C 也更新为 59,以保证在同一时刻,无论访问 A、B、C 中的哪个服务器,得到的数据均是 59。/ {% |: x: F1 D0 ]  ?% ?7 l6 F
; t( d& G" S' h( |
然后,看一下 A。A 代表 Availability,可用性,是指系统提供的服务一直处于可用状态,对于用户的请求可即时响应。
! M2 ?, I: x4 t2 @( d
" H! s" k) _0 M  S% Q: U( q: T在电商系统中,用户在任一时刻向 A、B、C 中的任一服务器发出请求时,均可得到即时响应,比如查询商品信息等。; r& j3 H2 Q5 E* b; _" B& w3 n
: v8 z9 W, \, {6 o1 i  H
最后,我们看一下 P。P 代表 Partition Tolerance,分区容错性,是指在分布式系统遇到网络分区的情况下,仍然可以响应用户的请求。网络分区是指因为网络故障导致网络不连通,不同节点分布在不同的子网络中,各个子网络内网络正常。
" ], p" u6 N% W  {+ q) |: T( H在电商系统中,假设 C 与 A 和 B 的网络都不通了,A 和 B 是相通的。也就是说,形成了两个分区{A, B}和{C},在这种情况下,系统仍能响应用户请求。
. L2 w$ }7 c, H+ ?- n( z4 v: V# H* F$ p6 T' l3 P
一致性可用性分区容错性,就是分布式系统的三个特征。
* q2 N8 P, ]' r* {: M, U0 o0 {. F% n; @1 V' v$ q3 u. q
CAP 理论又是什么呢?
2 ^# y0 ]! i% P  i" J4 m- }! Q- F
' \$ z( a1 Z$ fCAP 理论指的就是,在分布式系统中 C、A、P 这三个特征不能同时满足,只能满足其中两个,如下图所示。这,是不是有点像分布式系统在说,这顶“帽子”我不想要呢?
2 [7 Z2 x0 W7 s7 ^7 E

c3442zemq5l64067888744.png

c3442zemq5l64067888744.png

  i; x( `% g$ P- `9 Q$ H/ R8 E! F1 E7 l/ N6 Q* h7 J1 l: f: `4 ^- b
接下来,我就通过一个例子和你进一步解释下,什么是 CAP 以及 CAP 为什么不能同时满足吧。
$ \; t2 [, f" i8 {3 e
0 w3 }" c% L" N0 I# a- V如下图所示,网络中有两台服务器 Server1 和 Server2,分别部署了数据库 DB1 和 DB2,这两台机器组成一个服务集群,DB1 和 DB2 两个数据库中的数据要保持一致,共同为用户提供服务。用户 User1 可以向 Server1 发起查询数据的请求,用户 User2 可以向服务器 Server2 发起查询数据的请求,它们共同组成了一个分布式系统。
0 p6 I# d+ g1 f0 V  J$ X- R; Y0 W; N' P8 I  z+ \' D

wgyjheclahg64067888844.png

wgyjheclahg64067888844.png
! ?% ^& s1 l; x6 P+ N0 y& E! e
9 }- ?" s, K3 o# Y* H1 }
对这个系统来说,分别满足 C、A 和 P 指的是:
( i6 ?1 m, _& a4 d3 M/ _; |6 Y' j4 {# H
在满足一致性 C 的情况下,Server1 和 Server2 中的数据库始终保持一致,即 DB1 和 DB2 内容要始终保持相同;) l* _  E7 Q4 E; q% R
7 x) Q: V* ]( u( P0 N0 `6 {: l
在满足可用性 A 的情况下,用户无论访问 Server1 还是 Server2,都会得到即时响应;9 v" s/ Y3 v, E3 W3 l

1 ^1 d* M" l( p! g2 M  S在满足分区容错性 P 的情况下,Server1 和 Server2 之间即使出现网络故障也不会影响 Server1 和 Server2 分别处理用户的请求。
) x8 D5 W7 b6 z+ F" n, n, p
6 a% L4 ~$ t1 @当用户发起请求时,收到请求的服务器会及时响应,并将用户更新的数据同步到另一台服务器,保证数据一致性。具体的工作流程,如下所示:
& U3 d/ b9 K* h4 y' w& ^1 A% L* [2 |9 I! Q7 t
用户 User1 向服务器 Server1 发起请求,将数据库 DB1 中的数据 a 由 1 改为 2;$ M2 l  w) R. z( E& ^/ c8 f
系统会进行数据同步,即图中的 S 操作,将 Server1 中 DB1 的修改同步到服务器 Server2 中,使得 DB2 中的数据 a 也被修改为 2;
' e/ ~* A! N/ j* a+ G8 \# |当 User2 向 Server2 发起读取数据 a 的请求时,会得到 a 最新的数据值 2。1 f! f# F$ o: P8 B, h: n3 s+ J
( h) S% X4 P* W4 k3 X

be1divgxbyp64067888944.png

be1divgxbyp64067888944.png
( l4 n- [4 I, c4 b

* {- v6 E6 r( D- t* I* [: H8 M这其实是在网络环境稳定、系统无故障的情况下的工作流程。但在实际场景中,网络环境不可能百分之百不出故障,比如网络拥塞、网卡故障等,会导致网络故障或不通,从而导致节点之间无法通信,或者集群中节点被划分为多个分区,分区中的节点之间可通信,分区间不可通信。
4 ~! ?0 _2 @% v: F6 g! X1 t  a4 ~: l$ J) ~; {2 u2 n
这种由网络故障导致的集群分区情况,通常被称为“网络分区”。在分布式系统中,网络分区不可避免,因此分区容错性 P 必须满足。接下来,我们就来讨论一下在满足分区容错性 P 的情况下,一致性 C 和可用性 A 是否可以同时满足。$ q' G. b* f* t' s
1 @5 s1 A8 u/ C3 o; p

% k5 d% y6 t& t. ^假设,Server1 和 Server2 之间网络出现故障,User1 向 Server1 发送请求,将数据库 DB1 中的数据 a 由 1 修改为 2,而 Server2 由于与 Server1 无法连接导致数据无法同步,所以 DB2 中 a 依旧是 1。这时,User2 向 Server2 发送读取数据 a 的请求时,Server2 无法给用户返回最新数据,那么该如何处理呢?
( F) m- G2 B7 G8 N6 L3 \$ }
" @4 [# X' A8 N; t% s1 h我们能想到的处理方式有如下两种。
; \. h. n5 R0 J- U* f第一种处理方式是,保证一致性 C,牺牲可用性 A:Server2 选择让 User2 的请求阻塞,一直等到网络恢复正常,Server1 被修改的数据同步更新到 Server2 之后,即 DB2 中数据 a 修改成最新值 2 后,再给用户 User2 响应。5 c8 M# v# E9 k7 Y0 ?2 q

4xllgmabcno64067889044.png

4xllgmabcno64067889044.png

+ z& I' [- D* R0 F9 c! W4 Y3 |7 V* O4 A1 Y2 a0 p  A; m
第二种处理方式是,保证可用性 A,牺牲一致性 C:Server2 选择将旧的数据 a=1 返回给用户,等到网络恢复,再进行数据同步。) X/ f8 A! t) K5 [1 }

1 {& p( f7 @( I2 ?: H- Z+ {

3czjzp3kqvu64067889144.png

3czjzp3kqvu64067889144.png
7 y- T3 a, A1 T& Z
$ {1 u# Z. q9 Q' _: [6 b! V7 L
除了以上这两种方案,没有其他方案可以选择。可以看出:在满足分区容错性 P 的前提下,一致性 C 和可用性 A 只能选择一个,无法同时满足。6 r; [( {% S0 |/ `6 ^! V3 m1 Y6 z
CAP 选择策略及应用通过上面的分析,你已经知道了分布式系统无法同时满足 CAP 这三个特性,那该如何进行取舍呢?* k2 H9 W: N" \2 E, f% q4 g

' ~) Q- n- Z* g) ]8 X  ?其实,C、A 和 P,没有谁优谁劣,只是不同的分布式场景适合不同的策略。接下来,我就以一些具体场景为例,分别与你介绍保 CA 弃 P、保 CP 弃 A、保 AP 弃 C 这三种策略,以帮助你面对不同的分布式场景时,知道如何权衡这三个特征。* m2 d% A2 l* ^
比如,对于涉及钱的交易时,数据的一致性至关重要,因此保 CP 弃 A 应该是最佳选择。2015 年发生的支付宝光纤被挖断的事件,就导致支付宝就出现了不可用的情况。显然,支付宝当时的处理策略就是,保证了 CP 而牺牲了 A。3 w% R7 i: ^3 U: t6 K+ q: {& u! z

9 a* [! M( S! H' i( m/ W4 r而对于其他场景,大多数情况下的做法是选择 AP 而牺牲 C,因为很多情况下不需要太强的一致性(数据始终保持一致),只要满足最终一致性即可。' m& l. {( I. h4 X1 u, t- T$ c7 X# x2 M

+ O: ]$ u* v/ _$ p, M( U- H3 x0 S最终一致性指的是,不要求集群中节点数据每时每刻保持一致,在可接受的时间内最终能达到一致就可以了。不知道你是否还记得,在第 6 篇文章分布式事务中介绍的基于分布式消息的最终一致性方案?没错,这个方案对事务的处理,就是选择 AP 而牺牲 C 的例子。- m4 z. R; d$ t/ b7 D  G

; a& |$ f$ a# S' o这个方案中,在应用节点之间引入了消息中间件,不同节点之间通过消息中间件进行交互,比如主应用节点要执行修改数据的事务,只需要将信息推送到消息中间件,即可执行本地的事务,而不需要备应用节点同意修改数据才能真正执行本地事务,备应用节点可以从消息中间件获取数据。) c6 c, X  a' p" @7 _- U6 ]
保 CA 弃 P首先,我们看一下保 CA 弃 P 的策略。
+ p+ o, s& @! f% b) [: ]/ ?在分布式系统中,现在的网络基础设施无法做到始终保持稳定,网络分区(网络不连通)难以避免。牺牲分区容错性 P,就相当于放弃使用分布式系统。因此,在分布式系统中,这种策略不需要过多讨论。
9 u: g/ F' u, O3 U9 h* J( O" L" x; `+ z3 p. e: D& K
既然分布式系统不能采用这种策略,那单点系统毫无疑问就需要满足 CA 特性了。比如关系型数据库 DBMS(比如 MySQL、Oracle)部署在单台机器上,因为不存在网络通信问题,所以保证 CA 就可以了。0 D5 k' e/ p" Z- J
保 CP 弃 A如果一个分布式场景需要很强的数据一致性,或者该场景可以容忍系统长时间无响应的情况下,保 CP 弃 A 这个策略就比较适合。1 w( d3 |; U. i, d" [3 J: I
4 Y0 ?9 o9 @* v3 k& e2 i
一个保证 CP 而舍弃 A 的分布式系统,一旦发生网络分区会导致数据无法同步情况,就要牺牲系统的可用性,降低用户体验,直到节点数据达到一致后再响应用户。6 H- n( K! }  O6 ?. A) C5 j
& Q* u( S* C# m7 F5 O- o
我刚刚也提到了,这种策略通常用在涉及金钱交易的分布式场景下,因为它任何时候都不允许出现数据不一致的情况,否则就会给用户造成损失。因此,这种场景下必须保证 CP。
! E+ A% U; s! @8 K/ q2 @8 O( M8 B1 W9 t2 K) m0 K8 D  F
保证 CP 的系统有很多,典型的有 Redis、HBase、ZooKeeper 等。接下来,我就以 ZooKeeper 为例,带你了解它是如何保证 CP 的。2 @; y! R- a; c
首先,我们看一下 ZooKeeper 架构图。
5 b; ]+ }* I; }
! h7 S# ^9 j! G2 I% }- f

akurhpoy0hk64067889244.png

akurhpoy0hk64067889244.png
3 D+ w8 I6 s, b/ C% O# n+ @
备注:此图引自ZooKeeper 官网。/ ^, v3 r. ~$ |/ V0 s% F, I

% k& ?# C& f6 }6 ^ZooKeeper 集群包含多个节点(Server),这些节点会通过分布式选举算法选出一个 Leader 节点。在 ZooKeeper 中选举 Leader 节点采用的是 ZAB 算法。在 ZooKeeper 集群中,Leader 节点之外的节点被称为 Follower 节点,
- u+ {" W$ a9 E+ C6 Z+ X
; L2 `* Y$ k4 c- i- c1 YLeader 节点会专门负责处理用户的写请求:
7 l  p8 F3 c" L: V3 R# y0 |当用户向节点发送写请求时,如果请求的节点刚好是 Leader,那就直接处理该请求;
7 H8 i% X* U# Y' \5 X) d: M& }3 o% w! h2 Z) p/ A
如果请求的是 Follower 节点,那该节点会将请求转给 Leader,然后 Leader 会先向所有的 Follower 发出一个 Proposal,等超过一半的节点同意后,Leader 才会提交这次写操作,从而保证了数据的强一致性。
: T6 e' u; k# ?; F具体示意图如下所示:
6 ?: G0 y" `- I6 q2 @9 ]1 ?0 S0 ^5 C: M2 t/ K7 q$ P8 W7 Y  z

4okx4dhvzjg64067889344.png

4okx4dhvzjg64067889344.png

3 }+ V2 c& |' X
5 t; }! X4 t- H/ d  v当出现网络分区时,如果其中一个分区的节点数大于集群总节点数的一半,那么这个分区可以再选出一个 Leader,仍然对用户提供服务,但在选出 Leader 之前,不能正常为用户提供服务;如果形成的分区中,没有一个分区的节点数大于集群总节点数的一半,那么系统不能正常为用户提供服务,必须待网络恢复后,才能正常提供服务。
( h/ \* B; |- D8 Q- m: Y8 x7 X$ n  j% N
这种设计方式保证了分区容错性,但牺牲了一定的系统可用性。- s1 @0 o# m2 z3 m6 a  r# d* ]
保 AP 弃 C如果一个分布式场景需要很高的可用性,或者说在网络状况不太好的情况下,该场景允许数据暂时不一致,那这种情况下就可以牺牲一定的一致性了。
$ D6 s  O+ X9 x! O网络分区出现后,各个节点之间数据无法马上同步,为了保证高可用,分布式系统需要即刻响应用户的请求。但,此时可能某些节点还没有拿到最新数据,只能将本地旧的数据返回给用户,从而导致数据不一致的情况。
/ _' N7 O+ I6 ]4 g( z+ {1 w: F( X& D6 o. d6 _9 Z: _
适合保证 AP 放弃 C 的场景有很多。比如,很多查询网站、电商系统中的商品查询等,用户体验非常重要,所以大多会保证系统的可用性,而牺牲一定的数据一致性。
% f* B( p( ?7 @3 `& ?8 d% ?, x2 Y6 s7 b5 _/ k( B- O- s
以电商购物系统为例,如下图所示,某电吹风在北京仓库有 20 个,在杭州仓库有 10 个,在上海仓库有 30 个。初始时,北京、杭州、上海分别建立的服务器{A, B, C}存储该电吹风的数量均为 60 个。
# f& M, P" z7 N1 Z# g, T" `
2 I: V# ?$ G: J- k. S* w假如,上海的网络出现了问题,与北京和杭州网络均不通,此时北京的用户通过北京服务器 A 下单购买了一个电吹风,电吹风数量减少到 59,并且同步给了杭州服务器 B。也就是说,现在用户的查询请求如果是提交到服务器 A 和 B,那么查询到的数量为 59。但通过上海服务器 C 进行查询的结果,却是 60。% O6 I+ ], D' s9 @+ H+ J

7 i% W0 ^8 e" [2 K当然,待网络恢复后,服务器 A 和 B 的数据会同步到 C,C 更新数据为 59,最终三台服务器数据保持一致,用户刷新一下查询界面或重新提交一下查询,就可以得到最新的数据。而对用户来说,他们并不会感知到前后数据的差异,到底是因为其他用户购买导致的,还是因为网络故障导致数据不同步而产生的。
* V1 o3 A# {* n$ }+ d+ W& u2 T) U
4 I$ T1 g7 K9 J

ddy133dj33w64067889444.png

ddy133dj33w64067889444.png
' m3 s' {+ }3 E. b" ]  F& \9 [
当然,你可能会说,为什么上海服务器不能等网络恢复后,再响应用户请求呢?可以想象一下,如果用户提交一个查询请求,需要等上几分钟、几小时才能得到反馈,那么用户早已离去了。
8 I* U9 J, C: O8 B1 Y, m! c
. o) I( l1 o  K# u, F- Q5 z  ]: h也就是说这种场景适合优先保证 AP,因为如果等到数据一致之后再给用户返回的话,用户的响应太慢,可能会造成严重的用户流失。1 F$ V% S4 b) p& q
! u9 L! c; @$ }
目前,采用保 AP 弃 C 的系统也有很多,比如 CoachDB、Eureka、Cassandra、DynamoDB 等。
6 z3 {: H, O$ q$ V对比分析保 CA 弃 P、保 CP 弃 A 和保 AP 弃 C 这三种策略,以方便你记忆和理解。' K4 w3 z/ T/ ^# ]

5 K- g0 H. z8 ]2 @2 ?

xm2y0jcfaeg64067889544.jpg

xm2y0jcfaeg64067889544.jpg
$ A' j" h$ |. i% I2 o; M
知识扩展:CAP 和 ACID 的“C”“A”是一样的吗?首先,我们看一下 CAP 中的 C 和 ACID 中的 C 是否一致。% c6 O0 |7 ^' N1 }

- y& p- z  `$ k( Y+ Q+ e9 bCAP 中的 C 强调的是数据的一致性,也就是集群中节点之间通过复制技术保证每个节点上的数据在同一时刻是相同的。0 u% G  H) W4 ?5 s
/ t& U. U) W8 l9 ]7 t& ?
ACID 中的 C 强调的是事务执行前后,数据的完整性保持一致或满足完整性约束。也就是不管在什么时候,不管并发事务有多少,事务在分布式系统中的状态始终保持一致。2 K$ J0 g$ f" o$ C% @; {9 R
" g7 |7 T. Z6 O; |4 e
其次,我们看一下 CAP 中的 A 和 ACID 中的 A。1 ~. o$ \9 c! C  ?) |
. H/ l- q" }8 U
CAP 中的 A 指的是可用性(Availability),也就是系统提供的服务一直处于可用状态,即对于用户的请求可即时响应。
7 T, \  q/ g, _, {( X7 R* h
% ?6 D$ a& h6 G8 n& ~3 G1 W! a; UACID 中的 A 指的是原子性(Atomicity),强调的是事务要么执行成功,要么执行失败。6 O6 U% n9 X! P3 B- l3 b  X: e7 ?

3 f, @+ d; o) M因此,CAP 和 ACID 中的“C”和“A”是不一样的,不能混为一谈。
, h% j' L$ V* U( j总结今天,我主要与你分享的是 CAP 理论
2 R4 V1 f, `- i3 I# _+ L# n# E
' a$ r5 R" B8 Q& G) W首先,我通过电商的例子带你了解了 CAP 这三个字母在分布式系统中的含义以及 CAP 理论,并与你证明了,C、A 和 P 在分布式系统中最多只能满足两个。
* F% \% j1 s- G- {1 O7 ?然后,我为你介绍了分布式系统设计时如何选择 CAP 策略,包括保 CA 弃 P、保 CP 弃 A、保 AP 弃 C,以及这三种策略适用的场景。5 j" y% _  B6 j6 k: n; Q/ Z

* ^1 x! x$ m$ X0 k8 e$ A/ p2 U相信通过今天的学习,你不仅对 CAP 理论有了更深刻的认识,并且可以针对不同场景采用哪种策略给出自己的建议。加油,行动起来,为你的业务场景选择一种合适的策略,来指导分布式系统的设计吧。相信你,一定可以的!
! \9 I/ |# }& N1 v3 B7 ~7 ?思考题CAP 理论和 BASE 理论的区别是什么?4 [' x+ M* H- t& A
) |2 x: n! Y% _
作者:聂鹏程,文章来源极客时间 --分布式技术原理与算法解析0 b, T2 m4 e5 T! d
https://time.geekbang.org/column/article/166582% X0 }3 n. x1 x; m& H! N  D
——EOF——一个我十分佩服的朋友阿秀开发了一个互联网大厂面试真题记录网站,支持按照行业公司岗位科目考察时间等查看面试真题,有意者欢迎扫码下方二维码适用~
* Z" p1 |' N  {: |

4hmggubek0y64067889644.png

4hmggubek0y64067889644.png
1 P/ F0 v/ \; t3 u& H

& a9 }: X/ x3 q; n你好,我是飞宇,本硕均于某中流985 CS就读,先后于百度搜索以及字节跳动电商等部门担任Linux C/C++后端研发工程师。$ F9 x; `" d7 j  A6 y
同时,我也是知乎博主@韩飞宇,日常分享C/C++、计算机学习经验、工作体会,欢迎点击此处查看我以前的学习笔记&经验&分享的资源。
$ {* N/ u7 G5 p+ F( ]  g我组建了一些社群一起交流,群里有大牛也有小白,如果你有意可以一起进群交流。, M& ^0 F& v* f, O

oqssrsmovgu64067889745.png

oqssrsmovgu64067889745.png

; Y" z/ k0 V+ w. t欢迎你添加我的微信,我拉你进技术交流群。此外,我也会经常在微信上分享一些计算机学习经验以及工作体验,还有一些内推机会。
$ c; R- }  a* N9 A, s# J. e! r; S+ B: D8 T7 M) ^9 `

wtf1xod0vje64067889845.png

wtf1xod0vje64067889845.png
1 a) i# S  q3 p* Z  n
加个微信,打开另一扇窗
1 e$ Z! p9 b) i! G% K3 Q

s2gr0mtr35q64067889945.gif

s2gr0mtr35q64067889945.gif
回复

使用道具 举报

发表回复

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则


联系客服 关注微信 下载APP 返回顶部 返回列表