Java-泛型的使用

Java-泛型的使用

    在日常开发中,我们经常会遇到一种情况:为了通用性,不得不写很多重复代码。Java 泛型(Generic)就是为了解决这个问题而诞生的。本文将通过实际项目中的 UserHolder 优化过程,带你一步步理解并学会使用泛型。

   一、Java 泛型介绍

   泛型是一种代码参数化的机制,它允许类、接口、方法在声明时使用类型参数,从而提高代码的复用性、类型安全性和可读性。

List<String> list = new ArrayList<>();

   这里 List<String> 就是使用了泛型,限定了这个列表只能存储 String 类型的数据,编译器会自动进行类型检查。

   泛型最常见的用途是集合类(如 List<T>、Map<K, V>),但它不仅仅局限于此,还可以用于我们自己的类和方法中。

   二、项目实践

   1. 应用举例

   在我们的应用中,常常使用 ThreadLocal 保存当前请求线程的用户信息,例如:

public class UserHolder {

    /**
     * C端用户上下文
     */
    private static final ThreadLocal<UserContext> threadLocal = new ThreadLocal<>();

    /**
     * 管理端用户上下文
     */
    private static final ThreadLocal<AdminUserContext> adminThreadLocal = new ThreadLocal<>();

    public static UserContext get() {
        return threadLocal.get();
    }

    public static void set(UserContext context) {
        threadLocal.set(context);
    }

    public static void clear() {
        threadLocal.remove();
    }
    public static AdminUserContext getAdmin() {
        return adminThreadLocal.get();
    }
    public static void setAdmin(AdminUserContext context) {
        adminThreadLocal.set(context);
    }

    public static void clearAdmin() {
        adminThreadLocal.remove();
    }
}

     在这个 UserHolder 类中,hreadLocal 和 adminThreadLocal 用于分别存放C端用户上下文和管理端用户上下文,它们通过 ThreadLocal 提供线程隔离,确保每个线程能够独立存储和访问自己的用户信息。

     2.  问题分析

     1)代码重复

   上面代码中,threadLocal 和 adminThreadLocal 的功能完全相似,唯一的区别是存储的用户上下文不同。这种重复的代码会导致维护上的麻烦,一旦需要对存取逻辑进行修改,可能需要修改多个地方,增加了出错的风险。

     2)不便于扩展

     随着业务的扩展,可能会增加更多的用户类型(例如,律师端用户、财务端用户等),每添加一个新的用户类型,就需要新增一个 ThreadLocal 变量以及相应的 getter、setter 方法,这会导致代码变得越来越冗长且不易扩展。

     3. 使用泛型优化UserHolder

     我们可以通过泛型的方式来简化 UserHolder 类,减少重复代码,使其更加灵活、可扩展。我们将 ThreadLocal 的类型化抽象出来,利用泛型让同一个 ThreadLocal 类可以存储不同类型的用户上下文,从而实现统一管理。

     优化后的 UserHolder 代码如下:

public class UserHolder<T> {
    // 泛型化的ThreadLocal,用于存储不同类型的上下文
    private static final ThreadLocal<Object> threadLocal = new ThreadLocal<>();

    public static <T> T get(Class<T> clazz) {
        return clazz.cast(threadLocal.get());
    }

    public static <T> void set(T context) {
        threadLocal.set(context);
    }

    public static void clear() {
        threadLocal.remove();
    }
}

    在优化后的 UserHolder 中,我们利用了泛型,get 和 set 方法都变得更加通用,可以接受任意类型的上下文对象。

    使用时,我们可以根据具体的上下文类型进行操作,例如:

// 设置C端用户上下文
UserHolder.set(new UserContext());

// 获取C端用户上下文
UserContext userContext = UserHolder.get(UserContext.class);

// 设置管理端用户上下文
UserHolder.set(new AdminUserContext());

// 获取管理端用户上下文
AdminUserContext adminUserContext = UserHolder.get(AdminUserContext.class);

    通过这种方式,UserHolder 类的代码更加简洁且具有可扩展性,当需要增加新的用户类型时,只需要在 set 和 get 方法中指定相应的类型即可,不再需要为每个类型定义一个独立的 ThreadLocal 变量。

    三、总结

   上文中,我们以UserHolder 中的 ThreadLocal 优化为例,演示了如何使用泛型提高代码的简洁性、可扩展性和维护性。通过泛型的方式,我们不仅解决了原始代码中的重复问题,还使得 UserHolder 能够灵活地存储和访问不同类型的用户上下文。

   使用泛型优化后的代码更加简洁,且容易扩展。当业务需求发生变化时,我们只需要修改少量代码,新增用户类型变得更加容易,减少了重复代码和维护成本。

posted @ 2025-04-21 15:25  欢乐豆123  阅读(26)  评论(0)    收藏  举报