Java面试总结(2018 - 12 - 10)
1.缓存穿透,缓存雪崩,缓存击穿解决方案分析
1)缓存穿透就是比如拿Redis数据库来说,它是一个键值对,有可能用户查询某个数据的时候没有查到这个key,就跑到数据库中去查了,我们设置缓存的目的就是为了不让请求去数据库中查,但是穿透呢,没查到key会自动去数据库中去查了。这种透过缓存去数据库中查就叫缓存穿透。一般我们把那些空的也存到缓存中,这样就不会穿透缓存了,一般发生的也不是特别多。
2)缓存雪崩就是在高并发的情况下,因为在缓存中有缓存的时间,同一时间下有好多数据失效了,这样在高并发访问量比较大的时候呢,同时又去数据库查,这样数据库的压力就很大,有可能就挂掉了,这就是缓存雪崩。一般我们给实效时间设置点随机数,不让它一起失效。
3)缓存击穿就是指某一个数据失效的话,这个数据访问的也比较多,比如说淘宝在0点的时候某个数据实效了,这个时候又有很多用户提前预购了,在0点的时候去抢它,这个时候就会造成击穿的效果。一般这种击穿很难处理,因为你也不知道这个数据会不会用到,所以我也不是太清楚怎么处理。
2.缓存数据和数据库数据一致性问题
1)用MQ模式的消息队列,保证串行。
2)或者用Id先查询数据,保证每次操作数据是同一个,然后再设置队列。如果分布式的话就再加个服务Id,保证同一个服务。
3.设计模式
Java中有23种设计模式,常见的设计模式有:
1)单列模式
一句话总结:一个类在Java虚拟机中只有一个对象,并提供全局访问点。
生活中例子:太阳,月亮,国家主席等。
解决什么问题:对象的唯一性,性能浪费太多。
项目里面怎么用:数据库连接对象,属性配置文件的读取对象。
模式结构:分为懒汉模式和饿汉模式(如果考虑性能问题的话,就使用懒汉模式,因为懒汉模式是在方法里面进行初始化的,构造器私有化,对外提供方法加同步关键字)。
框架里面的使用:Struts1的Action。
Jdk里面使用:java.lang.Runtime#getRuntimejava.awt.Desktop#getDesktop。
懒汉模式:就是说比较懒,在用到它的时候它才去创建对象(懒汉模式线程不安全,懒汉加载慢,但是在内存作用域短,节省空间)。
/* * 懒汉模式 * 懒汉线程不安全 * 懒汉加载慢,但是在内存作用域短,节省空间 */ public class SingletonA { //SinglettonA的唯一实例 private static SingletonA instance = null; //将构造函数私有,防止外界构造SingletonA实例 private SingletonA() { } //获取SingletonA的唯一实例,用synchronized关键字保证某一时刻只有一个线程调用此方法 public static synchronized SingletonA getInstance(){ //如果instance为空,便创建一个新的SingletonA实例,否则,返回已有的实例 if(instance == null){ instance = new SingletonA(); } return instance; } public void print(){ System.out.println("我是懒汉模式!"); } public static void main(String[] args) { SingletonA s1 = SingletonA.getInstance(); SingletonA s2 = SingletonA.getInstance(); System.out.println(s1 == s2); } }
饿汉模式:就是说比较饿,在用到它的时候它早早的就加载完对象了(饿汉模式线程安全,饿汉加载快,但是在内存中作用域长,不节省空间)。
/* * 饿汉模式 * 饿汉线程安全 * 饿汉加载块,但是在内存中作用域长, 不节省空间 */ public class SingletonB { //SingletonB的唯一实例 private static SingletonB instance = new SingletonB(); //将构造函数私有,防止外界构造SingletonB实例 private SingletonB() { } //获取SingletonB的实例 public static SingletonB getInstance() { return instance; } public void print() { System.out.println("我是饿汉模式"); } public static void main(String[] args) { SingletonB s1 = SingletonB.getInstance(); SingletonB s2 = SingletonB.getInstance(); System.out.println(s1 == s2); } }
2)Factory(简单的工厂模式)
一句话总结:用一个方法来代替new关键字。
生活中的例子:制衣厂,面包厂等生产厂。
解决什么问题:对象产生过多,或者经常有子类替换生成。
项目里面怎么用:对于经常生成的对象,或者父子类替换的对象。
模式结构:写一个对外声明的方法,方法里面使用new关键字代替。
框架里面使用:Spring的核心就是工厂模式。
Jdk里面使用:newInstance。
工厂模式代码:
public class UserFactory { public static User createUser(int i){ //如果输入的是1,就创建它的子类,否则就创建父类 if(i == 1){ return new Alices(); } return new User(); } }
3)Proxy(代理模式)
一句话总结:为其他对象提供一个代理,以控制对当前对象的访问。
生活中的例子:房屋中介,婚姻介绍所。
解决什么问题:不能直接访问该对象,或者太大的资源耗费多。
项目里面怎么用:权限,或者大对象的访问权限。
模式结构:代理类和被代理类实现同一个接口,用户访问的时候先访问代理对象,然后让代理对象去访问被代理对象。
框架里面使用:Spring里面的AOP实现。
Jdk里面使用:java.lang.reflect.Proxy。
代理模式代码:
创建一个接口:
public interface SellHouse { void sell(double money); }
创建一个被代理类:
public class Hoster implements SellHouse { @Override public void sell(double money) { System.out.println("祝你居住愉快"); } }
创建一个代理类:
public class Medium implements SellHouse { SellHouse hoster=new Hoster(); @Override public void sell(double money) { if(money>=1000){ hoster.sell(money); }else{ System.out.println("你的价格太低了"); } } }
创建一个测试类:
public class Renter { public static void main(String[] args) { SellHouse renter=new Medium(); renter.sell(500); } }