@yexiaoqi
2018-12-03T12:14:19.000000Z
字数 2993
阅读 1072
java面试总结
java类中普通成员函数就是虚函数。网上有一个对比
| C++ | Java |
|---|---|
| 虚函数 | 普通函数 |
| 纯虚函数 | 抽象函数 |
| 抽象类 | 抽象类 |
| 虚基类 | 接口 |
双亲委派模式的工作原理的是;如果一个类加载器收到了类加载请求,它并不会自己先去加载,而是把这个请求委托给父类的加载器去执行,如果父类加载器还存在其父类加载器,则进一步向上委托,依次递归,请求最终将到达顶层的启动类加载器,如果父类加载器可以完成类加载任务,就成功返回,倘若父类加载器无法完成此加载任务,子加载器才会尝试自己去加载,这就是双亲委派模式。
当一个Hello.class这样的文件要被加载时。不考虑我们自定义类加载器,首先会在AppClassLoader中检查是否加载过,如果有那就无需再加载了。如果没有,那么会拿到父加载器,然后调用父加载器的loadClass方法。父类中同理会先检查自己是否已经加载过,如果没有再往上。注意这个过程,知道到达Bootstrap classLoader之前,都是没有哪个加载器自己选择加载的。如果父加载器无法加载,会下沉到子加载器去加载,一直到最底层,如果没有任何加载器能加载,就会抛出ClassNotFoundException。
采用双亲委派模式的是好处是Java类随着它的类加载器一起具备了一种带有优先级的层次关系,通过这种层级关可以避免类的重复加载,当父亲已经加载了该类时,就没有必要子ClassLoader再加载一次。其次是考虑到安全因素,java核心api中定义类型不会被随意替换,假设通过网络传递一个名为java.lang.Integer的类,通过双亲委托模式传递到启动类加载器,而启动类加载器在核心Java API发现这个名字的类,发现该类已被加载,并不会重新加载网络传递的过来的java.lang.Integer,而直接返回已加载过的Integer.class,这样便可以防止核心API库被随意篡改。
方法一:在for循环中使用entry实现Map的遍历
//最常见也是大多数情况下用的最多的,一般在键值对都需要使用Map <String,String>map = new HashMap<String,String>();map.put("熊大", "棕色");map.put("熊二", "黄色");for(Map.Entry<String, String> entry : map.entrySet()){String mapKey = entry.getKey();String mapValue = entry.getValue();System.out.println(mapKey+":"+mapValue);}
方法二:在for循环中遍历key或者values,一般适用于只需要map中的key或者value时使用,性能上比entrySet较好;
Map <String,String>map = new HashMap<String,String>();map.put("熊大", "棕色");map.put("熊二", "黄色");//keyfor(String key : map.keySet()){System.out.println(key);}//valuefor(String value : map.values()){System.out.println(value);}
方法三:通过Iterator遍历
Iterator<Entry<String, String>> entries = map.entrySet().iterator();while(entries.hasNext()){Entry<String, String> entry = entries.next();String key = entry.getKey();String value = entry.getValue();System.out.println(key+":"+value);}
方法四:通过键找值遍历,这种方式的效率比较低
for(String key : map.keySet()){String value = map.get(key);System.out.println(key+":"+value);}
类从类加载到JVM中开始,到卸载为止,整个生命周期包括:加载、验证、准备、解析、初始化、使用、卸载七个阶段。
其中类加载过程包括加载、验证、准备、解析、初始化。5个阶段
JVM有三种类加载器:BootstrapLoader负责加载系统类,ExtClassLoader负责加载扩展类,AppClassLoader负责加载应用类。他们主要是分工不一样,各自负责不同的区域,另外也是为了实现委托模型。什么是委托模型呢,其实就是当类加载器有加载需求的时候,先请示他的父类使用父类的搜索路径来加入,如果没有找到的话,才使用自己的搜索路径来来搜索类。
下面的图形可以表示三者之间的关系:
BootstrapLoader <---(Extends)----AppClassLoader <---(Extends)----ExtClassLoader
JVM内存模型主要由堆,方法区,程序计数器,虚拟机栈和本地方法栈组成。其中,堆和方法区是所有线程共有的,而虚拟机栈,本地方法栈和程序计数器则是线程私有的。
/*** 编写程序,输入一个字符,判断它是否为小写字符,如果是,将它转换为大写字母,否则不转换*/public class homework {public static void main(String[] args) {//小写字母的的ascii值为 97-122//大写字母的ascii值为 65-90System.out.print("请输入一个字母:");Scanner input = new Scanner(System.in);char c = input.next().charAt(0);if (c>=97 && c<=122){ //判断是不是小写字母System.out.println("该字母是小写字母");c = (char)(c-32); //如果是小写字母,将其转换为大写字母System.out.println("转换之后的大写字母是:"+c);}else{System.out.println("该子母不是小写字母!");}}}