JAVA面经整理(10)

news2025/1/14 18:08:01
一)MyBatis有什么优缺点?

Mybatis是⼀种典型的半自动化的ORM 框架,所谓的半自动,因为还需要⼿动的写 SQL 语句在XML文件里面再由框架根据SQL以及传入数据来进行组装成要执行的SQL,所谓的ORM框架,就是对象关系映射框架,就是让JAVA程序中的类里面的属性直接映射到数据库中的表里面的列,在Java中操作这个类的时候,就相当于直接操作数据库中的表和字段;

1)更灵活:支持自定义SQL映射,结果集映射,来满足不同的需求

2)更好的性能控制:SQL都是自己写的,可以加索引,可以进行SQL优化,不像Hibernate那样SQL是由程序自动生成的,Hibernate是自己调用API生成SQL语句,万一性能不好没有办法

3)可读性很好:将SQL查询和映射配置分离到XML文件中,使得代码更有可读性,便于维护和调试

4)二级缓存:MyBatis支持二级缓存,可以缓存查询结果提升查询性能,减少数据库访问次数

5)动态SQL:MyBatis会根据条件自动生成SQL语句

6)支持多种数据库:MYBATIS支持多种数据库,可以灵活的切换数据库供应商

缺点:成也萧何,败也萧何

1)繁杂的XML配置:处理大量SQL查询的时候比较麻烦;

2)手动写SQL:虽然手动编写SQL可以更好地进行性能控制,但是也需要更多的时间和精力来手动维护和优化SQL,如果有一天更换数据库,SQL都是需要重写的,移植性很差;

Hibernate是⼀种典型的全⾃动 ORM 框架,所谓的全⾃动,是 SQL 语句都不⽤在编写,基于框架的 API,可以将对象⾃动的组装为要执⾏的 SQL 语句,简单的SQL连SQL语句都不用编写,底层会提供一个Insert方法,只需要将对象的属性设置进去将这个对象放到方法的参数里面,就可以进行使用了最终调用save方法

二)MyBatis和Hibernate和Spring Data JPA有什么区别?

1)MYBatis是一个典型的半自动ORM框架,Hibernate和SpringDataJPA是全自动的ORM框架

2)MyBatis自己写比较灵活,但是MyBatis不能反向的生成表结构,但是Hibernate和Spring Data JPA对于SQL的要求没有那么高,只需要熟练使用API即可,他们俩都可以生成表结构

3)Mybatis在国内用的比较多,剩下这俩在国外用的比较多

4)Spring Data JPA底层是基于Hibernate来实现的,SDJ是一个SPI;

5)对于MyBatis来说,更换数据库的重新写SQL代价很大,但是对于剩下那两个不需要更换API

1)全⾃动 ORM 框架,调用固定的方法,⾃动的组装为 SQL 语句,单表查询比较简单;

2)可以跨数据库,底层框架提供了多套主流数据库的 SQL ⽣成规则,数据库从MYSQL转变成SQLServer基本上不需要改一行代码,因为你的是HQL语法,是中间语法,相当于是门面模式;

Hibernate的缺点:

1)学习⻔槛更⾼,要学习框架 API 与 SQL 之间的转换关系,况且如果实现一个简单的增删改查是非常简单的,但是说如果写一些复杂的SQL调用API就会很麻烦,Hibernate提供了一些自己复杂的语法进行复杂SQL查询,提供的HQL语法是一个中间语法,况且最终生成的SQL也不知道,Hibernate提供一些特殊的API来实现更复杂的SQL语句,使用精通非常难;

2)对数据库模型依赖⾮常⼤,在软件需求变更频繁的系统中,会导致⾮常难以调整及维护。可能数据库中随便改⼀个表或字段的定义,Java代码中要修改⼏⼗处

3)很难定位问题,也很难进⾏性能优化,需要精通框架,对数据库模型设计也⾮常熟悉

三)数据库属性名和类中的字段名冲突咋解决?

在MyBatis中,属性名和字段名如果不一致的话会出现查询和其他操作的时候是NULL,如果进行映射的话,那么查询和映射都不会出现为NULL的情况

1)更改属性或者是字段名让二者保持一致

2)查询的时候使用as重映射字段名,能够在查询的时候也能够找到相应的值

3)使用结果集映射,使用resultMap来映射相应的字段,要映射全部映射,否则多表联查会出现NULL值

4)使用MyBatis-plus中的@TableField字段来进行映射,再类中属性上面加上@TableField("数据库中的属性名")

四)MyBatis是如何实现对象映射?

1)MyBatis底层是基于JDBC实现的,返回查询的结果集

2)每一个查询结果集就是一个ResultSet,循环ResultSet将行数据赋值给返回对象

3)每一行数值赋值的流程如下:

3.1)先进行判断返回类型是否是ResultMap,如果是,那么就以ResultMap中设置的映射关系为主进行赋值

3.2)如果没有设置ResultMap那么就使用数据库的字段名和表中的字段名进行对比并完成赋值

4)遍历完成ResultMap将结果返回给用户

五)MyBatis的执行流程:

1)加载配置文件:MyBatis的执行流程首先从加载配置文件开始,通常MyBatis的配置文件是一个XML文件,其中包含了数据源配置,SQL映射配置和连接池配置等信息;

2)构建sqlSessionFactory对象:在配置文件进行加载以后,MyBatis使用配置信息来构建sqlSessionFactory对象,这是MyBatis的核心工厂类,SqlSessionFactory是线程安全的,他是用来创建sqlSession对象;

3)创建sqlSession对象:应用程序会通过SqlSeessionFactory来创建SqlSession对象,SqlSession代表一次数据库会话,它提供了SQL执行的方法,通常情况下,每一个线程都是有着自己的SqlSession对象;

4)执行SQL查询:在SqlSession中,开发人员可以执行SQL查询,可以通过两种方式来实现

4.1)使用注解来加SQL:MyBatis使用了注解+SQL的执行方式,MyBatis会为Mapper接口,生成代理对象,实际来执行SQL查询;

4.2)使用XML映射文件:开发人员可以通过XML来自定义SQL查询语句和映射关系,然后通过sqlSession来执行SQL查询,将结果集映射到JAVA对象上;

5)SQL解析和执行:MYBatis会解析SQL执行,进行SQL查询,并获取到执行结果,如果是select操作,就返回一个ResultSet,如果是DML操作,就返回一个数字

6)结果集映射:MyBatis将使用配置的结果映射规则,将查询结果映射成JAVA对象上面,将数据库的列映射到JAVA对象的属性上面,并来处理关联关系等等

7)返回结果:查询结果返回给应用程序,开发人员可以针对于接过来做进一步处理

8)sqlSession:关闭SqlSession释放资源

六)#和$有什么区别?

#和$都是MyBatis用来替换参数的,但是两者的区别主要体现在以下方面:

1)含义不同:#{}是预处理,$是直接替换

2)使用场景不同:普通参数使用#{}来进行处理,程序会认为你传递的参数都是一个值,会自动加双引号,但是$如果传递的是SQL命令或者是SQL中的关键字,但是需要做好安全验证

3)安全性不同

SQL注入:SQL注入指的是应用程序对于用户的输入的合法性数据没有判断或者说是过滤不严,攻击者可以在应用程序中实现定义好的查询语句的末尾加上额外的SQL语句,在管理员不知情的条件下进行非法的查询操作,以此来欺骗数据库服务器执行的非授权查询,从而进一步来得到用户的数据信息;

由此可以看出以上程序在应用程序不知情的情况下进行非法操作,以此来实现欺骗数据库服务器执行非授权的敏感数据的查询操作;

七)MyBatis是如何实现分页的?

MyBatis分页实现有两种方式

1)物理分页:物理分页是通过SQL查询语句,在数据库引擎层实现的,比如说使用MYSQL的limit语法实现分页

1.1)通过XML中直接拼接SQL来实现分页,传入limit和offset

1.2)通过pageHelper来实现物理分页

注意:这里面的数据库的属性名和表中的字段名必须要相同

	<dependency>
			<groupId>com.github.pagehelper</groupId>
			<artifactId>pagehelper</artifactId>
			<version>5.1.2</version>
		</dependency>
#pagehelper分页插件配置
#
pagehelper:
  helperDialect: mysql
  reasonable: true
  supportMethodsArguments: true
  params: count=countSql

2)逻辑分页:逻辑分页是在应用程序方面实现的分页,通常是先查询出所有符合条件的数据,然后在内存中,然后在内存中针对于数据进行查询操作,在内存效率很低;

2.1)注入SqlSession对象

2.2)新创建RowBounds对象,里面指定分页的规则

2.3)调用SqlSession对象的selectList方法,里面传入三个参数,mapper中的方法名对应的路径,第二个参数是传递的参数,第三个参数是rowBounds

在MyBatis中,为什么可以将Mapper接口直接进行注入使用?

为什么只在UserService里面写了一个@Resource就可以把标记了@Mapper的接口直接注入使用?

表面上是一个接口,但实际上咱们的程序时不会进行new这个接口的,而是使用代理的方式来进行实现的(调用动态代理子类)

八)RowBounds是一次性查询所有数据吗?

RowBounds表面上是在所有数据中进行检索数据,但是其实本质上并非是一次性查询出所有数据,因为MyBatis是基于对JDBC的封装,在JDBC驱动中有一个fetch-Size中的属性,规定了每一次最多从数据库中拉取多少数据,在select标签中可以指定fetchSize属性

Synchronized常见面试题:

1)synchronized在JVM从JDK1.6是有锁升级的一个过程的,况且是靠监视器锁来实现的,在方法进入之前加上一个moniterenter,离开监视器锁是moniterexit,监视器就类似于是一个房间,只能由同一个线程进去,无锁,偏向锁,轻量级锁,重量级锁

2)锁在在对象头里面有一个标识,用来标识当前有没有锁,是属于哪一个线程里面的,当有线程尝试获取这把锁的时候,会进行判断线程ID和我们对象头中的线程ID是不是一致的,如果是一致的,那么代表我是可以重入的,你已经拥有这把锁了,我就可以重新获取到这把锁,说明我是可重入的;

3)如果ID不同,当前锁已经被其他线程占有了,就会一直尝试获取锁使用自旋锁,这是自适应的;

九)什么是动态SQL

根据查询条件的不同来动态的生成最终查询的SQL,是一种动态构建查询查询语句的机制,允许在SQL查询中根据不同的条件来生成不同的SQL语句来满足查询的需求,动态SQL里面包含条件判断,循环,参数替换等功能,使得SQL查询更加的具有灵活性和可重用性

填写条件,查询的是特定条件的值,如果不填写条件,那么查询的是所有的值

1)if标签,有一个test属性,就是类似于JAVA中的if判断

2)trim标签,prefix,suffix,prefixOverrides,suffixOverrides

3)where标签,自动加where条件,并且会自动去除最前面的and

4)set标签,自动加上set条件,并且会自动去除最前面的逗号

5)forEach标签:循环迭代集合,collection,seperator,open,close

十)MyBatis的二级缓存:不适用于多写少读的问题,多写经常更新缓存的问题

MyBatis的缓存机制属于本地缓存,适用于单机系统,他的目的就是为了减少数据库的查询次数,从而来提升系统的查询性能;

MyBatis中一共包含着两级缓存:

1)SqlSession级别的,是MyBatis自带的缓存功能,默认情况下是开启的,并且无法进行关闭,但是当两个SqlSession访问同一条SQL语句的时候,,一级缓存也不会生效,需要查询两次数据库;

2)二级缓存是Mapper级别的,只要是同一个Mapper,无论使用多少个SqlSession来进行操作,数据都是共享的,也就是说多个不同的SqlSession可以共用二级缓存,MyBatis的二级缓存默认是关闭的,需要的时候需要手动开启,二级缓存也可以使用第三方的缓存,比如说使用Ehance作为二级缓存;

1)一级缓存是SqlSession级别的缓存,它的作用域就是同一个SqlSession,同一个SqlSession的多次查询会共享同一个缓存,二级缓存是Mapper级别的缓存,它的作用域就是同一个Mapper,同一个Mapper的多次查询会共享同一个缓存;

2)一级缓存是不需要手动开启的,不需要手动配置,二级缓存需要手动配置,需要在Mapper.xml文件中进行手动配置

3)一级缓存的生命周期是和SqlSession一样长的,当SqlSession关闭的时候,一级缓存也会被清空,二级缓存的生命周期适合MapperFactory是一样长的,当应用程序关闭的时候,二级缓存也会被清空;

4)一级缓存只能用于同一个SqlSession的多次查询,不能用于跨SqlSession的查询,二级缓存可以用于跨Sql之间的查询,多个SqlSession可以共享同一个二级缓存

5)一级缓存是线程私有的,不同的SqlSession之间的缓存数据不会受到干扰,二级缓存是线程之间共享的,多个SqlSession可以共享同一个二级缓存,就需要考虑到线程安全问题

总结:MyBatis的缓存机制属于本地缓存,适用于单机系统,他的目的就是为了减少数据库的查询次数,来提升系统性能,MyBatis本地缓存一共分成两类,一级缓存SqlSession级别,默认开启不能关闭,二级缓存Mapper级别,默认关闭,可以在XML配置中手动开启useCatche=true;

1)在mapper.xml添加catche标签

2)在需要使用查询数据的select标签上添加useCache等于true

也就是说多个用户,也就是多个sqlSession中只要有一个sqlSession拿到了缓存,就会把缓存存放到二级缓存中,那么以后其他的sqlSession就可以直接从二级缓存中来获取到数据

有了二级缓存,先查询二级缓存,再来查询一级缓存,最后再来查询数据库

十一)为什么MyBatis不默认开启二级缓存?

1)内存占用问题:开启二级缓存以后,缓存的数据可能占用一定的内存空间,如果没有合适的策略来管理缓存,那么很有可能造成内存占用过多的问题

2)并发性问题:在多线程环境下,开启二级缓存可能会引起并发性问题,如果多个线程或者是用户同时更新了数据,而没有及时的更新和刷新缓存,可能会造成内存占用过多的问题

3)复杂性和配置问题:二级缓存的配置需要考虑很多问题,包括缓存的刷新策略,缓存的清理策略,这就增加了配置的复杂性和可能引入配置错误的风险

十二)MyBatis有哪些缓存清除策略?

加上</cache>标签以后

1)映射语句文件中的所有的select查询操作都会被缓存

2)映射语句文件中的所有insert,delete和update会刷新缓存\

3)缓存会默认使用最近最少使用算法LRU算法来清除不需要的缓存

4)缓存会不定时进行刷新,没有刷新间隔,缓存默认会保存1024个引用

这个更高级的配置了一个FIFO缓存,每隔60S刷新,最多只能存储结果对象的或者是列表的512个引用,并且返回的对象默认是只读的,因此针对于他们进行修改可能在不同线程的调用者产生冲突;

1)LRU:最近最少使用算法,移除最长时间不使用的对象

2)FIFO:先进先出,按照对象进入缓存的顺序来清除他们

3)SOFT:软引用

4)WEAK:虚引用

十三)MyBatis中使用到了那些设计模式?

1)单例模式:异常的输出类ErrorContext,错误的上下文对象,用于打印线程执行的错误信息

2)工厂模式:工厂模式在MyBatis的典型设计模式是SqlSession,可以通过该接口来执行SQL命令,SqlSessionFactory是用来产生sqlSession对象的;

3)装饰器模式:

比如说我现在买了一个机器人,他有三个功能对话,唱歌,放音乐,但是有一天我希望这个机器人能够扫地,跳舞,打游戏,希望它有更多的功能,我于是开始打电话给厂商,这个厂商就直接回答说,现在这个机器人设计的时候就只有这三个功能,当然可以扩展新的功能,这个厂商承诺开始研制一个新的产品,他说会在第一代的产品中进行扩展,最终厂家针对于第一代产品的升级修改最终研制出了第二代产品,就好像下面这样他拥有了两条胳膊两条腿,他就可以跳舞和打游戏了,最终厂家邮寄了一个给我们,最后厂家开始销售第二代产品,这个厂家的所有客户拿到了新的产品以后,他就有了拖地和跳舞的功能;

但是现在李四是一个手工狂人,他觉得根本没有必要等待第二代产品,他就可以对第一代产品进行升级改造

这个时候张三只是在原来的机器人上面加了一个箱子,然后自己加上胳膊和大腿,自己就实现了跳舞和扫地的功能,她是自己扩展原有的功能,而不是需要等待厂家重新研发设计,他就可以瞬间将这个机器人进行改造,总而言之上面这两种方式可以实现给一个类或者是对象添加新的功能,第一种方式称之为是继承模式,继承机制就是在一个现有类的基础上,在子类进行扩展功能,第二种方式称之为是关联机制,把一个对象嵌入到另一个对象中,相当于是给它套一个壳子,来扩展功能,那么这个壳子就是装饰器,这种方式就称之为是装饰器模式

4)适配器模式:

电源适配器:将交流电变成直流电,中国的电源用的是交流电,而手机的使用的电流是直流电,那么需要使用适配器来整流,滤波和稳压转化成所需要的直流电,从而来给手机进行充电,就是手机本来是没有办法使用交流电的,那么就通过电源适配器,也就是手机的插头,通过充电线直接连到手机上,那么电源适配器就是一个桥梁,将两个不兼容的东西兼容到一起,就是将一个类的接口变成另一个客户端所期望的一种接口,从而使得原本接口不匹配的而无法工作在一起的两个类进行相互工作

Spring里面的适配器模式用在DispatcherServlet前端控制器里面有一个doDispatcher方法,在调度方法里面会根据用户输入的url地址来进行匹配相应的Handler方法,方法是被HandlerAdpter控制适配器来执行的,MyBatis中的适配器的是logging;

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

burpsuite进行拦截请求包

当我们进行正常访问浏览器时&#xff0c;进行勾选住如上内容&#xff0c;若我们需要进行拦截某个请求时&#xff0c;首先将如上勾选去掉&#xff0c;然后打开proxy下的Intercept&#xff0c;确保intercept is on 按钮打开。 然后打开intercept is on 按钮即可进行拦截请求

温故知新-Redis01

目录 开场白-追命3连 使用场景 01缓存穿透场景与方案 02布隆过滤器 03缓存击穿场景与方案 04缓存雪崩场景与方案 开场白-追命3连 看你项目中有说用到Redis&#xff0c;都是哪些场景使用了Redis呢&#xff1f; 如果发生了缓存穿透、击穿、雪崩如何应对呢&#xff1f;缓存…

LangChain+LLM实战---ChatGPT的即时插件套件制作

英文原文&#xff1a;Instant Plugins for ChatGPT: Introducing the Wolfram ChatGPT Plugin Kit 在一分钟内构建一个新插件 几周前&#xff0c;我们与OpenAI合作发布了Wolfram插件&#xff0c;使ChatGPT可以使用Wolfram语言和Wolfram|Alpha作为工具&#xff0c;在ChatGPT内部…

Day 6 登录页以及路由(四)Vue页面处理

系列文章目录 本系列记录一下通过Abp搭建后端&#xff0c;VueElement UI Plus搭建前端&#xff0c;实现一个小型项目的过程。 Day 1 Vue 页面框架Day 2 Abp框架下&#xff0c;MySQL数据迁移时&#xff0c;添加表和字段注释Day 3 登录页以及路由 (一&#xff09;Day 4 登录页及…

我终于学会的前端技能——代码调试、打断点

在技术的世界里&#xff0c;要用魔法来打败魔法 说来惭愧我做前端已近三年了竟然还没有学会如何调试代码&#xff0c;也就是给自己的代码打上断点一步步看它的运行状态以达到理清代码运行逻辑、排查问题提升开发效率的目的。直到最近我才学会了这一技能&#xff0c;在这之前我用…

抖音小店参与双十一活动:销售增长的策略与实施

双十一是中国最大的购物狂欢节&#xff0c;对于抖音小店商家来说&#xff0c;参与双十一活动是一个极好的机会&#xff0c;可以在这个繁忙的购物季节中大幅提升销售。下面四川不若与众将介绍一些抖音小店商家如何参与双十一活动的策略和注意事项。 1. 提前准备&#xff1a;在双…

Nginx编译安装和配置

官网:http://nginx.org/ 这里以1.20.2为例 Nginx是C语言写的 如果Linux系统上没有安装C编译环境 先执行下面命令 yum install -y gcc automake autoconf libtool make 如果不确定 可以使用命令查看 命令格式 rpm -q xxx 例如 说明有C编译环境 安装前需要安装4个依赖包…

任正非说:为更好地服务客户,我们把指挥所建到听得到炮声的地方

你好&#xff01;这是华研荟【任正非说】系列的第29篇文章&#xff0c;让我们聆听任正非先生的真知灼见&#xff0c;学习华为的管理思想和管理理念。 一、产品发展的路标是客户需求导向&#xff0c;企业管理的目标是流程化的组织建设&#xff0c;这两句话归纳了我形容的龙的组织…

海康Visionmaster-全局变量:全局变量关联流程中具体 模块结果的方法

将视觉流程中模板匹配算法模块运行的结果数据&#xff1a;特征匹配点 X 关联全局变量 MatchResultX。 在流程运行的主界面中&#xff0c;按照下面 1&#xff0c;2&#xff0c;3&#xff0c;4 步骤操作&#xff0c;第一步选中算法模块&#xff0c;第二步择模块结果 Tab 页&#…

【NeurIPS 2020】基于蒙特卡罗树搜索的黑箱优化学习搜索空间划分

Learning Search Space Partition for Black-box Optimization using Monte Carlo Tree Search 目标&#xff1a;从采样&#xff08;Dt ∩ ΩA&#xff09;中学习一个边界&#xff0c;从而最大化两方的差异 先使用Kmeans在特征向量上&#xff08; [x, f(x)] &#xff09;聚类…

Python算法例4 求平方根

1. 问题描述 实现int sqrt&#xff08;int x&#xff09;函数&#xff0c;计算并返回x的平方根。 2. 问题示例 sqrt&#xff08;3&#xff09;1&#xff1b;sqrt&#xff08;4&#xff09;2&#xff1b;sqrt&#xff08;5&#xff09;2&#xff1b;sqrt&#xff08;17&#…

GEE数据集——原住民土地(原住民土地地图)数据集

原住民土地&#xff08;原住民土地地图&#xff09; 土地承认是人们在日常生活中融入原住民存在和土地权利意识的一种方式。这通常在仪式、讲座或教育指南开始时进行。它可以是一种明确但有限的方式来认识殖民主义和第一民族的历史以及定居者殖民社会变革的需要。在这种情况下…

夜间灯光数据VIIRS Nighttime Day: Night Band Composites Version 1数据集

简介&#xff1a; 新一代对地观测卫星Suomi NPP,搭载的可见光红外成像辐射仪&#xff08;Visible Infrared Imaging Radiometer Suit,VIIRS&#xff09;&#xff0c;能够获取新的夜间灯光遥感影像&#xff08;Day/Night Band,DNB波段&#xff09;。VIIRS_VCMCFG夜光遥感数据的…

航模模拟器训练

准备物品 航模遥控器 aux线 即两端都是耳机插头的线 解密狗 电脑 phoenixRC 航模模拟软件(【飞舜极创】凤凰5.0飞行模拟器 安装和设置方法_哔哩哔哩_bilibili) 操控 美国手(俗称左手油门)——左手油门和方向舵&#xff0c;右手升降和副翼 美国手 左摇杆&#xff1a; 上下…

【qemu逃逸】HITB2017-babyqemu 2019数字经济-qemu

前言 由于本地环境问题&#xff0c;babyqemu 环境都没有起起&#xff0c;这里仅仅做记录&#xff0c;exp 可能不正确。 HITB2017-babyqemu 设备逆向 设备定位啥的就不说了&#xff0c;先看下实例结构体&#xff1a; 其中 dma_state 结构体如下&#xff1a; 这里看字段猜测…

【算法|二分查找No.3】leetcode 35. 搜索插入位置

个人主页&#xff1a;兜里有颗棉花糖 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 兜里有颗棉花糖 原创 收录于专栏【手撕算法系列专栏】【LeetCode】 &#x1f354;本专栏旨在提高自己算法能力的同时&#xff0c;记录一下自己的学习过程&#xff0c;希望…

网络运维Day01

文章目录 环境准备OSI七层参考模型什么是协议&#xff1f;协议数据单元(PDU)设备与层的对应关系什么是IP地址&#xff1f;IP地址分类IP的网络位和主机位IP地址默认网络位与主机位子网掩码默认子网掩码查看IP地址安装CISCO汉化CISCO(可选操作) CISCO之PC机器验证通信 CISCSO之交…

台球厅桌球室计时计算软件计费方法,台球厅的电脑怎么计时

台球厅桌球室计时计算软件计费方法&#xff0c;台球厅的电脑怎么计时 今天给大家分享的是 佳易王桌球计时计费软件V18.0版本&#xff0c;只需点开始计时即可&#xff0c;结账的时候&#xff0c;软件自动计算金额。 灯控为可选&#xff0c;点开始计时&#xff0c;相应的桌灯亮…

【算法|二分查找No.4】leetcode 852. 山脉数组的峰顶索引

个人主页&#xff1a;兜里有颗棉花糖 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 兜里有颗棉花糖 原创 收录于专栏【手撕算法系列专栏】【LeetCode】 &#x1f354;本专栏旨在提高自己算法能力的同时&#xff0c;记录一下自己的学习过程&#xff0c;希望…

一文带你掌握多继承,菱形继承以及虚拟继承

&#x1f388;个人主页:&#x1f388; :✨✨✨初阶牛✨✨✨ &#x1f43b;强烈推荐优质专栏: &#x1f354;&#x1f35f;&#x1f32f;C的世界(持续更新中) &#x1f43b;推荐专栏1: &#x1f354;&#x1f35f;&#x1f32f;C语言初阶 &#x1f43b;推荐专栏2: &#x1f354;…