socket连接池简单实现

当server碰到server,socket连接池简单实现

当我们在程序中启动了一个server,这个server需要跟另一个server(比如建立在老系统上的老的c程序)之间通讯,为了增加效率需要建立一个连接池的时候不妨考虑下我的解决方案,不足处清指正。特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系

1:SocketAdapter类,此类继承了socket,重载了socket类的close方法,目的是当用户关闭socket的时候,我们并不关闭它只是放在连接池内部。

package com.tarena.socketpool;

import https://www.360docs.net/doc/7d19234277.html,.*;

import java.io.IOException;

/**

*

socket连接的简单实现

*

Description:

*

Copyright: Copyright Tarena(c) 2005

*

Company: Tarena

* @author chengxing

* @version 1.0

*/

public class ConnectionAdapter extends Socket{

/**

* 连接状态

*/

private boolean status=true;

/**

* 默认的构造函数

*/

public ConnectionAdapter() {

super();

}

public ConnectionAdapter(String host,int port)throws UnknownHostException,IOException{

super(host,port);

}

/**

* 判断此连接是否空闲

* @return boolean 空闲返回ture,否则false

*/

public boolean isFree(){

return status;

}

/**

* 当使用此连接的时候设置状态为false(忙碌)

*/ 特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系. public void setBusy(){

this.status=false;

}

/**

* 当客户端关闭连接的时候状态设置为true(空闲)

*/

public void close(){

System.out.println("Close : set the status is free " );

status=true;

}

public void destroy(){

//Close socket connection

close();

// System.out.println("Close success " );

}

}

第二个类连接管理器。

package com.tarena.socketpool; .

import https://www.360docs.net/doc/7d19234277.html,ng.reflect.*;

import java.util.Properties;

/**

*

连接管理器

*

Copyright: Copyright Tarena(c) 2005

*

Company: Tarena

* @author chengxing

* @version 1.0

*/

public class ConnectionManager {

//测试程序默认的连接池实现类

public static final String PROVIDER_CLASS="com.tarena.socketpool.MyConnectionProvider";

//测试程序的默认ip

public static final String HOST="127.0.0.1";

//测试程序的默认端口号

public static final String PORT="9880";

/**

* 注册钩子程序的静态匿名块

*/

static {

//增加钩子控制资源的释放周期

Runtime runtime = Runtime.getRuntime();

Class c = runtime.getClass();

try {

Method m = c.getMethod("addShutdownHook", new Class[] { Thread.class } );

m.invoke(runtime, new Object[] { new ShutdownThread() });

}

catch (NoSuchMethodException e) {

e.printStackTrace(); .

// Ignore -- the user might not be running JDK 1.3 or later.

}

catch (Exception e) {

e.printStackTrace();

}

}

/**

* 默认的构造函数

*/

public ConnectionManager() {

}

/**

* 得到并初始化一个连接池

* 连接池的实现类通过系统参数来传递进来,通过命令行-DConnectionProvider=YourImplClass

* 如果没有指定的实现的话,则采用系统默认的实现类

* 通过命令行传入的参数列表如下

* 对方主机名-DHost=192.168.0.200

* 对方端口号-DPort=9880

* 最小连接数 -DMax_size=10

* 最大连结数-DMin_size=20

* 以上的值可以改变,但是参数不能改变,

* 最大连结数和最小连接数可以省略,默认值分别为20和10

* @return ConnectionProvider

*/

public static ConnectionProvider getConnectionProvider()throws Exception{

String provider_class=System.getProperty("ConnectionProvider");

if(provider_class==null)provider_class=PROVIDER_CLASS; 。

String host=System.getProperty("Host");

if(host==null)host=HOST; .

String port=System.getProperty("port");

if(port==null)port=PORT; 。

String max_size=System.getProperty("Max_size");

String min_size=System.getProperty("Min_size"); 版权申明:本站文章均来自网络,本站所有转载文章言论不代表本站观点

Properties pro=new Properties();

pro.setProperty(ConnectionProvider.SERVER_IP,host);

pro.setProperty(ConnectionProvider.SERVER_PORT,port);

if(max_size!=null)pro.setProperty(ConnectionProvider.MAX_SIZE,max_size);

if(min_size!=null)pro.setProperty(ConnectionProvider.MIN_SIZE,min_size);

//通过反射得到实现类

System.out.println(provider_class);

System.out.flush();

Class provider_impl=Class.forName(provider_class);

//由于是单子模式,采用静态方法回调

Method m=provider_impl.getMethod("newInstance",new Class[]{java.util.Properties.class});

ConnectionProvider provider=null;

try{

provider = (ConnectionProvider) m.invoke(provider_impl, new Object[]{pro});

}catch(Exception e){

e.printStackTrace();

} ...

return provider;

}

/**

*

*

一个钩子的线程: 在程序结束的时候调用注销连接池

*

Description:

*

Copyright: Copyright Tarena(c) 2005

*

Company: Tarena

* @author chengxing

版权申明:本站文章均来自网络,如有侵权,请联系028-********-215 ,我们收到后立即删除,谢谢!

特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有。

* @version 1.0

*/

private static class ShutdownThread extends Thread {

public void run() {

try{

ConnectionProvider provider = ConnectionManager.getConnectionProvider();

if (provider != null) {

provider.destroy();

}

}catch(Exception e){

e.printStackTrace();

}

}

} ..

} .

第三个类,连接池的接口定义

package com.tarena.socketpool; 特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系

import https://www.360docs.net/doc/7d19234277.html,.*;

import java.util.*;

import java.io.IOException; 版权申明:本站文章均来自网络,本站所有转载文章言论不代表本站观点

/**

*

*

定义的抽象类,所有的子类必须单子模式去实现,

* 统一方法为public ConnectionProvider newInstance();

* 连接提供器的抽象接口,每一个实现它的子类最好都是JAVABEAN,

* 这样它的方法就可以是被外界控制

* @see JiveBeanInfo

*

Copyright: Copyright Tarena(c) 2005

*

Company: Tarena

* @author chengxing

* @version 1.0

*/

public interface

ConnectionProvider {

public static final String SERVER_IP = "SERVER_IP_ADDRESS";

public static final String SERVER_PORT = "SERVER_IP_PORT";

public static final String MAX_SIZE = "MAX_SIZE";

public static final String MIN_SIZE = "MIN_SIZE"; .

/**

*判断连接池内是否有连接

* @return true 有连接返回true,否则返回false

*/

public boolean isPooled(); 。

/**

* 当此方法被调用的时候提供一个 socket

* @see Socket

* @return Socket a Connection object.

*/

public Socket getConnection() throws https://www.360docs.net/doc/7d19234277.html,.SocketException; 根据专家观察,这样的理论和现象都是值得各位站长深思的,所以希望大家多做研究学习,争取总结出更多更好的经验!

/**

* 连接池初始化

*/

public void init() throws UnknownHostException, IOException; 版权申明:本站文章均来自网络,本站所有转载文章言论不代表本站观点

/**

* 连接池重新启动

*/

public void restart() throws UnknownHostException, IOException; 特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系.

/**

* 注销连接池

*/

public void destroy();

}

第四个类MyConnectionProvider,自己写的一个连接池的简单实现

package com.tarena.socketpool; .

import java.util.*;

import https://www.360docs.net/doc/7d19234277.html,.*;

import https://www.360docs.net/doc/7d19234277.html,.SocketException;

import java.io.IOException; 。

/**

*

*

这是一个连接管理器的简单实现

*

Description: implements the Interface ConnectionProvider

*

Copyright: Copyright Tarena(c) 2005

*

Company: Tarena

* @author chengxing

* @version 1.0

*/

public class MyConnectionProvider

implements ConnectionProvider { .

private Properties pro = null;

private static ConnectionProvider provider = null;

private static Object object_lock = new Object();

private String ip;

private String port; .

/**

* 默认的最大连接数

*/

private int max_size = 20; 版权申明:本站文章均来自网络,如有侵权,请联系028-********-215,我们收到后立即删除,谢谢!

* 默认的最小连接数

*/

private int min_size = 10; ..

/**

* Socket connection池数组

*/

private ConnectionAdapter[] socketpool = null; 。

/**

* 构造对象的时候初始化连接池

* @throws UnknownHostException 未知的主机异常

* @throws IOException

*/

private MyConnectionProvider(Properties pro) throws UnknownHostException, IOException {

ip = pro.getProperty(SERVER_IP);

port = pro.getProperty(SERVER_PORT);

String max_size_s = pro.getProperty(MAX_SIZE);

String min_size_s = pro.getProperty(MIN_SIZE);

if (max_size_s != null) {

max_size = Integer.parseInt(max_size_s);

}

if (min_size_s != null) {

min_size = Integer.parseInt(min_size_s);

} ..

init(); //构造对象的时候初始化连接池

} !

/**

* 判断是否已经池化

* @return boolean 如果池化返回ture,反之返回false

*/

public boolean isPooled() {

if (socketpool != null) {

return true;

}

else return false;

} 。

/**

*返回一个连接

* @return a Connection object.

public Socket getConnection() {

Socket s = null;

for (int i = 0; i < socketpool.length; i ) {

if (socketpool[i] != null) {

//如果有空闲的连接,返回一个空闲连接,如果没有,继续循环

if (socketpool[i].isFree()) {

s = socketpool[i];

return s;

}

else continue;

}

else { //如果连接为空,证明超过最小连接数,重新生成连接

try {

s = socketpool[i] = new ConnectionAdapter(ip, Integer.parseInt(port));

}

catch (Exception e) {

//never throw

}

}

}

//如果连接仍旧为空的话,则超过了最大连接数

if (s == null) {

try { //生成普通连接,由客户端自行关闭,释放资源,不再由连接池管理

s = new Socket(ip, Integer.parseInt(port));

}

catch (Exception e) { //此异常永远不会抛出

}

}

return s; 特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系.

} .

/**

* 初始化连接池

* @throws UnknownHostException 主机ip找不到

* @throws IOException 此端口号上无server监听

*/

版权申明:本站文章均来自网络,如有侵权,请联系028-********-215 ,我们收到后立即删除,谢谢!

特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有。

public void init() throws UnknownHostException, IOException { !

socketpool = new ConnectionAdapter[max_size];

for (int i = 0; i < min_size; i ) {

socketpool[i] = new ConnectionAdapter(ip, Integer.parseInt(port));

System.out.print(" . ");

}

System.out.println();

System.out.println("System init success ....");

} ..

/**

* 重新启动连接池

* @throws UnknownHostException

* @throws IOException

*/

public void restart() throws UnknownHostException, IOException {

destroy();

init();

} 本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系。

/**

* 注销此连接池

*/

public void destroy() {

for (int i = 0; i < socketpool.length; i ) {

if (socketpool[i] != null) {

ConnectionAdapter adapter = (ConnectionAdapter) socketpool[i];

adapter.destroy();

System.out.print(" . ");

}

}

System.out.println("\ndestory success ....");

}

/**

* 静态方法,生成此连接池实现的对象

* @param pro Properties 此连接池所需要的所有参数的封装

* @throws UnknownHostException 主机无法找到

* @throws IOException 与服务器无法建立连接

* @return ConnectionProvider 返回父类ConnectionProvider

*/

public static ConnectionProvider newInstance(java.util.Properties pro) throws UnknownHostException, IOException {

if (provider == null) {

synchronized (object_lock) {

if (provider == null) {

provider = new MyConnectionProvider(pro);

}

}

}

return provider;

}

/** ..

*设置系统属性通过封装系统properties对象来封装所需要的不同值

* SERVER_IP,SERVER_PORT,MAX_SIZE,MIN_SIZE等父类定义的不同的参数* @param pro Properties 传进来的系统属性

*/

public void setProperties(Properties pro) {

this.pro = pro;

}

}

相关主题
相关文档
最新文档