【JAVA】分布式定时任务
问题描述
正式环境用负载均衡部署了两台服务器部四个Tomcat节点,同一套代码,导致了定时任务同时启用了四个,然后数据库死锁了。。。
解决方案一 分布式锁
不会
解决方案二 限定IP
在定时任务里加代码,判断服务器的ip和端口号,仅一个 ip+port 才能执行定时任务,其他都return。
大概的方法就是获取配置项中的ip和端口号,和本机一致的才能继续往下执行定时任务;额外配置一个开关,决定是否启用这个方案。
获取本机ip和port方法
package com.comtop.koron.common.utils;
import javax.management.MBeanServer;
import javax.management.MalformedObjectNameException;
import javax.management.ObjectName;
import javax.management.Query;
import javax.naming.Context;
import javax.naming.InitialContext;
import java.lang.management.ManagementFactory;
import java.net.InetAddress;
import java.util.Set;
public class IPPortUtil {
/**
* @return port
* @throws MalformedObjectNameException 获取当前tomcat机器的端口号
*/
public static String getLocalPort() throws MalformedObjectNameException {
MBeanServer beanServer = ManagementFactory.getPlatformMBeanServer();
Set<ObjectName> objectNames = beanServer.queryNames(new ObjectName("*:type=Connector,*"),
Query.match(Query.attr("protocol"), Query.value("HTTP/1.1")));
String port = objectNames.iterator().next().getKeyProperty("port");
return port;
}
/**
* 获取weblogic 端口号
*
* @return port
*/
public static String getWeblogicPort() {
try {
Context ctx = new InitialContext();
MBeanServer tMBeanServer = (MBeanServer) ctx.lookup("java:comp/env/jmx/runtime");
ObjectName tObjectName = new ObjectName("com.bea:Name=RuntimeService,Type=weblogic.management.mbeanservers.runtime.RuntimeServiceMBean");
ObjectName serverrt = (ObjectName) tMBeanServer.getAttribute(tObjectName, "ServerRuntime");
String port = String.valueOf(tMBeanServer.getAttribute(serverrt, "ListenPort"));
String listenAddr = (String) tMBeanServer.getAttribute(serverrt, "ListenAddress");
String[] tempAddr = listenAddr.split("/");
if (tempAddr.length == 1) {
listenAddr = tempAddr[0];
} else if (tempAddr[tempAddr.length - 1].trim().length() != 0) {
listenAddr = tempAddr[tempAddr.length - 1];
} else if (tempAddr.length > 2) {
listenAddr = tempAddr[tempAddr.length - 2];
}
return port;
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
/**
* @return 获取当前机器的IP
*/
public static String getLocalIP() {
InetAddress addr = null;
try {
addr = InetAddress.getLocalHost();
} catch (Exception e) {
e.printStackTrace();
}
byte[] ipAddr = addr.getAddress();
String ipAddrStr = "";
for (int i = 0; i < ipAddr.length; i++) {
if (i > 0) {
ipAddrStr += ".";
}
ipAddrStr += ipAddr[i] & 0xFF;
}
return ipAddrStr;
}
}
代码来源: 感谢大佬,侵权私信.