Java初级笔记No.5(更新中)

2018.7.30

1、Java发送邮件

在机器上安装JavaMail API和Java Activation Framework(JAF)。
具体可以参考java发送邮件

2、Java多线程编程

Java中线程的几个状态
· 新建状态: 使用new关键字和Thread类或子类建立一个线程对象后,该线程对象就处于新建状态。保持这个状态直到程序start()这个线程;
· 就绪状态: 当线程对象调用了start()方法之后,该线程就进入了就绪状态,就绪状态的线程处于就绪队列中,要等待JVM里线程调度器的调度;
· 运行状态: 如果就绪状态的线程获取CPU资源,就可以执行run(),此时线程便处于运行状态。处于运行状态的线程最为复杂,它可以变为阻塞状态、就绪状态、和死亡状态。
· 阻塞状态: 如果一个线程执行了sleepsuspend(挂起)等方法,失去所占用资源之后,该线程就从运行状态进入阻塞状态。在睡眠时间已到或获得设备资源后可以重新进入就绪状态。可以分为三种: 1)等待阻塞: 运行状态中的线程执行wait()方法使线程进入到等待阻塞状态; 2) 同步阻塞: 线程在获取synchronized同步锁失败(因为同步锁被其他线程占用); 3) 其他阻塞: 通过调用线程的sleep()join()发出了I/O请求时,线程就会进入到阻塞状态。当sleep()状态超时时,join()等待线程终止或超时,或者I/O处理完毕,线程重新转入就绪状态。

· 死亡状态: 一个运行状态的线程完成任务或者其他终止条件发生时,该线程就切换到终止状态。

3、线程的优先级
每个Java线程都有一个优先级,这样有助于操作系统确定线程的调度顺序。
Java线程的优先级是一个整数,范围是1(Thread.MIN_PRIORITY)- 10(Thread.MAX_PRIORITY)。
默认情况下,每一个线程都会分配一个优先级NORM_PRIORITY(5)
具有较高优先级的线程对程序更重要,并且应该在低优先级的线程之前分配处理器资源。但是线程优先级不能保证线程执行的顺序,而且非常依赖于平台。

4、创建一个线程
Java有三种创建线程的方法:
· 通过实现Runnable接口;
· 通过继承Thread类本身;
· 通过CallableFuture创建线程。

5、通过实现Runnable接口来创建线程
创建一个线程最简单的方法是实现Runnable接口的类。
为了实现Runnable,一个类只需要执行一个方法调用run()

public void run()

可以重写该方法,重要的是理解run()可以调用其他方法,使用其他类,并声明变量,就像主线程一样。

在创建一个实现Runnable接口的类之后,可以在类中实例化一个线程对象:

Thread(Runnable threadOb, String threadName);

这里threadOb是一个实现Runnable接口的类的实例,并且threadName指定新线程的名字。
新线程创建后,调用start()方法运行。

实例

package test4;

class RunnableDemo implements Runnable {
    private Thread t;
    private String threadName;
    
    RunnableDemo(String name) {
        threadName = name;
        System.out.println("Creating " + threadName);
    }
    
    public void run() {
        System.out.println("Running " + threadName);
        try {
            for (int i = 4; i > 0; --i) {
                System.out.println("Thread: " + threadName + ", " + i);
                //睡眠
                Thread.sleep(50);
            }
        } catch(InterruptedException e) {
            System.out.println("Thread " + threadName + " interrupted.");
        }
        System.out.println("Thread " + threadName + " exiting.");
    }
    
    public void start() {
        System.out.println("Starting " + threadName);
        if (t == null) {
            t = new Thread(this, threadName);
            t.start();
        }
    }
}

public class TestThread {
    public static void main(String args[]) {
        RunnableDemo R1 = new RunnableDemo("Thread-1");
        R1.start();
        
        RunnableDemo R2 = new RunnableDemo("Thread-2");
        R2.start();
    }
}

6、通过继承Thread来创建线程
创建一个线程的第二种方法时创建一个新的类,该类继承Thread类,然后创建一个该类的实例。
继承类必须重写run()方法,该方法时新线程的入口点。它必须调用start()方法才能执行。
该方法的本质其实也是实现了Runnable接口的一个实例。

实例

package test4;

class ThreadDemo extends Thread {
    private Thread t;
    private String threadName;
    
    ThreadDemo(String name) {
        threadName = name;
        System.out.println("Creating " + threadName);
    }
    
    public void run() {
        System.out.println("Running " + threadName);
        try {
            for (int i = 4; i > 0; --i) {
                System.out.println("Thread: " + threadName + ", " + i);
                
                Thread.sleep(50);
            }
        } catch(InterruptedException e) {
            System.out.println("Thread " + threadName + " interrupted.");
        }
        System.out.println("Thread " + threadName + "exiting.");
    }
    
    public void start() {
        System.out.println("Starting " + threadName);
        if (t == null) {
            t = new Thread(this, threadName);
            t.start();
        }
    }
}

public class TestThread2 {
    public static void main(String[] args) {
        ThreadDemo T1 = new ThreadDemo("Thread-1");
        T1.start();
        
        ThreadDemo T2 = new ThreadDemo("Thread-2");
        T2.start();
    }
}

public class TestThread2 {

}

7、通过Callable 和 Future创建线程
· 创建Callable接口的实现类,并实现call()方法,该call()方法将作为线程执行体,并且有返回值;
· 创建Callable实现类实例,使用FutureTask类来包装Callable对象,该FutureTask对象封装了该Callable对象的call()方法的返回值; · 使用FutureTask对象作为Thread对象的target创建并启动新线程; · 调用FutureTask对象的get()`方法来获得子线程执行结束后的返回值。

package test4;

import java.util.concurrent.ExecutionException;

public class CallableThreadTest implements Callable<Integer> {
    public static void main(String[] args) {
        CallableThreadTest ctt = new CallableThreadTest();
        FutureTask<Integer> ft = new FutureTask<>(ctt);
        for (int i = 0; i < 100; ++i) {
            System.out.println(Thread.currentThread().getName()+ "的循环变量i的值" + i);
            
            if (i == 20) {
                new Thread(ft, "有返回值的线程").start();
            }
        }
        try {
            System.out.println("子线程的返回值: " + ft.get());
        } catch(InterruptedException e) {
            e.printStackTrace();
        } catch(ExecutionException e) {
            e.printStackTrace();
        }
    }
    
    @Override
    public Integer call() throws Exception {
        int i = 0;
        for (; i<100; ++i) {
            System.out.println(Thread.currentThread().getName()+ " " + i);
        }
        return i;
    }
}

8、创建线程的三种方式对比
· 采用实现RunnableCallable接口的方式创建多线程时,线程类知识实现了Runnable接口或Callable接口,还可以继承其他类;
· 使用继承Thread类的方式创建多线程时,编写简单,如果需要访问当前线程,则无需使用Thread.currentThread()方法,直接使用this即可获得当前线程。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 进程和线程 进程 所有运行中的任务通常对应一个进程,当一个程序进入内存运行时,即变成一个进程.进程是处于运行过程中...
    小徐andorid阅读 2,865评论 3 53
  •   一个任务通常就是一个程序,每个运行中的程序就是一个进程。当一个程序运行时,内部可能包含了多个顺序执行流,每个顺...
    OmaiMoon阅读 1,719评论 0 12
  • 怎么办,有点羡慕了...... 越来越多的人过生日秀礼物,秀party!而我今天生日,我只有自己,看到别人...
    珊海阅读 233评论 0 0
  • 2018.6.5 晴 星期二 今天想写的内容,是继昨天的文章内容的。 关键词:坚持 美好 从小,我有很...
    林音书阅读 1,319评论 40 20
  • 帮朋友闺女完成的幼儿园作业。年年有余挂饰
    一个人的烟火Amber阅读 240评论 1 1