论坛首页 入门技术论坛

corejava辅导(11--2)

浏览 993 次
精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (0)
作者 正文
   发表时间:2008-12-03  

共享数据的并发处理

 

多线程同时并发访问的资源叫做临界资源。

 

多个线程同时访问对象并要求操作相同资源时分割了原子操作就会出现问题。(原子操作,不可再分的操作)会出现数据的不一致或数据不完整,为避免这种现象采用对访问的线程做限制的方法。

Synchronized关键字

 

1Synchronized修饰代码块(同步代码块),

   public void push(char c){

        synchronized(this)//只有持有当前对象的锁标记才能访问这个代码块

     {

              ...

        }

   }

 

对括号内的对象加锁,只有拿到锁标记的对象才能执行该代码块

2Synchronized修饰方法

   public synchronized void push(char c) {

      ...

      }

对当前对象的加锁,只有拿到锁标记的对象才能执行该方法。

 

互斥锁机制,利用每个对象都有一个monitor(锁标记),当线程拥有这个锁标记时才能访问加了同步的这个资源,没有锁标记便进入锁池。任何一个对象系统都会为其创建一个互斥锁,这个锁是为了分配给线程的,防止打断原子操作。每个对象的锁只能分配给一个线程

 

注意:构造方法不能Synchronized修饰,静态方法可以用Synchronized修饰(是对类对象加锁,类对象会在反射时讲到)。抽象方法不能用Synchronized修饰,不影响子类覆盖,子类在覆盖这个方法是可以加Synchronized,也可以不加Synchronized,所以根据Java不允许写废代码的特点是不能写在一起。

 

注意:对当前对象加锁,一个代码块或者方法是同步的(Synchronized),当前对象的锁标记没有分配出去时,这个对象的锁标记为分配时,线程来访问这个代码块,会得到这个对象的锁标记,直到这个线程结束才会释放着个锁标记,其他访问这个代码块或者是方法线程就会进入这个对象锁池,没有得到当前对象的锁标记,不能访问这个代码块或者方法。当一个线程想要获得某个对象锁标记而进入锁池,这个线程又持有其他对象的锁标记,这个线程也不会释放其持有的锁标记。

 

注:方法的Synchronized特性本身不会被继承。

线程因为未拿到锁标记而发生阻塞进入锁池(lock pool)。每个对象都有自己的一个锁池的空间,用于放置等待运行的线程。由系统决定哪个线程拿到锁标记并运行。

 

使用互斥锁的注意事项

 

举例:男孩和女孩例子,每个女孩是一个对象,每个男孩是个线程。每个女孩都有自己的锁池。每个男孩可能在锁池里等待

 

class Girl{

     public void hand(){

 

     }

     public syncronized void kiss(){

 

     }

}

class Boy extends Thread{

     public void run(){

        

     }

}

 

注意:只读不用加同步,只写也不用加同步,只有读写操作兼而有之时才加同步。

 

注意:在java.io包中Vector HashTable 之所以是线程安全的,是因为每个方法都有synchronized修饰。Static 方法可以加 synchronized , 锁的是类对象。但是Vector jdk 1.0   ArrayList jdk1.2 所以实际应用还是使用ArrayList

 

注意:内同步,外同步,内同步,即,类内的方法加同步(synchronized)修饰,外同步即,在需要控制只能由一个线程进行访问时,把需要控制的方法写在同步代码块里。

论坛首页 入门技术版

跳转论坛:
Global site tag (gtag.js) - Google Analytics