泛型用的好,下班走得早

简介:

泛型方法和类允许程序员编写高级化的单个方法或类,以用于不同的类型。

泛型方法和类允许程序员编写高级化的单个方法或类,以用于不同的类型。

我们每天都在使用它们,例如List ,Stream ,Map …

泛型类是普通类,其类名后跟一个类型参数部分。这些类称为参数化类或参数化类型,因为它们接受一个或多个参数。

特征:

  • 类型安全:泛型中只能容纳一种类型的对象。
  • 不需要类型转换:无需类型转换对象。
  • 编译时检查:在编译时进行检查,因此在运行时不会发生此问题。好的编程策略表明,在编译时处理问题比运行时要好得多。

类型参数命名约定:

  • T type
  • E element
  • K Key
  • N Number
  • V value

泛型通配符:

我们知道 ? 是通配符,它表示任何类型。如果我们写<? extends Number>,我们接受Number的任何子类,例如Integer,Float和double。

我们可以使用通配符作为参数,字段,返回类型或局部变量的类型。但是,不允许将通配符用作泛型方法调用,泛型类实例创建或超类型的类型参数。

通用类示例:

public class Response<T> {
 private List<T> data;
 private Integer page;
 private Integer elements;
}

在第一个简单示例中,我们创建了一个通用类Response,将用于返回分页的结果。我们不需要做魔术,也不必创建其他类。

如何写更少的代码?

尽管泛型在库中使用非常广泛,但是我看到很少有程序员使用泛型来泛化代码,例如,您有多少次看到控制器,服务和存储库几乎在做同样的事情?

在介绍完之后,我希望您展示一个简单的示例,说明如何编写一次性代码并在多个类中使用它。

一个简单的项目

在此示例中,我想向您展示如何定义BaseService和GeneralRepository来处理Java服务中的常见操作。

BaseEntity.java

@Data
@AllArgsConstructor
@NoArgsConstructor
public class BaseEntity {
    @Id
    protected String id;
    protected String tenantId;
    protected Date creationDate;
    protected Date updateDate;
}

我们正在创建一个简单的类,该类在我们的类中包含commons属性,下一个代码段将很有必要。

BaseRepository

@Repository
public interface GenericRepository<E extends BaseEntity> extends ReactiveCrudRepository<E, String> {
    Mono<Void> deleteByIdAndTenantId(String id, String tenantId);
    Mono<E> findFirstByIdAndTenantId(String id, String tenantId);
}

这是基本存储库,其中包含对我们的BaseEntity类的子类的常见查询。

BaseService

@Service
@Slf4j
public class BaseService<E extends BaseEntity> {

    @Autowired
    protected GenericRepository<E> generalRepo;

    public Mono<E> findEnitity(String entityId, String tenantId) {
        return generalRepo.findFirstByIdAndTenantId(entityId,tenantId);
    }

    public Mono<Void> delete(String entityId, String tenantId) {
        return findEnitity(entityId, tenantId)
                .switchIfEmpty(Mono.error(new NotFoundException(ITEM_NOT_FOUND)))
                .flatMap(e -> generalRepo.deleteByIdAndTenantId(e.getId(),e.getTenantId()));
    }
    
    public Mono<E> saveEntity(String tenantId, E entity) {
        entity.setTenantId(tenantId);
        return generalRepo.save(entity);
    }

    public Mono<E> updateEntity(String entityId, String tenantId,E entity) {
        return findEnitity(entityId,tenantId)
                .map(u -> saveEntity(tenantId,entity))
                .switchIfEmpty(Mono.error(new Exception(ITEM_NOT_FOUND)))
                .flatMap(m -> m);
    }
}

所有服务将继承BaseService继承继承的保存,更新,查找和删除方法,这些方法执行通用操作以将我们的数据持久保存在数据库中。显然,这段代码非常简单,您可以编写一种处理分页请求的方法,在执行插入/更新之前添加检查。

BaseController

public class BaseController<E extends BaseEntity> {

    @Autowired
    private BaseService<E> baseService;

    @PostMapping("/save")
    public Mono<E> save(Authentication auth,
                        @RequestBody E entity) {
        return baseService.save(getTenant(auth),entity);
    }
    
}

让我们以允许在多个控制器之间共享公共端点的BaseController结束该示例。

总结

因此,我们对泛型进行了一些小小的更新,并看到了如何编写更少的代码来使多个类之间的通用操作更加温和。

目录
相关文章
|
9月前
|
SQL Java 数据库
Lambda表达式你到哪个境界了?
日常开发中,我们很多时候需要用到Java 8的Lambda表达式,它允许把函数作为一个方法的参数,让我们的代码更优雅、更简洁。所以整理了一波工作中,我常用的,有哪些Lambda表达式。看完一定会有帮助的。
|
Oracle Java 关系型数据库
小朋友, 好好学学lambda表达式吧!
小朋友, 好好学学lambda表达式吧!
121 0
小朋友, 好好学学lambda表达式吧!
|
设计模式 Java 编译器
【Java泛型】你真的理解Java泛型吗?别再欺骗自己了。【建议小白收藏】
在学习设计模式之前,我对java泛型特性的了解仅限于表面的浅浅一层,最近突然对泛型来了兴趣,才想起详细的记录一下。泛型其实是通过类型参数使我们的程序具有更好的可读性和安全性。 本文参考java 泛型详解、Java中的泛型方法、 java泛型详解。
|
Java 编译器 C++
不懂泛型,怎么装逼,一文把泛型说的明明白白,安排!!!
泛型是Java中的高级概念,也是构建框架必备技能,比如各种集合类都是泛型实现的,今天详细聊聊Java中的泛型概念,希望有所收获。记得点赞,关注,分享哦。
108 0
不懂泛型,怎么装逼,一文把泛型说的明明白白,安排!!!
|
安全 编译器
我在一个构造方法中写了30个参数,老板看了想骂人
我在一个构造方法中写了30个参数,老板看了想骂人
86 0
|
存储 编译器 C++
萌新不看会后悔的C++基本类型总结(一)
萌新不看会后悔的C++基本类型总结(一)
萌新不看会后悔的C++基本类型总结(一)
|
存储 编译器 C++
萌新不看会后悔的C++基本类型总结(二)
萌新不看会后悔的C++基本类型总结(二)
萌新不看会后悔的C++基本类型总结(二)
|
C# C++
C#(五十)之泛型
泛型( Generic ) 允许您延迟编写类或方法中的编程元素的数据类型的规范,直到实际在程序中使用它的时候。换句话说,泛型允许您编写一个可以与任何数据类型一起工作的类或方法。可以理解为就是C++中的模板。
115 0
C#(五十)之泛型
|
设计模式 安全 Java
面试官:说说对单例模式的理解,最后的枚举实现我居然不知
说起单例模式(Singleton Pattern),想必大家都不不会陌生,它是 Java 中最简单的设计模式之一,属于创建型模式的一种,它提供了一种创建对象的最佳方式。 这种模式的意义在于保证一个类仅有一个实例,并提供一个访问它的全局访问点,避免重复的创建对象,节省系统资源。