当前位置: 首页 > news >正文

第 32 章 - Go语言 部署与运维

在Go语言的应用开发中,部署与运维是一个非常重要的环节。它不仅关系到应用能否顺利上线,还直接影响到应用的性能、安全性和可维护性。以下是根据您的需求整理的关于Go语言应用的打包和发布、容器化部署、监控和日志管理的相关内容。

1. 应用的打包和发布

打包
  • 静态编译:Go语言支持跨平台编译,可以使用GOOSGOARCH环境变量指定目标操作系统和架构。例如,要为Linux x86_64系统编译一个静态二进制文件,可以执行:
    GOOS=linux GOARCH=amd64 go build -o myapp .
    
  • 资源文件打包:如果应用依赖于外部资源文件(如配置文件、静态文件等),可以使用第三方工具如go-bindata将这些文件嵌入到二进制文件中,方便发布和部署。
发布
  • 版本控制:使用Git等版本控制系统来管理代码变更,确保每次发布的版本都是可追溯的。
  • 持续集成/持续部署(CI/CD):利用Jenkins、GitHub Actions、GitLab CI等工具实现自动化构建、测试和部署流程,提高效率并减少人为错误。

2. 容器化部署

  • Dockerfile:编写Dockerfile定义应用运行所需的环境,包括基础镜像、安装依赖、复制应用文件等步骤。例如:
    FROM golang:1.17-alpine AS builder
    WORKDIR /app
    COPY . .
    RUN CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o app .FROM alpine:latest
    RUN apk --no-cache add ca-certificates
    WORKDIR /root/
    COPY --from=builder /app/app .
    CMD ["./app"]
    
  • Docker Compose:对于多服务应用,可以使用Docker Compose来管理服务间的依赖关系,简化本地开发环境搭建。
  • Kubernetes:在生产环境中,推荐使用Kubernetes进行容器集群管理和调度,以实现高可用性和弹性伸缩。

3. 监控和日志

  • 日志管理:使用logrus等库来格式化输出日志信息,并通过日志级别控制输出量。生产环境中应将日志发送到集中式日志管理系统,如ELK Stack或Loki。
  • 性能监控:可以通过Prometheus搭配Golang客户端库(如prometheus/client_golang)收集应用指标数据,设置警报规则以便及时发现和处理问题。
  • 健康检查:在应用中实现HTTP接口供外部调用,用于检查应用的状态是否正常。这对于Kubernetes中的自动恢复机制尤为重要。

案例及源代码

假设我们有一个简单的Web服务,该服务提供了一个返回当前时间的API。下面是一些关键部分的示例代码:

main.go
package mainimport ("fmt""net/http""time"
)func main() {http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {fmt.Fprintf(w, "Current time is %s", time.Now().Format(time.RFC3339))})http.ListenAndServe(":8080", nil)
}
Dockerfile
FROM golang:1.17-alpine AS builder
WORKDIR /app
COPY . .
RUN CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o app .FROM alpine:latest
RUN apk --no-cache add ca-certificates
WORKDIR /root/
COPY --from=builder /app/app .
CMD ["./app"]
监控和日志

可以添加对logrusprometheus的支持,以便更好地监控应用状态和性能。这通常涉及到修改main.go以包含相应的库初始化和指标收集逻辑。

4. 监控和日志管理

日志管理
  • 日志格式化:使用logrus库可以方便地格式化日志输出,支持多种日志级别(如Debug、Info、Warn、Error等)。

    package mainimport (log "github.com/sirupsen/logrus"
    )func main() {log.Info("This is an info message")log.Warn("This is a warning message")log.Error("This is an error message")
    }
    
  • 日志输出:可以将日志输出到标准输出(stdout)、文件、甚至是远程日志服务器。

    log.SetOutput(os.Stdout)
    
  • 集中式日志管理:使用ELK Stack(Elasticsearch、Logstash、Kibana)或Loki等工具集中管理日志,便于搜索和分析。

    • ELK Stack:Logstash负责收集日志,Elasticsearch存储日志,Kibana提供可视化界面。
    • Loki:轻量级的日志聚合系统,适合Kubernetes环境。
性能监控
  • Prometheus:使用Prometheus进行性能监控,可以收集各种指标(如请求延迟、QPS等)。

    • 客户端库:使用prometheus/client_golang库在应用中暴露指标。
    • 中间件:可以使用中间件(如promhttp)来自动收集HTTP请求的指标。
    package mainimport ("net/http""github.com/prometheus/client_golang/prometheus""github.com/prometheus/client_golang/prometheus/promhttp"
    )var requestCounter = prometheus.NewCounterVec(prometheus.CounterOpts{Name: "http_requests_total",Help: "Total number of HTTP requests.",},[]string{"code"},
    )func init() {prometheus.MustRegister(requestCounter)
    }func handler(w http.ResponseWriter, r *http.Request) {// Your application logic herew.Write([]byte("Hello, world!"))requestCounter.WithLabelValues(http.StatusOK).Inc()
    }func main() {http.HandleFunc("/", handler)http.Handle("/metrics", promhttp.Handler())http.ListenAndServe(":8080", nil)
    }
    
  • Grafana:使用Grafana可视化Prometheus收集的数据,创建仪表板监控应用的性能。

5. 安全性

  • 输入验证:对用户输入进行严格的验证,防止SQL注入、XSS等攻击。

    func validateInput(input string) error {if len(input) > 100 {return errors.New("input too long")}// Additional validation logicreturn nil
    }
    
  • HTTPS:使用TLS证书启用HTTPS,保护数据传输的安全。

    certFile := "path/to/cert.pem"
    keyFile := "path/to/key.pem"
    http.ListenAndServeTLS(":443", certFile, keyFile, nil)
    
  • 身份验证和授权:实现身份验证(如JWT)和授权(如RBAC)机制,保护敏感操作。

    • JWT:使用jwt-go库生成和验证JSON Web Tokens。
    • RBAC:基于角色的访问控制,限制不同角色的权限。

6. 高可用性和容错

  • 负载均衡:使用Nginx或HAProxy等工具实现负载均衡,提高应用的可用性和性能。

    upstream myapp {server 192.168.1.1:8080;server 192.168.1.2:8080;
    }server {listen 80;location / {proxy_pass http://myapp;}
    }
    
  • 自动扩展:在Kubernetes中使用Horizontal Pod Autoscaler (HPA) 根据CPU或内存使用率自动扩展Pod数量。

    apiVersion: autoscaling/v2beta2
    kind: HorizontalPodAutoscaler
    metadata:name: myapp
    spec:scaleTargetRef:apiVersion: apps/v1kind: Deploymentname: myappminReplicas: 2maxReplicas: 10metrics:- type: Resourceresource:name: cputarget:type: UtilizationaverageUtilization: 50
    
  • 故障恢复:配置Kubernetes的Pod和Deployment策略,确保应用在出现故障时能够自动重启和恢复。

7. 持续集成/持续部署 (CI/CD)

  • GitHub Actions:使用GitHub Actions实现自动化构建、测试和部署流程。
    name: CI/CD Pipelineon:push:branches: [ main ]pull_request:branches: [ main ]jobs:build:runs-on: ubuntu-lateststeps:- name: Checkout codeuses: actions/checkout@v2- name: Set up Gouses: actions/setup-go@v2with:go-version: 1.17- name: Buildrun: go build -o myapp .- name: Testrun: go test -v ./...- name: Docker Loginuses: docker/login-action@v1with:username: ${{ secrets.DOCKER_USERNAME }}password: ${{ secrets.DOCKER_PASSWORD }}- name: Build and push Docker imagerun: |docker build -t myapp:latest .docker push myapp:latest- name: Deploy to Kubernetesuses: kubernetes-sigs/kustomize-github-actions/deploy@v1with:kubernetes-context: ${{ secrets.KUBE_CONTEXT }}kustomization-directory: k8s
    

8. 示例项目结构

一个典型的Go语言Web应用项目结构可能如下所示:

myapp/
├── cmd/
│   └── myapp/
│       └── main.go
├── internal/
│   ├── handlers/
│   │   └── handler.go
│   ├── middleware/
│   │   └── middleware.go
│   └── models/
│       └── model.go
├── pkg/
│   └── utils/
│       └── utils.go
├── Dockerfile
├── go.mod
├── go.sum
├── k8s/
│   ├── deployment.yaml
│   ├── service.yaml
│   └── hpa.yaml
├── .github/
│   └── workflows/
│       └── ci-cd.yml
└── README.md

结论

通过上述内容,您可以更好地理解和实践Go语言应用的部署与运维。从打包和发布、容器化部署,到监控和日志管理、安全性、高可用性和容错,再到CI/CD,每个环节都至关重要。

9. 安全性增强

认证和授权
  • OAuth2 和 OpenID Connect:使用OAuth2协议进行认证和授权,OpenID Connect是基于OAuth2的标准化身份层。
    • OAuth2 Provider:可以使用Keycloak、Auth0等开源或商业服务作为OAuth2提供商。
    • 中间件:使用中间件(如oauth2-proxy)保护应用的API端点。
    package mainimport ("net/http""github.com/oauth2-proxy/oauth2-proxy/pkg/apis/options""github.com/oauth2-proxy/oauth2-proxy/pkg/logger""github.com/oauth2-proxy/oauth2-proxy/pkg/oauthproxy"
    )func main() {config := options.Options{ClientID:     "your-client-id",ClientSecret: "your-client-secret",RedirectURL:  "http://localhost:8080/oauth2/callback",Provider:     "google",CookieSecret: "your-cookie-secret",}proxy, err := oauthproxy.NewOAuthProxy(config)if err != nil {logger.Fatalf("Failed to create OAuth proxy: %v", err)}http.Handle("/", proxy)http.HandleFunc("/protected", func(w http.ResponseWriter, r *http.Request) {w.Write([]byte("Protected content"))})http.ListenAndServe(":8080", nil)
    }
    
安全头设置
  • Security Headers:使用中间件设置HTTP安全头,如Content-Security-Policy、X-Frame-Options等。
    package mainimport ("net/http""github.com/unrolled/secure"
    )func main() {secureMiddleware := secure.New(secure.Options{AllowedHosts:          []string{"example.com"},SSLRedirect:           true,STSSeconds:            315360000,STSIncludeSubdomains:  true,STSPreload:            true,FrameDeny:             true,ContentTypeNosniff:    true,BrowserXssFilter:      true,ContentSecurityPolicy: "default-src 'self'",})http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {w.Write([]byte("Hello, world!"))})http.ListenAndServe(":8080", secureMiddleware.Handler(http.DefaultServeMux))
    }
    

10. 性能优化

代码优化
  • 基准测试:使用go test -bench .进行基准测试,找出性能瓶颈。

    package mainimport "testing"func BenchmarkMyFunction(b *testing.B) {for i := 0; i < b.N; i++ {MyFunction()}
    }
    
  • 性能剖析:使用pprof进行性能剖析,分析CPU和内存使用情况。

    package mainimport (_ "net/http/pprof""net/http"
    )func main() {go func() {http.ListenAndServe("localhost:6060", nil)}()// Your application logichttp.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {w.Write([]byte("Hello, world!"))})http.ListenAndServe(":8080", nil)
    }
    
数据库优化
  • 索引优化:合理使用索引,避免全表扫描。
  • 连接池:使用数据库连接池(如sqlx)管理数据库连接,提高性能。
    package mainimport ("database/sql""fmt"_ "github.com/go-sql-driver/mysql"
    )func main() {db, err := sql.Open("mysql", "user:password@tcp(127.0.0.1:3306)/dbname?parseTime=true")if err != nil {panic(err)}defer db.Close()db.SetMaxOpenConns(100)db.SetMaxIdleConns(10)rows, err := db.Query("SELECT * FROM users")if err != nil {panic(err)}defer rows.Close()for rows.Next() {var id intvar name stringif err := rows.Scan(&id, &name); err != nil {panic(err)}fmt.Println(id, name)}
    }
    

11. 微服务架构

服务拆分
  • 领域驱动设计(DDD):根据业务领域将应用拆分为多个微服务。
  • API网关:使用API网关(如Kong、Traefik)统一管理微服务的入口。
    # Kong API Gateway configuration
    services:- name: user-serviceurl: http://user-service:8080routes:- name: user-routepaths: ["/users"]
    
服务发现和注册
  • Consul:使用Consul进行服务发现和注册。
    package mainimport ("context""fmt""time"consul "github.com/hashicorp/consul/api"
    )func main() {config := consul.DefaultConfig()client, err := consul.NewClient(config)if err != nil {panic(err)}service := &consul.AgentServiceRegistration{ID:      "user-service-1",Name:    "user-service",Address: "127.0.0.1",Port:    8080,Check: &consul.AgentServiceCheck{HTTP:                           "http://127.0.0.1:8080/health",Interval:                       10 * time.Second,Timeout:                        5 * time.Second,DeregisterCriticalServiceAfter: 1 * time.Minute,},}if err := client.Agent().ServiceRegister(service); err != nil {panic(err)}defer client.Agent().ServiceDeregister("user-service-1")http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {w.Write([]byte("User service is running"))})http.HandleFunc("/health", func(w http.ResponseWriter, r *http.Request) {w.WriteHeader(http.StatusOK)})http.ListenAndServe(":8080", nil)
    }
    

12. 服务网格

Istio
  • Istio:使用Istio作为服务网格,提供流量管理、安全性和可观测性。
    • 安装Istio

      curl -L https://istio.io/downloadIstio | ISTIO_VERSION=1.10.0 sh -
      cd istio-1.10.0
      export PATH=$PWD/bin:$PATH
      istioctl install --set profile=demo -y
      
    • 部署应用

      apiVersion: apps/v1
      kind: Deployment
      metadata:name: myapplabels:app: myapp
      spec:replicas: 2selector:matchLabels:app: myapptemplate:metadata:labels:app: myappspec:containers:- name: myappimage: your-docker-imageports:- containerPort: 8080
      ---
      apiVersion: v1
      kind: Service
      metadata:name: myapplabels:app: myapp
      spec:ports:- port: 80targetPort: 8080selector:app: myapp
      ---
      apiVersion: networking.istio.io/v1alpha3
      kind: VirtualService
      metadata:name: myapp
      spec:hosts:- "myapp.example.com"http:- route:- destination:host: myappport:number: 80
      

13. 最佳实践

代码质量
  • 代码审查:定期进行代码审查,确保代码质量和一致性。
  • 代码规范:使用golintgo vet等工具检查代码风格和潜在问题。
    golint ./...
    go vet ./...
    
测试
  • 单元测试:编写单元测试,确保每个函数的行为符合预期。

    package mainimport ("testing"
    )func TestMyFunction(t *testing.T) {result := MyFunction()if result != expected {t.Errorf("Expected %v, got %v", expected, result)}
    }
    
  • 集成测试:编写集成测试,确保各个组件协同工作。

    package mainimport ("net/http""net/http/httptest""testing"
    )func TestIntegration(t *testing.T) {ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {w.Write([]byte("Hello, world!"))}))defer ts.Close()resp, err := http.Get(ts.URL)if err != nil {t.Fatal(err)}defer resp.Body.Close()if resp.StatusCode != http.StatusOK {t.Errorf("Expected status %v, got %v", http.StatusOK, resp.StatusCode)}
    }
    

结论

通过上述内容,您可以进一步提升Go语言应用的部署与运维水平。从安全性增强、性能优化,到微服务架构和服务网格,每个环节都有其独特的挑战和解决方案。希望这些内容能帮助您构建更加健壮、高效和可扩展的应用。

相关文章:

第 32 章 - Go语言 部署与运维

在Go语言的应用开发中&#xff0c;部署与运维是一个非常重要的环节。它不仅关系到应用能否顺利上线&#xff0c;还直接影响到应用的性能、安全性和可维护性。以下是根据您的需求整理的关于Go语言应用的打包和发布、容器化部署、监控和日志管理的相关内容。 1. 应用的打包和发布…...

噪杂环境离线语音通断器效果展示

介于之前的离线语音通断器模块的使用环境大部分为噪音比较小的环境中&#xff0c;部分客户对环境提出了更高的要求&#xff0c;能在噪杂、室外或者有一定的噪音的车内使用的模块开发需求被提高到了一个新的层次。最近找到某些能支持室外噪杂环境使用的芯片和模组&#xff0c;打…...

【django】扩展

1. Promise 1.1 对象和状态 是什么&#xff1f;是前端开发时js中的一个对象&#xff08;包裹&#xff09;。【对象】【异步请求】# 对象中有一个状态的值&#xff0c;status # 创建对象&#xff0c;不赋值&#xff0c;statuspendding let v1 new Promise(function(resolve, …...

逆向破解识别基础

找main函数&#xff1a; &#xff08;使用OllyDbg软件&#xff09; 方法一&#xff1a; 因为main函数需要三个参数&#xff0c;所以遇到三个参数加一个调用&#xff0c;那么可能是main函数。 方法二&#xff1a; 如果main函数中有字符串或者调用函数等一些唯一标识&#x…...

MFC 下拉框显示问题和控件自适应窗口大小

在mfc开发过程中&#xff0c;遇到了下拉框的两个问题&#xff0c;现在记录一下 1、把下拉框点开显示不全我放进去的多条文本 解决办法&#xff1a;把鼠标放到下拉框的倒三角去点一下&#xff0c;鼠标就会变成双向箭头&#xff0c;或者先选中一下下拉框再把鼠标移动到倒三角上去…...

Docker 的存储驱动及其优缺点

Docker 的存储驱动是容器文件系统的关键组成部分&#xff0c;决定了容器镜像和数据的存储方式。以下是常见的存储驱动及其优缺点&#xff1a; 1. OverlayFS/Overlay2 简介: Overlay 是现代 Linux 文件系统中一种高效的联合文件系统&#xff0c;Overlay2 是其改进版本&#xff0…...

单片机系统的性能指标有哪些?

单片机系统的性能指标涵盖了多个方面&#xff0c;这些指标共同决定了系统的整体性能。以下是单片机系统中常见的性能指标&#xff1a; 1. 处理器性能指标 1.1 时钟频率&#xff08;Clock Frequency&#xff09; 定义&#xff1a;处理器内核工作的时钟频率&#xff0c;通常以…...

波点音乐自动点击

波点音乐 import uiautomator2 as u2 import time import sys import os# 动态点击时间&#xff0c;打印剩余时间 def dynamic_sleep(seconds):wait_time secondsfor i in range(wait_time):print(f"Waiting... {wait_time - i} seconds remaining")sys.stdout.flu…...

HTTP 消息结构

HTTP 消息结构 1. 引言 超文本传输协议(HTTP)是互联网上应用最广泛的协议之一,它定义了客户端和服务器之间交换数据的格式和规则。HTTP消息是客户端和服务器之间通信的基本单位,包括请求消息和响应消息两种类型。本文将详细介绍HTTP消息的结构,包括其组成部分和格式。 …...

ESP32学习笔记_Bluetooth(1)——蓝牙技术与 BLE 通信机制简介

摘要(From AI): 这篇笔记详细讲解了蓝牙技术的分类、核心架构和通信流程&#xff0c;重点介绍了低功耗蓝牙&#xff08;BLE&#xff09;的协议栈、角色划分及连接机制&#xff0c;结合拓扑示例清晰阐述了蓝牙网络配置和操作过程 前言&#xff1a;本文档是本人在《ESP32-C3 物联…...

C0034.在Ubuntu中安装的Qt路径

Qt安装路径查询 在终端输入qmake -v如上中/usr/lib/x86_64-linux-gnu就是Qt的安装目录&#xff1b;...

jdbc学习——SQL注入

SQL注入介绍 SQL注入是通过操作输入来修改事先定义好的SQL语句&#xff0c;用以达到执行代码对服务器进行攻击的方法。 用户登录的本质&#xff1a;执行下面的sql语句 select * from tb_user where username zhangsan and password 123; 发生sql注入的本质 解决SQL注入的方…...

JavaEE-线程安全专题

文章目录 线程安全概述线程安全引入线程不安全原因概述 线程是随机调度的 线程安全概述 线程安全引入 线程安全问题是整个多线程专题的最核心也是最重要的章节, 如果不理解线程的安全, 是无法写出正确的多线程的代码的, 我们之前所写的代码都是在单一的线程环境之下写出的 “…...

Android 设备使用 Wireshark 工具进行网络抓包

背景 电脑和手机连接同一网络&#xff0c;想使用wireshark抓包工具抓取Android手机网络日志&#xff0c;有以下两种连接方法&#xff1a; Wi-Fi 网络抓包。USB 网络共享抓包。需要USB 数据线将手机连接到电脑&#xff0c;并在开发者模式中启用 USB 网络共享。 查看设备连接信…...

物联网无线局域网WiFi开发(一):WiFi智能家居解决方案

一、WiFi智能家居硬件设计 &#xff08;一&#xff09;WiFi智能家居硬件方案 &#xff08;二&#xff09;硬件选型方案 二、WiFi开发环境搭建 &#xff08;一&#xff09;开发环境搭建 虚拟机lubuntu VirtualBox下载地址&#xff1a;https://www.virtualbox.org/wiki/Downl…...

GMAN解读(论文+代码)

一、注意力机制 注意力机制与传统的卷积神经网络不同的是&#xff0c;前者擅长捕获全局依赖和长程关系&#xff0c;权重会动态调整。而后者对于所有特征都使用同一个卷积核。关于更多注意力机制内容&#xff0c;详见&#xff1a; 注意力机制、自注意力机制、多头注意力机制、通…...

速盾:ddos防御手段哪种比较好?高防cdn怎么样?

DDoS&#xff08;分布式拒绝服务&#xff09;攻击是一种威胁网络安全的常见攻击手段。为了保护网站和服务器免受DDoS攻击的影响&#xff0c;许多安全专家和公司开发了各种防御手段。在这篇文章中&#xff0c;我们将重点讨论一种常见的DDoS防御手段——高防CDN&#xff08;内容分…...

Spring:AOP切入点表达式

对于AOP中切入点表达式&#xff0c;我们总共会学习三个内容&#xff0c;分别是语法格式、通配符和书写技巧。 语法格式 首先我们先要明确两个概念: 切入点:要进行增强的方法切入点表达式:要进行增强的方法的描述方式 对于切入点的描述&#xff0c;我们其实是有两中方式的&a…...

《文件操作》

一 . 文本文件和二进制文件 根据数据的组织形式&#xff0c;数据文件被分为了二进制文件和文本文件 数据在内存中是以二进制的形式存储&#xff0c;如果不加转换的输出到外存的文件中&#xff0c;就是二进制文件。 如果要求在外存上以ASCII 码的形式存储&#xff0c;则需要再存…...

python特殊字符序列

字符 描述 \A 只匹配字符串的开始 \b 匹配一个单词边界 \B 匹配一个单词的非边界 \d 匹配任意十进制数字字符&#xff0c;等价于r&#xff3b;0-9] \D 匹配任意非十进制数字字符&#xff0c;等价于r[^0-9]’ \s 匹配任意空格字符&#xff08;空格符、tab制表符、换…...

卷积神经网络(CNN)中的批量归一化层(Batch Normalization Layer)

批量归一化层&#xff08;BatchNorm层&#xff09;&#xff0c;或简称为批量归一化&#xff08;Batch Normalization&#xff09;&#xff0c;是深度学习中常用的一种技术&#xff0c;旨在加速神经网络的训练并提高收敛速度。 一、基本思想 为了让数据在训练过程中保持同一分布…...

LLaMA-Mesh: Unifying 3D Mesh Generation with Language Models 论文解读

目录 一、概述 二、相关工作 1、LLMs到多模态 2、3D对象生成 3、自回归的Mesh生成 三、LLaMA-Mesh 1、3D表示 2、预训练模型 3、有监督的微调数据集 4、数据集演示 四、实验 1、生成的多样性 2、不同模型text-to-Mesh的比较 3、通用语境的评估 一、概述 该论文首…...

【ESP32CAM+Android+C#上位机】ESP32-CAM在STA或AP模式下基于UDP与手机APP或C#上位机进行视频流/图像传输

前言: 本项目实现ESP32-CAM在STA或AP模式下基于UDP与手机APP或C#上位机进行视频流/图像传输。本项目包含有ESP32源码(arduino)、Android手机APP源码以及C#上位机源码,本文对其工程项目的配置使用进行讲解。实战开发,亲测无误。 AP模式,就是ESP32发出一个WIFI/热点提供给电…...

ESP-KeyBoard:基于 ESP32-S3 的三模客制化机械键盘

概述 在这个充满挑战与机遇的数字化时代&#xff0c;键盘已经成为我们日常学习、工作、娱乐生活必不可少的设备。而在众多键盘中&#xff0c;机械键盘&#xff0c;以其独特的触感、清脆的敲击音和经久耐用的特性&#xff0c;已经成为众多游戏玩家和电子工程师的首选。本文将为…...

28.UE5游戏框架,事件分发器,蓝图接口

3-3 虚幻游戏框架拆解&#xff0c;游戏规则基础_哔哩哔哩_bilibili 目录 1.游戏架构 2.事件分发器 2.1UI控件中的事件分发器 2.2Actor蓝图中的事件分发器 2.2.1动态决定Actor的分发事件 2.2.2父类中定义事件分发器&#xff0c;子类实现事件分发器 2.3组件蓝图中实现事件…...

Puppeteer 和 Cheerio 在 Node.js 中的应用

Puppeteer 和 Cheerio 在 Node.js 中的应用 引言 在现代 Web 开发中&#xff0c;自动化测试、数据抓取和页面分析是常见的需求。Node.js 提供了丰富的工具和库来满足这些需求。本文将介绍两个在 Node.js 中常用的库&#xff1a;Puppeteer 和 Cheerio&#xff0c;它们分别用于…...

Unity2D 关于N方向俯视角 中 角色移动朝向的问题

通常对俯视角2d游戏的角色移动我们使用简单2d混合树的方式,但是其不移动时的朝向该如何定义&#xff1f; 十分简单&#xff1a;移动和不移动之间形成逻辑自锁 详细说明思路就是再创建一个简单2d混合树 定义其N方向的idle 并用lastDirc二维向量保存玩家输入&#xff0c;当玩家输…...

pytorch 和tensorflow loss.item()` 只能用于只有一个元素的张量. 防止显存爆炸

loss.item() 是 PyTorch 中的一个方法&#xff0c;它用于从一个只包含单个元素的张量&#xff08;tensor&#xff09;中提取出该元素的值&#xff0c;并将其转换为一个 Python 标量&#xff08;即 int 或 float 类型&#xff09;。这个方法在训练神经网络时经常用到&#xff0c…...

链表刷题|判断回文结构

题目来自于牛客网&#xff0c;本文章仅记录学习过程的做题理解&#xff0c;便于梳理思路和复习 我做题喜欢先把时间复杂度和空间复杂度放一边&#xff0c;先得有大概的解决方案&#xff0c;最后如果时间或者空间超了再去优化即可。 思路一&#xff1a;要判断是否为回文结构则…...

海盗王集成网关和商城服务端功能golang版

之前用golang把海盗王的商城服务端和网关服务端都重写了一次。 后来在同时开启网关和商城服务时&#xff0c;发现窗口数量有点多&#xff0c;有时要找到商城窗口比较麻烦。 既然2个都是用golang govcl写的&#xff0c;是不是可以集成到一起&#xff0c;方便使用呢&#xff1f;…...