多线程设计要点

2016-02-19 18:55 5 1 收藏

下面这个多线程设计要点教程由图老师小编精心推荐选出,过程简单易学超容易上手,喜欢就要赶紧get起来哦!

【 tulaoshi.com - 编程语言 】


  1.多线程中有主内存和工作内存之分, 在JVM中,有一个主内存,专门负责所有线程共享数据;而每个线程都有他自己私有的工作内存, 主内存和工作内存分贝在JVM的stack区和heap区。
  
  2.线程的状态有'Ready', 'Running', 'Sleeping', 'Blocked', 和 'Waiting'几个状态,
  
  'Ready' 表示线程正在等待CPU分配答应运行的时间。
  
  3.线程运行次序并不是按照我们创建他们时的顺序来运行的,CPU处理线程的顺序是不确定的,假如需要确定,那么必须手工介入,使用setPriority()方法设置优先级。
  
  4.我们无从知道一个线程什么时候运行,两个或多个线程在访问同一个资源时,需要synchronized
  
  5. 每个线程会注册自己,实际某处存在着对它的引用,因此,垃圾回收机制对它就“束手无策”了。
  
  6. Daemon线程区别一般线程之处是:主程序一旦结束,Daemon线程就会结束。
  
  7. 一个对象中的所有synchronized方法都共享一把锁,这把锁能够防止多个方法对通用内存同时进行的写操作。synchronized static方法可在一个类范围内被相互间锁定起来。
  
  8. 对于访问某个要害共享资源的所有方法,都必须把它们设为synchronized,否则就不能正常工作。
  
  9. 假设已知一个方法不会造成冲突,最明智的方法是不要使用synchronized,能提高些性能。
  
  10. 假如一个"同步"方法修改了一个变量,而我们的方法要用到这个变量(可能是只读),最好将自己的这个方法也设为 synchronized。
  
  11. synchronized不能继续, 父类的方法是synchronized,那么其子类重载方法中就不会继续“同步”。
  
  12. 线程堵塞Blocked有几个原因造成:
  
  (1)线程在等候一些IO操作 
  
  (2)线程试图调用另外一个对象的“同步”方法,但那个对象处于锁定状态,暂时无法使用。
  
  13.原子型操作(atomic), 对原始型变量(primitive)的操作是原子型的atomic. 意味着这些操作是线程安全的, 但是大部分情况下,我们并不能正确使用,来看看 i = i + 1 , i是int型,属于原始型变量:
  
  (1)从主内存中读取i值到本地内存. 
  
  (2)将值从本地内存装载到线程工作拷贝中. 
  
  (3)装载变量1. 
  
  (4)将i 加 1. 
  
  (5)将结果给变量i. 
  
  (6)将i保存到线程本地工作拷贝中. 
  
  (7)写回主内存. 
  
  注重原子型操作只限于第1步到第2步的读取以及第6到第7步的写, i的值还是可能被同时执行i=i+1的多线程中断打搅(在第4步)。
  
  double 和long 变量是非原子型的(non-atomic)。数组是object 非原子型。
  
  14. 由于13条的原因,我们解决办法是:
  
  class xxx extends Thread{
  
  //i会被经常修改
  
  private int i;
  
  public synchronized int read(){ return i;}
  
  public synchronized void update(){ i = i + 1;}
  
  ..........
  
  }
  
  15. Volatile变量, volatile变量表示保证它必须是与主内存保持一致,它实际是"变量的同步", 也就是说对于volatile变量的操作是原子型的,如用在long 或 double变量前。
  
  16. 使用yield()会自动放弃CPU,有时比sleep更能提升性能。
  
  17. sleep()和wait()的区别是:wait()方法被调用时会解除锁定,但是我们能使用它的地方只是在一个同步的方法或代码块内。
  
  18. 通过制造缩小同步范围,尽可能的实现代码块同步,wait(毫秒数)可在指定的毫秒数可退出wait;对于wait()需要被notisfy()或notifyAll()踢醒。
  
  19. 构造两个线程之间实时通信的方法分几步:
  
  (1). 创建一个PipedWriter和一个PipedReader和它们之间的管道;
  
  PipedReader in = new PipedReader(new PipedWriter())
  
  (2). 在需要发送信息的线程开始之前,将外部的PipedWriter导向给其内部的Writer实例out
  
  (3). 在需要接受信息的线程开始之前,将外部的PipedReader导向给其内部的Reader实例in
  
  (4). 这样放入out的所有东西度可从in中提取出来。
  
  20. synchronized带来的问题除性能有所下降外,最大的缺点是会带来死锁DeadLock,只有通过谨慎设计来防止死锁,其他毫无办法,这也是线程难以驯服的一个原因。不要再使用stop() suspend() resume()和destory()方法
  
  21. 在大量线程被堵塞时,最高优先级的线程先运行。

来源:http://www.tulaoshi.com/n/20160219/1619496.html

延伸阅读
Mutex是互斥体,广泛地应用在多线程编程中。本文以广为流程的Doug Lea的concurrent工具包的Mutex实现为例,进行一点探讨。在Doug Lea的concurrent工具包中,Mutex实现了Sync接口,该接口是concurrent工具包中所有锁(lock)、门(gate)和条件变量(condition)的公共接口,Sync的实现类主要有:Mutex、Semaphore及其子类、Latch、Cou...
多线程是Java语言的一大特性,多线程就是同时存在N个执行体,按几条不同的执行线索共同工作的情况。程序,进程,线程,可以从不同的角度去理解。 !-- frame contents -- !-- /frame contents -- 程序就是一段静态的代码,可以理解成一组计算机命令的集合.进行就是这个程序一次动态的执行过程,从代码的加载到执行完毕的一个过...
一、理解多线程 多线程是这样一种机制,它允许在程序中并发执行多个指令流,每个指令流都称为一个线程,彼此间互相独立。 线程又称为轻量级进程,它和进程一样拥有独立的执行控制,由操作 系统负责调度,区别在于线程没有独立的存储空间,而是和所属进程中的其它线程共享一个存储空间,这使得线程间的通信远较进程简单。 多个线程的执行...
import java.io.*;//多线程编程public class MultiThread{public static void main(String args[]){System.out.println("我是主线程!");//下面创建线程实例thread1ThreadUseExtends thread1=new ThreadUseExtends();//创建thread2时以实现了Runnable接口的THhreadUseRunnable类实例为参数Thread thread2=new Thread(new ThreadU...
大家都知道,在开发过程中应该尽可能减少用户等待时间,让程序尽可能快的完成运算。可是无论是哪种语言开发的程序最终往往转换成汇编语言进而解释成机器码来执行。但是机器码是按顺序执行的,一个复杂的多步操作只能一步步按顺序逐个执行。改变这种状况可以从两个角度出发:对于单核处理器,可以将多个步骤放到不同的线程,这样一来用户完成UI...

经验教程

754

收藏

79
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部