设为首页 收藏本站
查看: 583|回复: 0

[经验分享] Tomcat 源码学习 之 StandardServer

[复制链接]

尚未签到

发表于 2017-1-25 10:18:17 | 显示全部楼层 |阅读模式
  
类名StandardServer
继承关系  LifecycleMBeanBase
  Server

关联类  NaingResources
  PropertyChangeSupport
  Service

实现功能管理Service及全局的resources

 
 
分析
 
在catalina类中管理及维护的Server实例,实际上就是StandardServer的实例。这个类继承了Server接口,并实现了其中的管理并维护Servcie及全局的resource的方法。



  而StandardServer所继承的另一个抽象类则是LifecycleMBeanBase,这个类的结构相对比较复杂,主要目的就为了同时继承Lifecycle和MBeanRegistration的默认实现,这样就不需要再重复许多代码。而因为要提供两个不同的接口的默认实现,同时又不会产生重复的代码,LifecycleMBeanBase先继承LifecycleBase类(lifecyle的默认实现),同时又继承MBeanRegistration接口,并在实现了这个接口的方法。
  Lifecycle
  Tomcat中所有可以被“Container”包含的组件(包括Container本身),都实现了这个接口。它提供了修改组件状态的方法,并使用观察者模式使得监控组件状态变化及在生命周期的不同阶段进行操作的成为可能。
  修改组件状态得方法包括:
  init()
  start()
  stop()
  destroy()
  组件的状态包括:
  BEFORE_INIT
  AFTER_INIT
  BEFORE_START
  START
  AFTER_START
  BEFORE_STOP
  STOP
  AFTER_STOP
  BEFORE_DESTROY
  AFTER_DESTROY
  PERIODIC(周期性的事件)
  CONFIG_START
  CONFIG_STOP
  所有的这些状态发生时,都会触发相应的事件,想要监听到这些事件并执行相应的操作,就需要用以下的方法注册成为其监听者:
  addLifecycleListener(LifecycleListener listener)
  findLifecycelListener()
  removeLifecycleLisnter(LifecycleListener listener)
  同时,也可以随时调用以下方法获得当前状态:
  getState()
  getStateName()
  LifecycleListener
  想要注册监听某个类的状态变化,就需要继承接口LifecycleListener,这个接口只声明了一个方法:
  lifecyleEvent(LifecycleEvent event)
  你需要在具体实现中决定需要监听哪些事件,对这些事件相应的做出什么样的操作。
  LifecycleEvent
  用来实现某个事件的final类,它的构造函数接收3个参数:
  public LifecycleEvent(Lifecycle lifecycle, String type, Object Data)
  参数分别表示产生当前event的Lifecycle类,类型以及相应的数据。许多时候我们只需要设置第一个和第二个参数即可。
  LifecycleBase & LifecycleSupport
  LifecycleBase类提供了Lifecycle接口的默认实现,对于事件处理的具体实现是在LifecycleSupport类中,LifecycleBase的方法仅是简单的代理到LifecycleSupport的相应方法。
  LifecycleSupport中的维护了一个LifecycleListener的数组,同时利用一个同步锁来确保每次只有一个线程在操作。

    private LifecycleListener listeners[] = new LifecycleListener[0];//维护的LifecycleListener数组。
private final Object listenersLock = new Object();//同步锁
public void addLifecycleListener(LifecycleListener listener) {
synchronized (listenersLock) {//首先同步锁,保证单线程访问
LifecycleListener results[] =
new LifecycleListener[listeners.length + 1];//创建一个新的,长度加一的数组。
for (int i = 0; i < listeners.length; i++)
results = listeners;
results[listeners.length] = listener;//将新的listener加入。
listeners = results;
}
//我想这样实现的目的是为了尽量减少Collection的使用,提高性能。Just a guess。
}
  其它的方法实现也大同小异。
  LifecycleBase中还实现了其它修改状态的的方法,具体步骤如下
  1. 修改状态前确定该修改是否合法。
  2. 修改状态标记,发起状态变化事件。
  3. 利用模板方法给子类提供一个抽象方法来实现具体的修改状态的实现。
  4. 修改状态标记,发起状态变化事件。

    public final synchronized void init() throws LifecycleException {
if (!state.equals(LifecycleState.NEW)) {
invalidTransition(Lifecycle.BEFORE_INIT_EVENT);
}//判断状态修改是否合法
setStateInternal(LifecycleState.INITIALIZING, null, false); //状态标记修改,启动事件
try {
initInternal();//模板方法,将会由子类来实现。
} catch (LifecycleException e) {
setStateInternal(LifecycleState.FAILED, null, false);
throw e;
}
setStateInternal(LifecycleState.INITIALIZED, null, false); //状态标记修改,启动事件
}

protected abstract void initInternal() throws LifecycleException;//模板方法

  MBeanRegistration
  这个接口是JDK中的JMX组件提供的一个接口,实现这个接口的MBean可以在JMX注册和反注册的之前之后提供一些相应的操作。
  代码
  作为container,StandardServer维护着两种资源:Service和Global Resources。因此,StandardServer中也包含了管理维护这两者的代码,这里拿addservice方法作为一个例子:

    public void addService(Service service) {
service.setServer(this);
synchronized (services) {
Service results[] = new Service[services.length + 1];
System.arraycopy(services, 0, results, 0, services.length);
results[services.length] = service;
services = results;
if (getState().isAvailable()) {
try {
service.start();
} catch (LifecycleException e) {
// Ignore
}
}
// Report this property change to interested listeners
support.firePropertyChange("service", null, service);
}
}
   
  await方法
  await方法包含了启动一个SocketServer并等待接收由其它Socket信息的过程。当接收到的request是一个SHUTDOWN时,整个循环就会结束,Tomcat也随之关闭。 同时,也可以通过调用该类的StopAwait方法,来达到同样的效果。我想这个应该是Tomcat内部的关闭机制。但尚未找到证据。此处存疑。
  同时对于其它的非关闭request过来之后如何做处理的,这里好像也没有做处理,难道这里的SocketServer并非真正的HTTP Request监听者?尚未得知,需要进一步的学习和了解。

    public void await() {
// Negative values - don't wait on port - tomcat is embedded or we just don't like ports
if( port == -2 ) {
// undocumented yet - for embedding apps that are around, alive.
return;
}
if( port==-1 ) {
try {
awaitThread = Thread.currentThread();
while(!stopAwait) {
try {
Thread.sleep( 10000 );
} catch( InterruptedException ex ) {
// continue and check the flag
}
}
} finally {
awaitThread = null;
}
return;
}
// Set up a server socket to wait on
try {
awaitSocket = new ServerSocket(port, 1,
InetAddress.getByName(address));
} catch (IOException e) {
log.error("StandardServer.await: create[" + address
+ ":" + port
+ "]: ", e);
return;
}
try {
awaitThread = Thread.currentThread();
// Loop waiting for a connection and a valid command
while (!stopAwait) {
ServerSocket serverSocket = awaitSocket;
if (serverSocket == null) {
break;
}
// Wait for the next connection
Socket socket = null;
StringBuilder command = new StringBuilder();
try {
InputStream stream;
try {
socket = serverSocket.accept();
socket.setSoTimeout(10 * 1000);  // Ten seconds
stream = socket.getInputStream();
} catch (AccessControlException ace) {
log.warn("StandardServer.accept security exception: "
+ ace.getMessage(), ace);
continue;
} catch (IOException e) {
if (stopAwait) {
// Wait was aborted with socket.close()
break;
}
log.error("StandardServer.await: accept: ", e);
break;
}
// Read a set of characters from the socket
int expected = 1024; // Cut off to avoid DoS attack
while (expected < shutdown.length()) {
if (random == null)
random = new Random();
expected += (random.nextInt() % 1024);
}
while (expected > 0) {
int ch = -1;
try {
ch = stream.read();
} catch (IOException e) {
log.warn("StandardServer.await: read: ", e);
ch = -1;
}
if (ch < 32)  // Control character or EOF terminates loop
break;
command.append((char) ch);
expected--;
}
} finally {
// Close the socket now that we are done with it
try {
if (socket != null) {
socket.close();
}
} catch (IOException e) {
// Ignore
}
}
// Match against our command string
boolean match = command.toString().equals(shutdown);
if (match) {
log.info(sm.getString("standardServer.shutdownViaPort"));
break;
} else
log.warn("StandardServer.await: Invalid command '"
+ command.toString() + "' received");
}
} finally {
ServerSocket serverSocket = awaitSocket;
awaitThread = null;
awaitSocket = null;
// Close the server socket and return
if (serverSocket != null) {
try {
serverSocket.close();
} catch (IOException e) {
// Ignore
}
}
}
}
 

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-333196-1-1.html 上篇帖子: Tomcat 表单乱码解决办法 下篇帖子: tomcat 站内session共享
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表