您当前的位置:首页 > 电脑百科 > 程序开发 > 编程百科

几款主流消息队列之间的差异,我们应该如何选择

时间:2023-11-17 15:25:00  来源:微信公众号  作者:古明地觉的编程教室

为什么需要消息队列

消息队列是历史最悠久的中间件之一,它可以和不同的进程进行通信,从而实现上下游之间的消息传递。基于此特性,我们可以在以下三个场景中使用消息队列。

  • 解耦;
  • 限流;
  • 流量削峰;

1)解耦

先来看解耦,假设有两个服务:A 和 B,当服务 A 依赖服务 B 时,请求的耗时就是这两个服务之和。但如果服务 B 耗时比较长怎么办?

显然这时服务 A 可以将消息发送到队列中,服务 B 从队列里面去取即可,从而实现两个服务之间的逻辑解耦 + 物理解耦。

几款主流消息队列之间的差异,我们应该如何选择

当用户注册账号时,会将注册信息发给账号服务,账号服务将信息写入数据库后,会调用短信服务给用户发送短信。如果不使用消息队列,那么必须等短信发送成功之后才能返回。

但为了给用户更好的体验,我们可以将发送短信这一步独立出去,账号服务将用户手机号和短信内容投入消息队列中就可以返回了,这样用户就能立刻收到注册结果。而短信服务会消费消息,异步执行发送短信逻辑,这就是消息队列的作用之一:解耦。

使用消息队列进行解耦,不仅可以提升性能,还可以使整个系统更加的模块化。以电商为例,订单服务是电商系统中的核心部分,它会被一系列下游服务依赖。并且随着业务的发展,依赖订单的下游服务会不断增加、不断变化。

因此负责订单服务的开发团队不得不花费大量精力,应对不断增加变化的下游服务,不停地修改调试订单服务与这些下游服务的接口。任何一个下游服务的接口产生变更,都需要订单模块重新进行一次上线,对于一个电商的核心服务来说,这几乎是不可接受的。

因此所有的电商系统都选择用消息队列,来解决这种系统耦合过于紧密的问题。引入消息队列后,订单服务在订单变化时发送一条消息到消息队列的一个主题 order 中,所有下游服务都订阅主题 order,这样每个下游服务都可以获得一份实时完整的订单数据。并且此时下游服务发生变化,不会影响订单服务。

2)限流

一个完善的系统一定具备自我保护的能力,即使面对海量请求,也能尽最大努力去处理,处理不了的则会拒绝掉,从而保证系统运行正常。因此如果我们能预估出系统的最大处理能力,就可以用消息队列实现一个令牌桶,进行流量控制。

令牌桶控制流量的原理是:单位时间内发放固定数量的令牌到令牌桶中,规定服务在处理请求之前必须先从桶中取走一个令牌,如果桶里面没有令牌,则拒绝请求。这样就保证单位时间内,能处理的请求数不超过发放令牌的数量,起到了流量控制的作用。

令牌桶可以简单地用一个有固定容量的消息队列加一个令牌生成器来实现:令牌生成器按照预估的处理能力,匀速生产令牌并放入令牌队列(如果队列满了则丢弃令牌)。网关(流量的入口)在收到请求时从令牌队列消费一个令牌,获取到令牌则继续调用后端服务,如果获取不到令牌则直接返回失败。

3)流量削峰

任何的大型服务,特别是秒杀服务,都离不开消息队列。因为消息队列除了解耦和限流之外,还可以起到流量削峰的作用,就是缓冲瞬时的突发流量,使其更平滑。

对于那些发送能力很强的上游系统,如果没有消息队列的保护,脆弱的下游系统可能会直接被压垮导致全链路服务雪崩。而一旦有了消息队列,它就能够有效地对抗上游的流量冲击,避免了流量的震荡。

我们举一个实际的例子,比如在京东购买商品,当点击购买的时候,会调用订单服务生成对应的订单。然而要处理该订单则会依次调用下游的多个子服务,比如查询登录信息、验证商品信息、确认地址信息,调用银行等支付接口进行扣款等等。

显然上游的订单操作比较简单,它的 TPS 要远高于处理订单的下游服务。因此如果上游和下游直接对接,势必会出现下游服务无法及时处理上游订单从而造成订单堆积的情况。特别是当出现双十一以及秒杀业务的时候,上游订单流量会瞬间增加,可能出现的结果就是直接压垮下游子系统服务。

解决此问题的一个做法是对上游的订单服务进行限流,比如采用上面说的令牌桶。但对于一个电商系统来说,这么做很明显是不合适的,因为问题不是出现在订单服务上面,而且用户买东西还限流,这样钱送到嘴边都赚不到。

所以会引入消息队列来对抗这种上下游系统的 TPS 不一致以及瞬时的峰值流量,引入消息队列之后,上游系统不再直接与下游系统进行交互。当新订单生成之后它仅仅向队列中发送一条消息,而下游消费队列中的消息,从而实现上游订单服务和下游订单处理服务的解耦。

这样当出现秒杀业务的时候,消息队列能够将瞬时增加的订单流量全部以消息的形式保存在队列中,既不影响上游服务的 TPS,同时也给下游服务留出了足够的时间去消费,这就是消息队列存在的最大意义所在。

简单来说,我们在单体应用里面需要使用本地队列解决的问题,在分布式系统中大多都可以用消息队列来解决。但同时我们也要认识到,消息队列也有它的一些问题:

  • 引入消息队列带来的延迟问题;
  • 增加了系统的复杂度;
  • 可能产生数据不一致的问题;

所以在软件开发中没有银弹,需要根据业务的特点和自身条件选择合适的架构

消息队列该怎么选择

消息队列如同数据结构一样,没有最好的,只有最合适的。但不管哪种消息队列,如果想要用于生产环境中,都应该具备以下几个特点:

  • 消息的传递一定是可靠的;
  • 支持阻塞等待拉取消息;
  • 支持发布 / 订阅模式;
  • 具备 ack 机制,消费失败后可重新消费,消息不丢失;
  • 实例宕机了,消息也不会丢失,也就是支持数据持久化;
  • 消息可积压;
  • 支持集群部署;
  • 开源免费,社区具有一定的活跃度;
  • 生态完善;
  • 性能足够好,能满足绝大部分场景;

符合以上需求的消息队列,主要有以下几种。

1)RabbitMQ

RabbitMQ 是一个在 AMQP(高级消息队列协议)基础上完成的可复用的企业消息系统,最早为电信行业系统之间的可靠通信而设计,是当前最主流的消息队列之一。

早期的 RabbitMQ 只支持 AMQP 协议,现在也支持 MQTT 协议。

RabbitMQ 都具备哪些优点呢?

  • 采用 Erlang 语言编写,Erlang 语言最初用于交换机领域,它有着和原生 socket 一样的延迟,因此性能较好,吞吐量在万级,并且时效性在微秒级。
  • 功能完善,健壮、稳定、易用、跨平台。
  • 支持大部分主流语言,文档丰富,还提供了管理界面,并拥有非常高的社区活跃度和更新频率。

有优点,自然就有缺点,缺点如下:

  • RabbitMQ 对消息堆积的支持并不好,在它的设计理念里面,消息队列是一个管道,大量的消息积压是一种不正常的情况,应当尽量去避免。当大量消息积压的时候,会导致 RabbitMQ 的性能急剧下降。
  • RabbitMQ 的性能比较差,根据官方给出的测试数据以及使用经验,随着硬件配置的不同,它大概每秒钟可以处理几万到十几万条消息。其实这个性能也足够支撑绝大多数的应用场景了,但如果你的应用对消息队列的性能要求非常高,那就不适合选择 RabbitMQ 了。
  • RabbitMQ 使用的编程语言 Erlang,这个编程语言不仅非常小众,学习曲线也很陡峭。

2)RocketMQ

阿里巴巴开源的一款消息队列,用 JAVA 语言实现,在设计时参考了 Kafka,并做了一些改进。在阿里内部广泛应用于订单、交易、重置、流计算、消息推送、日志流式处理、以及 binlog 分发等场景,经历过多次双十一考验,其性能、稳定性和可靠性都是值得信赖的。

优点如下:

  • 支持单机吞吐量达到数十万级,可用性高,分布式架构保证消息零丢失;
  • 功能较为完善,扩展性好,支持 10 亿级别的消息堆积,不会因为消息堆积导致性能下降;
  • 对在线业务的响应时延做了很多的优化,大多数情况下可以做到毫秒级的响应;

所以 RocketMQ 在吞吐量和消息堆积方面要比 RabbitMQ 高很多,如果你比较在意这两个方面,那么可以使用 RocketMQ。而 RocketMQ 的缺点就是支持的客户端语言不多,社区活跃度一般。

3)Kafka

大数据的杀手锏,谈到大数据领域的消息传输,必然离不开 Kafka。这款为大数据而生的消息队列,有着百分级 TPS 的吞吐量,在数据采集、传输、存储的过程中发挥至关重要的作用,任何的大公司、或者做大数据的公司都离不开 Kafka。

  • Kafka 的特点是性能卓越,单机写入 TPS 在百万条每秒,时效性也在毫秒级。
  • Kafka 是分布式的,一个数据多个副本,少数的机器宕机也不会丢失数据。
  • 消费者采用 pull 方式获取消息,消息有序、并且可以保证所有消息被消费且仅被消费一次。
  • 拥有优秀的第三方 Kafka Web 管理界面 Kafka-Manager,在日志领域比较成熟,在大数据领域的实时计算以及日志采集等场景中被大规模使用。
  • 和周边生态系统的兼容性非常好,在大数据和流计算领域,几乎所有的开源软件系统都会优先支持 kafka。

Kafka 使用 Scala 和 Java 语言开发,设计上大量使用了批量和异步的思想,这种设计使得 Kafka 能做到超高的性能。但也正是这种异步批量设计使得 Kafka 的响应时延比较高,因为客户端发送消息的时候,不会立即发出,而是攒够一批之后一起发送。

所以 Kafka 不太适合在线业务场景,它的重点是吞吐量,而不是低延迟。并且 Kafka 还有如下缺点:

  • 单机超过 64 个分区,CPU 使用率会发生明显的飙高现象,队列越多 CPU 使用率越高,发送消息响应时间变长;
  • 使用短轮询方式,实时性取决于轮询间隔时间,消费失败不支持重试;
  • 虽然支持消息有序,但如果某台机器宕机,就会产生消息乱序。

那么这些消息队列,我应该选择哪一种呢?

RabbitMQ:如果说消息队列并不是你系统的主角之一,你对消息队列的功能和性能都没有很高的要求,只需要一个开箱即用易于维护的产品,建议使用 RabbitMQ。

RocketMQ:天生为金融领域而生,适合可靠性要求很高的场景,尤其是电商里面的订单扣款、以及业务削峰。RocketMQ 在稳定性上绝对值得信赖,毕竟这些业务场景在阿里双十一已经经历了多次考验,如果你的业务也有类似场景,那么建议选择 RocketMQ。

Kafka:基于 Pull 模式来处理消息,追求高吞吐量,一开始的目的就是用于日志收集和传输,高吞吐量是 Kakfka 的目标。因此如果要处理海量的消息(比如日志采集、监控信息、前端埋点),或者使用了大数据、流计算相关的开源产品,那么首选 Kafka。

消息队列的存储模型

任何一款消息队列,都可以视为三部分:生产者、broker、消费者。

  • 生产者和消费者都可以视为客户端;
  • broker 便是服务端启动之后的进程,比如 Kafka broker;

生产者会将消息发送至 broker 中,broker 会对消息进行存储以及持久化,消费者负责从 broker 中拉取消息。如果抛开那些花里胡哨的概念,其实整个过程是非常简单的。

而这里我们要探讨的是,消息在队列中是如何存储的?

最初的消息队列,就是一个严格意义上的队列,它是一个先进先出的线性表,通常使用链表或数组来实现。队列只允许在后端(称为 rear)进行插入操作,在前端(称为 front)进行删除操作。

这个定义里面包含几个关键点:

  • 先进先出:这意味着消息在入队和出队的过程中,需要保证这些消息严格有序,消息的写入顺序和读取顺序是一致的;
  • 早期的消息队列,就是按照队列的数据结构来设计的;
  • 生产者发消息是入队操作,消费者收消息是出队(删除)操作;
  • 服务端存放消息的容器自然就是队列

几款主流消息队列之间的差异,我们应该如何选择

如果有多个生产者往同一个队列里面发消息,这个队列中可以消费到的消息就是这些生产者生产的所有消息的合集,消息的顺序就是这些生产者发送消息的自然顺序。

同理,如果有多个消费者接收同一个队列的消息,这些消费者之间就是竞争关系,每个消费者只能收到队列中的一部分消息,也就是说任何一条消息只能被其中的一个消费者收到。

如果需要将一份消息数据分发给多个消费者,要求每个消费者都能收到全量的消息,比如一份订单数据,要求风控系统、分析系统、支付系统等都需要接收消息。这时单个队列就满足不了需求,一个可行的解决方式是,为每个消费者创建一个单独的队列,让生产者发送多份。

显然这是个比较笨的做法,同样的一份消息被复制到多个队列中会浪费资源。更重要的是,生产者必须知道有多少个消费者,为每个消费者单独发送一份消息,这实际上违背了消息队列的解耦这个设计初衷。

为了解决这个问题,演化出了另外一种消息模型:发布/订阅模型。

几款主流消息队列之间的差异,我们应该如何选择

在发布/订阅模型中,消息的发送方被称为发布者(Publisher),消息的接收方被称为订阅者(Subscriber),服务端存放消息的容器称为主题(Topic)。发布者将消息发送到主题中,订阅者在接收消息之前需要先订阅主题,每份订阅中,订阅者都可以接收到主题的所有消息。

在消息领域的历史上很长的一段时间,队列模式和发布/订阅模式是并存的,有些消息队列同时支持这两种消息模型。但我们仔细对比一下这两种模型,会发现它们并没有本质的区别,生产者就是发布者,消费者就是订阅者,队列就是主题。它们最大的区别其实就是,一份消息数据能不能被消费多次的问题。

实际上,在这种发布/订阅模型中,如果只有一个订阅者,那它和队列模型就基本是一样的了,因此发布/订阅模型在功能层面上可以兼容队列模型。

现代的消息队列产品使用的消息模型大多是这种发布/订阅模型,当然也有例外。

RabbitMQ 的消息模型

RabbitMQ 是少数依然坚持使用队列模型的产品之一,那它是怎么解决多个消费者的问题呢?在 RabbitMQ 里面有一个别的消息队列都没有的概念:Exchange(交换机),它位于生产者和队列之间,生产者并不关心将消息发送给哪个队列,而是将消息发送给 Exchange,由 Exchange 上配置的策略来决定将消息投递到哪些队列中。

几款主流消息队列之间的差异,我们应该如何选择

同一份消息如果需要被多个消费者消费,则需要配置 Exchange 将消息发送到多个队列,每个队列中都存放一份完整的消息数据,可以为一个消费者提供消费服务。这也可以变相地实现发布/订阅模型中,一份消息数据可以被多个订阅者来多次消费这样的功能。

几款主流消息队列之间的差异,我们应该如何选择

所以 RabbitMQ 消息传递模型的核心思想是:生产不会直接将消息发送到队列,而是先发送到交换机。交换机的工作内容就是接收生产者的消息,并且按照指定的规则将消息推入队列,因此交换机必须清楚地知道如何处理接收到的消息,是把这些消息推送到特定队列、还是多个队列,亦或是丢弃它们,这要由交换机的类型决定。

交换机有 4 种类型:direct、fanout、topic、headers,默认是 direct,不同的类型的交换机会有不同的表现。

RabbitMQ 会通过 Binding 将 Exchange 和 Queue 绑定在一起,并且在绑定 Exchange 和 Queue 的同时(可多次绑定),会指定一个 Binding key。而生产者将消息发送到 Exchange 的时候,也会带上一个 Routing key。

  • 如果交换机的类型是 direct,它会将消息推送到 Binding Key 和 Routing Key 相匹配的 Queue 中。因为交换机和队列可以多次绑定,所以一个队列可以有多个 Binding Key,只要一个能匹配上即可;
  • 如果交换机的类型是 fanout,它会直接把消息推送到所有与它绑定的队列中;
  • 如果交换机的类型是 topic,那么 Binding Key 会支持 * 通配符,从而和 Routing Key 进行模糊匹配;
  • 如果交换机类型是 headers,会根据发送的消息内容中的 headers 属性进行匹配;

RocketMQ 的消息模型

RocketMQ 使用标准的发布/订阅模型,但它除了生产者、消费者和主题之外,也有队列这个概念,并且队列在 RocketMQ 中是一个非常重要的概念。每个主题包含多个队列,通过多个队列来实现并行生产和消费。需要注意的是,RocketMQ 只在队列上保证消息的有序性,主题层面是无法保证消息的严格顺序的。

RocketMQ 中,订阅者的概念是通过消费者组(Consumer Group)来体现的,每个消费者组都消费主题中一份完整的消息,不同消费者组之间消费进度彼此不受影响。也就是说,一条消息被 Consumer Group1 消费过,也可以再给 Consumer Group2 消费。

但消费者组中包含多个消费者,同一个组内的消费者是竞争关系,每个消费者负责消费组内的一部分消息。如果一条消息被消费者 Consumer1 消费了,那同组的其它消费者就不会再收到这条消息。

几款主流消息队列之间的差异,我们应该如何选择

在 Topic 的消费过程中,由于消息需要被不同的组进行多次消费,所以消费完的消息并不会立即被删除,这就需要 RocketMQ 为每个消费者组在每个队列上维护一个消费位移(Consumer Offset)。这个位移之前的消息都被消费过,之后的消息都没有被消费过,每成功消费一条消息,消费位移就加一。

如果你对 RocketMQ 中的这些概念还有些困惑的话,那么没关系,我们看一下 Kafka 的消息模型。如果你熟悉 Kafka 的话,那么你会瞬间理解 RocketMQ。

Kafka 的消息模型

Kafka 的消息模型和 RocketMQ 是完全一样的,上面说的所有 RocketMQ 中的概念,和生产消费过程中的确认机制,都完全适用于 Kafka。唯一的区别是,在 Kafka 中,队列这个概念的名称不一样,Kafka 中对应的名称是分区(Partition),但含义以及功能和 RocketMQ 的队列是没有任何区别的。

所以如果你熟悉 Kafka,那么你会瞬间理解 RocketMQ,因为两者的消息模型是一样的。只不过 RocketMQ 是一个主题对应多个队列,而 Kafka 是一个主题对应多个分区。

小结

以上我们就探讨了消息队列的应用场景,以及它们存储模型之间差异。关于这些队列更详细的内容,可以参考网上的资料。

总之当你的数据量不大时,使用 RabbitMQ 是一个不错的选择。



Tags:消息队列   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系,我们将及时更正、删除。
▌相关推荐
如何使用 Redis 实现消息队列
Redis不仅是一个强大的内存数据存储系统,它还可以用作一个高效的消息队列。消息队列是应用程序间或应用程序内部进行异步通信的一种方式,它允许数据生产者将消息放入队列中,然...【详细内容】
2024-03-22  Search: 消息队列  点击:(17)  评论:(0)  加入收藏
消息队列备选架构选择,你选择哪个?
中间件团队的研发人员认为这个方案比较简单,实现成本低,但测试代表认为这个方案测试人力投入较大。运维团队认为这个方案的硬件成本比较高,一个数据分组就需要4台机器(2台服务器...【详细内容】
2023-11-30  Search: 消息队列  点击:(179)  评论:(0)  加入收藏
四种消息队列,如何选型?
最近发现很多号主发消息队列的文章,质量参差不齐,相关文章我之前也写过,建议直接看这篇。这篇文章,主要讲述 Kafka、RabbitMQ、RocketMQ 和 ActiveMQ 这 4 种消息队列的异同,无论...【详细内容】
2023-11-27  Search: 消息队列  点击:(193)  评论:(0)  加入收藏
几款主流消息队列之间的差异,我们应该如何选择
为什么需要消息队列消息队列是历史最悠久的中间件之一,它可以和不同的进程进行通信,从而实现上下游之间的消息传递。基于此特性,我们可以在以下三个场景中使用消息队列。 解耦; ...【详细内容】
2023-11-17  Search: 消息队列  点击:(124)  评论:(0)  加入收藏
常用消息队列框架与技术选型
又是一年双11季,土豪们买买买,程序员看看热闹,聊聊技术。海量的订单、支付请求以及库存更新等任务,离不开分布式架构(SOFAStack)、分布式数据库(OceanBase)、分布式缓存(Tair)、数据处...【详细内容】
2023-11-13  Search: 消息队列  点击:(204)  评论:(0)  加入收藏
Java中的消息队列实战,构建高效异步系统
随着互联网应用的发展,高效的异步系统变得越来越重要。在这样的系统中,消息队列起到了关键的作用。通过消息队列,可以将不同组件之间的耦合度降低,实现解耦和异步处理,提高系统的...【详细内容】
2023-11-07  Search: 消息队列  点击:(324)  评论:(0)  加入收藏
热门的消息队列框架比较、使用方法、优缺点,提供示例代码
消息队列(Message Queue)是一种在分布式系统中用于消息传递的通信模式。它可以将消息发送者和接收者解耦,提高系统的可靠性、可扩展性和可维护性。下面将详细介绍3-5个常用的...【详细内容】
2023-10-11  Search: 消息队列  点击:(151)  评论:(0)  加入收藏
为什么我们需要消息队列?
消息队列有着悠久的历史,它们经常用于不同系统之间的通信。图1通过将其与星巴克的工作方式进行比较,阐述了消息队列的概念。在星巴克,收银员接受订单并收取款项,然后在咖啡杯上...【详细内容】
2023-09-05  Search: 消息队列  点击:(353)  评论:(0)  加入收藏
使用 SQL 的方式查询消息队列数据以及踩坑指南
Pulsar-SQL 是一个非常有用的功能,只是我们使用过程中确实发现了一些问题,大部分都已经修复了;希望对后续使用该功能的朋友有所帮助。背景为了让业务团队可以更好的跟踪自己消...【详细内容】
2023-08-31  Search: 消息队列  点击:(275)  评论:(0)  加入收藏
Java消息队列开发实战,打造高效异步处理
随着互联网应用的发展,高效的异步系统变得越来越重要。在这样的系统中,消息队列起到了关键的作用。通过消息队列,可以将不同组件之间的耦合度降低,实现解耦和异步处理,提高系统的...【详细内容】
2023-08-30  Search: 消息队列  点击:(357)  评论:(0)  加入收藏
▌简易百科推荐
即将过时的 5 种软件开发技能!
作者 | Eran Yahav编译 | 言征出品 | 51CTO技术栈(微信号:blog51cto) 时至今日,AI编码工具已经进化到足够强大了吗?这未必好回答,但从2023 年 Stack Overflow 上的调查数据来看,44%...【详细内容】
2024-04-03    51CTO  Tags:软件开发   点击:(5)  评论:(0)  加入收藏
跳转链接代码怎么写?
在网页开发中,跳转链接是一项常见的功能。然而,对于非技术人员来说,编写跳转链接代码可能会显得有些困难。不用担心!我们可以借助外链平台来简化操作,即使没有编程经验,也能轻松实...【详细内容】
2024-03-27  蓝色天纪    Tags:跳转链接   点击:(12)  评论:(0)  加入收藏
中台亡了,问题到底出在哪里?
曾几何时,中台一度被当做“变革灵药”,嫁接在“前台作战单元”和“后台资源部门”之间,实现企业各业务线的“打通”和全域业务能力集成,提高开发和服务效率。但在中台如火如荼之...【详细内容】
2024-03-27  dbaplus社群    Tags:中台   点击:(8)  评论:(0)  加入收藏
员工写了个比删库更可怕的Bug!
想必大家都听说过删库跑路吧,我之前一直把它当一个段子来看。可万万没想到,就在昨天,我们公司的某位员工,竟然写了一个比删库更可怕的 Bug!给大家分享一下(不是公开处刑),希望朋友们...【详细内容】
2024-03-26  dbaplus社群    Tags:Bug   点击:(5)  评论:(0)  加入收藏
我们一起聊聊什么是正向代理和反向代理
从字面意思上看,代理就是代替处理的意思,一个对象有能力代替另一个对象处理某一件事。代理,这个词在我们的日常生活中也不陌生,比如在购物、旅游等场景中,我们经常会委托别人代替...【详细内容】
2024-03-26  萤火架构  微信公众号  Tags:正向代理   点击:(10)  评论:(0)  加入收藏
看一遍就理解:IO模型详解
前言大家好,我是程序员田螺。今天我们一起来学习IO模型。在本文开始前呢,先问问大家几个问题哈~什么是IO呢?什么是阻塞非阻塞IO?什么是同步异步IO?什么是IO多路复用?select/epoll...【详细内容】
2024-03-26  捡田螺的小男孩  微信公众号  Tags:IO模型   点击:(8)  评论:(0)  加入收藏
为什么都说 HashMap 是线程不安全的?
做Java开发的人,应该都用过 HashMap 这种集合。今天就和大家来聊聊,为什么 HashMap 是线程不安全的。1.HashMap 数据结构简单来说,HashMap 基于哈希表实现。它使用键的哈希码来...【详细内容】
2024-03-22  Java技术指北  微信公众号  Tags:HashMap   点击:(11)  评论:(0)  加入收藏
如何从头开始编写LoRA代码,这有一份教程
选自 lightning.ai作者:Sebastian Raschka机器之心编译编辑:陈萍作者表示:在各种有效的 LLM 微调方法中,LoRA 仍然是他的首选。LoRA(Low-Rank Adaptation)作为一种用于微调 LLM(大...【详细内容】
2024-03-21  机器之心Pro    Tags:LoRA   点击:(12)  评论:(0)  加入收藏
这样搭建日志中心,传统的ELK就扔了吧!
最近客户有个新需求,就是想查看网站的访问情况。由于网站没有做google的统计和百度的统计,所以访问情况,只能通过日志查看,通过脚本的形式给客户导出也不太实际,给客户写个简单的...【详细内容】
2024-03-20  dbaplus社群    Tags:日志   点击:(4)  评论:(0)  加入收藏
Kubernetes 究竟有没有 LTS?
从一个有趣的问题引出很多人都在关注的 Kubernetes LTS 的问题。有趣的问题2019 年,一个名为 apiserver LoopbackClient Server cert expired after 1 year[1] 的 issue 中提...【详细内容】
2024-03-15  云原生散修  微信公众号  Tags:Kubernetes   点击:(5)  评论:(0)  加入收藏
站内最新
站内热门
站内头条