《用Java写一个通用的服务器程序》02 监听器
在一个服务器程序中,监听器的作用类似于公司前台,起引导作用,因此监听器花在每个新连接上的时间应该尽可能短,这样才能保证最快响应。
回到编程本身来说:
1. 监听器最好由单独的线程运行
2. 监听器在接到新的连接之后,处理连接的方法需要尽快返回
在Java Push Framework中,因为需要同时监听普通客户端和服务器监视服务的客户端,所以定义两种监听器:Acceptor和MonitorAcceptor。
由于两者的关于监听部分的逻辑是相同的,因此首先定义了抽象类Listener来实现了监视器的功能,把处理socket的部分定义为抽象方法。
// 处理socket的抽象方法
protected abstract boolean handleAcceptedSocket(
PushClientSocket clientSocket);
对于监听功能的实现比较简单,还是那三步:create,bind,accept。
private boolean doListening(InetSocketAddress serverAddr) {
boolean ret = false;
int socketBufferSize = getServerImpl().getServerOptions()
.getSocketBufferSize();
int socketType = getServerImpl().getServerOptions()
.getSocketType();
try {
// Create
serverSocket = SocketFactory.getDefault().createServerSocket(
socketType, socketBufferSize);
// Bind
serverSocket.bind(serverAddr);
Debug.debug("Start to listen " + serverAddr.getHostName() + ":"
+ serverAddr.getPort());
// Accept
doAccept();
ret = true;
} catch (IOException e) {
e.printStackTrace();
if (serverSocket != null) {
serverSocket.close();
serverSocket = null;
}
}
return ret;
}
考虑Java中现在有好几种不同的socket:同步阻塞Socket,同步非阻塞Socket,以及JDK7新添加的异步Socket,如果直接使用Java的Socket类,不方便在不同类型的socket之间切换使用。所以我自定义了PushServerSocket和PushClientSocket两个新接口:
// 对于服务器socket来说,只定义了必须的bind和accept,
// 以及一个不会抛出异常的close。
public interface PushServerSocket { public void bind(InetSocketAddress serverAddr) throws IOException; public PushClientSocket accept() throws IOException; public void close();
}
// 客户端socket接口的定义是C++的风格,因为原来的代码是C++写的,这么定义便于翻译原来的C++代码
public interface PushClientSocket { public String getIP();
public int getPort(); // 这里直接使用Selector其实是有问题的,注定了只能使用NIO的方式
// 后面会考虑修改
public SelectionKey registerSelector(Selector selector, int ops,
Object attachment) throws IOException; public int send(byte[] buffer, int offset, int size) throws IOException; public int recv(byte[] buffer, int offset, int size) throws IOException; public boolean isOpen(); public boolean isConnected(); public void close();
}
两者对应的NIO版本实现是PushServerSocketImpl和PushClientSocketImpl,代码实现比较简单,这里就不贴出来了。
回到Listener,来看doAccept:
private void doAccept()
{
// Start a new thread
acceptorThread = new Thread(new Runnable() {
public void run() {
while (blnRunning) {
try {
PushClientSocket clientSocket = serverSocket.accept(); Debug.debug("New client from " + clientSocket.getIP()); // Start servicing the client connection
if (!handleAcceptedSocket(clientSocket)) {
clientSocket.close();
}
} catch (IOException e) {
e.printStackTrace();
return;
}
}
} }); // Start the thread
acceptorThread.start();
}
这里服务器socket的accept方法实现是阻塞的,这样可以避免不停地轮询,因此在用NIO实现accept时要不能调用configureBlocking设置成非阻塞模式。
后面停止监听时直接调用服务器socket的close方法,accept方法会抛出异常从而跳出循环,结束监听线程的运行。
结束监听时不要忘记使用线程的join方法等待线程结束。
public void stopListening() {
blnRunning = false;
// Close server socket
if (serverSocket != null) {
serverSocket.close();
serverSocket = null;
}
// Wait the thread to terminate
if (acceptorThread != null) {
try {
acceptorThread.join();
} catch (InterruptedException e) {
//e.printStackTrace();
}
acceptorThread = null;
}
}
Acceptor的实现相对复杂一些,需要记录访问的信息,做一些检查,然后再交给ClientFactory处理:
protected boolean handleAcceptedSocket(PushClientSocket clientSocket) {
// 记录日志
ClientFactory clientFactoryImpl = serverImpl.getClientFactory();
ServerStats stats = serverImpl.getServerStats();
ServerOptions options = serverImpl.getServerOptions();
stats.addToCumul(ServerStats.Measures.VisitorsSYNs, 1);
// 检查是否达到最大允许访问数
if (clientFactoryImpl.getClientCount() >= options.getMaxConnections()) {
Debug.debug("Reach maximum clients allowed, deny it");
return false;
}
//检查IP是否被允许
if (!clientFactoryImpl.isAddressAllowed(clientSocket.getIP())) {
Debug.debug("IP refused: " + clientSocket.getIP());
return false;
}
// 处理socket
return clientFactoryImpl.createPhysicalConnection(clientSocket,
false, listenerOptions);
}
MonitorAcceptor的实现比较简单,直接交给ClientFactory处理就可以。
protected boolean handleAcceptedSocket(PushClientSocket clientSocket) {
return serverImpl.getClientFactory().createPhysicalConnection(
clientSocket, true, listenerOptions);
}
关于ClientFactory的处理逻辑后面的文章里细讲。
实现一个监听器功能是很容易的,所以可以说的东西不多。
《用Java写一个通用的服务器程序》02 监听器的更多相关文章
- 《用Java写一个通用的服务器程序》01 综述
最近一两年用C++写了好几个基于TCP通信类型程序,都是写一个小型的服务器,监听请求,解析自定义的协议,处理请求,返回结果.每次写新程序时都把老代码拿来,修改一下协议解析部分和业务处理部分,然后就一个 ...
- 《用Java写一个通用的服务器程序》03 处理新socket
在讲监听器时说过处理的新的socket要尽快返回,监听器调用的是ClientFactory的createPhysicalConnection方法,那么就来看这个方法: public boolean c ...
- 五:用JAVA写一个阿里云VPC Open API调用程序
用JAVA写一个阿里云VPC Open API调用程序 摘要:用JAVA拼出来Open API的URL 引言 VPC提供了丰富的API接口,让网络工程是可以通过API调用的方式管理网络资源.用程序和软 ...
- 用JAVA写一个多线程程序,写四个线程,其中二个对一个变量加1,另外二个对一个变量减1
package com.ljn.base; /** * @author lijinnan * @date:2013-9-12 上午9:55:32 */ public class IncDecThrea ...
- CBrother脚本10分钟写一个拯救“小霸王服务器”的程序
CBrother脚本语言10分钟写一个拯救“小霸王服务器”的程序 到了一家新公司,接手了一坨c++服务器代码,到处内存泄漏,这服务器没有数据库,挂了后重启一下就好了,公司就这么凑活着用了几年了,定时重 ...
- (原创)如何使用boost.asio写一个简单的通信程序(一)
boost.asio相信很多人听说过,作为一个跨平台的通信库,它的性能是很出色的,然而它却谈不上好用,里面有很多地方稍不注意就会出错,要正确的用好asio还是需要花一番精力去学习和实践的,本文将通过介 ...
- 用JAVA写一个函数,功能例如以下: 随意给定一组数, 找出随意数相加之后的结果为35(随意设定)的情况
用JAVA写一个函数.功能例如以下:随意给定一组数,比如{12,60,-8,99,15,35,17,18},找出随意数相加之后的结果为35(随意设定)的情况. 能够递归算法来解: package te ...
- 用java写一个servlet,可以将放在tomcat项目根目录下的文件进行下载
用java写一个servlet,可以将放在tomcat项目根目录下的文件进行下载,将一个完整的项目进行展示,主要有以下几个部分: 1.servlet部分 Export 2.工具类:TxtFileU ...
- 使用JAVA写一个简单的日历
JAVA写一个简单的日历import java.text.DateFormat;import java.text.ParseException;import java.text.SimpleDateF ...
随机推荐
- JavaScript享元模式
通过两个例子的对比来凸显享元模式的特点:享元模式是一个为了提高性能(空间复杂度)的设计模式,享元模式可以避免大量非常相似类的开销. 第一实例,没有使用享元模式,计算所花费的时间和空间使用程度. 要求为 ...
- win10 uwp 上传Nuget 让别人用我们的库
Nuget 我们的开发经常使用别人的dll,那么我们需要每次都从网上下载,然后复制到我们的项目, 而不知道我们的dll是否安全? 当我们的库更新的时候,我们又需要从网上搜索,这样不好,于是我们就用Nu ...
- <Mastering KVM Virtualization>:第四章 使用libvirt创建你的第一台虚拟机
在第3章<搭建独立的KVM虚拟化>中,你安装并启动了libvirtd服务.你还引入了帮助你管理虚拟机的libvirt管理工具virt-manager和virsh. 相较于命令行,新用户总是 ...
- Mapper 动态代理方式
Mapper接口开发方法只需要程序员编写Mapper接口(相当于Dao接口),由Mybatis框架根据接口定义创建接口的动态代理对象,代理对象的方法体同上边Dao接口实现类方法. Mapper接口开发 ...
- 微信退款流程,以及在过程中遇见的错误和解决方式(php 语言)
官方下载demo 1:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=11_1 开发步骤 : https://pay.weix ...
- Asp.net Api中使用OAuth2.0实现“客户端验证”
一.实现继承自OAuthAuthorizationServerProvider的类,实现以"客户端验证"方式传入的相关认证和access_token发放. public class ...
- 使用ichartjs生成图表
官网:http://www.ichartjs.com/ ichartjs 是一款基于HTML5的图形库.使用纯javascript语言, 利用HTML5的canvas标签绘制各式图形. ichartj ...
- html网页中加载js脚本 下载下来是乱码(文件编码格式)
问题描述: 在一个index.html网页中,引入了jquery脚本,但是却出现错误,提示$ is not defined <!DOCTYPE html> <html lang=&q ...
- 一款特好用的JavaScript框架——JQuery
了解了解 jQuery是一个快速,小巧,功能丰富的JavaScript库.它使诸如HTML文档遍历和操纵,事件处理,动画和Ajax等事情变得简单得多,而且易于 ...
- Servlet 学习笔记
Servlet 运行在服务器上的 java 类: Servlet 容器为 javaWeb 应用提供运行时环境,负责管理 servlet 和 jsp 生命周期,以及管理他们的共享数据. 现在我们知道了 ...