当前位置: 首页 > news >正文

迁安社会信用体系建设官方网站/做竞价推广大概多少钱

迁安社会信用体系建设官方网站,做竞价推广大概多少钱,利州区住房和城乡建设部网站,专业企业网站建设定制目录 1.简述2.踩坑记录3.LoginController4.LoginService5.LoginLogService5.1 Async实现异步5.2 自定义线程池实现异步1)自定义线程池2)复制上下文请求3)自定义线程池实现异步 6.补充:LoginService 手动提交事务 背景:…

目录

    • 1.简述
    • 2.踩坑记录
    • 3.LoginController
    • 4.LoginService
    • 5.LoginLogService
      • 5.1 @Async实现异步
      • 5.2 自定义线程池实现异步
        • 1)自定义线程池
        • 2)复制上下文请求
        • 3)自定义线程池实现异步
    • 6.补充:LoginService 手动提交事务

背景: 模块调用之后,记录模块的相关日志,看似简单,其实暗藏玄机。

1.简述

模块日志的实现方式大致有三种:

  1. AOP + 自定义注解实现
  2. 输出指定格式日志 + 日志扫描实现
  3. 在接口中通过代码侵入的方式,在业务逻辑处理之后,调用方法记录日志。

这里我们主要讨论下第3种实现方式。

假设我们需要实现一个用户登录之后记录登录日志的操作。

调用关系如下:

在这里插入图片描述

2.踩坑记录

这里之所以不能在 LoginService.login() 方法中开启事务,是为了在日志处理中方便单独开启事务。

如果在 LoginService.login() 方法中开启了事务,日志处理的方法做异步和做新事务都会有问题:

  • 做异步:由于主事务可能没有执行完毕,导致可能读取不到主事务中新增或修改的数据信息;
  • 做新事务:可以通过 Propagation.REQUIRES_NEW 事务传播行为来创建新事务,在新事务中执行记录日志的操作,可能会导致如下问题:
    1. 由于数据库默认事务隔离级别是可重复读,意味着事物之间读取不到未提交的内容,所以也会导致读取不到主事务中新增或修改的数据信息;
    2. 如果开启的新事务和之前的事务操作了同一个表,就会导致锁表。
  • 什么都不做,直接同步调用:问题最多,可能导致如下几个问题:
    1. 不捕获异常,直接导致接口所有操作回滚;
    2. 捕获异常,部分数据库,如:PostgreSQL,同一事务中,只要有一次执行失败,就算捕获异常,剩余的数据库操作也会全部失败,抛出异常;
    3. 日志记录耗时增加接口响应时间,影响用户体验。

3.LoginController

@RestController
public class LoginController {@Autowiredprivate LoginService loginService;@RequestMapping("/login")public String login(String username, String pwd) {loginService.login(username, pwd);return "succeed";}
}

4.LoginService

@Service
public class LoginService {@Autowiredprivate LoginLogService loginLogService;/** 登录 */public void login(String username, String pwd) {// 用户登录loginUser(username, pwd);// 记录日志loginLogService.recordLog(username);}/** 用户登录 */@Transactional(rollbackFor = Exception.class)private void loginUser(String username, String pwd) {// TODO: 实现登录逻辑..}
}

5.LoginLogService

5.1 @Async实现异步

@Service
public class LoginLogService {/** 记录日志 */@Async@Transactional(rollbackFor = Exception.class)public void recordLog(String username) {// TODO: 实现记录日志逻辑...}
}

注意:@Async 需要配合 @EnableAsync 使用,@EnableAsync 添加到启动类、配置类、自定义线程池类上均可。

补充:由于 @Async 注解会动态创建一个继承类来扩展方法的实现,所以可能会导致当前类注入Bean容器失败 BeanCurrentlyInCreationException,可以使用如下方式:自定义线程池 + @Autowired

5.2 自定义线程池实现异步

1)自定义线程池

import com.demo.async.ContextCopyingDecorator;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.task.TaskExecutor;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;import java.util.concurrent.ThreadPoolExecutor;/*** <p> @Title AsyncTaskExecutorConfig* <p> @Description 异步线程池配置** @author ACGkaka* @date 2023/4/24 19:48*/
@EnableAsync
@Configuration
public class AsyncTaskExecutorConfig {/*** 核心线程数(线程池维护线程的最小数量)*/private int corePoolSize = 10;/*** 最大线程数(线程池维护线程的最大数量)*/private int maxPoolSize = 200;/*** 队列最大长度*/private int queueCapacity = 10;@Beanpublic TaskExecutor taskExecutor() {ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();executor.setCorePoolSize(corePoolSize);executor.setMaxPoolSize(maxPoolSize);executor.setQueueCapacity(queueCapacity);executor.setThreadNamePrefix("MyExecutor-");// for passing in request scope context 转换请求范围的上下文executor.setTaskDecorator(new ContextCopyingDecorator());// rejection-policy:当pool已经达到max size的时候,如何处理新任务// CALLER_RUNS:不在新线程中执行任务,而是有调用者所在的线程来执行executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());executor.setWaitForTasksToCompleteOnShutdown(true);executor.initialize();return executor;}
}

2)复制上下文请求

import org.slf4j.MDC;
import org.springframework.core.task.TaskDecorator;
import org.springframework.security.core.context.SecurityContext;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.web.context.request.RequestAttributes;
import org.springframework.web.context.request.RequestContextHolder;import java.util.Map;/*** <p> @Title ContextCopyingDecorator* <p> @Description 上下文拷贝装饰者模式** @author ACGkaka* @date 2023/4/24 20:20*/
public class ContextCopyingDecorator implements TaskDecorator {@Overridepublic Runnable decorate(Runnable runnable) {try {// 从父线程中获取上下文,然后应用到子线程中RequestAttributes requestAttributes = RequestContextHolder.currentRequestAttributes();Map<String, String> previous = MDC.getCopyOfContextMap();SecurityContext securityContext = SecurityContextHolder.getContext();return () -> {try {if (previous == null) {MDC.clear();} else {MDC.setContextMap(previous);}RequestContextHolder.setRequestAttributes(requestAttributes);SecurityContextHolder.setContext(securityContext);runnable.run();} finally {// 清除请求数据MDC.clear();RequestContextHolder.resetRequestAttributes();SecurityContextHolder.clearContext();}};} catch (IllegalStateException e) {return runnable;}}
}

3)自定义线程池实现异步

@Service
public class LoginLogService {@Qualifier("taskExecutor")@Autowiredprivate TaskExecutor taskExecutor;/** 记录日志 */@Transactional(rollbackFor = Exception.class)public void recordLog(String username) {taskExecutor.execute(() -> {// TODO: 实现记录日志逻辑...});}
}

6.补充:LoginService 手动提交事务

如果是已经开发好的项目,不好将核心逻辑单独抽离出来,可以通过手动提交事务的方式来实现,代码如下:

import org.springframework.transaction.PlatformTransactionManager;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.transaction.interceptor.TransactionAspectSupport;@Service
public class LoginService {@Autowiredprivate LoginLogService loginLogService;@Autowiredprivate PlatformTransactionManager transactionManager;/** 登录 */@Transactional(rollbackFor = Exception.class)public void login(String username, String pwd) {// 用户登录// TODO: 实现登录逻辑..// 手动提交事务TransactionStatus status = TransactionAspectSupport.currentTransactionStatus();if (status.isNewTransaction()) {transactionManager.commit(status);}// 记录日志loginLogService.recordLog(username);}
}

日志记录虽然小,坑是真的多,这里记录的只是目前遇到的问题。

大家有遇到其他坑的欢迎评论补充。

整理完毕,完结撒花~ 🌻





参考地址:

1.SpringBoot 关于异步与事务一起使用的问题,https://blog.csdn.net/qq_19922839/article/details/126322800

http://www.jmfq.cn/news/5245993.html

相关文章:

  • 手机网站信任从哪里设置/网络营销概念
  • 百度网站建设中心/测试自己适不适合做销售
  • 网站留言程序怎么做/怎么免费推广自己网站
  • 黑色炫酷灯饰照明科技企业商务网站模板/seo查询官网
  • 政府网站与门户网站的区别/seo的优化技巧和方法
  • 可以做微商的网站/网络促销方案
  • 做网站用什么web服务器吗/seo推广营销公司
  • 建设课程网站的目的/张家界百度seo
  • 阿里巴巴电子商务网站建设目的/b站视频推广app
  • 怎么选择网站开发/优化网站建设seo
  • 一般做网站用什么软件/推广普通话的重要意义
  • 教育网站模板/网上兼职外宣推广怎么做
  • 网站颜色正确搭配实例/达内教育
  • 普通网站和营销型网站的区别是什么/邀请注册推广赚钱
  • app导航网站源码/求几个微信推广平台
  • 电脑iis做网站/新闻平台发布
  • 9元建站节/成都网络推广哪家好
  • 专业集团门户网站建设公司/促销活动推广语言
  • 网站建设经典语录/百度开车关键词
  • 如何确定网站被k/上海seo关键词优化
  • 灯饰模板网站/东莞网站制作公司
  • 重庆建一科技发展有限公司/西安网站seo推广
  • 企业营销网站建设费用预算/网站建设方案及报价
  • 网站的开发流程分哪几步/百度推广服务
  • CMS源码就可以做网站吗/电脑培训学校哪家好
  • 网站设计遇到的问题/石家庄seo公司
  • 移动端网站和app区别/英文seo
  • 长春优化/seo流量增长策略
  • 冒用公司名义做网站/外贸建站推广公司
  • 阳江网站建设公司/平台网站开发公司