flea-msg使用之JMS初识

news2025/1/10 2:50:38

JMS初识

1. JMS 基本概念

1.1 什么是 JMS ?

Java 消息服务【Java Message Service】,又简称 JMS,它是 Java 平台上有关面向消息中间件(MOM)的技术规范。

在这里插入图片描述

1.2 JMS 规范

JMS 中定义了 Java 中访问消息中间件的接口,并没有给予实现,实现 JMS 接口的消息中间件称为 JMS Provider,例如 ActiveMQRocketMQ 等等。

下面我们来了解一下 JMS 中的一些专业名词:

英文名称中文名称描述
JMS ProviderJMS 提供者实现 JMS 接口和规范的消息中间件。提供者可以是 Java 平台的 JMS 实现,也可以是非 Java 平台的面向消息中间件的适配器。
JMS MessageJMS 消息可以在 JMS 客户之间传递数据的对象。 它由如下三部分组成:
(1) 消息头: 每个消息头字段都有相应的 gettersetter 方法。
(2) 消息属性: 如果需要除消息头字段以外的值,那么可以使用消息属性。
(3) 消息体: 封装具体的消息数据。
JMS ClientJMS 客户端生产或消费基于消息的 Java 的应用程序或对象。
JMS ProducerJMS 生产者创建和发送 JMS 消息的客户端应用。
JMS ConsumerJMS 消费者接收和处理 JMS 消息的客户端应用。可以有如下两种方法消费消息:
(1)同步消费: 通过调用消费者的 receive 方法从目的地中显式提取消息,receive 方法可以一直阻塞到消息到达。
(2)异步消费: 客户端可以为消费者注册一个消息监听器,当会话线程调用消息监听器对象的 onMessage() 方法时,客户端消费消息。
JMS DomainsJMS 消息传递域JMS 规范中定义了两种消息传递域:点对点point-topoint,简称 PTP) 和 发布/订阅publish/subscribe,简称 pub/sub

点对点消息传递域的特点
(1)每个消息只能有一个消费者;
(2)消息的生产者和消费者之间没有时间上的关联性,无论消费者在生产者发送消息的时候是否处于运行状态,它都可以提取消息。生产者不需要在接收者消费该消息期间处于运行状态,接收者也同样不需要在消息发送时处于运行状态。

发布/订阅消息传递域的特点
(1)每个消息可以有多个消费者;
(2)生产者和消费者之间有时间上的关联性。订阅一个主题的消费者只能消费自它订阅之后发布的消息。JMS 规范允许客户创建持久订阅,这在一定程度上放松了时间上的关联性要求。持久订阅允许消费者消费它在未处于激活状态时发送的消息。
ConnectionFactory连接工厂用来创建连接对象,以连接到 JMS 提供者。
JMS ConnectionJMS 连接封装了 JMS 客户端和 JMS 提供者【服务器端】 之间的一个活动的连接,是由客户端通过调用连接工厂的方法建立的。
JMS SessionJMS 会话JMS 客户端 与 JMS 提供者【服务器端】 之间的会话状态。JMS 会话建立在 JMS 连接上,表示 客户端与 服务器端 之间的一个会话线程。它提供了一个事务性的上下文,在这个上下文中,一组发送和接收被组合到了一个原子操作中。
JMS DestinationJMS 目的地消息发送到的目的地,是实际的消息源。
在点对点消息传递域中,目的地被称为队列(Queue);
在发布/订阅消息传递域中,目的地被称为主题(Topic)。
在这里插入图片描述
Administered Objects管理对象JMS 没有完全定义的两个消息传递元素是 连接工厂目的地。尽管这些是 JMS 编程模型中的基本元素,但在提供者定义和管理这些对象的方式上存在许多现有的和预期的差异,因此创建一个通用的定义既不可能也不可取。因此,这两个对象通常是使用管理工具创建和配置的,而不是以编程方式创建的。然后将它们存储在(提供者)的对象存储区中,并由 JMS 客户端通过标准 JNDI 查找进行访问。

连接工厂 管理对象用于生成客户端到 Broker 的连接。它们封装了特定于提供者的信息,这些信息控制消息传递行为的某些方面:连接处理、客户端标识、消息头覆盖、可靠性和流控制等。从给定连接工厂派生的每个连接都显示为该工厂配置的行为。

目的地 管理对象用于引用 Broker上的物理目的地。它们封装了特定于提供者的命名(地址语法)约定,并指定了使用目的地的消息传递域:队列(Queue)主题(Topic)

如下图显示了消息生产者和消息消费者如何使用目的地管理对象访问其对应的物理目的地。标记的步骤表示管理员和客户端应用程序使用此机制发送和接收消息所需采取的操作。
在这里插入图片描述
步骤1. 管理员在 Broker上创建物理目的地。
步骤2. 管理员创建目的地管理对象,并通过指定其对应的物理目的地的名称及其类型(队列或主题)进行配置。
步骤3. 消息生产者使用 JNDI 查找目的地管理对象。
步骤4. 消息生产者向目的地发送消息。
步骤5. 消息消费者查找其希望获取消息的目的地管理对象。
步骤6. 消息消费者从目的地获取消息。

2. JMS 编程对象

所谓的 JMS 编程对象,即为实现 JMS 消息传递的对象,包括 连接工厂、连接、会话、生产者、消费者、消息 和 目的地。

下图就展示了上面这些 JMS 编程对象之间的联系:

在这里插入图片描述

我们在上图中可以看到,有两个对象(连接工厂目的地)是在对象存储中的。它们通常是作为管理对象创建、配置和管理的。也就是说,连接工厂目的地 是以管理方式(而不是以编程方式)创建的。

如下表格总结了 发送 和 接收 消息所需的步骤【从上图中也可看出一二】:

生产消息消费消息
1. 管理员创建连接工厂管理的对象。
2. 管理员创建物理目的地和引用它的管理对象。
3. 客户端通过 JNDI 查找获得连接工厂对象。
4. 客户端通过 JNDI 查找获得目的地对象。
5. 客户端创建一个连接并设置针对此连接的属性。
6. 客户端创建一个会话并设置管理消息传递可靠性的属性。
7. 客户端创建消息生产者 客户端创建消息消费者
8. 客户端创建消息 客户端启动连接
9. 客户端发送消息 客户端接收消息

注意,步骤1到6 对于发送方和接收方是相同的。

下面我们来详细介绍下 JMS 编程对象:

2.1 连接工厂和连接

客户端使用连接工厂对象(ConnectionFactory)创建连接。连接对象(Connection)表示客户端到 Broker 的活跃连接。它使用基础连接服务,该服务在默认情况下启动,或者由该客户端的管理员显式启动。

通信资源的分配和客户端的身份验证都在创建连接时进行。它是一个相对重量级的对象,大多数客户端都使用一个连接来完成所有消息传递。连接支持并发使用:任何数量的生产者和消费者都可以共享连接

创建连接工厂时,可以通过设置其属性来配置从该工厂派生的所有连接的行为。对于消息队列,它们可以指定如下信息:

  • Broker 驻留的主机的名称、所需的连接服务以及客户端希望访问该服务的端口。
  • 如果连接失败,应如何处理与 Broker 的自动重新连接。(如果连接丢失,此功能会将客户端重新连接到同一个(或不同的 Broker)。无法保证数据故障切换:当重新连接到其他代理时,持久消息和其他状态信息可能会丢失。)
  • 需要 Broker 跟踪其持久订阅的客户端的ID。
  • 尝试连接的用户的默认名称和密码。如果在连接时未指定密码,则此信息用于验证用户并授权操作。
  • 对于那些不关心可靠性的客户端,是否应禁止 Broker 签收。
  • 如何管理 Broker 和客户端运行时之间的控制流和有效负载消息。
  • 应如何处理队列浏览(仅限Java客户端)。
  • 是否应重写某些消息头字段。

可以从启动客户端应用程序的命令行来覆盖连接工厂属性。也可以通过设置那连接的属性来覆盖任何给定连接的属性。

您可以使用连接对象来创建会话对象、设置异常监听器 或 获取 JMS 版本和提供者信息。

2.2 会话

如果 Connection 表示客户端和 Broker 之间的通信通道,那 Session 就将代表它们之间的单个会话。后面我们主要使用会话对象来创建消息、消息生产者和消息消费者。创建会话时,您可以通过多个确认选项 或者 事务 来配置可靠的传递。有关详细信息,请参阅 可靠性消息传递。

根据 JMS 规范,会话是用于生产和消费消息的单线程上下文。您可以为一个会话创建多个消息生产者和消费者,但您只能连续使用它们。JavaC 客户端的线程实现略有不同,

还可以使用会话对象执行以下操作:

  • 为那些不使用管理对象定义目的地的客户端创建和配置目的地。
  • 创建和配置临时主题和队列;这些被用作请求-应答模式的一部分。请参阅 请求-应答模式。
  • 支持事务处理。
  • 定义生产或消费消息的序列顺序。
  • 为异步消费者序列化消息监听器的执行。
  • 创建队列浏览器(仅限Java客户端)。

2.3 消息

上面我们了解到,消息由三部分组成,分别是 消息头消息属性消息体

2.3.1 消息头

每个 JMS 消息都需要一个消息头。消息头包含十个预定义字段,这些字段参考如下表格:

消息头字段描述
JMSDestination指定将消息发送到的目的地对象的名称(由提供者设置),也就是 QueueTopic,自动分配。
JMSDeliveryMode传送模式,指定消息是否持久(默认情况下,由 提供者 或 客户端 为生产者或单个消息显式设置)。有两种 :持久模式和非持久模式。
JMSExpiration指定消息过期的时间(默认情况下,由提供者 或 客户端为生产者或单个消息设置),它等于 Destinationsend 方法中的 timeToLive 值加上发送时刻的 GMT 时间值。如果 timeToLive 值等于零,则JMSExpiration 被设为零,表示该消息永不过期。如果发送后,在消息过期时间之后消息还没有被发送到目的地,则该消息被清除。
JMSPriority指定0(低)到9(高)范围内的消息优先级(默认情况下,由提供者设置 或 客户端为生产者或单个消息显式设置),其中 0-4普通消息5-9加急消息JMS 不要求 JMS Provider 严格按照这十个优先级发送消息,但 必须保证加急消息要先于普通消息到达
JMSMessageID为提供者上下文中的消息指定唯一ID(由提供者设置)
JMSTimestamp指定提供者接收消息的时间(由提供者设置)
JMSCorrelationID允许客户端定义两个消息之间的对应关系的值(如果需要,由客户端设置),典型的应用是在应答消息中连接到原消息。
JMSReplyTo指定消费者应发送回复的目的地(如果需要,由客户端设置)
JMSType消息类型的识别符,可以由消息选择器计算的值(如果需要,由客户端设置)
JMSRedelivered指定消息是否已传递但未确认(由提供者设置)。如果一个客户端收到一个设置了 JMSRedelivered 属性的消息,则表示可能客户端曾经在早些时候收到过该消息,但并没有签收(acknowledged)。

通过查看上述表格,我们可以看出,消息头字段有多种用途:标识消息配置消息路由提供有关消息处理的信息等等。消息生产者可能需要配置消息头以获得某些消息传递行为;消息消费者可能需要读取消息头,以了解消息是如何路由的,以及它可能需要的进一步的处理。

JMSDeliveryMode 是最重要的字段之一,它决定了消息传递的可靠性。此字段指示消息是否持久。

  • 持久消息。保证消息传递并成功消费一次。如果消息服务失败,持久消息不会丢失。

  • 非持久性消息。保证消息最多传递一次。如果消息服务失败,非持久性消息可能会丢失。

2.3.2 消息属性

JMS 规范中包含如下三种类型的属性:

  • 应用程序设置或添加的属性
  • JMS 定义的属性。
  • JMS 供应商特定的属性。

JMS 规范定义了九个标准属性,详见如下表格。其中一些由客户端设置,一些由提供者设置。它们的名称以保留字符 “JMSX” 开头。客户端或提供者可以使用这些属性来确定谁发送了消息、消息的状态、发送频率和时间。这些属性对于提供者路由消息和提供诊断信息很有用。

属性名描述
JMSXUserID发送消息的用户标识,发送时由提供者设置
JMSXAppID发送消息的应用标识,发送时由提供者设置
JMSXDeliveryCount转发消息重试次数,第一次是1,第二次是2,… ,发送时由提供者设置
JMSXGroupID消息所在消息组的标识,由客户端设置
JMSXGroupSeq组内消息的序号第一个消息是1,第二个是2,…,由客户端设置
JMSXProducerTXID产生消息的事务的事务标识,发送时由提供者设置
JMSXConsumerTXID消费消息的事务的事务标识,接收时由提供者设置
JMSXRcvTimestampJMS 转发消息到消费者的时间,接收时由提供者设置
JMSXState假定存在一个消息仓库,它存储了每个消息的单独拷贝,且这些消息从原始消息被发送时开始。每个拷贝的状态有:1(等待),2(准备),3(到期)或4(保留)。由于状态与生产者和消费者无关,所以它不是由生产者和消费者来提供。它只和在仓库中查找消息相关,因此JMS没有提供这种API。由提供者设置

消息队列也定义了消息属性,这些属性用于标识压缩消息以及在无法传递消息时应如何处理消息。

2.3.1 消息体

消息体包含客户端要交换的数据。

JMS 消息的类型决定了消息体可能包含的内容以及消费者应该如何处理它,详见如下表格。另外,Session 对象中包含了每种类型的消息体的创建方法。

消息类型描述
StreamMessage消息体包含 Java 原始值流的消息。它是按顺序填充和读取的。
MapMessage消息体包含一组键值对的消息。未定义条目的顺序。
TextMessage消息体包含Java字符串的消息,例如XML字符串消息。
ObjectMessage消息体包含序列化Java对象的消息。
BytesMessage消息体包含未解释字节流的消息
Message包含消息头和消息属性,但不包含消息体的消息

Java 客户端可以设置一个属性,让客户端运行时压缩正在发送的消息的消息体。消费者端的消息队列运行时在传递消息之前对消息进行解压缩。

2.4 生产者

上文中,我们知道生产者是创建和发送 JMS 消息的客户端应用,消息就是由消息生产者在连接和会话的上下文中发送或发布。生成消息其实非常简单:客户端使用消息生成器对象(MessageProducer)将消息发送到物理目的地(在 JMS API 中由目的地对象表示)。

创建生产者时,可以指定所有生产者发送消息的默认目的地。还可以为消息头字段指定默认值,这些字段控制持久性、优先级和生存时间。然后,从该生产者发出的所有消息都会使用这些默认值,除非在发送消息时通过指定备用目的地 或 为给定消息的消息头字段设置备用值 来覆盖这些默认值。

消息生产者还可以通过设置 JMSReplyTo 消息头字段来实现请求-应答模式。有关更多信息,请参阅 请求-应答模式。

2.5 消费者

消费者是接收和处理 JMS 消息的客户端应用,消息就是由消息消费者在连接和会话的上下文中接收和处理的。客户端使用消息消费者对象(MessageConsumer)从指定的物理目的地(在 JMS API 中表示为目的地对象)接收消息。

需要注意,有如下三个因素影响 Broker 向消费者传递消息的方式:

  • 消费是同步还是异步
  • 是否使用选择器筛选传入消息
  • 如果消息是从主题目标消费的,则订阅者是否持久

影响消息传递和客户端设计的另一个主要因素是消费者所需的可靠性程度。请参阅 可靠性消息。

2.5.1 同步和异步消费者

消息消费者可以支持消息的同步或异步消费。

  • 同步消费。它意味着消费者需明确请求传递消息,然后消费它。根据请求消息的方式,同步消费者可以选择(无限期地)等待消息到达,等待指定的消息时间,或者在没有消息可供使用时立即返回。(“Consumed” 表示客户端可以立即使用该对象。已成功发送但 Broker 尚未完成处理的消息【即尚未准备好消费】。)

  • 异步消费。它意味着消息将自动传递到已为消费者注册的消息监听器对象(MessageListener)上。当会话线程调用消息监听器对象的 onMessage() 方法时,客户端消费消息。

2.5.2 消息选择器

消息消费者可以使用消息选择器让消息服务仅传递其属性与特定选择条件匹配的消息。我们在创建消费者时可以指定此条件。

选择器使用类似 SQL 的语法来匹配消息属性。例如:

name = "Huazie"
age >= 18

Java 客户端还可以在浏览队列时指定选择器;这允许您查看 有哪些选定的消息正在等待使用

2.5.3 持久订阅者

我们可以使用会话对象创建主题的持久订阅者。即使订阅者处于非活跃状态,Broker 也会保留这些订阅者的消息。

因为 Broker 必须维护订阅者的状态,并在订阅者被重新激活时恢复消息的传递,所以 Broker 必须能够在其来来往往的过程中识别给定订阅者。订阅者的标识是根据创建它的连接的 ClientID 属性和创建订阅者时指定的订阅者名称构造的。

3. JMS 点对点 模型

点对点 模型中,消息生产者称为发送者,消息消费者称为接收者。它们通过一个称为 队列(Queue) 的目的地交换消息:发送方向队列生产消息,接收者消费队列中的消息

下图展示了 点对点 中一个最简单的消息传递操作。MyQueueSenderMsg1 发送到队列目的地 MyQueue1。然后,MyQueueReceiverMyQueue1 中获取消息。

在这里插入图片描述
至于更为复杂的场景,我们可以看下图。两个发送方 MyQSender1MyQSender2 使用 相同的连接MyQueue1 发送消息。MyQSender3 使用额外的连接向MyQueue1 发送消息。在接收端,MyQReceiver1 使用来自 MyQueue1 的消息,MyQRreceiver2MyQRreceive3 共享一个连接以使用来自 MyQueue1 的信息。

在这里插入图片描述
下面我们来总结一下,上图的场景中展示的 点对点 消息传递的一些附加要点:

  • 多个生产者可以向队列发送消息。生产者可以共享一个连接或使用不同的连接,但他们都可以访问同一个队列。
  • 多个接收方可以使用队列中的消息,但每个消息只能由一个接收方消费。因此,Msg1Msg2Msg3 由不同的接收器使用。
  • 接收方可以共享一个连接或使用不同的连接,但它们都可以访问同一个队列。
  • 发送方和接收方没有时间依赖性:无论客户端发送消息时消息是否正在运行,接收方都可以获取消息。
  • 发送方和接收方可以在运行时动态添加和删除,从而允许消息传递系统根据需要进行扩展或收缩。
  • 消息按照发送的顺序放置在队列中,但它们的消费顺序取决于消息过期日期、消息优先级以及是否使用选择器来使用消息等因素。

综合来说,点对点 模型具有如下的一些优势:

  • 如果消息的接收顺序不重要,那么多个接收者可以消费同一队列中的消息,这一事实允许您平衡消息消耗。
  • 即使没有接收方,也始终保留发往队列的消息。
  • Java 客户端可以使用队列浏览器对象来检查队列的内容。然后,他们可以根据从检查中获得的信息消费消息。也就是说,尽管消费模型通常是FIFO(先进先出),但如果消费者通过使用消息选择器知道他们想要什么消息,他们可以消费不在队列头部的消息。管理客户端还可以使用队列浏览器监视队列的内容。

4. JMS 发布/订阅 模型

发布/订阅 模型中,消息生产者称为发布者,消息消费者称为订阅者。他们通过一个称为 主题(Topic) 的目的地交换消息:发布者向主题发布消息;订阅者订阅主题并消费来自主题的消息

下图展示了发布/订阅域中的一个最简单的消息传递操作。MyTopicPublisherMsg1 发布到 MyTopic。然后,MyTopicSubscriber1MyTopicSubscriber2分别从 MyTopic 接收 Msg1 的副本。
在这里插入图片描述
虽然 发布/订阅 模型不需要有多个订阅者,但图中列出了两个订阅者,这就告诉我们该模型允许广播消息。主题的所有订阅者都会获得发布到该主题的任何消息的副本。

订阅服务器可以是持久的或者非持久的。Broker 将保留所有活跃订阅者的消息,但仅当这些订阅者是持久的,Broker 才会保留非活跃订阅者的信息。

下面我们来看下更为复杂的场景,如下图所示。三个生产者向 Topic1 发布消息,三个消费者消费来自 Topic1 的消息;除非订阅者使用选择器来筛选消息,否则每个订阅者都会获得发布到所选主题的所有消息【其中,MyTSubscriber2 过滤掉了 Msg2】。

在这里插入图片描述
通过上图的场景,我们来总结一下其展示的 发布/订阅 消息传递的一些附加要点:

  • 多个生产者可以向主题发布消息。生产者可以共享一个连接或使用不同的连接,但他们都可以访问同一主题。
  • 多个订阅者可以消费来自主题的消息。订阅服务器检索发布到主题的所有消息,除非它们使用选择器筛选出消息,或者消息在使用之前过期。
  • 订阅服务器可以共享一个连接或使用不同的连接,但它们都可以访问同一主题。
  • 持久订阅者可以是活跃的或非活跃的。Broker 在它们处于非活跃状态时将为它们保留消息。
  • 发布者和订阅者可以在运行时动态添加和删除,从而允许消息传递系统根据需要进行扩展或收缩。
  • 消息按照发送的顺序发布到主题,但使用它们的顺序取决于消息过期日期、消息优先级以及是否使用选择器来使用消息等因素。
  • 发布者和订阅者具有时间依赖性:主题订阅者只能使用在创建订阅后发布的消息

发布/订阅 模型的主要优点是它允许 向订阅者广播消息

5. JMS 请求-应答 模式

我们可以在同一个 连接(甚至使用统一API的 会话)中组合生产者和消费者。此外,JMS API 允许我们通过使用 临时目的地 来为 消息传递操作 实现 请求-应答 模式。

如果想要设置 请求-应答 模式,我们需要执行以下操作:

  1. 创建一个消费者可以发送应答的临时目的地。
  2. 在要发送的消息中,将消息头的 JMSReplyTo 字段设置为该临时目的地。

当消息消费者处理消息时,它检查消息的 JMSReplyTo 字段以确定是否需要应答,并将应答发送到指定的目的地。

请求-应答 机制为生产者省去了为应答目的地设置管理对象的麻烦,并使消费者更容易响应请求。当生产者在继续之前必须确保已经处理了请求时,该模式将非常有用。

下图就展示了 向主题发送消息并在临时队列中接收应答的 请求-应答 模式
在这里插入图片描述
如上图所示,MyTopicPublisher 向目标 MyTopic 生产了 Msg1MyTopicSubscriber1MyTopicSubscriber2 接收消息并向 MyTempQueue 发送应答,MyTQReceiver 从中检索消息。此模式可能适用于向大量客户端发布定价信息并将其订单排队进行顺序处理的应用程序。

临时目的地存在的时间仅与创建它们的连接一样长。任何生产者都可以发送到临时目的地,但唯一可以访问临时目的地的消费者是由创建目的地的同一连接创建的消费者。

由于 请求-应答 模式依赖于创建的临时目的地,所以在以下的情况下不应该使用此模式:

  • 如果你预计创建临时目的地的连接可能会在发送应答之前终止。
  • 如果需要将持久消息发送到临时目的地。

参考资料

  1. 【百度百科–JMS】
  2. 【JMS as a MOM Standard】
  3. 【Client Programming Model】

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

分类预测 | MATLAB实现SSA-CNN麻雀算法优化卷积神经网络多特征分类预测

分类预测 | MATLAB实现SSA-CNN麻雀算法优化卷积神经网络多特征分类预测 目录分类预测 | MATLAB实现SSA-CNN麻雀算法优化卷积神经网络多特征分类预测分类效果基本介绍模型描述程序设计参考文献分类效果 基本介绍 1.Matlab实现SSA-CNN麻雀算法优化卷积神经网络多特征分类预测&…

Python操作的5个坏习惯,你中了几个呢?

很多文章都有介绍怎么写好 Python,我今天呢相反,说说写代码时的几个坏习惯。有的习惯会让 Bug 变得隐蔽难以追踪,当然,也有的并没有错误,只是个人觉得不够完美。 注意:示例代码在 Python 3.6 环境下编写 …

数据与C(布尔类型和虚数和实数)

一._Bool类型(%d占位符) C99标准添加了_Bool类型,用于表示布尔值,既逻辑值true(1)和false(0)。原则上_Bool在原则上仅占用1位存储空间,因为对0和1而言,1位的…

数据与C(位,字节,进制转换和C数据内部存储)

程序的运行离不开数据,所以在本数据章节我们会比较详细的讲解不同数据的重要内容 本章主要讲解一些基础知识便于后面后面的数据类型学习,如果本章知识都懂的同学可以直接从下章开始阅读 目录 一.常量和变量 二.位,字节和字 三.四种进制形…

BI-SQL丨ALL、ANY、SOME

ALL、ANY、SOME ALL、ANY和SOME,这三个关键字,在SQL中使用频率较高,通常可以用来进行数据比较筛选。 注:SQL中ALL的用法和DAX中ALL的用法是完全不同的,小伙伴不要混淆了。 那么三者之间的区别是什么呢? A…

spring 笔记

一、spring概述 1.1 spring介绍 spring是一个轻量级的控制反转和面向切面的容器框架,用来解决企业项目开发的复杂度问题---解耦 轻量级:体积小,对代码没有侵入性控制反转:IOC inverse of control, 把创建对象的工作交…

JUC并发编程Ⅰ -- Java中的线程

文章目录线程与进程并行与并发进程与线程应用应用之异步调用应用之提高效率线程的创建方法一:通过继承Thread类创建方法二:使用Runnable配合Thread方法三:使用FutureTask与Thread结合创建查看进程和线程的方法线程运行的原理栈与栈帧线程上下…

MAC Boook打印长图

有时老师给留的作业是一张长图,直接打印或者通过把图放入word打印都不能实现把长页分成多页进行打印。通过网上找到思路可以通过EXCEL实现将长图分成多页打印。 测试版本 macos:ventura 13.1 office 365 注:同样适用windows版本的excel 第…

cass10.1+鸿业生成平纵横数据

cass10.1鸿业生成平纵横数据前言1 纵断面数据获取1.1 数据准备1.2 纵断面桩号设置(1)桩号设置(2)桩号标注(3)标注显示1.3 高程数据处理1.4 纵断面里程标高文件生成2. cass10.1生成横断面数据2.1 生成横断面…

区块链技术与应用2——BTC-数据结构

文章目录比特币中的数据结构1. 区块链(block chain)2. 默克尔树(Merkle tree)3.哈希指针的问题比特币中的数据结构 1. 区块链(block chain) 哈希指针: (1)保存数值的位置…

基于 Python 实时图像获取及处理软件图像获取;图像处理;人脸识别设计 计算机毕设 附完整代码+论文 +报告

界面结果:图像获取;图像处理;人脸识别 程序结构设计 图形用户界面设计与程序结构设计是互为表里的。或者说,程序结构设计是软件设计最本质、最核心的内容。徒有界面而内部逻辑结构混乱的软件一无是处。 Windows 操作系统是一款图形化的操作系统,相比于早期的计算机使用的命…

XXL-JOB 任务调度平台实践

XXL-JOB 任务调度平台实践一、调度中心(服务端)1、从gitbub 获取项目源码:[https://github.com/xuxueli/xxl-job](https://github.com/xuxueli/xxl-job)2、从源码中得到SQL脚本创建和初始化数据库3、Maven 编译打包 xxl-job-admin 并部署为调度中心4、启动运行 xxl-…

Linux 编译器 gcc/g++

本文已收录至《Linux知识与编程》专栏! 作者:ARMCSKGT 演示环境:CentOS 7 目录 前言 正文 gcc/g常用命令 自定义可执行程序名命令-o 预处理指令-E 编译指令-S 汇编指令-c 链接指令gcc 命令巧记口诀 链接库 动态库-动态链接 静态库…

双11大型互动游戏“喵果总动员” 质量保障方案总结

推荐语:互动游戏是一个系统化工程,在笔者的“喵果总动员”质量方案中,可以看到为保障用户体验,我们在各个难点的解决方案, 例如:用线上压测能力支持业务及时调整各服务容量、通过强化学习覆盖游戏行业的测试…

设计师一定要知道这几个网站,解决你80%的设计素材。

本期推荐一波设计师必备的设计素材网站,设计党赶紧马住!能解决你日常设计中80%的素材。 1、菜鸟图库 菜鸟图库-免费设计素材下载 这是一个为新手设计师提供免费素材的设计网站,站内有超多平面模板、海报、UI设计、电商设计等相关素材&#x…

数据与C(浮点数)

目录 一.基本概念 二.声明和初始化 三.浮点数上溢和下溢 浮点数在数据类型上一共就两个,一个是float,另一个是double。但两个唯一的区别就在于double精度方面是float的两倍 一.基本概念 首先介绍浮点数的三种书写形式 第一种 数字 :103…

K近邻算法和KD树详细介绍及其原理详解

相关文章 K近邻算法和KD树详细介绍及其原理详解 文章目录相关文章前言一、K近邻算法二、KD树总结前言 K近邻算法一般是我们学习机器学习的入门算法,本篇文章详细介绍了K近邻算法,并对其原理进行了说明。同时,为了优化K近邻算法查找最近K个邻…

Spring工厂模式

解决方案1:工厂模式 可以使用抽象工厂模式,让StudentDao、StudentService的实现在在工厂中生成,而工厂可以根据配置 文件的指定类型来创建不同的对象,而且工厂本身一般是不变的。从而降低了对可以变的业务逻辑类的 依赖,接近的软…

字节青训前端笔记 | 数据可视化基础

本课程主要内容可以分为三个章节: 数据可视化的基本概念可视化设计的基本原则面向前端的可视化工具 本手册可以作为学生学习数据可视化的“学习指南”,按照手册所列内容,结合扩展资料进行系统的学习和实践。本课程没有讨论更为前沿的可视化…

推荐5款干净又实用的软件

我们在使用电脑的时候,总是会用到一些好用的软件,今天分享这5款干净又实用的软件,实在是太好用了,我不允许你不知道。 1.桌面工具——火柴 火柴是一款集快速搜索,程序启动、本地文件查找、网站直达、网络搜索等多种功…