标签:
@学习thinking in java
二、控制程序流程
赋值是用等号运算符(=)进行的。它的意思是“取得右边的值,把它复制到左边”。右边的值可以是任何常
数、变量或者表达式,只要能产生一个值就行。但左边的值必须是一个明确的、已命名的变量。也就是说,
它必须有一个物理性的空间来保存右边的值。举个例子来说,可将一个常数赋给一个变量(A=4;),但不可
将任何东西赋给一个常数(比如不能4=A)。
@Test public void test() { int a=12; int b; b=a; a=13; System.out.println(b);//12 /* * 这里我们在栈中创建一个a=12,一个b * 然后把b=a也就是说b的值也等于a,即12 * 这个时候把a变成了13,操作的是a * b没有影响,故b还是12 * */ }
@Test public void test01() { // 本来想用Integer来做例子的,发现Integer除了new新对象改值,没有什么方法去修改,所以就用Demo1类吧 // 1.创建Demo对象d1并赋值为12,这里我们要注意的是程序首先在栈中创建一个d1的句柄,然后在堆中new 一个对象,然后这个d1的句柄 // 指向了这个对象 Demo1 d1 = new Demo1(); d1.i = 12; // 2.创建Demo对象d2,并让其也具体d1的功能,d1指向对象A,所以d2也指向对象A, Demo1 d2 = d1; // d1指向的对象A的值改变了 d1.i = 13; // d2指向的对象,即同样的A,所以当然也改变了 System.out.println(d2.i);// 13 } class Demo1 { int i; }
例:这个例子是我在网上看到的比较全的,讲的比较透彻的例子
@Test public void test04() { /** * 情景一:字符串池 JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多String对象; 并且可以被共享使用,因此它提高了效率。 * 由于String类是final的,它的值一经创建就不可改变。 * 字符串池由String类维护,我们可以调用intern()方法来访问字符串池。 */ String s1 = "abc"; // ↑ 在字符串池创建了一个对象 String s2 = "abc"; // ↑ 字符串pool已经存在对象“abc”(共享),所以创建0个对象,累计创建一个对象 System.out.println("s1 == s2 : " + (s1 == s2)); // ↑ true 指向同一个对象, System.out.println("s1.equals(s2) : " + (s1.equals(s2))); // ↑ true 值相等 // ↑------------------------------------------------------over /** * 情景二:关于new String("") * */ String s3 = new String("abc"); // ↑ 创建了两个对象,一个存放在字符串池中,一个存在与堆区中; // ↑ 还有一个对象引用s3存放在栈中 String s4 = new String("abc"); // ↑ 字符串池中已经存在“abc”对象,所以只在堆中创建了一个对象 System.out.println("s3 == s4 : " + (s3 == s4)); // ↑false s3和s4栈区的地址不同,指向堆区的不同地址; System.out.println("s3.equals(s4) : " + (s3.equals(s4))); // ↑true s3和s4的值相同 System.out.println("s1 == s3 : " + (s1 == s3)); // ↑false 存放的地区多不同,一个栈区,一个堆区 System.out.println("s1.equals(s3) : " + (s1.equals(s3))); // ↑true 值相同 // ↑------------------------------------------------------over /** * 情景三: 由于常量的值在编译的时候就被确定(优化)了。 在这里,"ab"和"cd"都是常量, * 这行代码编译后的效果等同于: String str3 = "abcd"; */ String str1 = "ab" + "cd"; // 1个对象 String str11 = "abcd"; System.out.println("str1 = str11 : " + (str1 == str11)); // ↑------------------------------------------------------over /** * 情景四: 局部变量str2,str3存储的是存储两个拘留字符串对象(intern字符串对象)的地址。 * * 第三行代码原理(str2+str3): 运行期JVM首先会在堆中创建一个StringBuilder类, * 同时用str2指向的拘留字符串对象完成初始化, 然后调用append方法完成对str3所指向的拘留字符串的合并, * 接着调用StringBuilder的toString()方法在堆中创建一个String对象, * 最后将刚生成的String对象的堆地址存放在局部变量str3中。 * * 而str5存储的是字符串池中"abcd"所对应的拘留字符串对象的地址。 str4与str5地址当然不一样了。 * * 内存中实际上有五个字符串对象: 三个拘留字符串对象、一个String对象和一个StringBuilder对象。 */ String str2 = "ab"; // 1个对象 String str3 = "cd"; // 1个对象 String str4 = str2 + str3; String str5 = "abcd"; System.out.println("str4 = str5 : " + (str4 == str5)); // false // ↑------------------------------------------------------over /** * 情景五: JAVA编译器对string + 基本类型/常量 是当成常量表达式直接求值来优化的。 * 运行期的两个string相加,会产生新的对象的,存储在堆(heap)中 */ String str6 = "b"; String str7 = "a" + str6; String str67 = "ab"; System.out.println("str7 = str67 : " + (str7 == str67)); // ↑str6为变量,在运行期才会被解析。 final String str8 = "b"; String str9 = "a" + str8; String str89 = "ab"; System.out.println("str9 = str89 : " + (str9 == str89)); // ↑str8为常量变量,编译期会被优化 // ↑------------------------------------------------------over }
补充:
代码中的字符串常量在编译的过程中收集并放在class文件的常量区中,如"123"、"123"+"456"等,含有变量的表达式不会收录,如"123"+a。
JVM在加载类的时候,根据常量区中的字符串生成常量池,每个字符序列如"123"会生成一个实例放在常量池里,这个实例是不在堆里的,也不会被GC
在执行到双引号包含字符串的语句时,如String a = "123",JVM会先到常量池里查找,如果有的话返回常量池里的这个实例的引用,否则的话创建一个新实例并置入常量池里。如果是 String a = "123" + b (假设b是"456"),前半部分"123"还是走常量池的路线,但是这个+操作符其实是转换成[SringBuffer].Appad()来实现的,所以最终a得到是一个新的实例引用,而且a的value存放的是一个新申请的字符数组内存空间的地址(存放着"123456"),而此时"123456"在常量池中是未必存在的。
要注意: 我们在使用诸如String str = "abc";的格式定义类时,总是想当然地认为,创建了String类的对象str。担心陷阱!对象可能并没有被创建!而可能只是指向一个先前已经创建的对象。只有通过new()方法才能保证每次都创建一个新的对象
在执行String a = new String("123")的时候,首先走常量池的路线取到一个实例的引用,然后在堆上创建一个新的String实例,走以下构造函数给value属性赋值,然后把实例引用赋值给a
存在于.class文件中的常量池,在运行期被JVM装载,并且可以扩充。String的 intern()方法就是扩充常量池的 一个方法;当一个String实例str调用intern()方法时,Java 查找常量池中 是否有相同Unicode的字符串常量,如果有,则返回其的引用,如果没有,则在常 量池中增加一个Unicode等于str的字符串并返回它的引用;看示例就清楚了:
@Test public void test06() { String s0 = "kvill"; String s1 = new String("kvill"); String s2 = new String("kvill"); System.out.println( s0 == s1 ); //false System.out.println( "**********" ); s1.intern(); //虽然执行了s1.intern(),但它的返回值没有赋给s1 s2 = s2.intern(); //把常量池中"kvill"的引用赋给s2 System.out.println( s0 == s1); //flase System.out.println( s0 == s1.intern() ); //true//说明s1.intern()返回的是常量池中"kvill"的引用 System.out.println( s0 == s2 ); //true }
算术运算
注意优先级(用“()"就能解决)
关于进阶的问题
整数与整数运算不存在进阶,直接砍掉小数:例:System.out.println(3/2)//1
@Test public void test08() { int i=0; int j=0; System.out.println(i++);//先用i所以输出为0 System.out.println(++j);//先运算后等于1,再打印为1 }
@Test public void test09() { Demo2_1 d = new Demo2_1(); System.out.println(d.A() || d.B() || d.C()); System.out.println("**************"); System.out.println(d.A() | d.B() | d.C()); } class Demo2_1 { public boolean A() { System.out.println("A"); return true; } public boolean B() { System.out.println("B"); return true; } public boolean C() { System.out.println("C"); return true; } } /* 运行结果: A true ************** A B C true */
@Test public void test11() { int i=3; System.out.println(i<<2);//12 }
@Test public void test12() { int i=3; int j=-3; System.out.println(i>>2); System.out.println(j>>2); }
@Test public void test13() { int i=3; int j=-3; System.out.println(i>>>2); System.out.println(j>>>2); }
三元 if-else 运算符(个人感觉就是if—else的减化版)
布尔表达式 ? 值 0:值 1 例:
@Test public void test14() { int a = 12; int b = 10; if (a > b) { System.out.println(a); } else { System.out.println(b); } System.out.println("*************"); System.out.println(a > b ? a : b); /* * 12 ************* * 12 */ }
二、执行控制
public class WhileTest { public static void main(String[] args) { double r = 0; while(r < 0.99d) { r = Math.random(); System.out.println(r); } } } ///:~
for(初始表达式; 布尔表达式; 步进)语句
@Test public void test() { for (int i = getI(); i < getJ(); i = IJaJa(i)) { System.out.println("****************"); } /* * 分析:首先执行语句之前初始表达式(这个过程只执行一次)故先打印一个"getI()" * 然后,判断条件,满足打印一个"getJ()",满足就执行语句块打印************** 再然后,步进,打印一个“i++" * 再然后,判断条件,满足打印一个"getJ()",满足就执行语句块打印************** 再然后,步进,打印一个“i++" * 再然后,判断条件,不满足,则结束程序 */ } public int getI() { System.out.println("getI()"); return 0; } public int getJ() { System.out.println("getJ()"); return 2; } public int IJaJa(int i) { System.out.println("i++"); return ++i; }
其实等效下列的语句:
@Test public void test01() { int i=0; while(i<2){ System.out.println("************"); ++i; } }
这是for例子的结果:
getI() getJ() **************** i++ getJ() **************** i++ getJ()
@Test public void test02() { for(int i=0,j=0;i<2&&j<1;i++,j++){ int count=1; System.out.println("******count: "+count);//******count: 1 count++; } }
@Test public void test03() { for(int i=0;;){//这是一个无限循环 System.out.println(i++); if(i>1)//当i>1的时候终止当前循环,注意if可不是循环 break;//所以输出0,1 } }
@Test public void test04() { for(int i=0;;){//这是一个无限循环 i++; if(i<2)//当i<2的时候,注意if可不是循环 continue;//不再后面的的内容,再次循环,所以执行下面的语句 System.out.println(i);//2 if(i>=2) break; } }
“标签”是后面跟一个冒号的标识符,就象下面这样:
label1:
对Java 来说,唯一用到标签的地方是在循环语句之前。进一步说,它实际需要紧靠在循环语句的前方——在
标签和循环之间置入任何语句都是不明智的。而在循环之前设置标签的唯一理由是:我们希望在其中嵌套另
一个循环或者一个开关。这是由于 break和 continue 关键字通常只中断当前循环,但若随同标签使用,它们
就会中断到存在标签的地方。如下所示:
label1:
外部循环{
内部循环{
//...
break; //1
//...
continue; //2
//...
continue label1; //3
//...
break label1; //4
}
}
在条件1 中,break 中断内部循环,继续外部循环。在条件2 中,continue 移回内部循环的起始处。但
在条件3 中,continue label1 却同时中断内部循环以及外部循环,并移至label1 处。随后,它实际是继续
循环,但却从外部循环开始。在条件4 中,break label1也会中断所有循环,并回到label1 处,但并不重
新进入循环。也就是说,它实际是完全中止了两个循环。
下面是for 循环的一个例子:
@Test public void test06() { int i = 0; outer: // 标签要紧邻着循环 for (; true;) { // 这是个无限循环 inner: // 标签要紧邻着循环 for (; i < 10; i++) { prt("i = " + i);// static void prt(String s) // {System.out.println(s);} if (i == 2) { prt("continue"); continue; } if (i == 3) { prt("break"); i++; // i增加的语句不能放break后面,不然永远不会得到增加 break; } if (i == 7) { prt("continue outer"); i++; // i增加的语句不能放break后面,不然永远不会得到增加 continue outer; } if (i == 8) { prt("break outer"); break outer; } for (int k = 0; k < 5; k++) { if (k == 3) { prt("continue inner"); continue inner; } } } } /* * i = 0 continue inner i = 1 continue inner i = 2 continue i = 3 break * i = 4 continue inner i = 5 continue inner i = 6 continue inner i = 7 * continue outer i = 8 break outer */ } static void prt(String s) { System.out.println(s); }
如果没有break outer 语句,就没有办法在一个内部循环里找到出外部循环的路径。这是由于break 本身只
能中断最内层的循环(对于continue 同样如此)。
当然,若想在中断循环的同时退出方法,简单地用一个return 即可。
备注:其实吧,这种也不推荐使用,我们完成可以做个flag来控制循环,达到同样的效果
switch(整数选择因子) {
case 整数值1 : 语句; break;
case 整数值2 : 语句; break;
case 整数值3 : 语句; break;
case 整数值4 : 语句; break;
case 整数值5 : 语句; break;
//..
default:语句;
}
要注意的是这里的break不能少,少了break不会报错,会一直往下执行
@Test public void test07() { String a="a"; switch(a){ case "b" : case "a" : System.out.println("a"); break;//a default :System.out.println("**"); } }
注意这里的String a不能为空,代码的原理其实是根据String的hashCode来匹配的,String 为null ,String.hashCode必然会报空指针异常
标签:
原文地址:http://www.cnblogs.com/wangyang108/p/5773904.html