Java经典面试题整理及答案详解(二)

1.说一下java类集
类集是java实现的数据结构应用,如果只是使用,那么类集的操作非常简单,因为类集的核心接口有List,Set、Map、Iterator、Enumeration;
List子接口:是可以根据索引号取得内容,而在list集合里最容易问到的问题:ArrayList(包装了数组的集合,比较常用,数组是可变的)、LinkedList(链表的实现,搜索数据的时间复杂度为:n) 区别;
Set子接口:排序子类、HashSet与hashCode()和equals的关系;
— Hashset:重复的判断依靠的是hashcode()和equals(),但是它是无序的
— TreeSet:是有序的,依靠的是Comparable排序
— LinkedHashSet:继承了HashSet的特点,但是属于有序(增加顺序为保存顺序)

Map接口:Map.Entry、Iterator输出、HashMap、WeakHashMap(弱引用)。
如果只回答这一个问题,按以上几点说,这道题基本上就可以拿下,但也有可能追问其他的问题,例如:可能要求你现场编写一个链表,或者编写一个二叉树。这时候就要熟悉链表与二叉树的数据结构。
2、字符串哈西相等,equals相等吗?反过来呢?
凡是这类问相不相等的,写个程序测试是比较方便的办法。
范例:首先观察直接赋值
*提到字符串,首先就要想到共享数据词,共享常量词


范例:采用构造法


范例:字符串


总结:从正常道理来讲,如果hashCode()相同,那么equals一定相同。但是反过来equals相同,那么hashCode()也应该相同。

3、Spring的工作原理,控制反转是怎么实现的?自己写过滤器过滤编码怎么实现?
Spring的核心组成:IOC&DI(工厂设计)、AOP(代理设计、动态代理设计);
-Spring之中针对于XML的解析处理采用的是DOM4J的实现;
-Anntation的时候必须要求有一个容器;

对于编码过滤需要考虑两种情况:
-Struts1.x、SpringMVC、JSP+Servlet:都可以以直接通过过滤器完成;
-Struts2.x:必须通过拦截器完成;
-实现:考虑到可扩展性的配置,所以在配置文件里面设置编码,在程序运行的时候动态取得设置的编码进行操作。但是需要设置两个操作:请求编码、回应编码。

4、框架的源码有没有看过?
不要回答没有,即使你真的没有,你也别回答没有看过;
框架的核心思想:反射+XML(Annotation)
-Struts2.X的设计:请求交由过滤器执行,而后过滤器交给控制器完成,后面由于将跳转路径等信息都写在了配置文件或知识Annoration里面,所以还需要进行这部分内容的加载;
-SpringMVC:它是基于方法的请求处理,所有的参数都提交到方法上,本质上还是一个DispatcherServlet;
-Hibernate:就是反射和DOM4J解析处理流程。

5、动态代理是怎么实现的?
直接使用Invocation Handler接口进行实现,同时利用Proxy类设置动态请求对象;
使用CGLIB来避免对于“代理设计模式需要使用接口实现”的限制。
6、action是单实例还是多实例?
Struts 2.x和Spring MVC中的Action都是多实例;
Struts 1.x的Action是单实例;
Struts 2.x和Spring MVC是否单实例可以控制,只要交由Spring管理的Action类,都可以通过“@scope=“prototype””来进行控制。
7、怎么配置bean?
这样的配置主要是在Spring里面,重点只有xml和annotation的扫描负责:

xml中直接使用"bean",这样在Spring容器启动的时候就可以通过容器进行初始化;
annotation必须设置context命名空间,而后进行扫描包的配置。
8、修改单实例多实例.
“@scope=“prototype””来进行控制。

9.Java的设计模式
工厂设计模式、代理设计模式、单例设计模式、合成设计模式、门面设计模式(JDBC)、装饰设计模式、模版设计模式。

10.事务的控制
MySQL数据库如果想要使用事务,必须使用“type=innodb”这个数据引擎;
事务的核心控制:commit、rollback;
在Spring里面,利用AspectJ可以设置AOP的切面,而后进行声明式事务控制。
11. 脏读、幻读、不可重复读
脏读:脏读就是指当一个事务正在访问数据,并且对数据进行了修改,而这种修改还没有提交到数据库中,这时,另外一个事务也访问这个数据,然后使用了这个数据。
幻读:是指当事务不是独立执行时发生的一种现象,例如第一个事务对一个表中的数据进行了修改,这种修改涉及到表中的全部数据行。同时,第二个事务也修改这个表中的数据,这种修改是向表中插入一行新的数据。那么,以后就会发生操作第一个事务的用户发现表中还有没有修改的数据行,就好像发生了幻觉一样。
不可重复读:是指在一个事务内,多次读同一数据。在这个事务还没有结束时,另外一个事务也访问该同一数据。那么,在第一个事务中的两次读数据之间,由于第二个事务的修改,第一个事务两次读到的数据可能是不一样的,这样就发生了在一个事务内两次读到的事务是不一样的,因此成为不可重复读。
12 .事务的传播属性
在Transaction Definition接口中定义了七个事务传播行为: 

PROPAGATION REQUIRED:如果存在一个事务,则支持当前事务,如果没有事务则开启一个新的事务;
PROPAGATION SUPPORTS:如果存在一个事务,支持当前事务,如果没有事务则非事务的执行,但是对于事务同步的事务管理器,PROPAGATION SUPPORTS与不使用事务有少许不同;
PROPAGATION MANDATORY:如果已经存在一个事务,支持当前事务,如果没有一个活动的事务,则抛出异常;
PROPAGATION REQUIRES NEW:总是开启一个新的事务,如果一个事务已经存在,则将这个存在的事务挂起;
PROPAGATION_NOT_SUPPORTED:总是非事务地执行,并挂起任何存在的事务;
PROPAGATION_NEVER:总是非事务地执行,如果存在一个活动事务,则抛出异常;
PROPAGATION NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中,如果没有活动事务,则按Transaction Definition.PROPAGATION_REQUIRED属性执行。
13.购物车是怎么实现的?
购物车的实现基于:

Session:浏览器关闭后消失;
Cookie:数据保存在本地,如果切换到手机上无效;
数据库:可以在不同的终端上持续操作。
你的实现:Ajax处理操作,处理购物车的应用,考虑到用户如果多的情况,并且访问量频繁,要单独设计一个购物车的子系统模块。
14.统计一天的订单量
如果一天的订单预估才1W条,随便你折腾;
绝对不可能使用COUNT()和WHERE;
分时统计,如果系统统计到位,可以单独配置一个文件进行计数(需要考虑同步,一同步就慢),不应该出现在抢购环节中。
15.IN HAVING exsit
IN是判断具体的几项数据:
HAVING是针对于分组后的数据的筛选,依然要使用统计函数处理;
EXISTS:只是需要判断子查询里面是否有数据。

16.有没有用过定时任务?
Java本身提供有定时任务:Time Task、Timer;但是此类操作对于定时很难完成,它只能够做频率,但是这个频率不准,所以在定时开发之中会使用quartz组件,而且spring里面也提供有自己的定时实现,这个实现的好处是可以在准确的时间上进行触发。

17.JVM的内存管理
内存分为:栈(java虚拟机栈)、堆、程序计数器(类似于寄存器)、方法区、本地方法栈。
垃圾处理操作指的是堆内存:年轻代、老年代、永生代(JDK1.8移除)。会直接牵扯到JVM的内存调优问题。
推荐《深入java虚拟机》

18.堆内存、栈内存溢出
栈溢出:栈帧,所有的方法调用都是通过栈帧的形式控制的。栈如果保存的数据过多,那么就会产生内存溢出,如果堆内存中产生的数据量过大,那么就有可能出现“OutOfMemoryError”错误。

19.说一下缓存
缓存的主要目的是提高查询的效率,常见的两种缓存组件:EhCache(用于数据库)、OSCache(用于页面),除了这样的操作之外,还有缓存的数据库:redis、memcached,其中redis可以将数据保存到磁盘上,并且支持的数据类型要多于memcached。这种redis的数据库每秒并发的访问量可以达到15W次。

20.统计所有重名用户
还是需要考虑数据量问题,如果数据量大,考虑使用位图索引,如果数据量小就直接分组统计即可。

你可能感兴趣的:(点滴)