Java多线程(上篇)

每一个分支都叫做一个线程,main()叫做主分支,也叫主线程。

程只是一个静态的概念,机器上的一个.class文件,机器上的一个.exe文件,这个叫做一个进程。程序的执行过程都是这样的:首先把程序的代码放到内存的代码区里面,代码放到代码区后并没有马上开始执行,但这时候说明了一个进程准备开始,进程已经产生了,但还没有开始执行,这就是进程,所以进程其实是一个静态的概念,它本身就不能动。平常所说的进程的执行指的是进程里面主线程开始执行了,也就是main()方法开始执行了。进程是一个静态的概念,在我们机器里面实际上运行的都是线程。

Windows操作系统是支持多线程的,它可以同时执行很多个线程,也支持多进程,因此Windows操作系统是支持多线程多进程的操作系统。Linux和Uinux也是支持多线程和多进程的操作系统。DOS就不是支持多线程和多进程了,它只支持单进程,在同一个时间点只能有一个进程在执行,这就叫单线程。

CPU难道真的很神通广大,能够同时执行那么多程序吗?不是的,CPU的执行是这样的:CPU的速度很快,一秒钟可以算好几亿次,因此CPU把自己的时间分成一个个小时间片,我这个时间片执行你一会,下一个时间片执行他一会,再下一个时间片又执行其他人一会,虽然有几十个线程,但一样可以在很短的时间内把他们通通都执行一遍,但对我们人来说,CPU的执行速度太快了,因此看起来就像是在同时执行一样,但实际上在一个时间点上,CPU只有一个线程在运行。

学习线程首先要理清楚三个概念:

进程:进程是一个静态的概念

线程:一个进程里面有一个主线程叫main()方法,是一个程序里面的,一个进程里面不同的执行路径。

在同一个时间点上,一个CPU只能支持一个线程在执行。因为CPU运行的速度很快,因此我们看起来的感觉就像是多线程一样。
  什么才是真正的多线程?如果你的机器是双CPU,或者是双核,这确确实实是多线程。

一、进程(process)

  1. 程序是在某个操作系统中运行的,在系统中运行的程序就叫进程,进程是操作系统中的概念或叫法,程序=进程

  2. 程序在运行期间,要占用CPU、内存等这些资源,这些资源的管理和调配都是由操作系统负责的

  3. 一个程序至少对应一个进程,有些程序采用的是多进程架构设计

  4. 进程间相互独立,不能互相访问或共享资源

二、线程(thread)

1.进程包含若干个线程 如果一个进程中只包含了一个线程,就叫单线程程序,如果一个进程中只包含了多个线程,就叫多线程程序

2.线程是程序(进程)中更小的一个执行单元

3.火车 - 某个车厢 QQ - 每个聊天窗口

4.一个进程可以拥有多个线程,但至少要有一个线程,即主执行线程(java的main方法),我们之前写的都是单线程程序。

5.线程不能够单独执行,它必须运行在处于活动状态的进程中,它可与同属一个进程的其他线程共享进程所拥有的全部资源

三、多线程编程

1.JVM是从main方法开始执行的,执行的时候把main方法当成一个线程去执行

2.作用 效果 :能让某些代码“同时”执行 让给两个for循环同时执行

3.大部分功能还是单线程即可,但是有时候也需要多线程:上传图片

A.一张一张传

B.多张图片同时上传

4.好处:提过代码执行的效率(多线程可以充分利用CPU和内存的资源)

5.生活案例:工厂刚成立时只有一条生产线,随着市场的开拓,为了提高生产效率,工厂不断增加生产线(生产线 = 线程)

四、多线程跟CPU

1.一心无二用,一个人能同时干多件事情吗?(边看片儿边写字)那计算机是怎么实现的呢?
真正的“同时”是不存在的,CPU在以极快的速度随机轮流执行程序,专业说法是程序在快速轮流抢占CPU资源。速度太快,人感觉不到

2.胶片电影

3.一个(程序)进程中如果只有一条生产线,这个程序称为单线程。
一个(程序)进程中如果有多条生产线,这个程序称为多线程。

4.多个线程同时执行时,CPU会随机性的、极快的、轮流的、执行每个线程。

五、Java的多线程

1.Java中有两类线程:User Thread(用户线程)、Daemon Thread(守护线程或后台服务线程) 。
守护线程的作用是为其他线程的运行提供服务,比如说GC线程。

2.在Java中实现多线程有两种方式:

方式一. 继承Thread类

方式二. 实现Runnable接口

六、方式一

1.一般的类都是很普通的类,类中的方法或其他代码都不能同时执行,都不是生产线类

2.在Java中,专门提供了一个现成的生产线类:Thread类

3.我们定义的普通的类可以继承Thread类,从而让这些普通类变成生产线类

4.步骤:

A.继承Thread类

B.重写run方法,把需要同时执行的代码放进去

C.在main方法中创建几条生产线

D.使用start()方法启动生产线

七、Thread类

public long getId(),获得该线程的标识符(线程 ID) ,它在创建该线程时生成。线程 ID 是唯一的,并终生不变。

public final void setName(String name),给线程命名,不命名的话都有默认的,格式为:Thread-0、Thread-1...

public final String getName(),获得线程的名字

public void start(),使该线程开始执行,JVM会自动调用该线程的 run() 方法。

public static void sleep(long millis),让线程休眠(暂停)多少毫秒

public static Thread currentThread(),获得当前正在执行的线程对象的引用。(方式二会用到)

八、多线程程序执行的流程

1.从main开始执行,这就是一个线程了,主线程

2.开始执行main方法中的代码,那么代码中的线程称之为其他线程

3.单线程程序,只要main方法执行完就表示程序结束;多线程程序,所有的线程都执行完,才表示程序结束

4.主线程的“主”不表示先执行或先执行完,表示这个线程是必须有的

九、方式二

1.自定义一个类实现Runnable接口,实现里面的run方法(需要同时执行的代码) ----不是生产线类

2.在main中创建几条生产线(Thread),需要把它要干的活儿传给它

3.启动生产线

当前线程:CPU当前正在干活的那个线程(currentThread())

代码举例:



/**
 * 继承Thread,生产线类
 */
public class T02_ThreadStudent extends Thread {

    public void run(){   //同时要执行的代码
        for (int i = 1; i <= 10; i++) {
            System.out.println(getName()+"在看第"+i+"页书");
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

/**
 * 实现Runnable接口,并不是生产线类
 */
public class T02_RunnableStudent implements Runnable{

    public void run() {
        for (int i = 1; i <= 10; i++) {
            System.out.println(Thread.currentThread().getName()+"在看第"+i+"页书");
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        
    }
}

/**
 * Created by AFinalStone on 2017/6/30.
 */
public class T02_ExampleRunnableAndThread {

    public void testRunnable() {
        Thread t1 =new Thread(new T02_RunnableStudent());   //创建了一条生产线
        Thread t2 =new Thread(new T02_RunnableStudent());   //创建了一条生产线

        t1.start();   //启动
        t2.start();
    }

    public void testThread() {
        //用方式一实现两个学生同时看书(10页/100ms)
        //多线程编程
        //1.同时干什么事      看书    这部分代码应该放在run方法中
        //2.哪些名词应该充当生产线     生产线类
        //3.应该创建或启动几条生产线

        T02_ThreadStudent ts1=new T02_ThreadStudent();   //创建了一条生产线
        ts1.setName("张三");
        T02_ThreadStudent ts2=new T02_ThreadStudent();  //创建了一条生产线
        ts2.setName("李四");

        ts1.start();
        ts2.start();

        //思考:为什么不直接调用run()方法,而是调用start()方法来启动线程?
        //run()其实就是一个很普通的方法,而start()方法内部调用了run(),可以当成生产线去执行
    }

}

十、线程安全问题

1.概念:多个线程操作的是同一个共享资源,但是线程之间是彼此独立的,因此就会出现数据不同步更新的情况,这就是线程安全问题

2.条件:

a.多个线程

b.有共享资源

c.每个线程都操作或使用了共享资源(线程之间是互相独立的)

3.模拟实现:除夕那天北京到杭州的动车票只剩下10张,两个窗口同时在卖

a.同时要干的活:卖这10张票

b.自定义一个类型(不是生产线类)

c.在main中创建两个生产线(窗口)并启动

4.在代码中,共享资源怎么体现出来?共享资源一般通过属性来编写出来,在创建生产线时需要传进去同一个对象(run())

5.怎么解决线程安全问题?

大概的思路:就是把某些代码当成一个整体,就是给某些代码上锁 同步(锁)

6.在火车的车厢里上厕所

7.Java同步机制的实现方式:

a.同步代码块

b.同步方法(函数)

十一、同步代码快

1.语法格式:

synchronized(对象){    //可以使用任意对象的锁,包括this
    操作共享数据的代码即需要同步的代码;
}

锁:监视器 遍地都是锁 任意一个对象内部都有一个监视器(锁)

十二、同步方法

public synchronized void sell(){  
    //锁是固定的this,即当前对象
    操作共享数据的代码即需要同步的代码;
}

把需要上锁的代码摘出来放到一个方法中,但是这个方法必须是同步方法,然后在run()中进行调用
同步方法的锁是固定不变的this,一个类中既可以有同步方法也可以有普通方法

  • 代码举例:

public class T03_SaleWindow implements Runnable {

    private int ticket = 10;   //火车票的编号     共享资源

    public void run() {

        for (int i = 0; i < 10; i++) {
        {
            if (ticket >= 1) {
                System.out.println(Thread.currentThread().getName()
                        + "在卖编号为" + ticket + "的火车票");
                ticket--;
                try {
                    Thread.sleep(500);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            } else {
                System.out.println("没票了");
                break;
            }
        }
        }
    }
}
public class T03_SaleWindowSynchronized01 implements Runnable{
    
    private int  ticket = 10;   //火车票的编号     共享资源

    public void run() {
        
        for (int i = 0; i < 10; i++) {
            synchronized (this) {   //同步代码快
                if (ticket >= 1) {
                    System.out.println(Thread.currentThread().getName()
                            + "在卖编号为" + ticket + "的火车票");
                    ticket--;
                    try {
                        Thread.sleep(500);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    System.out.println("没票了");
                    break;
                }
            }
        }
    }
}

public class T03_SaleWindowSynchronized02 implements Runnable {

    private int ticket = 10; // 火车票的编号 共享资源

    public synchronized void sale() {

        if (ticket >= 1) {
            System.out.println(Thread.currentThread().getName() + "在卖编号为"
                    + ticket + "的火车票");
            ticket--;
            try {
                Thread.sleep(500);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        } else {
            System.out.println("没票了");
        }

    }

    public void run() {
        for (int i = 0; i < 10; i++) {
            sale();
        }
    }
}
public class T03_ExampleSaleWindow {

    public void testSaleWindow(){
        T03_SaleWindow t03_saleWindow = new T03_SaleWindow();

        Thread t1 = new Thread(t03_saleWindow);   //窗口1
        Thread t2 = new Thread(t03_saleWindow);   //窗口2

        t1.start();
        t2.start();
    }

    public void testSaleWindowBySynchronized01(){
        T03_SaleWindowSynchronized01 sw = new T03_SaleWindowSynchronized01();

        Thread t1 = new Thread(sw);   //窗口1
        Thread t2 = new Thread(sw);   //窗口2

        t1.start();
        t2.start();
    }

    public void testSaleWindowBySynchronized02(){

        T03_SaleWindowSynchronized02 sw=new T03_SaleWindowSynchronized02();

        Thread t1=new Thread(sw);
        Thread t2=new Thread(sw);

        t1.start();
        t2.start();
    }
}

十三、细节

1.在解决线程安全问题时,多个线程使用的锁必须是同一把锁

同步的优劣:

2.优:能解决线程安全问题

3.劣:耗费大量资源,影响程序执行的速度(效率)

线程安全和性能不能兼得

4.静态方法能不能同步?类自身

项目地址:传送门

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352

推荐阅读更多精彩内容

  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,452评论 1 15
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,955评论 1 18
  • 又来到了一个老生常谈的问题,应用层软件开发的程序员要不要了解和深入学习操作系统呢? 今天就这个问题开始,来谈谈操...
    tangsl阅读 4,119评论 0 23
  • 一、认识多任务、多进程、单线程、多线程 要认识多线程就要从操作系统的原理说起。 以前古老的DOS操作系统(V 6....
    GT921阅读 1,013评论 0 3
  • 林炳文Evankaka原创作品。转载自http://blog.csdn.net/evankaka 本文主要讲了ja...
    ccq_inori阅读 653评论 0 4