一、线程的生命周期:新建状态 就绪状态 运行状态 阻塞状态 死亡状态
二、创建线程的三种方式:
1、创建类继承Thread
2、创建类实现Runnable接口传给Thread的构造方法(或直接匿名内部类)
3、先创建一个未来任务对象,构造方法的参数为新建类继承Callable接口(或匿名内部类),未来任务对象传给Thread类的构造方法。优点:有返回值,缺点:获取返回值必须等到该线程完全结束。效率低
三、线程的常用方法 :
1、线程对象.strat();启动线程
2、String 线程对象.getName(); 获取线程名字
3、线程对象.setName();修改线程名字
4、Thread.currentThread();获取当前线程对象
5、Thread.sleep();使当前线程睡眠进入阻塞状态
运行结果
6、线程对象.interrupt(); 叫醒线程对象(利用异常机制)
7、线程对象.stop();终止线程(已过时),有更好的方法终止线程
只运行了5秒
8、Thread.getPriority();获取当前线程优先级
9、Thread.setPriority();修改当前线程优先级
线程优先级最高:10,默认优先级:5,最低优先级:1
优先级越高抢夺到cpu时间片的概率越高
10、Thread.yield();当前线程暂停,回到就绪状态
11、线程对象.join();线程合并,其他线程停止,直到调用join方法的线程执行结束
12、线程对象.setDaemon();守护线程, 守护线程会在用户线程结束后结束
13、wait()和notify()方法
四、线程安全问题(synchronized关键字)同步编程
两个线程共享一个对象,同时对对象的成员变量或实例变量进行修改,会产生线程不安全问题。
比如去银行取钱,去柜台取的同时也在自动取款机取款。余额为1w,柜台和取款机同时取1w,如果其中一个方法来不及更新卡余额数据,就会都取出1w块。代码模拟如下:(简单演示,就不封装了..)
余额1w,取了2w,余额0.明显出了问题。这时候就要使用synchronized关键字,语法结构
1、synchronized(共享对象){
}
把共享对象当成锁,哪个线程先到先把锁从锁池拿走,这时候其他需要这把锁的线程只能在就绪状态等着,当抢到的cpu时间片用完后还锁。
2、写在实例方法前,表示对象锁。与第一个语法结构比,()内默认是这个对象,以上例子明显不共享AccountThread的对象,而是共享Account 的对象act,所以无法产生作用。
3、放在静态方法前,表示类锁。
对象锁:1个对象1把锁,100个对象100把锁。
类锁:100个对象,也可能只是1把类锁。
4、死锁,一般出现在synchronized嵌套,发生死锁现象不报错,不显示任何东西,需要避免。
sleep记得抛异常,不会长截屏删了好截的全一点
运行结果:
五、定时器
定时器的作用:间隔特定的时间,执行特定的程序。