1. 如果判断一个参数是否是奇数?

我们通过下面代码来尝试一下,看看方法可行不:

  public static boolean isOdd(int i) {
return i % 2 == 1;
} public static void main(String[] args) {
System.out.println(isOdd(8));
}

解惑:

(1)程序看起来是可以的,因为利用表达式 i % 2,计算的是 i 除以2时所产生的余数。但事实上,这种方法是不行的,在四分之一的时间里它返回的都是错误的答案;

(2)因为它对所有的负奇数的判断都会失败,因为当i是一个负奇数时,i % 2等于-1而不是1,因此这种判断方式是不正确的。

而通常正确的做法是这样的:

 public static boolean isOdd(int i) {
return i % 2 != 0;
}

只需将 i % 2与0而不是1比较,并且使用想反的比较符即可;

如果在一个强调性能的环境中使用此方法,那么使用位操作符(&)会更好;

public static boolean isOdd(int i) {
return (i & 1) != 0;
}

2. 条件表达式问题

  条件表达式,有一个更广泛的名字:问号冒号操作符。我们看以下程序,并考虑以下程序会输出什么内容呢?

public static void main(String[] args) {
char x = 'X';
int i = 0;
System.out.print(true ? x : 0);
System.out.print(false ? i : x);
}

  我一开始觉得,第一个打印的是char类型的x的值'X',第二个依旧是char类型的x的值'X',因此程序应该输出两个X。但结果却输出了X88。

  答案就在这个条件表达式的第二个和第三个操作数的类型不相同,x是char类型的,而0是int类型的常量,i是int类型的变量。

解惑:

(1)混合类型的计算会引起混乱,而这一点在条件表达式中比在其他任何地方都表现得更明显;

(2)条件表达式的规则很长,很复杂,很难完全记住,但核心就以下三点:

  • 如果第二个和第三个操作数具有相同的类型,那么它就是条件表达式的类型,那么就可以绕过混合类型的计算来避免大麻烦;
  • 如果一个操作数的类型是T,T表示byte、short或char,而另一个操作数是一个int类型的常量表达式,它的值可以用类型T表示,那么条件表达式的类型就是T;
  • 否则,将对操作数类型进行二进制数字提升,而条件表达式的类型就是第二个和第三个操作数被提升之后的类型;

(3)在本程序的两个表达式中,一个操作数的类型是char,另一个的类型是int。首先,在两个表达式中,int操作数都是0,它是可以被表示成一个char的。然后,只有第一个表达式中的int操作数是常量(0),而第二个表达式中的int操作数是变量(i),因此,第二点被应用到第一个表达式上,返回char,第三点被用到第二个表达式上,返回的是对int和char进行二进制提升之后的类型,即int;

(4)条件表达式的类型将确定调用哪一个重载的print方法,对第一个来说,调用的是PrintStream.print(char),第二个是调用的PrintStream.print(int)。前一个重载方法将变量x的值作为Unicode字符(X)打印,而后一个是将其作为一个十进制整数(88)打印。

(5)还有一种情况,是使用final修饰符用于i的声明,把i转换为一个常量表达式,从而让程序打印XX,但这会引起混乱,不建议使用。

总结:因此,在使用条件表达式时,最好在条件表达式中使用类型相同的第二和第三操作数。

3. 赋值表达式问题

请写两行代码,给出x和i的声明,使得

x += i;

合法,而

x = x + i;

不合法。

这个不太难,我们可以很快的写出来,但重要的是我们要理解这两个赋值表达式。

short x = 1;
int i = 123456;

Java语言规范中讲到:

A compound assignment expression of the form E1 op= E2 is equivalent to E1 = (T) ((E1) op (E2)), where T is the type of E1, except that E1 is evaluated only once. 

  即复合赋值 E1 op=E2等价于简单赋值 E1 = (T)((E1) op (E2)),其中T是E1的类型,并且E1只被计算一次。

  换句话说,就是复合赋值表达式自动的将所执行的结果转型为其左侧变量的类型。如果结果的类型与该变量的类型相同,那么这个转型不会有任何影响,然而,如果结果的类型范围要比变量类型范围要小的话,如左侧是short类型,变量类型是int类型,那么复合赋值操作符将自动的执行类型转换,并把变量类型的高位给截取掉;

  因此,x += i 的结果是一个负值:-7615

总结:

(1)请不要将复合赋值操作符作用于byte、short或char类型的变量上;

(2)在将复合赋值操作符作用于int类型的变量上时,要确保表达式右侧不是long、float或double类型;

(3)同样,作用于float上时,要确保右侧表达式不是double类型。

参考自:《Java解惑》

【Java解惑】表达式问题的更多相关文章

  1. Java解惑五:类之谜

    本文是依据JAVA解惑这本书,做的笔记.电子书见:http://download.csdn.net/detail/u010378705/7527721 谜题46 函数重载的问题. JAVA重载解析过程 ...

  2. java解惑之常常忘记的事

    java解惑之常常忘记的事 2012-10-17 18:38:57|  分类: JAVA |  标签:基础知识  软件开发  |举报|字号 订阅     针对刚接触java的菜鸟来说,java基础知识 ...

  3. Java Lambda表达式初探

    Java Lambda表达式初探 前言 本文受启发于Trisha Gee在JavaOne 2016的主题演讲Refactoring to Java 8. Java 8已经发行两年多,但很多人仍然在使用 ...

  4. Java Lambda表达式入门

    Java Lambda表达式入门 http://blog.csdn.net/renfufei/article/details/24600507 Java 8十个lambda表达式案例 http://w ...

  5. Java Web表达式注入

    原文:http://netsecurity.51cto.com/art/201407/444548.htm 0×00 引言 在2014年6月18日@终极修炼师曾发布这样一条微博: 链接的内容是一个名为 ...

  6. 利用Java实现表达式二叉树

    (*^-^*) 什么是二叉树,这里不再介绍,可以自行百度:二叉树.在这里利用java实现“表达式二叉树”. 表达式二叉树的定义 第一步先要搞懂表达式二叉树是个什么东东?举个栗子,表达式:(a+b×(c ...

  7. Java Lambda表达式入门[转]

    原文链接: Start Using Java Lambda Expressions http://blog.csdn.net/renfufei/article/details/24600507 下载示 ...

  8. ref:一种新的攻击方法——Java Web表达式注入

    ref:https://blog.csdn.net/kk_gods/article/details/51840683 一种新的攻击方法——Java Web表达式注入 2016年07月06日 17:01 ...

  9. Java Lambda表达式教程与示例

    Lambda表达式是Java 8中引入的一个新特性.一个lambda表达式是一个匿名函数,而且这个函数没有名称且不属于任何类.lambda表达式的概念最初是在LISP编程语言中引入的. Java La ...

随机推荐

  1. searchbar的使用介绍

    searchBar的使用介绍 首先如何创建一个SearchBar实例: self.searchBar = [[UISearchBar alloc] initWithFrame: CGRectMake( ...

  2. c++逆向 vector

    最近弄Android c/c++方面的逆向,发现c++的类,stl模板,在逆向的时候相比c语言都带来了不小的困难. 今天自己写了个小程序,然后逆向分析了一下 vector<int> arr ...

  3. 《Android开发艺术探索》读书笔记 (13) 第13章 综合技术、第14章 JNI和NDK编程、第15章 Android性能优化

    第13章 综合技术 13.1 使用CrashHandler来获取应用的Crash信息 (1)应用发生Crash在所难免,但是如何采集crash信息以供后续开发处理这类问题呢?利用Thread类的set ...

  4. Java基础知识强化之集合框架笔记07:Collection集合的遍历之迭代器遍历

    1. Collection的迭代器: Iterator iterator():迭代器,集合的专用遍历方式 2. 代码示例: package cn.itcast_03; import java.util ...

  5. Android中实现跨app之间数据的暴露与接收

    例如一个小项目:实现单词本的添加单词等功能 功能:不同的方式实现跨app之间数据的暴露与接收 暴露端app:实现单词的添加(Word.Translate),增删改查: 接收端app:模糊查询,得到暴露 ...

  6. 模拟电路"虚短" & "虚断"

    <虚短 & 虚断> 运算放大器组成的电路五花八门,令人眼花瞭乱,是模拟电路中学习的重点.遍观所有模拟电子技朮的书籍和课程,在介绍运算放大器电路的时候,无非是先给电路来个定性,比如这 ...

  7. ZOJ 3822 Domination(概率dp)

    一个n行m列的棋盘,每天可以放一个棋子,问要使得棋盘的每行每列都至少有一个棋子 需要的放棋子天数的期望. dp[i][j][k]表示用了k天棋子共能占领棋盘的i行j列的概率. 他的放置策略是,每放一次 ...

  8. centos7 部署ssserver

    centos7 部署shadowsocks服务端 为什么要选centos7? 以后centos7 肯定是主流,在不重要的环境还是尽量使用新系统吧 centos7 的坑 默认可能会有firewall 或 ...

  9. H.264视频在android手机端的解码与播放(转)

    随着无线网络和智能手机的发展,智能手机与人们日常生活联系越来越紧密,娱乐.商务应用.金融应用.交通出行各种功能的软件大批涌现,使得人们的生活丰富多彩.快捷便利,也让它成为人们生活中不可取代的一部分.其 ...

  10. 自己编写SqlHelper

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.W ...