欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

springboot升級(jí)到j(luò)dk21最新教程(2023年)

 更新時(shí)間:2023年10月27日 09:06:11   作者:JavaGPT  
你還在使用jdk8?快來(lái)看看最新出爐的SpringBoot+jdk21如何使用,下面這篇文章主要給大家介紹了關(guān)于springboot升級(jí)到j(luò)dk21的相關(guān)資料,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下

前言

隨著微服務(wù)的發(fā)展,越來(lái)越多的sql處理被放到j(luò)ava來(lái)處理,數(shù)據(jù)庫(kù)經(jīng)常會(huì)使用到對(duì)集合中的數(shù)據(jù)進(jìn)行分組求和,分組運(yùn)算等等。

那怎么樣使用java的stream優(yōu)雅的進(jìn)行分組求和或運(yùn)算呢?

一、準(zhǔn)備測(cè)試數(shù)據(jù)

這里測(cè)試數(shù)據(jù)學(xué)生,年齡類(lèi)型是Integer,身高類(lèi)型是BigDecimal,我們分別對(duì)身高個(gè)年齡進(jìn)行求和。

@Data
@AllArgsConstructor
@NoArgsConstructor
public class Student {
    /**
     * 姓名
     */
    private String name;
    /**
     * 年齡
     */
    private Integer age;
    /**
     * 身高
     */
    private BigDecimal stature;
}

public class LambdaLearn {
	// 初始化的測(cè)試數(shù)據(jù)集合
    static List<Student> list = new ArrayList<>();

    static {
    // 初始化測(cè)試數(shù)據(jù)
        list.add(new Student("張三", 18, new BigDecimal("185")));
        list.add(new Student("張三", 19, new BigDecimal("185")));
        list.add(new Student("張三2", 20, new BigDecimal("180")));
        list.add(new Student("張三3", 20, new BigDecimal("170")));
        list.add(new Student("張三3", 21, new BigDecimal("172")));
    }
}

二、按學(xué)生姓名分組求年齡和(Integer類(lèi)型的求和簡(jiǎn)單示例)

1.實(shí)現(xiàn)

// 按學(xué)生姓名分組求年齡和
public static void main(String[] args) {
    Map<String, Integer> ageGroup = list.stream().collect(Collectors.groupingBy(Student::getName
            , Collectors.summingInt(Student::getAge)));
    System.out.println(ageGroup);
}

執(zhí)行結(jié)果:
{張三=37, 張三3=41, 張三2=20}

三、按學(xué)生姓名分組求身高和(Collectors沒(méi)有封裝對(duì)應(yīng)的API)

1.實(shí)現(xiàn)一(推薦寫(xiě)法)

思路:先分組,再map轉(zhuǎn)換成身高BigDecimal,再用reduce進(jìn)行求和

public static void main(String[] args) {
   Map<String, BigDecimal> ageGroup = list.stream().collect(Collectors.groupingBy(Student::getName
            , Collectors.mapping(Student::getStature, Collectors.reducing(BigDecimal.ZERO, BigDecimal::add))));
    System.out.println(ageGroup);
}

執(zhí)行結(jié)果:
{張三=370, 張三3=342, 張三2=180}

2.實(shí)現(xiàn)二

思路:先分組,再收集成list,然后再map,再求和

public static void main(String[] args) {
   Map<String, BigDecimal> ageGroup = list.stream().collect(Collectors.groupingBy(Student::getName
            , Collectors.collectingAndThen(Collectors.toList()
                    , x -> x.stream().map(Student::getStature).reduce(BigDecimal.ZERO, BigDecimal::add))));
    System.out.println(ageGroup);
}

執(zhí)行結(jié)果:
{張三=370, 張三3=342, 張三2=180}

3.實(shí)現(xiàn)三

思路:業(yè)務(wù)時(shí)常在分組后需要做一些判斷邏輯再進(jìn)行累加業(yè)務(wù)計(jì)算,所以自己實(shí)現(xiàn)一個(gè)收集器

1.封裝一個(gè)自定義收集器

public class MyCollector {
    static final Set<Collector.Characteristics> CH_CONCURRENT_ID
            = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.CONCURRENT,
            Collector.Characteristics.UNORDERED,
            Collector.Characteristics.IDENTITY_FINISH));
    static final Set<Collector.Characteristics> CH_CONCURRENT_NOID
            = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.CONCURRENT,
            Collector.Characteristics.UNORDERED));
    static final Set<Collector.Characteristics> CH_ID
            = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.IDENTITY_FINISH));
    static final Set<Collector.Characteristics> CH_UNORDERED_ID
            = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.UNORDERED,
            Collector.Characteristics.IDENTITY_FINISH));
    static final Set<Collector.Characteristics> CH_NOID = Collections.emptySet();
    private MyCollector() {
    }
    @SuppressWarnings("unchecked")
    private static <I, R> Function<I, R> castingIdentity() {
        return i -> (R) i;
    }
    /**
     * @param <T> 集合元素類(lèi)型
     * @param <A> 中間結(jié)果容器
     * @param <R> 最終結(jié)果類(lèi)型
     */
    static class CollectorImpl<T, A, R> implements Collector<T, A, R> {
        private final Supplier<A> supplier;
        private final BiConsumer<A, T> accumulator;
        private final BinaryOperator<A> combiner;
        private final Function<A, R> finisher;
        private final Set<Characteristics> characteristics;
        CollectorImpl(Supplier<A> supplier,
                      BiConsumer<A, T> accumulator,
                      BinaryOperator<A> combiner,
                      Function<A, R> finisher,
                      Set<Characteristics> characteristics) {
            this.supplier = supplier;
            this.accumulator = accumulator;
            this.combiner = combiner;
            this.finisher = finisher;
            this.characteristics = characteristics;
        }
        CollectorImpl(Supplier<A> supplier,  // 產(chǎn)生結(jié)果容器
                      BiConsumer<A, T> accumulator,  // 累加器
                      BinaryOperator<A> combiner, // 將多個(gè)容器結(jié)果合并成一個(gè)
                      Set<Characteristics> characteristics) {
            this(supplier, accumulator, combiner, castingIdentity(), characteristics);
        }
        @Override
        public BiConsumer<A, T> accumulator() {
            return accumulator;
        }
        @Override
        public Supplier<A> supplier() {
            return supplier;
        }
        @Override
        public BinaryOperator<A> combiner() {
            return combiner;
        }
        @Override
        public Function<A, R> finisher() {
            return finisher;
        }
        @Override
        public Set<Characteristics> characteristics() {
            return characteristics;
        }
    }
    public static <T> Collector<T, ?, BigDecimal> summingDecimal(ToDecimalFunction<? super T> mapper) {
        return new MyCollector.CollectorImpl<>(
                () -> new BigDecimal[1],
                (a, t) -> {
                    if (a[0] == null) {
                        a[0] = BigDecimal.ZERO;
                    }
                    a[0] = a[0].add(Optional.ofNullable(mapper.applyAsDecimal(t)).orElse(BigDecimal.ZERO));
                },
                (a, b) -> {
                    a[0] = a[0].add(Optional.ofNullable(b[0]).orElse(BigDecimal.ZERO));
                    return a;
                },
                a -> a[0], CH_NOID);
    }
}

2.封裝一個(gè)函數(shù)式接口

@FunctionalInterface
public interface ToDecimalFunction<T> {
    BigDecimal applyAsDecimal(T value);
}

3.使用

public static void main(String[] args) {
    Map<String, BigDecimal> ageGroup = list.stream().collect(Collectors.groupingBy(Student::getName
            , MyCollector.summingDecimal(Student::getStature)));
    System.out.println(ageGroup);
}

總結(jié)

自定義實(shí)現(xiàn)收集器可以參考Collectors的內(nèi)部類(lèi)CollectorImpl的源碼,具體解析寫(xiě)到注釋中。

推薦通過(guò)模仿Collectors.summingInt()的實(shí)現(xiàn)來(lái)實(shí)現(xiàn)我們自己的收集器。

// T代表流中元素的類(lèi)型,A是中間處理臨時(shí)保存類(lèi)型,R代表返回結(jié)果的類(lèi)型
static class CollectorImpl<T, A, R> implements Collector<T, A, R> {
        private final Supplier<A> supplier;
        private final BiConsumer<A, T> accumulator;
        private final BinaryOperator<A> combiner;
        private final Function<A, R> finisher;
        private final Set<Characteristics> characteristics;

        CollectorImpl(Supplier<A> supplier,
                      BiConsumer<A, T> accumulator,
                      BinaryOperator<A> combiner,
                      Function<A,R> finisher,
                      Set<Characteristics> characteristics) {
            this.supplier = supplier;
            this.accumulator = accumulator;
            this.combiner = combiner;
            this.finisher = finisher;
            this.characteristics = characteristics;
        }

        CollectorImpl(Supplier<A> supplier,
                      BiConsumer<A, T> accumulator,
                      BinaryOperator<A> combiner,
                      Set<Characteristics> characteristics) {
            this(supplier, accumulator, combiner, castingIdentity(), characteristics);
        }

		// 這里提供一個(gè)初始化的容器,用于存儲(chǔ)每次累加。即使我們求和這里也只能使用容器存儲(chǔ),否則后續(xù)計(jì)算累加結(jié)果會(huì)丟失(累加結(jié)果不是通過(guò)返回值方式修改的)。
        @Override
        public Supplier<A> supplier() {
            return supplier;
        }
        
        // 累加計(jì)算:累加流中的每一個(gè)元素T到A容器存儲(chǔ)的結(jié)果中,這里沒(méi)有返回值,所以A必須是容器,避免數(shù)據(jù)丟失
        @Override
        public BiConsumer<A, T> accumulator() {
            return accumulator;
        }
        
        // 這里是當(dāng)開(kāi)啟parallelStream()并發(fā)處理時(shí),會(huì)得到多個(gè)結(jié)果容器A,這里對(duì)多個(gè)結(jié)果進(jìn)行合并
        @Override
        public BinaryOperator<A> combiner() {
            return combiner;
        }

		// 這里是處理中間結(jié)果類(lèi)型轉(zhuǎn)換成返回結(jié)果類(lèi)型
        @Override
        public Function<A, R> finisher() {
            return finisher;
        }
        
		// 這里標(biāo)記返回結(jié)果的數(shù)據(jù)類(lèi)型,這里取值來(lái)自于Collector接口的內(nèi)部類(lèi)Characteristics
        @Override
        public Set<Characteristics> characteristics() {
            return characteristics;
        }
    }
enum Characteristics {
// 表示此收集器是 并發(fā)的 ,這意味著結(jié)果容器可以支持與多個(gè)線程相同的結(jié)果容器同時(shí)調(diào)用的累加器函數(shù)。 
    CONCURRENT,

// 表示收集操作不承諾保留輸入元素的遇到順序。
    UNORDERED,
    
// 表示整理器功能是身份功能,可以被刪除。 
    IDENTITY_FINISH
}

到此這篇關(guān)于springboot升級(jí)到j(luò)dk21的文章就介紹到這了,更多相關(guān)springboot升級(jí)到j(luò)dk21內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Spring Boot項(xiàng)目使用Flyway的詳細(xì)教程

    Spring Boot項(xiàng)目使用Flyway的詳細(xì)教程

    這篇文章主要介紹了Spring Boot項(xiàng)目使用Flyway,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-07-07
  • 如何為Repository添加自定義方法

    如何為Repository添加自定義方法

    這篇文章主要介紹了如何為Repository添加自定義方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • 使用log4j2打印mybatis的sql執(zhí)行日志方式

    使用log4j2打印mybatis的sql執(zhí)行日志方式

    這篇文章主要介紹了使用log4j2打印mybatis的sql執(zhí)行日志方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • 揭秘SpringBoot!一分鐘教你實(shí)現(xiàn)配置的動(dòng)態(tài)神刷新

    揭秘SpringBoot!一分鐘教你實(shí)現(xiàn)配置的動(dòng)態(tài)神刷新

    在今天的指南中,我們將深入探索SpringBoot?動(dòng)態(tài)刷新的強(qiáng)大功能,讓你的應(yīng)用保持最新鮮的狀態(tài),想象一下,無(wú)需重啟,你的應(yīng)用就能實(shí)時(shí)更新配置,是不是很酷?跟我一起,讓我們揭開(kāi)這項(xiàng)技術(shù)如何讓開(kāi)發(fā)變得更加靈活和高效的秘密吧!
    2024-03-03
  • 詳解Java停止線程的四種方法

    詳解Java停止線程的四種方法

    如何停止java的線程一直是一個(gè)困惱我們開(kāi)發(fā)多線程程序的一個(gè)問(wèn)題嗎,本文主要介紹了詳解Java停止線程的四種方法,感興趣的可以了解一下
    2021-05-05
  • 最新評(píng)論