java线程基础

1.通过继承Thread类创建线程

public class Thread extends Object implements Runnabel

  • Thread类已经实现了Runnable接口
  • 继承Tread的类只需重写run方法
  • Thread类已经定义了final类型的getName和setName方法
  • 启动线程应调用start方法而不是run方法
  • 一个线程对象只能调用一次start方法,否则会抛IllegalThreadStartException错误
package com.gyp.thread.start;

public class Test1 extends Thread{
    private String name;
    
    public Test1(){
    }
    public Test1(String name){
        setName(name);
    }
    public void run(){
        for(int i=0;i<10;i++){
            System.out.println("当前运行的线程:"+getName()+",i="+i);
        }
    }
    
    public static void main(String[] args){
        Test1 t1 = new Test1();
        Test1 t2 = new Test1("线程2");
        t1.start();
        t2.start();
        //受单继承的影响
//      t2.start(); 
    }
}

2.通过实现Runnable接口创建线程

public interface Runnable{ public abstract void run(); }

  • Runnable只提供了run方法
  • Runnable对线程没有任何支持,在获得线程实例后必须通过Thread类的构造函数来实现
package com.gyp.thread.start;

public class Test2 implements Runnable {
    private String name;
    
    //实现接口的类要自己定义方法
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Test2(){
    }
    
    public Test2(String name){
        setName(name);
    }
    
    public void run(){
        for(int i=0;i<10;i++){
            System.out.println("当前运行的线程:"+getName()+",i="+i);
        }
    }
    
    public static void main(String[] args){
        Test2 r1 = new Test2();
        Test2 r2 = new Test2("线程2");
        Thread t1 = new Thread(r1);
        Thread t2 = new Thread(r2);
        t1.start();
        t2.start();
    }
}

2.5继承Thread类和实现Runnable接口的区别

  • 一般选择后者
    • 避免单继承带来的局限性
    • 适合多个相同程序代码的线程区处理同一资源的情况。
  • 前者处理同一资源可用静态变量。

3.主线程-main()

  • 主线程是自动创建的,是产生应用程序所有其他线程的线程。
  • 程序中只要还有其他非守护线程没有结束,主线程就不会结束,即使主线程已执行完或者调用了retuen。
  • 一个java程序最少有两个线程,一个是main,优先级5,一个是JVM垃圾收集线程
package com.gyp.thread.start;

public class Test3 implements Runnable {
    public void run(){
        for(int i=0; i<10; i++){
            System.out.println("当前运行线程:"+Thread.currentThread()+", i="+i);
        }
    }
    
    public static void main(String[] args){
        //获取当前对象的线程引用
        Thread mi = Thread.currentThread();
        System.out.println("线程的信息:"+mi);
        System.out.println("线程的名字:"+mi.getName());
        mi.setName("主线程");
        System.out.println("线程的名字:"+mi.getName());
        
        Test3 r = new Test3();
        Thread t = new Thread(r);
        t.start();
        t.run();    //直接调用run方法
    }
}

4.线程的加入join()

使当前运行的线程停下来,等待调用join的线程执行,可设最长等待时间,也可不设等他执行完。

package com.gyp.thread.start;

import java.util.Date;

/**
 * @author gyp19
 * 线程的加入join()
 * 等待调用的线程终止
 */

class ThreadTest4 implements Runnable{
    public void run(){
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        for(int i=0;i<5;i++){
            System.out.println("当前运行的线程:"+Thread.currentThread().getName()+" ,i="+i+"时间:"+new Date());
        }
    }
}
public class Test4 {
    public static void main(String[] arge){
        ThreadTest4 r = new ThreadTest4();
        Thread t = new Thread(r,"线程一");
        Thread mi = Thread.currentThread();//获得主线程的引用
        t.start();
        for(int i=0;i<10;i++){
            if(i==5){
                try {
                    t.join(3000);   //最大等待时间3s
//                  mi.join(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            System.out.println("main="+i+",时间:"+new Date());
        }
        
    }
}

5.线程的唤醒 interrupt()

线程从休眠的组赛状态(sleep)转化为运行状态。

package com.gyp.thread.start;

import java.util.Date;

class ThreadTest5 implements Runnable{
    public void run(){
        System.out.println("线程开始的时间:"+new Date());
        for(int i=0;i<5;i++){
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
//              e.printStackTrace();
                System.out.println("线程被唤醒啦啦啦");
            }
            System.out.println(Thread.currentThread().getName()+"的第"+i+1+"秒"+", 时间:"+new Date());
        }
        System.out.println("线程的结束时间:"+new Date());
    }
}

public class Test5 {
    public static void main(String[] args){
        ThreadTest5 r = new ThreadTest5();
        Thread t = new Thread(r,"线程一");
        t.start();
        try {
            Thread.sleep(2000);//使用t.sleep(2000);的效果也是一样一样的
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        t.interrupt();      //唤醒线程
    }
}

6.后台线程

  1. 后台执行的线程,也可称之为守护线程。
  2. JVM垃圾回回收线程就是典型的守护线程。
  3. 后台线程为其他线程服务,当所有非后台线程结束时后台线程才结束。
  4. main()是一个非后台线程。

setDaemon(true) : 设置后台线程
isDaemon() : 判断是否为后台线程,是则true

package com.gyp.thread.start;

public class Test6 implements Runnable{
    
    public void run(){
        int i=0;
        while(true){
            System.out.println("i="+(i++));
        }
    }
    
    public static void main(String[] args){
        Test6 r = new Test6();
        Thread t = new Thread(r,"线程一");
        t.setDaemon(true);
        if(t.isDaemon()){
            t.start();
        }
        System.out.println("say something");
    }
}

7.线程的礼让

1.当前正在运行的线程推出运行状态,暂时让其他线程执行。
2.不能让给指定的线程,一般优先级高的线程会先当到运行资源。
3.sleep和yield的区别

  • sleep让出运行权时不考虑别的线程的优先级。
  • sleep转到堵塞状态,yield转到就绪状态。
  • sleep会抛出异常。
  • slepp有着更好的移植性,不能依靠yield方法提高程序的并发性。

yield() : 调用礼让

package com.gyp.thread.start;

class ThreadDemo1 implements Runnable{
    public void run(){
        for(int i=0;i<10;i++){
            System.out.println("礼让->"+Thread.currentThread().getName()+":i="+i);
            //线程的礼让
            Thread.currentThread().yield();
        }
    }
}
class ThreadDemo2 implements Runnable{
    public void run(){
        for(int i=0;i<10;i++){
            System.out.println(Thread.currentThread().getName()+":i="+i);
        }
    }
}

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

推荐阅读更多精彩内容

  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,454评论 1 15
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,957评论 1 18
  • 写在前面的话: 这篇博客是我从这里“转载”的,为什么转载两个字加“”呢?因为这绝不是简单的复制粘贴,我花了五六个小...
    SmartSean阅读 4,730评论 12 45
  • 该文章转自:http://blog.csdn.net/evankaka/article/details/44153...
    加来依蓝阅读 7,353评论 3 87
  • 前言- CPU竞争策略 操作系统中,CPU竞争有很多种策略。Unix系统使用的是时间片算法,而Windows则属于...
    zhanglbjames阅读 450评论 1 1