滴滴三面 | Go后端研发

滴滴三面 | Go后端研发

码农世界 2024-05-30 后端 107 次浏览 0个评论

狠狠的被鞭打了快两个小时… 注意我写的题解不一定是对的,如果你认为有其他答案欢迎评论区留言

bg:23届 211本 社招

1. 自我介绍

2. 讲一个项目的点,因为用到了中间件平台的数据同步,于是开始鞭打数据同步。。

3. 如果同步的时候,插入了新数据怎么处理?

看我们业务对数据实时性的要求

  • 如果实时性要求不高,可以设置定时任务,T+1小时、T+30分钟进行同步或者统一处理
  • 如果实时性要求高,我们可以监听binlog进行消费,不过要做好幂等性方面的工作,防止重复消费

    4. binlog有什么用?

    binlog是存储mysql的数据变更,我们可以通过监听binlog知道数据库发生了哪些变更,通常可以使用binlog进行数据同步、数据备份以及主从复制等等…

    6. binlog的数据格式有哪些?

    binlog日志有三种数据格式

    1. STATEMENT:每一条修改数据的原声 SQL 语句都会被记录到 binlog 中。但有动态函数的问题,比如你用了 uuid 或者 now 这些函数,那么就会导致主库上执行的结果并不是从库执行的结果,这种随时在变的函数就会导致复制前后的数据不一致。
    2. ROW:记录行数据最终被修改成什么样了,不会出现 STATEMENT 下动态函数的问题。但 ROW 的缺点是每行数据的变化结果都会被记录,比如执行UPDATE user_user SET a = 1 WHERE id > 100 语句,那么有多少行数据产生了变化,日志就会记录多少,这会使 binlog 文件过大,而在 STATEMENT 格式下只会记录一个 update 语句而已;
    3. MIXED:包含了 STATEMENT 和 ROW 模式,它会根据不同的情况自动使用 ROW 模式和 STATEMENT 模式

    7. 如何监听binlog?

    一般有两种方案:

    1. 基于 canal 中间件进行监听binlog
    2. 基于flinkcdc监听binlog

    其实这两种区别不大,业内常用的是flinkcdc来监听。原理就是模拟主从复制,将自身模拟程一个slave节点,向master节点发送dump协议,当master节点收到dump协议请求之后,就开始推送binlog到slave。

    滴滴三面 | Go后端研发

    8. mysql dump 之后需要进行什么处理?

    一般接受到的是byte流数据,我们需要解析 binary log 对象才能拿到正在的变更后的数据。那么在接收到主服务器上的 dump 数据后,会根据数据的类型(SQL 查询语句或 Binlog)来进行处理,如果是STATEMENT格式就直接执行sql,如果是ROW格式就直接记录数据。(不确定是不是这样)

    9. 唯一索引和联合索引有什么区别?

    • 唯一索引是指索引列的取值必须是唯一的,索引列中的值不能重复。如果尝试插入重复值,数据库会抛出唯一性约束错误。
    • 联合索引是指索引包含多个列,通过这些列的组合值进行索引。当查询时涉及到联合索引的所有列,数据库会使用该联合索引进行优化查询,提高多列查询的效率,特别是当这些列经常一起作为查询条件时。联合索引中的列顺序很重要,查询时必须按照最左匹配原则进行查询,否则索引无法生效。

      10. 联合索引可以是唯一索引吗?

      可以的,这样就意味着索引列的组合值必须是唯一

      举个例子:

      创建下面一张表结构,如下结构,我们创建了一个联合的唯一索引,username和email作为联合的唯一索引列

      滴滴三面 | Go后端研发

      当我们插入数据

      滴滴三面 | Go后端研发

      插入相同索引数据的时候,就会报错了

      滴滴三面 | Go后端研发

      11. 那mysql索引结构是什么样的?

      mysql的索引结构是B+树,

      • 根节点:包含指向子节点的指针。
      • 中间节点:包含索引键值和指向子节点的指针。
      • 叶子节点:包含索引键值和指向实际数据行的指针。

        数据按照索引键值的顺序存储在叶子节点中,这样可以通过在树中进行一系列比较操作来快速定位到所需的数据行。叶子节点之间通过指针连接,形成一个有序的链表,这样可以支持范围查询和排序操作。

        滴滴三面 | Go后端研发

        12. 一个索引的建立过程是什么样的?

        1. 插入数据:当插入一个新的数据项时,首先在叶子节点中找到合适的位置插入数据。如果插入后叶子节点的数据项数量超过了阶数的限制,就需要进行节点分裂操作。
        2. 节点分裂:当一个节点中的数据项数量超过了阶数的限制时,该节点需要进行分裂。分裂操作会将该节点中的数据项分为两部分,然后将中间值上移到父节点中,以保持 B+树的平衡性。
        3. 向上递归:如果父节点也满足不了插入新数据项的条件,就需要继续向上递归进行节点分裂操作,直到根节点。如果根节点也满了,则根节点会分裂成两个节点,同时树的高度增加一层。
        4. 更新索引:在每次节点分裂后,需要更新父节点的索引信息,确保索引的正确性。
        5. 删除数据:删除数据时,首先在叶子节点中找到要删除的数据项,然后将其删除。如果删除后导致节点的数据项数量低于阶数要求的最小值,需要进行节点合并操作。
        6. 节点合并:当一个节点中的数据项数量低于阶数要求的最小值时,该节点需要与其兄弟节点进行合并操作。合并操作会将两个节点合并成一个节点,并将父节点中的相应索引项删除。

        13. 如果我对age字段建立索引,建立的过程是什么样的?

        举个age的例子:

        最开始的时候是一段链表

        滴滴三面 | Go后端研发

        然后当我们数据变多的时候,会对这个链表进行拆分,抽取

        滴滴三面 | Go后端研发

        当我们的数据越来越多的时候,会不断向上抽取,一般会抽成三层

        滴滴三面 | Go后端研发

        14. 为什么走索引加快了?

        1. 减少数据扫描:当数据库表中有索引时,MySQL可以通过索引快速定位到符合查询条件的数据行,而不需要对整个表进行扫描。这样可以大大减少需要扫描的数据量,提高查询速度。
        2. 加快数据定位:索引使得数据库系统能够更快速地定位到需要的数据行,而不需要逐行查找。通过索引,MySQL可以跳过大部分数据行,直接定位到目标数据行,从而减少了数据访问的时间。
        3. 降低磁盘I/O操作:索引可以减少磁盘I/O操作的次数。由于索引使得数据定位更快速,数据库系统需要读取的数据页数减少,从而减少了磁盘I/O操作,提高了查询效率。

        15. 为什么age可以建立索引?sex字段就不行?

        sex字段一般是个枚举值0,1,2,那么如果是sex的枚举的话,就会变成

        滴滴三面 | Go后端研发

        我们最终无法通过索引命中我们需要的节点,所以我们必须扫全表才能找到这条数据

        注意一点:

        InnoDB中的聚集索引的叶节点就是最终的数据节点,InnoDB中的非聚集索引叶子节点指向的是相应的主键值。而MyISAM中非聚集索引的主键索引树和二级索引树的叶节仍然是索引节点,但它有一个指向最终数据的指针

        16. 为什么sex建立索引还是会扫全表?

        性别字段因为可重复所以只能建立非聚集索引,然而因为非聚集索引叶子节点存储的是索引值和聚集索引值,所以非聚集索引不能直接获取到数据,需要通过逻辑指针进行二次查找来获取数据,也就是需要回表的。

        那么无论搜索哪个sex字段都可能得到1/3的数据。在这些情况下,还不如不要索引,而且数据库优化器最终很大概率也不会选择走这个索引,因为 MySQL 优化器发现某个值出现在表的数据行中的百分比很高的时候,它一般会忽略索引,进行全表扫描。

        17. 如果我订单表达到一定规模之后mysql单表是撑不住了,怎么办?

        首先要做好调研工作,根据当前业务的发展情况,去选择合适的技术方案。

        • 如果这个业务比较重要(能赚钱),那么我们可以做分库分表,用比较多的人力和精力去维护。
        • 如果这个业务相对来说不那么重要(辅助性业务),可以用当前比较热门的分布式数据库做DTS,比如 tidb、ES,减少我们的投入精力。

          18. 具体你会怎么分库分表?

          一般有水平扩展和垂直扩展,然后根据当前业务现状选择合适的分片键

          • 水平扩展:根据日期进行分库分表,同一时期分在一个表或者一个库中。
          • 垂直扩展:根据分片键进行hash命中哪一个库和表,就存在哪一个库表,分片键一般会取用户id的后5~7位

            滴滴三面 | Go后端研发

            19. 分库分表如果进行条件查询?

            分片键只是为了路由命中我们要查的数据在哪个库,哪个表,命中之后还是需要带条件走索引查询。

            一般会设置多个分片键,以防多个业务场景需要,如果实在没有命中任何库表,会兜底去查分布式数据库,tidb、es之类的,因为tidb、es会有dts进行同步。

            20. 同步ES?不使用其他组件,单单是mysql怎么操作,所有表遍历找过去吗?索引会不会失效?

            如果我们不走ES、TiDB这种并且没有分片键命中的话,那么我们就要在go里面用到协程池去做批量查询的操作了,每张表都用新协程去查询然后聚合结果,同时用协程池进行池化,减少协程的频繁创建与销毁。

            滴滴三面 | Go后端研发

            当然其实这种场景是很少的,99%的业务场景都可以使用分片键去处理,如果实在很难用分片键处理,我们一般会和业务沟通,甚至拒掉这个需求。索引是不会失效的,如果索引失效的话,那将会是P2级及以上的bug了。

            21. ok,redis源码有了解吗?他的线程模型是什么样的?

            其实我们说redis是单线程,说的是只有一个进程处理主线任务,主线任务就是将接受客户端命令,将命令传送到服务端,服务端处理完,再将数据返回给客户端。

            滴滴三面 | Go后端研发

            其实redis在启动的时候,是会启动一个后台线程去加载日志数据。

            22. redis 有哪些存储日志的形式?同步还是异步?

            • RDB持久化就是指的讲当前进程的全量数据生成快照存入到磁盘中,触发RDB机制又分为手动触发与自动触发。
            • AOF 持久化是以独立的日志记录每次写命令,也就是增量数据,所以AOF主要就是解决持久化的实时性。

              是否异步?

              • AOF 是同步的
              • RDB的save是同步的,bgsave是异步的

                AOF和主进程的关系如下:

                滴滴三面 | Go后端研发

                AOF的写入是同步的,AOF写入不会阻塞当前的写命令,但是有可能会阻塞下一个写命令

                注意是写命令进行日志写入,读命令才会记录日志

                23. 那AOF具体是怎么存储日志的?

                1. Redis 写操作命令结束后,会将命令重写到 AOF 缓冲区
                2. 然后通过系统调用,将 AOF 缓冲区的数据拷贝到了内核缓冲区
                3. 然后内核会将数据写入硬盘,具体内核缓冲区的数据什么时候写入到硬盘,由内核决定,而这也是数据丢失的一个隐患。

                  滴滴三面 | Go后端研发

                24. AOF不断的写日志不是会有很多的io操作吗?怎么避免?

                其实我们业务背景中,redis的读和写的比例一般会是9:1这个状态,写一次redis就意味着这一次请求可能打到DB了。如果是在一些极限的场景,比如滴滴的订单需要频繁的修改状态,那么我们可以设置redis的aof的落盘策略。

                • Always:每次写操作命令执行完后,实时将 AOF 日志数据写回硬盘
                • Everysec:每次写操作命令执行完后,先将命令写入到 AOF 文件的内核缓冲区,然后每隔一秒将缓冲区里的内容写回到硬盘
                • No: Redis 不控制写回硬盘的时间,每次写操作命令执行完后,先将命令写入到 AOF 文件的内核缓冲区,再由操作系统决定什么时候将缓冲区内容写回硬盘

                  如果我们需要极致的性能,选择No就可以了,当然高回报的背后是高风险,由于我们并不知道操作系统会什么时候写入硬盘,所以如果redis宕机了,我们将会丢失这段时间的数据。所以在生产当中我们一般会使用AOF + RDB的组合方式来保证AOF的持久化。

                  25. RDB是怎么进行操作?

                  redis 一般会有两个命令生产RDB,一个是save,另一个是bgsave,区别在于是否阻塞主线程

                  • save 是和操作命令在同一个线程里面,如果有大key写入RDB文件,会造成阻塞。
                  • bgsave是创建一个子进程来写入RDB文件,不会造成阻塞。

                    RDB和everysec模式的AOF区别就是一个是全量,一个是增量

                    那么在bgsave中,通过fork函数创建的子进程会和父进程共享同一片内存数据,因为在fork的时候,子进程会复制父进程的页表数据,而这两个进程的页表数据一摸一样也就表示会指向同一片物理内存地址。 这样是为了减少创建子进程时的性能损耗,加快子进程的创建速度,毕竟创建子进程的过程中,是会阻塞主线程的。

                    滴滴三面 | Go后端研发

                    26. 那如果数据库和redis需要缓存一致性怎么解决?

                    其实我们一般都是先更新数据库再更新redis的,在业务允许的下,可以接受一定时间段的数据不一致,因为我们业务场景都只是要求数据的最终一致性就可以了。

                    27. 那我不考虑最终一致性,我要强一致性,怎么解决?

                    1. 那我们可以需要牺牲一点性能了,我们将 redis 和 mysql 的数据绑定在一起,redis 使用 lua 脚本更新,和mysql 的更新数据绑定在一点,要么一起成功,一起失败,如果失败则需要告警,走降级兜底,然后人工快速介入。

                    滴滴三面 | Go后端研发

                    1. 又或者可以把缓存删掉,然后再更新数据库,此时大量的请求将会打到DB,这时候,我们就需要使用singlefight 这种在缓存失效的时候使用的合并请求的方式减轻DB的压力。

                    其实无论怎么选型都不会非常完美的,CAP必定会牺牲一个,即使是99.9999%高可用的阿里云也会有宕机的情况。这个一致性如果要深究一篇文章都讲不完…这里不再过度描述…

                    28. ok,网络这块我也问问。http和https的区别是什么?

                    http和https其实就是有一个加密的过程,这个加密就是tls、ssl加密。

                    29. https是怎么建立连接的?先建立什么?再建立什么?

                    先建立tcp连接再建立ssl、tls连接

                    30. 具体是怎么建立的ssl tls加密?

                    1. 客户端 Hello:客户端向服务器发送一个Client Hello消息,其中包含支持的加密算法、协议版本、随机数等信息。
                    2. 服务器 Hello:服务器收到Client Hello后,选择加密算法和协议版本,并向客户端发送一个Server Hello消息。
                    3. 证书验证:服务器还会发送自己的数字证书给客户端,客户端会验证证书的有效性,包括证书是否由受信任的证书颁发机构颁发。
                    4. 密钥交换:如果证书验证通过,客户端生成一个随机数,使用服务器的公钥(从证书中提取)加密后发送给服务器,服务器使用自己的私钥解密得到该随机数,用于生成对称加密的会话密钥。
                    5. 完成握手:握手阶段完成后,客户端和服务器都知道如何加密通信,并且共享一个会话密钥用于加密和解密数据。

                      滴滴三面 | Go后端研发

                      四次握手主要是交换以下信息:

                    • 数字证书:该证书包含了公钥等信息,一般是由服务器发给客户端,接收方通过验证这个证书是不是由信赖的CA签发,或者与本地的证书相对比,来判断证书是否可信;假如需要双向验证,则服务器和客户端都需要发送数字证书给对方验证;

                    • 三个随机数:这三个随机数构成了后续通信过程中用来对数据进行对称加密解密的对话密钥。

                      • 首先客户端先发第一个随机数N1,然后服务器回了第二个随机数N2(这个过程同时把之前提到的证书发给客户端),这两个随机数都是明文的;
                      • 而第三个随机数N3(预主密钥),客户端用数字证书的公钥进行非对称加密,发给服务器,服务器用只有自己知道的私钥来解密,获取第三个随机数。
                      • 服务端和客户端都有了三个随机数N1+N2+N3,然后两端就使用这三个随机数来生成“对话密钥”,在此之后的通信都是使用这个“对话密钥”来进行对称加密解密。因为这个过程中,服务端的私钥只用来解密第三个随机数,从来没有在网络中传输过,这样的话,只要私钥没有被泄露,那么数据就是安全的。
                      • 加密通信协议:就是双方商量使用哪一种加密方式,假如两者支持的加密方式不匹配,则无法进行通信;

                        一句话概括也就是用非对称加密来生成对称加密的密钥

                        31. http的请求头和响应头一般有什么信息,有什么用?

                        • HTTP请求头(Request Headers):

                          • Host:指定请求的目标主机。
                          • User-Agent:标识发起请求的客户端软件。
                          • Accept:指定客户端能够接受的内容类型。
                          • Content-Type:指定请求体的MIME类型。
                          • Authorization:用于身份验证的信息。
                          • Cookie:包含客户端发送给服务器的Cookie信息。
                          • Referer:指示请求的来源页面的URL。
                          • Cache-Control:指定缓存行为。
                          • Connection:指定是否保持持久连接。
                          • Accept-Encoding:指定客户端支持的内容编码方式。
                          • HTTP响应头(Response Headers):

                            • Content-Type:指定响应体的MIME类型。
                            • Content-Length:指定响应体的长度。
                            • Server:指定响应的服务器软件。
                            • Set-Cookie:在响应中设置Cookie。
                            • Cache-Control:指定缓存行为。
                            • Location:指定重定向的URL。
                            • Expires:指定响应过期的时间。
                            • Last-Modified:指定资源的最后修改时间。
                            • Access-Control-Allow-Origin:指定允许跨域请求的来源。

                              32. ok,页的概念你清楚吗?

                              对于进程来说,使用的都是虚拟地址, 虚拟地址空间划分为多个固定大小的虚拟页(VP),物理地址空间划分为多个固定大小的物理页(PP),虚拟页和物理页的大小是一样,通常为4KB。页的主要功能是做出虚拟地址对物理地址的映射。

                              • 页表:每个进程维护一个单独的页表,页表是一种数组结构,存放着各虚拟页的状态,是否映射,是否缓存。
                              • 页表项:页表中的每个条目称为页表项,其中包含了虚拟页号和物理页框号之间的映射关系。
                              • 页面调度:当程序访问一个虚拟地址,而对应的物理页不在内存中时,会发生页面调度,操作系统会从磁盘中将相应的页面加载到内存中。
                              • 页面置换:当内存中的页面不足时,操作系统需要选择一个页面进行置换(page replacement),将其写回磁盘并加载新的页面。

                                通过使用页面和页表,操作系统可以实现虚拟内存管理,提高内存利用率和程序的运行效率。虚拟内存技术允许程序看到一个比实际物理内存更大的地址空间,同时可以将不常用的页面置换到磁盘上,从而提高系统的整体性能和稳定性。

                                滴滴三面 | Go后端研发

                                33. 页碎是什么?

                                页碎片是指在虚拟内存系统中,由于分配和释放内存的过程中导致的页面不连续、零散的现象。

                                在虚拟内存管理中,内存通常被划分为固定大小的页面,而应用程序请求的内存空间可能不是页面大小的整数倍,这就导致了页面的碎片化。

                                而大量的页碎可能会导致以下问题:

                                • 分配性能下降:当系统需要分配大块连续内存时,如果内存中存在大量碎片,系统可能需要进行额外的合并操作,降低了分配性能。
                                • 缓存失效率增加:页面碎片化也可能导致缓存的失效率增加,因为数据分散存储在不同的页面中,需要更多的缓存行加载,降低了缓存的效率。
                                • 内存利用率下降:由于页面被分割成小块或者存在空隙,导致实际可用内存空间比总内存空间要少。
                                • 缓存失效率增加:页面碎片化也可能导致缓存的失效率增加,因为数据分散存储在不同的页面中,需要更多的缓存行加载,降低了缓存的效率。

                                  通常会采取一些策略来优化内存分配和释放,比如使用内存池、动态内存分配算法、碎片整理等技术来减少碎片化,提高内存利用率和系统性能。

                                  滴滴三面 | Go后端研发

                                  34. 为什么需要内存对齐?

                                  内存对齐,就是将数据存放到一个是字的整数倍的地址指向的内存之中。处理器在执行指令去操作内存中的数据,这些数据通过地址来获取。为了能让复杂数据结构对齐,编译器一般会对数据结构做一些填充。

                                  内存对齐总的来说就是两个原因:提升效率和避免出错。

                                  1. 某些处理器只能存取对齐的数据,存取非对齐的数据可能会引发异常;
                                  2. 某些处理不能保证在存取非对齐数据的时候的操作是原子操作;
                                  3. 相比于存取对齐的数据,存取非对齐数据需要额外花费更多的时钟周期;
                                  4. 有些处理器虽然支持非对齐数据访问,但是会引发对齐陷阱;
                                  5. 某些处理只支持简单数据指令非对齐存取,不支持复杂数据指令非对齐存取。

                                  而我们可能需要进行类型转换、位运算、使用特定平台的指令、内存拷贝这些额外的操作取非对齐的数据,而这也需要额外的指令来访问和处理这些数据,也造成了不必要的开销。

                                  35. go里面怎么样会发生死锁?死锁的场景具体有哪些?

                                  我们了解一下死锁的必要条件:互斥、请求与保持、不可抢占、循环等待

                                  • 相互等待资源:两个或多个goroutine相互持有对方需要的资源而无法释放,例如,goroutine A持有资源X并等待资源Y,而goroutine B持有资源Y并等待资源X。
                                  • 未释放锁:一个goroutine获取了一个锁,但在释放之前就阻塞了,导致其他goroutine无法获取该锁。
                                  • 未缓冲的通道:当一个goroutine试图向一个未缓冲的通道发送数据,但没有接收者时,它会被阻塞,这可能导致死锁。
                                  • 使用互斥锁的顺序问题:如果多个goroutine以不同的顺序获取多个互斥锁,可能会导致死锁。
                                  • 等待超时不处理:如果goroutine在等待资源时没有设置超时或者没有处理超时情况,可能会导致永久等待。

                                    36. 内存泄漏有哪些场景?怎么排查?

                                    内存泄漏是指程序在动态分配内存后,由于某种原因未能释放或回收这些内存,导致系统中的可用内存持续减少,最终可能耗尽系统资源。

                                    • 未释放资源:动态分配的内存或其他资源(如文件流、响应流、数据库连接等)在使用完后未被释放。
                                    • 循环引用:两个或多个对象相互引用,导致它们之间形成循环引用,即使程序不再需要这些对象,它们也无法被垃圾回收机制回收。
                                    • 缓存未及时清理:缓存中的对象长时间未被清理或更新,导致不再需要的对象仍然占用内存。
                                    • 大对象未释放:大对象(如大型数组、大文件等)在使用完后未被及时释放,占用大量内存。

                                      在go语言里面,其实内存泄漏不太好排查,因为gc和编辑器做的优化都太好了,不怎么容易发生内存泄漏。那么如果发生了内存泄露,我们可以采取以下方法:

                                      • 内存分析工具:使用内置或第三方的内存分析工具(如Go语言中的pprof工具)来检测程序的内存使用情况,查看内存分配情况和对象生命周期。
                                      • 代码审查:仔细审查代码,查找可能导致内存泄漏的地方,例如未释放资源、循环引用等。
                                      • 日志和监控:添加日志和监控,记录程序的内存使用情况,以便及时发现内存泄漏问题。
                                      • 压力测试:进行压力测试,模拟程序长时间运行,观察内存使用情况是否持续增长。
                                      • Heap Dump:在发生内存泄漏时,生成Heap Dump文件,用于分析程序中的对象和内存使用情况。

                                        37. goruntine 泄漏的场景有哪些?怎么排查?

                                        Goroutine泄漏指的是创建的goroutine没有被正确释放或终止,导致这些goroutine继续存在而不被使用,最终可能导致系统资源的浪费,影响程序的并发执行能力。

                                        1. 未等待goroutine完成:在启动goroutine后没有正确等待goroutine执行完成,导致goroutine未被正确处理而继续存在并且没有被gc回收。
                                        2. 阻塞导致无法退出:某些情况下,goroutine可能会被阻塞而无法正常退出,例如等待通道操作、锁操作等。
                                        3. 循环/递归创建goroutine:在循环中创建goroutine,但未控制goroutine的数量和生命周期,导致大量无法及时销毁的goroutine。在没有终止条件或终止条件不正确的情况下递归,导致goroutine的无限增长。

                                        排查:

                                        • 使用go vet工具:Go提供了go vet工具,可以静态分析检查代码。
                                        • 使用pprof工具:pprof工具可以分析程序的性能和资源使用情况,包括goroutine的使用情况。

                                          预防:

                                          • 监控日志告警:添加监控,日志和告警,以便及时发现异常情况。
                                          • 代码审查:仔细审查代码,特别是涉及goroutine启动和结束的地方,确保每个goroutine都能被正确处理。
                                          • 限制goroutine数量:在创建goroutine时,可以考虑限制goroutine的数量,比如使用协程池进行池化,避免无限制地创建goroutine。

                                            38. 进程、线程、协程有什么区别?

                                            39. 协程能被 kill 掉吗?

                                            不能,kill 作用对象是进程,是进程管理的常用命令,实施对象是操作系统。

                                            40. 那协程应该怎么处理?

                                            协程如果要实现“被kill”的效果,可以使用context包进行timeout处理,这样如果到了一定时间如果还没执行完就进行context cancel了。

                                            41. 那context一般有什么信息?有什么用途?

                                            42. 那如果我要 clone 一个context,子context 和 父context 是一摸一样吗?为什么?

                                            41、42 另外开一期讲讲context吧

                                            43. singleflight 是什么?什么时候用的?

                                            缓存失效,合并请求的时候用的

                                            44. 如果这个goruntine超时怎么办?

                                            45. doChan方法具体是怎么实现的?

                                            43-45 也单独开一期讲讲 singleflight 。

                                            46. 为什么会有饥饿模式?

                                            互斥锁在设计上主要有两种模式: 正常模式和饥饿模式。

                                            之所以引入了饥饿模式,是为了保证goroutine获取互斥锁的公平性。 所谓公平性,其实就是多个goroutine在获取锁时,goroutine获取锁的顺序,和请求锁的顺序一致,则为公平。

                                            1. 正常模式下,所有阻塞在等待队列中的goroutine会按顺序进行锁获取,当唤醒一个等待队列中的goroutine时,此goroutine并不会直接获取到锁,而是会和新请求锁的goroutine竞争。 通常新请求锁的goroutine更容易获取锁, 这是因为新请求锁的goroutine正在占用cpu片执行,大概率可以直接执行到获取到锁的逻辑
                                            2. 饥饿模式下, 新请求锁的goroutine不会进行锁获取,而是加入到队列尾部阻塞等待获取锁。

                                            47. 什么时候会让出时间片?

                                            1. 时间片用尽:当进程或线程的时间片用尽时,操作系统会强制进行上下文切换,将CPU资源分配给其他就绪状态的进程或线程。
                                            2. 阻塞操作:当进程或线程执行阻塞操作 (如等待I/O操作完成、等待信号量、等待锁等) 时,它会让出CPU时间片,以便其他就绪状态的进程或线程能够执行。
                                            3. 显式让出:进程或线程可以通过系统调用或特定的API显式让出CPU时间片,例如在 Golang 进程会有一个全局监控协程 monitor g 的存在,这个 g 会越过 p 直接与一个 m 进行绑定,不断轮询对所有 p 的执行状况进行监控. 倘若发现满足抢占调度的条件,则会从第三方的角度出手干预,主动发起该动作。
                                            4. 优先级调度:如果有更高优先级的进程或线程需要执行,当前进程或线程可能会让出时间片,以便高优先级任务能够及时执行。
                                            5. 信号处理:当进程接收到信号并需要处理时,它可能会让出CPU时间片来处理信号。毕竟信号也可以在多线程中传递信息。

                                            48. IO密集型和计算密集型的区别?

                                            在计算机科学中,我们通常将任务分为两类:IO密集型和计算密集型。

                                            1. IO密集型任务:

                                              • 这类任务主要涉及大量的输入/输出操作,如从磁盘读取数据、网络通信、数据库查询等。在执行过程中,任务会频繁地进行IO操作,而不是大量的计算操作。
                                              • IO密集型任务的特点是CPU通常会大部分时间处于空闲状态,等待IO操作完成。因此,是非阻塞的。
                                              • 计算密集型任务:

                                                • 这类任务主要涉及大量的计算操作,例如复杂的数学运算、图像处理、加密算法等。在执行过程中,任务会消耗大量的CPU资源进行计算操作。
                                                • 计算密集型任务的特点是CPU会长时间处于繁忙状态,执行大量的计算操作,因此是阻塞的。

                                            最后是几道代码题,飞书文档看着代码,不使用任何编辑器。

                                            这段代码会发生什么?为什么?具体是怎么溢出的?
                                            var a uint = 1
                                            var b uint = 2
                                            fmt.Println(a-b) 
                                            

                                            这个uint类型的溢出我就不过多赘述了。

                                            说出以下输出结果?为什么?
                                            func TestSlicePrint(t *testing.T) {
                                            	a := []byte("AAAA/BBBBB")
                                            	index := bytes.IndexByte(a, '/')
                                            	b := a[:index]
                                            	c := a[index+1:]
                                            	b = append(b, "CCC"...)
                                            	fmt.Println(string(a))
                                            	fmt.Println(string(b))
                                            	fmt.Println(string(c))
                                            }
                                            

                                            结果:

                                            AAAACCC
                                            CCBBB
                                            AAAACCCBBB
                                            

                                            切片是对底层数组的引用,因此对切片的修改会影响原始的切片。

                                            滴滴三面 | Go后端研发

                                            当一个切片被用做一个append函数调用中的基础切片时:

                                            • 如果添加的元素数量大于此(基础)切片的冗余元素槽位的数量,则一个新的底层内存片段将被开辟出来并用来存放结果切片的元素。 这时,基础切片和结果切片不共享任何底层元素。

                                              滴滴三面 | Go后端研发

                                              • 否则,不会有底层内存片段被开辟出来。这时,基础切片中的所有元素也同时属于结果切片。两个切片的元素都存放于同一个内存片段上。

                                                滴滴三面 | Go后端研发

                                                这个锁是什么用的?这段代码有什么问题?
                                                func TestNumPrint(t *testing.T) {
                                                	wg := sync.WaitGroup{}
                                                	lock := new(sync.Mutex)
                                                	var a int32 = 0
                                                	var b int32 = 2
                                                	for i := 0; i < 5; i++ {
                                                		go func() {
                                                			if a > b {
                                                				fmt.Println("done")
                                                				return
                                                			}
                                                			lock.Lock()
                                                			defer lock.Unlock()
                                                			a++
                                                			fmt.Printf("i: %d a: %d \n", i, a)
                                                		}()
                                                	}
                                                	wg.Wait()
                                                }
                                                

                                                我个人觉得有三点:

                                                1. wg没有add(1)导致主进程结束了,子进程还没开始。但这个被面试官否掉啦,“你就当主进程一直阻塞就好了”
                                                2. 这个i的变量有问题,都将会是最后一个数也就是5的这个值。也被否掉了,“go最新版里面这个不是问题,你就当是最新版吧”
                                                3. 还有就是这个锁有问题,虽然锁住了a的自增,但是没锁住a的读取,int 类型本身并不是并发安全的,我们必须加锁才能保证原子性,那么如果我们不加锁,还可以使用atomic.AddInt32() 进行原子操作。但是这里锁的粒度出现了问题,锁应该是对变量的读取和修改都进行锁,上面只锁住了修改,锁应该要锁出一段逻辑操作,而不是一个变量,所以只需要将锁提到if a>b的前面。
                                                最后手撕一题:整数反转。直接在飞书文档上写,写的真难受…

                                                面试官在我写的过程依次提示考虑正数、负数、正数溢出、负数溢出?正数溢出和负数溢出是不是一样的?

                                                这题是全场最简单的了,略,具体思路就是先计算这个整数的位数,然后反着来相乘再相加就好了。相加的过程中注意正数的溢出,如果是负数的话,做个标识,让负数变正数,最后再把符合加上就好了

                                                参考资料

                                                [1] https://www.jianshu.com/p/67600a8ddb8c

                                                [2] https://zh.m.wikipedia.org/wiki/数据结构对齐

                                                [3] https://www.cnblogs.com/rjzheng/p/12557314.html

                                                [4] https://golang.design/under-the-hood/

转载请注明来自码农世界,本文标题:《滴滴三面 | Go后端研发》

百度分享代码,如果开启HTTPS请参考李洋个人博客
每一天,每一秒,你所做的决定都会改变你的人生!

发表评论

快捷回复:

评论列表 (暂无评论,107人围观)参与讨论

还没有评论,来说两句吧...

Top