初识Java 12-2 流

news2025/1/13 14:04:39

目录

中间操作

跟踪与调试

对流元素进行排序

移除元素

将函数应用于每个流元素

在应用map()期间组合流

Optional类型

便捷函数

创建Optional

Optional对象上的操作

由Optional组成的流


本笔记参考自: 《On Java 中文版》


中间操作

||| 中间操作:从流中接收一个对象,并将对象作为另一个流送出后端,以连接其他操作。

跟踪与调试

        peek()操作是用于辅助调试的,它允许我们查看流对象而不修改它。

【例子:peek()的使用流】

public class Peeking {
    public static void main(String[] args)
            throws Exception {
        FileToWords.stream("Cheese.dat")
                .skip(21)
                .limit(4)
                .map(w -> w + " ")
                .peek(System.out::print) // 查看当前流的状态
                .map(String::toUpperCase)
                .peek(System.out::print)
                .map(String::toLowerCase)
                .forEach(System.out::print);
    }
}

        程序执行的结果是:

        其中,FileToWords类的形式如下(之后会再展示):

这个类的作用就是生成一个String对象组成的流。

        当流在管线中传输时,我们使用了peek()进行观察。这里要注意的是,peek()接受的是一个遵循Consumer函数式接口的函数:

这样的函数没有返回值Consumer一般用于对输入元素进行消费),所以也就不可能使用不同的对象替换流中的对象。因此我们只能“观察”这些对象。


对流元素进行排序

        sorted()方法可用于流的排序,它有一种可以接受一个Comparator参数的形式:

【例子:接受参数的sorted()

import java.util.Comparator;

public class SortedComparator {
    public static void main(String[] args)
            throws Exception {
        FileToWords.stream("D:\\code\\Java\\Test_Java\\src\\streams\\Cheese.dat")
                .skip(10)
                .limit(10)
                .sorted(Comparator.reverseOrder())
                .map(w -> w + " ")
                .forEach(System.out::print);
        System.out.println();
    }
}

        程序执行的结果是:

        也可以传入一个lambda表达式作为sorted()的参数,不过也有许多预设的Comparator


移除元素

        介绍两个用于从流中移除元素的操作:

  • distinct():可用于移除流中的重复元素。与Set相比,distinct()要更加简洁。
  • filter(Predicate):过滤,只保留符合特定条件(结果为true)的元素。

        distinct()在笔记12-1的Randoms.java中出现过。这里演示filter(Predicate)操作:

【例子:filter(Predicate)的使用例】

import java.util.stream.LongStream;

import static java.util.stream.LongStream.iterate;
import static java.util.stream.LongStream.rangeClosed;

public class Prime {
    public static boolean isPrime(long n) {
        return rangeClosed(2, (long) Math.sqrt(n)) // rangeClosed()在这里是类似于循环的作用
                .noneMatch(i -> n % i == 0); // 若流为空,或流中没有元素能够与当前谓词匹配,返回true
    }

    public LongStream numbers() {
        return iterate(2, i -> i + 1)
                .filter(Prime::isPrime); // 若返回结果是true,则保留下来
    }

    public static void main(String[] args) {
        new Prime().numbers()
                .limit(10)
                .forEach(n -> System.out.format("%d ", n));

        System.out.println();
        new Prime().numbers()
                .skip(90)
                .limit(10)
                .forEach(n -> System.out.format("%d ", n));
    }
}

        程序执行的结果是:

        rangeClosed()的参数是一个左开右闭的区间,包含了上界值。若没有一个取余操作的结果是0,则noneMatch()返回true,否则返回false。并且noneMatch()会在第一次结果为false时退出。


将函数应用于每个流元素

        本条目用于介绍一些作用于每个流元素的各种map操作:

  • map(Function):将Function应用于输入流中的每个对象,结果作为输出流继续传递。

        以下是map面对不同类型的不同版本:

  • mapToInt(ToIntFunction):应用于IntStream中。
  • mapToLong(ToLongFunction):应用于LongStream中。
  • mapToDouble(ToDoubleFunction):应用于LongStream中。

【例子:将Function(通过map)映射到一个用String组成的流中】

import java.util.Arrays;
import java.util.function.Function;
import java.util.stream.Stream;

public class FunctionMap {
    static String[] elements = {"12", "", "23", "45"};

    static Stream<String> testStream() {
        return Arrays.stream(elements);
    }

    static void test(String descr, Function<String, String> func) {
        System.out.println(" ---( " + descr + " )---");
        testStream()
                .map(func)
                .forEach(System.out::println);
    }

    public static void main(String[] args) {
        test("添加括号", s -> "[" + s + "]");

        test("数值增长", s -> {
            try {
                return Integer.parseInt(s) + 1 + ""; // parseInt()会尝试将String转换为Integer
            } catch (NumberFormatException e) { // 若无法完成转换,抛出异常,执行catch{}中的语句
                return s;
            }
        });

        test("替换", s -> s.replace("2", "9"));

        test("提取最后一位的数值",
                s -> s.length() > 0 ?
                        s.charAt(s.length() - 1) + "" : s); 
    }
}

        程序执行的结果是:

         在语句

return Integer.parseInt(s) + 1 + "";

中,表达式从左向右顺序进行,①先尝试将s转换为Integer,②再进行+1操作,③最后将其转换为字符串。

        另外,String.charAt()会返回指定索引处的char值。索引的范围是从0String.length()-1

---

        上述程序中,map()被用于将一个String映射到另一个String。而实际上,并没有约束要求生成的类型必须和输入的类型一致。因此还可以在map()中改变这个流的类型:

【例子:将基本类型的流映射到自定义类型中】

        在这个例子中,我们接受的是一个int类型的流,并且使用Numbered::new将其转换为了Numbered

---

        若Function生成的结果是某种数值类型,就需要使用相应的mapTo操作:

【例子:使用mapTo处理数值类型】

import java.util.stream.Stream;

public class FunctionMap3 {
    public static void main(String[] args) {
        Stream.of("5", "7", "9")
                .mapToInt(Integer::parseInt)
                .forEach(n -> System.out.format("%d ", n));

        System.out.println();
        Stream.of("17", "19", "23")
                .mapToLong(Long::parseLong)
                .forEach(n -> System.out.format("%d ", n));

        System.out.println();
        Stream.of("17", "1.9", ".23")
                .mapToDouble(Double::parseDouble)
                .forEach(n -> System.out.format("%f ", n));
    }
}

        程序执行的结果是:


在应用map()期间组合流

        假设我们现在有得到了一个由传入元素组成的流,我们需要对其运用map()

我们希望得到的是一个String类型的流,但结果却是一个由指向其他流的“头”组成的流。换句话说,我们想要的是一个由元素组成的流,但却生成了一个与元素流组成的流

        这种时候就需要使用flatMap()。该方法会做两件事:

  1. 接受生成流的函数,并将其应用于传入元素(这一点和map()一样)。
  2. 然后,将每个流“扁平化”处理,将其展开为元素。

    和map()类似,flatMap()也有面对不同类型的不同版本(flatMapToInt()flatMapToLong()等)。

【例子:flatMap()的使用例】

import java.util.stream.Stream;

public class FlatMap {
    public static void main(String[] args) {
        Stream.of(1, 2, 3)
                .flatMap(i -> Stream.of("A", "B", "C"))
                .forEach(System.out::println);
    }
}

        程序执行的结果是:

        从映射返回的每个流都被自动进行了扁平化处理,展开为组成这个流的String元素。

【例子:flatMapTo生成随机数】

import java.util.Random;
import java.util.stream.IntStream;
import java.util.stream.Stream;

public class StreamOfRandoms {
    static Random rand = new Random(47);

    public static void main(String[] args) {
        Stream.of(1, 2, 3, 4, 5)
                .flatMapToInt(i -> IntStream.concat( // concat()会按照参数的顺序将两个流组合到一起
                        rand.ints(0, 100).limit(i), // 其中的每一个流的长度不超过i
                        IntStream.of(-1)
                ))
                .forEach(n -> System.out.format("%d ", n));
        System.out.println();
    }
}

        程序执行的结果是:

        上述程序中出现了concat()

这个方法会按照参数的顺序将两个流组合到一起。所以,在每个有随机的Integer组成的流的末尾,都添加了一个-1作为标记。

    因为rand.ints()生成的是一个IntStream,所以这里使用的flatMap()concat()of()都是Integer的版本。

        在笔记12-1的FileToWordsRegexp.java中,我们将整个文件读取到内存的List中,这是需要存储空间的。但我们真正需要的,是一个不需要中间存储的单词流。这就是flatMap()解决的问题:

【例子:flatMap()创建占用少量内存的流】

import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.regex.Pattern;
import java.util.stream.Stream;

public class FileToWords {
    public static Stream<String> stream(String filepath)
            throws Exception {
        return Files.lines(Paths.get(filepath))
                .flatMap(line -> Pattern.compile("\\W+").splitAsStream(line));
    }
}

        stream()在这里是一个静态方法,因为它自己就可以完成整个流的创建。

        这里还出现了正则表达式\\W+\\W的意思是一个“非单词字符”,而+意味着“一个或是多个”。

    小写形式的\\w是指“单词字符”。

        由于语句

Pattern.compile().splitAsStream()

会生成一个流,所以若将其应用于map()操作中,我们会得到的是一个由单词流组成的流。而我们仅仅需要一个单词流而已。因此这里需要使用的是flatMap(),将其转化为一个由元素组成的简单流。

        除此之外,我们还可以使用String.split(),它会生成一个数组,这个数组可以经由Arrays.stream()转换为一个流:

.flatMap(line -> Arrays.stream(line.split("\\W+")))

此时得到的是一个真正的流(而不是像FileToWordsRegexp.java中那样,基于集合创建的流)。因此,每当我们想要一个新的流时,就必须从头创建,并且它无法复用

【例子:无法复用的流】

public class FileToWordsTest {
    public static void main(String[] args)
            throws Exception {
        FileToWords.stream("Cheese.dat")
                .limit(7)
                .forEach(s -> System.out.format("%s ", s));

        System.out.println();
        FileToWords.stream("Cheese.dat")
                .skip(7)
                .limit(2)
                .forEach(s -> System.out.format("%s ", s));
    }
}

        程序执行的结果如下:

Optional类型

        在进行编程时,我们或许会认为流被连接成了一条“快乐通道”(指没有异常或错误情形发生的默认场景),并假设没有什么能够中断这个流。但事实是,只需要在流中放入一个null就可以破坏它。

    当从流中提取null时,对null进行的类型转换会发生异常。

        因此就需要Optional类型。这一类型实现了这样一个概念:存在一个对象,即可以作为流元素来占位,也可以在需要寻找的元素不存在时提醒我们(不会抛出异常)。

        某些标准的流操作会返回Optional对象,因为这些操作不能确保所要的结果一定存在:

  • findFirst():返回包含第一个元素的Optional
  • findAny():返回包含任何元素的Optional
  • max()min():返回包含流中最大值或最小值的Optional

        以上三种操作在流为空的时候都返回Optional.empty

  • reduce()的其中一个版本:这个版本不会把一个“identity”对象作为其第一个参数(其他版本中,“identity”是默认结果),

    它的返回值会被包在一个Optional中。
  • average():将数值化的流(IntStreamLongStreamDoubleStream)包在一个Optional中。

【例子:返回Optional类型的操作】

import java.util.stream.IntStream;
import java.util.stream.Stream;

public class OptionalsFromEmptyStreams {
    public static void main(String[] args) {
        System.out.println(Stream.<String>empty()
                .findFirst());

        System.out.println(Stream.<String>empty()
                .findAny());

        System.out.println(Stream.<String>empty()
                .max(String.CASE_INSENSITIVE_ORDER));

        System.out.println(Stream.<String>empty()
                .min(String.CASE_INSENSITIVE_ORDER));

        System.out.println(Stream.<String>empty()
                .reduce((s1, s2) -> s1 + s2));

        System.out.println(IntStream.empty()
                .average());
    }
}

        程序执行的结果是:

        此时得到的结果不是抛出的异常,而是Optional.empty()对象。

        注意:这里通过Stream.<String>empty()创建了空流。若使用的是Stream.empty(),那么Java就无法通过这么有限的上下文信息推断出这个流的类型,但这种语法解决了这一问题:

Stream.<String> s = Stream.empty();

【例子:Optional的两个基本动作】

import java.util.Optional;
import java.util.stream.Stream;

public class OptionalBasics {
    static void test(Optional<String> optString) {
        if (optString.isPresent()) // isPresent():若存在值,返回true
            System.out.println(optString.get());
        else
            System.out.println("流中不存在数据");
    }

    public static void main(String[] args) {
        test(Stream.of("一个元素").findFirst());
        test(Stream.<String>empty().findFirst());
    }
}

        程序执行的结果是:

        当接收了一个Optional时,首先调用了isPresent(),对流中元素的存在与否进行测试。

便捷函数

        有许多便捷函数,由于获取Optional中的数据。这些函数简化了上述例子的“先检查、再处理”的过程。

  • isPresent(Consumer):若值存在,则使用这个值调用Consumer。否则不进行任何动作。
  • orElse(otherObject):若对象存在,则返回这个对象。否则返回otherObject
  • orElseGet(Supplier):若对象存在,则返回这个对象。否则返回使用Supplier函数创建的替代对象。
  • orElseThrow(Supplier):若对象存在,则返回这个对象。否则抛出一个使用Supplier函数创建的异常。

【例子:便捷函数】

import java.util.Optional;
import java.util.function.Consumer;
import java.util.stream.Stream;

public class Optionals {
    static void basics(Optional<String> optString) {
        if (optString.isPresent())
            System.out.println(optString.get());
        else
            System.out.println("流中不存在数据");
    }

    static void ifPresent(Optional<String> optString) {
        optString.ifPresent(System.out::println);
    }

    static void orElse(Optional<String> optString) {
        System.out.println(optString.orElse("从orElse()返回的对象"));
    }

    static void orElseGet(Optional<String> optString) {
        System.out.println(optString.orElseGet(() -> "orElseGet()替换的对象"));
    }

    static void orElseThrow(Optional<String> optString) {
        try {
            System.out.println(optString.orElseThrow(
                    () -> new Exception("orElseThrow()替换的对象")
            ));
        } catch (Exception e) { // 使用catch捕获Java库函数Optional::orElseThrow()抛出的异常
            System.out.println("捕获 " + e);
        }
    }

    static void test(String testName
            , Consumer<Optional<String>> cos) { // 一个可以接收所有示例方法的Consumer,避免代码重复
        System.out.println(" === " + testName + " === ");
        cos.accept(Stream.of("测试语句").findFirst());
        cos.accept(Stream.<String>empty().findFirst());
    }

    public static void main(String[] args) {
        test("basics", Optionals::basics);
        test("ifPresent", Optionals::ifPresent);
        test("orElse", Optionals::orElse);
        test("orElseGet", Optionals::orElseGet);
        test("orElseThrow", Optionals::orElseThrow);
    }
}

        程序执行的结果是:


创建Optional

        若要自己编写生成Optional的代码,可以使用以下三种静态方法:

  • empty():返回一个空的Optional
  • of(value):若已经确定这个value不是null,可以通过该方法将其包在一个Optional中。
  • ofNullable(value):若确定value是否为null,使用这个方法。若valuenull,该方法返回Optional.empty(),否则将这个value包在一个Optional中。

【例子:生成Optional的三个方法】

import java.util.Optional;

public class CreatingOptionals {
    static void test(String testName, Optional<String> opt) {
        System.out.println(" === " + testName + " === ");
        System.out.println(opt.orElse("Null"));
    }

    public static void main(String[] args) {
        test("empty", Optional.empty());

        test("of", Optional.of("Value"));
        try {
            test("of", Optional.of(null));
        } catch (Exception e) {
            System.out.println(e);
        }

        test("ofNullable", Optional.ofNullable("Value"));
        test("ofNullable", Optional.ofNullable(null));
    }
}

        程序执行的结果如下:

        若试图通过of()传递null来创建Optional,就会抛出空指针异常。相比之下,ofNullable()显得更加安全。


Optional对象上的操作

        若生成了一个Optional,有三种方法可以在最后再做一项处理:

  • filter(Predicate):将Predicate应用于Optional的内容,并返回其结果。若OptionalPredicate不匹配,则返回empty。若Optionalempty,返回其本身。
  • map(Function):若Optional不是empty,将Function应用于Optional包含的对象,并返回结果。否则返回empty
  • flatMap(Function):与map()类似,但所提供的映射函数会将结果包在Optional中,这样flatMap()最后就不会再做任何包装了。

    但是,数值化的Optional上没有上述的这些操作。

        filter()方法存在于普通的流中和在Optional中。但它们的行为并不相同,在普通的流中,若Predicate返回falsefilter()会将元素从流中删除。而Optional中的filter()则会将其转换为empty

【例子:filter()的用法】

import java.util.Arrays;
import java.util.function.Predicate;
import java.util.stream.Stream;

public class OptionalFilter {
    static String[] elements = {
            "Foo", "", "Bar", "Baz", "Bingo"
    };

    static Stream<String> testStream() {
        return Arrays.stream(elements);
    }

    static void test(String descr, Predicate<String> pred) {
        System.out.println(" ---( " + descr + " )---");
        for (int i = 0; i <= elements.length; i++) {
            System.out.println(
                    testStream() // 注意:每次进入for循环,都会重新获取一个流
                            .skip(i)
                            .findFirst()
                            .filter(pred)
            );
        }
    }

    public static void main(String[] args) {
        test("true", str -> true);
        test("false", str -> false);
        test("str != \"\"", str -> str != "");
        test("str.length() == 3", str -> str.length() == 3);
        test("startsWith(\"B\")", str -> str.startsWith("B"));
    }
}

        程序执行的结果是:

        这次的程序虽然输出结果像一个流,但实际上每次进入for循环都会重新获取一个流。只是skip()操作会跳过元素,以至于结果看上去是一个流。

        这次for循环语句的结束条件并不是i < elements.length,而是i <= elements.length。因此最后一个元素事实上会超出这个流,但超出的部分自动变为了一个Optional.empty

        Optional也有自己的map()。不同的是,只有当Optional不为empty时,map()才会应用其的映射函数。

    若Optional不为空,则将其传递给函数时,map()会首先提取Optional中的对象。

【例子:Optional.map()的使用例】

import java.util.Arrays;
import java.util.function.Function;
import java.util.stream.Stream;

public class OptionalMap {
    static String[] elements = {"12", "", "23", "45"};

    static Stream<String> testStream() {
        return Arrays.stream(elements);
    }

    static void test(String descr,
                     Function<String, String> func) {
        System.out.println(" ---( " + descr + " )---");
        for (int i = 0; i <= elements.length; i++) {
            System.out.println(
                    testStream()
                            .skip(i)
                            .findFirst() // 生成一个Optional
                            .map(func)
            );
        }
    }

    public static void main(String[] args) {
        test("加上括号", s -> "[" + s + "]");

        test("递增", s -> {
            try {
                return Integer.parseInt(s) + 1 + "";
            } catch (NumberFormatException e) {
                return s;
            }
        });

        test("替换", s -> s.replace("2", "9"));

        test("获取最后一位数字", s -> s.length() > 0 ?
                s.charAt(s.length() - 1) + "" : s);
    }
}

        程序执行的结果是:

        在函数完成之后,map()会先把结果包在一个Optional中,然后返回。并且,Optional.empty在遇到map()时直接通过,并没有被更改。

                OptionalflatMap()被应用于已经会生成Optional的映射函数,所以flatMap()并没有像map()一样进行包装的操作:

【例子:Optional.flatMap()的使用例】

import java.util.Arrays;
import java.util.Optional;
import java.util.function.Function;
import java.util.stream.Stream;

public class OptionalFlatMap {
    static String[] elements = {"12", "", "23", "45"};

    static Stream<String> testStream() {
        return Arrays.stream(elements);
    }

    static void test(String descr,
                     Function<String, Optional<String>> func) { // Function的返回类型是Optional<String>
        System.out.println(" ---( " + descr + " )---");
        for (int i = 0; i <= elements.length; i++) {
            System.out.println(
                    testStream()
                            .skip(i)
                            .findFirst()
                            .flatMap(func)
            );
        }
    }

    public static void main(String[] args) {
        // 用Optional.of()将函数括起来
        test("加上括号", s -> Optional.of("[" + s + "]"));

        test("递增", s -> {
            try {
                return Optional.of(Integer.parseInt(s) + 1 + "");
            } catch (NumberFormatException e) {
                return Optional.of(s);
            }
        });

        test("替换", s -> Optional.of(s.replace("2", "9")));

        test("获取最后一位数字", s -> Optional.of(s.length() > 0 ?
                s.charAt(s.length() - 1) + "" : s));
    }
}

        程序执行的结果是:

        flatMap()map()唯一的区别在于,flatMap()不会将结果包在Optional中,这件事会交由映射函数来做。


由Optional组成的流

        Optional可以处理null值。所以若存在一个可能会生成null值的生成器,并且这个生成器创建了一个流,我们自然会想要将这些元素包含在Optional中。

【例子:由Optional组成的流】

import java.util.Optional;
import java.util.Random;
import java.util.stream.Stream;

public class Signal {
    private final String msg;

    public Signal(String msg) {
        this.msg = msg;
    }

    public String getMsg() {
        return msg;
    }

    @Override
    public String toString() {
        return "Signal(" + msg + ")";
    }

    static Random rand = new Random(47);

    public static Signal morse() {
        switch (rand.nextInt(4)) {
            case 1:
                return new Signal("dot");
            case 2:
                return new Signal("dash");
            default:
                return null;
        }
    }

    public static Stream<Optional<Signal>> stream() {
        return Stream.generate(Signal::morse)
                .map(signal -> Optional.ofNullable(signal)); // 使用ofNullable将元素包入Optional中
    }
}

        当我们需要使用这个流的时候,我们需要考虑如何获取Optional中的对象:

【例子:从Optional的流中获取对象】

import java.util.Optional;

public class StreamOfOptionals {
    public static void main(String[] args) {
        Signal.stream()
                .limit(10)
                .forEach(System.out::println);

        System.out.println(" ---");
        Signal.stream()
                .limit(10)
                .filter(Optional::isPresent)
                .map(Optional::get)
                .forEach(System.out::println);
    }
}

        程序执行的结果如下:

        从Optional的流中提取对象时,往往会遇到“没有值”的情况。这就需要我们针对不同的应用采取不同的方法。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux嵌入式学习之Ubuntu入门(六)shell脚本详解

系列文章内容 Linux嵌入式学习之Ubuntu入门&#xff08;一&#xff09;基本命令、软件安装、文件结构、编辑器介绍 Linux嵌入式学习之Ubuntu入门&#xff08;二&#xff09;磁盘文件介绍及分区、格式化等 Linux嵌入式学习之Ubuntu入门&#xff08;三&#xff09;用户、用户组…

从0手写两轮差速机器人urdf模型

文章目录 前言一、基本理论二、实现步骤1.创建一个机器人建模功能包2.使用圆柱体创建一个车体模型2.同理创建机器人其它构件3.机器人模型添加传感器 前言 最近为找到与自己课题应用场景相适应的机器人结构&#xff0c;对机器人建模方面的内容进行了了解和学习&#xff0c;计划…

博途SCL区间搜索指令(判断某个数属于某个区间)

S型速度曲线行车位置控制,停靠位置搜索功能会用到区间搜索指令,下面我们详细介绍区间搜索指令的相关应用。 S型加减速行车位置控制(支持点动和停车位置搜索)-CSDN博客S型加减速位置控制详细算法和应用场景介绍,请查看下面文章博客。本篇文章不再赘述,这里主要介绍点动动和…

【nginx】Nginx配置:

文章目录 一、什么是Nginx&#xff1a;二、为什么使用Nginx&#xff1a;三、如何处理请求&#xff1a;四、什么是正向代理和反向代理&#xff1a;五、nginx 启动和关闭&#xff1a;六、目录结构&#xff1a;七、配置文件nginx.conf&#xff1a;八、location&#xff1a;九、单页…

嵌入式C 语言函数宏封装妙招

1. 函数宏介绍 函数宏&#xff0c;即包含多条语句的宏定义&#xff0c;其通常为某一被频繁调用的功能的语句封装&#xff0c;且不想通过函数方式封装来降低额外的弹栈压栈开销。 函数宏本质上为宏&#xff0c;可以直接进行定义&#xff0c;例如&#xff1a; #define INT_SWA…

Spring的注解开发-注解方式整合MyBatis代码实现

之前使用xml方式整合了MyBatis&#xff0c;文章导航&#xff1a;Spring整合第三方框架-MyBatis整合Spring实现-CSDN博客 现在使用注解的方式无非是就是将xml标签替换为注解&#xff0c;将xml配置文件替换为配置类而已。 非自定义配置类 package com.example.Configure;import c…

嵌入式系统中如何正确使用动态内存?

​ 大家好&#xff0c;今天给大家分享一下&#xff0c;动态内存的使用方法 一&#xff0e; 常见错误与预防 1. 分配后忘记释放内存 void func(void) {p malloc(len);do_something(p);return; /*错误&#xff01;退出程序时没有释放内存*/ }预防&#xff1a; 编写代码…

DevExpress ChartControl 画间断线

效果如下&#xff1a; 解决办法&#xff1a;数据源间断位置加入double.NaN demo下载

Linux 下如何调试代码

debug 和 release 在Linux下的默认模式是什么&#xff1f; 是release模式 那你怎么证明他就是release版本? 我们知道如果一个程序可以被调试&#xff0c;那么它一定是debug版本&#xff0c;如果它是release版本&#xff0c;它是没法被调试的&#xff0c;所以说我们可以来调试一…

基于SpringBoot+MyBatis实现的个人博客系统(一)

这篇主要讲解一下如何基于SpringBoot和MyBatis技术实现一个简易的博客系统(前端页面主要是利用CSS,HTML进行布局书写),前端的静态页面代码可以直接复制粘贴,后端的接口以及前端发送的Ajax请求需要自己书写. 博客系统需要完成的接口: 注册登录博客列表页展示博客详情页展示发布博…

【重拾C语言】二、顺序程序设计(基本符号、数据、语句、表达式、顺序控制结构、数据类型、输入/输出操作)

目录 前言 二、顺序程序设计 2.1 求绿化带面积——简单程序 2.2基本符号&#xff1a; 2.2.1 字符集 可视字符 不可视字符 2.2.2 C特定符 关键字 分隔符 运算符 2.2.3 标识符 2.2.4 间隔符 2.2.5 注释 2.3 数据 2.3.1 字面常量&#xff08;Literal Constants&am…

Flutter+SpringBoot实现ChatGPT流实输出

FlutterSpringBoot实现ChatGPT流式输出、上下文了连续对话 最终实现Flutter的流式输出上下文连续对话。 这里就是提供一个简单版的工具类和使用案例&#xff0c;此处页面仅参考。 服务端 这里直接封装提供工具类&#xff0c;修改自己的apiKey即可使用&#xff0c;支持连续…

FOC程序cubemx配置-ADC配置

具体配置步骤大家参考&#xff1a;这篇文章 我配置后用keil5自带的仿真工具查看引脚波形&#xff0c;在这里写一下遇到的问题。 1、波形仿真的时候出现 Unknown Signal&#xff1a;参考 这篇文章 2、生成的波形并不完全互补。 PS&#xff1a;出现以上这种情况时&#xff0…

【斗罗大陆2】动画新增12集备案,冰碧帝皇蝎形象被吐槽遭狂喷!

Hello,小伙伴们&#xff0c;我是小郑继续为大家深度解析斗罗大陆2绝世唐门。 《斗罗大陆2》动画新增12集备案 《斗罗大陆2》动画正在如火如荼的上映着&#xff0c;《斗罗大陆2》动画也在同步新增了。 在2023年9月全国重点网络动画片规划备案通过剧目信息中&#xff0c;《斗罗大…

【计算机网络】高级IO之select

文章目录 1. 什么是IO&#xff1f;什么是高效 IO? 2. IO的五种模型五种IO模型的概念理解同步IO与异步IO整体理解 3. 阻塞IO4. 非阻塞IOsetnonblock函数为什么非阻塞IO会读取错误&#xff1f;对错误码的进一步判断检测数据没有就绪时&#xff0c;返回做一些其他事情完整代码myt…

【算法|动态规划No.8】leetcode面试题 17.16. 按摩师

个人主页&#xff1a;兜里有颗棉花糖 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 兜里有颗棉花糖 原创 收录于专栏【手撕算法系列专栏】【LeetCode】 &#x1f354;本专栏旨在提高自己算法能力的同时&#xff0c;记录一下自己的学习过程&#xff0c;希望…

Python使用词云图展示

网上看到一个txt文本信息&#xff0c;共2351条饭否记录&#xff0c;据说是微信之父每天发的饭否记录&#xff0c;其实我不知道什么是饭否。我读取这个文本内容&#xff0c;展示到词语图上。之前也使用过&#xff0c;但是好久没有玩Python了&#xff0c;称假期空闲&#xff0c;练…

【从入门到起飞】IO高级流(1)(缓冲流,转换流,序列化流,反序列化流)

&#x1f38a;专栏【JavaSE】 &#x1f354;喜欢的诗句&#xff1a;天行健&#xff0c;君子以自强不息。 &#x1f386;音乐分享【如愿】 &#x1f384;欢迎并且感谢大家指出小吉的问题&#x1f970; 文章目录 &#x1f384;缓冲流&#x1f354;字节缓冲流&#x1f6f8;一次读取…

vue ant 隐藏列

vue ant 隐藏列 重要代码 type: FormTypes.hidden{ title: 序号, key: barCode, width: 10%, type: FormTypes.hidden},

YTM32的电源管理与低功耗系统详解

YTM32的电源管理与低功耗系统详解 苏勇&#xff0c;2023年10月 文章目录 YTM32的电源管理与低功耗系统详解缘起原理与机制电源管理模型的功耗模式正常模式&#xff08;Normal&#xff09;休眠模式&#xff08;Sleep&#xff09;深度休眠模式&#xff08;DeepSleep&#xff09;…