1.既然 Java 的垃圾回收机制能够自动的回收内存,怎么还会出现内存泄漏的情况呢?
2.java内存泄漏的根本原因是?
3.JVM、JRE、JDK
4.记事本编写java、cmd运行java。
5.变量类型
1. 变量大小
2. 定义变量
3. 运算时自动类型转换
4.强制类型转换
5.char和String
6.原码、反码、补码
7.Java命名规则。
8.运算
9.逻辑运算符
10.位运算符
11.运算符的优先级
这个问题,我们需要知道 GC 在什么时候回收内存对象,什么样的内存对象会被 GC 认为是“不再使用”的。Java中对内存对象的访问,使用的是引用的方式。在 Java 代码中我们维护一个内存对象的引用变量,通过这个引用变量的值,我们可以访问到对应的内存地址中的内存对象空间。在 Java 程序中,这个引用变量本身既可以存放堆内存中,又可以放在代码栈的内存中(与基本数据类型相同)。 GC 线程会从代码栈中的引用变量开始跟踪,从而判定哪些内存是正在使用的。如果 GC 线程通过这种方式,无法跟踪到某一块堆内存,那么 GC 就认为这块内存将不再使用了(因为代码中已经无法访问这块内存了)。
通过这种有向图的内存管理方式,当一个内存对象失去了所有的引用之后,GC 就可以将其回收。反过来说,如果这个对象还存在引用,那么它将不会被 GC 回收,哪怕是 Java 虚拟机抛出 OutOfMemoryError 。
答:内存对象明明已经不需要的时候,还仍然保留着这块内存和它的访问方式(引用)。
JVM:Java Virtual Machine(Java 虚拟机)JVM是JRE的一部分,它是一个虚拟出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的。JVM有自己完善的硬件架构,如处理器、堆栈、寄存器等,还具有相应的指令系统。
JVM是Java实现跨平台最核心的部分,所有的Java程序会首先被编译为.class的类文件,JVM的主要工作是解释自己的指令集(即字节码)并映射到本地的CPU的指令集或OS的系统调用。Java面对不同操作系统使用不同的虚拟机,依次实现了跨平台。JVM对上层的Java源文件是不关心的,它关心的只是由源文件生成的类文件。
JRE:Java Runtime Environment(Java运行环境)。在Java平台下,所有的Java程序都需要在JRE下才能运行。只有JVM还不能进行class的执行,因为解释class的时候,JVM需要调用解释所需要的类库lib。JRE里面有两个文件夹bin和lib,这里可以认为bin就是JVM,lib就是JVM所需要的类库,而JVM和lib合起来就称为JRE。
JRE包括JVM和JAVA核心类库与支持文件。与JDK不同,它不包含开发工具—-编译器、调试器和其它工具。
JDK:Java Development ToolKit(Java开发工具包)。JDK是整个JAVA的核心,包括了Java的运行环境(Java Runtime Environment)、一堆Java工具(javac、java、jdb等)和Java基础的类库(即Java API包括rt.jar)。
Java API是Java的应用程序接口,里面有很多已经写好的Java Class,包括一些重要的语言结构以及基本图形,网络和文件I/O等等,我们可以直接调用使用。
新建记事本文件改文件名为helloworld.java编写java文件内容为
class HelloWorld
{
public static void main(String[] args){
System.out.println("hello world");
}
}
打开cmd在cmd输入
javac helloworld.java
生成class文件然后输入
java HelloWorld
注:1.运行javac helloworld.java时生成的class文件名是代码中的class类名。不一定与原来的java文件名字一样。
2.输入命令java HelloWorld时不需要加后缀名class
3.输入javac的时候是在Windows层次上找文件名。不会区分大小写。但是在输入java时候是在找类名,此时区分大小写。
注意: String 属于引用数据类型。
byte一个字节-128~127
short两个字节
int四个字节
long八个字节
float 四个字节
double 八个字节
char 两个字节
定义long类型数据时要在数据后加大写或小写的L;long 类型不加L会被认为时int类型。
定义float类型数据时要在数据后面加小写或者大写的F;
定义char类型时要用单引号、只能有一个字符。char还可以使用unicode。String是双引号。
定义boolean类型时区别于c语言:在java中只有true和false;在c语言中零就是false,非零就是true。
byte、short、char其中任意两者相运算时结果都是int类型的。
注意:当byte和bytttte相加short和short相加char和char相加的时候运算结果也是int。
形式如下:
单引号和单引号相加表运算
双引号和双引号相加表连接
long a = 22222L;
long b = 33333l;
float c = 2.2F;
float d = 3.3f;
char char1 = 'a';
char char2 = '中文';
char char3 = '\n' //换行符
char char4 = '\t' //制表符
char char5 = '\u0043' //unicode 编码
boolean isMarried = true;
short t = (short)a;
char e = ''; //单引号内什么都没有报错
char e = ' '; //单引号内有空格不报错
String e = ""; //双引号内为空不报错
String e = 123; //编译不通过,必须加双引号
String str1 = 123 + "";
int num1 = (int)str1;
System.out.println(num1); // 编译不通过
int num1 = Integer.parseInt(str1) //可以将String转换成int
包名:xxxyyyzzz
类名、接口名:XxxYyyZzz
变量名、方法名:xxxYyyZzz
常量名:XXX_YYY_ZZZ
1.++运算不会改变数据类型,+=运算不会改变数据类型。
char char1 = 'a';
System.out.println(++char1); //输出b
short sh1 = 123;
System.out.println(sh1++); //输出为short类型的数据
short sh2 = 10;
//sh2 = sh2 + 1;编译失败
short sh2 += 2;
System.out.println(sh2); //输出12
int n = 10;
n += (n++) + (++n); //n = n + (n++) + (++n);
System.out.println(n); //输出为32
对布尔类型的进行操作。
区分逻辑与&和短路与&&。
当符号左边是false时&还会判断符号后面的操作
当符号左边时false时&&只会判断第一个false,后面的内容会被过滤掉。
开发中推荐使用&&,省时。
boolean b1 = true;
b1 = false;
int num1 = 10;
if(b1 & (num1++ > 0)){ //逻辑与正常操作
System.out.println("ture");
}else{
System.out.println("false");
}
System.out.println("num1 = " + num1);
boolean b2 = true;
b2 = false;
int num2 = 10;
if(b2 && (num2++ > 0)){ //短路与如果前面第一个条件时false他就不看后面的条件,把num2过滤掉了,num2 就不会执行++操作
System.out.println("ture");
}else{
System.out.println("false");
}
System.out.println("num2 = " + num2);
boolean b3 = true;
b3 = false;
int num3 = 10;
if((num3++ > 0) && b3){
System.out.println("ture");
}else{
System.out.println("false");
}
System.out.println("num3 = " + num3);
输出结果为:
false
num1 = 11
false
num2 = 10
false
num3 = 11
直接对整数的二进制进行操作