导航:
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
推荐视频:
黑马程序员全套Java教程_哔哩哔哩
尚硅谷Java入门视频教程_哔哩哔哩
推荐书籍:
《Java编程思想 (第4版)》
《Java核心技术·卷I(原书第12版) : 开发基础》
目录
零、引言
0.1 背景和写作目的
0.2 本期更新内容
一、环境准备
1.1 JDK8
1.1.1 下载
1.1.2 安装
1.1.3 配置环境变量
1.1.4 验证
1.1.5 知识加油站:JDK、JRE、JVM、Java的区别
1.1.6 知识加油站:Java8新特性
1.2 记事本体验Hello World
二、Java编译器:IDEA
2.1 下载安装配置
2.1.1 下载安装
2.1.2 配置
2.1.2.1 编码配置UTF-8
2.1.2.2 配置Maven路径
2.1.2.3 (根据情况配置)调试时跳过反射以及aop
2.1.3 安装插件
2.1.3.1 基础插件
2.1.3.2 高级插件
2.1.3.3 解决插件安装慢的问题
2.1.5 编写Hello World
2.1.6 常用快捷键
2.2 断点调试
2.2.1 断点
2.2.2 调试
2.2.3 知识加油站:高级断点调试
2.2.3 丢帧(退帧)
2.2.4 强制返回
2.2.5 stream流调试
2.2.6 评估表达式:ALT+F8
2.2.7 多线程调试
三、Java基本概念和语法
3.1 基本数据类型
3.1.1 概述
3.1.2 知识加油站:基本数据类型和引用类型
3.1.3 知识加油站:基本数据类型的内存空间
3.2 数组
3.2.1 创建
3.2.1.1 数组的两种创建方式
3.2.1.2 知识加油站:数组的创建原理
3.2.2 初始化
3.2.2.1 静态初始化
3.2.2.2 动态初始化
3.2.3 访问
3.2.3.1 基本访问方式
3.2.3.2 高级访问方式:迭代器、Stream流、toString
3.2.4 操作
3.2.4.1 数组和字符串的互相转换
3.2.4.2 数组和***的互相转换
3.3 流程控制语句
3.3.1 概述
3.3.2 IF分支语句
3.3.2.1 基本语法
3.3.2.2 练习
3.3.3 switch分支语句
3.3.3.1 基本语法
3.3.3.2 练习
3.4 修饰符
3.4.1 访问权限修饰符:public、protected、default、private
3.4.2 static
3.4.2.1 基本介绍
3.4.2.2 知识加油站:类变量
3.4.3 abstract
3.4.3.1 基本介绍
3.4.3.2 知识加油站:接口和抽象类的区别
3.4.6 final
3.4 常用关键字
3.4.1 this
3.4.2 super
3.4.3 知识加油站:this和super的区别
四、面向对象编程(OOP)
4.0 概述
4.1 类和对象
4.1.1 基本介绍
4.1.2 知识加油站:内部类
4.1.3 知识加油站:创建对象的几种方法
4.2 方法
4.2.1 基本介绍
4.2.2 基本用法
4.2.2.1 定义
4.2.2.2 调用
4.2.2.3 返回值
4.2.3 方法的重载
4.2.3.1 重载
3.2.3.2 重载和重写的区别
3.2.4 可变参数
3.2.5 构造方法
4.3 接口和抽象类
4.3.1 概述
4.3.2 接口
4.3.3 抽象类
4.3.4 知识加油站:抽象类和接口的区别
4.4 面对对象三特性
4.4.1 封装
4.4.2 继承
4.4.3 多态
五、常用类
5.1 String类
5.1.1 基本介绍
5.1.2 基本用法
5.1.2.1 创建
5.1.2.2 访问
5.1.2.3 连接
5.1.2.4 子串
5.1.2.5 删除
5.1.2.6 替换
5.1.2.7 获取长度
5.1.2.8 比较
5.1.2.8 知识加油站:==与equals()的区别
5.1.2.9 分割
5.1.2.10 转换大小写
5.1.2.11 字符串和数字的互相转换
5.1.2.12 数组和字符串的互相转换
5.1.3 知识加油站
5.1.3.1 字符串和字符数组的区别
5.1.3.2 字符串常量池
5.1.3.3 String不可被继承、不可变的原因
5.1.3.4 new String("abc")创建的字符串对象数量
5.2 StringBuilder类
5.2.1 基本介绍
5.2.2 知识加油站:String、StringBuffer、Stringbuilder的区别
5.3 Scanner类
5.4 Object类
5.4.1 基本介绍
5.4.2 知识加油站:JVM垃圾回收的可达性分析算法
5.4.3 知识加油站:hashCode()和equals()的区别
5.5 System类
5.6 Integer类
5.6.1 基本介绍
5.6.2 知识加油站:包装类的自动拆装箱与自动装箱
5.6.3 知识加油站:什么情况下用包装类?什么情况下用基本数据类型?
5.6.4 知识加油站:包装类和基本数据类型直接如何互相比较?(浮点数等号比较精度丢失问题)
5.6.5 知识加油站:Integer a1=127;Integer a2=127;a1==a2原因
5.7 Arrays类
5.6.1 排序
5.6.2 查找和判断
5.6.3 批量填充
5.6.4 转换字符串、链表
5.6.5 拷贝
5.8 Date类
5.8.1 常用方法
5.8.2 代码示例
5.8.3 知识加油站:Date类的线程安全问题
5.8.4 格式化日期:
5.8.4.1 SimpleDateformat类:线程不安全
5.8.4.2 DateTimeFormatter类:线程安全
5.9 Math类
5.10 Random
5.10.1 基本介绍
5.10.2 代码示例
1.提供一套完整的Java学习路线。
本专栏旨在提供一套完整的Java学习路线,覆盖Java基础知识、数据库、SSM/SpringBoot等框架、Redis/MQ等中间件、设计模式、架构设计、性能调优、源码解读、核心面试题等全面的知识点,并在未来不断更新和完善,帮助Java从业者在更短的时间内成长为高级开发。
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/黑马旅游/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码
2.遵循开源精神,贡献自身力量。
Java语言拥有一个非常活跃的社区,在笔者的成长过程中,从很多文章中学习到有价值的知识,所以也希望可以能帮助到其他人。
3.温故知新,提高技术深度。
人的记忆力有时候是不可靠的,所以在做事情的过程中要尽量有记录、有结果,这样随着自己的成长,技术深度的提升,可以不断地强化它们,并使自己可以有很多新的见解。
本期更新了以下内容:
下载jdk8(JDK8就是JDK1.8)
JDK8或者JDK1.8是由于自从JDK1.5/JDK5命名方式改变后遗留的新旧命令方式问题。所以JDK8或者JDK1.8也是同一个东西。
下载地址:Java Downloads | Oracle
选择windows64版本exe下载
需要登录才能下载:
登录后会自动下载:
下一步:
设置安装路径:
点击下一步,将安装完成。
此电脑右键-属性
高级系统设置:
环境变量:
设置环境变量:
JAVA_HOME
%JAVA_HOME%\bin
验证是否安装成功:
打开Windows的运行(可用win+R快捷键打开),输入cmd
验证Java版本:
java -version
如果提示找不到命令,就尝试重新打开一个cmd窗口,如果还不行,就检查环境变量是不是配错了。
验证Java编译命令是否可用:
javac
简而言之,JVM是Java虚拟机,JRE是Java运行环境,JDK是个Java开发的工具包,Java是门编程语言。
这里仅简单介绍,具体下面章节会详细介绍。
Java8新特性:
1.新增txt文件,输入内容如下,并将文件命名为Test.java
public class Test { public Test() { } public static void main(String[] args) { System.out.println("hello~~~~~~~"); } }
2.在当前目录下打开cmd命令行,编译、运行Java代码:
编译:
#javac指令编译java代码为Test.class文件 javac Test.java
运行:
#java指令运行Test.class文件。.class后缀可以省略 java Test
可以看到,命令行会输出
扩展:
1. 如何打开命令行?
方法一:在文件地址栏输入cmd,回车,可以快速在当前目录打开命令行:
方法二:win+R快捷键,输入cmd,回车,然后用cd命令到达指定目录:
2.javac指令和java指令是什么?
在Java中,
javac
和java
是两个关键的命令行工具,用于编译和运行Java程序。javac:
javac是Java编译器的命令。它用于将Java源代码文件(以.java为扩展名)编译成字节码文件(以.class为扩展名)。
语法:
javac <options> <source files>
例如,要编译一个名为
MyProgram.java
的Java源文件,可以使用以下命令:
javac MyProgram.java
javac的常见选项:
- javac -d <directory>:指定生成的类文件存放的目录。
- javac -cp 或 -classpath:指定查找用户类文件和注释处理程序的位置。
- javac -source:指定要使用的源代码版本。
- javac -target:生成的类文件的目标字节码版本。
java:
java是Java虚拟机(JVM)的命令。它用于运行已经编译好的Java程序。
语法:
java <options> <class file>
例如,如果你有一个名为
MyProgram.class
的已编译Java程序,可以使用以下命令来运行它:
java MyProgram
java的常见选项:
- java -cp 或 -classpath:指定查找用户类文件的位置。
- java -Xmx:设置JVM的最大堆内存大小。
- java -Xms:设置JVM的初始堆内存大小。
Other Versions - IntelliJ IDEA
安装一直点击下一步即可,只有两点需要注意:
1.设置安装路径:
2.创建桌面快捷方式:
注意:所有设置都必须要关闭项目后,再进行设置,这样才是全局设置。
先关闭项目再点击设置
这一步初学者可以直接略去,在后面学到Maven后再配置。
设置里搜索Maven,然后配置主路径、仓库路径、配置路径
Maven配置依赖和插件后自动刷新:
跳过反射
java.lang.reflect.*
跳过spring aop
org.springframework.aop
IDEA推荐安装以下插件,便捷开发,初学者可以仅安装靠前个插件,后面学到再安装:
1.Chinese(Simplified) Language Pack/中文语言包
用于汉化IDEA,官方已经加汉化做的很好了,例如checkout移位签出,rebase译为变基。当然习惯了英文可以继续用英文。
2.Alibaba Java Coding Guidelines
提供代码检查,对于不符合编程规范的代码,会进行提示:
/** * 没有注释作者 */ public class Test { public static void main(String[] args) { int a=123; System.out.println("hello");//不要使用行尾注释 } }
下面插件建议学到Maven和Mybatis后再安装:
1.Maven helper
安装后在代码中右键,可以选择Maven模块,会出现打包,编译等选择,另外Maven helper可以解决依赖冲突问题。
2.MybatisX
使用MybatisX插件后,对应的mapper.xml,mapper.java出现小鸟图标,编写接口定义方法后,会自动在xml中生成statement语句,提高Mybatis框架的开发效率。
3.RESTfulToolkit-fix
RestfulToolkit-fix是一套RESTful服务开发辅助工具集,可以根据URL直接跳转到对应的Controller方法定义,在Controller的方法上添加了能复制请求URL和方法参数的功能。
安装后可以通过ctrl+\快捷键,搜索url,插件会帮你找到url对应的Controller。
全局模糊匹配URL:
- Windows快捷键: Ctrl + \ 或者 Ctrl + Alt + N。
- MacOS快捷键: Command + \ 或者 Command + option + N。
5.Private Notes
用于在只读代码中添加注释,在阅读一些框架和中间件源码时使用比较方便。
在任何你想加注释的地方 按下Alt + Enter鼠标移出点击即可保存
已有私人注释 按下Alt + Enter即可快速编辑
Alt + p 可快速添加或者编辑私人注释
Alt + o 展示私人注释的其它操作
右键菜单私人注释查看操作
6.easy javadoc
首先安装插件后重启;
然后配置类注解:
/** * @Author: xxx * @CreateTime: $DATE$ * @Description: TODO * @Version: 1.0 */
修改快捷键:
在类上连按两次“\”,就会自动生成注释:
如果插件安装速度较慢,可以尝试以下方法进行解决。
1.访问DNS查询
https://tool.chinaz.com/
2.检测
在页面搜索框中输入plugins.jetbrains.com,点击检测按钮
plugins.jetbrains.com
3.寻找最优dns地址
下拉并复制靠前个ip地址:例如3.160.150.74
tip:注意并不一定要用上面这个ip地址,要选择自己网络环境下最优的ip地址。选择原值是TTL值越大越好,解析时间越小越好
4.更改hosts文件
打开C:\Windows\System32\drivers\etc目录下的hosts文件,在hosts文件中添加下述配置,然后保存退出即可。
3.160.150.74 plugins.jetbrains.com
5.刷新DNS
点击Win+R,输入cmd,点击Enter按键,然后在黑窗口中执行以下命令:
ipconfig /flushdns
1.创建空项目
2.新建模块
注意SDK开发工具包选JDK8
3.新建包
4.新建类
然后psvm回车自动生成静态main方法,sout回车自动生成System.out.println()输出方法。
package package1; /** * @author xxx */ public class Test { public static void main(String[] args) { System.out.println("hello world~~~~"); } }
运行:
如果控制台中文乱码,就在设置里把文件编码页都改成utf-8。
格式化:ctrl+alt+L
main方法psvm回车,输出sout回车,内容提示ctrl+alt+space。alt+insert生成构造、setget方法。
补全代码:Ctrl+Alt+v或.var或.for
例如写new Dog();按快捷键会自动补全声明Dog dog=new Dog();或者写new Dog().var回车会自动补全成Dog dog=new Dog();或者写lists.for回车会自动生成for(List<String> list:lists){}
多功能快捷键:alt+enter
获取警告、报错提示自动改正。
查找:ctrl+f
替换:ctrl+r
替换所有与选中内容相同的文本
ctrl+b:跟进
或者ctrl+鼠标左键、或者鼠标左键,跟进到该类的定义页面。如果选中的是标识符,可以查找页面内所有该标识符位置。
alt+左键:整列编辑
ctrl+d:复制光标行并粘贴
ctrl+x:删除光标行
Ctrl+h:查看该类继承关系:
ctrl+alt+m:抽取选中代码为方法
Ctrl+f12:查看类中所有方法。
评估表达式:ALT+F8
查看本地历史记录:
idea是会自动保存各文件历史记录的,右键文件-查看本地历史记录即可:
- 调试按钮:
- 运行按钮:
普通断点:如果断点打在普通代码行上,点击“debug”会自动运行到该行暂停;
方法断点:如果断点打在方法上,点击“debug”会自动运行到该方法入口处暂停;
接口断点:如果断点打在接口方法,点击“debug”会自动运行到实现类对应方法入口处暂停;
字段断点:如果断点打在字段上,点击“debug”会自动运行到修改此字段的代码行处暂停(可以在断点处右键设置成访问时也暂停);
条件断点:在断点处右键,可以输入布尔类型的条件,例如i==0、user.getName()!=null&&"zhangsan".equals(user.getName())。
示例:当i为10时暂停代码运行:
debug可以看到变量信息:
异常断点:在异常时暂停代码运行。在断点处右键-更多,设置断点为异常断点,程序将在发生异常时暂停。
打断点后代码运行到这一步会停止:下面红色方框是最常用的两个调试按钮:
蓝色代码行代表此时调试运行到这一行(这一行还没运行),能看到上一行的结果。
调试的五个核心图标分别是:
- “Step over”:步过。往下运行一行代码并跳过当前方法的调用。
- “Step into”:步入。进入方法内部并继续逐行执行。适用于自定义方法或第三方方法,但无法进入JDK方法的内部。注意如果蓝色光标行不是方法,则此按钮相当于step over。
- “Force step into”:强制步入(不常用)。强制进入方法的内部,即使常规的"Step into"操作无法进入方法内部时也可以使用。适用于无法通过常规方式进入的方法。例如Symtem.out.println()等JDK方法内部。
- “Step out”:步出。退出当前方法,即执行完当前方法后返回到调用该方法的位置。注意如果本方法是main方法,则将直接步出这个main方法。
- “Resume Program”:运行到光标处。继续运行程序直到下一个断点位置或程序结束。可用于暂停状态下的程序恢复运行。
退帧:回退到此方法被调用之前。
当我们 Debug 从 A 方法进入 B 方法时,通过降帧(退帧)可以返回到调用 B 方法前,这样我们就可以再一次调用 B 方法。
通常用于当我们快执行完 B 方法后,发现某个重要流程被我们跳过了,想再看一下,则此时可以先回退到 A 方法,然后再次进入 B 方法。
右键并点击“force return” 后,此方法栈将直接终止,恢复到上个方法栈。
使用场景:需要结束当前断点,并且不希望接下来的程序操作数据库时,使用强制返回。
注意:注意如果点击红色方格或重新调试虽然也是终止,但它其实是运行完剩余代码后再终止。
- 强制返回:接下来的程序将不再继续执行。
- 终止:接下来的程序将走完,然后再终止程序。
示例:在下面断点处强制返回,将不打印那行文字;终止则打印。
在调试中,断点到达Stream流,往往直接会跳过,而我们有时候是需要看Stream流的处理逻辑的,所以就需要对Stream流进行调试。
调试方法:在stream流处打断点,debug后点击“trace current stream chain”:
示例:Stream流过滤出长度大于5单词:
public class Test { public static void main(String[] args) { // 创建一个列表 List<String> words = Arrays.asList("apple", "banana", "orange", "grape", "watermelon"); // 使用 Stream 进行过滤,只选择长度大于5的单词 List<String> filteredWords = words.stream() .filter(word -> word.length() > 5) .collect(Collectors.toList()); System.out.println("Filtered Words: " + filteredWords); } }
点击“跟踪当前流链”即可看到过滤效果:
在if处打断点,debug后点击“evaluate expression”,可以在评估框下测试另一个分支:
多线程环境,打断点并右键设置Thread或All:
Java的八大基本数据类型分别是:
代码示例:
public class Test { public static void main(String[] args) { // 1. 整型 byte byteData = 120; short shortData = 30000; int intData = 2000000000; long longData = 9000000000000000000L; // 2. 字符型 char charData = 'A'; // 3. 浮点型 float floatData = 3.14f; double doubleData = 123.456; // 4. 布尔型 boolean booleanData = true; // 访问和输出变量值 System.out.println("整型:"); System.out.println("byteData: " + byteData); System.out.println("shortData: " + shortData); System.out.println("intData: " + intData); System.out.println("longData: " + longData); System.out.println("\n字符型:"); System.out.println("charData: " + charData); System.out.println("\n浮点型:"); System.out.println("floatData: " + floatData); System.out.println("doubleData: " + doubleData); System.out.println("\n布尔型:"); System.out.println("booleanData: " + booleanData); // 操作变量 intData++; // 增加 intData 的值 doubleData *= 2; // 将 doubleData 的值乘以 2 // 输出修改后的值 System.out.println("\n操作后的整型和浮点型变量:"); System.out.println("intData: " + intData); System.out.println("doubleData: " + doubleData); } }
结果:
基本数据类型
基本数据类型共有八大类,这八大数据类型又可分为四小类,分别是整数类型(byte/short/int/long)、浮点类型(float、double)、字符类型(char)和布尔类型(boolean)。
引用类型
引用类型包括数组引用、类引用、接口引用,还有一种特殊的null类型,所谓引用数据类型就是对一个对象的引用,对象包括实例和数组两种。
区别:
特征 | 基本数据类型 | 引用类型 |
---|---|---|
存储方式 | 直接存储数据值 | 存储对象的引用,实际数据存储在堆内存中 |
默认值 | 有默认值,不可为null | 有默认值为null |
赋值方式 | 直接赋值 | 使用new关键字创建对象 |
内存分配 | 栈上分配 | 在堆上分配 |
大小 | 固定大小,与具体类型有关 | 大小不固定,由对象本身和其内容决定 |
效率 | 更高效,直接操作数据 | 相对较低,需要间接操作对象引用 |
比较 | 用==比较 | 通常使用equals方法比较 |
范围 | 有限,具体范围取决于数据类型 | 无限,取决于系统的内存大小 |
传递方式 | 值传递,传递的是实际的数据值 | 引用传递,传递的是对象的引用 |
示例 | int num = 42; |
String str = new String("Hello"); |
JVM存储位置 | 方法参数和局部变量:存在本地方法栈的局部变量表; final常量、静态变量:存在类常量池 |
堆 |
对于基本数据类型,你需要了解每种类型所占据的内存空间,这是面试官喜欢追问的问题:
一维数组:
数组命名有两种方式。
// a和b都是一维数组 int[] a,b;
//c和d都是一维数组,不推荐这种命名方法 int c[],d[];
二维数组:
int[][] x;
阿里规约:
【强制】 类型与中括号紧挨相连来表示数组。
正例: 定义整形数组 int[] arrayDemo;
反例: 在 main 参数中,使用 String args[]来定义
JVM内存模型:
什么是JVM的内存模型?详细阐述Java中局部变量、常量、类名等信息在JVM中的存储位置-CSDN博客
数组存放在JVM的堆中,是一片连续的存储空间,下标依次为0,1,2,3,4...
数组在JVM中的创建过程:
Java数组初始化有两种方式,分别是静态初始化、动态初始化。
静态初始化:不指定数组长度,由JVM自己识别
格式:数据类型[] 数组名=new 数据类型 {元素1,元素2,元素3...};
示例:
// 静态初始化:不指定长度,由系统猜 // 简写格式(推荐) int[] c={1,2,3}; // 完整格式 int[] ff=new int[]{1,2,3};
动态初始化:指定数组长度
格式:数据类型[] 数组名=new 数据类型[数组的长度];
示例:
// 动态初始化:指定长度 int[] ff=new int[3];
通过下标访问数组元素:
数组存放在JVM的堆中,是一片连续的存储空间,下标依次为0,1,2,3,4...
使用数组的索引(下标)来访问特定位置的元素。数组的索引从 0 开始,一直到数组长度减一。
int[] numbers = {1, 2, 3, 4, 5}; int element = numbers[2]; // 访问索引为 2 的元素,即数组中的第三个元素 System.out.println(element); // 输出:3
for遍历数组:
使用循环结构(如 for
或 foreach
)遍历数组,访问每个元素。
int[] numbers = {1, 2, 3, 4, 5}; // 使用 for 循环 for (int i = 0; i < numbers.length; i++) { System.out.println(numbers[i]); } // 使用 foreach 循环 for (int num : numbers) { System.out.println(num); }
结果:
遍历:迭代器
对于***类(如 ArrayList
),可以使用迭代器来访问元素。
ArrayList<Integer> list = new ArrayList<>(); list.add(1); list.add(2); list.add(3); Iterator<Integer> iterator = list.iterator(); while (iterator.hasNext()) { int element = iterator.next(); System.out.println(element); }
结果:
遍历:Java 8 的 Stream
使用 Java 8 引入的 Stream API 进行数组遍历和操作。
int[] numbers = {1, 2, 3, 4, 5}; // 使用 Stream.forEach Arrays.stream(numbers).forEach(System.out::println);
打印全部:Arrays 类的 toString 方法
使用 Arrays
类的 toString
方法将整个数组转换为字符串。
int[] numbers = {1, 2, 3, 4, 5}; //输出:[1, 2, 3, 4, 5] System.out.println(Arrays.toString(numbers)); //直接输出数组会输出地址:[I@5b444398 System.out.println(a);
数组本身的方法很少,只有equals()、stream()等简单的方法,一些对数组的高级操作,可以将数组转为String或***,操作后再转回数组。
数组转字符串:
Arrays.toString(数组名)
示例:
int[] numbers = {1, 2, 3, 4, 5}; //输出:[1, 2, 3, 4, 5] System.out.println(Arrays.toString(numbers)); //直接输出数组会输出地址:[I@5b444398 System.out.println(a);
字符串转数组:
方法名:stringToIntArray(String str)、stringToCharArray(String str)等
示例代码:
// 字符串转整型数组 int[] intArray = stringToIntArray("1 2 3 4 5"); // 字符串转字符数组 char[] charArray = stringToCharArray("Hello");
List<T> arrayToList(T[] array)
T[] listToArray(List<T> list, Class<T> elementType)
Set<T> arrayToSet(T[] array)
T[] setToArray(Set<T> set, Class<T> elementType)
示例:
// 1. 数组转List String[] array = {"apple", "banana", "orange"}; List<String> listFromArray = Arrays.asList(array); System.out.println("List from Array: " + listFromArray); // 2. List转数组 List<String> fruitList = new ArrayList<>(List.of("apple", "banana", "orange")); String[] arrayFromList = fruitList.toArray(new String[0]); System.out.println("Array from List: " + Arrays.toString(arrayFromList)); // 3. List转Set Set<String> setFromList = new HashSet<>(fruitList); System.out.println("Set from List: " + setFromList); // 4. Set转List Set<String> fruitSet = new HashSet<>(Set.of("apple", "banana", "orange")); List<String> listFromSet = new ArrayList<>(fruitSet); System.out.println("List from Set: " + listFromSet); // 5. Set转数组 String[] arrayFromSet = fruitSet.toArray(new String[0]); System.out.println("Array from Set: " + Arrays.toString(arrayFromSet)); // 6. 数组转Set Set<String> setFromArray = new HashSet<>(Arrays.asList(array)); System.out.println("Set from Array: " + setFromArray); }
转为***后,可以通过***的api操作各元素。例如:
判断数组是否包含某个元素:
示例代码:
// 判断整型数组是否包含元素 int[] array = {1, 2, 3, 4, 5}; boolean isEle = Arrays.asList(array).contains("a"); System.out.println(isEle);
流程控制语句分类:
格式一:如果...就...
if (关系表达式) { 语句体; }
执行流程:
格式2:如果...就...,否则...
if (关系表达式) { 语句体1; } else { 语句体2; }
执行流程:
格式3:如果...就...,如果...就...,否则...
if (关系表达式1) { 语句体1; } else if (关系表达式2) { 语句体2; } // ... else { 语句体n+1; }
执行流程:
如果年龄小于20岁,则打印年轻人,如果年龄大于60岁,则打印老年人,否则打印中年人。
public class AgeCategory { public static void main(String[] args) { // 声明年龄变量 int age = 35; // 判断年龄范围并打印相应信息 if (age < 20) { System.out.println("年轻人"); } else if (age > 60) { System.out.println("老年人"); } else { System.out.println("中年人"); } } }
格式:
switch (表达式) { case 值1: 语句体1; break; case 值2: 语句体2; break; // ... default: 语句体n+1; break; //最后一个可以省略 }
格式说明:
执行流程:
注意事项:在switch语句中,如果case控制的语句后面不写break,将会出现"穿透"现象。即
写一段Java代码,输入数字几,就输出星期几:
import java.util.Scanner; public class DayOfWeek { public static void main(String[] args) { // 创建Scanner对象用于输入 Scanner scanner = new Scanner(System.in); // 提示用户输入一个数字代表星期几 System.out.print("请输入一个数字(1-7)代表星期几:"); // 读取用户输入的数字 int dayNumber = scanner.nextInt(); // 使用Switch语句判断并输出对应的星期几 switch (dayNumber) { case 1: System.out.println("星期日"); break; case 2: System.out.println("星期一"); break; case 3: System.out.println("星期二"); break; case 4: System.out.println("星期三"); break; case 5: System.out.println("星期四"); break; case 6: System.out.println("星期五"); break; case 7: System.out.println("星期六"); break; default: System.out.println("输入无效,应为1-7之间的数字。"); } // 关闭Scanner scanner.close(); } }
public : 对所有类可见。使用对象:类、接口、变量、方法
protected : 对同包可见、对不同包子类可见。使用对象:变量、方法。 注意:不能修饰类(外部类)。
default : 同包可见。使用对象:类、接口、变量、方法。
private : 在同一类内可见。使用对象:变量、方法。 注意:不能修饰类(外部类)
修饰符 | 当前类 | 同一包内 | 子孙类(同一包) | 子孙类(不同包) | 其他包 |
---|---|---|---|---|---|
public |
Y | Y | Y | Y | Y |
protected |
Y | Y | Y | Y/N(说明) | N |
default |
Y | Y | Y | N | N |
private |
Y | N | N | N | N |
private
在同一类内可见,保护成员不被别的类使用。可以修饰变量、方法。 注意:不能修饰类(外部类)
private变量不能被其他类直接访问,但可以通过get和set方法间接访问:
示例:
public class Phone { int price;//成员变量在堆内存 private int age; public void setAge(int a) { age = a; } //或者用this // public void setAge(int age) { // this.age = age; // } public int getAge() { return age; } } //使用 public class hello { public static void main(String args[]) { Phone p = new Phone(); p.setAge(12); System.out.println(p.getAge()); } }
静态成员变量被所有对象共享,可用类名调用。局部变量不能被声明为 static 变量。
public class Phone { static int price;//成员变量在堆内存 } //使用 public class hello { public static void main(String args[]) { Phone.price=4; //fun();会报错,静态方法只能访问静态方法或变量。 } public fun(){}; }
静态方法只能访问静态变量和方法。非静态方法都可以访问。
静态方法中不能使用 this 关键字,因为在静态方法的局部变量表中并不存在this变量。
static可以修饰什么?
Java类中包含了成员变量、方法、构造器、初始化块和内部类(包括接口、枚举)5种成员。
static关键字可以修饰成员变量、方法、初始化块和内部类,不能修饰构造器。
static访问规则:
被static修饰的成员先于对象存在,所以又称为类变量。
类成员不能访问实例成员,即静态不能访问非静态,静态中也没有this关键字。this是随着对象的创建存在的。
类加载过程中类变量是怎样创建的?
在类加载过程中的链接-准备阶段,JVM会给类变量赋零值,初始化阶段为类变量赋初值,执行静态代码块。
类加载过程:加载、链接(验证、准备、解析)、初始化。这个过程是在类加载子系统完成的。
加载:生成类的Class对象。
- 通过一个类的全限定名获取定义此类的二进制字节流(即编译时生成的类的class字节码文件)
- 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构。包括创建运行时常量池,将类常量池的部分符号引用放入运行时常量池。
- 在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类各种数据的访问入口。注意类的class对象是运行时生成的,类的class字节码文件是编译时生成的。
链接:将类的二进制数据合并到JRE中。该过程分为以下3个阶段:
- 验证:确保代码符合JAVA虚拟机规范和安全约束。包括文件格式验证、元数据验证、字节码验证、符号引用验证。
- 文件格式验证:验证字节码文件是否符合规范。
- 魔数:是否魔数0xCAFEBABE开头
- 版本号:版本号是否在JVM兼容范围
- 常量类型:类常量池里常量类型是否合法
- 索引值:索引值是否指向不存在或不符合类型的常量。
- 元数据验证:元数据是字节码里类的全名、方法信息、字段信息、继承关系等。
- 标识符:验证类名接口名标识符有没有符合规范
- 接口实现方法:有没有实现接口的所有方法
- 抽象类实现方法:有没有实现抽象类的所有抽象方法
- final类:是不是继承了final类。
- 指令验证:主要校验类的方法体,通过数据流和控制流分析,保证方法在运行时不会危害虚拟机安全。
- 类型转换:保证方法体中的类型转换是否有效。例如把某个类强转成没继承关系的类
- 跳转指令:保证跳转指令不会跳转到方法体以外的字节码指令上;
- 保证任意时刻操作数栈的数据类型与指令代码序列都能配合工作。
- 符号引用验证:确保后面解析阶段能正常执行。
- 类全限定名地址:验证类全限定名是否能找到对应的类字节码文件
- 引用地址:引用指向地址是否存在实例
- 引用权限:是否有权引用
- 准备:为类变量(即static变量)分配内存并赋零值。
- 解析:将方法区-运行时常量池内的符号引用(类的名字、成员名、标识符)转为直接引用(实际内存地址,不包含任何抽象信息,因此可以直接使用)。
初始化:类变量赋初值、执行静态语句块。
public abstract class Animal{ abstract void m(); //抽象方法 abstract void n(); //抽象方法 } class Dog extends Animal{ //非抽象子类 //实现父类所有抽象方法 void m(){ ......... } void n(){ ......... } } abstract class Cat extends Animal{}//抽象子类,不需要重写父抽象类的抽象方法
设计目的:
相同点:
不同点:
①this指代当前对象的引用。通过this可以获取当前对象中的成员变量、方法。
常可以用于方法的形参与成员变量同名时进行区分,
public class Phone { private int age; //如果这个方法是static的,则会报错。 //因为this指向当前对象的引用,不是指向当前类的引用。static方法是类方法,类方法不创建对象就可以通过类名调用。 public void setAge(int age) { this.age = age; } public int getAge() { return age; } }
②this指代构造方法的调用
package package1; public class Dog { private int age; public Dog() { System.out.println("无参构造方法。。"); } public Dog(int age) { //先执行一次带参构造方法 this(); System.out.println("带参构造方法,年龄:"+age); this.age = age; } public static void main(String[] args) { //带参构造方法创建对象 Dog dog = new Dog(23); } }
this指代构造方法的调用时,有以下几点需要注意:
①指代仅包括父类特征的当前对象的引用。可以看做是父类的引用。
通过super可以获取当前对象的父对象的成员变量、方法。
示例:
以age变量为例,父类的age变量是3,子类的age变量是4,子类方法的age局部变量是5。
public class Animal { public String name; int age = 3; public void eat() { //吃东西方法的具体实现 System.out.println("吃东西"); } } //子类 public class Dog extends Animal{ int age=4; public void show(){ int age=5; //输出5。访问上一行的局部变量 System.out.println(age); //4。通过this访问当前对象的成员变量age System.out.println(this.age); //3。通过super访问当前对象的父对象的成员变量age System.out.println(super.age); } }
②指代父类构造方法的调用
通过super(),可以调用父类的各个构造方法。
示例:
public class Animal { public String name; int age = 3; public void eat() { //吃东西方法的具体实现 System.out.println("吃东西"); } public Animal(String name) { System.out.println("动物类带参构造方法"+name); this.name = name; } public void sleep() { //睡觉方法的具体实现 } } public class Dog extends Animal { public Dog(int age,String name) { //调用父类的带参构造方法 super("小白"); System.out.println("带参构造方法"+age+","+name); } public static void main(String[] args) { new Dog(23,"小黑"); } }
结果:
this和super的区别:
注意:子类所有无参、带参构造方法靠前行都会隐式或显式地加super()。this()和super()方法不能显式的共存,但可以隐式的共存,且都只能显式地出现在构造方法的靠前行。
面向对象(Object-Oriented,简称 OOP)是一种程序设计的范式,它基于对象的概念,将数据和操作数据的行为封装在对象中,以模拟现实世界的问题和解决方案。
核心概念:
总结起来就是一句话:类是对象的抽象,对象是类的具体实现。
示例:
例如我声明一个Person类,有名字和年龄两个属性;
然后我就可以通过这个类创建两个对象,分别是Alice和Bob,他们的年龄不一样。
public class Person { // 成员变量 String name; int age; // 构造方法 public Person(String name, int age) { this.name = name; this.age = age; } // 成员方法 public void work() { System.out.println(name + " is working."); } public void study() { System.out.println(name + " is studying."); } }
public class Test { public static void main(String[] args) { // 创建 Person 类的对象 Person person1 = new Person("Alice", 25); Person person2 = new Person("Bob", 30); // 调用对象的方法 person1.work(); person2.study(); } }
结果:
Alice is working. Bob is working.
内部类:在一个类中定义类。
分为:
匿名内部类:一个继承了其他类或者实现了其他接口的子类对象。
内部类可以直接访问外部类私有、公有成员。外部类要访问内部类成员要创建对象。
public class Outer { int num=10; //定义成员内部类:在成员位置定义类 public class Inner{ //一般private,用外部类访问内部类更安全 public void show(){ System.out.println("innershow"+num); //内部类可直接访问外部类成员。 } } //外部类访问成员内部类 public void fun(){ Inner in=new Inner(); in.show(); } //局部内部类:在成员方法内定义类 public void fun2(){ class Inner2{ //成员内部类不能public和private void show(){ System.out.println("成员内部类"); } } Inner2 in=new Inner2(); in.show(); } //匿名内部类:在成员方法内定义子类,实现接口或继承抽象类 public void fun3(){ Phone p=new Phone() { //自己写的Phone接口 @Override public void show() { System.out.println("实现接口的匿名内部类"); } }; //注意是一个语句有分号 p.show(); } } //内部类创建对象(一般内部类私有,使用外部类访问内部类) public class Test { public static void main(String[] args) { Outer.Inner i=new Outer().new Inner(); i.show(); } }
概述:
详细:
使用 new
关键字:
Person person1 = new Person();
这是最常见的创建对象的方式,使用 new
关键字直接调用类的构造方法。
反射:
通过反射机制,可以在运行时获取类的信息,动态创建对象。
Class<?> clazz = Person.class; Person person2 = (Person) clazz.newInstance();
对象的 clone()
方法:
要实现克隆,类必须实现 Cloneable
接口,并重写 clone()
方法。这种方式可以实现对象的浅拷贝或深拷贝。
public class Person implements Cloneable { // 其他类定义 @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } } // 创建对象 Person person3 = (Person) person1.clone();
反序列化:
通过反序列化,可以将对象的状态从持久性存储中重新创建出来。需要让类实现 Serializable
接口,并使用 ObjectInputStream
类的 readObject()
方法从文件或网络中反序列化对象。
// 反序列化对象 try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("person.ser"))) { Person person4 = (Person) ois.readObject(); } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); }
方法(Method):一组执行特定任务的代码块。它可以在类中定义一次,然后在本方法、其他方法中被多次调用。
作用:提高代码的可读性和可维护性。
例如小明要吃多顿饭,每顿饭吃的食物不同,我们可以将“吃饭”这个行为抽象成方法,从而减少代码量:
public class Test { private static String name = "小明"; /** * 吃饭方法 * * @param food */ public static void eat(String food) { System.out.println("====="); System.out.println(name + " is eating " + food); System.out.println("====="); } public static void main(String[] args) { eat("西瓜"); eat("白菜"); eat("可乐"); eat("番茄"); } }
结果:
如果不用方法,将需要多写很多代码,可读性变差:
public class Test { private static String name = "小明"; /** * 吃饭方法 * * @param food */ public static void eat(String food) { System.out.println("====="); System.out.println(name + " is eating " + food); System.out.println("====="); } public static void main(String[] args) { System.out.println("====="); System.out.println(name + " is eating 西瓜" ); System.out.println("====="); System.out.println("====="); System.out.println(name + " is eating 白菜"); System.out.println("====="); System.out.println("====="); System.out.println(name + " is eating 可乐"); System.out.println("====="); System.out.println("====="); System.out.println(name + " is eating 番茄"); System.out.println("====="); } }
在 Java 中,方法的定义包括以下几个要素:
public
、private
、protected
或默认(包内可见)。void
。// 方法的定义 public int add(int num1, int num2) { int sum = num1 + num2; return sum; }
在程序中,可以通过方法名和参数列表来调用方法。方法调用是程序执行的一个重要步骤,它使代码更具可重用性。
// 方法的调用 int result = add(5, 3); System.out.println("Result: " + result);
方法可以返回一个值,也可以是 void
,表示不返回任何值。如果方法返回值,必须使用 return
语句返回对应类型的值。
// 方法的返回值 public int add(int num1, int num2) { int sum = num1 + num2; return sum; }
重载(Overload):指一个类中可以有多个方法具有相同的方法名,但这些方法的参数类型不同、个数不同、顺序不同。
注意:方法返回值和访问修饰符可以不同。
示例:
public class hello { public static void main(String args[]) { f(); } public static void f() { System.out.println("3f"); } public static void f(int a) { //重载,注意返回值同,参数不同 System.out.println(a); } // 下面两个注释的方法就不是重载,会报错 // public static int f(){ //返回值不同 // System.out.println("hello"); // } // void f(){ //修饰符不同 // return 666; // } }
示例:求和的数学类:
public class MathOperations { // 求和方法,接受两个整数参数 public int sum(int num1, int num2) { return num1 + num2; } // 重载的求和方法,接受三个整数参数 public int sum(int num1, int num2, int num3) { return num1 + num2 + num3; } // 重载的求和方法,接受两个双精度浮点数参数 public double sum(double num1, double num2) { return num1 + num2; } public static void main(String[] args) { // 创建 MathOperations 对象 MathOperations math = new MathOperations(); // 使用不同的方法进行求和 int result1 = math.sum(5, 10); int result2 = math.sum(3, 7, 12); double result3 = math.sum(2.5, 3.5); // 打印结果 System.out.println("Sum of two integers: " + result1); System.out.println("Sum of three integers: " + result2); System.out.println("Sum of two doubles: " + result3); } }
重载要求发生在同一个类中,多个方法之间方法名相同且参数列表不同。
重载与返回类型和访问修饰符无关。方法名相同返回类型不同会直接报错。
重写发生在父类子类或接口实现类中,若子类方法想要和父类方法构成重写关系,则它的方法名、参数列表、返回类型必须与父类方法相同。
重写时:
思考:构造方法能不能重写?
答案:构造方法不能重写。
Java 5 以后引入了可变参数(Varargs),允许方法接受可变数量的参数。可变参数在方法的参数列表中使用省略号 ...
表示。
// 可变参数的定义 public void printNumbers(int... numbers) { for (int num : numbers) { System.out.print(num + " "); } }
注意:
- 可变参数必须是方法的最后一个参数。例如void (int a,int... b)正确,而void (int... a,int b)会报错。
- 一个方法最多只能有一个可变参数。
- 可变参数在方法内部被当作数组处理。
示例:求和的数学类:
public class MathOperations { // 求和方法(整数数组) public static int sum(int[] numbers) { int result = 0; for (int num : numbers) { result += num; } return result; } // 求和方法(可变参数) public static int sum(int... numbers) { int result = 0; for (int num : numbers) { result += num; } return result; } // 求和方法(双精度浮点数数组) public static double sum(double[] numbers) { double result = 0; for (double num : numbers) { result += num; } return result; } // 求和方法(可变参数) public static double sum(double... numbers) { double result = 0; for (double num : numbers) { result += num; } return result; } public static void main(String[] args) { // 示例:整数求和 int[] intArray = {1, 2, 3, 4, 5}; int intSum = sum(intArray); System.out.println("Integer Sum: " + intSum); // 示例:可变参数整数求和 int varArgsIntSum = sum(1, 2, 3, 4, 5); System.out.println("VarArgs Integer Sum: " + varArgsIntSum); // 示例:双精度浮点数求和 double[] doubleArray = {1.1, 2.2, 3.3, 4.4, 5.5}; double doubleSum = sum(doubleArray); System.out.println("Double Sum: " + doubleSum); // 示例:可变参数双精度浮点数求和 double varArgsDoubleSum = sum(1.1, 2.2, 3.3, 4.4, 5.5); System.out.println("VarArgs Double Sum: " + varArgsDoubleSum); } }
构造方法是一种特殊的方法,与类同名,没有返回类型。
每次创建对象时,都会默认执行一次构造方法。
特点:
简单的构造方法:
public class Phone { public Phone() { //创建对象时会直接运行构造方法,输出hhh System.out.println("hhh"); } }
示例:
public class Car { private String brand; private String model; private int year; // 默认构造方法 public Car() { System.out.println("Default constructor called."); } // 带参数的构造方法 public Car(String brand, String model, int year) { this.brand = brand; this.model = model; this.year = year; System.out.println("Parameterized constructor called."); } // Getter 和 Setter 方法(省略) public static void main(String[] args) { // 使用默认构造方法创建对象 Car defaultCar = new Car(); // 使用带参数的构造方法创建对象 Car customCar = new Car("Toyota", "Camry", 2022); } }
接口:对行为的抽象,如吃饭类、睡觉类。
接口的特点:
public interface Phone { public static final int price=4; //修饰符可省略 public abstract void show(); //修饰符可省略 } public class PhoneImpl implements Phone{ public void show(){ //必须是public,否则报错 System.out.println("hello"); } }
抽象类:对事物的抽象,如动物类,小狗类。
抽象类包含抽象方法的类,它不能被实例化,通常用于作为其他类的基类。
特点:抽象类可以包含抽象方法、具体方法、字段和构造方法。
使用场景:
示例:
public abstract class Animal { private String name; public Animal(String name) { this.name = name; } // 抽象方法 public abstract void makeSound(); // 具体方法 public void eat() { System.out.println(name + " is eating"); } } public class Dog extends Animal { public Dog(String name) { super(name); } // 实现抽象方法 @Override public void makeSound() { System.out.println("Woof! Woof!"); } } public class Cat extends Animal { public Cat(String name) { super(name); } // 实现抽象方法 @Override public void makeSound() { System.out.println("Meow! Meow!"); } }
设计目的:
相同点:
不同点:
面向对象的三大基本特征是:封装、继承、多态。
封装:通过private修饰符,将类的某些信息隐藏在类的内部,不允许外部程序直接访问。
封装将对象的状态和行为包装在一个类中并对外界隐藏实现的细节,可以通过访问修饰符控制成员的访问权限,让外部程序通过该类提供的方法来实现对内部信息的操作和访问。
示例:将价格、年龄等信息封装到小狗类中,外部不能直接访问,只能通过get和set方法访问。
public class Dog{ int price;//成员变量在堆内存 private int age; public void setAge(int age) { this.age = age; } public int getAge() { return age; } }
优点:安全性和复用性。
继承:继承是指一个类通过从另一个类继承其属性和方法。这使得子类具有其父类的行为和属性,同时可以扩展或修改这些行为和属性以满足特定的需求。
优点:提高代码复用性,维护性,实现代码共享。
缺点:高耦合性,有侵入性,父类改变子类也会改变。
特点:子类拥有父类非 private 的属性、方法。可以拥有自己的属性和方法,即子类可以对父类进行扩展。可以用自己的方式实现父类的方法。
注意: Java 不支持多继承,但支持多重继承。
单继承是子类只能继承一个父类。相对于实现是一个子类可以实现多个接口。所以java是单继承多实现。
子类所有构造方法会默认先运行super();
public class Animal { public String name; int age=3; public Animal(String name, int age) { //初始化属性值 } public void show() { //吃东西方法的具体实现 } public void sleep() { //睡觉方法的具体实现 } } //子类 public class Penguin extends Animal{ int age=4; public Penguin(){ super(); //不写也会默认运行 } @Override //注解重写,检查重写是否正确。例如修饰符(子类重写方法的访问权限要≥父类)、函数名错误。 public void show(){ //重写父类中show(),如果去掉public会报错。想再用super.show(); int age=5; System.out.println(age); //5,子类可以访问父类非私有成员变量 System.out.println(this.age);//4 System.out.println(super.age);//3 } }
使用 implements 关键字可以变相的使java具有多继承的特性
public interface A { public void eat(); public void sleep(); } public interface B { public void show(); } public class C implements A,B { }
多态:同一行为具有多个不同的表现形式或形态。例如同一个接口,使用不同的实例而执行不同操作。使程序更灵活、易于扩展。
多态最常用的是接口引用指向实现类对象,这样当之后程序需要更新时候,只需要修改new后面的实现类即可,左边就不用修改了,从而降低耦合。
简单一句话,多态是:接口引用指向实现类对象。
在Spring框架中,甚至可以通过配置或注解连实例化的new也不用了,从而更高层次的降低耦合。
示例:
List<String> a=new ArrayList<String>();
这样写的话,等号右边换成Vector 或 LinkedList 时,就可以很少修改代码量,降低耦合。
向上转型:父类引用指向子类对象。编译看左边(调用子类特有变量、方法会报错),运行看右边(优先运行子类重写后的方法)。
Animal a=new Cat();
向下转型: 子类引用指向父类对象。编译看右边,运行看右边。
public static void main(String[] args) { Animal a = new Cat(); // 向上转型 a.eat(); // 调用的是 Cat 的 eat Cat c = (Cat)a; // 向下转型 c.work(); // 调用的是 Cat 的 work }
什么是String?
String是一个类,用于存储字符串,内部封装了一系列用于操作字符串的方法,底层是final修饰的char数组。
JDK9开始,为了节省内存,进而减少垃圾回收次数,String底层由char数组改成了byte[]。
Java的String和c++的string区别:
- java中字符串一个汉字长度是1;
- c++中字符串,一个汉字长度是2.
创建字符串的两种方式:
创建字符串有两种方式,一种是使用字符串直接量,另一种是使用new+构造器。采用new的方式会多创建出一个对象来,占用了更多的内存 ,所以建议采用直接量的方式来创建字符串。
// String 直接创建 String s1 = "Runoob"; // String 对象创建 String s2 = new String("Runoob"); // 引用赋值的方法创建字符串 String s3 = s1;
通过索引访问字符:charAt()
String str = "Hello, World!"; char firstChar = str.charAt(0); System.out.println("First Character: " + firstChar);
访问子串:substring()
String str = "0123456789ABCDEFG"; // 提取索引 7 到 11 的子串 String substring = str.substring(7, 12); System.out.println("Substring: " + substring);
遍历字符串每个字母:
- 使用for循环遍历字符串:
String str = "hello world"; for (int i = 0; i < str.length(); i++) { char c = str.charAt(i); System.out.println(c); }
- 使用增强for循环遍历字符串:
String str = "hello world"; for (char c : str.toCharArray()) { System.out.println(c); }
- 使用while循环遍历字符串:
String str = "hello world"; int i = 0; while (i < str.length()) { char c = str.charAt(i); System.out.println(c); i++; }
- 使用Iterator遍历字符串:
String str = "hello world"; Iterator<Character> iterator = str.chars().mapToObj(c -> (char) c).iterator(); while (iterator.hasNext()) { char c = iterator.next(); System.out.println(c); }
结果:
字符串可以通过“+”号拼接,拼接过程中可以将数字型转为字符串。
示例:
String s1,s2; //拼接字符串、数字、字符 s1="我的名字是 ".concat("s1"); s2="s2:"; s2+="abc"; //字符串拼接数字,数字会转成字符串 s2+=123; s2+='c'; System.out.println(s1); System.out.println(s2);
结果:
字符串连接字符数组:
//连接字符串 String s= "Hello"; s= s+ ", World!"; //连接字符数组 char[] greeting = {'H', 'e', 'l', 'l', 'o'}; char[] suffix = {',', ' ', 'W', 'o', 'r', 'l', 'd', '!'}; char[] result = new char[greeting.length + suffix.length]; System.arraycopy(greeting, 0, result, 0, greeting.length); System.arraycopy(suffix, 0, result, greeting.length, suffix.length); System.out.println(result);
在 Java 中,可以使用 substring() 方法获取字符串的子串:
substring(int beginIndex): 返回从指定索引开始到字符串末尾的子串。
String originalString = "Hello, World!"; String substring = originalString.substring(7); // 从索引 7 开始到字符串末尾的子串 System.out.println(substring); // 输出结果为 "World!"
substring(int beginIndex, int endIndex): 返回从指定索引开始到指定索引结束的子串(不包括 endIndex 处的字符)。
String originalString = "Hello, World!"; String substring = originalString.substring(7, 12); // 从索引 7 开始到索引 12 结束的子串 System.out.println(substring); // 输出结果为 "World"
注意:
因为String是不可变的,对字符串的任何操作都会返回一个新的字符串。所以substring 方法其实是创建了一个字符子串,而不是修改了原始字符串。
Java中字符串没有直接根据索引删除的方法,所以删除字符串中指定索引的字母时,可以通过子串substring()删除。
示例:删除字符串中下标是6的字母
String str = "Hello, World!"; String newStr = str.substring(0, 5) + str.substring(7); System.out.println("Modified String: " + newStr);
结果:
在Java中, 可以使用replaceAll()方法替换字符中的字符:
replaceAll(String regex, String replacement): 使用给定的替换字符串替换输入字符串中所有匹配正则表达式的部分。
String originalString = "Hello, World!"; String replacedString = originalString.replaceAll("o", "0"); System.out.println(replacedString); // 输出结果为 "Hell0, W0rld!"
在上面的例子中,所有的字母 "o" 都被替换为数字 "0"。
replaceAll(String regex, Function<MatchResult, String> replacer): 使用给定的 Function 替换输入字符串中所有匹配正则表达式的部分。该方法允许更加灵活的替换逻辑,并且可以基于匹配的结果进行自定义的替换。
String originalString = "Hello, World!"; String replacedString = originalString.replaceAll("o", match -> match.group().toUpperCase()); System.out.println(replacedString); // 输出结果为 "HellO, WOrld!"
在上面的例子中,将匹配到的字母 "o" 替换为大写的 "O"。
练习:将字符串中所有的 "Java" 都被替换为 "Python":
String original = "Hello, Java! Java is fun. Java is powerful."; // 将所有的 "Java" 替换为 "Python" String modifiedString = original.replaceAll("Java", "Python"); System.out.println("Original String: " + original); System.out.println("Modified String: " + modifiedString);
可以通过length()方法获取字符串的长度。
示例:获取字符串的长度
public class Test { public static void main(String[] args) { String text = "Java Programming"; // 返回字符串的长度 int length = text.length(); System.out.println(length); } }
结果:
字符串内容的互相比较一般用equals()方法。
Java中所有类都直接或间接继承了 Object 类,在 Object 类中,equals() 方法是 Java 中用于比较两个对象的引用是否相等(即内存地址是否相同)。
在 String 类中,equals() 方法被重写,用于比较两个字符串的内容是否相等。
示例:比较两个字符串内容是否相等
public class Test { public static void main(String[] args) { String str1 = "Hello"; String str2 = "Hello"; // 比较字符串内容是否相等 boolean isEqual = str1.equals(str2); System.out.println(isEqual); } }
结果:
注意:尽量不要用==进行比较,==比较的是地址。
String底层是常量池,使用享元设计模式,新创建的字符串会维护在常量池,下次再创建这个字符串,就直接从常量池取。
验证:
//比较地址 //只要new,就在堆内存开辟空间。直接赋值字符串在常量池里。 //常量池里无“hello”对象,创建“hello”对象,str1指向常量池“hello”对象。 String str1 = "hello"; //先检查字符串常量池中有没有"hello",如果字符串常量池中没有,则创建一个,然后 str1 指向字符串常量池中的对象,如果有,则直接将 str1 指向"hello"; //常量池里有“hello”对象,str2直接指向常量池“hello”对象。 String str2 = "hello"; //堆中new创建了一个对象。假如“hello”在常量池中不存在,Jvm还会常量池中创建这个对象“hello”。 String str3 = new String("hello"); String str4 = new String("hello"); //下面输出true,因为str1和str2指向的是常量池中的同一个内存地址 System.out.println(str1 == str2); //下面输出false,str1常量池旧地址,str3是new出的新对象,指向一个全新的地址 System.out.println(str1 == str3); //下面输出false,因为它们引用不同 System.out.println(str4 == str3); //比较内容 //下面输出true,因为String类的equals方法重写过,比较的是字符串值 System.out.println(str4.equals(str3));
结果:
示例:String源码中重写的equals()方法:
split() 方法是 Java 中 String 类的一个方法,用于将字符串分割成字符串数组,根据给定的正则表达式作为分隔符。该方法有两个重载的版本:
split(String regex): 使用给定的正则表达式作为分隔符,将字符串分割为字符串数组。
String sentence = "Hello, World! How are you?"; String[] words = sentence.split(" "); // 使用空格作为分隔符 for (String word : words) { System.out.println(word); }
上述代码会输出每个单词:
Hello, World! How are you?
split(String regex, int limit): 使用给定的正则表达式作为分隔符,将字符串分割为字符串数组,限制分割的次数。
String sentence = "apple,orange,banana,grape"; String[] fruits = sentence.split(",", 2); // 使用逗号作为分隔符,限制分割次数为 2 for (String fruit : fruits) { System.out.println(fruit); }
上述代码会输出:
apple orange,banana,grape
需要注意的是,split() 方法的参数是正则表达式,因此在传入正则表达式时,可能需要注意转义字符的使用。例如,如果要以点号 . 作为分隔符,由于点号在正则表达式中有特殊含义,需要使用双反斜杠 \ 进行转义。
String sentence = "one.two.three"; String[] parts = sentence.split("\\."); // 使用点号作为分隔符,需要转义 for (String part : parts) { System.out.println(part); }
上述代码会输出:
one two three
总的来说,split() 方法是在字符串处理中常用的方法,可以根据需要灵活地将字符串分割成多个部分。
//根据指定字符串分割 String s="1,2,3"; String[] arr=s.split(","); for (String s1 : arr) { System.out.println(s1); }
结果:
String text = "Java Programming"; String lowerCase = text.toLowerCase(); String upperCase = text.toUpperCase();
1. 字符串转整数:
使用 Integer.parseInt()
方法:
String strNumber = "123"; int intValue = Integer.parseInt(strNumber); System.out.println("Parsed Integer: " + intValue);
使用 Integer.valueOf()
方法:
String strNumber = "123"; Integer integerValue = Integer.valueOf(strNumber); System.out.println("Integer Value: " + integerValue);
2. 整数转字符串:
使用 String.valueOf()
方法:
int intValue = 123; String strNumber = String.valueOf(intValue); System.out.println("String Value: " + strNumber);
使用 Integer.toString()
方法:
int intValue = 123; String strNumber = Integer.toString(intValue); System.out.println("String Value: " + strNumber);
数组转字符串:
Arrays.toString(数组名)
示例:
int[] numbers = {1, 2, 3, 4, 5}; //输出:[1, 2, 3, 4, 5] System.out.println(Arrays.toString(numbers)); //直接输出数组会输出地址:[I@5b444398 System.out.println(a);
字符串转数组:
方法名:stringToIntArray(String str)、stringToCharArray(String str)等
示例代码:
// 字符串转整型数组 int[] intArray = stringToIntArray("1 2 3 4 5"); // 字符串转字符数组 char[] charArray = stringToCharArray("Hello");
5.1.2.13 格式化数字为字符串
double number = 123.456789; // 格式化浮点数为字符串,保留两位小数 String formattedString = String.format("%.2f", number); System.out.println(formattedString);
结果:
常量池:Java虚拟机有一个常量池机制,它会直接把字符串常量放入常量池中,从而实现复用。
Java字符串存储原理:
String为什么不可被继承?
因为String类底层的数组是由final修饰的,所以String类不可被继承。
String字符串为什么不可被变?
因为String底层char类型的value数组是private final修饰的。
不可变的优点:因为压根不会被改,所以线程安全、节省空间、效率高。
答案:一个或两个。
原因:
首先,new string 这边由于 new 关键字,所以这边肯定会在堆中直接创建一个字符串对象。
其次,如果字符串常量池中不存在 "abc"(通过equals比较)这个字符串的引用,则会在字符串常量池中创建一个字符串对象。如果已存在则不创建。注意这边说的在字符串常量池创建对象,最终对象还是在堆中创建,字符串常量池只放引用。
String拼接字符串后原字符串还存在于内存中,浪费内存。
StringBuffer 类的对象能够被多次的修改,并且不产生新的未使用对象,所以涉及到字符串拼接,优先用StringBuffer 。
//构造 StringBuilder ***= new StringBuilder("abc"); //String转StringBuilder ***.append("d").append("e").append("f"); //StringBuilder转String String s=***.toString(); //反转 ***.reverse();
String:不可变字符序列,效率低,但是复用率高、线程安全。
不可变是指String对象创建之后,直到这个对象销毁为止,对象中的字符序列都不能被改变。
复用率高是指String类型对象创建出来后归常量池管,可以随时从常量池调用同一个String对象。StringBuffer和StringBuider在创建对象后一般要转化成String对象才调用。
StringBuffer和StringBuilderStringBuffer和Stringbuilder都是字符序列可变的字符串,方法也一样,有共同的父类AbstractStringBuilder。
Scanner 类用于从各种输入源(例如控制台)中获取基本数据类型和字符串,如 int、double、String 等。常用于从控制台、文件等读取数据。
构造方法:
常用方法:
示例:
从控制台输入整数、浮点数、字符串,并在控制台打印。
import java.util.Scanner; public class Test { public static void main(String[] args) { // 创建 Scanner 对象,关联 System.in(标准输入流) Scanner scanner = new Scanner(System.in); // 从控制台读取整数 System.out.print("Enter an integer: "); int intValue = scanner.nextInt(); System.out.println("You entered: " + intValue); // 从控制台读取浮点数 System.out.print("Enter a double: "); double doubleValue = scanner.nextDouble(); System.out.println("You entered: " + doubleValue); // 从控制台读取字符串 System.out.print("Enter a string: "); String stringValue = scanner.next(); System.out.println("You entered: " + stringValue); // 关闭 Scanner 对象 scanner.close(); } }
结果:
在 Java 中,Object 类是所有类的根类,所有其他类都直接或间接地继承自 Object 类。
Object 类定义了一些基本的方法,这些方法可以被所有对象继承和使用:
toString() 方法: 返回对象的字符串表示。再不重写的情况下,toString() 返回的是对象的类名和散列码的十六进制表示。
public class MyClass { private int value; public MyClass(int value) { this.value = value; } //自定义转字符串逻辑 @Override public String toString() { return "MyClass{" + "value=" + value + '}'; } } // 使用 toString() 方法 MyClass obj = new MyClass(42); // toString将根据自己重新的逻辑返回字符串,而不是返回对象的类名和散列码的十六进制表示 System.out.println(obj.toString()); // 输出结果为 "MyClass{value=42}"
equals(Object obj) 方法: 比较对象是否相等。默认情况下,equals() 方法比较的是对象的引用(即内存地址),重写后可以自定义比较逻辑。
public class Person { private String name; private int age; // ... 其他代码 ... //自定义比较逻辑 @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null || getClass() != obj.getClass()) return false; Person person = (Person) obj; return age == person.age && Objects.equals(name, person.name); } }
hashCode() 方法: 返回对象的散列码。hashCode() 方法的默认实现返回对象的内存地址的散列码。通常情况下,如果 equals() 方法被覆盖,那么也应该同时覆盖 hashCode() 方法。
@Override public int hashCode() { return Objects.hash(name, age); }
getClass() 方法: 返回对象的类。该方法基于反射,返回类的 Class 对象,该对象包含有关对象的类的信息。
MyClass obj = new MyClass(42); Class<?> clazz = obj.getClass(); System.out.println(clazz.getName()); // 输出结果为 "MyClass"
clone() 方法: 创建并返回对象的拷贝。要实现 clone() 方法,类必须实现 Cloneable 接口。
public class CloneableClass implements Cloneable { private int value; public CloneableClass(int value) { this.value = value; } @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } }
finalize() 方法: 在对象被垃圾收集器回收之前调用。通常不推荐使用 finalize() 方法,因为它的行为是不确定的,并且可能导致一些问题。
@Override protected void finalize() throws Throwable { // 执行清理工作 // ... super.finalize(); }
JVM垃圾回收的可达性分析算法有用到Object类的finalize() 方法。
可达性分析算法:
以根对象***(GC Roots)的每个跟对象为起始点,根据引用关系向下搜索,将所有与GC Roots直接或间接有引用关系的对象在对象头的Mark Word里标记为可达对象,即不需要回收的有引用关系对象。搜索过程所走过的路径称为“引用链” 。
GC Roots:即GC根节点***,是一组必须活跃的引用。可作为GC Roots的对象:
非可达对象被回收需要两次标记:
finalize()方法:
finalize()方法是对象逃脱死亡命运的最后一次机会,需要注意的是,任何一个对象的finalize()方法都只会被系统自动调用一次,如果对象面临下一次回收,它的finalize()方法不会被再次执行。
另外,finalize()方法的运行代价高昂,不确定性大,无法保证各个对象的调用顺序,如今已被官方明确声明为不推荐使用的语法。
用途:
为什么重写equals()就要重写hashcode()?
因为二者之间有两个约定,相等对象的哈希码也要相等。
所以equals()方法重写时,通常也要将hashCode()进行重写,使得这两个方法始终满足相关的约定。 例如HashSet排序机制底层就是通过计算哈希码进行排序的,如果只重写equals()将达不到根据哈希码排序的效果。
如果两个对象相等,它们必须有相同的哈希码;但如果两个对象的哈希码相同,他们却不一定相等。
哈希碰撞:由于哈希码是一个有限的整数,因此有可能会出现不同的对象计算出相同的哈希码的情况。 例如某类里有两个int型成员变量,重写后equals()比较两个变量是否相等,hashcode()是两个变量的和,A对象变量是2和3,B对象变量是1和4,加起来都是5,它们哈希码相等而不equal。
System 类是 Java 标准库中的一个工具类,提供了与系统相关的一些操作。
它包含一些静态方法和字段,用于访问系统属性、标准输入输出、垃圾回收等。
常用方法和字段:
PrintStream、PrintStream等IO流详细看后文的I/O流。
out 字段: 是 PrintStream 类的一个实例,用于标准输出。可以使用 System.out.println() 来向标准输出打印信息。
System.out.println("Hello, World!"); // 将 "Hello, World!" 输出到标准输出
err 字段: 是 PrintStream 类的一个实例,用于标准错误输出。可以使用 System.err.println() 来向标准错误输出打印错误信息。
System.err.println("This is an error message."); // 将错误信息输出到标准错误输出
in 字段: 是 InputStream 类的一个实例,用于标准输入。可以使用 Scanner 或 BufferedReader 等类从标准输入读取用户输入的信息。
Scanner scanner = new Scanner(System.in); System.out.print("Enter your name: "); String name = scanner.nextLine();
currentTimeMillis() 方法: 返回当前时间与1970年1月1日午夜之间的毫秒数。通常用于测量代码的执行时间。
long startTime = System.currentTimeMillis(); // 执行一些操作 long endTime = System.currentTimeMillis(); long elapsedTime = endTime - startTime;
arraycopy(Object src, int srcPos, Object dest, int destPos, int length) 方法: 用于复制数组。
int[] sourceArray = {1, 2, 3, 4, 5}; int[] destinationArray = new int[5]; System.arraycopy(sourceArray, 0, destinationArray, 0, sourceArray.length); // 现在 destinationArray 中包含了 sourceArray 的内容
getProperty(String key) 方法: 获取系统属性。可以用于获取一些系统相关的信息,比如操作系统类型、Java 版本等。
String osName = System.getProperty("os.name"); System.out.println("Operating System: " + osName);
exit(int status) 方法: 终止当前运行的 Java 虚拟机。status 参数是一个整数,通常用于指示程序的退出状态。非零值通常表示发生了错误。
System.exit(0); // 正常退出程序 System.exit(1); // 异常退出程序
gc() 方法: 强制调用垃圾回收器。虽然 Java 具有自动垃圾回收机制,但可以使用 System.gc() 来显式地触发垃圾回收。
System.gc(); // 强制调用垃圾回收器
练习:获取for循环十万次的运行时长。
public class Test { public static void main(String[] args) { long start=System.currentTimeMillis(); for(int i=0;i<100000;i++){} long end=System.currentTimeMillis(); System.out.println("for循环十万次的运行时长:"+(end-start)+"毫秒"); } }
结果:
Integer类是 Java 中用于表示整数的包装类,它提供了许多方法来对整数进行操作和转换。
相比于基本数据类型,包装类的优势:可以有更多的方法操作改数据。
//装箱:基本数据类型转为包装类 Integer a=Integer.valueOf(32); Integer a2=32; //拆箱:包装类转为基本数据类型 int b=a.intValue(); //数字转String String s=String.valueOf(12); //String转数字 int c=Integer.parseInt(s); //String转Integer转数字 int c=Integer.valueOf(s).intValue();
包装类:包装类的主要作用是用于便于操作基本数据类型,将基本数据类型转换为对象,让基本数据类型拥有对象的特征,例如封装方法、泛型(基本数据类型不能作为泛型参数)、反射。
自动装箱是指把一个基本类型的数据直接赋值给对应的包装类型;
自动拆箱是指把一个包装类型的对象直接赋值给对应的基本类型;
向上转型:布尔型外的基本数据类型在互相比较或运算时会向上转型:byte,short,char → int → long → float → double。原理是将低字节数的数据类型转换为高字节数的数据类型,可以保证数据精度不丢失。c语言转型顺序:char→short→int→long→float→double
包装类适用场景:
基本数据类型适用场景:
包装类和基本数据类型:直接通过==比较。
Integer integer = new Integer(3000); double b=3000; //true,相等 System.out.println(b==integer);
整型:
浮点型:
浮点包装类先都转为BigDecimal,再进行运算、比较。
浮点基本类型直接比较,要声明误差,两浮点数差值在此范围内,认为是相等的。
浮点数正确比较方法:
基本数据类型:
float a = 1.0f - 0.9f; float b = 0.9f - 0.8f; float diff = 1e-6f; if (Math.abs(a - b) < diff) { System.out.println("a、b相等"); }
包装类:
//为了保证高精度 //BigDecimal推荐入参为 String 的构造方法,或使用 BigDecimal 的 valueOf 方法 //valueOf方法内部其实执行了Double 的 toString,而 Double 的 toString 按 double 的实际能表达的精度对尾数进行了截断 BigDecimal a = new BigDecimal("1.0"); BigDecimal b = BigDecimal.valueOf(0.9); BigDecimal c = new BigDecimal("0.8"); BigDecimal x = a.subtract(b); BigDecimal y = b.subtract(c); if (x.equals(y)) { System.out.println("a、b相等"); }
享元模式:
Integer 内部有享元模式设计,【-128,127】范围内的数字会被缓存,使用自动装箱方式赋值时,Java默认通过valueOf()方法对127这个数字进行装箱操作,触发缓存机制,使a1和a2指向同一个内存地址。
Integer a = 127; Integer b = 127; Integer c = 128; Integer d = 128; System.out.println(a==b); //true System.out.println(c==d); //false
Arrays
类包含了一系列用于操作数组的静态方法。可以对数组排序、搜索、比较、转换、填充等。以下是 Arrays
类的一些常用方法:
sort(T[] a)
: 对数组进行升序排序。
int[] numbers = {5, 2, 8, 1, 6}; Arrays.sort(numbers);
sort(T[] a, Comparator<? super T> c)
: 使用指定的比较器对数组进行排序。
String[] names = {"John", "Alice", "Bob", "Eve"}; Arrays.sort(names, Comparator.reverseOrder());
binarySearch(T[] a, T key)
: 在已排序的数组中使用二分查找算法查找指定元素的索引。
int[] numbers = {1, 2, 3, 4, 5, 6}; int index = Arrays.binarySearch(numbers, 3);
equals(T[] a, T[] a2)
: 比较两个数组是否相等。
int[] arr1 = {1, 2, 3}; int[] arr2 = {1, 2, 3}; boolean isEqual = Arrays.equals(arr1, arr2);
fill(T[] a, T val)
: 使用指定的值填充数组的所有元素。
int[] numbers = new int[5]; Arrays.fill(numbers, 42);
toString(T[] a)
: 返回包含数组元素的字符串。
int[] numbers = {1, 2, 3, 4, 5}; String arrayString = Arrays.toString(numbers);
asList(T... a)
: 将数组转换为固定大小的列表。
String[] names = {"John", "Alice", "Bob"}; List<String> nameList = Arrays.asList(names);
copyOf(T[] original, int newLength)
: 复制指定长度的数组。
int[] numbers = {1, 2, 3, 4, 5}; int[] copiedNumbers = Arrays.copyOf(numbers, 3);
Date 类是 Java 中用于表示日期和时间的类,位于 java.util 包中。在 Java 8 及之前的版本中,Date 类是主要的日期时间处理类,但在 Java 8 引入了 java.time 包,推荐使用新的日期时间 API(java.time 包中的 LocalDate、LocalTime、LocalDateTime 等类)。尽管如此,我们仍然可以了解 Date 类的基本使用。
getTime()
: 返回自 1970 年 1 月 1 日 00:00:00 GMT 以来的毫秒数。toString()
: 返回日期对象的字符串表示。import java.util.Date; public class Test { public static void main(String[] args) { // 创建当前日期和时间的 Date 对象 Date currentDate = new Date(); // 输出当前日期和时间 System.out.println("Current Date and Time: " + currentDate); // 获取毫秒表示的时间戳 long timestamp = currentDate.getTime(); System.out.println("Timestamp: " + timestamp); // 通过时间戳创建 Date 对象 Date newDate = new Date(timestamp); System.out.println("New Date: " + newDate); } }
输出结果:
Date类的缺点:
SimpleDateformat类用于格式化和解析日期。
public static void main(String[] args) throws ParseException { //throws只是把异常抛出,延迟处理。用trycatch处理比较好 //Date转String Date d=new Date(); SimpleDateFormat sdf=new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss"); String str=sdf.format(d); System.out.println(str); //String转Date d=sdf.parse(str); System.out.println(d); }
日期格式模式:
SimpleDateFormat
使用一组模式字母来定义日期时间格式。以下是一些常见的模式字母:
y
: 年份(如 "yy" 表示年份的后两位,"yyyy" 表示完整的年份)。M
: 月份(1-12,"MM" 表示两位数字,"MMM" 表示缩写,"MMMM" 表示全名)。d
: 日期(1-31,"dd" 表示两位数字)。H
: 小时(0-23,"HH" 表示两位数字)。m
: 分钟(0-59,"mm" 表示两位数字)。s
: 秒钟(0-59,"ss" 表示两位数字)。
DateTimeFormatter 类是 Java 8 引入的日期时间 API(java.time 包)的一部分,用于格式化和解析日期时间对象。
特点:
常用方法:
1. 格式化日期时间:format(TemporalAccessor temporal)
: 格式化指定的日期时间对象。
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); LocalDateTime dateTime = LocalDateTime.now(); String formattedDateTime = formatter.format(dateTime);
2. 解析字符串为日期时间:parse(CharSequence text)
: 解析输入的文本,返回一个解析后的日期时间对象。
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); String dateString = "2023-12-01 15:30:00"; LocalDateTime parsedDateTime = formatter.parse(dateString, LocalDateTime::from);
3. 获取格式化/解析模式:toString()
: 获取当前格式化/解析器的模式字符串。
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); String pattern = formatter.toString(); // 返回 "yyyy-MM-dd HH:mm:ss"
代码示例:将当前时间格式化为字符串
import java.time.LocalDateTime; import java.time.format.DateTimeFormatter; public class Test { public static void main(String[] args) { // 创建 DateTimeFormatter 对象,指定日期时间格式模式 DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); // 获取当前日期时间 LocalDateTime currentDateTime = LocalDateTime.now(); // 格式化日期时间为字符串 String formattedDateTime = currentDateTime.format(formatter); System.out.println("Formatted Date and Time: " + formattedDateTime); } }
Math 类是数学工具类,提供了一系列用于执行基本数**算的静态方法。
基本使用:
public class Test { public static void main(String[] args) { // 基本数**算 double absoluteValue = Math.abs(-5.5); // 返回绝对值,结果为 5.5 double ceilValue = Math.ceil(4.3); // 返回不小于参数的最小整数值,结果为 5.0 double floorValue = Math.floor(4.9); // 返回不大于参数的最大整数值,结果为 4.0 double maxValue = Math.max(10.2, 5.8); // 返回两个参数中的最大值,结果为 10.2 double minValue = Math.min(3.5, 7.1); // 返回两个参数中的最小值,结果为 3.5 long roundValue = Math.round(3.8); // 返回最接近参数的整数值,四舍五入,结果为 4 // 指数运算 double expValue = Math.exp(2.0); // 返回 e 的指数值,结果为 7.3890560989306495 double logValue = Math.log(10.0); // 返回以 e 为底的对数值,结果为 2.302585092994046 double powValue = Math.pow(2.0, 3.0); // 返回 2 的 3 次方,结果为 8.0 // 平方根和立方根 double sqrtValue = Math.sqrt(25.0); // 返回参数的平方根,结果为 5.0 double cbrtValue = Math.cbrt(27.0); // 返回参数的立方根,结果为 3.0 // 三角函数 double sinValue = Math.sin(Math.PI / 6); // 返回参数的正弦值,结果为 0.5 double cosValue = Math.cos(Math.PI / 3); // 返回参数的余弦值,结果为 0.5 double tanValue = Math.tan(Math.PI / 4); // 返回参数的正切值,结果为 1.0 // 角度和弧度转换 double degreesValue = Math.toDegrees(Math.PI / 2); // 将弧度转换为角度,结果为 90.0 double radiansValue = Math.toRadians(180.0); // 将角度转换为弧度,结果为 π // 随机数生成 double randomValue = Math.random(); // 返回一个大于等于 0.0 且小于 1.0 的随机浮点数 // 输出结果 System.out.println("Absolute Value: " + absoluteValue); System.out.println("Ceil Value: " + ceilValue); System.out.println("Floor Value: " + floorValue); System.out.println("Max Value: " + maxValue); System.out.println("Min Value: " + minValue); System.out.println("Round Value: " + roundValue); System.out.println("Exp Value: " + expValue); System.out.println("Log Value: " + logValue); System.out.println("Pow Value: " + powValue); System.out.println("Sqrt Value: " + sqrtValue); System.out.println("Cbrt Value: " + cbrtValue); System.out.println("Sin Value: " + sinValue); System.out.println("Cos Value: " + cosValue); System.out.println("Tan Value: " + tanValue); System.out.println("Degrees Value: " + degreesValue); System.out.println("Radians Value: " + radiansValue); System.out.println("Random Value: " + randomValue); } }
结果:
java.util.Random类:主要用于生成伪随机数。
伪随机:Random类产生的数字是伪随机的,在相同种子数(seed)下的相同次数产生的随机数是相同的。
基本使用方法:
创建 Random 对象:
Random random = new Random();
生成随机整数:
// 生成一个介于 0(包含)和指定范围(不包含)之间的整数 int randomNumber = random.nextInt(10); // 生成 0 到 9 之间的整数
生成随机浮点数:
// 生成一个介于 0(包含)和 1(不包含)之间的浮点数 double randomDouble = random.nextDouble();
生成随机布尔值:
boolean randomBoolean = random.nextBoolean();
生成随机字节数组:
byte[] randomBytes = new byte[10]; random.nextBytes(randomBytes); // 将随机字节填充到指定的字节数组中
设置种子值(可选):
// 如果需要重复生成相同的随机数序列,可以设置相同的种子值 long seed = 1234; Random seededRandom = new Random(seed);
不指定种子:
Random random = new Random(); //生成 0 到 10(不包括10)之间的随机整数 int num = random.nextInt(10); //生成 -1 到 10(不包括10)之间的随机整数 int num2 = random.nextInt(10)-1; System.out.println("生成 0 到 10(不包括10)之间的随机整数:"+num); // 生成随机浮点数 double randomDouble = random.nextDouble(); System.out.println("Random Double: " + randomDouble); // 生成随机布尔值 boolean randomBoolean = random.nextBoolean(); System.out.println("Random Boolean: " + randomBoolean); // 生成随机字节数组 byte[] randomBytes = new byte[5]; random.nextBytes(randomBytes); System.out.println("Random Bytes: " + Arrays.toString(randomBytes));
指定种子:
相同种子数(seed)下的相同次数,产生的随机数是相同的。
public class Test { public static void main(String[] args) { // 指定种子为42 long seed = 42; Random randomWithSeed = new Random(seed); // 生成随机整数 int randomNumber1 = randomWithSeed.nextInt(100); System.out.println("Random Number 1: " + randomNumber1); // 再次生成随机整数,因为种子相同,结果应该相同 int randomNumber2 = randomWithSeed.nextInt(100); System.out.println("Random Number 2: " + randomNumber2); // 创建另一个 Random 对象,没有指定种子 Random randomWithoutSeed = new Random(); // 生成随机整数,因为没有指定种子,结果不受前面的影响 int randomNumber3 = randomWithoutSeed.nextInt(100); System.out.println("Random Number 3: " + randomNumber3); } }
TAG:java学习笔记