⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 routingtable.java

📁 pastry的java实现的2.0b版
💻 JAVA
字号:
/*************************************************************************"FreePastry" Peer-to-Peer Application Development Substrate Copyright 2002, Rice University. All rights reserved.Redistribution and use in source and binary forms, with or withoutmodification, are permitted provided that the following conditions aremet:- Redistributions of source code must retain the above copyrightnotice, this list of conditions and the following disclaimer.- Redistributions in binary form must reproduce the above copyrightnotice, this list of conditions and the following disclaimer in thedocumentation and/or other materials provided with the distribution.- Neither  the name  of Rice  University (RICE) nor  the names  of itscontributors may be  used to endorse or promote  products derived fromthis software without specific prior written permission.This software is provided by RICE and the contributors on an "as is"basis, without any representations or warranties of any kind, expressor implied including, but not limited to, representations orwarranties of non-infringement, merchantability or fitness for aparticular purpose. In no event shall RICE or contributors be liablefor any direct, indirect, incidental, special, exemplary, orconsequential damages (including, but not limited to, procurement ofsubstitute goods or services; loss of use, data, or profits; orbusiness interruption) however caused and on any theory of liability,whether in contract, strict liability, or tort (including negligenceor otherwise) arising in any way out of the use of this software, evenif advised of the possibility of such damage.********************************************************************************/package rice.pastry.routing;import rice.environment.Environment;import rice.environment.logging.Logger;import rice.pastry.*;import java.util.*;/** * The Pastry routing table. <P> * * The size of this table is determined by two constants: <P> * * * <UL> *   <LI> {@link rice.pastry.Id#nodeIdBitLength nodeIdBitLength}which determines *   the number of bits in a node id (which we call <EM>n</EM> ). *   <LI> {@link RoutingTable#idBaseBitLength idBaseBitLength}which is the base *   that table is stored in (which we call <EM>b</EM> ). * </UL> * <P> * * We write out node ids as numbers in base <EM>2 <SUP>b </SUP> </EM> . They * will have length <EM>D = ceiling(log <SUB>2 <SUP>b </SUP> </SUB> 2 <SUP>n * </SUP>)</EM> . The table is stored from <EM>0...(D-1)</EM> by <EM>0...(2 * <SUP>b </SUP> - 1)</EM> . The table stores a set of node handles at each * entry. At address <EM>[index][digit]</EM> , we store the set of handles were * the most significant (numerically) difference from the node id that the table * routes for at the <EM>index</EM> th digit and the differing digit is <EM> * digit</EM> . An <EM>index</EM> of <EM>0</EM> is the least significant digit. * * @version $Id: RoutingTable.java 3274 2006-05-15 16:17:47Z jeffh $ * @author Andrew Ladd * @author Peter Druschel */public class RoutingTable extends Observable implements NodeSetEventSource {  /**   * The routing calculations will occur in base <EM>2 <SUP>idBaseBitLength   * </SUP></EM>   */  public byte idBaseBitLength;  // = 4;  private Id myNodeId;  /**   * DESCRIBE THE FIELD   */  public NodeHandle myNodeHandle;  private RouteSet routingTable[][];  private int maxEntries;  Logger logger;  ArrayList listeners = new ArrayList();  /**   * Constructor.   *   * @param me the node id for this routing table.   * @param max the maximum number of entries at each table slot.   * @param base DESCRIBE THE PARAMETER   * @param env DESCRIBE THE PARAMETER   */  public RoutingTable(NodeHandle me, int max, byte base, Environment env) {    logger = env.getLogManager().getLogger(RoutingTable.class, null);    idBaseBitLength = base;    myNodeId = me.getNodeId();    myNodeHandle = me;    maxEntries = max;    int cols = 1 << idBaseBitLength;    int rows = Id.IdBitLength / idBaseBitLength;    routingTable = new RouteSet[rows][cols];    for (int i = 0; i < rows; i++) {      int myCol = myNodeId.getDigit(i, idBaseBitLength);      // insert this node at the appropriate column      routingTable[i][myCol] = new RouteSet(maxEntries);      routingTable[i][myCol].put(myNodeHandle);      routingTable[i][myCol].setRoutingTable(this);    }  }  /**   * Gets the set of handles at a particular entry in the table.   *   * @param index the index of the digit in base <EM>2 <SUP>idBaseBitLength   *      </SUP></EM> .<EM>0</EM> is the least significant.   * @param digit ranges from <EM>0... 2 <SUP>idBaseBitLength - 1 </SUP> </EM> .   *      Selects which digit to use.   * @return a read-only set of possible handles located at that position in the   *      routing table, or null if none are known   */  public RouteSet getRouteSet(int index, int digit) {    RouteSet ns = routingTable[index][digit];    return ns;  }  /**   * Gets the set of handles that match at least one more digit of the key than   * the local Id.   *   * @param key the key   * @return a read-only set of possible handles, or null if none are known   */  public RouteSet getBestEntry(Id key) {    int diffDigit = myNodeId.indexOfMSDD(key, idBaseBitLength);    if (diffDigit < 0) {      return null;    }    int digit = key.getDigit(diffDigit, idBaseBitLength);    return routingTable[diffDigit][digit];  }  /**   * Gets the node handle associated with a given id.   *   * @param nid a node id   * @return the handle associated with that id, or null if none is known.   */  public NodeHandle get(Id nid) {    RouteSet ns = getBestEntry(nid);    if (ns == null) {      return null;    }    return ns.get(nid);  }  /**   * Get a row from the routing table.   *   * @param i which row   * @return an array which is the ith row.   */  public RouteSet[] getRow(int i) {    return routingTable[i];  }  /**   * return ths number of columns in the routing table   *   * @return number of columns   */  public int numColumns() {    return routingTable[0].length;  }  /**   * return the number of rows in the routing table   *   * @return number of rows   */  public byte numRows() {    return (byte) routingTable.length;  }  /**   * return the bit length of the base   *   * @return baseBitLength   */  public byte baseBitLength() {    return idBaseBitLength;  }  /**   * Determines an alternate hop numerically closer to the key than the one we   * are at. This assumes that bestEntry did not produce a live nodeHandle that   * matches the next digit of the key.   *   * @param key the key   * @return a nodeHandle of a numerically closer node, relative to the key   */  public NodeHandle bestAlternateRoute(Id key) {    return bestAlternateRoute(NodeHandle.LIVENESS_SUSPECTED, key);  }  /**   * Determines an alternate hop numerically closer to the key than the one we   * are at. This assumes that bestEntry did not produce a live nodeHandle that   * matches the next digit of the key.   *   * @param key the key   * @param minLiveness DESCRIBE THE PARAMETER   * @return a nodeHandle of a numerically closer node, relative to the key   */  public NodeHandle bestAlternateRoute(int minLiveness, Id key) {    final int cols = 1 << idBaseBitLength;    int diffDigit = myNodeId.indexOfMSDD(key, idBaseBitLength);    if (diffDigit < 0) {      return null;    }    int keyDigit = key.getDigit(diffDigit, idBaseBitLength);    int myDigit = myNodeId.getDigit(diffDigit, idBaseBitLength);    Id.Distance bestDistance = myNodeId.distance(key);    NodeHandle alt = null;    boolean finished = false;    for (int i = 1; !finished; i++) {      for (int j = 0; j < 2; j++) {        int digit = (j == 0) ? (keyDigit + i) & (cols - 1)          : (keyDigit + cols - i) & (cols - 1);        RouteSet rs = getRouteSet(diffDigit, digit);        for (int k = 0; rs != null && k < rs.size(); k++) {          NodeHandle n = rs.get(k);          if (n.getLiveness() <= minLiveness          /*           *  isAlive()           */            ) {            Id.Distance nDist = n.getNodeId().distance(key);            if (bestDistance.compareTo(nDist) > 0) {              bestDistance = nDist;              alt = n;            }          }        }        if (digit == myDigit) {          finished = true;        }      }    }    return alt;  }  /**   * Determines a set of alternate hops towards a given key.   *   * @param key the key   * @param max the maximal number of alternate hops requested   * @return a set of nodehandles, or null if no alternate hops exist   */  public NodeSet alternateRoutes(Id key, int max) {    NodeSet set = new NodeSet();    final int cols = 1 << idBaseBitLength;    int diffDigit = myNodeId.indexOfMSDD(key, idBaseBitLength);    if (diffDigit < 0) {      return set;    }    int keyDigit = key.getDigit(diffDigit, idBaseBitLength);    int myDigit = myNodeId.getDigit(diffDigit, idBaseBitLength);    Id.Distance myDistance = myNodeId.distance(key);    boolean finished = false;    int count = 0;    for (int i = 0; !finished; i++) {      for (int j = 0; j < 2; j++) {        int digit = (j == 0) ? (keyDigit + i) & (cols - 1)          : (keyDigit + cols - i) & (cols - 1);        RouteSet rs = getRouteSet(diffDigit, digit);        for (int k = 0; rs != null && k < rs.size(); k++) {          NodeHandle n = rs.get(k);          if (n.isAlive()) {            Id.Distance nDist = n.getNodeId().distance(key);            if (set != null && count < max && myDistance.compareTo(nDist) > 0) {              set.put(n);              count++;            }          }        }        if (digit == myDigit) {          finished = true;        }      }    }    return set;  }  /**   * Like getBestEntry, but creates an entry if none currently exists.   *   * @param key the key   * @return a read-only set of possible handles   */  private RouteSet makeBestEntry(Id key) {    int diffDigit = myNodeId.indexOfMSDD(key, idBaseBitLength);    if (diffDigit < 0) {      return null;    }    int digit = key.getDigit(diffDigit, idBaseBitLength);    if (routingTable[diffDigit][digit] == null) {      // allocate a RouteSet      routingTable[diffDigit][digit] = new RouteSet(maxEntries);      routingTable[diffDigit][digit].setRoutingTable(this);    }    return routingTable[diffDigit][digit];  }  /**   * Puts a handle into the routing table.   *   * @param handle the handle to put.   */  public void put(NodeHandle handle) {    if (logger.level <= Logger.FINER) {      logger.log("RT: put(" + handle + ")");    }    Id nid = handle.getNodeId();    RouteSet ns = makeBestEntry(nid);    if (ns != null) {      ns.put(handle);    }  }  /**   * Removes a node id from the table.   *   * @param nh DESCRIBE THE PARAMETER   * @return the handle that was removed, or null if it did not exist.   */  //  public NodeHandle remove(Id nid)  //  {  //RouteSet ns = getBestEntry(nid);  //  //if (ns == null) return null;  //  //return ns.remove(nid);  //  }  public NodeHandle remove(NodeHandle nh) {    if (logger.level <= Logger.FINER) {      logger.log("RT: remove(" + nh + ")");    }    RouteSet ns = getBestEntry(nh.getNodeId());    if (ns == null) {      return null;    }    return ns.remove(nh);  }  /**   * Is called by the Observer pattern whenever a RouteSet in this table has   * changed.   *   * @param o the RouteSet   * @param handle DESCRIBE THE PARAMETER   * @param added DESCRIBE THE PARAMETER   *///  public void update(Observable o, Object arg) {//    // pass the event to the Observers of this RoutingTable//    setChanged();//    notifyObservers(arg);//  }  public void nodeSetUpdate(Object o, NodeHandle handle, boolean added) {    // pass the event to the Observers of this RoutingTable    synchronized (listeners) {      for (int i = 0; i < listeners.size(); i++) {        ((NodeSetListener) listeners.get(i)).nodeSetUpdate(this, handle, added);      }    }    // handle deprecated interface    if (countObservers() > 0) {      setChanged();      notifyObservers(new NodeSetUpdate(handle, added));    }  }  /**   * produces a String representation of the routing table, showing the number   * of node handles in each entry   *   * @return DESCRIBE THE RETURN VALUE   */  public String toString() {    String s = "routing table: \n";    for (int i = routingTable.length - 1; i >= 0; i--) {      for (int j = 0; j < routingTable[i].length; j++) {        if (routingTable[i][j] != null) {          s += ("" + routingTable[i][j].size() + "\t");        } else {          s += ("" + 0 + "\t");        }      }      s += ("\n");    }    return s;  }  /**   * DESCRIBE THE METHOD   *   * @return DESCRIBE THE RETURN VALUE   */  public int numEntries() {    int count = 0;    int maxr = numRows();    int maxc = numColumns();    for (int r = 0; r < maxr; r++) {      for (int c = 0; c < maxc; c++) {        RouteSet rs = routingTable[r][c];        if (rs != null) {          count += rs.size();        }      }    }    return count;  }  /**   * DESCRIBE THE METHOD   *   * @return DESCRIBE THE RETURN VALUE   */  public int numUniqueEntries() {    HashSet set = new HashSet();    int maxr = numRows();    int maxc = numColumns();    for (int r = 0; r < maxr; r++) {      for (int c = 0; c < maxc; c++) {        RouteSet rs = routingTable[r][c];        if (rs != null) {          for (int i = 0; i < rs.size(); i++) {            set.add(rs.get(i));          }        }      }    }    return set.size();  }  /**   * Generates too many objects to use this interface   *   * @param o The feature to be added to the Observer attribute   * @deprecated use addNodeSetListener   */  public void addObserver(Observer o) {    if (logger.level <= Logger.WARNING) {      logger.logException("WARNING: Observer on RoutingTable is deprecated", new Exception("Stack Trace"));    }    super.addObserver(o);  }  /**   * Generates too many objects to use this interface   *   * @param o DESCRIBE THE PARAMETER   * @deprecated use deleteNodeSetListener   */  public void deleteObserver(Observer o) {    if (logger.level <= Logger.WARNING) {      logger.log("WARNING: Observer on RoutingTable is deprecated");    }    super.deleteObserver(o);  }  /**   * Adds a feature to the NodeSetListener attribute of the RoutingTable object   *   * @param listener The feature to be added to the NodeSetListener attribute   */  public void addNodeSetListener(NodeSetListener listener) {    synchronized (listeners) {      listeners.add(listener);    }  }  /**   * DESCRIBE THE METHOD   *   * @param listener DESCRIBE THE PARAMETER   */  public void removeNodeSetListener(NodeSetListener listener) {    synchronized (listeners) {      listeners.remove(listener);    }  }}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -