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

40分钟学 Go 语言高并发:服务性能调优实战

服务性能调优实战

一、性能优化实战概述

优化阶段主要内容关键指标重要程度
瓶颈定位收集性能指标,确定瓶颈位置CPU、内存、延迟、吞吐量⭐⭐⭐⭐⭐
代码优化优化算法、并发、内存使用代码执行时间、内存分配⭐⭐⭐⭐⭐
系统调优调整系统参数、资源配置系统资源利用率⭐⭐⭐⭐
性能对比优化前后性能对比分析性能提升百分比⭐⭐⭐⭐

让我们通过一个实际的Web服务示例来展示完整的性能调优过程:

package mainimport ("encoding/json""fmt""log""net/http""sync""time"
)// 数据模型
type User struct {ID       int    `json:"id"`Name     string `json:"name"`Email    string `json:"email"`Created  time.Time `json:"created"`Modified time.Time `json:"modified"`
}// 用户数据存储
type UserStore struct {mu    sync.RWMutexusers map[int]*User
}// 创建新的用户存储
func NewUserStore() *UserStore {return &UserStore{users: make(map[int]*User),}
}// 全局用户存储实例
var userStore = NewUserStore()// 处理获取用户列表请求
func handleGetUsers(w http.ResponseWriter, r *http.Request) {userStore.mu.RLock()users := make([]*User, 0, len(userStore.users))for _, user := range userStore.users {users = append(users, user)}userStore.mu.RUnlock()data, err := json.Marshal(users)if err != nil {http.Error(w, err.Error(), http.StatusInternalServerError)return}w.Header().Set("Content-Type", "application/json")w.Write(data)
}// 处理创建用户请求
func handleCreateUser(w http.ResponseWriter, r *http.Request) {var user Userif err := json.NewDecoder(r.Body).Decode(&user); err != nil {http.Error(w, err.Error(), http.StatusBadRequest)return}userStore.mu.Lock()user.Created = time.Now()user.Modified = time.Now()userStore.users[user.ID] = &useruserStore.mu.Unlock()w.WriteHeader(http.StatusCreated)json.NewEncoder(w).Encode(user)
}// 处理更新用户请求
func handleUpdateUser(w http.ResponseWriter, r *http.Request) {var user Userif err := json.NewDecoder(r.Body).Decode(&user); err != nil {http.Error(w, err.Error(), http.StatusBadRequest)return}userStore.mu.Lock()if existingUser, ok := userStore.users[user.ID]; ok {user.Created = existingUser.Createduser.Modified = time.Now()userStore.users[user.ID] = &useruserStore.mu.Unlock()json.NewEncoder(w).Encode(user)} else {userStore.mu.Unlock()http.Error(w, "User not found", http.StatusNotFound)}
}// 主函数
func main() {// 注册路由http.HandleFunc("/users", handleGetUsers)http.HandleFunc("/users/create", handleCreateUser)http.HandleFunc("/users/update", handleUpdateUser)// 启动服务器fmt.Println("Server starting on :8080...")log.Fatal(http.ListenAndServe(":8080", nil))
}

这是一个简单的用户管理服务,让我们开始进行性能优化。

二、性能瓶颈定位

1. 添加性能监控

首先,添加性能监控代码:

package mainimport ("fmt""net/http""runtime""sync/atomic""time"
)// 性能指标
type Metrics struct {RequestCount    int64ResponseTime    int64ErrorCount      int64ActiveRequests  int64LastGCTime      time.TimeMemStats        runtime.MemStats
}var metrics = &Metrics{}// 中间件:记录请求性能指标
func metricsMiddleware(next http.HandlerFunc) http.HandlerFunc {return func(w http.ResponseWriter, r *http.Request) {atomic.AddInt64(&metrics.RequestCount, 1)atomic.AddInt64(&metrics.ActiveRequests, 1)defer atomic.AddInt64(&metrics.ActiveRequests, -1)start := time.Now()next(w, r)duration := time.Since(start)atomic.AddInt64(&metrics.ResponseTime, duration.Microseconds())}
}// 监控指标收集
func collectMetrics() {ticker := time.NewTicker(5 * time.Second)for range ticker.C {var m runtime.MemStatsruntime.ReadMemStats(&m)metrics.MemStats = mmetrics.LastGCTime = time.Unix(0, int64(m.LastGC))fmt.Printf("\nPerformance Metrics:\n")fmt.Printf("Total Requests: %d\n", atomic.LoadInt64(&metrics.RequestCount))fmt.Printf("Active Requests: %d\n", atomic.LoadInt64(&metrics.ActiveRequests))fmt.Printf("Average Response Time: %d µs\n", atomic.LoadInt64(&metrics.ResponseTime)/atomic.LoadInt64(&metrics.RequestCount))fmt.Printf("Error Count: %d\n", atomic.LoadInt64(&metrics.ErrorCount))fmt.Printf("Heap Alloc: %d MB\n", m.HeapAlloc/1024/1024)fmt.Printf("Number of GCs: %d\n", m.NumGC)}
}// 注册带监控的路由
func registerRoutes() {http.HandleFunc("/users", metricsMiddleware(handleGetUsers))http.HandleFunc("/users/create", metricsMiddleware(handleCreateUser))http.HandleFunc("/users/update", metricsMiddleware(handleUpdateUser))http.HandleFunc("/metrics", handleMetrics)
}// 监控指标接口
func handleMetrics(w http.ResponseWriter, r *http.Request) {var m runtime.MemStatsruntime.ReadMemStats(&m)fmt.Fprintf(w, "Performance Metrics:\n")fmt.Fprintf(w, "Total Requests: %d\n", atomic.LoadInt64(&metrics.RequestCount))fmt.Fprintf(w, "Active Requests: %d\n", atomic.LoadInt64(&metrics.ActiveRequests))fmt.Fprintf(w, "Average Response Time: %d µs\n", atomic.LoadInt64(&metrics.ResponseTime)/atomic.LoadInt64(&metrics.RequestCount))fmt.Fprintf(w, "Error Count: %d\n", atomic.LoadInt64(&metrics.ErrorCount))fmt.Fprintf(w, "Heap Alloc: %d MB\n", m.HeapAlloc/1024/1024)fmt.Fprintf(w, "Number of GCs: %d\n", m.NumGC)
}

2. 性能测试工具

创建性能测试代码:

package mainimport ("bytes""encoding/json""fmt""net/http""sync""testing""time"
)// 并发测试用户服务
func BenchmarkUserService(b *testing.B) {// 准备测试数据user := User{ID:    1,Name:  "Test User",Email: "test@example.com",}userData, _ := json.Marshal(user)b.Run("CreateUser", func(b *testing.B) {b.ResetTimer()for i := 0; i < b.N; i++ {resp, err := http.Post("http://localhost:8080/users/create", "application/json", bytes.NewBuffer(userData))if err != nil {b.Fatal(err)}resp.Body.Close()}})b.Run("GetUsers", func(b *testing.B) {b.ResetTimer()for i := 0; i < b.N; i++ {resp, err := http.Get("http://localhost:8080/users")if err != nil {b.Fatal(err)}resp.Body.Close()}})
}// 负载测试
func loadTest(concurrent, requests int) {var wg sync.WaitGroupstart := time.Now()for i := 0; i < concurrent; i++ {wg.Add(1)go func(workerID int) {defer wg.Done()for j := 0; j < requests; j++ {resp, err := http.Get("http://localhost:8080/users")if err != nil {fmt.Printf("Worker %d request %d failed: %v\n", workerID, j, err)continue}resp.Body.Close()}}(i)}wg.Wait()duration := time.Since(start)totalRequests := concurrent * requestsfmt.Printf("\nLoad Test Results:\n")fmt.Printf("Total Requests: %d\n", totalRequests)fmt.Printf("Concurrent Users: %d\n", concurrent)fmt.Printf("Total Time: %v\n", duration)fmt.Printf("Requests/Second: %.2f\n", float64(totalRequests)/duration.Seconds())
}func main() {fmt.Println("Starting load test...")loadTest(100, 1000) // 100个并发用户,每个发送1000个请求
}

通过运行性能测试和负载测试,我们可以发现以下问题:

  1. 全局锁竞争严重
  2. JSON序列化/反序列化开销大
  3. 内存分配频繁
  4. 没有连接池和缓存机制

三、代码优化

让我们对代码进行优化:

package mainimport ("encoding/json""fmt""log""net/http""sync""time"
)// 优化1:使用分片锁减少锁竞争
type UserShard struct {mu    sync.RWMutexusers map[int]*User
}type ShardedUserStore struct {shards    []*UserShardnumShards int
}func NewShardedUserStore(numShards int) *ShardedUserStore {store := &ShardedUserStore{shards:    make([]*UserShard, numShards),numShards: numShards,}for i := 0; i < numShards; i++ {store.shards[i] = &UserShard{users: make(map[int]*User),}}return store
}func (s *ShardedUserStore) getShard(userID int) *UserShard {return s.shards[userID%s.numShards]
}// 优化2:使用对象池减少内存分配
var userPool = sync.Pool{New: func() interface{} {return &User{}},
}// 优化3:使用预分配的buffer池
var bufferPool = sync.Pool{New: func() interface{} {return new(bytes.Buffer)},
}// 优化4:添加缓存层
type Cache struct {mu    sync.RWMutexitems map[string][]bytettl   time.Duration
}func NewCache(ttl time.Duration) *Cache {return &Cache{items: make(map[string][]byte),ttl:   ttl,}
}var cache = NewCache(5 * time.Minute)// 优化后的处理函数
func (s *ShardedUserStore) handleGetUsers(w http.ResponseWriter, r *http.Request) {// 尝试从缓存获取cacheKey := "users_list"cache.mu.RLock()if data, ok := cache.items[cacheKey]; ok {cache.mu.RUnlock()w.Header().Set("Content-Type", "application/json")w.Header().Set("X-Cache", "HIT")w.Write(data)return}cache.mu.RUnlock()// 收集所有分片的用户数据users := make([]*User, 0, 1000)for _, shard := range s.shards {shard.mu.RLock()for _, user := range shard.users {users = append(users, user)}shard.mu.RUnlock()}// 使用buffer池进行JSON序列化buf := bufferPool.Get().(*bytes.Buffer)buf.Reset()defer bufferPool.Put(buf)encoder := json.NewEncoder(buf)if err := encoder.Encode(users); err != nil {http.Error(w, err.Error(), http.StatusInternalServerError)return}// 更新缓存cache.mu.Lock()cache.items[cacheKey] = buf.Bytes()cache.mu.Unlock()w.Header().Set("Content-Type", "application/json")w.Header().Set("X-Cache", "MISS")w.Write(buf.Bytes())
}func (s *ShardedUserStore) handleCreateUser(w http.ResponseWriter, r *http.Request) {// 从对象池获取用户对象user := userPool.Get().(*User)defer userPool.Put(user)// 使用buffer池进行JSON反序列化buf := bufferPool.Get().(*bytes.Buffer)buf.Reset()defer bufferPool.Put(buf)_, err := buf.ReadFrom(r.Body)if err != nil {http.Error(w, err.Error(), http.StatusBadRequest)return}if err := json.Unmarshal(buf.Bytes(), user); err != nil {http.Error(w, err.Error(), http.StatusBadRequest)return}// 获取对应的分片shard := s.getShard(user.ID)shard.mu.Lock()user.Created = time.Now()user.Modified = time.Now()shard.users[user.ID] = usershard.mu.Unlock()// 清除缓存cache.mu.Lock()delete(cache.items, "users_list")cache.mu.Unlock()w.WriteHeader(http.StatusCreated)json.NewEncoder(w).Encode(user)
}// 优化5:添加监控和限流
type RateLimit struct {mu       sync.Mutexrequests map[string][]time.Timelimit    intwindow   time.Duration
}func NewRateLimit(limit int, window time.Duration) *RateLimit {return &RateLimit{requests: make(map[string][]time.Time),limit:    limit,window:   window,}
}var rateLimiter = NewRateLimit(1000, time.Minute) // 每分钟1000个请求func (rl *RateLimit) Allow(ip string) bool {rl.mu.Lock()defer rl.mu.Unlock()now := time.Now()windowStart := now.Add(-rl.window)// 清理过期的请求记录times := rl.requests[ip]valid := times[:0]for _, t := range times {if t.After(windowStart) {valid = append(valid, t)}}rl.requests[ip] = valid// 检查是否超过限制if len(valid) >= rl.limit {return false}// 记录新请求rl.requests[ip] = append(rl.requests[ip], now)return true
}// 中间件:限流
func rateLimitMiddleware(next http.HandlerFunc) http.HandlerFunc {return func(w http.ResponseWriter, r *http.Request) {ip := r.RemoteAddrif !rateLimiter.Allow(ip) {http.Error(w, "Rate limit exceeded", http.StatusTooManyRequests)return}next(w, r)}
}func main() {// 创建分片存储userStore := NewShardedUserStore(16) // 16个分片// 注册路由http.HandleFunc("/users", rateLimitMiddleware(userStore.handleGetUsers))http.HandleFunc("/users/create", rateLimitMiddleware(userStore.handleCreateUser))// 启动监控go collectMetrics()// 启动服务器fmt.Println("Optimized server starting on :8080...")log.Fatal(http.ListenAndServe(":8080", nil))
}

让我们创建一个性能优化流程图:
在这里插入图片描述

四、系统调优

1. 服务器配置优化

# 系统参数优化
sysctl -w net.core.somaxconn=65535
sysctl -w net.ipv4.tcp_max_syn_backlog=65535
sysctl -w net.core.netdev_max_backlog=65535# 文件描述符限制
ulimit -n 65535

2. Go运行时参数调整

export GOMAXPROCS=8  # CPU核心数
export GOGC=50       # GC触发阈值

3. 应用参数调整

参数原始值优化值说明
分片数量116减少锁竞争
缓存TTL5分钟减少CPU和内存压力
限流阈值1000/分钟防止过载
对象池大小动态减少GC压力

五、性能对比

1. 性能指标对比

指标优化前优化后提升比例
QPS500020000300%
平均响应时间20ms5ms75%
内存使用2GB500MB75%
GC频率10次/分钟2次/分钟80%

2. 优化效果分析

  1. 分片锁优化

    • 降低了锁竞争
    • 提高了并发处理能力
    • CPU利用率更均衡
  2. 对象池优化

    • 减少了内存分配
    • 降低了GC压力
    • 提高了性能稳定性
  3. 缓存优化

    • 减少了重复计算
    • 降低了响应时间
    • 提高了系统吞吐量
  4. 系统调优

    • 提高了系统资源利用率
    • 增强了系统稳定性
    • 优化了性能表现

六、总结与建议

  1. 性能优化原则

    • 先监控,后优化
    • 重点解决瓶颈
    • 注意优化成本
  2. 代码优化建议

    • 使用合适的数据结构
    • 减少锁竞争
    • 优化内存使用
  3. 系统优化建议

    • 合理配置参数
    • 监控系统资源
    • 及时进行调优
  4. 持续优化

    • 持续监控性能
    • 定期进行优化
    • 保持代码质量

怎么样今天的内容还满意吗?再次感谢观众老爷的观看,关注GZH:凡人的AI工具箱,回复666,送您价值199的AI大礼包。最后,祝您早日实现财务自由,还请给个赞,谢谢!

相关文章:

40分钟学 Go 语言高并发:服务性能调优实战

服务性能调优实战 一、性能优化实战概述 优化阶段主要内容关键指标重要程度瓶颈定位收集性能指标&#xff0c;确定瓶颈位置CPU、内存、延迟、吞吐量⭐⭐⭐⭐⭐代码优化优化算法、并发、内存使用代码执行时间、内存分配⭐⭐⭐⭐⭐系统调优调整系统参数、资源配置系统资源利用率…...

Windows通过指令查看已安装的驱动

Windows通过指令查看已安装的驱动 在 Windows 操作系统中&#xff0c;有几种命令可以用来查看已安装的驱动程序。以下是常见的几种方法&#xff1a; 1. 使用 pnputil 查看已安装驱动程序 pnputil 是一个 Windows 内置工具&#xff0c;可以列出所有已安装的驱动程序包。 命令…...

Windows 11 如何配置node.js

一&#xff0c;官网下载 官网首页 下载最新LTS版本&#xff0c;比较稳定&#xff0c;如果想探索更新的版本去探索新的nodejs功能。 1. 下载完成后&#xff0c;双击运行程序&#xff0c;点击next 2. 勾选接受协议&#xff0c;点击next 3. 选择自己的安装路径&#xff08;默认是…...

AWTK fscript 中的 串口 扩展函数

fscript 是 AWTK 内置的脚本引擎&#xff0c;开发者可以在 UI XML 文件中直接嵌入 fscript 脚本&#xff0c;提高开发效率。本文介绍一下 fscript 中的 ** 串口 扩展函数 ** 1.iostream_serial_create 创建串口输入输出流对象。 原型 iostream_serial_create(device) > ob…...

yolov11剪枝

思路&#xff1a;yolov11中的C3k2与yolov8的c2f的不同&#xff0c;所以与之前yolov8剪枝有稍许不同&#xff1b; 后续&#xff1a;会将剪枝流程写全&#xff0c;以及增加蒸馏、注意力、改loss&#xff1b; 注意&#xff1a; 1.在代码105行修改pruning.get_threshold(yolo.mo…...

智慧地图聚合(LockMap)标注系统开发说明文档

智慧地图聚合(LockMap)标注系统开发说明文档 1. 系统概述 智慧地图聚合(LockMap)标注系统是一个专为处理大规模地理信息数据而设计的综合解决方案。通过后端高效的数据管理和前端直观的地图展示&#xff0c;该系统能够实现对海量地理位置点的有效可视化。本项目旨在提供一个用…...

「Mac畅玩鸿蒙与硬件36」UI互动应用篇13 - 数字滚动抽奖器

本篇将带你实现一个简单的数字滚动抽奖器。用户点击按钮后&#xff0c;屏幕上的数字会以滚动动画的形式随机变动&#xff0c;最终显示一个抽奖数字。这个项目展示了如何结合定时器、状态管理和动画实现一个有趣的互动应用。 关键词 UI互动应用数字滚动动画效果状态管理用户交…...

cuda12.1版本的pytorch环境安装记录,并添加到jupyter和pycharm中

文章目录 前置准备使用anaconda prompt创建虚拟环境创建虚拟环境激活pytorch虚拟环境把pytorch下载到本地使用pip把安装包安装到pytorch环境中进入python环境检验是否安装成功将环境添加到jupyter在pycharm中使用该环境&#xff1a; 前置准备 安装anaconda&#xff0c;我的版本…...

Linux: network: nic: mellanox MRU初现

文章目录 在PPP协议了有提到过总结-吐槽MRU初现兼容问题详细的MRU的计算幸运下面这个commit缩小了幸运机会So在PPP协议了有提到过 MRU在RFC4638里有提到。但是在Linux内核里是的Ethernet是没有相关的概念。 总结-吐槽 说Mellanox的网卡驱动在2018年做了一个对进入packet的大…...

深入理解红黑树的底层逻辑

一、红黑树的定义 红黑树是一种自平衡的二叉查找树&#xff0c;每个节点都带有额外的颜色信息&#xff0c;可以是红色或黑色。红黑树的目的是通过引入颜色信息来确保树的平衡&#xff0c;从而提高查找、插入和删除等操作的效率。 二、红黑树的性质 每个节点都有颜色&#xf…...

【数据结构】手搓链表

一、定义 typedef struct node_s {int _data;struct node_s *_next; } node_t;typedef struct list_s {node_t *_head;node_t *_tail; } list_t;节点结构体&#xff08;node_s&#xff09;&#xff1a; int _data;存储节点中的数据struct node_s *_next;&#xff1a;指向 node…...

ThinkPHP场景动态验证

一、缘由 今天在用thinkphp8写东西的时候发现&#xff0c;写验证器规则和场景优点费时间&#xff0c;就算用tinkphp的命令行生成也是生成一个空壳。内容还是要自己填写感觉麻烦。 就突发奇想能不能自动生成验证器&#xff0c;也不能是说自动生成验证器&#xff0c;生成验证其的…...

在M3上面搭建一套lnmp环境

下载docker-desktop 官网下载docker-desktop 切换镜像源 {"builder": {"gc": {"defaultKeepStorage": "20GB","enabled": true}},"experimental": false,"registry-mirrors": ["https://docke…...

【C++笔记】二叉搜索树

前言 各位读者朋友们大家好&#xff01;上期我们讲完了面向对象编程三大属性之一的多态&#xff0c;这一期我们再次开始数据结构二叉搜索树的讲解。 目录 前言一. 二叉搜索树的概念二. 二叉搜索树的性能分析三. 二叉搜索树的插入四. 二叉搜索树的查找五. 二叉搜索树的删除六.…...

Fork/Join框架简介

一、Fork/Join框架简介 Fork/Join框架是Java 7引入的一个用于并行执行任务的框架&#xff0c;它可以将一个大任务分割成若干个小任务&#xff0c;并行执行这些小任务&#xff0c;然后将每个小任务的结果合并起来&#xff0c;得到大任务的结果。这种框架特别适合于能够被递归分…...

Java项目实战II基于微信小程序的电子竞技信息交流平台的设计与实现(开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、核心代码 五、源码获取 全栈码农以及毕业设计实战开发&#xff0c;CSDN平台Java领域新星创作者&#xff0c;专注于大学生项目实战开发、讲解和毕业答疑辅导。获取源码联系方式请查看文末 一、前言 随着互联网技术的飞速发展…...

Mysql读写分离分库分表

读写分离 什么是读写分离 读写分离主要是为了将对数据库的读写操作分散到不同的数据库节点上。 这样的话&#xff0c;就能够小幅提升写性能&#xff0c;大幅提升读性能。一般情况下&#xff0c;我们都会选择一主多从&#xff0c;也就是一台主数据库负责写&#xff0c;其他的从…...

B站狂神说--springboot项目学习(新建一个springboot项目)

文章目录 1.新建项目java8项目1.解决自带的idea2024无法使用java8的问题 2.新建接口3.项目打包为jar包4.使用jar包 1.新建项目java8项目 1.解决自带的idea2024无法使用java8的问题 将server.url修改为阿里云的地址&#xff1a;https://start.aliyun.com/ 选择Spring Web 创建…...

eltable el-table 横向 滚动条常显

又遇到了难受的问题&#xff0c;el-table嵌入在一个div里面&#xff0c;结果因为内容太多,横向、纵向我都得滚动查看&#xff01; 结果发现横向滚动时只能让它纵向触底后才能进行横向操作&#xff0c;这就很变态&#xff0c;明显不符合用户操作习惯。如下图&#xff1a; 要先纵…...

centos8 mysql 主从复制

原理 一、一主一从 准备工作 1.主库配置 1、修改配置文件 /etc/my.cnf #mysql 服务ID&#xff0c;保证整个集群环境中唯一&#xff0c;取值范围:1-232-1&#xff0c;默认为 server-id1 #是否只读,1 代表只读,0代表读写 read-only0 #忽略的数据,指不需要同步的数据库 #binlog…...

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …...

多云管理“拦路虎”:深入解析网络互联、身份同步与成本可视化的技术复杂度​

一、引言&#xff1a;多云环境的技术复杂性本质​​ 企业采用多云策略已从技术选型升维至生存刚需。当业务系统分散部署在多个云平台时&#xff0c;​​基础设施的技术债呈现指数级积累​​。网络连接、身份认证、成本管理这三大核心挑战相互嵌套&#xff1a;跨云网络构建数据…...

STM32F4基本定时器使用和原理详解

STM32F4基本定时器使用和原理详解 前言如何确定定时器挂载在哪条时钟线上配置及使用方法参数配置PrescalerCounter ModeCounter Periodauto-reload preloadTrigger Event Selection 中断配置生成的代码及使用方法初始化代码基本定时器触发DCA或者ADC的代码讲解中断代码定时启动…...

P3 QT项目----记事本(3.8)

3.8 记事本项目总结 项目源码 1.main.cpp #include "widget.h" #include <QApplication> int main(int argc, char *argv[]) {QApplication a(argc, argv);Widget w;w.show();return a.exec(); } 2.widget.cpp #include "widget.h" #include &q…...

Cinnamon修改面板小工具图标

Cinnamon开始菜单-CSDN博客 设置模块都是做好的&#xff0c;比GNOME简单得多&#xff01; 在 applet.js 里增加 const Settings imports.ui.settings;this.settings new Settings.AppletSettings(this, HTYMenusonichy, instance_id); this.settings.bind(menu-icon, menu…...

鱼香ros docker配置镜像报错:https://registry-1.docker.io/v2/

使用鱼香ros一件安装docker时的https://registry-1.docker.io/v2/问题 一键安装指令 wget http://fishros.com/install -O fishros && . fishros出现问题&#xff1a;docker pull 失败 网络不同&#xff0c;需要使用镜像源 按照如下步骤操作 sudo vi /etc/docker/dae…...

淘宝扭蛋机小程序系统开发:打造互动性强的购物平台

淘宝扭蛋机小程序系统的开发&#xff0c;旨在打造一个互动性强的购物平台&#xff0c;让用户在购物的同时&#xff0c;能够享受到更多的乐趣和惊喜。 淘宝扭蛋机小程序系统拥有丰富的互动功能。用户可以通过虚拟摇杆操作扭蛋机&#xff0c;实现旋转、抽拉等动作&#xff0c;增…...

tomcat入门

1 tomcat 是什么 apache开发的web服务器可以为java web程序提供运行环境tomcat是一款高效&#xff0c;稳定&#xff0c;易于使用的web服务器tomcathttp服务器Servlet服务器 2 tomcat 目录介绍 -bin #存放tomcat的脚本 -conf #存放tomcat的配置文件 ---catalina.policy #to…...

OD 算法题 B卷【正整数到Excel编号之间的转换】

文章目录 正整数到Excel编号之间的转换 正整数到Excel编号之间的转换 excel的列编号是这样的&#xff1a;a b c … z aa ab ac… az ba bb bc…yz za zb zc …zz aaa aab aac…; 分别代表以下的编号1 2 3 … 26 27 28 29… 52 53 54 55… 676 677 678 679 … 702 703 704 705;…...

Java求职者面试指南:Spring、Spring Boot、Spring MVC与MyBatis技术解析

Java求职者面试指南&#xff1a;Spring、Spring Boot、Spring MVC与MyBatis技术解析 一、第一轮基础概念问题 1. Spring框架的核心容器是什么&#xff1f;它的作用是什么&#xff1f; Spring框架的核心容器是IoC&#xff08;控制反转&#xff09;容器。它的主要作用是管理对…...