java编译的语句_Java中的条件编译

在代码中添加大量log,对于CPU和内存的影响如何,会不会降低性能?相信有不少人对此有疑问,本文将详细解答该问题。

一、概述

条件编译是指源程序的代码行,可以在满足一定条件的情况下才进行编译,而未选中的源码,不会生成中间码或机器码,即部分内容参与编译。

条件编译的好处:对于不同硬件平台或者软件平台,或者不同功能模块的代码,编写到在同一个源文件,从而方便程序的维护和移植。

很多程序设计语言都提供条件编译的功能,比如C/++c采用预处理器指示符来达到条件编译。而Java语言并没有提供直接的预处理器,那么Java是不是就没有条件编译呢?先告诉大家,答案是Java存在条件编译,在这之前先说说C/C++的条件编译。

二、C/C++条件编译

对于C/C++,常见的预处理指令:

#include 引入源代码文件

#define 宏定义

#undef 取消已存在的宏定义

#if 如果条件为真,则编译后面的代码

#ifdef 如果宏已定义,则编译后面的代码

#ifndef 如果宏未定义,则编译后面的代码

#elif 如果前面的#if条件为假,并且当前条件为真,则编译后面的代码

#endif 结束前面的#if……#else条件编译语句块

条件编译常见形式:

(1) 当通过#define已定义过该 标识符,则程序编译阶段会选择编译代码段1,否则编译代码段2

#ifdef 标识符

代码段1

#else

代码段2

#endif

(2) 当通过#define未定义过该 标识符,则程序编译阶段会选择编译代码段1,否则编译代码段2。功能正好与(1)相反

#ifndef 标识符

代码段1

#else

代码段2

#endif

(3) 当 表达式为真,则程序编译阶段会选择编译代码段1,否则编译代码段2

#if 表达式

代码段1

#else

代码段2

#endif

三、Java条件编译

Java语法的条件编译,是通过判断条件为常量的if语句实现的。其原理是Java语言的语法糖,根据if判断条件的真假,编译器直接把分支为false的代码块消除。通过该方式实现的条件编译,必须在方法体内实现,而无法在正整个Java类的结构或者类的属性上进行条件编译,这与C/C++的条件编译相比,确实更有局限性。在Java语言设计之初并没有引入条件编译的功能,虽有局限,但是总比没有更强。

反编译分析技术:

对于Debug.java文件,执行:

javac Debug.java //编译后 生成Debug.class文件

javap -c Debug.class //通过javap,反编译class文件

接下来,展开几项对比分析:

3.1 final对比

该对比项是针对是否采用final变量与非final变量的对比:

源码:

// 空方法

public void voidMethod(){

}

//final常量

private final boolean FINAL_FLAG_FALSE = false;

public void constantFalseFlag(){

if(FLAG_FALSE){

System.out.println("debug log...");

}

}

// 非final

private boolean falseFlag= false;

public void falseFlag(){

if(falseFlag){

System.out.println("debug log...");

}

}

反编译解析后的结果如下:

// 空方法

public void voidMethod();

Code:

0: return

//final常量

public void constantFalseFlag();

Code:

0: return

// 非final

public void falseFlag();

Code:

0: aload_0

1: getfield #3 // Field falseFlag:Z

4: ifeq 15

7: getstatic #5 // Field java/lang/System.out:Ljava/io/PrintStream;

10: ldc #6 // String debug log...

12: invokevirtual #7 // Method java/io/PrintStream.println:(Ljava/lang/String;)V

15: return

从反编译的Code字段,可以看出constantFalseFlag()方法体内的内容经过编译后,对于常量false分支,是不可达分支,则在编译成class字节码文件时剪出该分支,最终效果等价于voidMethod()。而对于falseFlag()方法,则多了5条指令。

可见,对于常量为false的if语句,由于恒为false,等同于条件编译的功能。

另外除了反编译的方式来对比分析,如果不了解反编译后的语法,还可以简单地对比编译后的.Class文件的大小,也会发现if(false){}内部的代码块会自动剪除。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值