Java编程思想第五版精粹(四)-运算符

简介: Java编程思想第五版精粹(四)-运算符

1 作用

接受参数并生成新值。与普通方法调用殊途同归。所有运算符都能根据自己的运算对象生成一个值。

1.1 副作用(Side Effect)

可改变运算对象的值。

1.2 最常见用途

修改自己的运算对象,从而产生副作用。注意生成的值亦可由没有副作用的运算符生成。


几乎所有运算符都只能操作基本类型(Primitives)。


唯一例外是 =、== 和 !=,它们能操作所有对象(这也是令人混淆的一个地方)。


此外,String 类支持 + 和 +=。编译器会将 + 连接的非字符串尝试转换为字符串

2 优先级

2.1 作用

决定了存在多个运算符时一个表达式各部分的运算顺序。程序员经常习惯性忘记优先级规则,所以要用括号明确规定运算顺序。

2.2 赋值运算符 (=)

2.2.1 作用

获取右值并赋给左值。右值可以是任何常量、变量或者可产生一个返回值的表达式。左值必须是一个明确的、已命名的变量。即要有一个物理空间存放右值。

基本类型的赋值很简单。因为基本类型持有的是实际的值而非一个对象的引用。只是将内容从一个地方复制到另一个地方。


赋值对象时,实际上是把引用从一个地方复制到另一个地方。这种现象通常称为别名(aliasing),这是 Java 处理对象的一种基本方式。这种场景经常出现在方法参数的传递。

2.3 算术运算符

  • 整数除法会直接截断,而非进位
  • +=,-= 与 C++ 相同的简写形式同时进行运算和赋值操作,由运算符后跟等号表示,并且与语言中的所有运算符一致(只要有意义)


一元减号可以得到数据的负值。一元加号的作用相反,不过它唯一能影响的就是把较小的数值类型自动转换为 int 类型


布尔值只能表示 true 或 false,所以比较它们之间的“大于”或“小于”没有意义


== 和 != 比较的是对象引用,所以比较两个对象的内容是否相同必须使用所有对象(不包括基本类型)中都存在的 equals() 方法

2.4 逻辑运算符

  • 作用
    每个逻辑运算符 && (AND)、||(OR)和 !(非)根据参数的逻辑关系生成布尔值 true 或 false。在 Java 逻辑运算中,不能像 C/C++ 那样使用非布尔值, 而仅能使用 AND、 OR、 NOT。


如果在预期为 String 类型的位置使用 boolean 类型的值,则结果会自动转为适当的文本格式(即 “true” 或 “false” 字符串)。


“短路”(short-circuiting)

整个表达式会在运算到可以明确结果时就停止并返回结果,这意味着该逻辑表达式的后半部分不会被执行到

2.5 字面值常量(Literal)

向程序中插入一个字面值常量时,编译器会确切地识别它的类型。当类型不明确时,必须辅以字面值常量关联来帮助编译器识别。

在文本值的后面添加字符可以让编译器识别该文本值的类型。对于 Long 型数值,结尾使用大写 L 或小写 l 皆可(不推荐使用 l,因为容易与阿拉伯数值 1 混淆)。大写 F 或小写 f 表示 float 浮点数。大写 D 或小写 d 表示 double 双精度。


十六进制(以 16 为基数),适用于所有整型数据类型,由前导 0x 或 0X 表示,后跟 0-9 或 a-f (大写或小写)


八进制(以 8 为基数)由 0~7 之间的数字和前导零 0 表示


Java 7 引入了二进制的字面值常量,由前导 0b 或 0B 表示,它可以初始化所有的整数类型。使用整型数值类型时,显示其二进制形式会很有用。在 Long 型和 Integer 型中这很容易实现,调用其静态的 toBinaryString() 方法即可。若将较小的类型传递给 Integer.tobinarystring() 时,类型将自动转换为 int


Java 7 中有一个深思熟虑的补充:我们可以在数字字面量中包含下划线_,以使结果更清晰

仅限单 _,不能多条相连

数值开头和结尾不允许出现 _

F、D 和 L的前后禁止出现 _

二进制前导 b 和 十六进制 x 前后禁止出现 _


指数计数法


"e" 表示 10 的几次幂

2.6 位运算符

对两个整数对应的位执行布尔代数,从而产生结果。源自 C 语言的底层操作,嵌入式开发中经常要直接操纵硬件,频繁设置硬件寄存器内的二进制位。Java 的设计初衷是电视机顶盒嵌入式开发,所以这种底层的操作仍被保留了下来。

2.7 移位运算符

对象也是二进制的“位”。只能用于处理整数类型

左移位运算符 << 能将其左边的运算对象向左移动右侧指定的位数(在低位补 0)


右移位运算符 >> 则相反,右移位运算符有“正”、“负”值:若值为正,则在高位插入 0;若值为负,则在高位插入 1。Java 也添加了一种“不分正负”的右移位运算符(>>>),它使用了“零扩展”(zero extension):无论正负,都在高位插入 0。这一运算符是 C/C++ 没有的。

2.8 三元运算符

也称为条件运算符。

布尔表达式 ? 值 1 : 值 2   若表达式计算为 true,则返回结果 值 1 ;如果表达式的计算为 false,则返回结果 值 2。

当然,也可以换用普通的 if-else 语句(在后面介绍),但三元运算符更加简洁

2.9 字符串运算符

这个运算符在 Java 里有一项特殊用途:连接字符串。

若表达式以一个 String 类型开头(编译器会自动将双引号 "" 标注的的字符序列转换为字符串),那么后续所有运算对象都必须是字符串。

2.10 类型转换

作用

与一个模型匹配。在适当的时候,Java 会将一种数据类型自动转换成另一种。除了布尔类型的数据,Java 允许任何基本类型的数据转换为另一种基本类型的数据



可以在程序未自动转换时显式、强制地使此类型发生转换。要执行强制转换,需要将所需的数据类型放在任何值左侧的括号内。


为了程序逻辑清晰或提醒自己,也可以使用显式类型转换。在其他情况下,类型转换型只有在代码编译时才显出其重要性。

2.11 类型提升

表达式中最大的数据类型是决定表达式结果的数据类型。

对小于 int 的基本数据类型(即 char、byte 或 short)执行任何算术或按位操作,这些值会在执行操作之前类型提升为 int,并且结果值的类型为 int。若想重新使用较小的类型,必须使用强制转换(由于重新分配回一个较小的类型,结果可能会丢失精度)。Java 不需要 sizeof() 方法,因为所有类型的大小在不同平台上是相同的。


目录
相关文章
|
2月前
|
Java 开发者
Java多线程编程中的常见误区与最佳实践####
本文深入剖析了Java多线程编程中开发者常遇到的几个典型误区,如对`start()`与`run()`方法的混淆使用、忽视线程安全问题、错误处理未同步的共享变量等,并针对这些问题提出了具体的解决方案和最佳实践。通过实例代码对比,直观展示了正确与错误的实现方式,旨在帮助读者构建更加健壮、高效的多线程应用程序。 ####
|
1月前
|
Java 程序员
Java编程中的异常处理:从基础到高级
在Java的世界中,异常处理是代码健壮性的守护神。本文将带你从异常的基本概念出发,逐步深入到高级用法,探索如何优雅地处理程序中的错误和异常情况。通过实际案例,我们将一起学习如何编写更可靠、更易于维护的Java代码。准备好了吗?让我们一起踏上这段旅程,解锁Java异常处理的秘密!
|
5天前
|
存储 Java
java中的常见运算符的计算方式
本文介绍了计算机中二进制数的原码、反码和补码的概念及其转换方式。原码是符号位加真值的绝对值;反码中正数不变,负数其余位取反;补码在反码基础上加1。文章还详细解释了Java中的常见运算符(如按位与、或、异或、移位等)如何基于二进制进行计算,并探讨了使用补码的原因,包括统一符号位处理和扩展表示范围。通过具体代码示例帮助理解这些概念。
java中的常见运算符的计算方式
|
12天前
|
存储 缓存 Java
Java 并发编程——volatile 关键字解析
本文介绍了Java线程中的`volatile`关键字及其与`synchronized`锁的区别。`volatile`保证了变量的可见性和一定的有序性,但不能保证原子性。它通过内存屏障实现,避免指令重排序,确保线程间数据一致。相比`synchronized`,`volatile`性能更优,适用于简单状态标记和某些特定场景,如单例模式中的双重检查锁定。文中还解释了Java内存模型的基本概念,包括主内存、工作内存及并发编程中的原子性、可见性和有序性。
Java 并发编程——volatile 关键字解析
|
16天前
|
算法 Java 调度
java并发编程中Monitor里的waitSet和EntryList都是做什么的
在Java并发编程中,Monitor内部包含两个重要队列:等待集(Wait Set)和入口列表(Entry List)。Wait Set用于线程的条件等待和协作,线程调用`wait()`后进入此集合,通过`notify()`或`notifyAll()`唤醒。Entry List则管理锁的竞争,未能获取锁的线程在此排队,等待锁释放后重新竞争。理解两者区别有助于设计高效的多线程程序。 - **Wait Set**:线程调用`wait()`后进入,等待条件满足被唤醒,需重新竞争锁。 - **Entry List**:多个线程竞争锁时,未获锁的线程在此排队,等待锁释放后获取锁继续执行。
50 12
|
13天前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
93 2
|
2月前
|
设计模式 Java 开发者
Java多线程编程的陷阱与解决方案####
本文深入探讨了Java多线程编程中常见的问题及其解决策略。通过分析竞态条件、死锁、活锁等典型场景,并结合代码示例和实用技巧,帮助开发者有效避免这些陷阱,提升并发程序的稳定性和性能。 ####
|
2月前
|
缓存 Java 开发者
Java多线程编程的陷阱与最佳实践####
本文深入探讨了Java多线程编程中常见的陷阱,如竞态条件、死锁和内存一致性错误,并提供了实用的避免策略。通过分析典型错误案例,本文旨在帮助开发者更好地理解和掌握多线程环境下的编程技巧,从而提升并发程序的稳定性和性能。 ####
|
29天前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
29天前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
50 3