为什么你写的代码有时候和预期不一致

简介: 为什么你写的代码有时候和预期不一致

为什么你写的代码有时候和预期不一致

一.友情链接

目录
视频讲解
文字版

二.前端编译

2.1javac编译器

即一般所说的编译,从javac代码的总体结构来看,编译过程大致可以分为1个准备过程和3个处理过程,它们分为
1)准备过程:初始化插入式注解处理器
2)解析与填充符号表过程,构造抽象语法树
3)插入式注解处理器的注解处理过程
4)分析与字节码生成过程:语法检查->控制流分析->解析语法糖->字节码生成

语法糖主要有以下几种:泛型、自动拆装箱遍历循环、条件编译
类型擦除(编译前)

public static void main(String[] args) {
    Map<String,String> map = new HashMap<String,String>(16);
    map.put("name","csx-mg");
    String name = map.get("name");
    System.out.println(name);
}

类型擦除(编译后)

public static void main(String[] args) {
    //类型擦除
    Map map = new HashMap();
    map.put("name", "csx-mg");
    //强制转换
    String name = (String)map.get("name");
    System.out.println(name);
}

例如如下代码不可编译


public static void method(List<String> list){
    }

    public static void method(List<Integer> list){

    }

自动拆装箱

Integer a= 200;
Integer b = 200;
System.out.println(a==b);

输出为false

三.后端编译

3.1即时编译器

首先主流的java虚拟机同时包含解释器与编译器,这样可以保证同时使用两者的优点。
热点代码会被编译成本地代码,主要针对方法和被多次执行的循环体,这里还会触发一个有意思的现象栈上替换
主要有两种热点代码采集方式:基于采样的热点探测、基于计数的热点探测。Hotspot采用了第二种,使用了方法调用计数器(默认10000次)和回边计数器(默认10700)

3.2提前编译器

目的是改善java的启动时间,演变为现在所熟知的JIT
目前也在积极开发更多更优秀的提前编译器,用于匹配新出的一些垃圾回收器

3.3编译优化

1.最重要的优化:方法内联:把目标方法的代码原封不动地“复 制”到发起调用的方法之中,避免发生真实的方法调用。
2.最前沿的优化:逃逸分析:栈上分配(对象往栈上分配)、标量替换(大对象拆成基础数据类型)、同步消除(消除无用的线程同步)
3.最经典的优化:公共子表达式消除(例如 a=1和b=1先后执行顺序并没有区别,可以先执行初始化a,然后所有用到b的地方都替换为a)
4.最显著的优化:无用代码消除
5.java经典优化:数组边界消除检查
1)数组下标是一个常量,如foo[3],只要在编译期根据数据流分析来确定foo.length的值,并判断下标“3”没有越界,执行的时候就无须判断了。
2)数组访问发生在循环之中,并且使用循环变量来进行数组的访问。如果编译器只要通过数据流分析就可以判定循环变量的取值范围永远在区间[0,foo.length)之内,那么在循环中就可以把整个数组的上下界检查消除掉。

if(foo!=null){
    return foo.value;
}else{
    throw new NullPointException
}

3)如果foo几乎不会发生NPE,那么上述方法无疑是浪费了一次判断性能,就会被直接优化成trycatch
其它优化策略:编译器策略、基于性能监控的优化技术、基于证据的优化技术、数据流敏感重写、语言相关的优化技术、内存及代码位置变换、循环变换、全局代码调整、控制流图变换等

相关文章
|
8月前
在代码优化过程中,常见的错误和bug包括以下几点
在代码优化过程中,常见的错误和bug包括以下几点
|
4月前
警惕 Visual Studio 属性求值副作用导致逻辑不符合预期
【9月更文挑战第27天】在Visual Studio调试过程中,查看对象属性值时需谨慎,因为某些属性的`get`访问器可能包含改变对象状态的代码,导致副作用。例如,`Counter`类的`Count`属性每次访问都会增加计数。这可能导致调试逻辑混乱,如条件判断不符合预期。为避免此类问题,应修改属性设计,将状态改变操作分离出来,或在调试时留意属性求值的副作用。
|
5月前
|
Cloud Native 数据处理
项目环境测试问题之当异步任务在运行过程中抛出非预期的异常会导致后果如何解决
项目环境测试问题之当异步任务在运行过程中抛出非预期的异常会导致后果如何解决
|
6月前
|
开发者
软件交付问题之有效地减少代码中的逻辑错误,如何解决
软件交付问题之有效地减少代码中的逻辑错误,如何解决
|
6月前
|
JSON 前端开发 Java
前后端数据交互-----表单数据获取不到,出错的原因,在编写接口时,没有考虑数据如何返回,解决问题的思路,找到自己出错的地方,围绕着出错的地方进行考虑(很重要),找对解决问题的视频,理清返回数据的思路
前后端数据交互-----表单数据获取不到,出错的原因,在编写接口时,没有考虑数据如何返回,解决问题的思路,找到自己出错的地方,围绕着出错的地方进行考虑(很重要),找对解决问题的视频,理清返回数据的思路
|
8月前
|
算法 程序员
为何程序员在编写程序时难以一次性将所有代码完美无瑕地完成,而是需要经历反复修改Bug的过程?
为何程序员在编写程序时难以一次性将所有代码完美无瑕地完成,而是需要经历反复修改Bug的过程?
81 7
|
8月前
|
测试技术
需求不明确的情况下,测试该如何处理?
需求不明确的情况下,测试该如何处理?
151 0
|
设计模式 消息中间件 JavaScript
干掉 “重复代码”,这三种方式绝了!
干掉 “重复代码”,这三种方式绝了!
36977 2
干掉 “重复代码”,这三种方式绝了!
|
Unix Apache C++
给代码写注释时有哪些讲究?
给代码写注释时有哪些讲究?
172 0
给代码写注释时有哪些讲究?
|
测试技术 数据处理
pytest(9)-标记用例(指定执行、跳过用例、预期失败)
pytest中提供的mark模块,可以实现很多功能,如: 1. 标记用例,即打标签 2. skip、skipif标记跳过,skip跳过当前用例,skipif符合情况则跳过当前用例 3. xfail标记为预期失败
pytest(9)-标记用例(指定执行、跳过用例、预期失败)

热门文章

最新文章