总结一下面试的时候没答好的问题
1、short i = I+1 和 i+=1 问题
short i = 1; i = i+1;
说明: 上面的代码会发生编译器错误,即编译过不去,会提示Add cast to short
;
short i = 1; i += 1;
这里不会报错。
解析:
- 对于
short i =1;i = i+1;
由于1是int
类型,因此i+1运算结果也是int类型,需要强制转换类型才能赋值给short类型; - 而
short i =1;i+=1;
可以正确编译,因为i+=1
;相当于i=(short)(i+1);
其中有隐含的强制类型转换。
拓展:short i =1;i++;
也是正确的。需要注意的是,整数都默认为int类型,除非你将其定义为short类型,因此1+short类型的i还是int类型,所以会报错;而i++并没有经过i+1赋值语句,因此不会报错,而i+=1等价于i++,所以也不会报错。
2、Redis设置过期时间原子性问题
Redis
的setnx
命令是当key
不存在时设置key
,但setnx
不能同时完成expire
设置失效时长,不能保证setnx
和expire
的原子性。我们可以使用set
命令完成setnx
和expire
的操作,并且这种操作是原子操作。
set key value [EX seconds] [PX milliseconds] [NX|XX]
EX seconds:设置失效时长,单位秒
PX milliseconds:设置失效时长,单位毫秒
NX:key不存在时设置value,成功返回OK,失败返回(nil)
XX:key存在时设置value,成功返回OK,失败返回(nil)
案例:设置name=p7+,失效时长100s,不存在时设置
1.1.1.1:6379> set name p7+ ex 100 nx
OK
1.1.1.1:6379> get name
"p7+"
1.1.1.1:6379> ttl name
(integer) 94
代码实现
redisTemplate.execute(new RedisCallback<Boolean>() {
@Override
public Boolean doInRedis(RedisConnection connection) throws DataAccessException {
Object obj = connection.execute("set", serialize(key), serialize(value), SafeEncoder.encode("NX"), SafeEncoder.encode("EX"), Protocol.toByteArray(expire));
return obj != null;
}
});
3、Spring事务失效的场景
- 数据库引擎不支持事务,这里以 MySQL 为例,其 MyISAM 引擎是不支持事务操作的,InnoDB才是支持事务的引擎,一般要支持事务都会使用 InnoDB。
- 没有被 Spring 管理
- 方法不是 public 的,如果要用在非 public 方法上,可以开启 AspectJ 代理模式。
- 自身调用问题
@Service
public class OrderServiceImpl implements OrderService {
public void update(Order order) {
updateOrder(order);
}
@Transactional
public void updateOrder(Order order) {
// update order
}
}
- 不支持事务传播行为
- 异常被catch了
- 异常跟rolleback异常不一致