A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© j420984 中级黑马   /  2015-6-21 17:53  /  461 人查看  /  6 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

多线程
概述:
一.        进程和线程的概念
a)        进程:是一个正在执行中的程序,每一个进程执行都有一个执行顺序,该顺序是一个执行路径,或者叫一个控制单元。
b)        线程:就是进程中的一个独立的控制单元,线程在控制着进程的执行。一个进程中至少有一个线程。
c)        Java VM启动的时候会有一个进程java.exe,该进程中至少一个线程负责java程序的执行,而且这个线程运行的代码存在于main方法中。该线程称之为主线程。
i.        扩展:从细节上说明JVM,JVM启动不止一个线程,还有负责垃圾回收机制的线程。
二.        如何在自定义的代码中自定义一个线程
a)        通过对API的查找,java已经提供了对线程这类事物的描述,就是Thread类。
b)        创建线程的第一种方式:继承Thread类。
i.        定义类继承Thread类
ii.        复写Thread类中的run()方法。
iii.        创建并启动一个线程。调用线程的start()方法。该方法有两个作用,启动线程; 调用run()方法。
c)        代码如下:
public class LearnTread extends Thread {
        public void run() {
                System.out.println("+++");
        }
}

public class ThreadTest {
        public static void main(String[] args) {
                LearnTread lt = new LearnTread();
                lt.start();
        }
}

d)        多个线程都获取CPU的执行权时,CPU执行到谁,谁就运行,在某一个时刻,只能有一个程序在运行。CPU在做着快速的切换,以达到看上去是同时运行的效果。我们可以形象的把多线程的运行行为在相互抢夺CPU的执行权。所以每一次运行的结果都不相同。
e)        多线程的特性:随机性,谁抢到谁执行,至于执行多长,CPU说的算。
三.        为什么要覆盖run()方法?
a)        Thread类用于描述线程,该类就定义了一个功能,用于储存线程要运行的代码,该存储功能就是run()方法,也就是说Thread类中的run()方法,用于存储线程要运行的代码。
b)        注意:d.start()开启线程并执行该线程的run();方法。d.run()仅仅是对象调用方法,而线程创建了,并没有运行。
四.        练习:创建两个线程,和主线程交替运行。
a)        代码如下:

public class ThreadTest {
        public static void main(String[] args) {
                LearnTread lt1 = new LearnTread("1");
                LearnTread lt2 = new LearnTread("2");
                lt1.start();
                lt2.start();
                for (int i = 0; i < 60; i++) {
                        System.out.println("main");
                }
        }
}
public class LearnTread extends Thread {
        private String name;

        LearnTread(String name) {
                this.name = name;
        }

        public void run() {
                for (int i = 0; i < 60; i++) {
                        System.out.println(this.name);
                }
        }
}

五.        线程的四种状态
a)        被创建,运行,冻结,消亡;
六.        获取线程对象以及名称
a)        线程都有自己默认的名称,Thread-编号,该编号从0开始。
b)        static Thread currentThread():获取当前线程对象。
c)        getName():获取线程名称。
d)        设置线程名称:setName或者构造函数。
七.        创建线程的第二种方式,实现Runnalble接口
a)        定义类实现Runnable接口
b)        覆盖Runnable接口中的run方法
c)        通过Thread类建立线程象
d)        将Runnable接口的子类对象作为实际参数传递给Thread类的构造方法。为什么要将Runnalbe接口的子类对象传递给Thread的构造函法———因为自定义的run方法所属的对象是Runnable接口的子类对象,所以要让线程去指定对象的run方法,就必须该run方法所属的对象。
e)        调用Thread类的start方法开启线程并调用Runnable接口子类的run方法。
f)        实现方式和继承方式的区别:
i.        实现方式的好处,避免了单继承的局限性,定义线程时,建议使用实现方式
ii.        继承Thread:线程代码存放在Thread子类run方法中
iii.        实现Runnable:线程代码存放在接口子类的run方法中。
八.        多线程的安全问题
a)        通过分析,发现可能会出现-1.-2.0等错票。
b)        原因:当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一部分,还没有执行完,另一个线程参与进来执行,导致共享数据的错误。
c)        解决方法:对多条操作共享数据的语句只能让一个线程都执行完,再执行过程中,其他线程不可以参与执行。
d)        java对于多线程的安全问题提供了专业的解决方式:同步代码块。sychronized(对象){需要被同步的代码}对象如同锁,持有锁的线程,可以在同步中执行,不执有锁的线程,即使获取CPU的执行权,也进不去,因为没有获取锁。
e)        同步的前提:
i.        必须要有两个或者两个以上的线程。
ii.        必须是多个线程使用同一个锁。
iii.        必须保证同步中只能有一个线程在运行。
iv.        好处:解决了多线程的安全问题
v.        弊端:多个线程需要判断锁,比较消耗资源。
vi.        同步的第二种表现开式:同步函数
1.        同步函数用的是哪一个锁:同步函数需要被对象调用,那么函数都有一个所属对象引用,就是this所以同步函数的锁是this
2.        静态同步函数的锁是Class对象。
3.        静态方法中不可以定义this。
4.        静态进内存时,内存中没有本类对象,但是一定有该类对应的字节码文件,就是类名.class,该对象的类型是Class。

九.        死锁
a)        同步中嵌套同步,而锁不同。

public class DeadLockTest implements Runnable {
        private boolean flag;

        DeadLockTest(boolean flag) {
                this.flag = flag;
        }

        public void run() {
                if(flag){
                        synchronized(MyLock.locka){
                                System.out.println("if locka");
                                synchronized(MyLock.lockb){
                                        System.out.println("if lockb");
                                }
                        }
                }else{
                        synchronized(MyLock.lockb){
                                System.out.println("else lockb");
                                synchronized(MyLock.locka){
                                        System.out.println("else locka");
                                }
                        }
                }
        }
}

6 个回复

倒序浏览
写的很好,楼主真厉害
回复 使用道具 举报
不错不错哦
回复 使用道具 举报
支持下
回复 使用道具 举报
哇,楼主辛苦了
回复 使用道具 举报
总结的很好....
回复 使用道具 举报
顶一下!!
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马