【项目】抽奖系统bug历程(持续更新)

发布于:2025-05-23 ⋅ 阅读:(24) ⋅ 点赞:(0)

1. 构建错误码

在 common 层中,设置错误码

2. 构建自定义异常

注意事项:

  1. 继承自 RuntimeException
  2. @EqualsAndHashCode(callSuper = true)// 使用父类的equals和hashcode方法,不使用lombok生成的

3. 构建统一结果返回

common 层 中,设置统一结果返回(CommonResult)

注意事项:

  1. 使用泛型的格式:
    public static <T> CommonResult<T> error(Integer code, String msg) {
  1. Jackson 在进行序列化时,一般需要类有无参构造函数,并且属性要有对应的 getter 方法。

记得加上 @Data

默认的 lombok 依赖会出现错误:

<build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.8.1</version>// 加上版本号
                <configuration>
                    <annotationProcessorPaths>
                        <path>
                            <groupId>org.projectlombok</groupId>
                            <artifactId>lombok</artifactId>
                            <version>1.18.24</version>// 加上版本号
                        </path>
                    </annotationProcessorPaths>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <configuration>
                    <excludes>
                        <exclude>
                            <groupId>org.projectlombok</groupId>
                            <artifactId>lombok</artifactId>
                        </exclude>
                    </excludes>
                </configuration>
            </plugin>
        </plugins>
    </build>

在 Java 中,方法返回类型前的<T>不能省略,因为它是泛型方法的类型参数声明,用于告诉编译器:“这个方法使用一个名为T的泛型类型”。如果省略<T>,编译器会将T视为一个实际的类(如java.lang.Object的子类),而不是泛型类型参数,从而导致编译错误。

3.1. 返回类型前的<T> 为什么不能省略?

3.1.1. (1)语法规则

泛型方法的类型参数声明必须紧跟在方法修饰符(如publicstatic)之后,返回类型之前。省略<T>会导致语法错误:java

// 错误:缺少类型参数声明,编译器无法识别T
public void printArray(T[] array) { ... }  // 报错:Cannot resolve symbol 'T'

// 正确:声明泛型参数T
public <T> void printArray(T[] array) { ... }
3.1.2. (2)类型推断机制
  • 若省略<T>,编译器会认为T是一个已存在的具体类型(如类或接口),而非泛型类型参数。
  • 例如,若代码中没有定义名为T的类,编译器会报错 “无法解析符号 'T'”。

3.2. 泛型类型擦除机制

在 Java 中,泛型类型参数T在运行时必须被具体类型替换或擦除,否则会导致编译错误或运行时异常。

也就是说,运行时,不能存在还不知道替换成什么具体类型的T。

示例:

public static <T> T readValue(String content, T valueType) {
        return JacksonUtil.tryParse(()->{
           return JacksonUtil.getObjectMapper().readValue(content, valueType);
        });
    }

这段代码种的声明部分,存在运行时还未知的 T,所以错误!

更正:

public static <T> T readValue(String content, Class<T> valueType) {
        return JacksonUtil.tryParse(()->{
           return JacksonUtil.getObjectMapper().readValue(content, valueType);
        });
    }
3.2.1. 为什么是Class<T>,而不是class<T>?

在 Java 中,Class 和 class 的区别源于大小写的不同,这实际上代表了两种完全不同的语法概念:

  1. ClassJava 中的类型字面量

Class 是 Java 中的一个内置类,用于表示类的运行时类型信息(RTTI)。

泛型形式:Class 中的 T 表示该 Class 对象所代表的实际类型。

  1. class<T>:Java 中的语法错误
  • class 是 Java 的关键字,用于定义类(如 public class MyClass {})。
  • 泛型类定义:应使用 class MyClass<T> {},但不能单独作为类型使用。

Class 就像是 class 的类。

4. 构建序列化工具

工具有:

  1. fastjson
  2. jackson(选中)
  3. protobuf

可视化差、但是速度快。

演示 List 的:

/**
         * List序列化
         */
        List<CommonResult<String>> list = Arrays.asList(
                CommonResult.success("success1"),
                CommonResult.success("success2")
        );
        String s1;
        try {
            s1 = objectMapper.writeValueAsString(list);
            System.out.println("List序列化:" + s1);
        } catch (JsonProcessingException e) {
            throw new RuntimeException(e);
        }


        /**
         * List反序列化
         */
        JavaType javaType = objectMapper.getTypeFactory()
                .constructParametricType(List.class, CommonResult.class);
        try {
            List<CommonResult<String>> o = objectMapper.readValue(s1, javaType);
            System.out.println(Arrays.toString(new List[]{o}));
        } catch (JsonProcessingException e) {
            throw new RuntimeException(e);
        }

创建一个表示List<T>JavaType对象,其中TparameterClasses指定。


一直 try-catch,冗余,学习 spring 框架中是怎么处理的:

check.isAssignableFrom(ex.getClass())作用:

判断 ex 是否是 check 的实例/子类


5. 日志配置