------<a href="http://www.itheima.com" target="blank">Java培训、Android培训、iOS培训、.Net培训</a>、期待与您交流! -------
进程:是一个正在执行中的程序。
每一个进程执行都有一个执行顺序。该顺序就是一个执行路径,或者叫一个控制单元。
线程:就是进程中的一个独立的控制单元。
线程在控制着进程的执行。
一个进程中至少有一个线程。
Java VM 启动的时候会有一个进程java.exe
该进程中至少有一个线程负责java程序的执行。而且这个线程运行的代码存在于main()方法中。
该线程称之为主线程。
扩展:其实更细节说明jvm,jvm启动时不止一个线程,还有负责垃圾回机制的线程。
如何在自定义的代码中,自定义一个线程呢?
通过对API文档的查找,java已经提供了对线程这类事物的描述,就是Thread类。
一,创建线程的第一种方式:继承Thread类。
步骤:
1),定义类继承Thread类。
2),复写Thread类中的run方法。
目的:将自定义代码存储在run()方法中,让线程运行。
3),调用现成的start方法,该方法有两个作用:1.启动线程,2.调用run方法。
<span style="font-size:14px;">class Test extends Thread
{
Test(String name)
{
super(name);
}
public void run()
{
for(int x = 0; x<60; x++)
System.out.println(getName()+" run: "+x);
}
}
class ThreadTest
{
public static void main(String[] args)
{
Test t1 = new Test("one----");
Test t2 = new Test("two++++");
t1.start();
t2.start();
for(int x = 0; x<60; x++)
System.out.println("Main run: "+x);
}
}</span>
二,创建线程的第二种方式:实现 Runnable接口
步骤:
1,定义类实现Runnable接口。
2,覆盖Runnable接口中的run方法。
将线程要运行的代码存放在run方法中。
3,通过Thread类建立线程对象。
4,将Runnable接口的子类对象作为实际参数传递给Thread类的构造函数。
为什么要将Runnable接口的子类对象传递给Thread的构造函数。
因为,自定义的run方法所属的对象时Runnable接口的子类对象。
所以要让线程去执行指定对象的run方法,就必须明确该run方法所属的对象。
5,调用Thread类的start方法开启线程并调用Runnable接口子类的run方法。
<span style="font-size:14px;">class Ticket implements Runnable
{
private int tick = 100;
public void run()
{
while(tick>0)
{
System.out.println(Thread.currentThread().getName()+".sale: "+ tick--);
}
}
}
class TicketDemo1
{
public static void main(String[] args)
{
Ticket t = new Ticket();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
}
}</span>
实现方式和继承方式有什么区别呢?
实现方式好处:避免了单继承的局限性。
在定义线程时,建议使用实现方式。
两种方式的区别:
继承Thread:线程代码存放在Thread的子类run方法中。
实现Runnable:线程代码存放在接口子类的run方法中。
发现运行结果每一次都不同,是因为多个线程都获取CPU的使用权,CPU执行到谁,谁就运行。
明确一点,在某一时刻,只能有一个程序在运行(多核除外)。
CPU在做着快速切换,已达到看上去是同时运行的效果。
我们可以形象的把多线程的运行形容为在互相抢夺CPU资源/执行权。
这就是多线程的特性:随机性。(谁抢到谁执行,至于执行多长时间,CPU说的算)
为什么要覆盖run()方法:
Thread类用于描述线程。
该类就定义了一个功能,用于存储线程要运行的代码,该存储功能就是run()方法。
也就是说,Thread类中的run()方法,用于存储线程要运行的代码。
同步代码,同步函数同时存在,并且同步函数为static函数
static同步函数中没有this,所以使用的锁不是this。
static进入内存时,内存中没有本类对象,但是一定有该类对应的字节码文件对象:
类名.class,该对象的类型是Class。
即静态的同步方法,使用的锁是该方法所在类的字节码文件对象:类名.class