反射,枚举,lambda表达式
目录
1、反射
1.1 基本概念
1.2 反射相关的类
1.3 创建 Class 对象
1.4 反射的使用
1.4.1 通过反射创建对象:
1.4.2 获取私有的构造方法
1.4.3 获取私有的成员变量
1.4.4 获取私有的方法
1.5 总结
2、枚举
2.1 认识枚举
2.2 使用枚举
2.3 枚举与反射的那些事
3、Lambda 表达式
3.1 认识 Lambda 表达式
3.2 语法
3.3 函数式接口
3.4 Lambda 的基本使用
1、反射
1.1 基本概念
Java的反射(reflection)机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性,既然能拿到,那么我们就可以修改部分类型信息;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射(reflection)机制。
1.2 反射相关的类
类名 | 用途 |
---|---|
Class 类 | 代表类的实体,在运行的Java程序中表示类和接口 |
Field 类 | 代表类的成员变量/类的属性 |
Method 类 | 代表类的方法 |
Constructor 类 | 代表类的构造方法 |
每个类里面都有很多相关的方法啊,这里具体的方法我就不一一列举出来了,详细的可以去查看 Java 的官方文档。
1.3 创建 Class 对象
创建一个 Class 对象,通常使用以下三种方法:
1. 调用某个对象里面的 getClass 方法:
public static void main(String[] args) {Student student = new Student();Class<?> c1 = student.getClass();
}
2. 采取类名.class的方法:
public static void main(String[] args) {Class<?> c2 = Student.class; //这种方法说明每个类默认隐式包含一个静态的成员变量 class
}
3. 通过 Class.forName() 获取:
public static void main(String[] args) {Class<?> c3 = null;try {c3 = Class.forName("Student");} catch (ClassNotFoundException e) {e.printStackTrace();}
}
注意:这个 forName 中要放传入类的完整路径,比如如果是 String 的话,即:java.lang.String
1.4 反射的使用
这里我们自定义一个 Student 类:
public class Student {private String name;private int age;public Student() {System.out.println();}private Student(String name, int age) {this.name = name;this.age = age;}public void eat() {System.out.println(name + "正在吃饭!");}private void sleep() {System.out.println(name + "正在睡觉!");}@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", age=" + age +'}';}
}
1.4.1 通过反射创建对象:
public static void reflectClassDemo() {try {Class<?> c = Class.forName("Student");Object objectStudent = c.newInstance();} catch (ClassNotFoundException | IllegalAccessException | InstantiationException e) {e.printStackTrace();}
}
1.4.2 获取私有的构造方法
public static void reflectPrivateConstructor() {try {Class<?> c = Class.forName("Student");Constructor<?> constructor = c.getDeclaredConstructor(String.class, int.class);constructor.setAccessible(true); //设置为true后可修改访问权限Object objectStudent = constructor.newInstance("张三", 12);System.out.println(objectStudent);} catch (ClassNotFoundException | NoSuchMethodException | InvocationTargetException | IllegalAccessException | InstantiationException e) {e.printStackTrace();}
}
1.4.3 获取私有的成员变量
public static void reflectPrivateField() {try {Class<?> c = Class.forName("Student");Field field = c.getDeclaredField("name"); //获取名为 name 的成员变量field.setAccessible(true);Object student = c.newInstance();field.set(student, "张三"); //将 student 对象的name 设置成 "张三"System.out.println(field.getName());} catch (ClassNotFoundException | NoSuchFieldException | IllegalAccessException | InstantiationException e) {e.printStackTrace();}
}
1.4.4 获取私有的方法
public static void reflectPrivateMethod() {try {Class<?> c = Class.forName("Student");Method method = c.getDeclaredMethod("eat");method.setAccessible(true);//方法有参数的写法://Method methodStudent = classStudent.getDeclaredMethod("function",String.class);Object objectStudent = c.newInstance();Student student = (Student)objectStudent;System.out.println("私有方法方法名: " + method.getName());method.invoke(objectStudent); // 调用获取到的方法, student对象中的} catch (InstantiationException | InvocationTargetException | NoSuchMethodException | IllegalAccessException | ClassNotFoundException e) {e.printStackTrace();}
}
1.5 总结
在反射眼中,对应任意一个类,都能够知道这个类的属性和方法,对于任意一个对象,都能调用它任意一个方法, 这样使程序的灵活性大大提高,以及可扩展性,但是这样一来,似乎就在告诉大家,之前封装的一些方法和属性在反射面前就是一个摆设。
反射是一把双刃剑,是一种非常规的编程手段,不到必要的时候,不建议使用反射,使用反射会有效率问题。会导致程序效率降低,而且反射技术绕过了源代码的技术,因而会带来维护问题。反射代码比相应的直接代码更复杂 。
2、枚举
2.1 认识枚举
枚举顾名思义,一一列举,作用将一组常量组织起来,Java中常量就是常量,并没有常变量这种说法,利用我们现在的知识,如果让你定义三个常量,分别表示 红色,黑色,绿色,你可能会这样定义:
public static final int RED = 1;
public static final int BLACK = 3;
public static final int GREEN = 3;
这样的话,代码中出现的 1,有可能会被误认为是 RED,是可能会出现歧义的,于是就有一种类型枚举来进行组织:
public enum MyEnum {RED, BLACK, GREEN;
}
这样里面定义的每个都是枚举类型,不再是普通的数字了,我们自己写的 enum 会默认继承 Enum 类,所以不用显示的去继承 Enum 这个抽象类。
2.2 使用枚举
public static void main(String[] args) {MyEnum myEnum = MyEnum.BLACK;switch (myEnum) {case RED:System.out.println("红色!");break;case BLACK:System.out.println("黑色!");break;case GREEN:System.out.println("绿色!");break;default:System.out.println("其他颜色!");break;}
}
使用场景:错误状态码,消息类型,颜色的划分,状态机等等....
Enum 类的常用方法:
方法名称 | 描述 |
---|---|
values() | 以数组的形式返回枚举类型的所有成员 |
ordinal() | 获取枚举成员的索引位置 |
valueOf() | 将普通字符串转换为枚举类型 |
compareTo() | 比较两个枚举成员在定义时候的顺序 |
枚举是一种类型,也就是Java中的枚举就是一个类,那么就可以这样去写代码:
public enum MyEnum {RED("红色", 1), BLACK("黑色", 2), GREEN("绿色", 3);private String name;private int key;private MyEnum(String name, int key) {this.name = name;this.key = key;}
}
枚举的构造方法默认是私有的.
枚举常量是更简单安全的,安全体现在哪,马上就说到了,而且枚举拥有内置方法,使用起来方便,缺点也有,由于Java中支持单继承,因此枚举类型不能再继承其他类,无法扩展。
2.3 枚举与反射的那些事
通过反射,能否拿到枚举的私有构造方法呢?我们写个代码来测试一下:
public class TestMyEnum {public static void main(String[] args) {try {Class<?> c = Class.forName("MyEnum");Constructor<?> constructor = c.getDeclaredConstructor(String.class, int.class);constructor.setAccessible(true);Object myEnum = constructor.newInstance("红色", 123);} catch (Exception ex) {ex.printStackTrace();}}
}
这里居然报错了,报错信息提示没有该构造方法???为什么会没有呢,我们的 MyEnum 默认继承了 Enum 类,实例化子类对象的时候,先调用父类的构造方法,那么这里我们就去看一下 Enum 的构造方法。
Enum 只有这一个构造方法,还带有两个参数,但是在 JavaSE 的学习中,如果在子类的构造方法中,没有显式写明 super(),则会在子类构造方法第一行默认有 super(),也就是调用父类的无参构造,枚举比较特殊虽然我们写的是两个,但默认他还添加了 name,和 ordinal 参数。也就是说,我们需要提供四个参数:
Constructor<?> constructor = c.getDeclaredConstructor(String.class, int.class, String.class, int.class);
constructor.setAccessible(true);
Object myEnum = constructor.newInstance("红色", 123, "红色", 321);
这里还是报错了,但是这里的报错是第10行的,newInstance 方法报错,那么我们就进入该方法源码去一看究竟:
在JavaSE语法上,if 中的 & 会被认为 &&,所以枚举在这里被过滤了,这也就是你不能通过反射获取到枚举类的实例!
所以在这里可以发现,枚举是安全的,可以避免反射的问题。
3、Lambda 表达式
3.1 认识 Lambda 表达式
Lambda表达式是Java SE 8中一个重要的新特性。lambda表达式允许你通过表达式来代替功能接口。 lambda表达式就和方法一样,它提供了一个正常的参数列表和一个使用这些参数的主体(body,可以是一个表达式或一个代码 块)。 Lambda 表达式(Lambda expression),基于数学中的 λ 演算得名,也可称为闭包(Closure)。
3.2 语法
基本语法: (parameters) -> expression 或 (parameters) -> { statements; }
Lambda表达式由三部分组成:
- 1. paramaters:类似方法中的形参列表,这里的参数是函数式接口里的参数。这里的参数类型可以明确的声明 也可不声明而由JVM隐含的推断。另外当只有一个推断类型时可以省略掉圆括号。 2. ->:可理解为“被用于”的意思
- 3. 方法体:可以是表达式也可以代码块,是函数式接口里方法的实现。代码块可返回一个值或者什么都不反 回,这里的代码块块等同于方法的方法体。如果是表达式,也可以返回一个值或者什么都不反回。
3.3 函数式接口
函数式接口:一个接口中,只有一个抽象方法。
@FunctionalInterface 注解:如果我们在某个方法上声明了该注解,那么编译器就会按照函数式接口的定义来要求该接口。如果不符合函数式接口的语法,那么则会报错!
例子:
@FunctionalInterface
public interface TestFuncInterface {void work();
}
也可也这样写:
@FunctionalInterface
public interface TestFuncInterface {void work();default void test() {System.out.println("hello");}
}
在 JDK 1.8 中,default 默认方法可以有具体的实现。
3.4 Lambda 的基本使用
//无返回值无参数
@FunctionalInterface
interface NoParameterNoReturn {void work();
}
//无返回值一个参数
@FunctionalInterface
interface OneParameterNoReturn {void work(int a);
}
//无返回值多个参数
@FunctionalInterface
interface MoreParameterNoReturn {void work(int a,int b);
}
//有返回值无参数
@FunctionalInterface
interface NoParameterReturn {int work();
}//有返回值一个参数
@FunctionalInterface
interface OneParameterReturn {int work(int a);
}
//有返回值多参数
@FunctionalInterface
interface MoreParameterReturn {int work(int a,int b);
}public class TestLambda {public static void main(String[] args) {NoParameterNoReturn n1 = () -> System.out.println("NoParameterNoReturn");n1.work();// 只有一个参数, 可以省略小括号OneParameterNoReturn n2 = x -> {System.out.println(x + "OneParameterNoReturn");System.out.println("多条语句则不能省略大括号!!!");};n2.work(5);MoreParameterNoReturn n3 = (x, y) -> System.out.println(x + y + "OneParameterNoReturn");n3.work(5, 8);NoParameterReturn n4 = () -> 88; //只有 return 一条语句可以省略 returnint ret1 = n4.work();OneParameterReturn n5 = (x) -> {System.out.println("OneParameterReturn");return x + 10; //多条语句时, return 和 {} 都不能省略};int ret2 = n5.work(12);// 如果不省略形参类型, 必须都不省略, 省略的话必须全部省略MoreParameterReturn n6 = (int x, int y) -> (x + y + 8);int ret3 = n6.work(5, 5);}
}
Lambda表达式的优点很明显,在代码层次上来说,使代码变得非常的简洁。缺点也很明显,代码不易读。至于 Lambda 的更多使用,会在后续文章中慢慢体现出来。这里我们了解下语法即可。
优点:
- 代码简洁,开发迅速
- 方便函数式编程
- 非常容易进行并行计算
- ava 引入 Lambda,改善了集合操作(比如传比较器)
缺点:
- 代码可读性变差
- 在非并行计算中,很多计算未必有传统的 for 性能要高
- 不容易进行调试
下期预告:【MySQL】数据库的基本认识
相关文章:

反射,枚举,lambda表达式
目录 1、反射 1.1 基本概念 1.2 反射相关的类 1.3 创建 Class 对象 1.4 反射的使用 1.4.1 通过反射创建对象: 1.4.2 获取私有的构造方法 1.4.3 获取私有的成员变量 1.4.4 获取私有的方法 1.5 总结 2、枚举 2.1 认识枚举 2.2 使用枚举 2.3 枚举与反射…...

.Net Core对于RabbitMQ封装分布式事件总线
首先我们需要了解到分布式事件总线是什么; 分布式事件总线是一种在分布式系统中提供事件通知、订阅和发布机制的技术。它允许多个组件或微服务之间的协作和通信,而无需直接耦合或了解彼此的实现细节。通过事件总线,组件或微服务可以通过发布…...
GPIO功能描述
GPIO 文章目录 GPIO1. 功能描述1.1 OSCI/OSCO 引脚1.3 HSEIN/HSEOUT引脚1.2 Bit-Band1.4 VRTCAFx引脚1.5 EWKUPx引脚1.6 QSPI0 引脚1.7 LVDIN引脚1.8 SARADC引脚1.9 ADCIN引脚2. 测试项描述2.1 PAD Location2.2 LBOR和BOR复位2.3 驱动能力2.4 模拟态\高阻态2.5 SWD\JTAG2.6 输出…...

指派问题与匈牙利法讲解
指派问题概述:实际中,会遇到这样的问题,有n项不同的任务,需要n个人分别完成其中的1项,每个人完成任务的时间不一样。于是就有一个问题,如何分配任务使得花费时间最少。通俗来讲,就是n*n矩阵中&a…...

day5——冒泡排序,选择排序和插入排序的学习
选择排序冒泡排序插入排序 选择排序 选择排序的基本思路就是: 首先假定第一个的下表为所有元素中最小的一个, 然后用后面的每一个元素跟这个元素进行比较, 如果后面的元素比这个元素更小一点, 那么就将找到的最小的元素的下标和…...
Windows 数据类型 (Windows Data Types)
参考:https://learn.microsoft.com/en-us/windows/win32/winprog/windows-data-types 要求 要求值最低受支持的客户端Windows XP [仅限桌面应用]最低受支持的服务器Windows Server 2003 [仅限桌面应用]HeaderBaseTsd.h;WinDef.h;WinNT.hAPIENTRY 系统函数的调用约…...

九龙证券|本周5只新股申购,特斯拉、蔚来、理想的供应商来A股了!
据现在组织,2月13日到17日共有5只新股申购,其间上证主板2只,深证主板1只,北交所2只。 2月14日发动打新的深证主板新股多利科技成立于2010年,是一家专心于轿车冲压零部件及相关模具的开发、出产与出售的企业。从2020年…...
设计模式(持续更新)
本文主要是记录java的设计模式在实际工作中的应用案例,或者是对设计模式的个人理解及备忘 一、单例模式Singleton 工作场景(静态类): 在外部系统对接中,需要调用外部系统A的接口,但是接口是有身份校验的…...

Prometheus 告警规则
Prometheus 告警规则 Prometheus官方内置的第三方报警通知包括:邮件、 即时通讯软件(如Slack、Hipchat)、移动应用消息推送(如Pushover)和自动化运维工具(例如:Pagerduty、Opsgenie、Victorops) Promethe…...
mulesoft MCIA 破釜沉舟备考 2023.02.13.02
mulesoft MCIA 破釜沉舟备考 2023.02.13.03 1. According to MuleSoft, which deployment charcateristic applies to a microservices application architecture?2. A mule application designed to fulfil two requirements3. A mule application must periodically process…...
获取DLL运行时路径的方法
之前项目中发现的问题,记录下解决方案1. 问题背景OVVRNTool项目中,底层图像基本操作功能由DLL库函数提供,上层基于DLL封装了两个应用CMD和GUI,然后通过Qt打包分发;发布是直接采用绿色免安装的方式打包,具体…...
“华为杯”研究生数学建模竞赛2006年-【华为杯】D题:学生面试中教师安排的优化与算法(附获奖论文)
赛题描述 高校自主招生是高考改革中的一项新生事物,现在仍处于探索阶段。某高校拟在全面衡量考生的高中学习成绩及综合表现后再采用专家面试的方式决定录取与否。该校在今年自主招生中,经过初选合格进入面试的考生有N人,拟聘请老师M人。每位学生要分别接受4位老师(简称该学…...

【JavaScript】复习 【对象参数】【函数参数】
js不会检查任何参数类型,任何参数都可以作为参数传递 1、对象参数 改变量随便改,改对象要看这个对象是不是有多个变量同时指向这个对象 const 用来定义常量,只能赋值一次。 变量------->对象------->属性 被const修饰的对象 …...

如何批量提取文件名到excel表格?
批量提取文件名到excel表格?关于这个问题相信很多人都遇到过,大多数人在第一次碰到的时候都不知道如何下手,大家都会立即在百度里面搜索相关方法教程,小编也试着搜索了一下,发现找到的很多方法都大同小异,需…...

CUDA线程层次一文搞懂|参加CUDA线上训练营
设备术语 Host:CPU 和 内存 (host memory)Device:GPU 和显存 (device memory) CUDA 线程层次 CUDA 线程层次分为: Thread 所有线程执行相同的核函数并行执行 Thread Block 执行在一个 Streaming Multiprocessor (SM)…...

Linux文件默认权限:umask
umask就是指定目前用户在建立文件或目录时候的权限默认值 查看方式有两种:一种可以直接输入umask,就可以看到数字类型的权限设置值,一种则是加入umask后加入-S(Symbolic)选项,就会以符号类型的方式来显示出…...

SonicWall:请立即修复SMA 1000 漏洞
近日,网络安全供应商SonicWall发布了关于安全移动访问 (SMA) 1000设备的三个安全漏洞的紧急报告,其中包括一个高威胁性的身份验证绕过漏洞。SonicWall指出,攻击者可以利用这些漏洞绕过授权,并可能破坏易受攻击的设备。 从报告中可…...

基于VS调试分析 + 堆栈观察问题代码段
文章目录问题代码段1 —— 阶乘之和问题代码段2 —— 越界的危害① 发现问题② 分析问题③ 思考问题【⭐堆栈原理⭐】④ 解决问题【DeBug与Release】👨程序员与测试人员👩✒总结与提炼问题代码段1 —— 阶乘之和 先来看一道C语言中比较基础的题目&#x…...

QFramework框架学习
主要学习内容TypeEventSystemActionKitTimer类1、TypeEventSystem-适用于一个条件触发,多个组件响应的情况例如:动物园系统中,点击肉食动物按钮,动物园中有肉食属性的动物都进行显示。步骤:1、动物自身脚本上进行判断是…...
移动OA系统,联动企业协作让办公高效无间断
移动oa系统,近年来随着企业办公节奏的变化及人们个性化办公需求的增加迎来了快速发展。一方面,它兼具OA系统诸多优势,既凝聚了企业基础管理工作,联动了企业协作、沟通交流,又进一步提高了企业的综合实力与市场竞争力。…...

手游刚开服就被攻击怎么办?如何防御DDoS?
开服初期是手游最脆弱的阶段,极易成为DDoS攻击的目标。一旦遭遇攻击,可能导致服务器瘫痪、玩家流失,甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案,帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...
python爬虫:Newspaper3k 的详细使用(好用的新闻网站文章抓取和解析的Python库)
更多内容请见: 爬虫和逆向教程-专栏介绍和目录 文章目录 一、Newspaper3k 概述1.1 Newspaper3k 介绍1.2 主要功能1.3 典型应用场景1.4 安装二、基本用法2.2 提取单篇文章的内容2.2 处理多篇文档三、高级选项3.1 自定义配置3.2 分析文章情感四、实战案例4.1 构建新闻摘要聚合器…...

从零实现STL哈希容器:unordered_map/unordered_set封装详解
本篇文章是对C学习的STL哈希容器自主实现部分的学习分享 希望也能为你带来些帮助~ 那咱们废话不多说,直接开始吧! 一、源码结构分析 1. SGISTL30实现剖析 // hash_set核心结构 template <class Value, class HashFcn, ...> class hash_set {ty…...
Java入门学习详细版(一)
大家好,Java 学习是一个系统学习的过程,核心原则就是“理论 实践 坚持”,并且需循序渐进,不可过于着急,本篇文章推出的这份详细入门学习资料将带大家从零基础开始,逐步掌握 Java 的核心概念和编程技能。 …...
精益数据分析(97/126):邮件营销与用户参与度的关键指标优化指南
精益数据分析(97/126):邮件营销与用户参与度的关键指标优化指南 在数字化营销时代,邮件列表效度、用户参与度和网站性能等指标往往决定着创业公司的增长成败。今天,我们将深入解析邮件打开率、网站可用性、页面参与时…...

OPenCV CUDA模块图像处理-----对图像执行 均值漂移滤波(Mean Shift Filtering)函数meanShiftFiltering()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 在 GPU 上对图像执行 均值漂移滤波(Mean Shift Filtering),用于图像分割或平滑处理。 该函数将输入图像中的…...

AI书签管理工具开发全记录(十九):嵌入资源处理
1.前言 📝 在上一篇文章中,我们完成了书签的导入导出功能。本篇文章我们研究如何处理嵌入资源,方便后续将资源打包到一个可执行文件中。 2.embed介绍 🎯 Go 1.16 引入了革命性的 embed 包,彻底改变了静态资源管理的…...
今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存
文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...

初探Service服务发现机制
1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能:服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源…...
C++课设:简易日历程序(支持传统节假日 + 二十四节气 + 个人纪念日管理)
名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 专栏介绍:《编程项目实战》 目录 一、为什么要开发一个日历程序?1. 深入理解时间算法2. 练习面向对象设计3. 学习数据结构应用二、核心算法深度解析…...