Hello java
第一个Java程序,打印Hello java.
package com.example;
//package com.example 包,包路径
/**
* public 访问控制符(public 代表公有,谁都可以访问)
* class 类
* Demo 类名
*/
public class Demo {
//相当于C的main函数,程序的入口
/**
* public 访问控制符(public 代表公有,谁都可以访问)
* static 静态的
* void 标志没有返回值
* main 方法名
* String[] 参数类型
*
* @param args 参数
*/
public static void main(String[] args) {
//在控制台输出Hello Java!
System.out.println("Hello Java!");
//说明参数用途是接收的java运行参数
for (String s : args) {
System.out.println(s);
}
}
}
运行时你可能不知道的事
当程序启动时,JVM会将main方法压入栈中,当执行完后,则弹出栈。你可能会问什么是栈?
在程序运行的时候,会开辟运行时数据区:
JVM简介
Jvm工作原理学习记
下面我截取了一些比较常用易懂的,详细的请移步参考地址.
JVM生命周期
JVM实例对应了一个独立运行的java程序它是进程级别
- 启动。启动一个Java程序时,一个JVM实例就产生了,任何一个拥有public static void main(String[] args)函数的class都可以作为JVM实例运行的起点
- 运行。main()作为该程序初始线程的起点,任何其他线程均由该线程启动。JVM内部有两种线程:守护线程和非守护线程,main()属于非守护线程,守护线程通常由JVM自己使用,java程序也可以标明自己创建的线程是守护线程
消亡。当程序中的所有非守护线程都终止时,JVM才退出;若安全管理器允许,程序也可以使用Runtime类或者System.exit()来退出
JVM体系结构
类装载器(ClassLoader)(用来装载.class文件)
执行引擎(执行字节码,或者执行本地方法)
运行时数据区(方法区、堆、java栈、PC寄存器、本地方法栈)
JVM运行时数据区
第一块:PC寄存器
PC寄存器是用于存储每个线程下一步将执行的JVM指令,如该方法为native的,则PC寄存器中不存储任何信息。
第二块:JVM栈
JVM栈是线程私有的,每个线程创建的同时都会创建JVM栈,JVM栈中存放的为当前线程中局部基本类型的变量(java中定义的八种基本类型:boolean、char、byte、short、int、long、float、double)、部分的返回结果以及Stack Frame,非基本类型的对象在JVM栈上仅存放一个指向堆上的地址
第三块:堆(Heap)
它是JVM用来存储对象实例以及数组值的区域,可以认为Java中所有通过new创建的对象的内存都在此分配,Heap中的对象的内存需要等待GC进行回收。
(1) 堆是JVM中所有线程共享的,因此在其上进行对象内存的分配均需要进行加锁,这也导致了new对象的开销是比较大的
(2) Sun Hotspot JVM为了提升对象内存分配的效率,对于所创建的线程都会分配一块独立的空间TLAB(Thread Local Allocation Buffer),其大小由JVM根据运行的情况计算而得,在TLAB上分配对象时不需要加锁,因此JVM在给线程的对象分配内存时会尽量的在TLAB上分配,在这种情况下JVM中分配对象内存的性能和C基本是一样高效的,但如果对象过大的话则仍然是直接使用堆空间分配
(3) TLAB仅作用于新生代的Eden Space,因此在编写Java程序时,通常多个小的对象比大的对象分配起来更加高效。
第四块:方法区域(Method Area)
(1)在Sun JDK中这块区域对应的为PermanetGeneration,又称为持久代。
(2)方法区域存放了所加载的类的信息(名称、修饰符等)、类中的静态变量、类中定义为final类型的常量、类中的Field信息、类中的方法信息,当开发人员在程序中通过Class
对象中的getName、isInterface等方法来获取信息时,这些数据都来源于方法区域,同时方法区域也是全局共享的,在一定的条件下它也会被GC,当方法区域需要使用的内存超过其允许的大小时,会抛出OutOfMemory的错误信息。
第五块:运行时常量池(Runtime Constant Pool)
存放的为类中的固定的常量信息、方法和Field的引用信息等,其空间从方法区域中分配。
第六块:本地方法堆栈(Native Method Stacks)
JVM采用本地方法堆栈来支持native方法的执行,此区域用于存储每个native方法调用的状态。
基本数据类型
基本数据类型
- 基本数据类型
- 整型: byte short int long
- 浮点型: float double
- 逻辑型: boolean (false true)
- 字符型: char
变量
有数据类型的,且可以改变内容的标识符
命名规则: $ 、字母、下划线开头都行,后面的可以是数字、字母、下划线
变量
变量与常量的相同点是在声明之前都指定变量的数据类型,变量无final关键字声明。
-
声明变量语法:<类型> 变量名;如:
int a = 20;
常量
存储在常量池中
常量表示的是在程序中不能被修改的数据。
声明常量必须用final关键字。
-
常量分为:整型,浮点型,字符型,布尔型,字符串型常量,null。例如,语法:final<类型>常量名=值;如:
final int b = 30;
案例
public class Demo {
public static void main(String[] args) {
//方法调用
bianLiang();
numberTypes();
}
private static void numberTypes() {
byte b1 = 10; //-128~127
//byte b2=200; //错误:超出最大值
int i = b1; //合法,小的数据类型向大的数据类型自动转换
//short s=i; //错误: 大的数据类型不能向小的数据类型转换
short s2 = (short) i; //合法,强转将大的向小的转
System.out.println(s2);
byte b = 20;
short s = 3000;
int ii = 89;
long l = 90;
//float f=8.0;//非法,因为8.0默认为double类型,因此后面增加f或F
float f = 8.0f;
double dd = b + s + ii + l + f;//合法的,自动将小的向大的转型
String tag = "Hello";
boolean flag = true;
System.out.println(flag);
System.out.println(tag + " java");//字符串可以通过+连接任一变量,最终生成新的字符串
System.out.println(dd);
}
//命名规则:$ 、字母、下划线开头都行,后面的可以是数字、字母、下划线
private static void bianLiang() {
final int b = 30;
System.out.println(a);
System.out.println(b);
}
}
补充
基本数据类型
boolean(布尔型) byte(字节类型,1byte) char(字符类型,2) short(短整型,2) int(整形,4) float(浮点类型,单精度,4) long(长整型,8) double(浮点型,双精度,8)
Java默认整型是int类型,若要定义成float型,则要在数值后面加上f或F
Java默认浮点型是double类型,若要定义成float型,则要在数值后面加上f或F
基本数据类型自动装换
byte->short,char->int->long
float->double
int->float
long->double
小可转大,大要转小则损失精度
转义字符,运算符
转义字符
'':转义
-
常见的转义字符
"\t" 制表 ;"\n" 换行System.out.println("name=\"小红\",\"age\":20");//name="小红","age":20 System.out.println("name=\"小黑\",\n\"age\":20");//name="小黑", //"age":20 System.out.println("\\5\\2\\");// \5\2\
运算符
算数运算符
+(加)
-(减)
*(乘)
-
/(两数相除之后取整)
int f=19/10; //取整数,f=1
-
%(两数相除之后取余)
int h=19%10; //取余数,h=9
-
++(自加)
int c=8; int d=++c; //先自加,后返回(表达式的结果),因此d=9,c=9 int e=c++; //先返回,后自加,e=9,c=10
--(自减)同自加
赋值运算符
-
=
int a=4;
-
+=
a+=2;相当于 a=a+2;
-
-=
a-=2;相当于 a=a-2;
*= 同理
/= 同理
%= 同理
比较运算符
- ">"(大于) 忽略双引号
- ">="(大于等于)
- <(小于)
- <=(小于等于)
- ==(等于)
- !=(不等于)
逻辑运算符
- &(与)
* true & true true
* true & false false
* false & true false
* false & false false
* &符号就是你两边都是true才真,两边只要有一个false,就是false
-
|(或)
* true | true true * true | false true * false | true true * false | false false * 只要就一个成立就可以,也就是返回true,只有当两个都是false才会返回false
-
^(异或)
* true | true false * true | false true * false | true true * false | false false * 相同为false,不同为真
-
!(非)
不是.
&&(短路与)
功能同&,区别在于短路,就是只要发现一个为false,直接返回false,不在做第二个的判断,&必须判断两个||(短路或)功能同|,区别在于短路,就是只要发现一个为true,直接返回true,不在做第二个的判断,|必须判断两个
位运算
- & 按位与,两个都为1时,则为1,否则为0 ,eg. 1&0=0,1&1=1
- | 按位或,两个中任一个是1时,则为1, eg. 1|0=1,1|1=1,0|0=0
- ^ 按位异或, 两个相同时,为0,不同时为1, eg. 10=1,00=0,1^1=0
- ~ 按位取反,如果取反之后为负数,则按负数的正数取反加1得出结果,eg. ~10=-11
位移运算符
“>>”(忽略双引号) 向右移,相当于除于2^n << 向左移,相当于乘于2^n; >>> 无符号移动
三目运算符
-
三目运算符 条件表达式?为真的值:为假的值
int max= a>b?a:b;
作用相当于:
//使用if-else分支
if(a>b){
//为真时
max=a;
}else{
//为假时
max=b;
}
System.out.println("最大数为:"+max);
例子
public class Demo{
public static void main(String[] args) {
//'\'转义
zhuanYi();
//运算符 算术、赋值、比较、逻辑、位运算、三目运算符
yunSuanFu();
}
private static void yunSuanFu() {
int a=18;
int b=10;
int c=a-b;
//算术 + , - , *, /, %, ++,--
////////////////////////////////////////
int d=++c; //先自加,后返回(表达式的结果),因此d=9,c=9
int e=c++; //先返回,后自加,e=9,c=10
int f=a/b; //取整数,f=1
int h=a%b; //取余数,h=8
///////////////////////////////////////
//赋值: = , +=,-=,/=, *=, %=
h*=2; //h=h*2;
byte b1=127;
b1+=1; // -128
//b1=127+1; //错误的,128已超出为byte类型大小
System.out.println(b1);//-128
///////////////////////////////////////
//比较: >, >= , < , <= , ==, != ,表达式结果为boolean
System.out.println("c>d="+(c>d)); //c>d=true
boolean b2=c>d;
System.out.println("a==c ->"+(a==c)); //a==c ->false
///////////////////////////////////////
//逻辑: &&, || , !, &,|
// true && true = true, true && false = false
// true || false =true, false || true = true
// !true =false
// true & false : false,
// &与&&区别: && 如果第一个是false则不再判断后面的值 & 两个都要判断
// | 与|| 区别: || 如果第一个是true,则不再判断后面的值,| 两个都要判断
///////////////////////////////////////
//位运算: & 按位与,两个都为1时,则为1,否则为0 ,eg. 1&0=0,1&1=1
// | 按位或,两个中任一个是1时,则为1, eg. 1|0=1,1|1=1,0|0=0
// ^ 按位异或, 两个相同时,为0,不同时为1, eg. 1^0=1,0^0=0,1^1=0
// ~ 按位取反,如果取反之后为负数,则按负数的正数取反加1得出结果,eg. ~10=-11
System.out.println(~10000);//-10001
///////////////////////////////////////
//移位运算符:>> 向右移,相当于除于2^n << 向左移,相当于乘于2^n >>> 无符号移动
// -10 >>2 ,与-10/4=-2 不同, 8>>3 , 8/8=1; -10>>>2, 无符号移动
System.out.println(-10>>>2);//无符号移动,高位补0 //1073741821
}
//常用的转义字符 "\t" 制表 "\n" 换行
private static void zhuanYi() {
System.out.println("\t*\t*\t*");// * * *
//\t 制表符,\" , \\, \r\n (window 换行),\n (linux 换行)
System.out.println("name=\"小红\",\"age\":20");//name="小红","age":20
System.out.println("name=\"小黑\",\n\"age\":20");//name="小黑",
//"age":20
System.out.println("\\5\\2\\");// \5\2\
}
}
补充
String.format(String ,Object ...) 格式化字符串,将第一个参数中%位置的值用后面的参数替换,按先后顺序替换,可用的%格式: %d 整数, %f,%.2f 带小数位的float/double,%s 字符串,如:
System.out.println(String.format("a=%d,b=%d,c=%d,d=%d,e=%d,f=%d,h=%d",a,b,c,d,e,f,h));//a=18,b=10,c=10,d=9,e=9,f=1,h=8