如何同步操作?

时间:2011-04-10 11:42:31

标签: java multithreading locking deadlock synchronized

我有两阶段锁定的实现。问题是它在大多数情况下都是完美的,但不是全部。我弄清楚问题来了,因为我使用了Synchronized,这是以错误的方式使用的一些方法!我需要操作顺序如下:

  1. 解锁任何锁的线程应该在任何其他线程开始锁定同一个锁之前完全完成此阶段(ul方法)。
  2. 当它被唤醒时等待锁定的线程应该在它开始执行它的操作之前获得该锁定的所有服务员。
  3. 任何addEdge或removeEdge都需要同步,因为我在所有线程中都使用相同的图形对象。
  4. 以下是代码:

    class LockTable{
         private HashMap<Integer,MyLock> locks;
         public LockTable(){
             locks= new HashMap<Integer,MyLock>();   
        }
    
    /*******************************************************************************
     * Acquire a shared/read lock on data object oid.
     * @param tid  the transaction id
     * @param oid  the data object id
     */
    void rl (int tid, int oid) throws InterruptedException
    {
    
        MyLock lock=null;
        boolean wait = false;
        boolean getIt = false;
        synchronized(this) {
            try {
                lock = locks.get(oid);             // find the lock
    
                if((lock != null) && (lock.lock.isWriteLocked())){
    
                   wait = true;
                }
                if(lock == null){
                 getIt = true;
                 lock = new MyLock(tid, true);
                 lock.lock.readLock().lock();
                 lock.readers.add(tid);
                 locks.put(oid, lock);
              }
    
            } catch(Exception e) {
                System.out.println(e.getStackTrace());        // lock not found, so oid is not locked;
            } // try
        }//synch
        if (wait){
            synchronized(this){
                //System.out.println("Transaction " + tid + " is waiting..");
                lock.waiters.add(tid);
                lock.waitersType.add('s');
                Main.g.addEdge(tid, lock.tid);
                //System.out.println("Edge has been added "+tid + "==>" + lock.tid);
            }//sync
             if(Main.g.hasCycle()){
                //System.out.println("Detect Cycle in rl..Executing Restart");
                restart(tid);
            }
            //to exclude the restarted thread
            if(!Main.trans[tid].terminate){
                lock.lock.readLock().lock();
                lock.readers.add(tid);
              synchronized(this){
                lock.waitersType.remove(lock.waiters.indexOf(tid));
                lock.waiters.remove(lock.waiters.indexOf(tid));
                }//sync
                for(int i =0 ; i < lock.waiters.size();i++){
                    if(lock.waitersType.get(i) == 'w'){
                        synchronized(Main.g){
                        Main.g.addEdge(lock.waiters.get(i), tid);
                        //System.out.println("Edge has been added "+lock.waiters.get(i) + "==>" + tid);
                        }//sync
                        if(Main.g.hasCycle())
                            restart(lock.waiters.get(i));
                    }//if lock.waitersType
                }//for
            }//if terminate
            else
                return;
        }
       else
        if(!getIt){
                  //System.out.println("Getting Shared Lock without Waiting");
                  lock.lock.readLock().lock();
                  lock.readers.add(tid);
                  for(int i =0 ; i < lock.waiters.size();i++){
                    if(lock.waitersType.get(i) == 'w'){
                        synchronized(Main.g){
                        Main.g.addEdge(lock.waiters.get(i), tid);
                        //System.out.println("Edge has been added "+lock.waiters.get(i) + "==>" + tid);
                        }
                        if(Main.g.hasCycle())
                            restart(lock.waiters.get(i));
                    }//if lock.waitersType
                  } 
        }
    
    } // rl
    
    /*******************************************************************************
     * Acquire an exclusive/write lock on data object oid.
     * @param tid  the transaction id
     * @param oid  the data object id
     */
    void wl (int tid, int oid) throws InterruptedException
    {
         //type to determine the last lock type in order
         //to be able to remove the edges from waitfor graph
        int type = 0;
        MyLock lock = null;
        boolean wait = false;
        boolean getIt = false;
        synchronized(this) {
            try {
                lock = locks.get(oid);             // find the lock
                if(lock != null && (lock.lock.isWriteLocked() || lock.readers.size() > 0))
                {
                    wait = true;
                }
                if(lock == null){
                    getIt = true;
                    lock = new MyLock(tid);
                    lock.lock.writeLock().lock();
                    locks.put(oid,lock);
                }
            } catch(Exception e) {
                System.out.println(e.getStackTrace());        // lock not found, so oid is not locked;
            } // try
      }
         if (wait){
                //System.out.println("Transaction " + tid + " is waiting..");
               synchronized(this) {
                if(lock.lock.isWriteLocked()){
                    Main.g.addEdge(tid, lock.tid);
                    //System.out.println("Edge has been added "+tid + "==>" + lock.tid);
                   }
                else{
                    type = 1;
                    for(int reader : lock.readers){
                         Main.g.addEdge(tid, reader);
                         //System.out.println("Edge has been added "+tid + "==>" + reader);
                    }
                    }//else
                if(Main.g.hasCycle()){
                   //System.out.println("Detect Cycle");
                   restart(tid);
                 }//if
                //System.out.println("Graph: "+Main.g.toString());
             }//sync
    
    
    
           if(!Main.trans[tid].terminate){
                synchronized(this){
                lock.waiters.add(tid);
                lock.waitersType.add('w');
               }
                //System.out.println("I'm waiting here in wl");
                lock.lock.writeLock().lock();
                lock.tid = tid;
                //System.out.println("Wakeup..");
                synchronized(this){
                    lock.waitersType.remove(lock.waiters.indexOf(tid));
                    lock.waiters.remove(lock.waiters.indexOf(tid));
                    //System.out.println("the number of waiters after wakeup: " + lock.waiters.size());
               }//sync
                    for(int i =0 ; i < lock.waiters.size();i++){
                        synchronized(Main.g){
                        Main.g.addEdge(lock.waiters.get(i), tid);
                       // System.out.println("Edge has been added "+lock.waiters.get(i) + "==>" + tid);
                        //System.out.println("Graph: "+Main.g.toString());
                        }//sync
                        if(Main.g.hasCycle())
                            restart(lock.waiters.get(i));
                    }//for
             }
            else
                return; 
        }// if(wait) ==> for the lock to be released
        else
            if(!getIt){
                lock.lock.writeLock().lock();
                lock.tid = tid;
                for(int i =0 ; i < lock.waiters.size();i++){
                    synchronized(Main.g){
                    Main.g.addEdge(lock.waiters.get(i), tid);
                    //System.out.println("Edge has been added "+lock.waiters.get(i) + "==>" + tid);
                    }
                    if(Main.g.hasCycle())
                        restart(lock.waiters.get(i));
                }//for
    
         }
    
    } // wl
    
    void restart(int tid){
     synchronized(this) {
        Main.rollBack++;
        MyLock lock;
        List<Integer> toRemove = new ArrayList();
        for(int i : locks.keySet()){
           lock = locks.get(i);
           //for all the locks in the lock table delete the restarted thread from the waiters list
           if(lock.waiters.contains(tid)){
               lock.waitersType.remove(lock.waiters.indexOf(tid));
               lock.waiters.remove(lock.waiters.indexOf(tid));
            }
    
               //lock.sem.release();
               if(lock.lock.isWriteLockedByCurrentThread()){
    
    
                   //remove the edges between the waiters of this lock and the thread that unlocked it
                      for(int j=0;j<lock.waiters.size();j++)
                          Main.g.removeEdge(lock.waiters.get(j), lock.tid);
                   //System.out.println("Transaction"+tid+" unlock object "+ i +" in order to restart");
                   lock.lock.writeLock().unlock();
                    //System.out.println("number of write holders: " + lock.lock.writeLock().getHoldCount());
                    //System.out.println("number of read holders: " + lock.lock.getReadHoldCount());
                    //System.out.println("number of waiters: " + lock.lock.getQueueLength());
                   toRemove.add(i);
    
               }
           if(!lock.lock.isWriteLocked())
               if(lock.readers.contains(tid) && lock.lock.getReadLockCount()>0){
                      //remove the edges between the waiters of this lock and the thread that unlocked it
                      for(int j=0;j<lock.waiters.size();j++)
                          Main.g.removeEdge(lock.waiters.get(j), tid);
                       // lock.numberOfReaders --;
                      //System.out.println("Transaction"+tid+" unlock object "+ i +" in order to restart");
                      lock.readers.remove(lock.readers.indexOf(tid));
                      lock.lock.readLock().unlock();
                      //System.out.println("number of write holders: " + lock.lock.getWriteHoldCount());
                      //System.out.println("number of read holders: " + lock.lock.getReadHoldCount());
                      //System.out.println("number of waiters: " + lock.lock.getQueueLength());
                      toRemove.add(i);  
    
                  }//if
        }//for
        Main.g.removeEdges(tid);
        Main.trans[tid].terminate = true;
        //System.out.println("Transaction" + tid + " restarted");
    
        }//sync
    }
    
    /*******************************************************************************
     * Unlock/release the lock on data object oid.
     * @param tid  the transaction id
     * @param oid  the data object id
     */
    void ul (int tid, int oid)
    {
       MyLock lock = null;
        boolean error = false;
        synchronized(this) {
            try {
                lock = locks.get(oid);                    // find the lock
                if( lock == null)
                    System.out.println("println: lock not found");
            } catch(Exception e) {
                System.out.println("lock not found");   // lock not found
            } // try
        }
            if((lock != null) && (lock.lock.isWriteLockedByCurrentThread())){
                      //remove the edges between the waiters of this lock and the thread that unlocked it
                      for(int i=0;i<lock.waiters.size();i++)
                          synchronized(Main.g){
                          Main.g.removeEdge(lock.waiters.get(i), lock.tid);
                          }//sync
                       //System.out.println("tid: " + tid + " unlock object: " + oid);
                       lock.lock.writeLock().unlock();
                      //print out 
                      //System.out.println("done with unlock");
                      //System.out.println("number of write holders: " + lock.lock.writeLock().getHoldCount());
                      //System.out.println("number of read holders: " + lock.lock.getReadHoldCount());
                      //System.out.println("number of waiters: " + lock.lock.getQueueLength());
          }// if lock != null
            else
              if((lock != null) && (lock.readers.size()>0)){
                  synchronized(this){
                  if(lock.readers.contains(tid)){
                    lock.readers.remove(lock.readers.indexOf(tid));
                    }
                    //remove the edges between the waiters of this lock and the thread that unlocked it
                      for(int i=0;i<lock.waiters.size();i++)
                          synchronized(Main.g){
                          Main.g.removeEdge(lock.waiters.get(i), tid);
                          }
                    lock.lock.readLock().unlock();
                    //System.out.println("Transaction"+tid+" unlocked shared lock on object "+oid);
                    //System.out.println("number of write holders: " + lock.lock.readLock().);
                    //System.out.println("number of read holders: " + lock.lock.getReadHoldCount());
                    //System.out.println("number of waiters: " + lock.lock.getQueueLength());
    
                  }//if lock.readers
              }//if
        if (error) 
            System.out.println ("Error: ul: no lock for oid = " + oid + " found/owned");
    } // ul
    

1 个答案:

答案 0 :(得分:1)

使用synchronized时会出现几个问题。

  1. 除非极少数情况下,对共享对象的所有读/写(尤其是写)访问都必须在同一对象上同步。在某些地方写信时,您正在Main.g进行同步,但是有些地方没有同步(例如if(wait)方法的wl内)。

  2. 据我了解您的代码的目标,lock.{waiters|readers}的状态必须同步与图的状态(Main.g)。这意味着每当您对waiters进行更改,然后相应地更新Main.g时,必须以原子方式执行这两个操作。要做到这一点,您需要将它们包装在一个未破坏的synchronized语句中。我可以看到你理解这个概念,因为你在某些地方这样做,但是你似乎在其他地方错过了它。例如:在rl方法中,在if(!getIt){内更新lock.readers,然后Main.g,而不进行任何同步。

  3. 总的来说,我不能给你任何具体的东西,因为代码非常复杂,所以很难说出它的意图。但我认为您可以通过锁定代码的较大部分来解决一些锁定问题。例如,您可以通过在方法声明中添加rl关键字,在wl上同步整个方法ulthissynchronized,并摆脱所有方法方法内部的细粒度锁定。细粒度锁定可以提供更好的性能,但代价是高复杂性。我建议你先从简单的锁定方案开始,然后逐步改进它们(如果你真的需要!)。