3、手把手教你搭建Java环境
任务
想要进行java语言的学习与开发,我们要先搭建java的开发环境。就好比我们做一道油焖大虾,首先要有厨房这个环境。只有在厨房里,我们才可以使用燃气炉灶,油,盐等做出一道菜。同理,只有安装了java的开发环境,我们才能够编写和运行java的程序。
Java已经被Oracle收购,进入Oracle官网下载。
https://www.oracle.com/cn/index.html
- 1、首先,打开浏览器,输入以上Oracle的官网网址
- 2、然后点开产品,打开产品的列表,点击Java
- 3、点击Download Java,进入Java的下载页
- 4、滚动到下面,找到Java8版本。虽然最新版本是Java14,但是我们强烈建议选择Java8版本。
- 5、根据自己电脑的操作系统,进行下载。
当然,你也可以通过以下网盘链接来获取
课前准备工具包:链接:https://pan.baidu.com/s/1Kqg-O0r7H7KpXAXT3xvXww 密码:dqh4
JDK的安装是无脑安装,就是一路下一步下一步。。直到完成。默认安装在C盘下的Program Files目录下,傻瓜式下一步下一步直到完成即可。但是如果想更改安装目录,那么请你小心按照以下操作进行安装。
采坑:在jdk的安装过程中,包含了jre的安装,如果不想装在默认目录下,那么在安装jre的时候,要注意安装路径,不要覆盖刚刚安装好的jdk。
1、找到下载后的jdk的安装包,双击进行安装
2、首先安装jdk。在安装过程中,默认会安装到C盘的Program Files目录下,如果想改变安装目录,可以点击更改按钮。当然,你也可以安装到默认目录下。
3、直接手动的修改安装目录,将C盘改为你要安装的盘符即可。
4、修改好jdk的安装路径后,点击下一步。
5、一路下一步就安装好了jdk,然后就到了jre的安装。如果不想安装到默认的C盘,点击更改按钮。一般建议将jre和jdk安装到一个目录下。
6、此处会弹出浏览框,我们点击D盘后,再点击Java。jre的安装需要我们自己手动创建一个安装的文件夹。
7、安装jre的文件夹名,我们手动输入:jre后面跟着版本号。jre1.8.0_77。
8、点击确定后,就可以继续安装了。
9、到此jdk和jre安装完成。
10、然后查看一下安装目录,打开D盘,Program Files目录,多了一个Java文件夹,点进去后有两个文件夹,一个是jdk的安装目录,一个是jre的安装目录。
很多新同学在安装的时候,不注意安装目录,在安装jre的时候,会不小心覆盖掉刚刚安装好的jdk。所以一定要注意。
11、打开jdk文件夹
- bin目录:JDK开发工具的可执行文件。javac.exe和java.exe等命令就在此目录下。
- db:包含Java DB,Java DB是一个机遇Java编程语言的SQL关系数据库管理系统。
- include:包含C语言头文件,支持Java本地接口与Java虚拟机调试程序接口的本地编程技术。
- jre:Java运行时环境的根目录,包含Java虚拟机,运行时的类包和Java应用启动器。但不包含开发环境中的开发工具。
- lib目录:开发工具使用的归档包文件。
- src.zip:Java源码压缩包
- 。。。
此电脑上右键选择属性,然后点击高级系统设置,就会打开系统属性窗口。
然后点击高级,点击下面的环境变量,就会打开环境变量的设置窗口。
1、在系统变量里,点击新建,配置JAVA_HOME变量。变量值就是刚刚jdk的安装路径。
2、配置CLASSPATH路径:
3、修改path路径:
在我们的windows系统中,path变量是内置好的,我们需要将jdk下的bin目录配置到path里。所以此处选择编辑:
然后两次新建,将jdk下的bin目录和jdk下jre里的bin目录配置进去。
4、配置后一定要点击确定,这些配置才会生效。
踩坑1:配置环境变量后,要点击确认按钮,配置才会生效。
踩坑2:配置环境变量后,要重启dos黑窗口,命令才可以使用
打开终端:
方法一:开始|Windows系统|命令提示符
方法二:搜索框里直接搜索:命令提示符或者cmd
方法三:快捷键:win+R键,然后输入cmd
注意:如果你在配置环境变量前已经打开了dos窗口,那么配置环境变量后,需要重新打开dos窗口。
测试1:
测试2:
如果安装错了路径,或者想换个安装路径,那么你要选择卸载jdk,然后重新安装。切记不要直接剪切安装包。
如果你有360或者腾讯管家等软件可以操作他们来卸载jdk。否则可以按照以下操作卸载:
1、打开控制面板,点击应用程序,选择卸载:
2、点击Java8 Update,选择卸载,然后下面的Java SE Development Kit 8 Update也要卸载。
3、然后查看以下刚刚的安装目录,正常情况下,该目录会随着卸载jdk而删除,但是如果没有删除,你可以手动删除它,然后再重新安装jdk。
JDK的下载安装与环境变量的配置
一、JDK下载
1.jdk下载地址 官网 https://www.oracle.com
2.选择Products
oracle官网
3.选择Java
4.选择Java SE,进入Oracle JDK
5.根据需要选择版本
6.我所使用的版本
二、JDK安装
1.按正常软件安装即可,双击安装包,下一步,下一步 ,… ,完成。
2.如果不想安装在C盘,可以更改安装盘符,其他目录最好不变。
3.也可以更改安装目录,但所在目录不要有中文字符。
三、配置环境变量
1.我的电脑 右击 —— 属性 —— 高级系统设置 —— 环境变量 —— 系统变量
2.新建系统变量JAVA_HOME,值为安装目录的全路径:D:\\Program Files\\Java\\jdk1.8.0_241(根据自己的安装目录配置)
3.在Path中添加,%JAVA_HOME%\\bin
4.验证jdk是否安装成功
5.按win + R键,输入cmd + 回车键,打开命令提示符
6.在命令提示符中输入 java -version 回车
- 检验是不是自己所安装的版本
注:对初学者来说,可以直接在Path路径中添加javac.exe文件所在的全路径D:\\Program Files\\Java \\jdk1.8.0_241\\bin
四、运行java程序
1.新建 类名.java文件(public修饰的类的类名),编写源文件,保存。
2.在源文件所在目录下,打开命令提示符
3.输入 javac 类名.java 回车键,生成 类名.class 文件。
4.输入 java 类名 回车键,程序执行
五、JDK、JRE、JVM简介
1.JDK (Java Development Kit),java 开发工具包
1)安装目录介绍
(1)bin binary 二进制
- javac.exe 编译工具。将 java 源代码编译为 字节码二进制文件。
- java.exe 解释器。
2.JRE,javaruntime Environment Java 运行环境
不包括开发环境,没有编译的环境。
3.JVM,java virtual machine,Java虚拟机
- 负责解释执行 字节码 .class 文件,是一个使用软件模拟的计算机。
- class 文件中是 虚拟机指令 和 数据的集合
- jvm 也有自己的指令集合。
注:JDK包含JRE,JRE包含JVM
Java基础八股文背诵版
兄弟们求个赞,关注我,可以收到我整理的最新的IT大厂八股文更新
背之前先来看看阿里巴巴的常问Java基础你们会几个?(该站点长期更新各大厂最新面试题,欢迎大家关注)
八股文全集PDF打包下载:
链接:https://pan.baidu.com/s/1jSJWShuC__vcMHqEQyntNw
提取码:ozs7
- Java为纯面向对象的语言。它能够直接反应现实生活中的对象。
- 具有平台无关性。java利用Java虚拟机运行字节码,无论是在Windows、Linux还是MacOS等其它平台对Java程序进行编译,编译后的程序可在其它平台运行。
- Java为解释型语言,编译器把Java代码编译成平台无关的中间代码,然后在JVM上解释运行,具有很好的可移植性。
- Java提供了很多内置类库。如对多线程支持,对网络通信支持,最重要的一点是提供了垃圾回收器。
- Java具有较好的安全性和健壮性。Java提供了异常处理和垃圾回收机制,去除了C++中难以理解的指针特性。
- Java语言提供了对Web应用开发的支持。
- 继承:对象的一个新类可以从现有的类中派生,派生类可以从它的基类那继承方法和实例变量,且派生类可以修改或新增新的方法使之更适合特殊的需求。
- 封装:将客观事物抽象成类,每个类可以把自身数据和方法只让可信的类或对象操作,对不可信的进行信息隐藏。
- 多态:允许不同类的对象对同一消息作出响应。不同对象调用相同方法即使参数也相同,最终表现行为是不一样的。
字节序是指多字节数据在计算机内存中存储或网络传输时个字节的存储顺序。通常由小端和大端两组方式。
- 小端:低位字节存放在内存的低地址端,高位字节存放在内存的高地址端。
- 大端:高位字节存放在内存的低地址端,低位字节存放在内存的高地址端。
Java语言的字节序是大端。
- JDK:Java开发工具包(Java Development Kit),提供了Java的开发环境和运行环境。
- JRE:Java运行环境(Java Runtime Environment),提供了Java运行所需的环境。
JDK包含了JRE。如果只运行Java程序,安装JRE即可。要编写Java程序需安装JDK.
- default: 默认访问修饰符,在同一包内可见
- private: 在同一类内可见,不能修饰类
- protected : 对同一包内的类和所有子类可见,不能修饰类
- public: 对所有类可见
先后顺序:静态成员变量、成员变量、构造方法。详细的先后顺序:父类静态变量、父类静态代码块、子类静态变量、子类静态代码块、父类非静态变量、父类非静态代码块、父类构造函数、子类非静态变量、子类非静态代码块、子类构造函数。
相同点:
- 都不能被实例化。
- 接口的实现类或抽象类的子类需实现接口或抽象类中相应的方法才能被实例化。
不同点:
- 接口只能有方法定义,不能有方法的实现,而抽象类可以有方法的定义与实现。
- 实现接口的关键字为implements,继承抽象类的关键字为extends。一个类可以实现多个接口,只能继承一个抽象类。
- 当子类和父类之间存在逻辑上的层次结构,推荐使用抽象类,有利于功能的累积。当功能不需要,希望支持差别较大的两个或更多对象间的特定交互行为,推荐使用接口。使用接口能降低软件系统的耦合度,便于日后维护或添加删除方法。
- 为了程序的结构能够更加清晰从而便于维护。假设Java语言支持多重继承,类C继承自类A和类B,如果类A和B都有自定义的成员方法f(),那么当代码中调用类C的f()会产生二义性。Java语言通过实现多个接口间接支持多重继承,接口由于只包含方法定义,不能有方法的实现,类C继承接口A与接口B时即使它们都有方法f(),也不能直接调用方法,需实现具体的f()方法才能调用,不会产生二义性。
- 多重继承会使类型转换、构造方法的调用顺序变得复杂,会影响到性能。
Java提供了两种用于多态的机制,分别是重载与覆盖。
- 重载:重载是指同一个类中有多个同名的方法,但这些方法有不同的参数,在编译期间就可以确定调用哪个方法。
- 覆盖:覆盖是指派生类重写基类的方法,使用基类指向其子类的实例对象,或接口的引用变量指向其实现类的实例对象,在程序调用的运行期根据引用变量所指的具体实例对象调用正在运行的那个对象的方法,即需要到运行期才能确定调用哪个方法。
- 覆盖是父类与子类之间的关系,是垂直关系;重载是同一类中方法之间的关系,是水平关系。
- 覆盖只能由一个方法或一对方法产生关系;重载是多个方法之间的关系。
- 覆盖要求参数列表相同;重载要求参数列表不同。
- 覆盖中,调用方法体是根据对象的类型来决定的,而重载是根据调用时实参表与形参表来对应选择方法体。
- 重载方法可以改变返回值的类型,覆盖方法不能改变返回值的类型。
- final用于声明属性、方法和类,分别表示属性不可变、方法不可覆盖、类不可继承。
- finally作为异常处理的一部分,只能在try/catch语句中使用,finally附带一个语句块用来表示这个语句最终一定被执行,经常被用在需要释放资源的情况下。
- finalize是Object类的一个方法,在垃圾收集器执行的时候会调用被回收对象的finalize()方法。当垃圾回收器准备好释放对象占用空间时,首先会调用finalize()方法,并在下一次垃圾回收动作发生时真正回收对象占用的内存。
当遇到下面情况不会执行。
- 当程序在进入try语句块之前就出现异常时会直接结束。
- 当程序在try块中强制退出时,如使用System.exit(0),也不会执行finally块中的代码。
其它情况下,在try/catch/finally语句执行的时候,try块先执行,当有异常发生,catch和finally进行处理后程序就结束了,当没有异常发生,在执行完finally中的代码后,后面代码会继续执行。值得注意的是,当try/catch语句块中有return时,finally语句块中的代码会在return之前执行。如果try/catch/finally块中都有return语句,finally块中的return语句会覆盖try/catch模块中的return语句。
static的主要作用有两个:
- 为某种特定数据类型或对象分配与创建对象个数无关的单一的存储空间。
- 使得某个方法或属性与类而不是对象关联在一起,即在不创建对象的情况下可通过类直接调用方法或使用类的属性。
具体而言static又可分为4种使用方式:
- 修饰成员变量。用static关键字修饰的静态变量在内存中只有一个副本。只要静态变量所在的类被加载,这个静态变量就会被分配空间,可以使用\’\’类.静态变量\’\’和\’\’对象.静态变量\’\’的方法使用。
- 修饰成员方法。static修饰的方法无需创建对象就可以被调用。static方法中不能使用this和super关键字,不能调用非static方法,只能访问所属类的静态成员变量和静态成员方法。
- 修饰代码块。JVM在加载类的时候会执行static代码块。static代码块常用于初始化静态变量。static代码块只会被执行一次。
- 修饰内部类。static内部类可以不依赖外部类实例对象而被实例化。静态内部类不能与外部类有相同的名字,不能访问普通成员变量,只能访问外部类中的静态成员和静态成员方法。
- 父类静态代码块(只执行一次)
- 子类静态代码块(只执行一次)
- 父类构造代码块
- 父类构造函数
- 子类构造代码块
- 子类构造函数
- 普通代码块
一维数组的声明方式:
- type arrayName[]
- type[] arrayName
二维数组的声明方式:
- type arrayName[][]
- type[][] arrayName
- type[] arrayName[]
其中type为基本数据类型或类,arrayName为数组名字
String用于字符串操作,属于不可变类。String对象一旦被创建,其值将不能被改变。而StringBuffer是可变类,当对象创建后,仍然可以对其值进行修改。
== 比较的是引用,equals比较的是内容。
- 如果变量是基础数据类型,== 用于比较其对应值是否相等。如果变量指向的是对象,== 用于比较两个对象是否指向同一块存储空间。
- equals是Object类提供的方法之一,每个Java类都继承自Object类,所以每个对象都具有equals这个方法。Object类中定义的equals方法内部是直接调用 == 比较对象的。但通过覆盖的方法可以让它不是比较引用而是比较数据内容。为什么要把String设计为不变量?
- 节省空间:字符串常量存储在JVM的字符串池中可以被用户共享。
- 提高效率:String会被不同线程共享,是线程安全的。在涉及多线程操作中不需要同步操作。
- 安全:String常被用于用户名、密码、文件名等使用,由于其不可变,可避免黑客行为对其恶意修改。序列化是什么?序列化是一种将对象转换成字节序列的过程,用于解决在对对象流进行读写操作时所引发的问题。序列化可以将对象的状态写在流里进行网络传输,或者保存到文件、数据库等系统里,并在需要的时候把该流读取出来重新构造成一个相同的对象。
java中对象可以分为实例对象和Class对象,每一个类都有一个Class对象,其包含了与该类有关的信息。
获取Class对象的方法:
- Class.forName(“类的全限定名”)
- 实例对象.getClass()
- 类名.class
Java反射机制是指在程序的运行过程中可以构造任意一个类的对象、获取任意一个类的成员变量和成员方法、获取任意一个对象所属的类信息、调用任意一个对象的属性和方法。反射机制使得Java具有动态获取程序信息和动态调用对象方法的能力。可以通过以下类调用反射API。
- Class类:可获得类属性方法
- Field类:获得类的成员变量
- Method类:获取类的方法信息
- Construct类:获取类的构造方法等信息
Java 注解用于为 Java 代码提供元数据。作为元数据,注解不直接影响你的代码执行,但也有一些类型的注解实际上可以用于这一目的。
其可以用于提供信息给编译器,在编译阶段时给软件提供信息进行相关的处理,在运行时处理写相应代码,做对应操作。
元注解可以理解为注解的注解,即在注解中使用,实现想要的功能。其具体分为:
- @Retention: 表示注解存在阶段是保留在源码,还是在字节码(类加载)或者运行期(JVM中运行)。
- @Target:表示注解作用的范围。
- @Documented:将注解中的元素包含到 Javadoc 中去。
- @Inherited:一个被@Inherited注解了的注解修饰了一个父类,如果他的子类没有被其他注解修饰,则它的子类也继承了父类的注解。
- @Repeatable:被这个元注解修饰的注解可以同时作用一个对象多次,但是每次作用注解又可以代表不同的含义。
Java异常分为Error(程序无法处理的错误),和Exception(程序本身可以处理的异常)。这两个类均继承Throwable。
Error常见的有StackOverFlowError,OutOfMemoryError等等。
Exception可分为运行时异常和非运行时异常。对于运行时异常,可以利用try catch的方式进行处理,也可以不处理。对于非运行时异常,必须处理,不处理的话程序无法通过编译。
throw一般是用在方法体的内部,由开发者定义当程序语句出现问题后主动抛出一个异常。
throws一般用于方法声明上,代表该方法可能会抛出的异常列表。
泛型,即“参数化类型”,解决不确定对象具体类型的问题。在编译阶段有效。在泛型使用过程中,操作的数据类型被指定为一个参数,这种参数类型在类中称为泛型类、接口中称为泛型接口和方法中称为泛型方法。
Java编译器生成的字节码是不包涵泛型信息的,泛型类型信息将在编译处理是被擦除,这个过程被称为泛型擦除。
- byte: 占用1个字节,取值范围-128 ~ 127
- short: 占用2个字节,取值范围-2^15^ ~ 2^15^-1
- int:占用4个字节,取值范围-2^31^ ~ 2^31^-1
- long:占用8个字节
- float:占用4个字节
- double:占用8个字节
- char: 占用2个字节
- boolean:占用大小根据实现虚拟机不同有所差异简述自动装箱拆箱对于Java基本数据类型,均对应一个包装类。
装箱就是自动将基本数据类型转换为包装器类型,如int->Integer
拆箱就是自动将包装器类型转换为基本数据类型,如Integer->int
重写即子类重写父类的方法,方法对应的形参和返回值类型都不能变。
重载即在一个类中,方法名相同,参数类型或数量不同。
Java多态可以分为编译时多态和运行时多态。
编译时多态主要指方法的重载,即通过参数列表的不同来区分不同的方法。
运行时多态主要指继承父类和实现接口时,可使用父类引用指向子类对象。
运行时多态的实现:主要依靠方法表,方法表中最先存放的是Object类的方法,接下来是该类的父类的方法,最后是该类本身的方法。如果子类改写了父类的方法,那么子类和父类的那些同名方法共享一个方法表项,都被认作是父类的方法。因此可以实现运行时多态。
抽象类:体现的是is-a的关系,如对于man is a person,就可以将person定义为抽象类。
接口:体现的是can的关系。是作为模板实现的。如设置接口fly,plane类和bird类均可实现该接口。
一个类只能继承一个抽象类,但可以实现多个接口。
对于==,在基本数据类型比较时,比较的是对应的值,对引用数据类型比较时,比较的是其内存的存放地址。
对于equals方法,在该方法未被重写时,其效果和==一致,但用户可以根据对应需求对判断逻辑进行改写,比如直接比较对象某个属性值是否相同,相同则返回true,不同则返回false。需保证equals方法相同对应的对象hashCode也相同。
- hashCode:通过对象计算出的散列码。用于map型或equals方法。需要保证同一个对象多次调用该方法,总返回相同的整型值。
- equals:判断两个对象是否一致。需保证equals方法相同对应的对象hashCode也相同。
- toString: 用字符串表示该对象
- clone:深拷贝一个对象
- 成员内部类:作为成员对象的内部类。可以访问private及以上外部类的属性和方法。外部类想要访问内部类属性或方法时,必须要创建一个内部类对象,然后通过该对象访问内部类的属性或方法。外部类也可访问private修饰的内部类属性。
- 局部内部类:存在于方法中的内部类。访问权限类似局部变量,只能访问外部类的final变量。
- 匿名内部类:只能使用一次,没有类名,只能访问外部类的final变量。
- 静态内部类:类似类的静态成员变量。
String类采用利用final修饰的字符数组进行字符串保存,因此不可变。如果对String类型对象修改,需要新建对象,将老字符和新增加的字符一并存进去。
StringBuilder,采用无final修饰的字符数组进行保存,因此可变。但线程不安全。
StringBuffer,采用无final修饰的字符数组进行保存,可理解为实现线程安全的StringBuilder。
序列化:将java对象转化为字节序列,由此可以通过网络对象进行传输。
反序列化:将字节序列转化为java对象。
具体实现:实现Serializable接口,或实现Externalizable接口中的writeExternal()与readExternal()方法。
List是一个有序队列,在JAVA中有两种实现方式:
ArrayList 使用数组实现,是容量可变的非线程安全列表,随机访问快,集合扩容时会创建更大的数组,把原有数组复制到新数组。
LinkedList 本质是双向链表,与 ArrayList 相比插入和删除速度更快,但随机访问元素很慢。
HashTable: 哈希表的线程安全版,效率低ConcurrentHashMap:哈希表的线程安全版,效率高,用于替代HashTableVector:线程安全版ArraylistStack:线程安全版栈BlockingQueue及其子类:线程安全版队列
Set 即集合,该数据结构不允许元素重复且无序。JAVA对Set有三种实现方式:
HashSet 通过 HashMap 实现,HashMap 的 Key 即 HashSet 存储的元素,Value系统自定义一个名为 PRESENT 的 Object 类型常量。判断元素是否相同时,先比较hashCode,相同后再利用equals比较,查询O(1)
LinkedHashSet 继承自 HashSet,通过 LinkedHashMap 实现,使用双向链表维护元素插入顺序。
TreeSet 通过 TreeMap 实现的,底层数据结构是红黑树,添加元素到集合时按照比较规则将其插入合适的位置,保证插入后的集合仍然有序。查询O(logn)
JDK8 之前底层实现是数组 + 链表,JDK8 改为数组 + 链表/红黑树。主要成员变量包括存储数据的 table 数组、元素数量 size、加载因子 loadFactor。HashMap 中数据以键值对的形式存在,键对应的 hash 值用来计算数组下标,如果两个元素 key 的 hash 值一样,就会发生哈希冲突,被放到同一个链表上。
table 数组记录 HashMap 的数据,每个下标对应一条链表,所有哈希冲突的数据都会被存放到同一条链表,Node/Entry 节点包含四个成员变量:key、value、next 指针和 hash 值。在JDK8后链表超过8会转化为红黑树。
若当前数据/总数据容量>负载因子,Hashmap将执行扩容操作。默认初始化容量为 16,扩容容量必须是 2 的幂次方、最大容量为 1<< 30 、默认加载因子为 0.75。
在JDK1.7中,HashMap采用头插法插入元素,因此并发情况下会导致环形链表,产生死循环。
虽然JDK1.8采用了尾插法解决了这个问题,但是并发下的put操作也会使前一个key被后一个key覆盖。
由于HashMap有扩容机制存在,也存在A线程进行扩容后,B线程执行get方法出现失误的情况。
TreeMap是底层利用红黑树实现的Map结构,底层实现是一棵平衡的排序二叉树,由于红黑树的插入、删除、遍历时间复杂度都为O(logN),所以性能上低于哈希表。但是哈希表无法提供键值对的有序输出,红黑树可以按照键的值的大小有序输出。
- Collection是一个集合接口,它提供了对集合对象进行基本操作的通用接口方法,所有集合都是它的子类,比如List、Set等。
- Collections是一个包装类,包含了很多静态方法、不能被实例化,而是作为工具类使用,比如提供的排序方法:Collections.sort(list);提供的反转方法:Collections.reverse(list)。
- ArrayList、Vector和LinkedList都是可伸缩的数组,即可以动态改变长度的数组。
- ArrayList和Vector都是基于存储元素的Object[] array来实现的,它们会在内存中开辟一块连续的空间来存储,支持下标、索引访问。但在涉及插入元素时可能需要移动容器中的元素,插入效率较低。当存储元素超过容器的初始化容量大小,ArrayList与Vector均会进行扩容。
- Vector是线程安全的,其大部分方法是直接或间接同步的。ArrayList不是线程安全的,其方法不具有同步性质。LinkedList也不是线程安全的。
- LinkedList采用双向列表实现,对数据索引需要从头开始遍历,因此随机访问效率较低,但在插入元素的时候不需要对数据进行移动,插入效率较高。
- HashMap是Hashtable的轻量级实现,HashMap允许key和value为null,但最多允许一条记录的key为null.而HashTable不允许。
- HashTable中的方法是线程安全的,而HashMap不是。在多线程访问HashMap需要提供额外的同步机制。
- Hashtable使用Enumeration进行遍历,HashMap使用Iterator进行遍历。
如果对Map进行插入、删除或定位一个元素的操作更频繁,HashMap是更好的选择。如果需要对key集合进行有序的遍历,TreeMap是更好的选择。
- fail-fast直接在容器上进行,在遍历过程中,一旦发现容器中的数据被修改,就会立刻抛出ConcurrentModificationException异常从而导致遍历失败。常见的使用fail-fast方式的容器有HashMap和ArrayList等。
- fail-safe这种遍历基于容器的一个克隆。因此对容器中的内容修改不影响遍历。常见的使用fail-safe方式遍历的容器有ConcurrentHashMap和CopyOnWriteArrayList。
equals和hashCode这两个方法都是从object类中继承过来的,equals主要用于判断对象的内存地址引用是否是同一个地址;hashCode根据定义的哈希规则将对象的内存地址转换为一个哈希码。HashSet中存储的元素是不能重复的,主要通过hashCode与equals两个方法来判断存储的对象是否相同:
- 如果两个对象的hashCode值不同,说明两个对象不相同。
- 如果两个对象的hashCode值相同,接着会调用对象的equals方法,如果equlas方法的返回结果为true,那么说明两个对象相同,否则不相同
本文作者及来源:Renderbus瑞云渲染农场https://www.renderbus.com
文章为作者独立观点不代本网立场,未经允许不得转载。