Raft Lab3A

Lab3 需要在 Raft 层上实现一个 fault-tolerant key-value service,满足强一致性,也就是线性一致性 (Linearizable Consistency)。线性一致性保证整个系统看起来好像只有一个副本,其中所有的操作都是原子性的。简单地说,线性一致性系统的读写操作有以下特征:

  • 读写并不能瞬间完成,而是在一个时间段内进行。
  • 读在写开始前完成,读到的一定是旧值;读在写完成之后开始,读到的一定是新值。(读写操作无重合部分)
  • 读写并发进行,即有重合部分时,既可能读到新值,也可能读到旧值。
  • 一旦有一个客户端读取到了新值,那么之后的客户端一定也都会读取到新值。

先来看看整个系统的架构:

img

整个 KV Service 由多个 Server 组成,每个 Server 包含一个 State Machine,具体在 lab 中是一个 KV 数据库;Server 还包含一个 Raft 节点。需要注意的是,各 Server 之间并不会直接通信,而是靠其 Raft 节点进行通信。

整个系统的理想运行流程是:

  • Client 通过 RPC 向 KV Service 发送请求,例如 Put(x,1)
  • KV Service 将请求转发给当前拥有 Leader Raft 节点的 Server
  • Leader Server 将请求包含的 command 传递给 Raft 层
  • Raft 层对 command 进行共识,生成相同的 log replica
  • 在达成共识后,Raft 层将 command Apply 回 Server
  • Server 收到 Raft 层的 Apply 后,将 command 应用到状态机,即状态机此时状态为 {x: 1}
  • 成功应用至状态机后,Server 对 Client 的 RPC 进行回复,返回结果和错误码。

当系统能够正常运行时,一切看起来都很清晰美好。但是一旦出现问题,如节点挂掉、RPC丢包、网络分区等等,情况就变得比较复杂了。

Implement

在 Lab3 中,我们主要需要实现的部分是 Client 、 Server 和 Server KV Database。

KV Database

在 lab 中,KV 数据库并不是主要内容。因此直接用 Hashmap 模拟即可。键值均为 String。

type kvdb struct {
    m  map[string]string
}

func (db *kvdb) put(key string, value string) {
    db.m[key] = value
}

func (db *kvdb) append(key string, value string) {
    db.m[key] += value
}

func (db *kvdb) get(key string) (string, Err) {
    if _, ok := db.m[key]; !ok {
        return "", ErrNoKey
    }
    return db.m[key], OK
}

Client

Client 也相对简单。Client 可向 Server 发送三种不同的 RPC:Put(key,value)Append(key,arg)Get(key)。Put 和 Append 均为写请求,Get 为读请求。

一开始,Client 并不知道 Leader Server 是哪台 Server。Client 可向随机一台 Server 发送 RPC 请求。假如请求的 Server 不是当前的 Leader Server,或者由于网络中断、Server Crash 等原因,无法与 Server 取得联系,则无限地尝试更换 Server 重新发送请求,直到请求成功被处理。这里有一个小优化,在得知 Leader Server 后,Client 可以保存 Leader 的 id,避免下次发起请求时又需要随机地选择一台 Server 多次尝试。

type Clerk struct {
    servers []*labrpc.ClientEnd

    seq    int64 // write op index, increasing from 1
    id     int64 // client uuid
    leader int
}

func (ck *Clerk) Get(key string) string {
    args := GetArgs{
        Key:      key,
        ClientId: ck.id,
    }

    i := ck.leader

    defer func() {
        ck.leader = i
    }()

    for {
        reply := GetReply{}
        ok := ck.servers[i].Call("KVServer.Get", &args, &reply)
        if !ok || reply.Err == ErrWrongLeader || reply.Err == ErrTimeout {
            // cannot reach the server, or it's a wrong leader: retry
            i = (i + 1) % len(ck.servers)
            continue
        }

        if reply.Err == ErrNoKey {
            return ""
        }

        return reply.Value
    }
}

func (ck *Clerk) PutAppend(key string, value string, op string) {
    args := PutAppendArgs{
        Key:      key,
        Value:    value,
        Op:       op,
        Seq:      ck.seq,
        ClientId: ck.id,
    }

    i := ck.leader

    defer func() {
        ck.seq++
        ck.leader = i
    }()

    for {
        reply := PutAppendReply{}
        ok := ck.servers[i].Call("KVServer.PutAppend", &args, &reply)
        if !ok || reply.Err == ErrWrongLeader || reply.Err == ErrTimeout {
            // cannot reach the server, or it's a wrong leader: retry
            i = (i + 1) % len(ck.servers)
            continue
        }
        // successfully PutAppend
        return
    }
}

关于 Client 还需要维护的另一些状态,如 id 和 seq。

Server

Server 应该是 Lab3 中最为复杂的部分。我们先讨论一切正常的情况下 Server 的设计。

读写 RPC Handler 在接收到 Client 的请求后,通过调用 raft.Start() 将请求包含的 command 传递到 Raft 层,达成共识。当然,如果当前 Server 不为 Leader,则向 Client 返回 ErrWrongLeader 错误,Client 在收到回复后重新尝试向另一台 Server 发起请求。Raft 层达成共识后,通过 applyCh 通知 Server 该 command 已达成共识。

在一开始,可能会这样设计 Server:

_, _, isLeader := kv.rf.Start(op) // push op to raft layer to reach the agreement
if !isLeader {
    reply.Err = ErrWrongLeader
    return
}
<- applyCh
// agreed! apply to statemachine
kv.apply(op)

对于不出错的单 Client,这样设计似乎没有问题。但如果有多个 Client 并行地向 Server 发起请求时,就显然不能保证从 applyCh 传回的数据恰好是此前提交的 command 了。为了解决这个问题,我们需要在 Server 中对特定的 command 进行等待。如何区分不同的 command?用 command 在 raft log 中的 index 区分即可。Server 需要维护一张 Map,Key 为 index,Value 为 Server 等待此 index 对应 command 的 channel。

type Result struct {
    value string
    err   Err
}

notifyCh := map[int]chan Result

在 Raft 层,command 一定是按序向 applyCh 传输的。为了能够按序将 command 应用至状态机,Server 应启动一后台 goroutine 监听 applyCh,对需要 apply 的 command 进行互斥的处理。同时,这个 goroutine 也负责将 applyCh 传来的信息转发给对应的正在阻塞等待的 RPC Handler:

// 后台goroutine监听applyCh是否有Raft层传回的需要apply的command
func (kv *KVServer) notifier() {
    for !kv.killed() {
        select {
        case msg := <-kv.applyCh:
            op := msg.Command.(Op)
            result := kv.apply(op)  // apply to state machine
            
            index := msg.CommandIndex
            ch := kv.getNotifyCh(index) 
            ch <- result    // notify the blocked server 
        }
    }
}

// 后台goroutine转发apply给RPC处理handler,RPC handler需要向client返回结果
func (kv *KVServer) getNotifyCh(index int) chan Result {
    kv.mu.Lock()
    defer kv.mu.Unlock()
    if _, ok := kv.notifyCh[index]; !ok {
        kv.notifyCh[index] = make(chan Result)
    }
    return kv.notifyCh[index]
}

Server 阻塞的代码改写如下:

index, _, isLeader := kv.rf.Start(op)

if !isLeader {
    reply.Err = ErrWrongLeader
    return
}

ch := kv.getNotifyCh(index)

select {
case result := <-ch:
    // agreed! reply to client 成功,回复客户端
    reply.Value, reply.Err = result.value, result.err
case <-time.After(AGREE_TIMEOUT):
    // too slow response, reply to client and let it retry another server 超时重试
    reply.Value, reply.Err = "", ErrTimeout
}

go func() {
    // asynchronously release notifying channel
    kv.delNotifyCh(index)
}()

需要注意的是,如果 Raft 层长时间无法完成共识 (由于网络分区等原因),不要让 Server 一直阻塞。及时向 Client 返回 Timeout 错误,使其重新选择另一台 Server 重试。

这样一来,多个 Client 并行发送请求的情况似乎也可以应对了。

然而我们要实现的系统有一个重要的性质,fault-tolerant。目前为止,fault 还没有出现。

实际上,Raft 层的各种 fault 我们在 lab2 中已经妥善处理了,因此我们主要需要关注的是 Server 层的 fault。首先不考虑 Server 直接挂掉的情况 (需要在 lab3B 中用 snapshot 解决),那么剩下的就是 Client 和 Server 之间的 RPC 丢失问题了。

假如 Client 向 Server 发送请求,因网络问题 Server 无法接收,这种情况 Server 无需应对 (也无力应对),让 Client 自己慢慢重试就好。比较严重的问题是,Client 发送的请求 Server 成功接收,Server 也将请求中的 command 成功在 Raft 层达成共识并应用至状态机,然而在回复 Client 时出现了问题,RPC 回复丢失。这样就有可能导致一次请求多次重复提交的情况。比如下面一种简单的情况:

  1. Client 向 Server 发送 Append(x, 1) 的请求;
  2. Server 成功接收,Raft 层达成共识,应用至状态机。此时状态机状态 {x: 1};
  3. 由于网络原因,Server 向 Client 返回的结果丢失;
  4. Client 苦苦等待,也没有收到 Server 返回的结果,于是超时重试。绕了一圈后又回到了这个 Server (此 Server 仍为 Leader);
  5. Client 又向 Server 发送 Append(x, 1) 的请求,Server 成功接收,Raft 层达成共识,应用至状态机。此时状态机状态 {x: 11};
  6. 这次 Server 成功向 Client 返回了结果;
  7. Client 成功收到了返回的结果,结束请求。然而原本的 Append(x, 1) 请求,造成了 Append(x, 11) 的后果。

出现这种情况的根本原因是,Raft 层允许同样的 command commit 多次 (Raft 层并不知道这是不是相同的 command,只要有 command 来,就尝试共识),但实际上,同样的 command 只能 apply 一次。这就需要我们在 Server 层对请求进行去重

上面只介绍了 Append 请求的情况,Put 请求也类似。虽然在只有一个 Client 时,Put 请求多次执行不会改变结果,但如果有多个 Client,重复的 Put 请求也可能造成互相覆盖的后果。因此也需要进行去重。

至于 Get 请求,多次重复并不会改变状态机的状态,无需进行去重处理。

说到 Get 请求,在这里小小地偏一下题:

按我们目前的实现,Get/Put/Append 请求均需先推至 Raft 层达成共识,记录在 Raft 层的 Log 中。然而 Get 请求并不会改变系统的状态,记录在 Log 中,对崩溃后回放 Log 恢复数据也没有什么帮助。那么实际上是不是不需要将 Get 请求传入 Raft 层进行共识呢?是的。并且这样会使系统效率更高。那么为什么我们要将 Get 请求也传入 Raft 层呢?这么做实际上是为了简化 KV Service 的实现难度。KV Service 要求我们永远不在 minority 中读取数据,因为这样可能会破坏线性一致性。假如我们不将 Get 传入 Raft 层,直接读取 Leader Server 状态机中的数据,试想下面这种情况:

一共有 5 台 Server。一开始,Server1 为 Leader,Client 发送了一些请求,Raft 成功共识。

  • 此后,Server1、Server2 与 Server3、Server4、Server5 由于网络问题被划分成两个部分。第一部分中,Server1 仍认为自己是 Leader。第二部分中,Server3 成功当选 Leader。
  • Server3 又接收了一些来自 Client 的请求,且在 Server3、Server4、Server5 间达成了共识。

有两个 Client 希望 Get 同一个 key:

  • Client1 首先联系了 Server1,Server1 认为它自己是 Leader (实际已经 outdated),便向 Client1 返回了 outdated value。
  • Client2 首先联系 Server3,Server3 向其返回了 updated value。

这两个 Get 操作间并没有写操作,却读到了不同的数据,违背了线性一致性。

为什么将 Get 传入 Raft 进行共识就可以避免这种错误?依然考虑上述情况:Server1 在接收到 Client1 的 Get 请求后,将其传入 Raft 层试图达成共识。然而 Server1 只能获得 Server2 的响应,无法将 Get 请求同步到大多数节点上,所以迟迟无法达成共识,Server 层也会被长期阻塞。Client1 久久等不到答复,便会更换 Server 重新进行请求,此时就会找到新的 Leader Server3 并成功执行 Get 请求。所以,将 Get 请求一同传入 Raft 层是最简单地避免读取到 minority 数据的方法。

Raft 论文在 session 8 中提到了 read-only operations 等优化,避免将 Get 写入 Log,同时解决了可能获取 outdated 数据的问题。

去重具体的执行方式,就和之前在 Client 中还没有讲到的 id、seq 等变量有关了。

id 是 Client 的 uuid,用于标识不同的 Client,直接用 skeleton code 中的 nrand() 方法生成即可 (测试时开不了那么多 Client,碰撞概率极低,可以凑合当uuid用)。seq 则是 Client 写操作的最大操作数,从 1 开始递增。每当 Client 完成一次写操作,就将 seq 加 1。

Server 端则维护一张 map,用于记录不同 Client 成功应用至状态机的最大 seq 数:

maxSeq := map[int64]int64

在遇到来自 client x 的 y seq 请求时,如果

maxSeq[x] >= y

此时又有新的问题出现了,我们应该在哪里拦截重复的请求,在哪里更新 maxSeq?

我一开始的想法是,直接在 RPC handler 的最开始判断请求是否重复,若是重复请求则直接拦截并返回。并在 RPC handler 返回前更新 maxSeq。然而这种处理方法存在问题。试想如下情况:

  • Client 首先向 Leader Server1 发起 Append 请求。Server1 成功完成共识并将请求应用至状态机,也更新了 maxSeq。但在返回时 RPC 结果丢失。
  • 此时,恰好 Server1 由于崩溃或网络隔离等原因,失去 Leader 身份,Server2 当选 Leader。
  • 由于 RPC 结果丢失,Client 长时间得不到响应,便尝试更换 Server 重新发起请求。
  • Client 向 Server2 发起了同样的 Append 请求。由于 Server2 的 maxSeq 中并没有此 Client 此 Seq 的信息 (上次仅是存储在了 Server1 的 maxSeq),于是 Server2 再次执行了请求。也导致了一次请求多次应用的后果。

出现这种情况的根本原因是 Server 并不会直接联系,不同 Server 的 maxSeq 无法共享,因此在 Client 切换 Server 提交重复请求时,Server 无法察觉。

解决方法也比较简单:在 command 达成共识后,将 command 应用至状态机前,对 command 进行去重。并在成功应用 command 后,更新 maxSeq。**即 maxSeq 实际上由状态机维护。这样做能成功的原因是,不同的 Server 通过 Raft 层的交流,间接地共享了 maxSeq。**所有请求都需要先尝试应用至状态机,而状态机维护的 maxSeq 恰好可以拦截试图应用的重复请求。

func (kv *KVServer) apply(op Op) Result {
    result := Result{}

    if op.T == "Get" {
        result.value, result.err = kv.db.get(op.Key)
    } else if op.T == "Put" {
        if kv.maxSeq[op.ClientId] < op.Seq {
            kv.db.put(op.Key, op.Value)
            kv.maxSeq[op.ClientId] = op.Seq
        }
        result.err = OK
    } else {
        if kv.maxSeq[op.ClientId] < op.Seq {
            kv.db.append(op.Key, op.Value)
            kv.maxSeq[op.ClientId] = op.Seq
        }
        result.err = OK
    }

    return result
}

到此为止,似乎我们的 KV Service 已经完美无缺了。当时我就是这么认为的,然而它还存在两个逻辑上的小问题 TAT

我们用来转发 applyCh 信息的 notifier 协程是这样的:

func (kv *KVServer) notifier() {
    for !kv.killed() {
        select {
        case msg := <-kv.applyCh:
            op := msg.Command.(Op)
            result := kv.apply(op)  // apply to state machine
            
            index := msg.CommandIndex
            ch := kv.getNotifyCh(index) 
            ch <- result    // notify the blocked server 
        }
    }
}

需要意识到的是,不是所有 Server 都是 Leader 节点,Follower 节点也会通过 applyCh 向 Server 层转递需要 apply 至状态机的数据。此时 Server 层并没有 RPC Handler 在等待 applyCh 的数据。如果我们仍尝试获取对应的 notifyCh 并转发数据,则会造成 notifier 的无限阻塞。改写如下:

func (kv *KVServer) notifier() {
    for !kv.killed() {
        select {
        case msg := <-kv.applyCh:
            op := msg.Command.(Op)
            result := kv.apply(op)  // apply to state machine
            
            if _, isLeader := kv.rf.GetState(); !isLeader {
                continue
            }
            index := msg.CommandIndex
            ch := kv.getNotifyCh(index) 
            ch <- result    // notify the blocked server 
        }
    }
}

Server 不为 Leader 的情况已经解决。假如 Server 当前是 Leader,有没有可能部分 applyCh 传来的数据也无需转发呢?也有可能,最简单的情况就是新当选的 Leader 的 Log 中还存在已提交未应用的 command。将这个 command 传入 Server 层后,按照上面的写法,也会尝试向并不存在的 RPC Handler 转发数据并造成阻塞。这种情况解决起来也比较简单,不属于当前 term 的 command 无需转发,直接给状态机应用就可以了

func (kv *KVServer) notifier() {
    for !kv.killed() {
        select {
        case msg := <-kv.applyCh:
            op := msg.Command.(Op)
            result := kv.apply(op)

            if term, isLeader := kv.rf.GetState(); !isLeader || term != msg.CommandTerm {
                continue
            }

            index := msg.CommandIndex
            ch := kv.getNotifyCh(index)
            ch <- result
        }
    }
}
t {
        case msg := <-kv.applyCh:
            op := msg.Command.(Op)
            result := kv.apply(op)

            if term, isLeader := kv.rf.GetState(); !isLeader || term != msg.CommandTerm {
                continue
            }

            index := msg.CommandIndex
            ch := kv.getNotifyCh(index)
            ch <- result
        }
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/296208.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Unity之预制体与变体

PS:不用说了&#xff0c;我在写博客就是在摸鱼 一、预制体 不知道大家小时候有没有看过火影&#xff0c;记得剧情最开始的时候水木哄骗鸣人去偷封印之书&#xff0c;反而让鸣人学会了多重影分身之术&#xff1a; 好了&#xff0c;小编绞尽脑子终于想好怎么向大家介绍预制体了&a…

高级分布式系统-第3讲 网络与网络互联

万维网的诞生 1957年10月4日&#xff0c; 苏联发射了人类第一颗人造卫星—斯普特尼克一号 美国政府震惊不已。 他们认为&#xff0c; 在日趋激烈的冷战对抗中&#xff0c; 自己已经全面落后于苏联。 为了扭转这一局面&#xff0c; 美国国防部很快于1958 年 2 月组建了一个神秘…

红队专题-Web安全/渗透测试-文件上传/下载/包含

文件上传/下载/包含 招募六边形战士队员利用目录穿越反弹SHELL实战测试2.2 提交报文修改检测3.2 文件内容检测绕过完整文件结构 检测 第四章&#xff1a;解析漏洞第一节 常见解析漏洞iis/nginx php fastcgi 取值错误 解析漏洞 &#xff08;配置错误&#xff09;nginx 文件名逻…

2024年,Web3技术的“iPhone时刻”会出现吗?

出品&#xff5c;欧科云链研究院 关于2024年的Web3市场&#xff0c;大多数人目前是持乐观态度的。随着宏观政策稳定&#xff0c;美国和香港地区不断释放加密资产现货ETF的利好&#xff0c;叠加铭文热潮及减半周期临近&#xff0c;市场对Web3及加密资产的兴趣正持续上涨。年后的…

项目从npm迁移到pnpm

场景如下&#xff1a;在安装Vue3时默认为使用Npm安装&#xff0c;如图所示&#xff1a; 安装完后项目就包含了基于NPM的node_modules、package.json&#xff0c;以及package-lock.json 如果想使用pnpm去安装依赖项的话&#xff0c;可以通过如下几个步骤实现&#xff1a; ①删…

UICollection Compositional Layout全详解

本文字数&#xff1a;8325字 预计阅读时间&#xff1a;45分钟 01 Collection View Layout全详解 UICollectionView在iOS中是构建复杂布局的强大工具。iOS13中引入的 UICollectionViewCompositionalLayout为创建自定义布局提供了全新的可能性。本文将深入探讨Compositional Lay…

复试 || 就业day04(2024.01.05)项目一

文章目录 前言线性回归房价预测加载数据数据查看数据拆分数据建模模型的验证、应用模型的评估 总结 前言 &#x1f4ab;你好&#xff0c;我是辰chen&#xff0c;本文旨在准备考研复试或就业 &#x1f4ab;本文内容来自某机构网课&#xff0c;是我为复试准备的第一个项目 &#…

JAVA中小型医院信息管理系统源码 医院系统源码

开发框架&#xff1a;SpringBootJpathymeleaf 搭建环境&#xff1a;jdk1.8idea/eclipsemaven3mysql5.6 基于SpringBoot的中小型医院信息管理系统&#xff0c;做的比较粗糙&#xff0c;但也实现了部分核心功能。 就诊卡提供了手动和读卡两种方式录入&#xff0c;其中IC读卡器使用…

Python(30):非对称加密算法RSA的使用(openssl生成RSA公私钥对)

Python(30)&#xff1a;非对称加密算法RSA的使用(openssl生成RSA公私钥对) 1、openssl生成RSA公私钥对 1.1、生成RSA公私钥对命令 [rootloaclhost ~]# openssl OpenSSL> genrsa -out rsa_private_key.pem 1024 Generating RSA private key, 1024 bit long modulus .. ...…

PN协议下,上位机如何通过RJ45口远程控制PLC?

在实际系统中&#xff0c;车间里分布多台PLC&#xff0c;需要用上位机软件集中控制。通常所有设备距离在几十米到上百米不等。在有通讯需求的时候&#xff0c;如果布线的话&#xff0c;工程量较大且不美观&#xff0c;这种情况下比较适合采用无线通信方式。 本方案以组态王和2…

【AI视野·今日Robot 机器人论文速览 第六十八期】Tue, 2 Jan 2024

AI视野今日CS.Robotics 机器人学论文速览 Tue, 2 Jan 2024 Totally 12 papers &#x1f449;上期速览✈更多精彩请移步主页 Daily Robotics Papers Edge Computing based Human-Robot Cognitive Fusion: A Medical Case Study in the Autism Spectrum Disorder Therapy Author…

【Nginx】#安装配置及应用场景使用说明

文章目录 常见公网DNS服务1.Nginx的安装版本区别解压文件1.1编译安装1.2 启动安装1.3 安装成系统服务 2.运行原理3.Nginx配置及应用场景3.1最小配置worker_processesworker_connectionsinclude mime.types;default_type application/octet-stream;sendfifile on;keepalive_time…

云原生战专题 | 深入浅出分析云原生微服务的技术结构和架构设计

深入浅出分析云原生微服务的技术结构和架构设计 云原生容器技术背景容器编排Kubernetes控制平面的四大组件Kubernetes在容器编排中的设计要点 云原生微服务典型架构第一代微服务架构第二代微服务架构第三代微服务架构第四代微服务架构 未来的云原生架构 — Serverless 云原生容…

gitlab 配置 二

一 环境说明 群晖Nas DS418DELL XPS serverGitlab ce 二 需要实现的功能 外网可以访问&#xff0c;gitlab使用https的方式访问。wiki issue 等都可以上传图片和附件。 三 操作步骤 因为群晖上有证书&#xff0c;并且由群晖做转发功能。因此证书上&#xff0c;采用群晖的证书…

Linux LVM逻辑卷

一、LVM的定义 LVM 是 Logical Volume Manager 的简称&#xff0c;译为中文就是逻辑卷管理。它是 Linux 下对硬盘分区的一种管理机制。LVM 适合于管理大存储设备&#xff0c;并允许用户动态调整文件系统的大小。此外&#xff0c;LVM 的快照功能可以帮助我们快速备份数据。LVM 为…

聚观早报 |谷歌起草“机器人宪法”;极越与福耀集团达成合作

聚观早报每日整理最值得关注的行业重点事件&#xff0c;帮助大家及时了解最新行业动态&#xff0c;每日读报&#xff0c;就读聚观365资讯简报。 整理丨Cutie 1月6日消息 谷歌起草“机器人宪法” 极越与福耀集团达成合作 三星电子宣布与现代汽车集团合作 OpenAI下周将推GP…

【elfboard linux 开发板】9. 虚拟机扩容和内核编译

1. 虚拟机扩容 需要将虚拟机的快照全都删除&#xff0c;并且将运行的系统关机点击扩展&#xff0c;改为需要的磁盘大小安装gparted工具 sudo apt-get install gparted 如果报错&#xff0c;则按照出错内容修改&#xff0c;一般是出现下载错误&#xff0c;可以使用下列命令&…

05、Kafka ------ CMAK 各个功能的作用解释(主题和分区 详解,用命令行和图形界面创建主题和查看主题)

目录 CMAK 各个功能的作用解释&#xff08;主题&#xff09;★ 主题★ 分区★ 创建主题&#xff1a;★ 列出和查看主题 CMAK 各个功能的作用解释&#xff08;主题&#xff09; ★ 主题 Kafka 主题虽然也叫 topic&#xff0c;但它和 Pub-Sub 消息模型中 topic 主题及 AMQP 的 t…

Java经典框架之Zookeeper

Zookeeper Java 是第一大编程语言和开发平台。它有助于企业降低成本、缩短开发周期、推动创新以及改善应用服务。如今全球有数百万开发人员运行着超过 51 亿个 Java 虚拟机&#xff0c;Java 仍是企业和开发人员的首选开发平台。 课程内容的介绍 1. Zookeeper的介绍和安装 2. …

web学习笔记(九)

目录 1.初识JS(JavaScript) 1.1什么是JavaScript&#xff1f; 1.2HTML5 CSS3 javaScript三者的关系 1.3 JAVAScript的作用 1.4JAVAScript的组成部分 1.5JS注释 1.6补充知识 2.JS的引入方法 2.1行内式 2.2嵌入式&#xff08;内嵌式&#xff09; 2.3外链式 3.输入和…