需求描述:
有两台服务器,一个上面8个tomcat,运维将项目源码放到一个目录下,所有tomcat都跑这个目录。现在有个定时任务要执行,那么问题来了,8个tomcat就会执行8次,但是我们只要一次就够了。
解决方案:
因为是一台机器有多个tomcat跑一套代码,所以最简单的就是拿到tomcat端口号,通过限制端口号来解决。问题是平时获得端口号的途径一般是通过request对象。但是定时方法写在service层(没写在service层就用request吧),获得reques比较麻烦,及时得到该对象,因为定时是程序自己调用,并没有request请求,request也是null。怎么办呢?这就要通过java的JMX。
JMX中有个MBeanServer,简单说就是一个注册器,一个Mbean是一个可以被管理的资源,比如tomcat容器。当他运行时,会被MBeanServer注册到JMX中,java的API中有MBeanServer的操作方法。我们可以通过方法取到MBean的信息。同样能拿到端口号。直接看代码。
先写一个工具类
@SuppressWarnings("unused")
public static Map<String,Object> getPortByMBean() {
//通过java的InetAddress获得ip
Map<String,Object> ipAndPort =new HashMap<>();
InetAddress addr = null;
String ip = "";
try {
addr = InetAddress.getLocalHost();
ip = addr.getHostAddress().toString();
ipAndPort.put("ip",ip);
} catch (UnknownHostException e1) {
e1.printStackTrace();
}
//通过mBeanServer获得端口号
MBeanServer mBeanServer = null;
ArrayList<MBeanServer> mBeanServers = MBeanServerFactory.findMBeanServer(null);
if (mBeanServers.size() > 0) {
for (MBeanServer _mBeanServer : mBeanServers) {
mBeanServer = _mBeanServer;
break;
}
}
if (mBeanServer == null) {
throw new IllegalStateException("没有发现JVM中关联的MBeanServer.");
}
Set<ObjectName> objectNames = null;
try {
objectNames = mBeanServer.queryNames(new ObjectName("Catalina:type=Connector,*"), null);
} catch (MalformedObjectNameException e) {
e.printStackTrace();
} catch (NullPointerException e) {
e.printStackTrace();
}
if (objectNames == null || objectNames.size() <= 0) {
throw new IllegalStateException("没有发现JVM中关联的MBeanServer : "
+ mBeanServer.getDefaultDomain() + " 中的对象名称.");
}
try {
for (ObjectName objectName : objectNames) {
String protocol = (String) mBeanServer.getAttribute(objectName, "protocol");
if (protocol.equals("HTTP/1.1")) {
int port = (Integer) mBeanServer.getAttribute(objectName, "port");
ipAndPort.put("port",port);
}
// String scheme = (String) mBeanServer.getAttribute(objectName,
// "scheme");
// int port = (Integer) mBeanServer.getAttribute(objectName,
// "port");
}
} catch (AttributeNotFoundException e) {
e.printStackTrace();
} catch (InstanceNotFoundException e) {
e.printStackTrace();
} catch (MBeanException e) {
e.printStackTrace();
} catch (ReflectionException e) {
e.printStackTrace();
}
System.out.println("======================="+ipAndPort);
return ipAndPort;
}
方法中通过java的InetAddress拿到本机的ip地址(因为没有request),然后通过MBeanServer获取port。在service中调用这个工具类是可以拿到ip和port的,那么后边就好办了。这样我们就可以限制一个服务器(通过ip)上的一个tomcat(通过端口)跑定时,而不是所有的都跑。
当然jmx还有更多更大的用处,各位自行研究。。。
注意:1.使用maven的tomcat的插件,拿不到objectNames,如果要本地测试请安装本地容器tomcat。
2.在阿里云服务器取到的是内网地址ip