面试手撕题积累

1、实现滑动窗口限流,允许每分钟最多有100个请求

阿里一面题。

核心:
时间窗口管理:滑动窗口会根据时间流逝不断更新,需要记录请求的时间戳,并根据当前时间计算窗口内的请求数量。
限流判断:每次请求到来时,判断当前时间窗口内的请求数是否超过限制。如果没有超过限制,允许请求;如果超过限制,拒绝请求。

优化:
队列优化:如果每次都遍历 timestamps 切片来清理过期请求,当请求量大时,性能可能受到影响。为了提升性能,可以考虑使用双端队列(deque)来高效地删除最旧的请求。
并发控制:这个实现使用了 sync.Mutex 来保护共享资源的并发访问。在高并发环境下,如果请求量非常大,可以考虑其他更高效的并发控制机制,例如 sync.RWMutex 或无锁队列。

package mainimport ("fmt""sync""time"
)type SlidingWindowLimiter struct {// 请求时间戳队列,用来记录每个请求的时间timestamps []time.Time// 限制的最大请求数maxRequests int// 限制的时间窗口,单位为秒windowDuration time.Duration// 锁,保证并发安全mu sync.Mutex
}// 创建一个新的滑动窗口限流器
func NewSlidingWindowLimiter(maxRequests int, windowDuration time.Duration) *SlidingWindowLimiter {return &SlidingWindowLimiter{timestamps:    []time.Time{},maxRequests:   maxRequests,windowDuration: windowDuration,}
}// 检查是否允许请求
func (limiter *SlidingWindowLimiter) AllowRequest() bool {limiter.mu.Lock()defer limiter.mu.Unlock()// 当前时间now := time.Now()// 移除窗口外的请求时间戳limiter.cleanUp(now)// 如果当前请求时间戳个数小于限制,则允许请求if len(limiter.timestamps) < limiter.maxRequests {// 将当前请求的时间戳加入队列limiter.timestamps = append(limiter.timestamps, now)return true}// 否则,拒绝请求return false
}// 清理过期的请求时间戳
func (limiter *SlidingWindowLimiter) cleanUp(now time.Time) {// 移除超出时间窗口的请求windowStart := now.Add(-limiter.windowDuration)for len(limiter.timestamps) > 0 && limiter.timestamps[0].Before(windowStart) {limiter.timestamps = limiter.timestamps[1:]}
}func main() {// 创建一个滑动窗口限流器,限制每分钟最多100个请求limiter := NewSlidingWindowLimiter(100, time.Minute)// 模拟请求for i := 0; i < 120; i++ {allowed := limiter.AllowRequest()if allowed {fmt.Printf("Request %d allowed\n", i+1)} else {fmt.Printf("Request %d denied (too many requests)\n", i+1)}// 模拟请求间隔,假设每500毫秒发起一个请求time.Sleep(500 * time.Millisecond)}
}

2、把三千一十万五千零一十转换成对应数字

package mainimport ("fmt"
)func chineseToNumber(text string) int {numbersMap := map[rune]int{'零': 0, '一': 1, '二': 2, '三': 3, '四': 4, '五': 5, '六': 6, '七': 7, '八': 8, '九': 9,'十': 10, '百': 100, '千': 1000, '万': 10000,}total := 0num := 0for _, char := range text {if val, ok := numbersMap[char]; ok {if val < 10 {num += val} else {if num == 0 {num = 1}num *= val}} else {total += numnum = 0}}total += numreturn total
}func main() {text := "三千一十万五千零一十"number := chineseToNumber(text)fmt.Println(number)
}

3、用两个队列去模拟栈

package mainimport "fmt"type Stack struct {mainQueue    []intauxQueue     []int
}func NewStack() *Stack {return &Stack{mainQueue: make([]int, 0),auxQueue:  make([]int, 0),}
}func (s *Stack) Push(val int) {// 将元素推入主队列s.mainQueue = append(s.mainQueue, val)
}func (s *Stack) Pop() int {if len(s.mainQueue) == 0 {return -1}// 将主队列中除了最后一个元素外的其他元素依次出队并放入辅助队列for len(s.mainQueue) > 1 {val := s.mainQueue[0]s.mainQueue = s.mainQueue[1:]s.auxQueue = append(s.auxQueue, val)}// 弹出最后一个元素作为栈顶元素popVal := s.mainQueue[0]s.mainQueue = s.mainQueue[:0]// 交换主队列和辅助队列,以便下一次操作s.mainQueue, s.auxQueue = s.auxQueue, s.mainQueuereturn popVal
}func main() {stack := NewStack()stack.Push(1)stack.Push(2)stack.Push(3)fmt.Println(stack.Pop()) // Output: 3fmt.Println(stack.Pop()) // Output: 2
}

4、用两个栈模拟队列

package mainimport "fmt"type Queue struct {stack1 []intstack2 []int
}func NewQueue() *Queue {return &Queue{stack1: make([]int, 0),stack2: make([]int, 0),}
}func (q *Queue) Enqueue(val int) {// 将元素推入stack1q.stack1 = append(q.stack1, val)
}func (q *Queue) Dequeue() int {if len(q.stack2) == 0 {// 如果stack2为空,将stack1中的元素依次弹出并推入stack2for len(q.stack1) > 0 {val := q.stack1[len(q.stack1)-1]q.stack1 = q.stack1[:len(q.stack1)-1]q.stack2 = append(q.stack2, val)}}if len(q.stack2) == 0 {return -1 // 队列为空}// 弹出stack2的栈顶元素作为出队元素popVal := q.stack2[len(q.stack2)-1]q.stack2 = q.stack2[:len(q.stack2)-1]return popVal
}func main() {queue := NewQueue()queue.Enqueue(1)queue.Enqueue(2)queue.Enqueue(3)fmt.Println(queue.Dequeue()) // Output: 1fmt.Println(queue.Dequeue()) // Output: 2fmt.Println(queue.Dequeue()) // Output: 3
}

5、判断B树是否为A树的子结构

package mainimport "fmt"type TreeNode struct {Val   intLeft  *TreeNodeRight *TreeNode
}func isSubStructure(A *TreeNode, B *TreeNode) bool {if A == nil || B == nil {return false}return isSubTree(A, B) || isSubStructure(A.Left, B) || isSubStructure(A.Right, B)
}func isSubTree(A *TreeNode, B *TreeNode) bool {if B == nil {return true}if A == nil || A.Val != B.Val {return false}return isSubTree(A.Left, B.Left) && isSubTree(A.Right, B.Right)
}

6、实现timer类,功能是注册函数定期执行

package mainimport ("fmt""time"
)type Timer struct {interval time.Durationticker   *time.Tickerdone     chan bool
}func NewTimer(interval time.Duration) *Timer {return &Timer{interval: interval,ticker:   nil,done:     make(chan bool),}
}func (t *Timer) Start() {t.ticker = time.NewTicker(t.interval)go func() {for {select {case <-t.ticker.C:fmt.Println("Timer tick")// 在这里添加想要执行的函数case <-t.done:t.ticker.Stop()return}}}()
}func (t *Timer) Stop() {t.done <- true
}func main() {timer := NewTimer(1 * time.Second)timer.Start()time.Sleep(5 * time.Second)timer.Stop()fmt.Println("Timer stopped")
}

7、生产者消费者

基本实现:

package mainimport ("fmt""math/rand""time"
)func producer(ch chan int) {for {num := rand.Intn(100)ch <- numtime.Sleep(time.Duration(rand.Intn(1000)) * time.Millisecond)}
}func consumer(ch chan int) {for {num := <-chfmt.Println("Consumed:", num)time.Sleep(time.Duration(rand.Intn(1000)) * time.Millisecond)}
}func main() {rand.Seed(time.Now().Unix())ch := make(chan int)go producer(ch)go consumer(ch)time.Sleep(5 * time.Second) // Let the program run for a few seconds
}

8、三个协程轮流打印 abc

参考:https://juejin.cn/post/7262243685898436667

// 使用三个管道实现三个协程同步顺序打印a b c  
func printLetter(letter string, prevCh, nextCh chan struct{}, wg *sync.WaitGroup) {  defer wg.Done()  for i := 0; i < 10; i++ {  // 等待上一个协程通知  <-prevCh  fmt.Print(letter)  // 发送信号给下一个协程  nextCh <- struct{}{}  }if letter == "a" {<-prevCh}
}func main() {var wg sync.WaitGroup  wg.Add(3)  ch1 := make(chan struct{})  ch2 := make(chan struct{})  ch3 := make(chan struct{})  go printLetter("a", ch1, ch2, &wg)  go printLetter("b", ch2, ch3, &wg)  go printLetter("c", ch3, ch1, &wg)  // 启动第一个协程  ch1 <- struct{}{}  wg.Wait()
}

进阶:26个字母

// 使用26个协程分别顺序打印字母表  
func printAlphabet(letter rune, prevCh, nextCh chan struct{}, wg *sync.WaitGroup) {  defer wg.Done()for i := 0; i < 10; i++ {  <-prevChfmt.Printf("%c", letter)nextCh <- struct{}{}}  // 第一个协程必须要退出,因为最后一个协程往管道里面写入数据了,需要破环而出不然就会死锁  if letter == 'a' {<-prevCh}
}func main() {  var wg sync.WaitGroupwg.Add(26)var signals []chan struct{}  for i := 0; i < 26; i++ {  signals = append(signals, make(chan struct{}))  }  for letter, i := 'a', 0; letter <= 'z'; letter++ {  if letter == 'z' {  go printAlphabet(letter, signals[i], signals[0], &wg)break}go printAlphabet(letter, signals[i], signals[i+1], &wg)  i++}// 启动第一个协程  signals[0] <- struct{}{}  wg.Wait()  
}

9、

参考:https://blog.csdn.net/zss192/article/details/138610657#159_32_5898

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java——spring容器启动流程

Spring容器的启动流程是一个复杂但有序的过程&#xff0c;它涉及多个步骤来确保应用程序的组件被正确加载、配置和初始化。以下是Spring容器启动的主要步骤&#xff1a; 一、加载配置文件 Spring容器首先会加载配置文件&#xff0c;这些配置文件通常包含了应用程序的组件、依…

九、Ubuntu Linux操作系统

一、Ubuntu简介 Ubuntu Linux是由南非人马克沙特尔沃思(Mark Shutteworth)创办的基于Debian Linux的操作系统&#xff0c;于2004年10月公布Ubuntu是一个以桌面应用为主的Linux发行版操作系统Ubuntu拥有庞大的社区力量&#xff0c;用户可以方便地从社区获得帮助其官方网站:http…

python excel接口自动化测试框架!

今天采用Excel继续写一个接口自动化测试框架。 设计流程图 这张图是我的excel接口测试框架的一些设计思路。 首先读取excel文件&#xff0c;得到测试信息&#xff0c;然后通过封装的requests方法&#xff0c;用unittest进行测试。 其中&#xff0c;接口关联的参数通过正则进…

基本功能实现

目录 1、环境搭建 2、按键控制灯&电机 LED 电机 垂直按键(机械按键) 3、串口调试功能 4、定时器延时和定时器中断 5、振动强弱调节 6、万年历 7、五方向按键 1、原理及分析 2、程序设计 1、环境搭建 需求: 搭建一个STM32F411CEU6工程 分析: C / C 宏定义栏…

Android 13 Aosp 默认允许应用动态权限

图库 frameworks/base/services/core/java/com/android/server/pm/permission/DefaultPermissionGrantPolicy.java 修改 public void grantDefaultPermissions(int userId) {DelayingPackageManagerCache pm new DelayingPackageManagerCache();grantPermissionsToSysCompon…

操作系统 内存管理——针对实习面试

目录 操作系统 内存管理什么是虚拟内存&#xff1f;什么是物理内存&#xff1f;解释虚拟内存和物理内存的区别什么是分页式存储&#xff1f;什么是分段式存储&#xff1f;解释分页式存储和分段式存储的区别什么是内存碎片&#xff1f;描述几种常见的内存分配算法描述几种常见的…

开源免费的 分布式配置中心 介绍 与 选型 建议

分布式配置中心的应用场景介绍 在微服务架构中&#xff0c;配置管理变得尤为复杂。首先&#xff0c;我们可以想象下&#xff0c;如果没有配置中心&#xff0c;我们的项目可能是这样的&#xff1a;不同环境的配置文件都放在项目里面&#xff0c;部署时可以通过启动参数来指定使…

Linux入门攻坚——39、Nginx入门

Nginx&#xff1a;engine X Tengine&#xff1a;淘宝改进维护的版本 Registry&#xff1a; 使用了libevent库&#xff1a;高性能的网络库 epoll()函数 Nginx特性&#xff1a; 模块化设计、较好的扩展性&#xff1b;&#xff08;但不支持动态加载模块功能&#…

Asp.net core Autofac 案例 注入、AOP 启用接口代理拦截 启用 类代理拦截=== 只会拦截虚方法

资料 core 实现autofac 》》》 安装 如下工具包 安装之后 如出现 这种 》》》编写 AOP类 using Castle.DynamicProxy; using System.Diagnostics;namespace Web01.AOP {/// <summary>/// 日志记录/// </summary>public class LoggingInterceptor : IInterc…

网络安全事件管理

一、背景 信息化技术的迅速发展已经极大地改变了人们的生活&#xff0c;网络安全威胁也日益多元化和复杂化。传统的网络安全防护手段难以应对当前繁杂的网络安全问题&#xff0c;构建主动防御的安全整体解决方案将更有利于防范未知的网络安全威胁。 国内外的安全事件在不断增…

详谈面试题:Vue、React为什么使用虚拟DOM

虚拟DOM是一种在前端框架中广泛使用的技术&#xff0c;它可以提升开发效率。那么国外流行的框架svelte没有使用虚拟DOM&#xff0c;而是直接操作真实DOM&#xff0c;效率依然很高。为什么Vue和React不采用这种方式呢&#xff1f; 目录 一、框架设计 二、解耦运行环境 三、总…

前端JavaScript(一)---基本介绍

Javascript是一种由Netscape(网景)的LiveScript发展而来的原型化继承的面向对象的动态类型的区分大小写的客户端脚本语言&#xff0c;主要目的是为了解决服务器端语言&#xff0c;比如Perl&#xff0c;遗留的速度问题&#xff0c;为客户提供更流畅的浏览效果。当时服务端需要对…

(免费送源码)计算机毕业设计原创定制:Java+B/S+SSM+Web前端开发技术+IDEA+MySQL+Navicat 有风小院

摘 要 信息化社会内需要与之针对性的信息获取途径&#xff0c;但是途径的扩展基本上为人们所努力的方向&#xff0c;由于站在的角度存在偏差&#xff0c;人们经常能够获得不同类型信息&#xff0c;这也是技术最为难以攻克的课题。针对有风小院等问题&#xff0c;对有风小院信息…

Soul App创始人张璐团队亮相GITEX GLOBAL 2024,展示多模态AI的交互创新

随着全球AI领域的竞争加剧,越来越多的科技巨头和创新企业纷纷致力于多模态AI的开发。2024年10月14日至18日,GITEX GLOBAL海湾信息技术博览会在迪拜举行,吸引了超过6700家全球科技巨头和创新公司参与,展示了智能互联、人工智能等领域的新成果。 此次展会中,Soul App创始人张璐团…

新版布谷直播软件源码开发搭建功能更新明细

即将步入2025年也就是山东布谷科技专注直播系统开发,直播软件源码出售开发搭建等业务第9年,山东布谷科技不断更新直播软件功能&#xff0c;以适应当前新市场环境下的新要求。山东布谷科技始终秉承初心&#xff0c;做一款符合广大客户需求的直播系统软件。支持广大客户提交更多个…

Dockerfile打包部署

Dockerfile打包 先找到打包完的目录下创建一个Dockerfile文件 touch Dockerfile 进去文件内编写 vim Dockerfile # 基础镜像 FROM openjdk:8 # author MAINTAINER yxh # 挂载目录 VOLUME /home/project # 创建目录 RUN mkdir -p /home/project # 指定路径 WORKDIR /home/pr…

c++趣味编程玩转物联网:基于树莓派Pico控制有源蜂鸣器

有源蜂鸣器是一种简单高效的声音输出设备&#xff0c;广泛应用于电子报警器、玩具、计时器等领域。在本项目中&#xff0c;我们结合树莓派Pico开发板&#xff0c;通过C代码控制有源蜂鸣器发出“滴滴”声&#xff0c;并解析其中涉及的关键技术点和硬件知识。 一、项目概述 1. 项…

【NLP高频面题 - 分布式训练】ZeRO1、ZeRO2、ZeRO3分别做了哪些优化?

【NLP高频面题 - 分布式训练】ZeRO1、ZeRO2、ZeRO3分别做了哪些优化&#xff1f; 重要性&#xff1a;★★ NLP Github 项目&#xff1a; NLP 项目实践&#xff1a;fasterai/nlp-project-practice 介绍&#xff1a;该仓库围绕着 NLP 任务模型的设计、训练、优化、部署和应用&am…

路由引入中次优路由和路由环路问题

A公司用的是IS-IS&#xff0c;B公司用的是OSPF&#xff0c;现在这两个公司要合并&#xff0c;网络要相通 项目目标 前期准备 配置IP地址&#xff1a;完成IP地址规划&#xff0c;A公司和B公司内部网络通过路由器R2和R4环回接口模拟。配置路由器接口的IP地址并测试所有直连链路的…

shell脚本基础学习_总结篇(完结)

细致观看可以&#xff0c;访问shell脚本学习专栏&#xff0c;对应章节会有配图https://blog.csdn.net/2201_75446043/category_12833287.html?spm1001.2014.3001.5482 导语 一、shell脚本简介 1. 定义&#xff1a; 2. 主要特点&#xff1a; 3. shell脚本的基本结构 4. S…