Java最全Java注解-一文就懂,全网最经典26道Spring面试题总结

线程、数据库、算法、JVM、分布式、微服务、框架、Spring相关知识

一线互联网P7面试集锦+各种大厂面试集锦

学习笔记以及面试真题解析

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

methodPoint.proceed();

} else {

// 数据库服务异常捕获,防止异常中异常导致StackOverFlowError

try {

// ToDo

// 此处可以保存到数据库,做周期性失败重试,重试次数达到阈值后通过消息平台通知到运维及时处理异常任务

// 移除threadLocal,防止线程生命周期过长导致内存泄露

threadLocal.remove();

} catch (Exception e) {

e.printStackTrace();

}

}

} catch (Throwable throwable) {

logger.error(throwable.getMessage(),throwable);

// 失败后继续重试

tryAgain(point);

}

}

}

4.2 日志记录注解

定义注解

@Target(ElementType.METHOD)

@Retention(RetentionPolicy.RUNTIME)

public @interface Log {

String value() default “”;

/**

  • 是否启用

*/

boolean enable() default true;

LogActionType type() default LogActionType.SELECT;

}

日志类型

public enum LogActionType {

ADD(“新增”),

SELECT(“查询”),

UPDATE(“更新”),

DELETE(“删除”);

private String value;

LogActionType(String value) {

this.value = value;

}

public String getValue() {

return value;

}

public void setValue(String value) {

this.value = value;

}

}

定义注解处理器 - 切面

@Component

@Aspect

@Slf4j

public class LogAspect {

// 保存日志信息服务

private final LogService logService;

// 线程隔离,用于计算每个方法的执行时间

ThreadLocal currentTime = new ThreadLocal<>();

public LogAspect(LogService logService) {

this.logService = logService;

}

/**

  • 配置切入点

  • 该方法无方法体,主要为了让同类中其他方法使用此切入点

*/

@Pointcut(“@annotation(com.liziba.annotation.Log)”)

public void logPointcut() {

}

/**

  • 配置环绕通知,使用在方法logPointcut()上注册的切入点

  • @param joinPoint join point for advice

*/

@Around(“logPointcut()”)

public Object logAround(ProceedingJoinPoint joinPoint) throws Throwable {

Object result;

// 计算方法操作时间

currentTime.set(System.currentTimeMillis());

result = joinPoint.proceed();

Log log = new Log(“INFO”,System.currentTimeMillis() - currentTime.get());

currentTime.remove();

HttpServletRequest request = RequestHolder.getHttpServletRequest();

// 记录用户名、浏览器信息、ip地址、切入点、日志信息

logService.save(getUsername(), StringUtils.getBrowser(request), StringUtils.getIp(request),joinPoint, log);

return result;

}

/**

  • 配置异常通知

  • @param joinPoint join point for advice

  • @param e exception

*/

@AfterThrowing(pointcut = “logPointcut()”, throwing = “e”)

public void logAfterThrowing(JoinPoint joinPoint, Throwable e) {

Log log = new Log(“ERROR”,System.currentTimeMillis() - currentTime.get());

currentTime.remove();

// 获取日志堆栈信息,并设值

log.setExceptionDetail(ThrowableUtil.getStackTrace(e).getBytes());

HttpServletRequest request = RequestHolder.getHttpServletRequest();

// 记录用户名、浏览器信息、ip地址、切入点、日志信息

logService.save(getUsername(), StringUtils.getBrowser(request), StringUtils.getIp(request), (ProceedingJoinPoint)joinPoint, log);

}

/**

  • 获取用户名信息

*/

public String getUsername() {

try {

return SecurityUtils.getCurrentUsername();

}catch (Exception e){

return “”;

}

}

}

4.3 限流注解 - redis+lua限流

注解定义

@Target(ElementType.METHOD)

@Retention(RetentionPolicy.RUNTIME)

public @interface Limit {

// 资源名称,用于描述接口功能

String name() default “”;

// 资源 key

String key() default “”;

// key prefix

String prefix() default “”;

// 时间的,单位秒

int period();

// 限制访问次数

int count();

// 限制类型

LimitType limitType() default LimitType.CUSTOMER;

}

注解类型

public enum LimitType {

// 默认

CUSTOMER,

// IP限流

IP

}

定义注解处理器 - 切面

@Aspect

@Component

public class LimitAspect {

// redis用于执行lua脚本

private final RedisTemplate<Object,Object> redisTemplate;

private static final Logger logger = LoggerFactory.getLogger(LimitAspect.class);

public LimitAspect(RedisTemplate<Object,Object> redisTemplate) {

this.redisTemplate = redisTemplate;

}

/**

  • 配置切入点

  • 该方法无方法体,主要为了让同类中其他方法使用此切入点

*/

@Pointcut(“@annotation(com.liziba.annotation.Limit)”)

public void pointcut() {

}

@Around(“pointcut()”)

public Object around(ProceedingJoinPoint joinPoint) throws Throwable {

HttpServletRequest request = RequestHolder.getHttpServletRequest();

// 获取方法的Limit注解

MethodSignature signature = (MethodSignature) joinPoint.getSignature();

Method signatureMethod = signature.getMethod();

Limit limit = signatureMethod.getAnnotation(Limit.class);

LimitType limitType = limit.limitType();

String key = limit.key();

if (StringUtils.isEmpty(key)) {

if (limitType == LimitType.IP) {

key = StringUtils.getIp(request);

} else {

key = signatureMethod.getName();

}

}

// 通过方法或者ip 结合资源请求路径定义限流key

ImmutableList keys = ImmutableList.of(StringUtils.join(limit.prefix(), “", key, "”, request.getRequestURI().replaceAll(“/”,“_”)));

// Lua限流脚本

String luaScript = buildLuaScript();

RedisScript redisScript = new DefaultRedisScript<>(luaScript, Number.class);

最后

按照上面的过程,4个月的时间刚刚好。当然Java的体系是很庞大的,还有很多更高级的技能需要掌握,但不要着急,这些完全可以放到以后工作中边用别学。

学习编程就是一个由混沌到有序的过程,所以你在学习过程中,如果一时碰到理解不了的知识点,大可不必沮丧,更不要气馁,这都是正常的不能再正常的事情了,不过是“人同此心,心同此理”的暂时而已。

道路是曲折的,前途是光明的!”

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

个月的时间刚刚好。当然Java的体系是很庞大的,还有很多更高级的技能需要掌握,但不要着急,这些完全可以放到以后工作中边用别学。

学习编程就是一个由混沌到有序的过程,所以你在学习过程中,如果一时碰到理解不了的知识点,大可不必沮丧,更不要气馁,这都是正常的不能再正常的事情了,不过是“人同此心,心同此理”的暂时而已。

道路是曲折的,前途是光明的!”

[外链图片转存中…(img-0mnqijm0-1715314719719)]

[外链图片转存中…(img-CbuPC3UD-1715314719719)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值