SpringBoot 自定义注解实现操作日志记录

文章目录

  • 前言
  • 正文
    • 一、项目结构介绍
    • 二、核心类
    • 2.1 核心注解
      • 2.1.1 CLog 日志注解
      • 2.1.2 ProcessorBean 处理器bean
    • 2.2 切面类
    • 2.3 自定义线程池
    • 2.4 工具类
      • 2.4.1 管理者工具类
    • 2.5 测试
      • 2.5.1 订单创建处理器
      • 2.5.2 订单管理者
      • 2.5.3 订单控制器
      • 2.5.4 测试报文
      • 2.5.5 测试结果
  • 附录
    • 1、其他相关文章

前言

关于操作日志记录,在一个项目中是必要的。
本文基于 java8 和 SpringBoot 2.7 来实现此功能。

之前写过一个简单的接口报文日志打印的,和本文的起始思路相同,都是使用切面。但是本文功能更为强大,也更复杂。文章见本文附录《SpringBoot自定义starter之接口日志输出》。

本文代码仓库:https://gitee.com/fengsoshuai/custom-log2.git

正文

本文知识点如下:
自定义注解,SpringBoot使用切面,全局异常处理器,ThreadLocal的使用,MDC传递日志ID,登录拦截器,日志拦截器,自定义线程池,SPEL表达式解析,模版方法设计模式等。

一、项目结构介绍

在这里插入图片描述
其中 org.feng.clog 是核心代码区域。org.feng.test 是用于测试功能写的。

二、核心类

在这里插入图片描述

在项目启动时,会把AbstractProcessorTemplate 的子类放入Spring容器。同时会执行注册处理器的方法,其定义如下:

package org.feng.clog;

import lombok.extern.slf4j.Slf4j;
import org.feng.clog.annotation.ProcessorBean;
import org.feng.clog.utils.SpringBeanUtils;

import javax.annotation.PostConstruct;

/**
 * 处理器模板
 *
 * @author feng
 */
@Slf4j
public abstract class AbstractProcessorTemplate<T, R> implements Processor<T, R> {

    protected void init(ProcessorContext<T> context) {
    }

    protected void after(ProcessorContext<T> context, R result) {
    }

    public R start(ProcessorContext<T> context) {
        init(context);

        // 直接调用handle会导致aop失效
        // R result = handle(context);

        AbstractProcessorTemplate<T, R> template = SpringBeanUtils.getByClass(this.getClass());
        R result = template.handle(context);

        after(context, result);

        return result;
    }


    @PostConstruct
    private void registerProcessor() {
        if (this.getClass().isAnnotationPresent(ProcessorBean.class)) {
            ProcessorBean processorBean = this.getClass().getDeclaredAnnotation(ProcessorBean.class);

            log.info("ProcessorBean Register, action is {}, processor is {}", processorBean.action(), this.getClass().getName());
            ProcessorFactory.register(processorBean.action(), this);
        }
    }
}

2.1 核心注解

2.1.1 CLog 日志注解

package org.feng.clog.annotation;

import org.feng.clog.enums.ActionTypeEnum;
import org.feng.clog.enums.ModuleEnum;

import java.lang.annotation.*;

/**
 * 日志注解</br>
 * <pre>
 * <ul>使用示例:
 * <li>@CLog(template = "这是简单模版,无参数",actionType = ActionTypeEnum.UPDATE,actionIdEl = "{#userReq.id}",moduleEl = "1")</li>
 * <li>@CLog(template = "带参数模版,学生名称:{#userReq.name},班级名称:{#userReq.classReq.name}",actionTypeStr = "这是操作",actionIdEl = "{#userReq.id}")</li>
 * <li>@CLog(template = "带参数计算模版,{#userReq.classReq.number > 20?'大班':'小班'}",actionTypeStr = "这是操作",actionIdEl = "{#userReq.id}")</li>
 * <li>@CLog(template = "复杂模版,{#userReq.classReq.number > 20?'大班':('这是名称:').concat(#userReq.name).concat(',这是年龄:').concat(#userReq.age)}",actionTypeStr = "这是操作",actionIdEl = "{#userReq.id}")</li>
 * <li>@CLog(template = "自定义表达式处理,{SfObjectUtil.isEmpty(#userReq.id)?'id为0或者为空':'id不为0或者为空'}",actionTypeStr = "这是操作",actionIdEl = "{#userReq.id}")</li>
 * <li>@CLog(template = "自定义处理,{logDesc}",actionTypeStr = "这是操作",actionIdEl = "{id}")</li>
 * </ul>
 * </pre>
 *
 * @author feng
 */
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface CLog {

    /**
     * 日志模版
     */
    String template();

    /**
     * 模块
     */
    ModuleEnum module() default ModuleEnum.DEFAULT;

    /**
     * 所属模块名
     */
    String moduleStr() default "";

    /**
     * 所属模块名</br>
     * 变量/表达式获取
     */
    String moduleEl() default "";

    /**
     * 操作类型
     */
    ActionTypeEnum actionType() default ActionTypeEnum.DEFAULT;

    /**
     * 操作类型,优先级高于枚举;不为空时强制读取此值
     */
    String actionTypeStr() default "";

    /**
     * 操作类型</br>
     * 变量/表达式获取
     */
    String actionTypeEl() default "";

    /**
     * 业务操作唯一值</br>
     * 变量/表达式获取
     */
    String actionIdEl() default "";

    /**
     * 业务操作唯一值,多值
     */
    String actionIds() default "";

    /**
     * 扩展字段
     */
    String ext() default "";
}

2.1.2 ProcessorBean 处理器bean

package org.feng.clog.annotation;

import org.feng.clog.enums.ActionTypeEnum;

import java.lang.annotation.*;

/**
 * 处理器bean
 *
 * @author feng
 */
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface ProcessorBean {

    ActionTypeEnum action();
}

2.2 切面类

package org.feng.clog.aspect;

import com.alibaba.fastjson.JSONObject;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.feng.clog.LogId;
import org.feng.clog.LogRecordContext;
import org.feng.clog.annotation.CLog;
import org.feng.clog.config.LogCustomerConfig;
import org.feng.clog.enums.ActionTypeEnum;
import org.feng.clog.enums.ModuleEnum;
import org.feng.clog.utils.SpELParserUtils;
import org.feng.clog.utils.StringUtil;
import org.feng.clog.utils.UserUtil;
import org.feng.clog.vo.UserVo;
import org.springframework.stereotype.Component;

import javax.annotation.Resource;
import java.util.*;
import java.util.concurrent.Executor;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * 日志切面
 *
 * @author feng
 */
@Aspect
@Component
@Slf4j
public class LogAspect {
    private static final Pattern BRACES_PATTERN = Pattern.compile("\\{.*?}");

    @Resource(name = "logThreadPoolTaskExecutor")
    private Executor  executor;

    @Pointcut("@annotation(org.feng.clog.annotation.CLog)")
    private void pointCut() {
    }

    @AfterReturning(value = "pointCut()")
    public void after(JoinPoint joinPoint) {
        try {
            addLog(joinPoint);
        } finally {
            LogRecordContext.clean();
        }
    }


    public void addLog(JoinPoint joinPoint) {
        String logId = LogId.get();
        UserVo userVo = UserUtil.get();
        Map<String, String> logRecordMap = LogRecordContext.get();

        executor.execute(() -> {
            try {
                // 传递logId到异步线程
                LogId.put(logId);

                // 获取方法+入参
                MethodSignature signature = (MethodSignature) joinPoint.getSignature();
                Object[] args = joinPoint.getArgs();

                // 获取注解
                CLog cLog = signature.getMethod().getDeclaredAnnotation(CLog.class);

                // 获取模版中的参数(如果存在参数),并拼接
                List<String> templateParameters = getTemplateParameters(cLog.template());
                buildTemplateData(templateParameters, signature, args, logRecordMap);
                String template = cLog.template();
                for (String templateParameter : templateParameters) {
                    template = template.replace(templateParameter, logRecordMap.get(templateParameter));
                }

                // 获取module
                String module = getModule(cLog, signature, args, logRecordMap);

                // 获取actionType
                String actionType = getActionType(cLog, signature, args, logRecordMap);

                // 获取actionId
                List<String> actionIds = getActionId(cLog, signature, args, logRecordMap);

                // 获取扩展字段
                JSONObject ext = getExt(cLog, signature, args, logRecordMap);

                if (StringUtil.isNotBlank(template)) {
                    for (String actionId : actionIds) {
                        log.info("记录日志,user={}, template={}, module={}, actionType={}, actionId={}, ext={}", userVo, template, module, actionType, actionId, ext);

                        // todo 日志落库
                    }
                } else {
                    log.info("设置日志数据失败:不满足注解条件");
                }
            } catch (Exception e) {
                log.warn("设置日志异常:", e);
            }
        });
    }


    private List<String> getTemplateParameters(String template) {
        List<String> parameters = new ArrayList<>();
        Matcher matcher = BRACES_PATTERN.matcher(template);
        while (matcher.find()) {
            parameters.add(matcher.group());
        }
        return parameters;
    }


    private void buildTemplateData(List<String> parameters, MethodSignature signature, Object[] args, Map<String, String> map) {
        for (String el : parameters) {
            // 如果EL表达式为空,则直接下一个
            if (!StringUtil.isNotBlank(el)) {
                continue;
            }

            String spEl = el;
            // 兼容自定义数据
            spEl = getEl(spEl);
            if (map.containsKey(spEl)) {
                map.put("{" + spEl + "}", map.get(spEl));
                continue;
            }

            // 自定义类处理
            spEl = parseCustomerMethodEl(spEl);

            // El执行
            if (spEl.contains("#")) {
                String value = SpELParserUtils.parse(signature.getMethod(), args, spEl, String.class);
                map.put(el, value);
            } else {
                map.put(el, "");
            }
        }
    }

    private String getModule(CLog cLog, MethodSignature signature, Object[] args, Map<String, String> map) {
        // 设置了module枚举时,优先获取枚举对应的描述
        if (!ModuleEnum.DEFAULT.equals(cLog.module())) {
            return cLog.module().getDesc();
        }

        // 设置了moduleStr时
        if (StringUtil.isNotBlank(cLog.moduleStr())) {
            return cLog.moduleStr();
        }

        // 设置了moduleEl时
        if (StringUtil.isNotBlank(cLog.moduleEl())) {
            try {
                String el = cLog.moduleEl();
                el = getEl(el);
                // 处理自定义的el
                if (map.containsKey(el)) {
                    return map.get(el);
                }

                // 处理自定义方法el
                el = parseCustomerMethodEl(el);

                // 执行el
                return SpELParserUtils.parse(signature.getMethod(), args, el, String.class);
            } catch (Exception e) {
                log.error("日志切面获取module错误", e);
            }
        }
        return null;
    }


    private String getActionType(CLog cLog, MethodSignature signature, Object[] args, Map<String, String> map) {
        // 设置了actionType枚举时,优先获取枚举对应的描述
        if (!ActionTypeEnum.DEFAULT.equals(cLog.actionType())) {
            return cLog.actionType().getDesc();
        }

        // 设置了actionTypeStr时
        if (StringUtil.isNotBlank(cLog.actionTypeStr())) {
            return cLog.actionTypeStr();
        }

        // 设置了actionTypeEl时
        if (StringUtil.isNotBlank(cLog.actionTypeEl())) {
            String el = cLog.actionTypeEl();
            el = getEl(el);
            // 处理自定义的el
            if (map.containsKey(el)) {
                return map.get(el);
            }

            // 处理自定义方法el
            el = parseCustomerMethodEl(el);

            // 执行el
            return SpELParserUtils.parse(signature.getMethod(), args, el, String.class);
        }
        return null;
    }


    private List<String> getActionId(CLog cLog, MethodSignature signature, Object[] args, Map<String, String> map) {
        // 设置了actionIdEl时
        if (StringUtil.isNotBlank(cLog.actionIdEl())) {
            if (map.containsKey(cLog.actionIdEl())) {
                return Collections.singletonList(map.get(cLog.actionIdEl()));
            }
            String el = cLog.actionIdEl();
            el = getEl(el);
            // 处理自定义el
            if (map.containsKey(el)) {
                return Collections.singletonList(map.get(el));
            }

            // 执行el
            return Collections.singletonList(SpELParserUtils.parse(signature.getMethod(), args, el, String.class));
        }

        // 设置了actionIds时
        if (StringUtil.isNotBlank(cLog.actionIds())) {
            String el = getEl(cLog.actionIds());
            if (map.containsKey(el)) {
                return Arrays.asList(map.get(el).split(","));
            }
        }
        return Collections.singletonList(System.currentTimeMillis() * 10 + new Random().nextInt(10000) + "");
    }

    private JSONObject getExt(CLog cLog, MethodSignature signature, Object[] args, Map<String, String> map) {
        // 如果EL表达式为空,则直接结束
        if (!StringUtil.isNotBlank(cLog.ext())) {
            return null;
        }
        String spEl = cLog.ext();
        //兼容自定义数据
        spEl = getEl(spEl);
        if (map.containsKey(spEl)) {
            String value = map.get(spEl);
            if (StringUtil.isNotBlank(value)) {
                try {
                    return JSONObject.parseObject(value);
                } catch (Exception e) {
                    log.info("JSON转换失败:{},{}", value, e.getMessage());
                    return null;
                }
            }
            return null;
        }

        // 自定义类处理
        spEl = parseCustomerMethodEl(spEl);

        // El执行
        if (spEl.contains("#")) {
            String value = SpELParserUtils.parse(signature.getMethod(), args, spEl, String.class);
            if (StringUtil.isNotBlank(value)) {
                try {
                    return JSONObject.parseObject(value);
                } catch (Exception e) {
                    log.info("JSON转换失败:{},{}", value, e.getMessage());
                    return null;
                }
            }
            return null;
        }
        return null;
    }


    private String parseCustomerMethodEl(String el) {
        for (String key : LogCustomerConfig.getCustomerMethod().keySet()) {
            if (el.contains(key)) {
                String className = key.split("\\.")[0];
                el = el.replace(className, "T(" + LogCustomerConfig.getCustomerMethod().get(key) + ")");
            }
        }
        return el;
    }


    private String getEl(String str) {
        str = str.replaceAll("\\{", "");
        str = str.replaceAll("}", "");
        return str;
    }

}

2.3 自定义线程池

package org.feng.clog.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import java.util.concurrent.Executor;
import java.util.concurrent.ThreadPoolExecutor;

/**
 * 线程池配置
 *
 * @author feng
 */
@Configuration
@EnableAsync
public class ThreadPoolConfig {

    @Bean(name = "logThreadPoolTaskExecutor")
    public Executor initLogCpuExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        executor.setCorePoolSize(Runtime.getRuntime().availableProcessors() + 1);
        executor.setMaxPoolSize(150);
        executor.setQueueCapacity(50);
        executor.setKeepAliveSeconds(60);
        executor.setThreadNamePrefix("log-thread-pool-");
        executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
        executor.initialize();
        executor.setTaskDecorator(runnable -> runnable);
        return executor;
    }
}

2.4 工具类

2.4.1 管理者工具类

package org.feng.clog.utils;

import org.feng.clog.AbstractProcessorTemplate;
import org.feng.clog.ProcessorContext;
import org.feng.clog.ProcessorFactory;

/**
 * 管理工具
 *
 * @author feng
 */
public class ManagerUtil {
    public static <R, T> R handle(ProcessorContext<T> context) {
        AbstractProcessorTemplate<T, R> processor = ProcessorFactory.getProcessor(context.getAction());
        if (processor == null) {
            throw new RuntimeException("未找到 " + context.getAction() + "对应的处理器");
        }
        return processor.start(context);
    }
}

2.5 测试

2.5.1 订单创建处理器

package org.feng.test;

import lombok.extern.slf4j.Slf4j;
import org.feng.clog.AbstractProcessorTemplate;
import org.feng.clog.LogRecordContext;
import org.feng.clog.ProcessorContext;
import org.feng.clog.annotation.CLog;
import org.feng.clog.annotation.ProcessorBean;
import org.feng.clog.enums.ActionTypeEnum;
import org.feng.clog.enums.ModuleEnum;
import org.feng.clog.utils.StringUtil;
import org.springframework.stereotype.Service;

/**
 * 创建订单处理器
 *
 * @author feng
 */
@Slf4j
@Service
@ProcessorBean(action = ActionTypeEnum.ORDER_CREATE)
public class OrderCreateProcessor extends AbstractProcessorTemplate<OrderCreateReq, Boolean> {

    @Override
    protected void init(ProcessorContext<OrderCreateReq> context) {
        preHandleReq(context.getData());
    }

    @Override
    @CLog(template = "测试日志记录,{testK1}", module = ModuleEnum.ORDER, actionType = ActionTypeEnum.ORDER_CREATE,
            actionIdEl = "{#context.data.orderNum}", ext = "{JacksonUtil.toJSONString(#context.data)}"
    )
    public Boolean handle(ProcessorContext<OrderCreateReq> context) {

        LogRecordContext.put("testK1", "3wewd2");

        OrderCreateReq orderCreateReq = context.getData();
        log.info("处理--创建订单{}", orderCreateReq.getOrderNum());

        return true;
    }

    @Override
    protected void after(ProcessorContext<OrderCreateReq> context, Boolean result) {
        // todo 后置操作
    }

    private void preHandleReq(OrderCreateReq req) {
        // todo 参数校验

        // 例如校验参数
        if (StringUtil.isBlank(req.getOrderNum())) {
            throw new IllegalArgumentException("订单号不能为空");
        }
    }
}

2.5.2 订单管理者

package org.feng.test;

import org.feng.clog.ProcessorContext;
import org.feng.clog.enums.ActionTypeEnum;
import org.feng.clog.utils.ManagerUtil;
import org.springframework.stereotype.Component;

/**
 * 订单管理
 *
 * @author feng
 */
@Component
public class OrderManager {

    /**
     * 创建订单
     */
    public Boolean createOrder(OrderCreateReq req) {
        ProcessorContext<OrderCreateReq> processorContext = new ProcessorContext<>();
        processorContext.setAction(ActionTypeEnum.ORDER_CREATE);
        processorContext.setData(req);
        return ManagerUtil.handle(processorContext);
    }
}

2.5.3 订单控制器

package org.feng.test;

import org.feng.clog.utils.ResultUtil;
import org.feng.clog.vo.ResultVo;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.annotation.Resource;

/**
 * 控制器
 *
 * @author feng
 */
@RestController
@RequestMapping("order")
public class OrderController {

    @Resource
    private OrderManager orderManager;

    // @WithoutLogin
    @PostMapping("/test1")
    public ResultVo<String> test1(@RequestBody OrderCreateReq req) {
        // 创建
        Boolean started = orderManager.createOrder(req);

        return ResultUtil.success("success " + started);
    }
}

2.5.4 测试报文

{
  "orderNum": "1001",
  "type": 1,
  "senderName": "",
  "likes": ["1", "2", "3"]
}

2.5.5 测试结果

控制台日志输出:

2024-02-28 11:48:40.102  INFO  92309 --- [log-thread-pool-1] org.feng.clog.aspect.LogAspect.lambda$addLog$0(LogAspect.java:95) : [logId=d3b0dc267ce64dfa8a987e8eb6aad4ba] 记录日志,user=UserVo(id=1001, username=feng123, phone=18143431243, email=null), template=测试日志记录,3wewd2, module=订单, actionType=订单创建, actionId=1001, ext={"senderName":"","orderNum":"1001","type":1,"likes":["1","2","3"]}

可以看到,日志中记录了logId,以及日志注解对应的信息。

附录

1、其他相关文章

  • SpringBoot自定义starter之接口日志输出
  • SpringBoot使用线程池之ThreadPoolTaskExecutor和ThreadPoolExecutor

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/417279.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数据结构·栈和队列

1. 栈 1.1 栈的概念及结构 栈&#xff1a;一种特殊的线性表&#xff0c;其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一段称为栈顶&#xff0c;另一端称为栈底。 栈中的数据元素遵守 后进先出 LIFO(Last In First Out)的原则&#xff0c;后进来的数…

[linux]进程信号(信号的概念,信号的产生方式,信号的相关接口、指令,函数,信号怎么保存(原理),信号怎么处理)

目录 一、信号的概念 二、信号的产生方式 通过键盘发送信号 通过系统调用&#xff0c;指令 异常 软件条件 三、信号怎么保存&#xff08;原理&#xff09; 信号其他相关常见概念 在内核中表示 sigset_t 四、信号的相关接口、指令&#xff0c;函数 signal sigpr…

数据结构--树的遍历

数据结构--树的遍历 1. 前序中序后序遍历2. 前序中序后序遍历代码 1. 前序中序后序遍历 2. 前序中序后序遍历代码 /** public class TreeNode {int val 0;TreeNode left null;TreeNode right null;public TreeNode(int val) {this.val val;}} */// 前序遍历顺序&#xff1…

SQL函数学习记录

聚合函数 函数是编程语言的基础之一&#xff0c;在对数字的运算中&#xff0c;我们用的最多的就是聚合函数&#xff0c;本篇接下来就详细阐述下SQL中聚合函数的运用。 什么是聚合函数&#xff08;aggregate function&#xff09;&#xff1f; 聚合函数指的是对一组值执行计算…

MYSQL05高级_查看修改存储引擎、InnoDB和MyISAM对比、其他存储引擎介绍

文章目录 ①. 查看、修改存储引擎②. InnoDB和MyISAM对比③. Archive引擎 - 归档④. Blackhole引擎丢数据⑤. CSV - 引擎⑥. Memory引擎 - 内存表⑦. Federated引擎 - 访问远程表⑧. Merge引擎 - 管理多个MyISAM⑨. NDB引擎 - 集群专用 ①. 查看、修改存储引擎 ①. 查看mysql提…

C++ 之LeetCode刷题记录(三十六)

&#x1f604;&#x1f60a;&#x1f606;&#x1f603;&#x1f604;&#x1f60a;&#x1f606;&#x1f603; 开始cpp刷题之旅。 目标&#xff1a;执行用时击败90%以上使用 C 的用户。 16. 最接近的三数之和 给你一个长度为 n 的整数数组 nums 和 一个目标值 target。请你…

备战蓝桥杯Day20 - 堆排序的实现

一、每日一题 蓝桥杯真题之互质数的个数 我的解法&#xff1a; 两个数互质&#xff0c;说明两个数的最大公约数是1&#xff0c;定义了一个函数判断两个数的最大公约数&#xff0c;再在循环中判断&#xff0c;并实现计数。可以实现运行&#xff0c;缺点是时间复杂度很高&#…

【Leetcode每日一题】二分查找 - 寻找旋转排序数组中的最小值(难度⭐⭐)(22)

1. 题目解析 Leetcode链接&#xff1a;153. 寻找旋转排序数组中的最小值 这个题目乍一看很长很复杂&#xff0c;又是旋转数组又是最小值的 但是仔细想想&#xff0c;结合题目给的示例&#xff0c;不难看出可以用二分的方法来解决 核心在于找到给定数组里面的最小值 2. 算法原…

预训练大模型LLM的PEFT之—— Prefix Tuning

简介 Prefix Tuning是2021.01提出来的&#xff0c;在它之前&#xff0c;我们使用prompt主要是人工设计模板或者自动化搜索模板&#xff0c;也就是prompt范式的第一阶段&#xff0c;就是在输入上加上prompt文本&#xff0c;再对输出进行映射。这种离散模板对模型的鲁棒性很差。…

如何在Window系统部署BUG管理软件并结合内网穿透实现远程管理本地BUG

文章目录 前言1. 本地安装配置BUG管理系统2. 内网穿透2.1 安装cpolar内网穿透2.2 创建隧道映射本地服务3. 测试公网远程访问4. 配置固定二级子域名4.1 保留一个二级子域名5.1 配置二级子域名6. 使用固定二级子域名远程 前言 BUG管理软件,作为软件测试工程师的必备工具之一。在…

MYSQL学习笔记:索引

MYSQL学习笔记&#xff1a;索引 文章目录 MYSQL学习笔记&#xff1a;索引索引的分类索引的创建删除索引优化B树索引B树InnoDB主键和二级索引树聚集索引与非聚集索引哈希索引INNODB的自适应哈希索引索引和慢查询 用索引也是要涉及磁盘I/O的操作的索引也是一种数据结构&#xff0…

c语言day4 运算符 表达式 三大控制结构

1&#xff1a; 2&#xff1a; 输入一个年月日 计算这是这一年的第几天 17 int year,month,day;18 printf("请输入年份 月份 日期");19 scanf("%d %d %d",&year,&month,&day);20 int feb28;21 if(year%40&&year%1…

哈工大中文mistral介绍(Chinese-Mixtral-8x7B)

Chinese-Mixtral-8x7B基于Mistral发布的模型Mixtral-8x7B进行了中文扩词表增量预训练。扩充后的词表显著提高了模型对中文的编解码效率&#xff0c;并通过大规模开源语料对扩词表模型进行增量预训练&#xff0c;使模型具备了强大的中文生成和理解能力。 开源地址见https://gith…

前端-DOM树

dom树描述网页元素关系的一个专有名词&#xff0c;如html内包含了head、body&#xff0c;而head内包含meta、title、script等&#xff0c;body内包含div等元素&#xff1b;网页所有内容都在document里面&#xff0c;网页内容以树状形式排列&#xff0c;所以称之为dom树 dom树内…

centos服务配置springboot服务开机启动

在做后端服务运维时&#xff0c;经常遇到服务器重启时&#xff0c;需要移动一堆后端服务。服务器故障自动重启时&#xff0c;通常无人通知。把springboot服务的jar包配置开机启动太有必要了&#xff0c;虽然不是很复杂&#xff0c;这里记录一下太有必要了。 创建jar包启动和停…

看完这篇,终于理解如何制作产品使用说明书啦!

制作产品说明书是一项重要的任务&#xff0c;它不仅提供了产品的详细信息&#xff0c;还可以帮助用户正确地使用和维护产品&#xff0c;确保产品说明书的质量和可使用性。在制作产品说明书时&#xff0c;掌握注意事项&#xff0c;可以帮助你更加高效地制作产品说明书。以下是Lo…

lv20 QT主窗口

熟悉创建主窗口项目 1 QAction 2 主窗口 菜单栏&#xff1a;fileMenu menuBar()->addMenu(tr("&File")); 工具栏&#xff1a;fileToolBar addToolBar(tr("File")); 浮动窗&#xff1a;QDockWidget *dockWidget new QDockWidget(tr("Dock W…

基于Python3的数据结构与算法 - 06 topk问题

一、引入 问题&#xff1a;目前共有n个数&#xff0c;设计算法得到前k大的数。&#xff08;m<n&#xff09; 解决思路&#xff1a; 排序后切片&#xff1a;O(n*lognm) O(n*logn)排序LowB三人组&#xff1a;O(mn) 例如冒泡排序&#xff0c;交换m次&#xff0c;即可取前m…

uniapp 安装安卓、IOS模拟器并调试

一、安装Android模拟器并调试 1.下载并安装Android Studio。 2.创建简单project。 3.安装模拟器。 完成安卓模拟器的安装。 4.启动模拟器。 5.hbuilderx选择模拟器、运行。 点击刷新按钮后出现模拟器&#xff0c;勾选并运行。 6.调试。 在 HBuilderX 中&#xff0c;项目启…

如何将字体添加到 ONLYOFFICE 桌面编辑器8.0

作者&#xff1a;VincentYoung 为你写好的文字挑选一款好看的字体然而自带的字体列表却找不到你喜欢的怎么办&#xff1f;这只需要自己手动安装一款字体即可。这里教你在不同的桌面操作系统里的多种字体安装方法。 ONLYOFFICE 桌面编辑器 ONLYOFFICE 桌面编辑器是一款免费的办…