Java Web技术经验总结(十二)
2016-09-21 本文已影响1532人
程序熊大
- 今天解决多线程问题的时候,用到了优先级队列模型,涉及到“按照优先级”从高到低遍历,因此查到这个问题:如何利用Java原因逆序便利一个TreeMap?参考How to iterate TreeMap in reverse order in Java
import java.util.*;
class TreeMapDemo {
public static void main(String args[]) {
Map<String, String> treemap =
new TreeMap<String, String>(Collections.reverseOrder());
// Put elements to the map
treemap.put("Key1", "Jack");
treemap.put("Key2", "Rick");
treemap.put("Key3", "Kate");
treemap.put("Key4", "Tom");
treemap.put("Key5", "Steve");
Set set = treemap.entrySet();
Iterator i = set.iterator(); // Display elements
while(i.hasNext()) {
Map.Entry me = (Map.Entry)i.next();
System.out.print(me.getKey() + ": ");
System.out.println(me.getValue());
}
}
}
- 在Java中,使用ReetrantLock作为互斥锁,使用Condition作为通知机制,支持多路通知。参考一篇文章:Java多线程11:ReentrantLock的使用和Condition
- 预发/线上的远程debug利器greys:在Java开发中,如果服务部署在日常环境,我们可以通过IDEA 的远程DEBUG功能进行单步跟踪;但是如果线上服务出现问题,需要快速在预发环境复现问题的时候,并不能单步跟踪,这时候可以使用greys。利用greys可以实现类似BTrace的功能,打印出某个接口的调用链、可以打印出某个方法的调用情况和入参。
- Collections.emptyMap()的使用场景?近期在CodeReview的时候发现有同学试图给Collections.emptyMap()的返回值put值,从而引发错误。其实查看源码就可以知道,该接口返回的是一个不可变的EmptyMap对象。在《Effective Java》一书中指出,这个接口的作用是:“在能返回数组的情况下,不要返回null”,即为了避免返回null值,从而使得上层调用不需要检查返回值是否为null。参考stackoverflow上的问答:Collections.emptyMap() vs new HashMap()
- 近期遇到一个异常:“Caused by: java.lang.IllegalStateException: Duplicate spring bean id realnameAuthPubService ” 之前阅读Spring源码的时候知道,在Spring 中支持xml文件的重复导入,参考DefaultListableBeanFactory类的这个方法。可以看出,根据allowBeanDefinitionOverriding这个标志位的设置,Spring 也可以配置成不允许重复扫描xml。
public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition) throws BeanDefinitionStoreException {
Assert.hasText(beanName, "Bean name must not be empty");
Assert.notNull(beanDefinition, "BeanDefinition must not be null");
if(beanDefinition instanceof AbstractBeanDefinition) {
try {
((AbstractBeanDefinition)beanDefinition).validate();
} catch (BeanDefinitionValidationException var7) {
throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(), beanName, "Validation of bean definition failed", var7);
}
}
Map var4 = this.beanDefinitionMap;
BeanDefinition oldBeanDefinition;
synchronized(this.beanDefinitionMap) {
oldBeanDefinition = (BeanDefinition)this.beanDefinitionMap.get(beanName);
if(oldBeanDefinition != null) {
if(!this.allowBeanDefinitionOverriding) {
throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(), beanName, "Cannot register bean definition [" + beanDefinition + "] for bean \'" + beanName + "\': There is already [" + oldBeanDefinition + "] bound.");
}
if(this.logger.isInfoEnabled()) {
this.logger.info("Overriding bean definition for bean \'" + beanName + "\': replacing [" + oldBeanDefinition + "] with [" + beanDefinition + "]");
}
} else {
this.beanDefinitionNames.add(beanName); this.frozenBeanDefinitionNames = null;
}
this.beanDefinitionMap.put(beanName, beanDefinition);
}
if(oldBeanDefinition != null || this.containsSingleton(beanName)) {
this.resetBeanDefinition(beanName);
}
}
而我们遇到的问题是Dubbo框架在启动时引发的异常,查看Dubbo的源码可以发现,Dubbo 3.0之后改了策略,不允许bean的重复定义。这个问题是参考Duplicate spring bean id问题排查这篇文章解决的。
- 使用Spring AOP进行日志记录时,对于某个类中的方法,如果这个方法不是实现其对应的接口的方法,则不能使用基于接口的代理。即必须使用基于类的代理模式,也就是说,对应的Spring AOP的配置文件应该为:
<aop:aspectj-autoproxy proxy-target-class="true"/>
<context:component-scan base-package="com.*.*.*.example" />
- Java中的接口回调:深入浅出Java回调机制 和 回调函数。今天遇到一个问题,应用A和应用B共用了同一套底层代码L,业务需要在执行L中的某个函数的时候给调用B中的一个方法,对于B来说,只需要本地函数调用即可;对于A来说,则需要通过RPC调用实现,但是L只有一种实现,因此不能写死。这时候可以利用Java的回调机制,将具体的调用逻辑作为一个回调接口传入到L中。这样A和B可以自由选择自己对L的调用策略。
- 在使用git过程中,有时候需要排除某个已经被跟踪的文件,这时候可以使用命令:
git rm --cached logs/xx.log
。参考:git忽略已经跟踪的文件 - redis的使用技巧,DEL命令不支持通配符,但是我们可以结合Linux的管道和xargs命令自己实现删除所有符合规则的键。比如要删除所有以"bar"开头的键,就可以执行
redis-cli KEYS "bar*" | xargs DEL
命令。另外,由于DEL命令支持多个键作为参数,所以还可以用命令redis-cli DEL
redis-cli KEYS "bar*"``,后者效率更高。