含山县查询建设工程的网站,网站建设网站营销网站托管一体化,网站地址验证失败,莱芜最新话题个人评价 难度还是有的#xff0c;中等难度吧#xff0c;可能是因为项目使用的是物流项目#xff0c;该项目本来就比较庞大难度比较高#xff0c;流的八股文我真的是一点不会#xff0c;还需要加强#xff0c;reidis的多路io复用模型没有深问#xff0c;要是问了就寄了中等难度吧可能是因为项目使用的是物流项目该项目本来就比较庞大难度比较高流的八股文我真的是一点不会还需要加强reidis的多路io复用模型没有深问要是问了就寄了这个模型的底层太就没有复习了这次面试题差不多就是 50%项目50%八股文。
1.实习项目有哪些功能技术栈有哪些 实习项目就是基于springCloud-alibaba那套微服务方案落地的物流调度微服务项目。主要就是从用户下单到快递员收件物流调度到最终的快递员派送其中最主要的就是调度模块及运输状态的展示模块。 技术栈主要就是 Spring全家桶SpringCloud-alibabamysqlredisRabbitMqxxl-job。skywalking。
2.实习中遇到过的最大难点怎么解决的呢 在调度模块中的订单转运单到运单合并的这个流程还是比较难的。业务要求运单的id的格式为两个字母16为数字并且需要保证id为分布式id我们这里就无法使用UUID因为生成的是32为字符不符合条件雪花算法也不去使用因为雪花算法基于时间戳所以还是存在id冲突的情况。因此我使用美团leaf即可自定义格式还支持号段模式减少DB操作但是能在后续我们进行测试的时候发现了一个严重的问题在每次使用完id后需要长时间的db操作出现了尖刺问题为了解决这个问题我们使用它的双Buffer模式其底层就是基于双号段异步实现的解决了尖刺问题。 在运单合并中为了模拟等待队列的结构使用redis的list结构key存储两个网点id的拼接value就是存储对应当前网点和下一网点的运单后续定时任务进行运力计算也就是消费运单。但是能在后续测试的时候发现高并发的场景下运单会被多次消费为了保证运单的幂等性我们使用redis的set的结构,key和value的存储数据类似list在消费运单的时候需要先判断运单是否还存在。
3.刚刚听你提到运输信息模块你说说这个模块的实现吧 这个模块的主要功能就是展示运输的位置及状态刚开始的时候考虑使用mysql存储位置数据但是呢运单的数据比较庞大如果用mysql存储的话数据量就非常的庞大。经过讨论决定使用Mongdb进行存储因为Mongdb存在嵌套的document结构使用一个list属性存储运输位置。数据量就不会那么庞大并且Mongdb非常适合存储海量且不重要的数据。该调度模块中每次进行车辆入库的适合就会异步的发送消息修改list中的数据。退单的时候将棕垫和起点互换并做list数据的拼接即可。 这个业务的解决不算太难主要就是后续的优化比较难运输信息的模块算是用户访问比较多的模块。所以就就存在高并发的场景同一时间做大量的DB操作就会压垮数据库。最先开始的时就是做缓存用springCache。因为这里的缓存我们设置的ttl是相同的所以就存在缓存雪崩的问题。因此我们就不单单使用redis做缓存我们还使用Caffaine做二级缓存在进行写操作的时候就会清空缓存。但是呢在后续进行测试还是出现Caffaine缓存数据不一致的情况只要就是因为Caffaine是基于一个JVM的问题。我们就使用redis的发布与订阅模式来解决这个问题我们会让服务节点去订阅一个频道该频道在会在做写操作的时候收到消息最终每个节点的都会清除Caffaine中的缓存解决缓存不一致的问题。
4.你说使用redis的发布与订阅模式来解决缓存不一致的情况能不能使用MQ来解决呢 其实在最开始的时候就是想用RabbitMQ发送信息的方式来做清除缓存的通知但是呢在后续测试的时候还是发现数据不一致的情况只就是因为消息只能被一个服务节点监听并消费到其他的服务节点的缓存还是没有被清除。
5.谈一下你对分布式锁的理解吧 在我的项目中分布式锁的使用还是比较多的就拿我的电影院项目来说吧优惠劵模块中在解决同用户并发超领的问题上就使用了分布式锁锁住userId保证每次相同用户只能有一个线程进行领卷操作。我们只要就是使用redisson来做分布式锁的。 redisson分布式锁底层的实现我还是有了解的。redisson实现的分布式锁的特性就是支持锁的重入支持ttl重置支持阻塞重试机制。 锁的重入机制主要就是通过redis的hash结构来实现的大key存储业务id小key存储线程idvalue存储锁的重入次数。 锁的ttl重置机制主要就是通过Watch dog来实现的Watch dog本质就是一个定时任务每30秒就会去重置锁的过期时间当我们获取锁的时候没有设置超时时间就会触发ttl重置机制。 锁的阻塞重试机制主要就是通过redis的发布与订阅模式实现的获取锁失败的线程会去订阅一个频道其他线程解锁后就会向该频道发送消息让获取锁失败的线程进行重试。获取锁失败的线程可能存在很多个所以还会存在锁的争抢。
6.谈一下你对数据库索引的理解吧 从索引的数据结构方面Myisam和Innodb索引的底层都是使用B树实现的因为叶子节点存储数据所以在范围查询的时候速度是很快的。 索引分为聚簇索引和非聚簇索引聚簇索引的叶子节点存储一整行的数据非聚簇索引叶子节点存储主键。所以做查询操作的时候没有覆盖索引就会走非聚簇索引最终导致回表降低查询速度。 在编写SQL语句的时候要避免索引失效的问题。 我们可以通过explain索引的覆盖情况和回表的情况。
7.你说到索引失效的问题拿索引失效的场景有哪些 没有遵循最左前缀原则在做条件的跳过顺序的字段导致部分索引失效。 使用范围查询,时会导致符号右侧的部分索引失效可以通过,进行避免。 对条件字段使用聚合函也会导致部分索引失效。 当条件字段发生自动的类型转化的时候也会导致索引失效。 当条件中使用or进行拼接也会导致右侧索引失效。 当使用模糊匹配的时候字符串中如果使用%开头的话就会导致索引失效。 数据的分布影响导致索引失效当查询的结果数量大于等于30%就不会走索引。
8.java的垃圾回收算法有哪些 标记清除法通过可达性分析算法主要就是通过GCRoot进行判断标记存活的对象将其他对象进行回收。缺点很明显会导致空间碎片话在极端情况会无法创建数组。 标记整理法通过可达性分析算法标记存活的对象将标记的对象全部移动到内存的一侧在进行垃圾的回收缺点就是需要大量的移动操作效率比较低。 复制算法会将内存分为两个部分在通过可达性分析算法标记对象后将对象复制到另一侧的内存中并清除当前内存中的对象下次进行垃圾回收的时候就会复制另一内存中以此规则进行垃圾回收。
9.谈一下Java的内存管理 java的内存主要就是堆和栈。 内存堆主要就是存储实例对象及数组其中分为新时代老年代方法区。新时代中又分为Eden区,from幸存者区to幸存者区。方法区中主要就是存储类的信息常量等等。但是呢在jdk1.8之后移除了方法区而是在本地内存中添加了一个元空间用来存储这些数据解决OOM的问题。 内存栈中主要就是存储当前的执行方法及局部变量等数据栈帧不会涉及JVM的内存回收因为栈只会又一个移动的栈头且在栈头运行完成后就会自动释放内存。 内存堆中的数据是共享的也就是线程共享而内存栈中的数据是私有的也就是线程私有的。
10.什么是线程安全 就是数据是否存在逃逸的问题当数据被当做实参或者作为返回的数据此时就出现了逃逸的问题它就是线程不安全的反之就是线程安全的。
11.类的加载流程有了解过吗 主要就是七个步骤 加载加载class字节码文件。 校验校验符号引用是否合理。 准备为静态变量分配空间赋初值此时初始化还未完成。 解析将符号引用替换为指令的真实地址。主要就是基于运行时常量池实现的。 初始化完成静态初始化的赋值及其他相关的方法包括。 使用执行new和其他的类方法。 销毁将类实例进行销毁。
12.Java里面实现自定义注解的方式是什么 主要就是通过interface来实现的其中属性就是在后续使用自定义注解的时候可以可以设置的参数。
13.谈谈你对Java反射的理解 在创建实例的时候不再需要使用new但是呢反射严重的破坏了封装性它可以无视属性作用范围。 获取的反射的方法主要就是类.class实例.getclass()方法class.getName()方法包装类.TYPEclassLoader.loadClass方法。
14.线程池的实现方式有了解过吗 主要就是7个参数核心线程数总线程数救急线程的存活时间存活时间的单位阻塞队列拒绝策略线程工厂。 在阻塞队列中主要就是ArrayBlockQueue和LinkedBlockQueue。 拒绝策略报错策略使用主线程执行策略删除队列中存在时间最长的任务策略丢弃任务策略。
15.Java的流有哪些 字节流字符流。缓冲流对象流数据流。
16.Redis为什么会这么快 redis的操作都是单线程的速度很快。 因为redis是单线程的缘故所以不需要考虑上下文资源切换的问题。 Redis的底层主要就是基于 多路io复用模型来实现的。