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

📄 outgoingconnectionmanager.java

📁 P2P协议GUNTELLA的java源代码
💻 JAVA
字号:
/* * Copyright (C) 2000-2001  Ken McCrary * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA * * Email: jkmccrary@yahoo.com */package com.kenmccrary.jtella;import java.io.IOException;import java.util.List;import java.util.LinkedList;import java.util.Collections;import com.kenmccrary.jtella.util.Log;/** *  Manages the outgoing connections, attempts to connect *  to the network agressively. Initiates more than required *  connections in an attempt to quickly achieve connections *   */class OutgoingConnectionManager extends ConnectionManager{  private HostCache hostCache;  private StarterPool starterPool;     /**    *  Constructs the outgoing connection manager    *    */   OutgoingConnectionManager(Router router,                             HostCache hostCache,                             ConnectionList connectionList,                             ConnectionData connectionData) throws IOException  {     super(router, connectionList, connectionData);     this.hostCache = hostCache;     starterPool = new StarterPool(connectionData);  }       /**    *  Main processing loop    *    */   public void run()   {     while ( !isShutdown() )       {           try       {         Thread.currentThread().sleep(1000);       }       catch(InterruptedException e)       {         // ignore         continue;         }       // clean dead connections       int liveCount = connectionList.cleanDeadConnections                                    (NodeConnection.CONNECTION_OUTGOING);       // precondition, need hosts       if ( hostCache.size() == 0 )        {         Log.getLog().logDebug("no hosts cached");         continue;       }              // count number of active connections       int activeCount = connectionList.getActiveOutgoingConnectionCount();              // attempt connections on more servants than necessary to       // accelerate achieving desired connections       if ( activeCount == connectionData.getOutgoingConnectionCount() )       {             // connection requests satisfied           Log.getLog().logInformation(activeCount + " connections achieved");           connectionList.stopOutgoingConnectionAttempts();         continue;         }           else if( activeCount > connectionData.getOutgoingConnectionCount() )       {             // connection requests satisfied           Log.getLog().logInformation("Reducing outgoing connections to: " +                                      connectionData.getOutgoingConnectionCount() );           connectionList.stopOutgoingConnectionAttempts();         connectionList.reduceActiveOutgoingConnections                                      (connectionData.getOutgoingConnectionCount());         continue;              }           else if( (liveCount - activeCount) <              ( (connectionData.getOutgoingConnectionCount() - activeCount) * 2) )       {              Host host = hostCache.nextHost();          Log.getLog().logInformation(liveCount +                                      " live connections " +                                      connectionData.getOutgoingConnectionCount() +                                      " requested");                    Log.getLog().logInformation("Connection control attempting a connection: " +                                       host.getIPAddress() +                                       ":" +                                       host.getPort());                    if ( connectionList.contains(host.getIPAddress()) )          {            // avoid duplicate connection              Log.getLog().logInformation("Aborting start on duplicate host: " +                                         host.toString());              continue;          }                    ConnectionStarter starter = starterPool.getStarter();          starter.setHost(host);        }       }   }          /**    *  Attempts to add an immediate connection, opening a slot if needed    *    *  @param ipAddress host IP address    *  @param port port number    */    void addImmediateConnection(String ipAddress, int port)    {      int activeCount = connectionList.getActiveOutgoingConnectionCount();      if ( activeCount == connectionData.getOutgoingConnectionCount() )      {             connectionList.reduceActiveOutgoingConnections                                      (connectionData.getOutgoingConnectionCount());      }            Host host = new Host(ipAddress, port, 0, 0);      ConnectionStarter starter = starterPool.getStarter();      starter.setHost(host);   }   /**    *  Asynchronously attempts to start a connection    *    */    class ConnectionStarter extends Thread   {     private Host host;     private ConnectionData connectionData;     private StarterPool starterPool;     private boolean shutdown;          ConnectionStarter(StarterPool starterPool, ConnectionData connectionData)     {       super("ConnectionStarter");       this.connectionData = connectionData;       this.starterPool = starterPool;     }              void shutdown()     {       shutdown = true;       interrupt();     }          /**      *  Set the host for this connection starter to work on      *      *  @param host host to work on      */     void setHost(Host host)     {       this.host = host;       hostCache.removeHost(host); //todo rethink this       synchronized(this)       {         notify();       }       }          /**      *  Run the starter      */     public void run()     {       while ( !shutdown )       {           while ( null == host )          {           synchronized (this)           {             try             {              wait();             }             catch (InterruptedException e)             {               break;             }           }         }                    if ( !shutdown && null != host )         {            try          {            NodeConnection connection = new NodeConnection(router,                                                             host.getIPAddress(),                                                             host.getPort(),                                                            connectionData);              connectionList.addConnection(connection);            connection.startOutgoingConnection();            host = null;            starterPool.putStarter(this);          }          catch (Exception e)          {             Log.getLog().log(e);            }         }         }       }       }          /**    *  Pool of connection starters    *    */   class StarterPool   {      private List starterList;      private ConnectionData connectionData;            /**       *  Construct the starter pool and populate it       *       */      StarterPool(ConnectionData connectionData)      {        starterList = new LinkedList();        this.connectionData = connectionData;                for (int i = 0;              i < (connectionData.getOutgoingConnectionCount() * 2);             i++)        {          ConnectionStarter starter = new ConnectionStarter(this,                                                             connectionData);          starter.start();          starterList.add(starter);        }        }            /**       *  Get a starter thread       *        *  @return starter thread       */      synchronized ConnectionStarter getStarter()      {        if ( starterList.size() == 0 )        {          ConnectionStarter starter = new ConnectionStarter(this,                                                            connectionData);          starter.start();          return starter;        }                return (ConnectionStarter)starterList.remove(0);      }            /**       *  Put a starter back into the pool       *       *  @param connectionStarter connection starter       */      synchronized void putStarter(ConnectionStarter connectionStarter)      {        if ( starterList.size() >              (connectionData.getOutgoingConnectionCount() * 2) )        {          // not needed          connectionStarter.shutdown();          return;        }                starterList.add(connectionStarter);      }   }}

⌨️ 快捷键说明

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