Java线程:什么是线程
Posted lingtongwanr
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java线程:什么是线程相关的知识,希望对你有一定的参考价值。
一 基本概念
多任务:同一时刻运行多个程序的能力。每一个任务称为一个线程。可以同时运行一个以上线程的程序称为多线程程序。
进程是指一个内存中运行的应用程序,每个进程都有自己独立的一块内存空间,一个进程中可以启动多个线程。比如在Windows系统中,一个运行的exe就是一个进程。
线程是指进程中的一个执行流程,一个进程可以运行多个线程。比如java.exe进程可以运行很多线程。线程总是输入某个进程,进程中的多个线程共享进程的内存。
Java中线程是指java.lang.Thread类的一个实例或线程的执行。使用java.lang.Thread或java.lang.Runnable接口编写代码定义、实例化、启动新线程。
Java中每个线程都有一个调用栈,即使不在程序中创建任何新的线程,线程也在后台运行。main()方法运行在一个线程内,称为主线程。一旦创建一个新的线程,就产生一个新的调用栈。
线程分为两类:用户线程和守候线程。当所有用户线程执行完毕后,JVM自动关闭。但是守候线程却不独立与JVM,守候线程一般是有操作系统或用户自己创建的。
二 定义线程
1 扩展java.lang.Thread类以及实现java.lang.Runnable接口。
此类中有run()方法,public void run(),如果该线程是独立的Runnable运行对象构造的,则调用该Runnable对象的run()方法;否则,该方法不执行任何操作。Thread的子类也应该重写该方法。
三 实例化线程
1 如果是扩展了java.lang.Thread类的线程,则直接调用new即可。
2 如果是实现了jav.lang.Runnable接口的类,则调用Thread的构造方法:
Thread(Runnable target)
Thread(Runnable target,String name)
Thread(ThreadGroup group, Runnable target)
Thread(ThreadGroup group, Runnable target, String name)
Thread(ThreadGroup group, Runnable target, String name, long stackSize)
四 启动线程
在线程的Thread对象上调用start()方法,而不是run()或别的方法。
在调用start()方法之前,线程处于新状态中,新状态有一个Thread对象,但没有一个真正的线程。
在调用start()方法之后,发生了一系列复杂的事情:
启动新的执行线程(具有新的调用栈);
该线程从新状态转移到可运行状态;
当该线程获得执行机会时,其目标run()方法将运行。
五 注意事项
1 获取当前线程的对象的方法是:Thread.currentThread();
2 当线程目标run()方法结束时该线程完成。
3 一旦线程启动,它就永远不能再重新启动。只有一个新的线程可以被启动,并且只能一次。一个可运行的线程或死线程可以被重新启动。
4 线程的调度是JVM的一部分,在一个CPU的机器上上,实际上一次只能运行一个线程。一次只有一个线程栈执行。JVM线程调度程序决定实际运行哪个处于可运行状态的线程。众多可运行线程中的某一个会被选中做为当前线程。可运行线程被选择运行的顺序是没有保障的。
5 尽管通常采用队列形式,但这是没有保障的。队列形式是指当一个线程完成“一轮”时,它移到可运行队列的尾部等待,直到它最终排队到该队列的前端为止,它才能被再次选中。事实上,我们把它称为可运行池而不是一个可运行队列,目的是帮助认识线程并不都是以某种有保障的顺序排列一个队列的事实。
6 尽管我们没有无法控制线程调度程序,但可以通过别的方式来影响线程调度的方式。
六 示例
当点击start按钮时,程序从屏幕左上角弹出一个球,这个球开始移动,调用addBall方法,循环运行1000次move。每调用一次move,球就会移动一点,当碰到墙壁时,就会调整方向,但是这个程序有个弊端:当你想在移动1000次之前,就想退出程序,点击close发现,其仍在移动。
Thread类的静态方法sleep()将暂停给定的毫秒数。调用Thread.sleep不会创建一个新线程,sleep是Thread类的静态方法,用于暂停当前线程活动。
Bounce.java
技术分享
1 package Thread;
2 import java.awt.*;
3 import java.awt.event.*;
4 import javax.swing.*;
5 public class BounceThread {
6 public static void main(String[] args){
7 EventQueue.invokeLater(new Runnable(){
8 public void run(){
9 JFrame frame=new BounceFrame();
10 frame.setTitle("BounceFrame");
11 frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
12 frame.setVisible(true);
13 }
14 });
15 }
16 }
17 /*class BallRunnable implements Runnable{
18 private Ball ball;
19 private Component component;
20 public static final int STEPS=1000;
21 public static final int DELAY=5;
22 public BallRunnable(Ball aBall,Component aComponent){
23 ball=aBall;
24 component=aComponent;
25 }
26 public void run(){
27 try{
28 for(int i=1;i<=STEPS;i++){
29 ball.move(component.getBounds());
30 component.repaint();
31 Thread.sleep(DELAY);
32 }
33 }
34 catch(InterruptedException e){}
35 }
36 }*/
37 class BounceFrame extends JFrame{
38 private BallComponent comp;
39 public static final int STEPS=1000;
40 public static final int DELAY=100;
41 public BounceFrame(){
42 comp=new BallComponent();
43 add(comp,BorderLayout.CENTER);
44 JPanel buttonPanel=new JPanel();
45 addButton(buttonPanel,"Start",new ActionListener(){
46 public void actionPerformed(ActionEvent event){
47 addBall();
48 }
49 });
50 addButton(buttonPanel,"Close",new ActionListener(){
51 public void actionPerformed(ActionEvent event){
52 System.exit(0);
53 }
54 });
55 add(buttonPanel,BorderLayout.SOUTH);
56 pack();
57 }
58 public void addButton(Container c,String title,ActionListener listener){
59 JButton button=new JButton(title);
60 c.add(button);
61 button.addActionListener(listener);
62 }
63 /*public void addBall(){
64 Ball b=new Ball();
65 comp.add(b);
66 Runnable r=new BallRunnable(b,comp);
67 Thread t=new Thread(r);
68 t.start();
69 }*/
70 public void addBall(){
71 try{
72 Ball ball=new Ball();
73 comp.add(ball);
74 for(int i=1;i<=STEPS;i++){
75 ball.move(comp.getBounds());
76 comp.paint(comp.getGraphics());
77 Thread.sleep(DELAY);
78 }
79 }
80 catch(InterruptedException e){}
81 }
82 }
View Code
BollComponent.java
技术分享
1 package Thread;
2 import java.awt.*;
3
4 import java.util.*;
5 import javax.swing.*;
6 public class BallComponent extends JPanel{
7 private static final int DEFAULT_WIDTH=450;
8 private static final int DEFAULT_HEIGHT=350;
9 private java.util.List<Ball>balls=new ArrayList<>();
10 public void add(Ball b){
11 balls.add(b);
12 }
13 public void paintComponent(Graphics g){
14 super.paintComponent(g);
15 Graphics2D g2=(Graphics2D)g;
16 for(Ball b:balls){
17 g2.fill(b.getShape());
18 }
19 }
20 public Dimension getPreferredSize(){
21 return new Dimension(DEFAULT_WIDTH,DEFAULT_HEIGHT);
22 }
23 }
View Code
Ball.java
技术分享
1 package Thread;
2 import java.awt.geom.*;
3 import java.awt.geom.Ellipse2D.Double;
4 public class Ball {
5 private static final int XSIZE=15;
6 private static final int YSIZE=15;
7 private double x=0;
8 private double y=0;
9 private double dx=1;
10 private double dy=1;
11 public void move(Rectangle2D bounds){
12 x+=dx;
13 y+=dy;
14 if(x<bounds.getMinX()){
15 x=bounds.getMinX();
16 dx=-dx;
17 }
18 if(x+XSIZE>=bounds.getMaxX()){
19 x=bounds.getMaxX()-XSIZE;
20 dx=-dx;
21 }
22 if(y<bounds.getMinY()){
23 y=bounds.getMinY();
24 dy=-dy;
25 }
26 if(y+YSIZE>=bounds.getMaxY()){
27 y=bounds.getMaxY()-YSIZE;
28 dy=-dy;
29 }
30 }
31 public Ellipse2D getShape(){
32 return new Ellipse2D.Double(x,y,XSIZE,YSIZE);
33 }
34 }
View Code
针对上述的情况,下面的代码是改进后的,当点击close时,就会退出当前线程。
BounceThread.java
技术分享
1 package Thread;
2 import java.awt.*;
3 import java.awt.event.*;
4 import javax.swing.*;
5 public class BounceThread {
6 public static void main(String[] args){
7 EventQueue.invokeLater(new Runnable(){
8 public void run(){
9 JFrame frame=new BounceFrame();
10 frame.setTitle("BounceFrame");
11 frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
12 frame.setVisible(true);
13 }
14 });
15 }
16 }
17 class BallRunnable implements Runnable{
18 private Ball ball;
19 private Component component;
20 public static final int STEPS=1000;
21 public static final int DELAY=5;
22 public BallRunnable(Ball aBall,Component aComponent){
23 ball=aBall;
24 component=aComponent;
25 }
26 public void run(){
27 try{
28 for(int i=1;i<=STEPS;i++){
29 ball.move(component.getBounds());
30 component.repaint();
31 Thread.sleep(DELAY);
32 }
33 }
34 catch(InterruptedException e){}
35 }
36 }
37 class BounceFrame extends JFrame{
38 private BallComponent comp;
39 //public static final int STEPS=1000;
40 //public static final int DELAY=100;
41 public BounceFrame(){
42 comp=new BallComponent();
43 add(comp,BorderLayout.CENTER);
44 JPanel buttonPanel=new JPanel();
45 addButton(buttonPanel,"Start",new ActionListener(){
46 public void actionPerformed(ActionEvent event){
47 addBall();
48 }
49 });
50 addButton(buttonPanel,"Close",new ActionListener(){
51 public void actionPerformed(ActionEvent event){
52 System.exit(0);
53 }
54 });
55 add(buttonPanel,BorderLayout.SOUTH);
56 pack();
57 }
58 public void addButton(Container c,String title,ActionListener listener){
59 JButton button=new JButton(title);
60 c.add(button);
61 button.addActionListener(listener);
62 }
63 public void addBall(){
64 Ball b=new Ball();
65 comp.add(b);
66 Runnable r=new BallRunnable(b,comp);
67 Thread t=new Thread(r);
68 t.start();
69 }
70 /*public void addBall(){
71 try{
72 Ball ball=new Ball();
73 comp.add(ball);
74 for(int i=1;i<=STEPS;i++){
75 ball.move(comp.getBounds());
76 comp.paint(comp.getGraphics());
77 Thread.sleep(DELAY);
78 }
79 }
80 catch(InterruptedException e){}
81 }*/
82 }
View Code
BollComponent.java
技术分享
1 package Thread;
2 import java.awt.*;
3
4 import java.util.*;
5 import javax.swing.*;
6 public class BallComponent extends JPanel{
7 private static final int DEFAULT_WIDTH=450;
8 private static final int DEFAULT_HEIGHT=350;
9 private java.util.List<Ball>balls=new ArrayList<>();
10 public void add(Ball b){
11 balls.add(b);
12 }
13 public void paintComponent(Graphics g){
14 super.paintComponent(g);
15 Graphics2D g2=(Graphics2D)g;
16 for(Ball b:balls){
17 g2.fill(b.getShape());
18 }
19 }
20 public Dimension getPreferredSize(){
21 return new Dimension(DEFAULT_WIDTH,DEFAULT_HEIGHT);
22 }
23 }
View Code
Ball.java
技术分享
1 package Thread;
2 import java.awt.geom.*;
3 import java.awt.geom.Ellipse2D.Double;
4 public class Ball {
5 private static final int XSIZE=15;
6 private static final int YSIZE=15;
7 private double x=0;
8 private double y=0;
9 private double dx=1;
10 private double dy=1;
11 public void move(Rectangle2D bounds){
12 x+=dx;
13 y+=dy;
14 if(x<bounds.getMinX()){
15 x=bounds.getMinX();
16 dx=-dx;
17 }
18 if(x+XSIZE>=bounds.getMaxX()){
19 x=bounds.getMaxX()-XSIZE;
20 dx=-dx;
21 }
22 if(y<bounds.getMinY()){
23 y=bounds.getMinY();
24 dy=-dy;
25 }
26 if(y+YSIZE>=bounds.getMaxY()){
27 y=bounds.getMaxY()-YSIZE;
28 dy=-dy;
29 }
30 }
31 public Ellipse2D getShape(){
32 return new Ellipse2D.Double(x,y,XSIZE,YSIZE);
33 }
34 }
以上是关于Java线程:什么是线程的主要内容,如果未能解决你的问题,请参考以下文章
Java多线程 -- 什么是线程和进程线程和进程的联系(Java内存结构)并发与并行为何使用多线程使用多线程可能带来什么问题