【JavaSe】高级特性篇(八) Java8新特性函数式接口

2023-11-02

JavaSe·高级特性篇(八) Java8新特性函数式接口


1. 函数式接口

1.1 概念

函数式接口在Java中是指:有且仅有一个抽象方法的接口。

函数式接口,即适用于函数式编程场景的接口。而Java中的函数式编程体现就是Lambda,所以函数式接口就是可以适用于Lambda使用的接口。只有确保接口中有且仅有一个抽象方法,Java中的Lambda才能顺利地进行推导。

“语法糖”是指使用更加方便,但是原理不变的代码语法。例如在遍历集合时使用的for-each语法,其实底层的实现原理仍然是迭代器,这便是“语法糖”。从应用层面来讲,Java中的Lambda可以被当做是匿名内部类的“语法糖”,但是二者在原理上是不同的

1.2 格式

只要确保接口中有且仅有一个抽象方法即可:

修饰符 interface 接口名称 {
	public abstract 返回值类型 方法名称(可选参数信息);
	// 其他非抽象方法内容
}

由于接口当中抽象方法的 public abstract 是可以省略的,所以定义一个函数式接口很简单:

public interface MyFunctionalInterface {
	void myMethod();
}

1.3 @FunctionalInterface注解

与 @Override 注解的作用类似,Java 8中专门为函数式接口引入了一个新的注解: @FunctionalInterface 。该注解可用于一个接口的定义上:

@FunctionalInterface
public interface MyFunctionalInterface {
	void myMethod();
}

一旦使用该注解来定义接口,编译器将会强制检查该接口是否确实有且仅有一个抽象方法,否则将会报错。需要注意的是,即使不使用该注解,只要满足函数式接口的定义,这仍然是一个函数式接口,使用起来都一样。

1.4 自定义函数式接口

对于刚刚定义好的 MyFunctionalInterface 函数式接口,典型使用场景就是作为方法的参数:

// 使用自定义的函数式接口作为方法参数
private static void doSomething(MyFunctionalInterface inter) {
	inter.myMethod(); // 调用自定义的函数式接口方法
}
public static void main(String[] args) {
	// 调用使用函数式接口的方法
	doSomething(()> System.out.println("Lambda执行啦!"));
}

2. 函数式编程

2.1 Lambda的延迟执行

有些场景的代码执行后,结果不一定会被使用,从而造成性能浪费。而Lambda表达式是延迟执行的,这正好可以作为解决方案,提升性能。

性能浪费的日志案例

日志可以帮助我们快速的定位问题,记录程序运行过程中的情况,以便项目的监控和优化。一种典型的场景就是对参数进行有条件使用,例如对日志消息进行拼接后,在满足条件的情况下进行打印输出:

private static void log(int level, String msg) {
	if (level == 1) {
		System.out.println(msg);
	}
}
public static void main(String[] args) {
	String msgA = "Hello";
	String msgB = "World";
	String msgC = "Java";
	log(1, msgA + msgB + msgC);
}

这段代码存在问题:无论级别是否满足要求,作为 log 方法的第二个参数,三个字符串一定会首先被拼接并传入方法内,然后才会进行级别判断。如果级别不符合要求,那么字符串的拼接操作就白做了,存在性能浪费。

SLF4J是应用非常广泛的日志框架,它在记录日志时为了解决这种性能浪费的问题,并不推荐首先进行字符串的拼接,而是将字符串的若干部分作为可变参数传入方法中,仅在日志级别满足要求的情况下才会进行字符串拼接。例如: LOGGER.debug(“变量{}的取值为{}。”, “os”, “macOS”) ,其中的大括号 {} 为占位符。如果满足日志级别要求,则会将“os”和“macOS”两个字符串依次拼接到大括号的位置;否则不会进行字符串拼接。这也是一种可行解决方案,但Lambda可以做到更好。

体验Lambda的更优写法:

使用Lambda必然需要一个函数式接口:

@FunctionalInterface
public interface MessageBuilder {
	String buildMessage();
}

然后对 log 方法进行改造:

private static void log(int level, MessageBuilder builder) {
	if (level == 1) {
		System.out.println(builder.buildMessage());
	}
}
public static void main(String[] args) {
	String msgA = "Hello";
	String msgB = "World";
	String msgC = "Java";
	log(1, ()> msgA + msgB + msgC );
}

这样一来,只有当级别满足要求的时候,才会进行三个字符串的拼接;否则三个字符串将不会进行拼接。

证明Lambda的延迟
private static void log(int level, MessageBuilder builder) {
	if (level == 1) {
		System.out.println(builder.buildMessage());
	}
}
public static void main(String[] args) {
	String msgA = "Hello";
	String msgB = "World";
	String msgC = "Java";
	log(2, ()> {
		System.out.println("Lambda执行!");
		return msgA + msgB + msgC;
	});
}

在不符合级别要求的情况下,Lambda将不会执行。从而达到节省性能的效果。

实际上使用内部类也可以达到同样的效果,只是将代码操作延迟到了另外一个对象当中通过调用方法来完成。而是否调用其所在方法是在条件判断之后才执行的。

2.2 使用Lambda作为参数和返回值

如果抛开实现原理不说,Java中的Lambda表达式可以被当作是匿名内部类的替代品。如果方法的参数是一个函数式接口类型,那么就可以使用Lambda表达式进行替代。使用Lambda表达式作为方法参数,其实就是使用函数式接口作为方法参数.

例如 java.lang.Runnable 接口就是一个函数式接口,假设有一个startThread 方法使用该接口作为参数,那么就可以使用Lambda进行传参。这种情况其实和 Thread 类的构造方法参数为 Runnable 没有本质区别。

private static void startThread(Runnable task) {
	new Thread(task).start();
}
public static void main(String[] args) {
	startThread(()> System.out.println("线程任务执行!"));
}

类似地,如果一个方法的返回值类型是一个函数式接口,那么就可以直接返回一个Lambda表达式。当需要通过一个方法来获取一个 java.util.Comparator 接口类型的对象作为排序器时,就可以调该方法获取。

private static Comparator<String> newComparator() {
	return (a, b)> b.length() ‐ a.length();
}
public static void main(String[] args) {
	String[] array = { "abc", "ab", "abcd" };
	System.out.println(Arrays.toString(array));
	Arrays.sort(array, newComparator());
	System.out.println(Arrays.toString(array));
}

直接return一个Lambda表达式即可

3. 常用函数式接口

JDK提供了大量常用的函数式接口以丰富Lambda的典型使用场景,它们主要在 java.util.function 包中被提供。

3.1 Supplier接口

java.util.function.Supplier 接口仅包含一个无参的方法:

  • T get()
    用来获取一个泛型参数指定类型的对象数据。由于这是一个函数式接口,这也就意味着对应的Lambda表达式需要“对外提供”一个符合泛型类型的对象数据。
private static String getString(Supplier<String> function) {
	return function.get();
}
public static void main(String[] args) {
	String msgA = "Hello";
	String msgB = "World";
	System.out.println(getString(()> msgA + msgB));
}
求数组元素最大值

使用 Supplier 接口作为方法参数类型,通过Lambda表达式求出int数组中的最大值。提示:接口的泛型请使用 java.lang.Integer 类

//定一个方法,方法的参数传递Supplier,泛型使用Integer
public static int getMax(Supplier<Integer> sup){
	return sup.get();
}
public static void main(String[] args) {
	int arr[] = {2,3,4,52,333,23};
	//调用getMax方法,参数传递Lambda
	int maxNum = getMax(()>{
		//计算数组的最大值
		int max = arr[0];
		for(int i : arr){
			if(i>max){
			max = i;
			}
		}
		return max;
	});
	System.out.println(maxNum);
}

3.2 Consumer接口

java.util.function.Consumer 接口则正好与Supplier接口相反,它不是生产一个数据,而是消费一个数据,其数据类型由泛型决定。

抽象方法:accept

Consumer 接口中包含抽象方法 :

  • void accept(T t) 意为消费一个指定泛型的数据。
private static void consumeString(Consumer<String> function) {
    function.accept("Hello");
}

public static void main(String[] args) {
    consumeString(s -> System.out.println(s));
}

更好的写法是使用方法引用

默认方法:andThen

如果一个方法的参数和返回值全都是 Consumer 类型,那么就可以实现效果:消费数据的时候,首先做一个操作,然后再做一个操作,实现组合。而这个方法就是 Consumer 接口中的default方法 andThen 。下面是JDK的源代码:

default Consumer<T> andThen(Consumer<? super T> after) {
	Objects.requireNonNull(after);
	return (T t)> { accept(t); after.accept(t); };
}

java.util.Objects 的 requireNonNull 静态方法将会在参数为null时主动抛出NullPointerException 异常。这省去了重复编写if语句和抛出空指针异常的麻烦。

要想实现组合,需要两个或多个Lambda表达式即可,而 andThen 的语义正是“一步接一步”操作。

private static void consumeString(Consumer<String> one, Consumer<String> two) {
	one.andThen(two).accept("Hello");
}
public static void main(String[] args) {
	consumeString(
		s ‐> System.out.println(s.toUpperCase()),
		s ‐> System.out.println(s.toLowerCase())
	);
}

运行结果将会首先打印完全大写的HELLO,然后打印完全小写的hello。当然,通过链式写法可以实现更多步骤的组合

格式化打印信息

下面的字符串数组当中存有多条信息,请按照格式“ 姓名:XX。性别:XX。 ”的格式将信息打印出来。要求将打印姓名的动作作为第一个 Consumer 接口的Lambda实例,将打印性别的动作作为第二个 Consumer 接口的Lambda实例,将两个 Consumer 接口按照顺序“拼接”到一起。

private static void printInfo(Consumer<String> one, Consumer<String> two, String[] array) {
	for (String info : array) {
		one.andThen(two).accept(info); // 姓名:迪丽热巴。性别:女。
	}
}
public static void main(String[] args) {
	String[] array = { "迪丽热巴, 女", "古力娜扎, 女", "马尔扎哈, 男" };
	printInfo(
		s ‐> System.out.print("姓名:" + s.split(",")[0]),
		s ‐> System.out.println("。性别:" + s.split(",")[1] + "。"),
		array
	);
}

3.3 Predicate接口

有时候我们需要对某种类型的数据进行判断,从而得到一个boolean值结果。这时可以使用
java.util.function.Predicate 接口

抽象方法:test

Predicate 接口中包含一个抽象方法:

  • boolean test(T t) 。用于条件判断的场景
private static void method(Predicate<String> predicate) {
	boolean veryLong = predicate.test("HelloWorld");
	System.out.println("字符串很长吗:" + veryLong);
}
public static void main(String[] args) {
	method(s ‐> s.length() > 5);
}

条件判断的标准是传入的Lambda表达式逻辑,只要字符串长度大于5则认为很长。

默认方法:and

既然是条件判断,就会存在与、或、非三种常见的逻辑关系。其中将两个 Predicate 条件使用“与”逻辑连接起来实现“并且”的效果时,可以使用default方法 and 。其JDK源码为:

default Predicate<T> and(Predicate<? super T> other) {
	Objects.requireNonNull(other);
	return (t)> test(t) && other.test(t);
}

判断一个字符串既要包含大写“H”,又要包含大写“W”

private static void method(Predicate<String> one, Predicate<String> two) {
	boolean isValid = one.and(two).test("Helloworld");
	System.out.println("字符串符合要求吗:" + isValid);
}
public static void main(String[] args) {
	method(s ‐> s.contains("H"), s ‐> s.contains("W"));
}
默认方法:or

与 and 的“与”类似,默认方法 or 实现逻辑关系中的“或”。JDK源码为:

default Predicate<T> or(Predicate<? super T> other) {
	Objects.requireNonNull(other);
	return (t)> test(t) || other.test(t);
}

实现逻辑“字符串包含大写H或者包含大写W”,那么代码只需要将“and”修改为“or”名称即可,其他都不变

private static void method(Predicate<String> one, Predicate<String> two) {
	boolean isValid = one.or(two).test("Helloworld");
	System.out.println("字符串符合要求吗:" + isValid);
}
public static void main(String[] args) {
	method(s ‐> s.contains("H"), s ‐> s.contains("W"));
}

默认方法:negate

“与”、“或”已经了解了,剩下的“非”(取反)也会简单。默认方法 negate 的JDK源代码为:

default Predicate<T> negate() {
	return (t)> !test(t);
}

从实现中很容易看出,它是执行了test方法之后,对结果boolean值进行“!”取反而已。一定要在 test 方法调用之前调用 negate 方法,正如 and 和 or 方法一样

private static void method(Predicate<String> predicate) {
	boolean veryLong = predicate.negate().test("HelloWorld");
	System.out.println("字符串很长吗:" + veryLong);
}
public static void main(String[] args) {
	method(s ‐> s.length() < 5);
}
集合信息筛选

数组当中有多条“姓名+性别”的信息如下,请通过 Predicate 接口的拼装将符合要求的字符串筛选到集合ArrayList 中,需要同时满足两个条件:

  1. 必须为女生
  2. 姓名为4个字
public static void main(String[] args) {
	String[] array = { "迪丽热巴,女", "古力娜扎,女", "马尔扎哈,男", "赵丽颖,女" };
	List<String> list = filter(
							array,
							s ‐> "女".equals(s.split(",")[1]),
							s ‐> s.split(",")[0].length() == 4
						);
	System.out.println(list);
}
private static List<String> filter(String[] array, Predicate<String> one,
	Predicate<String> two) {
		List<String> list = new ArrayList<>();
		for (String info : array) {
		if (one.and(two).test(info)) {
			list.add(info);
		}
	}
	return list;
}

3.4 Function接口

java.util.function.Function<T,R> 接口用来根据一个类型的数据得到另一个类型的数据,前者称为前置条件,后者称为后置条件。

抽象方法:apply

Function 接口中最主要的抽象方法为:

  • R apply(T t) ,根据类型T的参数获取类型R的结果。

将 String 类型转换为 Integer 类型

private static void method(Function<String, Integer> function) {
	int num = function.apply("10");
	System.out.println(num + 20);
}
public static void main(String[] args) {
	method(s ‐> Integer.parseInt(s));
}

最好是通过方法引用的写法

默认方法:andThen

Function 接口中有一个默认的 andThen 方法,用来进行组合操作。JDK源代码如:

default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
	Objects.requireNonNull(after);
	return (T t)> after.apply(apply(t));
}

该方法同样用于“先做什么,再做什么”的场景,和 Consumer 中的 andThen 差不多:

private static void method(Function<String, Integer> one, Function<Integer, Integer> two) {
	int num = one.andThen(two).apply("10");
	System.out.println(num + 20);
}
public static void main(String[] args) {
	method(str‐>Integer.parseInt(str)+10, i ‐> i *= 10);
}

第一个操作是将字符串解析成为int数字,第二个操作是乘以10。两个操作通过 andThen 按照前后顺序组合到了一起

请注意,Function的前置条件泛型和后置条件泛型可以相同。

自定义函数模型拼接

请使用 Function 进行函数模型的拼接,按照顺序需要执行的多个函数操作为:

String str = “赵丽颖,20”;

  1. 将字符串截取数字年龄部分,得到字符串;
  2. 将上一步的字符串转换成为int类型的数字;
  3. 将上一步的int数字累加100,得到结果int数字。
public static void main(String[] args) {
	String str = "赵丽颖,20";
	int age = getAgeNum(
					str, s ‐> s.split(",")[1],
					s ‐>Integer.parseInt(s),
					n ‐> n += 100
				);
	System.out.println(age);
}
private static int getAgeNum(String str, Function<String, String> one, Function<String, Integer> two, Function<Integer, Integer> three) {
	return one.andThen(two).andThen(three).apply(str);
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【JavaSe】高级特性篇(八) Java8新特性函数式接口 的相关文章

  • Java 中等效的并行扩展

    我在 Net 开发中使用并行扩展有一些经验 但我正在考虑在 Java 中做一些工作 这些工作将受益于易于使用的并行库 JVM 是否提供任何与并行扩展类似的工具 您应该熟悉java util concurrent http java sun
  • Play框架运行应用程序问题

    每当我尝试运行使用以下命令创建的新 Web 应用程序时 我都会收到以下错误Play http www playframework org Error occurred during initialization of VM Could no
  • 在 HTTPResponse Android 中跟踪重定向

    我需要遵循 HTTPost 给我的重定向 当我发出 HTTP post 并尝试读取响应时 我得到重定向页面 html 我怎样才能解决这个问题 代码 public void parseDoc final HttpParams params n
  • JAXb、Hibernate 和 beans

    目前我正在开发一个使用 Spring Web 服务 hibernate 和 JAXb 的项目 1 我已经使用IDE hibernate代码生成 生成了hibernate bean 2 另外 我已经使用maven编译器生成了jaxb bean
  • 多个 Maven 配置文件激活多个 Spring 配置文件

    我想在 Maven 中构建一个环境 在其中我想根据哪些 Maven 配置文件处于活动状态来累积激活多个 spring 配置文件 目前我的 pom xml 的相关部分如下所示
  • 控制Android的前置LED灯

    我试图在用户按下某个按钮时在前面的 LED 上实现 1 秒红色闪烁 但我很难找到有关如何访问和使用前置 LED 的文档 教程甚至代码示例 我的意思是位于 自拍 相机和触摸屏附近的 LED 我已经看到了使用手电筒和相机类 已弃用 的示例 但我
  • 禁止的软件包名称:java

    我尝试从数据库名称为 jaane 用户名 Hello 和密码 hello 获取数据 错误 java lang SecurityException Prohibited package name java at java lang Class
  • 如何将 pfx 文件转换为 jks,然后通过使用 wsdl 生成的类来使用它来签署传出的肥皂请求

    我正在寻找一个代码示例 该示例演示如何使用 PFX 证书通过 SSL 访问安全 Web 服务 我有证书及其密码 我首先使用下面提到的命令创建一个 KeyStore 实例 keytool importkeystore destkeystore
  • 使用Caliper时如何指定命令行?

    我发现 Google 的微型基准测试项目 Caliper 非常有趣 但文档仍然 除了一些示例 完全不存在 我有两种不同的情况 需要影响 JVM Caliper 启动的命令行 我需要设置一些固定 最好在几个固定值之间交替 D 参数 我需要指定
  • 如何在 javadoc 中使用“<”和“>”而不进行格式化?

    如果我写
  • 在 Mac 上正确运行基于 SWT 的跨平台 jar

    我一直致力于一个基于 SWT 的项目 该项目旨在部署为 Java Web Start 从而可以在多个平台上使用 到目前为止 我已经成功解决了由于 SWT 依赖的系统特定库而出现的导出问题 请参阅相关thread https stackove
  • Google App Engine 如何预编译 Java?

    App Engine 对应用程序的 Java 字节码使用 预编译 过程 以增强应用程序在 Java 运行时环境中的性能 预编译代码的功能与原始字节码相同 有没有详细的信息这是做什么的 我在一个中找到了这个谷歌群组消息 http groups
  • 如何从终端运行处理应用程序

    我目前正在使用加工 http processing org对于一个小项目 但是我不喜欢它附带的文本编辑器 我使用 vim 编写所有代码 我找到了 pde 文件的位置 并且我一直在从 vim 中编辑它们 然后重新打开它们并运行它们 重新加载脚
  • 如何从泛型类调用静态方法?

    我有一个包含静态创建方法的类 public class TestClass public static
  • 如何在桌面浏览器上使用 webdriver 移动网络

    我正在使用 selenium webdriver 进行 AUT 被测应用程序 的功能测试自动化 AUT 是响应式网络 我几乎完成了桌面浏览器的不同测试用例 现在 相同的测试用例也适用于移动浏览器 因为可以从移动浏览器访问 AUT 由于它是响
  • 获取 JVM 上所有引导类的列表?

    有一种方法叫做findBootstrapClass对于一个类加载器 如果它是引导的 则返回一个类 有没有办法找到类已经加载了 您可以尝试首先通过例如获取引导类加载器呼叫 ClassLoader bootstrapLoader ClassLo
  • 静态变量的线程安全

    class ABC implements Runnable private static int a private static int b public void run 我有一个如上所述的 Java 类 我有这个类的多个线程 在里面r
  • 编译器抱怨“缺少返回语句”,即使不可能达到缺少返回语句的条件

    在下面的方法中 编译器抱怨缺少退货声明即使该方法只有一条路径 并且它包含一个return陈述 抑制错误需要另一个return陈述 public int foo if true return 5 鉴于Java编译器可以识别无限循环 https
  • 使用 JMF 创建 RTP 流时出现问题

    我正处于一个项目的早期阶段 需要使用 RTP 广播DataStream创建自MediaLocation 我正在遵循一些示例代码 该代码目前在rptManager initalize localAddress 出现错误 无法打开本地数据端口
  • 如何实现仅当可用内存较低时才将数据交换到磁盘的写缓存

    我想将应用程序生成的数据缓存在内存中 但如果内存变得稀缺 我想将数据交换到磁盘 理想情况下 我希望虚拟机通知它需要内存并将我的数据写入磁盘并以这种方式释放一些内存 但我没有看到任何方法以通知我的方式将自己挂接到虚拟机中before an O

随机推荐

  • Java 统计连续签到天数

    说明 签到功能根据业务情况的不同大致分为两种 一种是只要进入系统就会默认签到 另外一种是进入系统之后需要用户手动点击签到按钮进行签到 两种情况最大的区别就是 自动签到连续签到天数肯定至少为1天 而手动签到的情况则需要考虑断签的情况 1 自动
  • 关于list容器(链表)

    list容器基本概念 list优点 list容器的迭代器 以下代码为验证list是否为循环链表 list构造函数 list数据元素插入和删除操作 list大小操作 list赋值操作 list数据的存取 list常用操作 list容器基本概念
  • QGLViewer+VS(MSVC2013)配置

    工具 libQGLViewer 2 7 2 Visual Studio 2013 Uitimate QT5 9 1 MSVC2013 另外在VS中使用QT显示界面需要下载qt vsaddin msvc2013 2 2 0 vsix官网下载
  • STM32的HAL库开发系列 - GPIO中断/外部中断EXTI

    STM32的HAL库开发系列 GPIO中断 外部中断EXTI 中断它可以在GPIO口的电平发生变化时产生中断 从而使得STM32微控制器能够及时响应外部设备的变化 STM32的GPIO中断 外部中断EXTI可以配置为上升沿中断 下降沿中断和
  • c/c++教程 - 2.4.3 this指针作用,链式编程思想,空指针访问成员函数,const修饰成员函数,常函数,常对象

    目录 4 3 C 对象模型和this指针 4 3 1 成员变量和成员函数分开存储 4 3 2 this指针概念 非常重要 4 3 3 空指针访问成员函数 4 3 4 const修饰成员函数 相关教程 4 3 C 对象模型和this指针 4
  • C++基本数据类型

    一 数据类型 C 的数据类型有四类 基本类型 结构类型 指针类型 和空类型 void 二 部分基本数据类型说明 1 整型 1 int 整数就是没有小数点部分的数字 如2 98 100和0 在C 中提供多种整型 按示数范围分为短整数 shor
  • "Mac OS X"想要进行更改。键入管理员的名称和密码以允许执行此操作("Mac OS X"想使用系统钥匙串)...

    不知什么时候开始 每次我在运行xcode在真机上 或者archive打包的时间 都会弹出输入用户名和密码的框 搞的烦死了 解决方法 打开钥匙串访问 双击那些密钥弹出框 改变到允许所有应用程序访问此项目 输入一次密码即可 解决 不用每次输入了
  • Qt Widgets 与 QML/Qt Quick

    Qt Widgets 更老 更成熟 可以做gui界面 适合传统的桌面程序 QML Qt Quick 开发更 现代 的 UI 与高级应用 渲染效果更佳 界面更炫酷 参考 https my oschina net u 4000302 blog
  • Ant Design 常见用法与坑点总结(一)

    前言 Ant Design 是蚂蚁出品的出色优秀的 React 组件库 相信使用 React 进行管理系统开发的小伙伴们或多或少都接触过 Ant Design 很多公司基于 React 开发的管理端系统也都是使用 Ant Design 的组
  • 运动估计与运动补偿

    运动估计与运动补 偿 改正 2010 11 17 10 29 29 运动补偿是通过先前的局部图像来预测 补偿当前的局部图像 它是减少帧序列冗余信息的有效方法 运动估计是从视频序列中抽取运动信息的一整套技术 运动估计与运动补偿技术 MPEG
  • 基于matlab的长短期神经网络LSTM的电力负荷预测

    目录 背影 摘要 LSTM的基本定义 LSTM实现的步骤 基于长短期神经网络LSTM的电力负荷预测 MATALB代码 效果图 结果分析 展望 参考论文 背影 电力负荷预测的实质是从已知的电力系统 经济 社会 气象等情况出发 根据历史负荷变化
  • Vue入门基础(环境搭建+软件安装+问题解决)

    如何创建一个vue项目 打开cmd 输入如下指令 因人而异自己电脑上就进入想创建的目录 输入vue init webpack 项目名称 接下来打开vscode找到新建的文件夹 package json经过观察可以发现 现在没有配置Vuex
  • HTMLCanvasElement.toDataURL()

    HTMLCanvasElement toDataURL 方法返回一个包含图片展示的 data URI 可以使用 type 参数其类型 默认为 PNG 格式 图片的分辨率为96dpi 如果画布的高度或宽度是0 那么会返回字符串 data 如果
  • 基于FPGA的频率计

    好久没更了 百忙之中写一篇so easy的代码 基于FPGA的频率计设计 一 简介 废话不多说 下面是百度搜索关于频率计的简洁概念 数字频率计是一种基本的测量仪器 被广泛应用于航天 电子 测控等领域 基于传统测频原理的频率计的测量精度将随被
  • 2020最新Ubutu18.04安装NVIDIA出现如下报错解决方法[亲测可用]

    我使用的NVIDIA版本为 NVIDIA Linux x86 64 455 28 run 以下只是报错过程中尝试解决的方法 前提条件是你自身的机器具有nvidia显卡 如需正确安装nvidia 请参考这篇文章https blog csdn
  • MYSQL基础管理-auto_increment测试应用

    mysql gt create table tid id int not null auto increment name varchar 100 prima ry key id Query OK 0 rows affected 0 11
  • Flask 和 Bootstrap 应用控制 WiFi 模块温度控制

    根据室温 控制风扇 或者程序会为您完成 并在温度过高时发送短信 所需材料 简述 该项目具有三个主要功能 打开风扇 关闭风扇 自动模式 这些功能由 3 个按钮控制 这些按钮是使用 Flask 和 Html 以及 Bootstrap 设计的 分
  • 6 * 4 = 24从上面的乘法算式可以看出,我们可以把24只羊分的不同的组合方式,而且这些组合方式是通过乘法来计算出来的。因此,我们可以把乘法组合用于羊群分组的问题中。通过本文,可以看出,乘法组合是...

    有一个叫做安妮的小女孩 她有一群24只羊 安妮经常做这样的家务 把这些羊分成不同的组合 以便它们更好地享受活动 但是 安妮发现 把羊分成组合以后 有时会出现混乱的情况 她不知道如何把它们分成不同的组合 这时 安妮的爸爸想起了乘法组合 他给安
  • Linux内核实时线程throtted机制

    Linux内核支持实时线程和非实时线程同时工作 并且可以给实时线程分配适当的执行比例 当超过设置的比例的时候 调度器将不再调度实时线程工作 这样做保证了即便在实时线程不主动放弃CPU的情况下 CPU的占用率也不会到 100 保证了其它线程总
  • 【JavaSe】高级特性篇(八) Java8新特性函数式接口

    JavaSe 高级特性篇 八 Java8新特性函数式接口 1 函数式接口 1 1 概念 函数式接口在Java中是指 有且仅有一个抽象方法的接口 函数式接口 即适用于函数式编程场景的接口 而Java中的函数式编程体现就是Lambda 所以函数