SpringBootWeb 篇-深入了解 AOP 面向切面编程与 AOP 记录操作日志案例

🔥博客主页: 【小扳_-CSDN博客】
❤感谢大家点赞👍收藏⭐评论✍

文章目录

        1.0 AOP 概述

        1.1 构造简单 AOP 类

        2.0 AOP 核心概念

        2.1 AOP 执行流程

        3.0 AOP 通知类型

        4.0 AOP 通知顺序

        4.1 默认按照切面类的类名字母排序

        4.2 用 @Order(数字) 注解加在切面类上来控制顺序

        5.0 AOP 切入点表达式

        5.1 使用 execution() 创建切入点表达式

        5.2 使用 @annotation 创建切入点表达式

        6.0 AOP 连接点

        7.0 AOP 案例 - 记录操作日志


        1.0 AOP 概述

        AOP,Aspect Oriented Programming 面向切面编程,在 AOP 中,横切关注点被称为切面(Aspect),切面通过特定的注入方式被应用到程序的不同部分,从而实现对这些部分的增强或修改。AOP 能够帮助开发者更好地管理程序的复杂性,提高代码的重用性和易读性。

        简单来说,就是面向特定的方法编程,也或者说给原始的方法进行升级改造。这样原始的方法就不需要进行改变,从而实现方法升级了。如日志记录、权限控制等功能。通过AOP,可以实现方法的升级改造,提高代码的可维护性和可重用性。

        1.1 构造简单 AOP 类

        1)首先导入 AOP 依赖:

        在 pom.xml 中导入 AOP 的依赖。

    <!--AOP依赖-->
    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-aop</artifactId>
    </dependency>

        2)编写 AOP 程序:

        针对特定的方法业务需要进行编程。

        首先创建类,在类上加上 @Component 注解进行控制反转,成为 IOC 容器中的 Bean 对象。继续在类上加上 @Aspect 注解,代表当前类不是普通类而是 AOP 类。在方法上加上通知类型,根据切入点表达式来筛选出连接点。

代码演示:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class demo1 {
    @Around("execution(* org.example.controller.DeptController.getList())")
    public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {
        System.out.println("正在执行目标代码之前的代码");
        Object result = joinPoint.proceed();
        System.out.println("正在执行目标代码之后的代码");
        return result;
    }
}

        2.0 AOP 核心概念

        连接点:JoinPoint,可以被 AOP 控制的方法(暗含方法执行时的相关信息)。

        通知:Advice,指哪些重复的逻辑,也就是共性功能(最终体现为一个方法)。

        切入点:PointCut,匹配连接点的条件,通知仅会在切入点方法执行时被应用。

        切面:Aspect,描述通知与切入点的对应关心(通知+切入点)。

        目标对象:Target,通知所应用的对象。

连接点与切入点的区别:

        简而言之,连接点是具体的程序执行事件,而切入点是一种筛选连接点的机制,可以帮助我们选择在哪些连接点应用切面逻辑。对于切入点来说,是通过切入点表达式来描述切入点。

        2.1 AOP 执行流程

        1)首先定义切面:

        开发人员定义一个切面,包含通知和切入点的定义。通知定义了切面逻辑,包括前置通知、后置通知、环绕通知等。切入点定义了在哪些连接点上应用切面逻辑。

        2)创建目标对象和代理对象:

        确定目标对象,即需要进行增强的对象。AOP 框架会创建一个代理对象来包含目标对象和切面。应用程序中会通过代理对象来调用目标对象的方法。

        3)选择连接点:

        在应用程序执行过程中,AOP框架根据切入点的定义选择适当的连接点。连接点是指程序执行过程中可以被增强的具体事件。

        4)执行切面逻辑:

        对于选择的连接点,AOP 框架会在该连接点上执行相应的增强逻辑,即通知。根据通知的类型,在连接点执行前、执行后或执行前后都可能执行切面逻辑。

        5)织入切面:

        织入是将切面与应用程序的目标对象结合起来创建代理对象的过程。AOP 框架会动态地将切面织入到目标对象的方法调用中,从而实现横切关注点的功能。织入可以发生在编译时、加载时、运行时或动态切入时。

        6)执行增强后的程序:

        当应用程序使用代理对象调用目标对象的方法时,会触发代理对象的增强逻辑。代理对象会在适当的连接点上执行切面逻辑,从而实现对应用程序的增强功能。

        简单来说,当程序运行时,执行到了与切入点匹配适合的连接点,也就是匹配到对应的方法时,那么就会由代理对象替代原始的方法,代理对象的方法包含了切面方法和原始的方法,也就是包含了通知与原始方法的代码,到最后,当程序执行原始方法的方法名的时候,不会继续往下执行原始方法里面的内容了,会执行代理对象中的方法。

        3.0 AOP 通知类型

        1)@Around:环绕通知,此注解标注的通知方法在目标方法前、后都被执行。

        2)@Before:前置通知,此注解标注的通知方法在目标方法前被执行。

        3)@After:后置通知,此注解标注的通知方法在目标方法后被执行,无论是否有异常都会执行。

        4)@AfterReturning:返回后通知,此注解标注的通知的方法在目标方法后被执行,有异常不会被执行,也就是说,当目标方法出现异常时,那么该通知方法就不会执行。

        5)@AfterThrowing:异常后通知,此注解标注的通知方法发生异常后执行。

需要注意的是:

        @Around 环绕通知需要自己调用 ProceedingJoinPoint.proceed() 来让原始方法执行,其他通知不需要考虑目标方法执行。@Around 环绕通知方法的返回值,必须指定为 Object ,来接收原始方法的返回值。

        4.0 AOP 通知顺序

        如果有多个通知类型都绑定在同一个连接点上,其执行顺序可能会有所不同。因此,在配置 AOP 时,需要谨慎考虑通知的顺序以保证业务逻辑的正确执行。

        也就是说,当连接点匹配到多个通知类型时,是按照什么顺序执行的呢?

        1)默认按照切面类的类名字母排序

        2)用 @Order(数字) 注解加在切面类上来控制顺序

        4.1 默认按照切面类的类名字母排序

        目标方法前的通知方法:字母排名靠前的先执行。

        目标方法后的通知方法:字母排名靠前的后执行。

代码演示:

demo1 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class demo1 {

    @Around("execution(* org.example.controller.DeptController.getList())")
    public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {

        System.out.println("正在执行 demo1 切面类");

        Object result = joinPoint.proceed();

        System.out.println("正在执行 demo1 切面类");

        return result;

    }
}

demo2 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class demo2 {

    @Around("execution(* org.example.controller.DeptController.getList())")
    public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {

        System.out.println("正在执行 demo2 切面类");

        Object result = joinPoint.proceed();

        System.out.println("正在执行 demo2 切面类");

        return result;

    }

}

demo3 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class demo3 {

    @Around("execution(* org.example.controller.DeptController.getList())")
    public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {

        System.out.println("正在执行 demo3 切面类");

        Object result = joinPoint.proceed();

        System.out.println("正在执行 demo3 切面类");

        return  result;
    }
}

运行结果:

        4.2 用 @Order(数字) 注解加在切面类上来控制顺序

        目标方法前的通知方法:数字小的先执行。

        目标方法后的通知方法:数字小的后执行。

代码演示:

demo1 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

@Component
@Aspect
@Order(3)
public class demo1 {

    @Around("execution(* org.example.controller.DeptController.getList())")
    public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {

        System.out.println("正在执行 demo1 切面类");

        Object result = joinPoint.proceed();

        System.out.println("正在执行 demo1 切面类");

        return result;

    }
}

demo2 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

@Component
@Aspect
@Order(2)
public class demo2 {

    @Around("execution(* org.example.controller.DeptController.getList())")
    public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {

        System.out.println("正在执行 demo2 切面类");

        Object result = joinPoint.proceed();

        System.out.println("正在执行 demo2 切面类");

        return result;
    }
}

demo3 切面类:

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

@Component
@Aspect
@Order(1)
public class demo3 {

    @Around("execution(* org.example.controller.DeptController.getList())")
    public Object demo(ProceedingJoinPoint joinPoint) throws Throwable {

        System.out.println("正在执行 demo3 切面类");

        Object result = joinPoint.proceed();

        System.out.println("正在执行 demo3 切面类");

        return  result;
    }
}

运行结果:

        5.0 AOP 切入点表达式

        切入点表达式是描述切入点方法的一种表达式,用来筛选连接点也就是选择目标方法,主要用来决定项目中的哪些方法需要加入通知。

常见的形式:

        1)execution():根据方法的签名来匹配。

        2)@annotation():根据注解匹配。

补充:什么是方法签名?

        方法签名是一个方法在源代码中的表示,它由方法的名称、返回类型、参数列表以及可能的抛出异常列表组成。而权限修饰符是不属于方法签名的一部分。

        5.1 使用 execution() 创建切入点表达式

        1)根据方法的签名来匹配连接点。

        execution 主要根据方法的返回值、包名、类名、方法名、方法参数等信息来匹配,语法为:

execution(访问修饰符  返回值  包名.类名.方法名(方法参数) throws 异常)

        其中访问修饰符、包名.类名、throws 异常这些部分代码是可以省略的。需要注意的是 throws 异常是方法上声明抛出的异常,不是实际抛出的异常。

        2)可以使用通配符描述切入点:

        *:单个独立的任意符号,可以通配任意返回值、包名、类名、方法名、任意类型的一个参数,也可以通配包、类、方法名的一部分。

        ..:多个连续的任意符号,可以通配任意层级的包,或任意类型、任意个数的参数。

举个例子:

        execution(* org.example.controller.DeptController.getList()) 切入点为:在任意返回值类型下的 org.example.controller 包下的 DeptController 类下的 getList 没有参数的方法。

        还可以根据业务需要可以使用 && 、 ||、 ! 来组合比较复杂的切入点表达式。 

        5.2 使用 @annotation 创建切入点表达式

        用于匹配标识有特定注解的方法。

        1)首先创建一个注解

代码演示:

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Log {
    
}

        在注解上加上两个元注解 @Retention 用来表示该注解什么时候生效,会被保留到运行时,可以通过反射机制在运行时获取注解信息。@Target 指定了注解可以应用的目标类型。

        2)接着手动给连接点也就是目标方法上加上自定义的注解,最后在 AOP 通知类型的注解属性中添加自定义注解的全类名。

代码演示:

    @Around("@annotation(org.example.Anto.Log)")
    public Object log(ProceedingJoinPoint proceedingJoinPoint){
        //目标方法执行之前,需要执行的代码

        proceedingJoinPoint.proceed();

        //目标方法执行之后,需要执行的代码
}

        6.0 AOP 连接点

        连接点简单来说就是 AOP 所控制的方法。

        在 Spring 中使用 JoinPoint 抽象了连接点,用它可以获得方法执行时的相关信息,如目标类名、方法名、方法参数等。

        对于 @Around 通知,获取连接点信息只能使用 ProceedingJoinPoint 。

        对于其他四种通知,获取连接点信息只能使用 JoinPoint,它是 ProceedingJoinPoint 的父类型。

代码演示:

//获取目标类的类名
String className = proceedingJoinPoint.getTarget().getClass().getName();

//获取目标方法名
String methodName = proceedingJoinPoint.getSignature().getName();

//获取目标方法的方法参数
Object[] args = proceedingJoinPoint.getArgs();

//获得目标方法的返回值
Object result = proceedingJoinPoint.proceed();

        7.0 AOP 案例 - 记录操作日志

         实现将每一次操作的操作信息记录到数据库中。

实现思路:

        先创建数据库:

        实现类:

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

import java.time.LocalDateTime;

@Data
@AllArgsConstructor
@NoArgsConstructor
public class OperateLog {
    private Integer id;
    private  Integer operateUser;
    private LocalDateTime operateTime;
    private String className;
    private String methodName;
    private String methodParams;
    private String returnValue;
    private Long costTime;
}

        AOPMapper 接口:

import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Mapper;
import org.example.Pojo.OperateLog;

@Mapper
public interface AOPMapper {


    @Insert("insert into operate_log(operate_time,class_name,method_name,method_params,return_value,cost_time) " +
            "values (#{operateTime},#{className},#{methodName},#{methodParams},#{returnValue},#{costTime})")
    public void log(OperateLog operateLog);
}

        定义 AOP 切面类:

import com.alibaba.fastjson.JSONObject;
import io.jsonwebtoken.Claims;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.example.Pojo.OperateLog;
import org.example.Utilities.JWT;
import org.example.mapper.AOPMapper;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import javax.servlet.http.HttpServletRequest;
import java.time.LocalDateTime;
import java.util.Arrays;

@Component
@Aspect
public class AOPLog {

    @Autowired
    HttpServletRequest request;

    @Autowired
    AOPMapper aopMapper;
    
    @Around("@annotation(org.example.Anto.Log)")
    public Object log(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {

        String jwt = request.getHeader("token");
        Claims claims = JWT.parse(jwt);

        LocalDateTime operateTime = LocalDateTime.now();

        //获取目标类的类名
        String className = proceedingJoinPoint.getTarget().getClass().getName();

        //获取目标方法名
        String methodName = proceedingJoinPoint.getSignature().getName();

        //获取目标方法的方法参数
        Object[] args = proceedingJoinPoint.getArgs();
        String methodParams = Arrays.toString(args);
        System.out.println("方法执行之前");

        long start = System.currentTimeMillis();
        //获得目标方法的返回值
        Object result = proceedingJoinPoint.proceed();
        long end = System.currentTimeMillis();

        String returnValue = JSONObject.toJSONString(result);
        Long costTime = end - start;

        OperateLog operateLog = new OperateLog(null,null,operateTime,className,methodName,methodParams,returnValue,costTime);

        aopMapper.log(operateLog);
        System.out.println(operateLog);
        System.out.println("方法执行之后");
        return result;
    }
}

        自定义的注解:

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Log {

}

         接着将 @Log 注解加到需要进行操作时记录的方法上即可。

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

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

相关文章

b端系统类管理平台设计前端开发案例

b端系统类管理平台设计前端开发案例

(学习笔记)数据基建-数据质量

数据基建-数据质量 数据质量数据质量保障措施如何推动上下游开展数据质量活动数据质量保障如何量化产出数据质量思考全链路数据质量保障项目 数据质量 概念&#xff1a;数据质量&#xff0c;意如其名&#xff0c;就是数据的准确性&#xff0c;他是数据仓库的基石&#xff0c;控…

【案例分享】印前制版工单系统:“鹿山科技”助力“铭匠数据”重塑业务流程

内容概要 本文介绍了鹿山信息科技通过明道云HAP平台的数字化解决方案提升了铭匠数据在印前制版行业的效率。周口铭匠数据科技有限公司位于河南省周口市沈丘县&#xff0c;是一家专注于印前制版设计服务的公司&#xff0c;成立于2023年。企业在销售业务、版材制作生产和美工设计…

CATIA入门操作案例——草图绘制案例,导入草图图片,尺寸约束直径/半径切换,草图分析闭合检查,草图固定

目录 引出草图绘制&#xff0c;导入图片方便绘制新建product&#xff0c;进入sketch tracer模块技巧&#xff1a;尺寸直径 / 半径切换技巧&#xff1a;右键&#xff0c;自动搜索 草图分析&#xff1a;检查闭合警告&#xff1a;Change it to material mode to see the Paintings…

60V大功率半桥GaN半桥驱动器替代LMG1210

1. 产品特性&#xff08;替代LMG1210&#xff09; ➢ 工作频率高达 10MHz ➢ 20ns 典型传播延迟 ➢ 5ns 高侧/低侧匹配 ➢ 两种输入控制模式 ➢ 具有可调死区时间的单个 PWM 输入、 独立输入模式 ➢ 1.5A 峰值拉电流和 3A 峰值灌电流 ➢ 内置 5V LDO ➢ 欠压保护 ➢ 过…

小程序简单版录音机

先来看看效果 结构 先来看看页面结构 <!-- wxml --><view class"wx-container"><view id"title">录音机</view><view id"time">{{hours}}:{{minute}}:{{second}}</view><view class"btngroup"…

【人工智能】第七部分:ChatGPT的未来展望

人不走空 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌赋&#xff1a;斯是陋室&#xff0c;惟吾德馨 目录 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌…

kafka安装流程

安装kafka前需要安装zookeeper zookeeper安装教程 1.新建一个logs文件夹 2.修改配置文件 3.修改listeners参数 4.以管理员身份启动kafka服务 .\bin\windows\kafka-server-start.bat .\config\server.properties 如果报 输入行太长。 命令语法不正确。 解决方案如下&#x…

基于工业互联网打造敏捷供应链的实现方式:创新路径与实践应用

引言 工业互联网和敏捷供应链是当今制造业发展中的两个重要概念。工业互联网以数字化、网络化和智能化为核心&#xff0c;致力于将传统工业生产与互联网技术相融合&#xff0c;从而实现生产过程的高效、智能和灵活。而敏捷供应链则强调快速响应市场需求、灵活调整生产和供应计划…

调用华为云实现人证核身证件版(二要素)

目录 1.作者介绍2.华为云人证核身2.1什么是人证核身2.2应用场景2.3限制要求 3.流程介绍3.1调用API实现3.2调用SDK实现 1.作者介绍 高凡平&#xff0c;男&#xff0c;西安工程大学电子信息学院&#xff0c;2023级研究生 研究方向&#xff1a;数码印花缺陷检测 电子邮件&#xf…

基于扩散动力学模型的乳腺癌在不完整DCE-MRI中的分割

文章目录 Diffusion Kinetic Model for Breast Cancer Segmentation in Incomplete DCE-MRI摘要方法实验结果 Diffusion Kinetic Model for Breast Cancer Segmentation in Incomplete DCE-MRI 摘要 针对现有方法需要完整时间序列数据(尤其是增强后图像)的问题,DKM仅利用预增…

wordpress里面嵌入哔哩哔哩视频的方法

我们正常如果从blibli获取视频分享链接然后在wordpress里面视频URL插入&#xff0c;发现是播放不了的 而视频嵌入代码直接粘贴呢窗口又非常的小 非常的难受&#xff0c;就需要更改一下代码。你可以在在allowfullscreen"true"的后面&#xff0c;留1个空格&#xff…

掌控数据流:深入解析 Java Stream 编程

Java 8 引入了一种新的抽象称为流&#xff08;Stream&#xff09;&#xff0c;它可以让你以一种声明的方式处理数据。Java 8 Stream API 可以极大提高 Java 程序员的生产力&#xff0c;使代码更简洁&#xff0c;更易读&#xff0c;并利用多核架构进行外部迭代。这里将详细介绍 …

Python Flask实现蓝图Blueprint配置和模块渲染

Python基础学习&#xff1a; Pyhton 语法基础Python 变量Python控制流Python 函数与类Python Exception处理Python 文件操作Python 日期与时间Python Socket的使用Python 模块Python 魔法方法与属性 Flask基础学习&#xff1a; Python中如何选择Web开发框架&#xff1f;Pyth…

SQLserver通过CLR调用TCP接口

一、SQLserver启用CLR 查看是否开启CRL&#xff0c;如果run_value1&#xff0c;则表示开启 EXEC sp_configure clr enabled; GO RECONFIGURE; GO如果未启用&#xff0c;则执行如下命令启用CLR sp_configure clr enabled, 1; GO RECONFIGURE; GO二、创建 CLR 程序集 创建新项…

【算法】深入浅出爬山算法:原理、实现与应用

人不走空 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌赋&#xff1a;斯是陋室&#xff0c;惟吾德馨 目录 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌…

简单聊下办公白环境

在当今信息化时代&#xff0c;办公环境对于工作效率和员工满意度有着至关重要的影响。而白名单作为一种网络安全策略&#xff0c;其是否适合办公环境&#xff0c;成为了许多企业和组织需要思考的问题。本文将从白名单的定义、特点及其在办公环境中的应用等方面&#xff0c;探讨…

Excel 生成所在月份的每一天列表

Excel 的 A2 格是日期 A1Fecha201/03/24 需要生成该日期所在月份的每一天的列表 A1WholeMonth201/03/24302/03/24403/03/24504/03/24605/03/24706/03/24807/03/24908/03/241009/03/241110/03/241211/03/241312/03/241413/03/241514/03/241615/03/241716/03/241817/03/241918…

操作系统入门系列-MIT6.828(操作系统工程)学习笔记(三)---- xv6初探与实验一(Lab: Xv6 and Unix utilities)

系列文章目录 操作系统入门系列-MIT6.S081&#xff08;操作系统&#xff09;学习笔记&#xff08;一&#xff09;---- 操作系统介绍与接口示例 操作系统入门系列-MIT6.828&#xff08;操作系统工程&#xff09;学习笔记&#xff08;二&#xff09;----课程实验环境搭建&#x…

Git - 详解 创建一个新仓库 / 推送现有文件夹 / 推送现有的 Git 仓库 到私有Gitlab

文章目录 【推送现有文件夹】详细步骤指令说明Git 全局设置设置Git全局用户名设置Git全局电子邮件地址 推送现有文件夹1. 进入现有文件夹2. 初始化Git仓库并设置初始分支为main3. 添加远程仓库4. 添加所有文件到暂存区5. 提交更改6. 推送代码到远程仓库并设置上游分支 创建一个…