精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

反問面試官:如何實現集群內選主

開發 前端
本文主要演示了一個簡易的多Server的選主過程,以下代碼是一個簡單的基于Netty實現的集群選舉過程的示例。

面試官經常喜歡問什么zookeeper選主原理、什么CAP理論、什么數據一致性。經常都被問煩了,我就想問問面試官,你自己還會實現一個簡單的集群內選主呢?估計大部分面試官自己也寫不出來。

本篇使用 Java 和 Netty 實現簡單的集群選主過程的示例。

這個示例展示了多個節點通過投票選舉一個新的主節點的過程。Netty 用于節點間的通信,而每個節點則負責發起和響應選舉消息。

集群選主流程

1.選主流程

咱們且不說zookeeper如何選主,單說人類選主,也是采用少數服從多數的原則。人類選主時,中間會經歷如下過程:

  • 如果我沒有熟悉的或者沒找到能力比我強的,首先投給自己一票。
  • 隨著時間推移,可能后面的人介紹了各自的特點和實力,那我可能會改投給別人。
  • 所有人將投票信息放入到統計箱中。
  • 最終票數最多的人是領導者。

同樣的,zookeeper在選主時,也是這樣的流程。假設有五大服務器:

  • 服務器1先給自身投票
  • 后續起來的服務器2也會投自身一票,然后服務器1觀察到服務器2的id比較大,則會改投服務器2
  • 后續起來的服務器3也會投自身一票,然后服務1和服務器2發現服務器3的id比較大,則都會改投服務器3。服務器3被確定為領導者。
  • 服務器4起來后也會投自身一票,然后發現服務器3已經有3票了,立馬改投服務器3。
  • 服務器5與服務器4的操作一樣。

2.選主協議

在選主過程中采用的是超過半數的協議。在選主過程中,會需要如下幾類消息:

  • 投票請求:節點發出自己的投票請求。
  • 接受投票:其余節點作出判斷,如果覺得id較大,則接受投票。
  • 選舉勝出:當選主節點后,廣播勝出消息。

代碼實現

下面模擬3個節點的選主過程,核心步驟如下:

(1) 定義消息類型、消息對象、節點信息

public enum MessageType {
        VOTE_REQUEST, // 投票請求
        VOTE,         // 投票
        ELECTED       // 選舉完成后的勝出消息
}
    
public class ElectionMessage implements Serializable {
    private MessageType type;
    private int nodeId;   // 節點ID
    private long zxId;    // ZXID:類似于ZooKeeper中的邏輯時鐘,用于比較
    private int voteFor;  // 投票給的節點ID
}

public class ElectionNode {
    private int nodeId; // 當前節點ID
    private long zxId;  // 當前節點的ZXID
    private volatile int leaderId; // 當前選舉的Leader ID
    private String host;
    private int port;
    private ConcurrentHashMap<Integer, Integer> voteMap = new ConcurrentHashMap<>(); // 此節點對每個節點的投票情況
    private int totalNodes; // 集群總節點數
}

(2) 每個節點利用Netty啟動Server

public void start() throws Exception {
        EventLoopGroup bossGroup = new NioEventLoopGroup();
        EventLoopGroup workerGroup = new NioEventLoopGroup();
        try {
            ServerBootstrap serverBootstrap = new ServerBootstrap();
            serverBootstrap.group(bossGroup, workerGroup)
                    .channel(NioServerSocketChannel.class)
                    .childHandler(new ChannelInitializer<SocketChannel>() {
                        @Override
                        protected void initChannel(SocketChannel ch) {
                            ch.pipeline().addLast(
                                    new ObjectDecoder(ClassResolvers.cacheDisabled(null)),
                                    new ObjectEncoder(),
                                    new ElectionHandler(ElectionNode.this));
                        }
                    });

            ChannelFuture future = serverBootstrap.bind(port).sync();
            System.out.println("Node " + nodeId + " started on port " + port);

            // 啟動后開始選舉過程
            startElection();
//            future.channel().closeFuture().sync();


        } catch (Exception e) {

        } finally {
//            bossGroup.shutdownGracefully();
//            workerGroup.shutdownGracefully();
        }
    }

(3) 啟動后利用Netty發送投票請求

public void sendVoteRequest(String targetHost, int targetPort) {
        EventLoopGroup group = new NioEventLoopGroup();
        try {
            Bootstrap bootstrap = new Bootstrap();
            bootstrap.group(group)
                    .channel(NioSocketChannel.class)
                    .handler(new ChannelInitializer<SocketChannel>() {
                        @Override
                        protected void initChannel(SocketChannel ch) {
                            ch.pipeline().addLast(
                                    new ObjectDecoder(ClassResolvers.cacheDisabled(null)),
                                    new ObjectEncoder(),
                                    new ElectionHandler(ElectionNode.this));
                        }
                    });

            ChannelFuture future = bootstrap.connect(targetHost, targetPort).sync();
            ElectionMessage voteRequest = new ElectionMessage(ElectionMessage.MessageType.VOTE_REQUEST, nodeId, zxId, nodeId);
            future.channel().writeAndFlush(voteRequest);
//            future.channel().closeFuture().sync();
        } catch (Exception e) {

        } finally {
//            group.shutdownGracefully();
        }
    }

(4) 節點接受到投票請求后,做相關處理

節點在收到消息后,做相關邏輯處理:處理投票請求、處理確認投票、處理選主結果。

處理投票請求:判斷是否是否接受投票信息。只有在主節點沒確定并且zxId較大時,才發送投票消息。如果接受了投票請求的話,則更新本地的投票邏輯,然后給投票節點發送接受投票的消息

處理確認投票:如果投票消息被接受了,則更新本地的投票邏輯。

處理選主結果:如果收到了選主結果的消息,則更新本地的主節點。

public class ElectionHandler extends ChannelInboundHandlerAdapter {
    private final ElectionNode node;

    public ElectionHandler(ElectionNode node) {
        this.node = node;
    }

    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg) {
        ElectionMessage electionMessage = (ElectionMessage) msg;
        System.out.println("Node " + node.getNodeId() + " received: " + electionMessage);

        if (electionMessage.getType() == ElectionMessage.MessageType.VOTE_REQUEST) {
            // 判斷是否是否接受投票信息。只有在主節點沒確定并且zxId較大時,才發送投票消息
            // 如果接受了投票請求的話,則更新本地的投票邏輯,然后給投票節點發送接受投票的消息
            if (electionMessage.getZxId() >= node.getZxId() && node.getLeaderId() == 0) {
                node.receiveVote(electionMessage.getNodeId());
                ElectionMessage voteMessage = new ElectionMessage(ElectionMessage.MessageType.VOTE, electionMessage.getNodeId(), electionMessage.getZxId(), electionMessage.getNodeId());
                ctx.writeAndFlush(voteMessage);
            } else {
                // 如果已經確定主節點了,直接發送ELECTED消息
                sendLeaderInfo(ctx);
            }
        } else if (electionMessage.getType() == ElectionMessage.MessageType.VOTE) {
            // 如果投票消息被接受了,則更新本地的投票邏輯。
            if (electionMessage.getZxId() >= node.getZxId() && node.getLeaderId() == 0) {
                node.receiveVote(electionMessage.getNodeId());
            } else {
                // 如果已經確定主節點了,直接發送ELECTED消息
                sendLeaderInfo(ctx);
            }
        } else if (electionMessage.getType() == ElectionMessage.MessageType.ELECTED) {
            if (node.getLeaderId() == 0) {
                node.setLeaderId(electionMessage.getVoteFor());
            }
        }
    }

(5) 接受別的節點的投票

這里是比較關鍵的一步,當確定接受某個節點時,則更新本地的投票數,然后判斷投票數是否超過半數,超過半數則確定主節點。同時,再將主節點廣播出去。

此時,其余節點接收到選主確認的消息后,都會更新自己的本地的主節點信息。

public void receiveVote(int nodeId) {
    voteMap.merge(nodeId, 1, Integer::sum);
    // 比較出votes里值,取出最大的那個對應的key
    int currentVotes = voteMap.values().stream().max(Integer::compareTo).get();

    if (currentVotes > totalNodes / 2 && leaderId == 0) {
        setLeaderId(nodeId);
        broadcastElected();
    }
}

(6) 廣播選主結果

/**
 * 廣播選舉結果
 */
private void broadcastElected() {
    for (int i = 1; i <= totalNodes; i++) {
        if (i != nodeId) {
            sendElectedMessage(host, 9000 + i);
        }
    }
}

/**
 * 發送選舉結果
 *
 * @param targetHost
 * @param targetPort
 */
public void sendElectedMessage(String targetHost, int targetPort) {
    EventLoopGroup group = new NioEventLoopGroup();
    try {
        Bootstrap bootstrap = new Bootstrap();
        bootstrap.group(group)
                .channel(NioSocketChannel.class)
                .handler(new ChannelInitializer<SocketChannel>() {
                    @Override
                    protected void initChannel(SocketChannel ch) {
                        ch.pipeline().addLast(
                                new ObjectDecoder(ClassResolvers.cacheDisabled(null)),
                                new ObjectEncoder(),
                                new ElectionHandler(ElectionNode.this));
                    }
                });

        ChannelFuture future = bootstrap.connect(targetHost, targetPort).sync();
        ElectionMessage electedMessage = new ElectionMessage(ElectionMessage.MessageType.ELECTED, leaderId, zxId, leaderId);
        future.channel().writeAndFlush(electedMessage);
//            future.channel().closeFuture().sync();
    } catch (Exception e) {

    } finally {
//            group.shutdownGracefully();
    }
}

(7) 完整代碼

完整代碼:https://gitee.com/yclxiao/specialty/blob/master/javacore/src/main/java/com/ycl/election/ElectionHandler.java

總結

本文主要演示了一個簡易的多Server的選主過程,以上代碼是一個簡單的基于Netty實現的集群選舉過程的示例。在實際場景中,選舉邏輯遠比這個復雜,需要處理更多的網絡異常、重復消息、并發問題等。

責任編輯:趙寧寧 來源: 程序員半支煙
相關推薦

2024-04-03 00:00:00

Redis集群代碼

2024-09-24 10:28:22

2024-03-20 15:12:59

KafkaES中間件

2024-02-20 14:10:55

系統緩存冗余

2024-09-11 22:51:19

線程通訊Object

2023-11-20 10:09:59

2024-01-26 13:16:00

RabbitMQ延遲隊列docker

2024-01-19 14:03:59

Redis緩存系統Spring

2024-04-09 10:40:04

2024-10-22 16:39:07

2015-08-13 10:29:12

面試面試官

2021-12-15 06:58:13

List 集合LinkedHashS

2021-05-20 08:54:16

Go面向對象

2024-02-04 10:08:34

2024-12-25 15:44:15

2021-10-26 10:29:45

掃碼登錄功能

2021-05-20 08:34:03

CDN原理網絡

2024-09-09 15:09:30

2021-05-19 06:07:21

CSS 斜線效果技巧

2023-02-16 08:10:40

死鎖線程
點贊
收藏

51CTO技術棧公眾號

亚洲日本激情| 欧洲精品一区二区三区久久| 天天操天天爽天天射| 国产情侣av在线| 九九久久婷婷| 亚洲电影激情视频网站| 中文字幕av一区| 成人免费观看cn| 亚洲第一视频在线播放| 亚洲精品极品少妇16p| 91精品福利视频| 极品尤物一区二区三区| 激情综合网五月天| 国产精品麻豆| 亚洲视频中文字幕| 国产日韩精品入口| 国产黄色大片免费看| 中文在线免费视频| 91在线一区二区三区| 久久91亚洲精品中文字幕奶水| 三级在线视频观看| 久草在线资源站资源站| 欧美 日韩 国产 一区| 欧美亚洲国产一区在线观看网站| 久久99精品久久久久久久久久| 久草成人在线视频| 日韩欧美视频| 欧美日韩亚洲综合| 亚洲精品白虎| 中文字幕av片| 区一区二视频| 欧美日韩国产高清一区二区三区 | 国产亚洲自拍一区| 91精品国产高清久久久久久| 无码人妻一区二区三区在线| a视频在线观看| 国产一区二区三区黄视频 | 美女亚洲精品| 成人免费毛片视频| 国产精品欧美日韩一区| 欧美这里有精品| 亚洲一区二区三区午夜| 夜夜爽8888| 在线精品国产| 亚洲国产福利在线| 欧美一级在线看| 精品电影在线| 久久国产福利国产秒拍| 另类专区欧美制服同性| 最新国产精品自拍| 国产成人视屏| 7777精品伊人久久久大香线蕉最新版| 视频一区二区视频| 亚洲产国偷v产偷v自拍涩爱| 精品一区二区免费视频| 国产精品福利网站| 亚洲一二三四视频| 亚州一区二区| 欧美日韩激情网| 午夜精品电影在线观看| 国产精品自偷自拍| 一本久道久久综合婷婷鲸鱼| 国产亚洲精品日韩| 初高中福利视频网站| 蜜臀久久精品| 中文字幕亚洲一区二区av在线 | 欧美精品在线免费观看| 日本一卡二卡在线| www.26天天久久天堂| 亚洲精品视频自拍| 日本一区精品| 精品人妻少妇嫩草av无码专区| 亚洲综合不卡| 久久视频这里只有精品| 日本一级二级视频| 国产videos久久| 夜夜躁日日躁狠狠久久88av| 午夜男人的天堂| 欧美a一欧美| 777a∨成人精品桃花网| 真实乱偷全部视频| 老司机精品在线| 欧美日韩国产一级片| 99精品视频国产| 三级成人在线| 五月婷婷色综合| 欧美少妇在线观看| 国产爆初菊在线观看免费视频网站| 国产呦精品一区二区三区网站| 92看片淫黄大片看国产片| www.日韩一区| 亚洲乱亚洲高清| 日韩免费观看网站| 日韩乱码一区二区| 欧美freesex交免费视频| 久久久久久一区二区三区 | 最好看的中文字幕久久| 精品欧美一区二区三区久久久| 日本成人一区| 成人av手机在线观看| 成人美女av在线直播| 中文字幕av网站| 国产99久久久久| 亚洲aⅴ男人的天堂在线观看| 丰满熟妇乱又伦| 国产在线精品一区二区三区不卡| 不卡一卡2卡3卡4卡精品在| 亚洲天堂视频网| 国产成人在线色| 99久久精品久久久久久ai换脸| 熟妇高潮一区二区高潮| 丁香六月久久综合狠狠色| 91在线观看免费网站| 国产影视一区二区| 久久国产人妖系列| 国产一区深夜福利| 欧美在线观看在线观看| 亚洲精品网站在线观看| 不卡av免费在线| 国产极品模特精品一二| 亚洲国产成人久久综合| 日本免费网站视频| 日本久久黄色| 538国产精品一区二区在线| av资源免费观看| 久久激情久久| 国产福利精品在线| 中文在线观看av| 91在线视频免费91| 国产911在线观看| 精品久久在线| 91精品国产麻豆国产自产在线 | 粉嫩91精品久久久久久久99蜜桃| 337p日本欧洲亚洲大胆色噜噜| 国产伦理在线观看| 久久精品亚洲成在人线av网址| www.欧美免费| 九九免费精品视频| 久久99精品久久久久久| 欧美一区二区福利| 依依综合在线| 欧美日韩国产经典色站一区二区三区| 精品无码一区二区三区| 日韩欧美伦理| 国产精品免费看久久久香蕉| 国产美女无遮挡永久免费| 国产日韩欧美综合在线| wwwxxx黄色片| 91麻豆精品国产综合久久久 | 撸视在线观看免费视频| 中文字幕+乱码+中文字幕一区| 亚洲日本欧美在线| 精品成人免费一区二区在线播放| 欧美日韩视频专区在线播放| 高潮毛片无遮挡| 久久激情婷婷| 色综合视频二区偷拍在线| av亚洲一区二区三区| 亚洲视频在线观看免费| 日本高清不卡免费| 国产欧美日韩亚洲一区二区三区| 国产精品久久久久影院日本| 不卡在线视频| 亚洲国产精品影院| 亚洲一区二区在线免费| 国产日韩欧美高清免费| 蜜桃传媒视频麻豆第一区免费观看 | 国产成人福利av| 97久久精品人搡人人玩| 中文字幕在线网站| 中文字幕在线不卡视频| 妖精视频在线观看| 一区二区国产在线观看| 日本精品一区二区三区高清 久久| 韩国成人在线| 久久精品一偷一偷国产| 久久久久亚洲av成人毛片韩| 国产在线不卡一区| 无码熟妇人妻av在线电影| 国产精品久久久久久久久久齐齐| 色一情一乱一区二区| 91在线视频在线观看| 国产偷v国产偷v亚洲高清| 美女在线视频一区二区| 亚洲激情77| 欧美精品久久久久久久免费观看| 性插视频在线观看| 亚洲一线二线三线视频| 特级西西444www| 亚洲黄页一区| 午夜免费电影一区在线观看| 国产亚洲久久| 国产va免费精品高清在线| 巨大荫蒂视频欧美另类大| 欧美无砖砖区免费| 久久伊人成人网| 国产成人一级电影| 无码人妻h动漫| 中文精品久久| 欧美高清视频一区| 超级白嫩亚洲国产第一| 日韩免费一区二区| 欧美日韩综合一区二区| 国产九色精品成人porny| 正义之心1992免费观看全集完整版| 日本精品不卡| 亚洲免费精彩视频| 国产午夜免费福利| 中文字幕在线一区| www.超碰97| 久久在线精品| 日韩福利在线| 免费成人美女女| 久久久久久久一区二区| 黄色在线观看网站| 亚洲人成网站在线播| 欧美性猛交 xxxx| 日韩电影一二三区| 国产一区二区三区日韩精品 | 亚洲精品视频专区| 91国偷自产一区二区开放时间| 欧美在线视频第一页| 国产日韩欧美不卡| 中出视频在线观看| 狠狠干综合网| 国产免费一区二区| 天堂av中文在线观看| 欧美不卡视频一区发布| av在线日韩国产精品| 亚洲精品乱码久久久久久按摩观| 日韩av电影网| 亚洲欧美另类久久久精品2019| 成人激情五月天| 久久人人97超碰com| 亚洲综合欧美在线| 久久人人超碰| 六月丁香激情网| 一区二区三区四区五区精品视频 | 成人91免费视频| 成人在线视频www| 国产日韩在线视频| 久久麻豆视频| 成人免费视频在线观看超级碰| 精品亚洲a∨| 91精品国产自产在线老师啪| 国产成人福利夜色影视| 国产精品视频一区国模私拍| 在线日本中文字幕| 欧美一区二区高清| 国产精品久久无码一三区| 欧美日韩国产三级| 怡春院在线视频| 欧美久久久久久久久中文字幕| 国产性猛交普通话对白| 一级精品视频在线观看宜春院| 风间由美一二三区av片| 97久久超碰精品国产| a天堂视频在线观看| 91在线小视频| 中文字幕第20页| 成人一区在线观看| 中文在线观看免费视频| 麻豆精品一区二区综合av| 麻豆一区二区三区视频| 欧美在线1区| 国产又粗又猛又爽又黄的网站| 国产欧美一区| 亚洲精品国产精品国自产| 91精品国产91久久综合| japanese在线播放| 在线观看一区视频| 亚洲免费视频播放| 一呦二呦三呦国产精品| 先锋影音亚洲资源| 自拍欧美日韩| www.99热这里只有精品| 老牛嫩草一区二区三区日本 | 国产成人免费视| 特级西西人体4444xxxx| 欧美国产精品中文字幕| 亚洲国产精品免费在线观看| 亚洲黄色片在线观看| 日本一区二区视频在线播放| 国产精品萝li| 少妇太紧太爽又黄又硬又爽小说| 成人网男人的天堂| 成人网站免费观看| 中文字幕一区二区三区四区不卡| 国产在线成人精品午夜| 亚洲精品国产第一综合99久久 | 国产精品一区二区你懂得| 少妇高潮一区二区三区99| 国产国产精品人在线视| 成人黄色理论片| 久久久久久久久久久久久久久久av| 欧美日韩一区二区三区视频播放| 久久久久资源| 97视频热人人精品免费| 污视频在线免费观看一区二区三区| 日韩影院二区| jizzjizz国产精品喷水| 久久99精品久久久久久动态图| 亚洲一区二区三区四区av| 国产性做久久久久久| 精品视频久久久久| 精品视频1区2区3区| 西西人体44www大胆无码| 久久色精品视频| 97久久香蕉国产线看观看| 91精品网站| 97一区二区国产好的精华液| 精品高清美女精品国产区| 国产乱人伦偷精品视频不卡| 国产日韩亚洲欧美| 国产视频一区二| 日本在线一区| 亚洲二区精品| 中文字幕在线视频一区二区| 蜜桃在线一区二区三区| 男人搞女人网站| av电影在线观看一区| 极品白嫩丰满美女无套| 亚洲素人一区二区| 亚洲永久精品视频| 亚洲欧美日韩在线一区| 91超碰在线播放| 91精品成人久久| 国产一区二区三区亚洲综合| 亚洲精品成人a8198a| 可以看av的网站久久看| 五月婷婷综合在线观看| 91丝袜呻吟高潮美腿白嫩在线观看| 强伦人妻一区二区三区| 亚洲一二三专区| www精品国产| 欧美成人午夜视频| 99视频有精品高清视频| 亚洲精品在线视频观看| 日本成人在线不卡视频| 爽爽爽在线观看| av激情亚洲男人天堂| 国产亚洲精品久久久久久无几年桃| 日韩三级在线观看| 日韩二区三区| 97超碰色婷婷| 亚洲+变态+欧美+另类+精品| 亚洲韩国在线| 日本女优在线视频一区二区| 婷婷色一区二区三区| 欧亚洲嫩模精品一区三区| 高清福利在线观看| 国产精品亚发布| 99re66热这里只有精品8| 欧美视频亚洲图片| 一个色综合av| 日本高清视频在线| 欧美一区二区三区艳史| 国产精品一区二区99| 黄色一级二级三级| 丰满白嫩尤物一区二区| 亚洲精品在线观看av| 亚洲成人久久一区| 欧美激情20| 天堂社区 天堂综合网 天堂资源最新版 | 性做爰过程免费播放| 国产成人免费视频网站| 日韩欧美高清在线观看| 在线不卡中文字幕| 日韩激情av| 国产欧美日韩精品专区| 97久久夜色精品国产| 精品久久久久久无码人妻| 欧美性猛交xxx| 精品女同一区二区三区| 国内成人精品一区| 日韩视频1区| a在线视频观看| 国产精品污www在线观看| 国产成人精品无码高潮| 色偷偷亚洲男人天堂| 国产一区二区av在线| 国产日本在线播放| 国产精品视频看| 黑人操亚洲女人| 国产精品色午夜在线观看| 亚洲调教视频在线观看| 这里只有久久精品| 67194成人在线观看| 欧美私密网站| 公共露出暴露狂另类av| 97久久久精品综合88久久| 一区二区视频免费观看| 欧美肥老妇视频| 亚洲一区二区三区免费| 一道本在线观看视频| www.欧美.com| 中文字幕一区二区三区免费看| 欧美精品videos| 大片网站久久| www.免费av|