获得日志记录之外的新视角:应用程序性能监控简介(APM)
作者:来自 Elastic David Hope
日志记录领域即将发生改变。在这篇文章中,我们将概述从单纯的日志记录到包含日志、跟踪和 APM 的完全集成解决方案的推荐流程。
通过 APM 和跟踪优先考虑客户体验
企业软件开发和运营已成为一个有趣的领域。我们拥有一些非常强大的工具,但作为一个行业,我们未能采用许多可以让我们的生活更轻松的工具。目前未充分利用的一种工具是应用程序性能监控 (application performance monitoring - APM) 和跟踪,尽管 OpenTelemetry 已经使采用它成为可能。
然而,日志记录无处不在。每个软件应用程序都有某种日志,故障排除的默认工作流程(即使在今天)是从客户和系统遇到的异常转到日志,然后从那里开始寻找解决方案。
这有各种挑战,其中主要挑战之一是日志通常没有提供足够的信息来解决问题。如今,许多服务返回模棱两可的 500 错误,几乎没有任何线索。如果根本没有错误或日志文件,或者问题是系统非常慢,该怎么办?仅靠日志记录无法帮助解决这些问题。这使得用户只能使用半损坏的系统和糟糕的用户体验。我们都曾遇到过这种情况,这令人非常沮丧。
我发现自己在问一个问题,为什么客户体验往往排在错误之后?如果客户体验是重中之重,那么应该制定策略来采用跟踪和 APM,并使其与日志记录一样重要。用户应该停止默认使用日志,而主要考虑日志,就像现在许多人所做的那样。这也需要对思维模型进行一些必要的改变。
实现这一目标的途径是什么?这正是我们将在这篇博文中探讨的内容。我们将首先讨论支持组织变革,然后概述从单纯的日志记录转向具有日志、跟踪和 APM 的完全集成解决方案的建议历程。
培养新的监控思维:如何推动 APM 和跟踪采用
要让团队转变故障排除思维,需要进行哪些组织变革?
首先,企业应该考虑需要在团队之间广泛分享的战略优先事项和目标。在非常大的组织中,有一件事可以帮助推动这一点,那就是考虑一个致力于可观察性的产品团队或一个拥有自己的路线图和优先事项的 CoE(Center of Excellence - 卓越中心)。
这个团队(无论是虚拟的还是永久的)应该从客户的角度出发,然后逆向工作,从关键问题开始,例如:我需要收集什么?我需要观察什么?我该如何行动?一旦团队成员理解了这些问题的答案,他们就可以开始思考推动这些结果所需的技术决策。
从跟踪和 APM 的角度来看,最令人担忧的领域是客户体验、服务水平目标和服务水平结果。从这里开始,组织可以开始实施工作计划,以不断改进和跨团队共享知识。这将有助于让团队围绕一个具有共同目标的共同框架保持一致。
在接下来的几个部分中,我们将经历一个四步旅程,以帮助你最大限度地利用 APM 和跟踪。此旅程将带你完成成功采用 APM 的以下关键步骤:
- 提取:你必须做出哪些选择才能激活跟踪并开始将跟踪数据提取到可观察性工具中?
- 集成:跟踪如何与日志集成以实现完整的端到端可观察性,除了简单的跟踪之外,你还可以利用哪些其他方法来更好地解决数据问题?
- 分析和 AIOPs:通过机器学习改善客户体验并减少噪音。
- 规模和总拥有成本:推出企业范围的跟踪并采用策略来处理数据量。
1. 采集 - ingest
为 APM 目的采集数据通常涉及 “instrumenting - 检测” 应用程序。在本节中,我们将探讨检测应用程序的方法,讨论一下采样,最后总结一下使用常见模式进行数据表示的注意事项。
开始使用检测
我们有哪些选项可以采集 APM 和跟踪数据?我们将讨论许多选项来帮助指导你,但首先让我们回顾一下。APM 有着悠久的历史 —— 在 APM 的最初实现中,人们主要关注计时方法,如下所示:
通常,你会有一个配置文件来指定要计时的方法,而 APM 实现会使用方法计时来检测指定的代码。
从这里开始,事情开始发展,APM 的第一批新增功能之一就是添加跟踪。
对于 Java,使用所谓的 Java 代理来实现一个系统来执行此操作相当简单。你只需指定 -javagent 命令行参数,代理代码就可以访问 Java 中的动态编译例程,这样它就可以在将代码编译为机器代码之前对其进行修改,从而允许你使用计时或跟踪例程 “wrap - 包装” 特定方法。因此,自动检测 Java 是原始 APM 供应商所做的第一件事。
OpenTelemetry 有这样的代理,大多数提供 APM 解决方案的可观察性供应商都有自己的专有方法来实现这一点,通常具有比开源工具更高级和不同的功能。
从那时起,事情发生了变化,Node.JS 和 Python 现在很流行。
因此,出现了自动检测这些语言运行时的方法,这些方法大多通过在启动代码之前将库注入代码来实现。OpenTelemetry 有一种方法可以在 Kubernetes 上使用 Operator 和 sidecar 来实现这一点,它支持 Python、Node.JS、Java 和 DotNet。
另一种选择是开始在你自己的代码中添加 APM 和跟踪 API 调用,这与添加日志记录功能没有什么不同。你甚至可能希望在代码中创建一个抽象来处理这个横切问题,尽管现在有了可以实现这一点的开放标准,这不再是一个问题。
你可以在下面和此处看到一个示例,了解如何将 OpenTelemetry 跨度(spans)和属性(attributes)添加到你的代码中以进行手动检测。
from flask import Flask
import monitor # Import the module
from opentelemetry.exporter.otlp.proto.grpc.trace_exporter import OTLPSpanExporter
import urllib
import osfrom opentelemetry import trace
from opentelemetry.sdk.resources import SERVICE_NAME, Resource
from opentelemetry.sdk.trace import TracerProvider
from opentelemetry.sdk.trace.export import BatchSpanProcessor
from opentelemetry.instrumentation.requests import RequestsInstrumentor# Service name is required for most backends
resource = Resource(attributes={SERVICE_NAME: "your-service-name"
})provider = TracerProvider(resource=resource)
processor = BatchSpanProcessor(OTLPSpanExporter(endpoint=os.getenv('OTEL_EXPORTER_OTLP_ENDPOINT'),headers="Authorization=Bearer%20"+os.getenv('OTEL_EXPORTER_OTLP_AUTH_HEADER')))provider.add_span_processor(processor)
trace.set_tracer_provider(provider)
tracer = trace.get_tracer(__name__)
RequestsInstrumentor().instrument()# Initialize Flask app and instrument it
app = Flask(__name__)@app.route("/completion")
@tracer.start_as_current_span("do_work")
def completion():span = trace.get_current_span()if span:span.set_attribute("completion_count",1)
通过以这种方式实施 APM,你甚至可以通过将所需的所有日志记录信息存储在跨度属性(span attributes)、异常(exceptions)和指标(metrics)中来消除进行任何日志记录的需要。缺点是你只能使用自己拥有的代码来执行此操作,因此你将无法通过这种方式删除所有日志。
采样 - sampling
许多人没有意识到 APM 是一个昂贵的过程。它会为你的应用程序增加大量 CPU 周期和内存,尽管有很多价值,但肯定需要做出权衡。
你是否应该 100% 地采样所有内容并承担成本?还是应该考虑明智的权衡,使用更少的样本甚至基于尾部的采样,许多产品通常都支持这种采样?在这里,我们将讨论两种最常见的采样技术 —— 基于头部的采样(head-based)和基于尾部的采样(tail-based) —— 以帮助你做出决定。
基于头部的采样
在这种方法中,采样决策是在跟踪开始时做出的,通常是在服务或应用程序的入口点。跟踪的采样速率是固定的,并且此决策会传播到分布式跟踪中涉及的所有服务。
使用基于头部的采样,你可以使用配置来控制速率,从而控制采样并报告给 APM 服务器的请求百分比。例如,采样率为 0.5 意味着只有 50% 的请求被采样并发送到服务器。这对于减少收集的数据量非常有用,同时仍能保持应用程序性能的代表性样本。
基于尾部的采样
与基于头部的采样不同,基于尾部的采样在整个跟踪完成后做出采样决策。这允许根据实际跟踪数据做出更智能的采样决策,例如仅报告有错误的跟踪或超过特定延迟阈值的跟踪。
我们建议使用基于尾部的采样,因为它最有可能减少噪音并帮助你专注于最重要的问题。它还有助于降低数据存储方面的成本。然而,基于尾部的采样的一个缺点是它会导致从 APM 代理生成更多数据。这可能会占用应用程序的更多 CPU 和内存。
OpenTelemetry 语义约定和 Elastic Common Schema
OpenTelemetry 规定了语义约定或语义属性,以便为各种操作和数据类型建立统一的名称。遵守这些约定有助于实现代码库、库和平台之间的标准化,最终简化监控流程。
创建 OpenTelemetry 跨度进行跟踪非常灵活,允许实施者使用特定于操作的属性对其进行注释。这些跨度表示系统内部和系统之间的特定操作,通常涉及广泛认可的协议,如 HTTP 或数据库调用。为了有效地表示和分析监控系统中的跨度,需要补充信息,具体取决于协议和操作类型。
统一不同语言的归因方法对于操作员来说至关重要,这样他们就可以轻松地关联和交叉分析来自多语言微服务的遥测数据,而无需掌握特定于语言的细微差别。
Elastic 最近将 Elastic Common Schema 贡献给 OpenTelemetry,增强了语义约定以涵盖日志和安全性。
遵守共享模式可带来巨大好处,使操作员能够快速识别复杂的交互并关联日志、指标和跟踪,从而加快根本原因分析并减少搜索日志和确定特定时间范围所花费的时间。
我们提倡在应用程序中定义跟踪、指标和日志数据时遵守 ECS 等既定模式,尤其是在开发新代码时。这种做法将在解决问题时节省时间和精力。
2. 集成 - integrate
集成对于 APM 非常重要。你的解决方案与其他工具和技术(如云)的集成程度,以及将日志和指标集成到跟踪数据中的能力,对于充分了解客户体验至关重要。此外,大多数 APM 供应商都有用于综合监控(synthetic monitoring)和分析的邻近解决方案,以便获得更深入的视角来增强你的 APM。我们将在下一节中探讨这些主题。
APM + 日志 = 超能力!
由于 APM 代理可以检测代码,因此它们也可以检测用于日志记录的代码。这样,你可以直接在 APM 中捕获日志行。这通常很容易启用。
启用此功能后,你还可以自动注入以下有用的字段:
- service.name、service.version、service.environment
- trace.id、transaction.id、error.id
这意味着日志消息将自动与事务(transaction)相关联,如下所示,从而更容易减少平均解决时间 (MTTR) 并找到大海捞针:
如果你可以使用此功能,我们强烈建议你将其打开。
在 Kubernetes 中部署 APM
人们通常希望在 Kubernetes 环境中部署 APM,而跟踪对于监控云原生环境中的应用程序至关重要。有三种不同的方法可以解决这个问题。
1. 使用 sidecar 自动检测
使用 Kubernetes,可以使用 init 容器和一些可以动态修改 Kubernetes 清单的东西来自动检测你的应用程序。
init 容器将仅用于在启动时将所需的库或 jar 文件复制到容器中,你需要将其添加到主 Kubernetes pod 中。然后,你可以使用 Kustomize 添加所需的命令行参数来引导你的代理。
如果你不熟悉它,Kustomize 会动态添加、删除或修改 Kubernetes 清单。它甚至可以作为 Kubernetes CLI 的标志使用 - 只需执行 kubectl -k。
OpenTelemetry 有一个 operator,可以自动为你完成所有这些操作(无需 Kustomize),适用于 Java、DotNet、Python 和 Node.JS,许多供应商也有自己的运算符或 helm charts,可以实现相同的结果。
2. 将 APM 烘焙到容器或代码中
在 Kubernetes(实际上是任何容器化环境)中部署 APM 的第二种选择是使用 Docker 将 APM 代理和配置烘焙到 dockerfile 中。
在此处查看使用 OpenTelemetry Java 代理的示例:
# Use the official OpenJDK image as the base image
FROM openjdk:11-jre-slim# Set up environment variables
ENV APP_HOME /app
ENV OTEL_VERSION 1.7.0-alpha
ENV OTEL_JAVAAGENT_URL https://github.com/open-telemetry/opentelemetry-java-instrumentation/releases/download/v${OTEL_VERSION}/opentelemetry-javaagent-${OTEL_VERSION}-all.jar# Create the application directory
RUN mkdir $APP_HOME
WORKDIR $APP_HOME# Download the OpenTelemetry Java agent
ADD ${OTEL_JAVAAGENT_URL} /otel-javaagent.jar# Add your Java application JAR file
COPY your-java-app.jar $APP_HOME/your-java-app.jar# Expose the application port (e.g. 8080)
EXPOSE 8080# Configure the OpenTelemetry Java agent and run the application
CMD java -javaagent:/otel-javaagent.jar \-Dotel.resource.attributes=service.name=your-service-name \-Dotel.exporter.otlp.endpoint=your-otlp-endpoint:4317 \-Dotel.exporter.otlp.insecure=true \-jar your-java-app.jar
3. 使用服务网格 (Envoy/Istio) 进行跟踪
最后一个选项是使用服务网格。服务网格是专用的基础设施层,用于处理微服务架构中的服务间通信。它提供了一种透明、可扩展且高效的方式来管理和控制服务之间的通信,使开发人员能够专注于构建应用程序功能,而不必担心服务间通信的复杂性。
这样做的好处是,我们可以在代理中激活跟踪,从而了解服务之间的请求。我们不必为此更改任何代码,甚至不必运行 APM 代理;我们只需打开代理中存在的 OpenTelemetry 收集器 — 因此这可能是开销最低的解决方案。了解有关此选项的更多信息。
合成通用分析 - sythetic universal profiling
大多数 APM 供应商都为主要 APM 用例添加了附加组件。通常,我们会看到合成和持续分析被添加到 APM 解决方案中。APM 可以与两者集成,将这些技术结合在一起以提供更多问题见解具有一定的价值。
合成
合成监控是一种通过模拟用户交互和流量来衡量 Web 应用程序、网站和 API 的性能、可用性和可靠性的方法。它涉及创建模拟真实用户行为的脚本或自动化测试,例如浏览页面、填写表单或单击按钮,然后从不同位置和设备定期运行这些测试。
这使开发和运营团队能够比他们可能更早地发现问题,在许多情况下比真实用户更早发现问题。
合成可以与 APM 集成 - 在脚本运行时将 APM 代理注入网站,因此即使你最初没有将最终用户监控放入你的网站,也可以在运行时注入。这通常发生在没有任何用户输入的情况下。从那里,每个请求的跟踪 ID 可以通过系统的各个层传递下去,从而允许团队从合成脚本一直跟踪请求到应用程序堆栈的最低层(例如数据库)。
通用分析 - universal profiling
“分析”是一种分析程序复杂性的动态方法,例如 CPU 利用率或函数调用的频率和持续时间。通过分析,你可以准确找到应用程序中消耗最多资源的部分。“持续分析 - Continuous profiling ” 是分析的更强大版本,它增加了时间维度。通过了解系统随时间变化的资源,你可以找到、调试和修复与性能相关的问题。
通用分析是对此的进一步扩展,它允许你始终捕获有关系统中运行的所有代码的分析信息。使用 eBPF 之类的技术可以让你查看系统中的所有函数调用,包括 Kubernetes 运行时之类的内容。这样做可以让你最终看到未知的未知数 —— 你不知道是问题的东西。这与 APM 非常不同,后者实际上是关于跟踪单个跟踪和请求以及整体客户体验。通用分析可以解决你甚至不知道存在的问题,甚至可以回答 “我最昂贵的代码行是什么?” 这个问题。
通用分析可以链接到 APM,例如,向你显示特定客户问题期间发生的配置文件,或者通过查看线程级别存在的全局状态将配置文件直接链接到跟踪。这些技术一起使用时可以产生奇效。
通常,配置文件被视为如下所示的 “火焰图”。方框表示执行特定函数所花费的 “CPU 上” 时间量。
3. 分析和 AIOps
APM 的有趣之处在于它开辟了一个全新的分析世界,而不仅仅是日志。突然之间,你就可以访问应用程序内部的信息流。
这使你可以轻松捕获诸如特定客户当前在你最重要的电子商务商店上花费的金额之类的信息,或者查看经纪应用程序中的失败交易,以了解这些失败造成的收入损失。你甚至可以应用机器学习算法来预测未来的支出或查看这些数据中发生的异常,从而为你提供一个了解业务运作方式的新窗口。
在本节中,我们将介绍如何做到这一点以及如何充分利用这个新世界,以及如何将 AIOps 实践应用于这些新数据。我们还将讨论如何为 APM 数据设置 SLI 和 SLO。
将业务数据放入你的跟踪中
通常有两种方法可以将业务数据放入你的跟踪中。你可以修改代码并添加 Span 属性,此处提供了一个示例,如下所示。或者你可以编写扩展或插件,这样做的好处是避免代码更改。OpenTelemetry 支持在其自动检测代理中添加扩展。大多数其他 APM 供应商通常都有类似的东西。
def count_completion_requests_and_tokens(func):@wraps(func)def wrapper(*args, **kwargs):counters['completion_count'] += 1response = func(*args, **kwargs)token_count = response.usage.total_tokensprompt_tokens = response.usage.prompt_tokenscompletion_tokens = response.usage.completion_tokenscost = calculate_cost(response)strResponse = json.dumps(response)# Set OpenTelemetry attributesspan = trace.get_current_span()if span:span.set_attribute("completion_count", counters['completion_count'])span.set_attribute("token_count", token_count)span.set_attribute("prompt_tokens", prompt_tokens)span.set_attribute("completion_tokens", completion_tokens)span.set_attribute("model", response.model)span.set_attribute("cost", cost)span.set_attribute("response", strResponse)return responsereturn wrapper
使用业务数据既有趣又有利可图
一旦你跟踪了业务数据,就可以开始享受其中的乐趣了。请看下面金融服务欺诈团队的示例。在这里,我们正在跟踪交易 — 大型商业客户的平均交易价值。至关重要的是,我们可以查看是否存在任何异常交易。
其中很多都是由机器学习驱动的,它可以对交易进行分类或进行异常检测。一旦你开始捕获数据,就可以做很多有用的事情,而且有了灵活的平台,将机器学习模型集成到这个过程中就变得轻而易举了。
SLI 和 SLO
服务级别指标 (service level indicators - SLIs) 和服务级别目标 (service level objectives - SLOs) 是维护和增强应用程序性能的关键组件。SLI 代表关键性能指标,例如延迟、错误率和吞吐量,有助于量化应用程序的性能,而 SLO 则建立目标性能级别以满足用户期望。
通过选择相关的 SLI 并设置可实现的 SLO,组织可以使用 APM 工具更好地监控其应用程序的性能。根据应用程序要求、用户期望或竞争格局的变化不断评估和调整 SLI 和 SLO 可确保应用程序保持竞争力并提供卓越的用户体验。
为了定义和跟踪 SLI 和 SLO,APM 成为了解用户体验所需的关键视角。实施 APM 后,我们建议组织执行以下步骤。
- 定义跟踪它们所需的 SLO 和 SLI。
- 定义 SLO 预算及其计算方式。反映业务观点并设定切合实际的目标。
- 从用户体验的角度定义要衡量的 SLI。
- 定义不同的警报和寻呼规则,仅在面向客户的 SLO 降级时寻呼,记录症状警报,在关键症状警报时通知。
综合监控和最终用户监控 (EUM) 还可以帮助从用户的角度获取了解延迟、吞吐量和错误率所需的更多数据,这对于从中获得良好的业务重点指标和数据至关重要。
4. 规模和总拥有成本
随着视角的扩大,客户经常会遇到可扩展性和总拥有成本问题。所有这些新数据可能会让人不知所措。幸运的是,你可以使用各种技术来处理这个问题。跟踪本身实际上可以帮助解决容量挑战,因为你可以分解非结构化日志并将其与跟踪相结合,从而提高效率。你还可以使用不同的采样方法来应对规模挑战(即我们之前提到的两种技术)。
除此之外,对于大型企业规模,我们可以使用 Kafka 或 Pulsar 等流式管道来管理数据量。这还有一个额外的好处,你可以免费获得:如果你关闭使用数据的系统或它们面临中断,你丢失数据的可能性就会降低。
有了这种配置,你的 “可观察性管道” 架构将如下所示:
这将彻底将你的数据源与你选择的可观察性解决方案分离开来,这将为你的可观察性堆栈提供未来保障,使你能够实现大规模,并减少对特定供应商代码的依赖,从而收集数据。
我们建议你做的另一件事是智能地进行检测。这将带来两个好处:你将在被检测的应用程序中重新获得一些 CPU 周期,并且你的后端数据收集系统将有更少的数据需要处理。例如,如果你知道你不想跟踪对特定端点的调用,则可以将这些类和方法从检测中排除。
最后,数据分层是一种管理数据存储的变革性方法,可以显着降低企业的总拥有成本 (total cost of ownership - TCO)。首先,它允许组织根据其可访问性需求和数据的价值将数据存储在不同类型的存储介质中。例如,经常访问的高价值数据可以存储在昂贵的高速存储中,而访问频率较低的低价值数据则可以存储在更便宜、更慢的存储中。
这种方法通常包含在云存储解决方案中,通过确保企业只需支付他们在任何给定时间所需的存储费用,可以实现成本优化。此外,它还提供了根据需求扩展或缩减的灵活性,无需在存储基础设施上投入大量资本支出。这种可扩展性还减少了为满足潜在未来需求而进行昂贵的过度配置的需要。
结论
在当今竞争激烈、节奏飞快的软件开发领域,仅仅依靠日志记录已不足以确保一流的客户体验。通过采用 APM 和分布式跟踪,组织可以更深入地了解其系统,主动检测和解决问题,并保持强大的用户体验。
在本博客中,我们探讨了从仅日志记录方法转变为集成日志、跟踪和 APM 的全面可观察性策略的过程。我们讨论了培养优先考虑客户体验的新监控思维的重要性,以及推动 APM 和跟踪采用所需的必要组织变革。我们还深入研究了旅程的各个阶段,包括数据提取、集成、分析和扩展。
通过理解和实施这些概念,组织可以优化其监控工作,减少 MTTR,并让客户满意。最终,通过 APM 和跟踪优先考虑客户体验可以让企业在当今充满挑战的环境中更成功、更具弹性。
在 Elastic 了解有关 APM 的更多信息。
原文:Gaining new perspectives beyond logging: An introduction to application performance monitoring — Elastic Observability Labs
相关文章:
获得日志记录之外的新视角:应用程序性能监控简介(APM)
作者:来自 Elastic David Hope 日志记录领域即将发生改变。在这篇文章中,我们将概述从单纯的日志记录到包含日志、跟踪和 APM 的完全集成解决方案的推荐流程。 通过 APM 和跟踪优先考虑客户体验 企业软件开发和运营已成为一个有趣的领域。我们拥有一些非…...
如何避免缓存击穿?超融合常驻缓存和多存储池方案对比
作者:SmartX 解决方案专家 钟锦锌 很多运维人员都知道,混合存储介质配置可能会带来“缓存击穿”的问题,尤其是大数据分析、数据仓库等需要频繁访问“冷数据”的应用场景,缓存击穿可能会更频繁地出现,影响业务运行。除…...
口语笔记——祈使句用法
省略主语 (You give me) a cup of tea, please. 一杯茶(You wait for) another minute. 两等一分钟(You) keep quiet. 保持安静give me a break. 饶了我吧take your hand off. 把你的手拿开take this thing away 把这东西拿开never talk to strangers. 永远不要跟陌生人说话Do…...
SQL连续登录问题(详细案例分析)
如果要统计用户活跃度,那就涉及连续登录问题,接下来将举一个简单的例子来详细说明这个问题: 一、创建一些模拟数据 一些测试数据如下: deviceid1,2022-10-26,2022-10-26,2022-11-01 deviceid1,2022-10-26,2022-11-03,2022-11-0…...
Next.js 系统性教学:深入理解缓存与数据优化策略
更多有关Next.js教程,请查阅: 【目录】Next.js 独立开发系列教程-CSDN博客 目录 前言 1. 缓存的基本概念 1.1 缓存的作用 1.2 Next.js 中的缓存策略 2. Next.js 的缓存机制 2.1 请求记忆化(Request Memoization) 2.1.1 什…...
【PyTorch】(基础六)---- 搭建卷积神经网络
关于神经网络中激活函数、卷积层、池化层等底层原理,我不会在本文中详解,但是关于pytorch中如何使用对应的方法实现这些层的功能我会进行解释,如果你想要了解一些关于神经网络底层的知识,我十分推荐你去看一下吴恩达老师的深度学习…...
【JAVA项目】基于ssm的【美食推荐管理系统】
【JAVA项目】基于ssm的【美食推荐管理系统】 技术简介:采用JSP技术、B/S架构、SSM框架、MySQL技术等实现。 系统简介:美食推荐管理系统,在系统首页可以查看首页、热门美食、美食教程、美食店铺、美食社区、美食资讯、我的、跳转到后台等内容。…...
adb 常用命令笔记
adb connect <ip> #连接指定ip adb disconnect <ip> #断开连接指定ip adb devices #查看连接中的设备 adb install <flie> #安装apk adb uninstall <packageName> #卸载app adb -s install <flie> #指定设备安装 adb shell pm list package…...
[代码随想录Day32打卡] 理论基础 509. 斐波那契数 70. 爬楼梯 746. 使用最小花费爬楼梯
理论基础 题型 动归基础(这一节就是基础题)背包问题打家劫舍股票问题子序列问题 动态规划五部曲 确定dp数组及其下标的含义确定递推公式dp数组如何初始化遍历顺序打印dp数组 509. 斐波那契数 简单~ dp数组及下标含义: dp[i]表示第i各斐…...
android NumberPicker隐藏分割线或修改颜色
在 Android 中,可以通过以下几种方法隐藏 NumberPicker 的分割线: 使用 XML 属性设置 在布局文件中的 NumberPicker 标签内添加 android:selectionDividerHeight"0dp" 属性,将分割线的高度设置为 0,从而达到隐藏分割线…...
7-2 二分查找
输入n值(1<n<1000)、n个非降序排列的整数以及要查找的数x,使用二分查找算法查找x,输出x所在的下标(0~n-1)及比较次数。若x不存在,输出-1和比较次数。 输入格式: 输入共三行: 第一行是n值࿱…...
mid360使用cartorapher进行3d建图导航
1. 添加urdf配置文件: 添加IMU配置关节点和laser关节点 <!-- imu livox --> <joint name"livox_frame_joint" type"fixed"> <parent link"base_link" /> <child link"livox_frame" /> <o…...
Ubuntu安装grafana
需求背景:管理服务器,并在线预警,通知 需求目的: 及时获取服务器状态 技能要求: 1、ubuntu 2、grafana 3、prometheus 4、node 步骤: 一、grafana安装 1、准备系统环境,配置号网络 2、…...
Java版-图论-最短路-Floyd算法
实现描述 网络延迟时间示例 根据上面提示,可以计算出,最大有100个点,最大耗时为100*wi,即最大的耗时为10000,任何耗时计算出来超过这个值可以理解为不可达了;从而得出实现代码里面的: int maxTime 10005…...
可视化建模以及UML期末复习篇----UML图
这是一篇相对较长的文章,如你们所见,比较详细,全长两万字。我不建议你们一次性看完,直接跳目录找你需要的知识点即可。 --------欢迎各位来到我UML国! 一、UML图 总共有如下几种: 用例图(Use Ca…...
HTML常见标签列表,涵盖了多种用途的标签。
文档结构标签 <!DOCTYPE html>:声明文档类型,告诉浏览器使用HTML5标准。<html>:HTML文档的根元素。<head>:包含文档的元数据(meta-data),如标题、字符集、样式表链接、脚本等…...
FPGA 16 ,Verilog中的位宽:深入理解与应用
目录 前言 一. 位宽的基本概念 二. 位宽的定义方法 1. 使用向量变量定义位宽 ① 向量类型及位宽指定 ② 位宽范围及位索引含义 ③ 存储数据与字节数据 2. 使用常量参数定义位宽 3. 使用宏定义位宽 4. 使用[:][-:]操作符定义位宽 1. 详细解释 : 操作符 -: 操作符 …...
vue-生命周期
Vue 的生命周期是指 Vue 实例从创建到销毁期间经历的一系列阶段。每个阶段都有相应的钩子函数(Lifecycle Hooks),允许开发者在这些关键时刻执行自定义逻辑。 一、钩子函数 1. 创建阶段 beforeCreate 在实例初始化之后,数据观测 …...
浅谈Kubernetes(K8s)之RC控制器与RS控制器
1.RC控制器 1.1RC概述 Replication Controller 控制器会持续监控正在运行的Pod列表,并保证相应类型的Pod的数量与期望相符合,如果Pod数量过少,它会根据Pod模板创建新的副本,反之则会删除多余副本。通过RC可实现了应用服务的高可用…...
本题要求采用选择法排序,将给定的n个整数从大到小排序后输出。
#include <stdio.h> #define MAXN 10 int main() { int i, index, k, n, temp; int a[MAXN]; scanf("%d", &n); for (i 0; i < n; i) { scanf("%d", &a[i]); } // 外层循环控制排序轮数,一共需要n-1轮 for (k 0; k < n…...
Linux: glibc: 频繁调用new/delete会不会导致内存的碎片
最近同事问了一个问题:频繁调用new/delete会不会导致内存的碎片。 下面是我想到的一些回答, glibc的内存处理机制,是在释放的时候会自动将小块内存整合成大块内存,为接下来满足大块的需求的可能。而且程序也不是一直占着内存不释放(如果是一直不释放,要考虑是不是内存泄漏…...
量子变分算法---损失函数
引子 关于损失函数,我们知道在强化学习中,会有一个函数,用来表示模型每一次行为的分数,通过最大化得分,建立一个正反馈机制,若模型为最优则加分最多,若决策不佳则加很少分或者扣分。而在神经网络…...
计算机的性能评估
目录 计算机的性能评估 确定性能指标 考虑通讯因素 考虑机器过热因素 综合评估模型 动态评估与调整 计算机的性能评估 在分布式计算机系统中,综合考虑各种因素来评估性能是一个复杂但重要的问题。以下是一种可能的方法来综合考虑评估分布式计算机性能,动态地考虑实际情…...
大数据之国产数据库_OceanBase数据库002_在centos7.9上_安装部署OceanBase001_踩坑指南_亲测可用
部署前最好看一下,部署前的要求, 主要是系统 以及系统内核版本,还有比如清理一下缓存等,按照做一做. 这些都是前置条件. 清一下缓存. 也就是说官网给的前置的条件,都要根据说明去执行一遍,如果不执行可能后面安装会报错. 然后用户最好也去创建一个用户. 注意前置...
【ETCD】【源码阅读】深入解析 EtcdServer.run 函数
EtcdServer.run 是 etcd 的核心运行逻辑之一,负责管理 Raft 状态机的应用、事件调度以及集群的核心操作。本文将逐步从源码层面分析 run 函数的逻辑,帮助读者理解其内部机制和设计思想。 函数签名与关键职责 func (s *EtcdServer) run() {... }关键职责…...
springboot/ssm校内订餐系统Java代码web项目美食外卖点餐配送源码
springboot/ssm校内订餐系统Java代码web项目美食外卖点餐配送源码 基于springboot(可改ssm)vue项目 开发语言:Java 框架:springboot/可改ssm vue JDK版本:JDK1.8(或11) 服务器:tomcat 数据库ÿ…...
floodfill算法
目录 什么是floodfill算法 题目一——733. 图像渲染 - 力扣(LeetCode) 题目二——200. 岛屿数量 - 力扣(LeetCode) 题目三——695. 岛屿的最大面积 - 力扣(LeetCode) 题目四—— 130. 被围绕的区域 …...
【JAVA】六亮增加贴
James Gosling(詹姆斯.高斯林) Java 语言源于 1991 年 4 月,Sun 公司 James Gosling博士 领导的绿色计划(Green Project) 开始启动,此计划最初的目标是开发一种能够在各种消费性电子产品(如机顶盒、冰箱、收音机等)上运行的程序…...
git提交时出现merge branch main of xxx
git提交时出现merge branch main of xxx 原因: 1、同事commit了一个修改A,push到remote 2、我把这个修改直接pull了下来(pull是fetchmerge的操作,自动合并到本地workspace) 3、同事因为后续的commit有冲突,…...
lstm 输入数据的形状是怎么样的,他有两种输入方式,通过参数 batch_first来设置 默认是False
lstm 输入数据的形状是怎么样的,他有两种输入方式,通过参数 batch_first来设置 默认是False 当batch_firstFalse时,LSTM输入的数据形状通常是一个三维张量,其维度顺序为[sequence_length, batch_size, input_size]。下面是对这些维…...
新手如何搭建自己的网站/互联网营销师培训课程
一、Spring中Bean及Bean的理解[1] Bean在Spring和SpringMVC中无所不在,将这个概念内化很重要,下面分享一下我的想法: 一、Bean是啥 1、Java面向对象,对象有方法和属性,那么就需要对象实例来调用方法和属性(…...
中小型网站设计哪家好/官网seo优化找哪家做
347. 前 K 个高频元素 本题还是细节处理比较多,我不太熟悉小顶堆大顶堆,这个题花了一个多小时在搞基础,具体细节在代码部分。 class Solution {public int[] topKFrequent(int[] nums, int k) {Map<Integer,Integer> map…...
响应式网站软件/头条广告入口
CSS作为Web标准的一部分,已经成为现代网页设计中必不可少的关键要素。网页学习网CSS教程专题系统地讲解了CSS样式表的基础理论和实际运用技术,通过大量实例对CSS进行深入浅出的分析;透彻地讲解CSS核心技术的基础上,深入到各个CSS实…...
wordpress能做什么/微信广告平台推广
1,软硬件准备 软件:推荐使用VMwear,我用的是VMwear 10 镜像:CentOS 7 ,如果没有镜像可以去官网下载:https://www.centos.org/download/ 选择一个节点下载CentOS 7: 2.虚拟机准备 1)打开VMwea,点击左上角文…...
宁波网站制作流程/宁波seo优化外包公司
[hadoopmaster1 ~]$ cat zookeeper/conf/zoo.cfg # The number of milliseconds of each tick 每个心跳的时长 单位为毫秒 tickTime2000 # The number of ticks that the initial # synchronization phase can take 初始化同步时期的心跳数 initLimit10 # The number of tick…...
做网站调用无广告视频/引流推广方案
实例在所有的例子中,我们将使用 XML 文件 books.xml,以及 JavaScript 函数 loadXMLDoc()。下面的代码片段创建并向第一个 元素追加了一个节点,然后输出第一个 元素的的所有子节点:xmlDocloadXMLDoc("books.xml");xxmlDo…...