【Java 学习】详讲代码块:控制流语句代码块、方法代码块、实例代码块(构造代码块)、静态代码块、同步代码块

【Java 学习】详讲代码块:控制流语句代码块、方法代码块、实例代码块(构造代码块)、静态代码块、同步代码块
💬 欢迎讨论:如对文章内容有疑问或见解,欢迎在评论区留言,我需要您的帮助!

👍 点赞、收藏与分享:如果这篇文章对您有所帮助,请不吝点赞、收藏或分享,谢谢您的支持!

🚀 传播技术之美:期待您将这篇文章推荐给更多对需要学习Java语言、低代码开发感兴趣的朋友,让我们共同学习、成长!

1. 什么是代码块?

在学习各种语言的时候,有些语句需要使用{}将代码围起来,有些语句确不用,但是总体来说所有的代码被包围在一个{}中,这是为什么呢?

答:想一想我们学数学时的复杂运算,是不是有很多的括号?如:() []{}。有了这些括号就使得运算有条理。

换言之,大括号{}用于明确地定义代码块的开始和结束,使得代码的组织结构更加清晰。这有助于阅读和理解代码,尤其是在复杂的程序中,代码块可能嵌套在其他代码块内部。

学习Java编程语言时,理解代码块的概念是非常重要的。代码块是Java中的基本组成部分之一,它允许你将一组语句组织在一起,以便可以作为一个单元进行处理。

代码块的定义
代码块是一组用大括号{}包围的语句。在Java中,代码块可以出现在多个地方,比如在方法内部、循环结构中、条件语句中等。

代码块的类型

  1. 控制流语句代码块:定义在控制流语句中(如if、else、for、while、do-while、switch等)中,用于指定在特定条件下或循环迭代中应该执行的代码。
  2. 方法代码块:定义在方法内部的代码块,通常用于执行特定的任务。
  3. 实例初始化代码块:没有static关键字的代码块,用于初始化实例变量。
  4. 静态代码块:用static关键字标记的代码块,通常用于初始化类变量。
  5. 同步代码块:使用synchronized关键字,用于多线程编程中的同步操作。

代码块的作用域
代码块中定义的变量只能在该代码块内部被访问,这称为局部变量。局部变量的作用域仅限于定义它们的代码块。

2. 控制流代码块

2.1 定义

控制流语句代码块是普通代码块的一种特殊应用,它们出现在控制流语句(如if、else、for、while、do-while、switch等)中,用于指定在特定条件下或循环迭代中应该执行的代码。这些代码块没有特殊的关键字修饰,它们仅仅是由大括号{}包围的一组语句。

2.2 示例

switch 语句中的代码块

int month =4;switch(month){case1:case2:case3:// 这是一个代码块,用于处理1月、2月和3月System.out.println("第一季度");break;case4:case5:case6:// 另一个代码块,用于处理4月、5月和6月System.out.println("第二季度");break;// 可以有更多的case和代码块default:// 默认情况下的代码块System.out.println("其他季度");}

while 循环代码块

int count =0;while(count <3){// 这是一个代码块,用于循环直到count等于3System.out.println("计数: "+ count); count++;}

for 循环中的代码块

// 使用for循环打印1到5的数字for(int i =1; i <=5; i++){// 这是一个代码块,用于每次循环迭代System.out.println(i);}

if-else 语句中的代码块

int score =85;if(score >=90){// 这是一个代码块,用于处理分数大于等于90的情况System.out.println("优秀");}else{// 另一个代码块,用于处理分数小于90的情况System.out.println("良好");}

3. 方法代码块

3.1 定义

方法代码块是方法体的具体实现,它定义了方法的行为。方法代码块以一对大括号{}开始和结束,位于方法声明之后。

方法代码块的主要作用是实现方法的逻辑。它包含了方法执行时需要执行的所有操作,比如变量声明、条件判断、循环、数据计算等。

3.2 示例及说明

publicclassControlFlowInMethod{publicstaticvoidmain(String[] args){// 调用方法并传递参数printMultiplesOfTen(5);}// 方法声明,用于打印10的倍数publicstaticvoidprintMultiplesOfTen(int count){// 方法代码块开始for(int i =1; i <= count; i++){// 控制流语句的代码块if(i %2==0){// 检查是否为偶数System.out.println("10 * "+ i +" = "+(10* i));}}// 方法代码块结束}}

在这个例子中,printMultiplesOfTen方法用于打印10的倍数。

方法代码块中包含了一个for循环,这是一个控制流语句的代码块。在for循环内部,还有一个if语句,它也是一个控制流语句的代码块,用于检查当前的迭代次数i是否为偶数,如果是偶数,则打印出10乘以当前迭代次数的结果。

当main方法被调用时,它会调用printMultiplesOfTen方法,并传递一个参数5,这意味着方法会打印出10的1倍到5倍的偶数倍数。

4. 实例代码块(构造代码块)

4.1 定义

实例初始化代码块(Instance Initializer Block),也简称为实例代码块,是在类体内部、方法外部定义的代码块,它不属于任何方法。实例初始化代码块主要用于初始化类的实例变量,可以看作是对象创建时的初始化设置。

它没有访问修饰符、返回类型、名称或参数列表,只有一对大括号{}包含的代码。

实例初始化代码块在对象被创建时执行,且在任何构造方法执行之前执行。如果有多个构造方法,实例初始化代码块在每个构造方法之前都会执行。

4.2 示例及说明

publicclassExample{int number;String name;// 实例初始化代码块{ number =10;// 初始化实例变量number name ="Code哈哈笑";// 初始化实例变量nameSystem.out.println("实例代码块已经运行");}// 构造方法publicExample(){System.out.println("构造函数已经被运行");System.out.println("--------------------------------");System.out.println("Constructor is called.");}publicstaticvoidmain(String[] args){Example obj =newExample();System.out.println("Number: "+ obj.number +", Name: "+ obj.name);}}}

在这个例子中,实例初始化代码块在对象obj被创建时执行,它初始化了numbername两个实例变量,并且会打印出被使用的消息。随后,当构造方法Example()被调用时,它会打印出构造方法被调用的信息。最后,在main方法中,我们可以看到对象的实例变量已经被初始化。

在这里插入图片描述

4.3 实例代码块与构造方法的区别

虽然构造方法实例代码块都可以用于初始化对象,但它们之间存在一些区别:

  1. 构造方法可以被重载(即一个类可以有多个构造方法),而实例代码块只能有一个。
  2. 构造方法可以访问参数并使用this关键字调用其他构造方法,而实例代码块不能。
  3. 实例代码块在任何构造方法执行之前执行,而构造方法在对象创建时执行。

5. 静态代码块

5.1 定义

静态代码块(Static Initializer Block),也称为静态初始化块,是Java类中的一个特殊代码块,它使用static关键字声明。静态代码块主要用于初始化类的静态变量或执行只需要执行一次的操作。

静态代码块定义在类的成员变量声明之后、方法定义之前。它使用static关键字标记,并且没有访问修饰符、返回类型、名称或参数列表,只有一对大括号{}包含的代码。

静态代码块在类被Java虚拟机(JVM)加载时执行,且仅执行一次。这意味着静态代码块的执行与类的任何对象的创建无关,它只与类的加载有关。

5.2 示例及说明

publicclassExample{staticint staticNumber;staticString staticName;// 静态代码块static{ staticNumber =10;// 初始化静态变量staticNumber staticName ="Code哈哈笑";// 初始化静态变量staticNameSystem.out.println("Static block is executed.");}publicstaticvoidmain(String[] args){System.out.println("Static number: "+ staticNumber);System.out.println("Static name: "+ staticName);}}

在这个例子中,静态代码块在Example类被加载时执行,它初始化了两个静态变量staticNumberstaticName,并打印了一条消息。当main方法被调用时,它打印出这两个静态变量的值。

在这里插入图片描述

5.3 静态代码块与构造方法的区别

静态代码块与构造方法的主要区别在于:

  1. 静态代码块用于初始化静态变量,而构造方法用于初始化实例变量。
  2. 静态代码块在类加载时执行,而构造方法在对象创建时执行。
  3. 静态代码块中的代码不能访问类的实例变量或调用实例方法,因为这些实例成员依赖于具体的对象实例。

5.4 使用场景

静态代码块适用于以下场景:

  1. 初始化静态常量。
  2. 执行只需要执行一次的资源密集型操作,如加载配置文件、连接数据库等。
  3. 在类加载时执行一些必要的初始化操作,这些操作与任何对象的创建无关。

6. 同步代码块

6.1 定义

同步代码块(Synchronized Block)是Java中用于实现线程同步的一种机制。它允许多个线程在访问共享资源时,保证同一时间只有一个线程可以执行特定的代码段,从而避免发生线程安全问题,如数据不一致、竞态条件等。

同步代码块可以通过synchronized关键字来定义。它可以出现在任何方法中,或者作为一个独立的代码块出现在方法内部。当使用synchronized关键字时,你需要指定一个锁对象(lock object),代码块内的代码只有在获得该对象的锁之后才能执行。

同步代码块的主要作用是确保在多线程环境中,当一个线程访问同步代码块时,其他线程将被阻塞,直到锁被释放。这样可以防止多个线程同时修改同一个资源,确保数据的一致性和线程安全。

6.2 示例及说明

publicclassExample{privatestaticfinalObject lock =newObject();publicvoidmethodOne(){synchronized(lock){// 同步代码块System.out.println("Method One is executing.");try{Thread.sleep(2000);}catch(InterruptedException e){ e.printStackTrace();}System.out.println("Method One finished.");}}publicvoidmethodTwo(){synchronized(lock){// 同步代码块System.out.println("Method Two is executing.");try{Thread.sleep(2000);}catch(InterruptedException e){ e.printStackTrace();}System.out.println("Method Two finished.");}}publicstaticvoidmain(String[] args){finalExample example =newExample();Thread threadOne =newThread(()-> example.methodOne());Thread threadTwo =newThread(()-> example.methodTwo()); threadOne.start(); threadTwo.start();}}

在这个例子中,methodOnemethodTwo都包含了一个同步代码块,它们使用相同的锁对象lock。当main方法启动两个线程时,它们将尝试执行这两个方法。由于同步代码块使用了相同的锁对象,所以这两个方法不会同时执行,从而避免了线程安全问题。

6.3 同步代码块与同步方法的区别

同步代码块与同步方法的主要区别在于

  1. 同步方法使用synchronized关键字修饰整个方法,而同步代码块只同步方法中的特定部分。
  2. 同步代码块允许更细粒度的控制,你可以只同步需要同步的部分,而不是整个方法。

同步代码块适用于以下场景

  1. 当只需要同步方法中的一小部分代码时。
  2. 当多个方法需要同步同一个资源,但不是整个方法时。
  3. 当需要提高性能,减少不必要的同步开销时。

7. 实例代码块(构造代码块)、静态代码块和构造函数的运行顺序

7.1 在同一个类中

publicclassExample{// 静态代码块static{System.out.println("静态代码块已经运行");}// 实例初始化代码块{System.out.println("实例代码块已经运行");}// 构造方法publicExample(){System.out.println("构造函数已经被运行");}publicstaticvoidmain(String[] args){Example obj =newExample();}}

运行结果:

在这里插入图片描述


有同学疑问,运行顺序是不是和代码块写的顺序有关呢?

那打乱顺序再次运行:

publicclassExample{// 构造方法publicExample(){System.out.println("构造函数已经被运行");}// 实例初始化代码块{System.out.println("实例代码块已经运行");}// 静态代码块static{System.out.println("静态代码块已经运行");}publicstaticvoidmain(String[] args){Example obj =newExample();}}
在这里插入图片描述

显然,代码块运行的顺序和代码块的位置无关,先运行静态代码块,再运行实例代码块,最后运行构造函数。

7.2 在继承关系中

同学们思考一下将会打印出什么呢?

classFather{static{System.out.println("父类的静态代码块");}{System.out.println("父类的实例代码块");}publicFather(){System.out.println("父类的构造函数");}}publicclassSonextendsFather{static{System.out.println("子类的静态代码块");}{System.out.println("子类的实例代码块");}publicSon(){//super();System.out.println("子类的构造函数");}publicstaticvoidmain(String[] args){Son son =newSon();}

运行结果

在这里插入图片描述


下面的代码又会打印出什么呢?是不是上面的内容打印两边呢?

classFather{static{System.out.println("父类的静态代码块");}{System.out.println("父类的实例代码块");}publicFather(){System.out.println("父类的构造函数");}}publicclassSonextendsFather{static{System.out.println("子类的静态代码块");}{System.out.println("子类的实例代码块");}publicSon(){//super();System.out.println("子类的构造函数");}publicstaticvoidmain(String[] args){Son son1 =newSon();System.out.println("=======================");Son son2 =newSon();}}

打印结果:

在这里插入图片描述

8. 练习题

下面的代码运行后会发生什么?

publicclassTest{publicintaMethod(){staticint i =0; i++;return i;}publicstaticvoidmain(String args[]){Test test =newTest(); test.aMethod();int j = test.aMethod();System.out.println(j);}}
在这里插入图片描述


答:D
解析:局部代码块中不能有静态变量。只有成员变量才能是静态变量。

下列代码运行后会输出什么?

publicclassTest{staticint cnt =6;static{ cnt +=9;}publicstaticvoidmain(String[] args){System.out.println("cnt = "+ cnt);}static{ cnt /=3;};}
在这里插入图片描述


答案:A
解析:
(1)静态变量初始化:static int cnt = 6; 在类加载时进行初始化,因此cnt的初始值为6。
(2)第一个静态代码块先被执行。在第一个静态代码块中,cnt的值被增加了9,即:cnt = 6 + 9 = 15。
(3)紧接着,第二个静态代码块执行,此时cnt = 15。在第二个静态代码块中,cnt的值被除以3,即:cnt = 15 / 3 = 5。

Read more

震惊!C++中这个逻辑运算符的坑,90%程序员都踩过!

今天我要分享一个C++开发中常见的逻辑陷阱,以及如何优雅地处理多个容器的非空检查。这不仅是一个语法问题,更是一种编程思维的体现。 🎯 问题背景 在实际开发中,我们经常需要检查多个容器是否都为空,或者是否有任意一个非空。比如在图形处理、数值计算等场景,我们需要确保多个数据源都已经准备就绪。 原始需求是这样的:我们有一个车辆轨迹曲线的容器 GwheelCurve,需要检查它是否为空,并将结果取反: bool success =!GwheelCurve.empty(); 但现在需求升级了!我们需要同时检查5个不同的数据容器,只要任意一个非空,就认为检查失败(success = false)。 💡 核心思想 这个问题看似简单,但隐藏着几个关键点: 1. 逻辑运算符的选择:应该用 && 还是用 ||? 2. 取反的时机:是先各自判断再组合,还是先组合再整体取反? 3. 可读性和性能的平衡 让我用几个完整的代码示例来详细说明。 📊 示例1:错误的直观写法 #include<iostream>#include&

By Ne0inhk
基于java的蛋糕烘焙方法经验分享平台

基于java的蛋糕烘焙方法经验分享平台

目录 * 平台概述 * 技术架构设计 * 核心功能模块 * 关键技术实现 * 扩展与运维 * 示例代码片段 * 开发技术 * 源码文档获取/同行可拿货,招校园代理 :文章底部获取博主联系方式! 平台概述 介绍蛋糕烘焙经验分享平台的核心目标,如促进烘焙爱好者交流、分享食谱技巧、提供互动功能等。说明采用Java技术栈的原因,如稳定性、跨平台性、丰富的生态支持。 技术架构设计 后端框架:Spring Boot作为核心框架,集成Spring MVC、Spring Security、Spring Data JPA。 数据库:MySQL存储用户数据、食谱信息,Redis缓存热门内容。 前端技术:Thymeleaf模板引擎或前后端分离方案(如Vue.js + RESTful API)。 部署环境:Docker容器化,Nginx负载均衡,AWS或阿里云托管。 核心功能模块 用户系统:注册/登录(

By Ne0inhk

Java 常见异常梳理

Java 常见异常全面梳理(分类+含义+典型场景) Java 异常体系核心分为 运行时异常(RuntimeException,非受检) 和 编译时异常(Checked Exception 受检),前者编译期无需强制捕获,由程序逻辑错误导致;后者编译期必须显式捕获/抛出,由外部环境(如IO、网络、输入)异常导致。此外,还有错误(Error) 属于JVM层面严重问题,通常无需程序处理。 区分「运行时异常(预防为主)、受检异常(强制处理)、JVM错误(提前规避)」 一、运行时异常(RuntimeException)—— 逻辑错误主导,预防优先,捕获兜底 继承自java.lang.RuntimeException,编译期无需强制处理,多由代码逻辑不严谨导致,核心处理原则是通过严谨的代码逻辑从根源避免,仅对无法预防的场景做捕获兜底。 1.

By Ne0inhk
飞算JavaAI赋能企业级电商管理系统开发实践——一位资深开发者的技术选型与落地总结

飞算JavaAI赋能企业级电商管理系统开发实践——一位资深开发者的技术选型与落地总结

目录 * 一、背景与选型考量 * 二、开发环境与工具适配 * 1. 基础环境搭建 * 2. 飞算JavaAI插件配置 * 3. 版本控制与协作配置 * 三、核心模块设计与实现 * 1. 需求分析与模块拆分 * 2. 核心代码实现与技术亮点 * (1)实体类设计(带审计字段与枚举约束) * (2)服务层实现(带事务控制与业务校验) * (3)控制器实现(带权限控制与参数校验) * (4)网页端 * 四、系统架构与扩展性设计 * 1. 分层架构设计 * 2. 接口设计规范 * 3. 扩展性保障 * 五、资深开发者视角的工具评价 * 1. 代码规范性与可维护性 * 2. 对企业级业务的理解深度 * 3. 与资深开发者工作流的适配性 * 六、项目成果与经验总结 一、背景与选型考量 作为一名从业20余年的开发者,我亲历了从JSP+

By Ne0inhk