hi,
今天无意中看到有个兄弟在java程序中多次用到了goto语句,感觉很是怪异,copy来编译下发现有很多错误!
为此,我付出了一下午时间去研究这个块被人遗忘了的可怕的"魔鬼".
“可怕”的goto语句是c和c++的“遗物”,它是该语言技术上的合法部分,引用goto语句引起了程序结构的混乱,不易理解,goto语句子要用于无条件转移子程序和多结构分支技术。鉴于以广理由,Java不提供goto语句,它虽然指定goto作为关键字,但不支持它的使用,使程序简洁易读。在Java中goto语句被完全抛弃了,与此同时,Java又扩大了break语句和continue语句的功能,通过使用break和continue,程序流程被允许在多层循环中跳转。
大家再看看以下代码:
import java.sql.*;
import java.util.Enumeration;
import java.util.Vector;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
// Referenced classes of package fiyucore.db:
// DBOptions
class DBConnectionManager
{
class DBConnectionPool
{
private String URL;
private int checkedOut;
private Vector freeConnections;
private int maxConn;
private String password;
private String user;
synchronized void freeConnection(Connection con)
{
if(con == null)
break MISSING_BLOCK_LABEL_63;
if(checkedOut > 0)
break MISSING_BLOCK_LABEL_39;
DBConnectionManager.log.debug("DBConnectionManager: 关闭连接");
con.close();
goto _L1
SQLException ex;
ex;
_L1:
break MISSING_BLOCK_LABEL_62;
freeConnections.addElement(con);
checkedOut--;
notifyAll();
return;
}
Connection getConnection(long timeout)
{
long startTime = System.currentTimeMillis();
goto _L1
_L3:
long timeToWait;
long elapsedTime = System.currentTimeMillis() - startTime;
if(elapsedTime >= timeout)
return null;
timeToWait = timeout - elapsedTime;
if(timeToWait > 500L)
timeToWait = 500L;
Thread.sleep(timeToWait);
continue; /* Loop/switch isn't completed */
InterruptedException e;
e;
_L1:
Connection con;
if((con = getConnection()) == null) goto _L3; else goto _L2
_L2:
return con;
}
synchronized Connection getConnection()
{
Connection con;
for(con = null; freeConnections.size() > 0 && con == null;)
{
con = (Connection)freeConnections.firstElement();
freeConnections.removeElementAt(0);
try
{
if(con.isClosed())
{
DBConnectionManager.log.info("删除坏连接!");
con = null;
}
}
catch(SQLException e)
{
con = null;
}
}
if(con == null)
{
if(maxConn == 0 || checkedOut < maxConn)
con = newConnection();
}
if(con != null)
checkedOut++;
return con;
}
private Connection newConnection()
{
Connection con = null;
if(user == null)
con = DriverManager.getConnection(URL);
else
con = DriverManager.getConnection(URL, user, password);
con.setAutoCommit(true);
break MISSING_BLOCK_LABEL_83;
SQLException e;
e;
DBConnectionManager.log.error("不能在连接池中创建新的连接. URL = " + URL, e);
return null;
return con;
}
synchronized boolean release()
{
boolean retValue;
Enumeration allConnections;
retValue = true;
allConnections = freeConnections.elements();
goto _L1
_L3:
Connection con = (Connection)allConnections.nextElement();
con.close();
continue; /* Loop/switch isn't completed */
SQLException e;
e;
DBConnectionManager.log.error("不能关闭连接!");
_L1:
if(allConnections.hasMoreElements()) goto _L3; else goto _L2
_L2:
freeConnections.removeAllElements();
if(checkedOut != 0)
{
retValue = false;
DBConnectionManager.log.warn("DBConnectionManager: 内置的连接池没有正确配置好");
}
checkedOut = 0;
return retValue;
}
public DBConnectionPool(String URL, String user, String password, int maxConn)
{
checkedOut = 0;
freeConnections = new Vector();
this.maxConn = 0;
this.password = null;
this.URL = null;
this.user = null;
this.URL = URL;
this.user = user;
this.password = password;
this.maxConn = maxConn;
}
}
private static final int TIME_BETWEEN_RETRIES = 500;
static Class class$fiyucore$db$DBConnectionManager; /* synthetic field */
private static DBConnectionManager instance = null;
private static Log log;
private DBConnectionPool pool;
private DBConnectionManager(DBOptions option)
{
pool = null;
Class.forName(option.driverClassName).newInstance();
break MISSING_BLOCK_LABEL_59;
Exception e;
e;
log.fatal("DBConnectionManager: 不能加载驱动 = " + option.driverClassName);
pool = new DBConnectionPool(option.databaseURL, option.databaseUser, option.databasePassword, option.maxConnection);
return;
}
static Class class$(String x0)
{
return Class.forName(x0);
ClassNotFoundException x1;
x1;
throw new NoClassDefFoundError(x1.getMessage());
}
void freeConnection(Connection con)
{
pool.freeConnection(con);
}
Connection getConnection(long time)
{
return pool.getConnection(time);
}
Connection getConnection()
{
return pool.getConnection();
}
public static synchronized DBConnectionManager getInstance(DBOptions option)
{
if(instance == null)
{
if(option == null)
option = new DBOptions();
instance = new DBConnectionManager(option);
}
return instance;
}
public static synchronized DBConnectionManager getInstance()
{
if(instance == null)
{
DBOptions option = new DBOptions();
instance = new DBConnectionManager(option);
}
return instance;
}
boolean release()
{
return pool.release();
}
static
{
log = LogFactory.getLog(class$fiyucore$db$DBConnectionManager != null ? class$fiyucore$db$DBConnectionManager : (class$fiyucore$db$DBConnectionManager = class$("fiyucore.db.DBConnectionManager")));
}
}
这程序初看起来是多么的恐怖啊!多次用到goto,混乱。关于流程控制的方法,c++,java都提供了很好的语法:
分支语句 :if-else,break,switch, return.
循环语句 :while,do-while,for, continue.
例外处理语句 :try-catch-finally,throw
最后我们简单介绍一下注释语句
分
支语句
分支语句提供了一种控制机制,使得程序的执行可以跳过某些语句不执行,而转去执行特定的语句。
条件语句if-else.
if-else语句根据判定条件的真假来执行两种操作中的一种,格式为:
if(boolean-expression)
statement1;
[else statement2;]
1.布尔表达式boolean-expression是任意一个返回布尔型数据的达式(这比C、 C++的限制要严格)。
2.每个单一的语句后都必须有分号。
3.语句statement1,statement2可以为复合语句,这时要用大括号{}。建议对单一的语句也用大括号括起,这样程序的可读性强,而且有利于程序的扩充(可以在其中填加新的语句)。{}外面不加分号。
4.else子句是任选的。
5.若布尔表达式的值为true,则程序执行statement1,否则执行statement2。
6.if-else语句的一种特殊形式为:
if(expression1){
statement1
}else if (expression2){
statement2
}……
}else if (expressionM){
statementM
}else {
statementN
}
[else子句不能单独作为语句使用,它必须和if配对使用。else总是与离它最近的if配对。可以通过使用大括号{}来改变配对关系。]
多分支语句switch
switch语句根据表达式的值来执行多个操作中的一个,它的般格式如下:
switch (expression){
casevalue1:statement1;
break;
casevalue2:statement2;
break;
…………
casevalueN:statemendN;
break;
[default:defaultStatement;]
}
1.表达式expression可以返回任一简单类型的值(如整型、实数型、字符型),多分支语句把表达式返回的值与每个case子句中的值相比。如果匹配成功,则运行该case子句后的语句序列。
2.case子句中的值values必须是常量,而且所有case子句中的值是不同的。
3.default子句是任选的。当表达式的值与任一case子句中的都不匹配时,程序执行default后面的语句。如果表达式的值与任一case子句中的值都不匹配且没有default子句,则程序不作任何操作,而是直接跳出switch语句。
4.break语句用来在执行完一个case分支后,使程序跳出switch语句,即终止switch语句的执行。因为case子句只是起到一个标号的作用,用来查找匹配的入口,从此处开始执行,对后面的case子句不再进行匹配,而是直接执行其后的语句序列,因此该在每个case分支后,要用break来终止后面的case分支语句的执行。
在一些特殊情况下,多个不同的case值要执行一组相同的操作,这时可以不用break。
5.case分支中包括多个执行语句时,可以不用大括号{}括起。
6.switch语句的功能可以用if-else来实现,但在某些情况下,使switch语句更简炼,可读性强,而且程序的执行效率提高。
break语句
1.在switch语中,break语句用来终止switch语句的执行。使程序switch语句后的第一个语句开始执行。
2.在Java中,可以为每个代码块加一个括号,一个代码块通常用大括号{}括起来的一段代码。加标号的格式如下:
BlockLabel:{codeBlock}
break语句的第二种使用情况就是跳出它所指定的块,并从紧挨该块的第一条语句处执行。其格式为:
break BlockLabel;
例如:
a:{……//标记代码块a
b:{……//标记代码块b
c:{……//标记代码块c
break b;
…… //willnotbeexecuted
}
…… //willnotbeexecuted
}
…… //willnotbeexecuted
}
…… /executefromhere
}
3.与C、 C++不同,Java中没有goto语句来实现任意的跳转,因为goto语句破坏程序的可读性,而且影响编译的优化。但是从上例可以看出,Java用break来实现goto语句所特有的一些优点。如果break后所指定的标号不是一个代码块的标号,而是一个语句,则这时break完全实现goto的功能。不过应该避免这种方式的使用。
返回语句return
return语句从当前方法中退出,返回到调用该方法的语句处,继续程序的执行。返回语句有两种格式:
1.return expression
返回一个值给调用该方法的语句,返回值的数据类型必须用方法声明中的返回值类型一致。可以使用强制类型转换来使类型一致。
2.return
当方法说明中用void声明返回类型为空时,应使用这种格式,不返回任何值。
return语句通常用在一个方法体的最后,以退出该方法并返一个值。Java中,单独的return语句用在一个方法体的中间时,会产生编译错误,因为这时有一些语句执行不到。但
可以通过把return语句嵌入某些语句(如if-else)来使程序在未执行完方法中的所有语句时退出,例如:
int method (int num) {
∥ return num; ∥will cause compile time error
if (num>0)
return num;
…… ∥ may or may not be executed
∥depending on the value of num
循环语句
循环语句的作用是反复执行一段代码,直到满足终止循环条件为止,一个循环一般应包括四部分内容:
1.初始化部分(initialization):用来设置循环的一些初始条件,计数器清零等。
2.循环体部分(body):这是反复循环的一段代码,可以是单一一条语句,也可以是复合语句。
3.迭代部分(iteration):这是在当前循环结束,下一次循环开始执行的语句,常常用来使计数器加1或减1。
4.终止部分(termination):通常是一个布尔表达式,每一次循环要对该表达式求值,以验证是否满足循环终止条件。
Java中提供的循环语句有:while语句,do-while语句和for语句,下分别介绍。
while语句
while语句实现"当型"循环,它的一般格式为;
[initialization]
while (termination){
body;
[iteration;]
}
1.当布尔表达式(termination)的值为true时,循环执行大括号中语句。并且初始化部分和迭代部分是任选的。
2.while语句首先计算终止条件,当条件满足时,才去执行循环当中的语句。这是"当型"循环的特点。
do-while语句
do-while语句实现"直到型"循环,它的一般格式为:
[initialization]
do{
body;
[iteration;]
}while (termination);
1.do-while语句首先执行循环体,然后计算终止条件,若结果为true,则循环执行大括号中的语句,直到布尔表达式的结果为false。
2.与while语句不同的是,do-while语句的循环体至少执行一次,这是"直到型"循环的特点。
for语句
for语句也用来实现"当型"循环,它的一般格式为:
for(initialization;termination; iteration){
body;
}
1.for语句执行时,首先执行初始化操作,然后判断终止条件否满足,如果满足,则执行循环体中的语句,最后执行迭代部分。完成一次循环后,重新判断终止条件。
2.可以在for语句的初始化部分声明一个变量,它的作用域为一个for语句。
3.for语句通常用来执行循环次数确定的情况(如对数组元素进行操作),也可以根据循环结束条件执行循环次数不确定的情况。
4.在初始化部分和迭代部分可以使用逗号语句,来进行多个操作。逗号语句是用逗号分隔的语句序列。例如:
for(i=0,j=10;i<j;i++, j--){
……
}
5.初始化、终止以及迭代部分都可以为空语句,三者均为空的时候,相当于一个无限循环。
continue语句
1.continue语句用来结束本次循环,跳过循环体中下面尚未执的语句,接着进行终止条件的判断,以决定是否继续循环。对于for语句,在进行终止条件判断前,还要先执行迭代
语句。它的格式为:
continue;
2.也可以用continue跳转到括号指明的外层循环中,这时的格为
continue outerLable;
例 如 :
outer: for( int i=0; i<10; i++ ){ ∥外层循环
for( int j=0; j<20; j++ ){ ∥内层循环
if( j>i ){
……
continue outer;
}
……
}
……
}
该例中,当满足j>i的条件时,程序执行完相应的语句后跳转到外层循环,执行外层循环的迭代语句i++;然后开始下一次循环。
可以从中来比较这三种循环语句,从而在不同的场合选择合适的语句。
例外处理语句
例外处理语句包括try、catch、finally以及throw语句
注释语
Java中可以采用三种注释方式:
1∥ 用于单行注释。注释从∥开始,终止于行尾。
2/*…*/用于多行注释。注释从/*开始,到*/结束,且这种注释不能互相嵌套。
3/**…*/是Java所特有的doc注释。它以/**开始,到*/结束。这种注释主要是为支持JDK工具javadoc而采用的。javadoc能识别注释中用标记@标识的一些特殊变量,并把doc注释加入它所生成的HTML文件。