【面试题】SpringCloud架构中如何保证定时任务只在一个服务在执行

  • @author qiang220316

  • @date 2019/2/28

*/

@Component

public class WrokTask {

@Autowired

private IJobService jobService;

private static String serviceName=“provider”;

/**

  • 5秒更新一次

*/

@Scheduled(fixedDelay = 5000)

public void doWork(){

if (!IPV4Util.ipCompare(this.jobService.serviceUrl(serviceName))) {

return;

}

System.out.println(serviceName+“服务,地址为:”+IPV4Util.getIpAddress()+“,正在执行task任务”);

}

}

定时任务中我们可以看到this.jobService.serviceUrl方法,这个方法的作用则是获取SpringCloud集群中服务信息,IPV4Util.ipCompare这个作用就是将当前服务IP和集群所有IP进行对比,如果当前服务IP是集群服务IP最小则返回true,反之返回false。

接下来我们再看一下,如果来获取SpringCloud集群信息:

@Service

public class JobServiceImpl implements IJobService {

@Autowired

private DiscoveryClient discoveryClient;

@Override

public List serviceUrl(String serviceName) {

List serviceInstanceList = discoveryClient.getInstances(serviceName);

List urlList = new ArrayList();

if (CollectionUtils.isNotEmpty(serviceInstanceList)) {

serviceInstanceList.forEach(si -> {

urlList.add(si.getUri());

});

}

return urlList;

}

}

其实主要还是用到DiscoveryClient类中方法,我们就可以很轻松获取到集群信息。

最后我们再来看看IPV4Util这个工具类到底是怎么进行对比的呢?

public class IPV4Util {

/**

  • @param ipAddress

  • @return

*/

public static long ipToLong(String ipAddress) {

long result = 0;

String[] ipAddressInArray = ipAddress.split(“\.”);

for (int i = 3; i >= 0; i–) {

long ip = Long.parseLong(ipAddressInArray[3 - i]);

// left shifting 24,16,8,0 and bitwise OR

// 1. 192 << 24

// 1. 168 << 16

// 1. 1 << 8

// 1. 2 << 0

result |= ip << (i * 8);

}

return result;

}

/**

  • @param ip

  • @return

*/

public static String longToIp(long ip) {

StringBuilder result = new StringBuilder(15);

for (int i = 0; i < 4; i++) {

result.insert(0, Long.toString(ip & 0xff));

if (i < 3) {

result.insert(0, ‘.’);

}

ip = ip >> 8;

}

return result.toString();

}

/**

  • @param ip

  • @return

*/

public static String longToIp2(long ip) {

return ((ip >> 24) & 0xFF) + “.” + ((ip >> 16) & 0xFF) + “.” + ((ip >> 8) & 0xFF) + “.” + (ip & 0xFF);

}

/**

  • 获取当前机器的IP

  • @return

*/

public static String getIpAddress() {

try {

for (Enumeration enumNic = NetworkInterface.getNetworkInterfaces();

enumNic.hasMoreElements(); ) {

NetworkInterface ifc = enumNic.nextElement();

if (ifc.isUp()) {

for (Enumeration enumAddr = ifc.getInetAddresses();

enumAddr.hasMoreElements(); ) {

InetAddress address = enumAddr.nextElement();

if (address instanceof Inet4Address && !address.isLoopbackAddress()) {

return address.getHostAddress();

}

}

}

}

return InetAddress.getLocalHost().getHostAddress();

} catch (IOException e) {

//log.warn(“Unable to find non-loopback address”, e);

e.printStackTrace();

}

return null;

}

/**

  • 对比方法

  • @param serviceUrl

  • @return

*/

public static boolean ipCompare(List serviceUrl) {

try {

String localIpStr = IPV4Util.getIpAddress();

long localIpLong = IPV4Util.ipToLong(localIpStr);

int size = serviceUrl.size();

if (size == 0) {

return false;

}

最后

无论是哪家公司,都很重视基础,大厂更加重视技术的深度和广度,面试是一个双向选择的过程,不要抱着畏惧的心态去面试,不利于自己的发挥。同时看中的应该不止薪资,还要看你是不是真的喜欢这家公司,是不是能真的得到锻炼。

针对以上面试技术点,我在这里也做一些分享,希望能更好的帮助到大家。

lse;

}

最后

无论是哪家公司,都很重视基础,大厂更加重视技术的深度和广度,面试是一个双向选择的过程,不要抱着畏惧的心态去面试,不利于自己的发挥。同时看中的应该不止薪资,还要看你是不是真的喜欢这家公司,是不是能真的得到锻炼。

针对以上面试技术点,我在这里也做一些分享,希望能更好的帮助到大家。

[外链图片转存中…(img-W3Zdk302-1721165979742)]

[外链图片转存中…(img-mf0091dl-1721165979743)]

[外链图片转存中…(img-EXLoXTIK-1721165979743)]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值