前后端传参 三种传参方式
x-www-form-urlencoded :只能上传键值对,并且键值对都是间隔分开的
multipart/form-data :既可以上传文件等二进制数据,也可以上传表单键值对,只是最后会转化为一条信息;
x-www-form-urlencoded : Post传参的默认格式,使用js中URLencode转码方法。包括将name、value中的空格替换为加号;将非ascii字符做百分号编码;将input的name、value用‘=’连接,不同的input之间用‘&’连接。这种post格式跟get的区别在于:get把转换、拼接完的字符串用‘?’直接与表单的action连接作为URL使用,所以请求体里没有数据;而post把转换、拼接后的字符串放在了请求体里,不会在浏览器的地址栏显示,因而更安全一些。对于一段utf8编码的字节,用application/x-www-form-urlencoded传输其中的ascii字符没有问题,但对于非ascii字符传输效率就很低了,因此在传很长的字节(如文件)时应用multipart/form-data格式。smtp等协议也使用或借鉴了此格式。
multipart/form-data :将表单中的每个input转为了一个由boundary分割的小格式,没有转码,直接将utf8字节拼接到请求体中,在本地有多少字节实际就发送多少字节,极大提高了效率,适合传输长字节。
json :现在很多时候也把它作为请求头,用来告诉服务端消息主体是序列化的JSON字符串,除了低版本的IE,基本都支持。除了低版本的IE都支持JSON.stringify()的方法,服务端也有处理JSON的函数,使用JSON不会有任何麻烦。
当传入参数是x-www-form-urlencoded,接收参数@RequestBody+ String/Map/LinkedHashMap即可接收成功
post和get请求
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 $.ajax({ url: "/order/userPage?page=" +page+"&pageSize=" +pageSize, cache: "false" , async: "true" , dataType: "json" , type:"get" , }) async initData () { paging: { page: 1 , pageSize: 5 } const res = await orderPagingApi (this .paging) } function orderPagingApi (data) { return $axios({ 'url' : '/order/userPage' , 'method' : 'get' , params: {...data} }) } async initData () { paging: { page: 1 , pageSize: 5 } const res = await orderPagingApi (this .paging) } function orderPagingApi (data) { return $axios({ 'url' : '/order/userPage' , 'method' : 'get' , 'data' : data }) } @GetMapping("/xxx") public R<xxx> userPage (@RequestParam("page") Integer page,Integer pageSize) {}
将参数拼接在url中,后台通过占位符接收参数 /{id},前端传递参数直接将参数拼在url中,如 /xxx/1
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 async initData () { const res = await addressFindOneApi (params.id) } function addressFindOneApi (id) { return $axios({ 'url' : `/addressBook/${id}`, 'method' : 'get' , }) } @GetMapping("/xxx/{id}") public R<xxx> backList (@PathVariable("id") Long id) {}
通过post提交方式将form表单中的数据序列化后传递到后台
1 2 3 4 5 6 7 8 9 10 11 12 13 14 async initData () { const res = await formAjax () ; } function formAjax () { $.ajax({ url: "/login" , type: "post" , data: $("#form" ).serialize(), }) } @RequestMapping("/login") public String login (User user) {}
通过post提交方式将form表单的类型是 json
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 async initData () { const res = await formAjax () ; } function formAjax () { $.ajax({ url: "/login" , type: "post" , contentType: 'application/json' , }) } @RequestMapping("/login") public String login (@RequestBody User user) {}
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 @GetMapping("/xxx/xxx") public R<xxx> userPage (@RequesBody Map<Integer,Integer> map) { Integer page = map.get("page" ); Integer pageSize = map.get("pageSize" ); }async initData () { paging: { page: 1 , pageSize: 5 } const res = await orderPagingApi (this .paging) } function orderPagingApi (data) { return $axios({ 'url' : '/order/userPage' , 'method' : 'post' , data: JSON.stringify(data), }) } @GetMapping("/xxx/xxx") public R<xxx> userPage (@RequesBody Map<Integer,Integer> map) { Integer page = map.get("page" ); Integer pageSize = map.get("pageSize" ); }
1 2 3 4 5 6 @PostMapping("/regiest") public R<String> createUser (@RequestBody Map<String,String> map) {}@PostMapping("/regiest") public R<String> createUser (@RequestBody Map<String,Object> map) {}
前端传参 {params: params} ,后端用 @RequestParams(“id”) 接参
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 this .$axios .delete("login/deleteUserInfo" , { params: { userId: this .id } }) .then( this .$message({ type: "success" , message: "删除用户信息成功" , })) .catch ( this .$message({ type: "false" , message: "删除用户信息失败" , }) @DeleteMapping("xxx") public R deleteUserInfo (@RequestParam("userId") String userId) {}
前端通过 {data : param} 传参,后端通过 @RequestBody 接参
1 2 3 4 5 6 7 8 9 10 11 deleteMessage() { axios.delete('login/deleteUserInfo' , { data : { userId: this .id }}).then((response) => { }); } @DeleteMapping("deleteUserInfo") public R deleteUserInfo (@RequestBody String userId) {}
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 <!-- :value='1' v-model="1"(数值 1)则匹配(number=number string=string类型要一致),显示label,否则显示value--> <el-form-item label="角色名称:" label-width="100px"> <el-select placeholder="请选择角色名称" v-model="form.roleId" :label-width="formLabelWidth" style="width: 200px" > <el-option v-for="item in roleInfo" :key="item.value" :label="item.roleName" :value="item.roleId" > </el-option> </el-select> </el-form-item>
ajax请求 ajax请求规范
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 function test (value ) { axios ({ method :'get' , url :'/xxx' , params : { attributeName :value } }).then ((response ) => { }).catch ((error ) => { }) }
1 2 3 4 5 @GetMapping("/xxx") public void test (@RequestParam(value = "attributeName") String attributeName) {}
路径后面有反斜杠’/ ‘,不要写漏了,不然value就会变成路径的一部分,请求路径就会变成/xxx后面跟上value的值,导致404。
1 2 3 4 5 6 7 8 9 10 11 12 function test (value ) { axios ({ method :'get' , url :'/xxx/' +value, }).then ((response ) => { }).catch ((error ) => { }) }
1 2 3 4 @GetMapping("/xxx/{id}") public void test (@PathVariable("id") String() attributeName){}
除了method的请求方式和接收请求用@PostMapping,其他都和普通的get请求一致
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 function test (value ) { axios ({ method :'post' , url :'/xxx' , params : { attributeName :value } }).then ((response ) => { }).catch ((error ) => { }) }
1 2 3 4 @PostMapping("/xxx") public void test (@RequestParam(value = "attributeName") String attributeName) { }
注意:obj对象里的属性名要和people实体类里的属性名一样,因为obj对象里的属性名与people实体类里的属性名匹配时,大小写敏感,即区分大小写。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 function test (obj ) { axios ({ method :'post' , url :'/xxx' , data : obj }).then ((response ) => { }).catch ((error ) => { }) }
1 2 3 4 5 6 7 8 9 @Data public class People { private String userName; private Integer age; } @PostMapping("/xxx") public void test (@RequestBody People people) {
传参问题
② 把接口的@RequestBody 注解去掉;
Postman传参方式
第一种是使用 raw:
第二种是使用 form-data:
第三种是直接使用变量的方式:
日志 日志框架和日志门面
常见的日志框架
Log4j :Apache Log4j 是一个广泛使用的日志框架,它提供了丰富的配置选项和多种输出目标。
Log4j2 : Apache Log4j 2 是 Apache Software Foundation 开发的日志框架,它是 Log4j 的升级版本,提供了更好的性能和功能。
Logback :Logback 是由 Log4j 的创始人设计的日志框架,旨在取代 Log4j。它在性能和灵活性方面表现出色。
JUL : (Java Util Logging) JUL 是 Java 平台自带的日志框架,位于java.util.logging
包下。它是标准的 Java 日志框架,使用简单,但功能相对较弱。
常见的日志门面
SLF4J: 常见的日志门面,它提供了统一的日志 API,并允许你在项目中使用 SLF4J 的接口,而不用直接依赖于具体的日志实现。
ACL: (Apache Commons Logging)常见的日志门面,类似于 SLF4J,它允许开发者使用通用的 API 来进行日志记录。
注意 :日志门面是一个抽象层,它提供了一组通用的日志 API,可以将这些 API 与不同的日志框架进行绑定。这样,你可以在项目中使用统一的日志 API,而无需直接和特定的日志框架耦合。根据项目需求和个人喜好,可以选择使用不同的日志门面和底层的日志实现。其中,SLF4J 是目前使用最广泛的日志门面之一,并且很多日志框架都支持与 SLF4J 的集成
日志如何使用 两个步骤:引入依赖 和配置日志框架
1 2 3 4 5 6 7 8 9 10 11 12 13 <dependency > <groupId > org.slf4j</groupId > <artifactId > slf4j-api</artifactId > <version > 1.7.32</version > </dependency > <dependency > <groupId > ch.qos.logback</groupId > <artifactId > logback-classic</artifactId > <version > 1.2.6</version > </dependency >
1 2 3 4 5 6 7 8 9 10 11 12 <configuration > <appender name ="CONSOLE" class ="ch.qos.logback.core.ConsoleAppender" > <encoder > <pattern > %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern > </encoder > </appender > <root level ="debug" > <appender-ref ref ="CONSOLE" /> </root > </configuration >
上述配置定义了一个输出到控制台的 Appender,并将根日志记录器的级别设置为 debug
,这意味着会记录所有级别的日志信息。
一旦你完成了依赖的引入和日志框架的配置,你就可以在项目中使用日志门面的 API(例如 SLF4J 提供的 Logger
接口)来记录日志。根据日志门面的 API 设计,你可以使用不同的日志级别(如 debug
、info
、warn
、error
等)和相应的日志方法来记录不同级别的日志信息。
例如,在代码中使用 SLF4J 来记录日志:当项目运行时,具体的日志输出将由你在配置文件中指定的 Appender 和日志级别来决定。
1 2 3 4 5 6 7 8 9 10 11 12 13 import org.slf4j.Logger;import org.slf4j.LoggerFactory;public class MyClass { private static final Logger logger = LoggerFactory.getLogger(MyClass.class); public void doSomething () { logger.debug("This is a debug message." ); logger.info("This is an info message." ); logger.warn("This is a warning message." ); logger.error("This is an error message." ); } }
在项目中集成日志门面和日志框架,首先引入相应的依赖(日志门面和具体日志框架的实现),然后配置日志框架以指定日志输出的方式和级别。在代码中,你可以使用日志门面的 API 来记录日志。这样可以在不改变代码的情况下切换不同的日志框架,以满足项目的日志需求
异常 try-with-resource语法
后续补充细节…
网络编程 Socket
:套接字,是计算机网络编程中的一个核心概念,它提供了一种跨网络进程间通信(IPC, Inter-Process Communication)的机制。Socket允许两个不同计算机上的应用程序通过网络进行数据交换,就像两个程序在同一条线上“对话”一样。这一概念广泛应用于互联网通信、分布式系统以及各种客户端-服务器模型的程序中。
Socket的基本原理 Socket基于客户端-服务器模型工作,主要涉及以下几个关键点:
地址家族:指定通信的地址类型,如IPv4、IPv6等。
套接字类型:主要有流式套接字(Stream Socket,对应TCP协议)和数据报套接字(Datagram Socket,对应UDP协议)两种。流式套接字提供双向的、有序的、无重复的数据流服务,保证数据的可靠传输;而数据报套接字则是无连接的,不保证数据的顺序和可靠性,但传输效率较高。
通信协议:最常用的是 TCP
(Transmission Control Protocol,传输控制协议)和 UDP
(User Datagram Protocol,用户数据报协议)。
Socket的典型用途
Web服务:HTTP服务器与浏览器之间的通信通常基于TCP Socket。
文件传输:FTP、某些类型的下载和上传服务依赖TCP Socket保证数据的完整传输。
即时通讯:如聊天应用,可能使用TCP Socket保持稳定的连接或使用UDP Socket实现实时性较高的消息传递。
在线游戏:游戏服务器与客户端间的实时交互,可能结合TCP和UDP,TCP用于确保重要指令的可靠传输,UDP用于低延迟的游戏状态更新。
分布式计算:在分布式系统中,不同节点间的数据交换可能通过Socket实现。
使用流程
创建Socket:客户端和服务器端都需要创建Socket实例。
绑定与监听:服务器端Socket需要绑定到一个IP地址和端口号上,并开始监听来自客户端的连接请求。
连接:客户端Socket尝试连接到服务器的指定地址和端口。
数据交换:连接建立后,双方可以通过输入输出流进行数据的发送和接收。
关闭Socket:数据交换完毕后,关闭Socket释放资源。
Socket编程为开发者提供了灵活的网络通信能力,是构建网络应用的基础。
集合 List 1 2 3 4 5 List list = new ArrayList ()ArrayList list = new ArrayList ()List list = new ArrayList ();
java是面向对象编程,面向对象一个重要的原则就是“依赖倒置原则”。依赖抽象(接口),而非具体(实现类)
List是接口,ArrayList是实现类,它允许list可以轻松地在接口的不同实现之间切换
List的实现类包括List,Vector,LinkedList , Stack等等,使用List list = new ArrayList()
将来如果需要改成线程安全的Vector
,只需要直接把ArrayList
改成Vector
就行了。即List list = new Vector()
这种方法实现解耦合,大大提高代码使用的灵活性和通用性,并且在暴露公共接口的时候尤其有用
Map Map是java中的接口,Map.Entry是Map的一个内部接口。常用方法:keySet()
和 entrySet()
keySet()方法 :返回值是Map中key值的集合
entrySet()方法 :返回值是一个Set集合,此集合的类型为Map.Entry
,它是Map声明的一个内部接口,此接口为泛型,定义为Entry<K,V>
。它表示 Map中的一个实体即一个key-value
对。接口中有 getKey()
,getValue()
等方法
1 2 3 4 Map<String, String> map = new HashMap <String, String>(); map.put(“key1”, “value1”); map.put(“key2”, “value2”); map.put(“key3”, “value3”);
第一种:普遍使用,二次取值。即通过 Map.keySet()
遍历key和value
1 2 3 for (String key : map.keySet()) { System.out.println("key= " + key + " and value= " + map.get(key)); }
第二种:通过 Map.entrySet
使用 iterator
遍历key和value
1 2 3 4 5 Iterator<Map.Entry<String, String>> it = map.entrySet().iterator(); while (it.hasNext()) { Map.Entry<String, String> entry = it.next(); System.out.println("key= " + entry.getKey() + " and value= " + entry.getValue()); }
第三种:推荐,尤其是容量大时,即通过 Map.entrySet
遍历key和value
1 2 3 for (Map.Entry<String, String> entry : map.entrySet()) { System.out.println("key= " + entry.getKey() + " and value= " + entry.getValue()); }
第四种:通过Map.values()遍历所有的value,但不能遍历key
1 2 3 for (String v : map.values()) { System.out.println("value= " + v); }
Iterator和Iterable的区别
为什么Collection接口要继承于Iterable接口,而不是Iterator接口?
在jdk 1.5以后,引入了 Iterable
,使用 foreach
语句必须使用 Iterable
类
Java设计者让 Collection 继承于 Iterable
而不是 Iterator
接口
Iterable
的子类是 Collection,Collection 的子类是List,Set等,这些是数据结构或者说放数据的地方
Iterator
是定义了迭代逻辑的对象,让迭代逻辑和数据结构分离开来,这样的好处是可以在一种数据结构上实现多种迭代逻辑
更重要的一点是:每一次调用 Iterable
的 Iterator()
方法,都会返回一个从头开始的 Iterator
对象,各个 Iterator
对象之间不会相互干扰,这样保证了可以同时对一个数据结构进行多个遍历。这是因为每个循环都是用了独立的迭代器 Iterator
对象
Stream流
1 list.stream().map(条件).collect(Collectors.toList());
1 list.stream().foreach(条件).collect(Collectors.toList());
1 list.stream().collect(Collectors.groupingBy(条件));
1 list.stream().sorted(条件).collect(Collectors.toList());
1 list.stream().filter(条件).collect(Collectors.toList());
1 list.stream().reduce(条件)
1 2 list.stream().max(Comparator.comparing(条件)); list.stream().min(Comparator.comparing(条件));
1 2 list.stream().filter(条件).findFirst(); list.stream().filter(条件).findAny();
1 list.stream().filter(条件).count();
1 list.stream().skip(m).collect(Collectors.toList());
1 list.stream().limit(n).collect(Collectors.toList());
1 list.stream().distinct().collect(Collectors.toList());
合并的stream类型必须相同,只能两两合并
1 Stream.concat(stream1, stream2).distinct().collect(Collectors.toList());
1 stream().map().collect(Collectors.joining("," ));
语法 正则表达式
匹配:使用matches()
方法来判断一个字符串是否与正则表达式匹配。例如:
1 2 3 String regex = "a*b" ;String input = "aab" ;boolean isMatch = input.matches(regex);
搜索与替换:使用Pattern
和Matcher
类来进行搜索和替换操作。例如:
1 2 3 4 5 6 String regex = "\\bcat\\b" ;String input = "I have a cat and a dog." ;Pattern pattern = Pattern.compile(regex);Matcher matcher = pattern.matcher(input);boolean found = matcher.find(); String replacedInput = matcher.replaceAll("dog" );
分割字符串:使用split()
方法来根据正则表达式将字符串分割成数组。例如:
1 2 3 String regex = "\\s+" ; String input = "Hello World" ;String[] parts = input.split(regex);
提取内容:使用分组(Grouping)来提取正则表达式中的特定部分。例如:
1 2 3 4 5 6 7 8 9 10 String regex = "(\\d{4})-(\\d{2})-(\\d{2})" ; String input = "Today is 2021-09-30" ;Pattern pattern = Pattern.compile(regex);Matcher matcher = pattern.matcher(input);if (matcher.find()) { String year = matcher.group(1 ); String month = matcher.group(2 ); String day = matcher.group(3 ); String group = matcher,group(); }
Java中的反斜杠 \
在字符串中需要进行转义,因此正则表达式中的反斜杠需要使用两个反斜杠 \\
进行表示。
1 2 3 4 5 6 7 8 9 10 * :匹配前一个元素零次或多次。 . :匹配任意单个字符(除了换行符 \n)。 + :匹配前一个元素一次或多次。 ? :匹配前一个元素零次或一次。 \b :当它是由单词边界包围时才进行匹配 如上例 \s :用于匹配任意连续的空白字符。包括空格、制表符和换行符 \d* :匹配任意数量的数字字符(0 次或多次)。 \. :匹配实际的点号字符。 \d+ :匹配至少一个数字字符。 \? :匹配实际的问号字符。
字符串处理
1 StringUtils.strip(collect.toString(), "[]" )
数字和字母转换
For inputString “A”错误
字符串截取操作
1 2 3 4 5 String code = "A" ;int a1 = (int )'A' int a2 = (int )code.toUpperCase().charAt(0 )
转换对象
1 BeanUtils.copyProperties(item,voItem);
1 2 3 4 BeanUtils.copytoList(list1,list2); list2<b> = BeanUtils.copytoList(list1,b.class);
遍历
1 2 3 list.stream().map((item)->{ return item; }).collect(Collectors.toList());
list.foreach()
和 list.stream.foreach()
1 2 3 4 5 list.forEach(item->{ }); lsit.stream.foreach(item->{ });
1 2 3 4 Iterator<xxx> iterator = xxx.iterator(); while (iterator.hasNext()) { xxx xxx = iterator.next(); }
运算符
1 2 3 int i = 1 ;int j = ++i + i++ + ++i + ++i + i++;System.out.println(j);
1 2 3 4 5 6 7 8 9 10 11 12 public class HelloWorld { public static void main (String[] args) { int i = 2 ; System.out.println( i== 1 & i++ ==2 ); System.out.println(i); int j = 2 ; System.out.println( j== 1 && j++ ==2 ); System.out.println(j); } }
Swagger
Api注解不是Spring自带的,他是swagger里面的,代码编写的时候需要在pom文件中引入相关swagger的依赖.
1 2 3 4 5 6 7 8 9 10 <dependency > <groupId > io.springfox</groupId > <artifactId > springfox-swagger2</artifactId > <version > 2.2.2</version > </dependency > <dependency > <groupId > io.springfox</groupId > <artifactId > springfox-swagger-ui</artifactId > <version > 2.2.2</version > </dependency >
常用注解
1 2 3 4 @ApiResponses({ @ApiResponse(code = 0, message = "请求成功"), @ApiResponse(code = 1, message = "请求失败") })
@ApiModelProperty:用在属性上,描述响应类的属性
value-字段说明;
name-重写属性名字;
dataType-重写属性类型;
required-是否必填;
example-举例说明;
hidden-隐藏
@ApiImplicitParams:用在请求的方法上,表示一组参数说明
@ApiImplicitParam:用在@ApiImplicitParams注解中,指定一个请求参数的各个方面
name:参数名;value:参数的汉字说明、解释;required:参数是否必须传
paramType:参数放在哪个地方
header –> 请求参数的获取:@RequestHeader
query –> 请求参数的获取:@RequestParam
path(用于restful接口)–> 请求参数的获取:@PathVariable
body(不常用)
form(不常用)
dataType:参数类型,默认String,其它值dataType=”Integer”
defaultValue:参数的默认值
@ApiModel:用于响应类上,表示一个返回响应数据的信息。这种一般用在post创建的时候,使用@RequestBody这样的场景,请求参数无法使用@ApiImplicitParam注解进行描述的时候
ObjectUtil的 isNull和 isEmpty isNull(Object obj)
方法:
该方法用于判断对象是否为 null
。
如果传入的对象 obj
是 null
或者等于 null
(通过调用 equals
方法进行判断),则返回 true
,否则返回 false
。
这个方法只关注对象是否为 null
,不考虑对象的具体内容。
isEmpty(Object obj)
方法:
该方法用于判断对象是否为空。
如果传入的对象 obj
是 null
,则返回 true
。
如果传入的对象是 CharSequence
(如字符串)、Map
(如Map集合)、Iterable
(如List、Set等可迭代对象)或Iterator
(迭代器),则会调用对应的工具类方法来判断对象是否为空。比如,如果是字符串类型,则使用 StrUtil.isEmpty
方法来判断字符串是否为空。
如果传入的对象是数组类型,则使用 ArrayUtil.isEmpty
方法来判断数组是否为空。
如果传入的对象不属于上述类型,那么返回 false
,表示对象不为空。
注意:isEmpty()
方法会针对不同类型的对象调用相应的工具类方法来判断对象是否为空;isNull()
方法只关注对象是否为 null
Jsoup Jsoup官方文档
解析HTML
Jsoup.parse(String html)
: 将HTML字符串解析成一个Document对象。
Jsoup.parse(File in, String charsetName)
: 从文件中解析HTML文档。
Jsoup.parse(URL url, int timeoutMillis)
: 从URL中解析HTML文档。
遍历和选择元素:
getElementsByTag(String tagName)
: 选择指定标签名的元素,返回一个Elements对象。
getElementById(String id)
: 选择具有指定id属性的元素,返回一个Element对象。
getElementsByClass(String className)
: 选择具有指定CSS类名的元素,返回一个Elements对象。
Document.select(String cssQuery)
: 使用CSS选择器选择元素,返回一个Elements对象。
Element.select(String cssQuery)
: 在当前元素的子元素中使用CSS选择器选择元素,返回一个Elements对象。
Elements.first()
: 返回Elements集合中的第一个元素。
Elements.last()
: 返回Elements集合中的最后一个元素。
Elements.get(int index)
: 返回Elements集合中指定索引位置的元素。
Elements.hasAttr(String attributeKey)
: 选择具有指定属性的元素。
Elements.not(String cssQuery)
: 排除符合指定选择器的元素。
Element.traverse(NodeVisitor nodeVisitor)
: 使用自定义的NodeVisitor遍历元素及其子元素。
Element.wrap(String html)
: 将元素用指定的HTML包装起来。
Element.unwrap()
: 移除元素的父元素,保留元素本身。
过滤元素
Elements.filter(NodeFilter nodeFilter)
: 使用自定义的NodeFilter过滤元素。
操作元素
Element.tagName()
: 获取元素的标签名。
Element.text()
: 获取元素的文本内容。
Element.html()
: 获取元素的HTML内容。
Element.attr(String attributeKey)
: 获取元素指定属性的值。
Element.attr(String attributeKey, String attributeValue)
: 设置元素指定属性的值。
Element.append(String html)
: 在元素的末尾添加HTML内容。
Element.prepend(String html)
: 在元素的开头添加HTML内容。
Element.after(String html)
: 在元素之后插入HTML内容。
Element.before(String html)
: 在元素之前插入HTML内容。
Element.remove()
: 移除元素。
提取和修改HTML文档结构
Document.body()
: 获取HTML文档的<body>
元素。
Element.parent()
: 获取元素的父元素。
Element.children()
: 获取元素的直接子元素。
Element.clone()
: 克隆元素及其子元素。
Element.replaceWith(Node replacement)
: 用指定的节点替换元素。
Element.replaceWith(String html)
: 用指定的HTML内容替换元素。
操作CSS类
Element.hasClass(String className)
: 检查元素是否具有指定的CSS类。
Element.addClass(String className)
: 为元素添加CSS类。
Element.removeClass(String className)
: 从元素中移除指定的CSS类。
处理表单数据
FormElement.formData()
: 获取表单元素中的所有表单数据。
获取和设置表单数据
Element.val()
: 获取或设置元素的值。
Element.val(String value)
: 设置元素的值。
处理相对和绝对URL
Element.absUrl(String attributeKey)
: 获取元素指定属性的绝对URL。
处理特殊字符和编码
Entities.escape(String string)
: 对HTML文本进行转义。
Entities.unescape(String string)
: 解码HTML文本中的特殊字符。
Java数据脱敏 ① SQL数据脱敏实现 MYSQL(电话号码,身份证)数据脱敏的实现
1 2 3 4 5 6 7 8 SELECT mobilePhone AS 脱敏前电话号码,CONCAT(LEFT (mobilePhone,3 ), '********' ) AS 脱敏后电话号码 FROM t_s_userSELECT idcard AS 未脱敏身份证, CONCAT(LEFT (idcard,3 ), '****' ,RIGHT (idcard,4 )) AS 脱敏后身份证号 FROM t_s_user
② JAVA数据脱敏 可参考:sensitive-plus
数据脱敏插件,目前支持地址脱敏、银行卡号脱敏、中文姓名脱敏、固话脱敏、身份证号脱敏、手机号脱敏、密码脱敏 一个是正则脱敏、另外一个根据显示长度脱敏,默认是正则脱敏,可以根据自己的需要配置自己的规则。
③ mybatis-mate-sensitive-jackson mybatis-mate-sensitive-jackson
根据定义的策略类型,对数据进行脱敏,当然策略可以自定义。
BeanUtils.copyProperties的坑 我们日常开发中,经常涉及到DO、DTO、VO
对象属性拷贝赋值,很容易想到org.springframework.beans.BeanUtils
的copyProperties
。它会自动通过反射机制获取源对象和目标对象的属性,并将对应的属性值进行复制 。可以减少手动编写属性复制代码的工作量,提高代码的可读性和维护性。
第1个坑:类型不匹配 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 @Data public class SourceBean { private Long age; } @Data public class TargetBean { private String age; } public class Test { public static void main (String[] args) { SourceBean source = new SourceBean (); source.setAge(25L ); TargetBean target = new TargetBean (); BeanUtils.copyProperties(source, target); System.out.println(target.getAge()); } }
源对象SourceBean
的age
属性是Long
类型,目标对象TargetBean
的age
属性是String
类型。由于类型不匹配,BeanUtils.copyProperties
赋值失败
第2个坑:浅拷贝 什么是深拷贝?什么是浅拷贝?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 public class Address { private String city; } public class Person { private String name; private Address address; } Person sourcePerson = new Person (); sourcePerson.setName("John" ); Address address = new Address (); address.setCity("New York" ); sourcePerson.setAddress(address); Person targetPerson = new Person (); BeanUtils.copyProperties(sourcePerson, targetPerson); sourcePerson.getAddress().setCity("London" ); System.out.println(targetPerson.getAddress().getCity());
源对象Person
的属性address
是引用类型。使用BeanUtils.copyProperties
方法进行属性复制时,只复制了引用。即目标对象targetPerson
的 address
属性引用和源对象 sourcePerson
的 address
属性引用指向同一个对象。因此,当修改源对象的address
对象时,目标对象的address
对象也会被修改。
第3个坑:属性名称不一致 1 2 3 4 5 6 7 8 9 10 11 12 13 public class SourceBean { private String username; } public class TargetBean { private String userName; } SourceBean source = new SourceBean (); source.setUsername("捡田螺的小男孩" ); TargetBean target = new TargetBean (); BeanUtils.copyProperties(source, target); System.out.println(target.getUserName());
源对象SourceBean
的属性名称是username
,而目标对象TargetBean
的属性名称也是userName
。但是,两个 username
,一个N是大写,一个n是小写,即属性名称不一致 ,BeanUtils.copyProperties
方法无法自动映射这些属性(无法忽略大小写自动匹配),因此目标对象的userName
属性值为null
。
日常开发注意:大小写不一致,差一两个字母等等
第4个坑:Null 值覆盖 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 @Data public class SourceBean { private String name; private String address; } @Data public class TargetBean { private String name; private String address; } SourceBean source = new SourceBean (); source.setName("John" ); source.setAddress(null ); TargetBean target = new TargetBean (); target.setAddress("田螺address" ); BeanUtils.copyProperties(source, target); System.out.println(target.getAddress());
源对象 SourceBean
的 address 属性值为 null。默认情况下,BeanUtils.copyProperties
方法会将源对象中的 null 值属性覆盖到目标对象中。因此,目标对象的 address 属性值也为 null。
如果不希望 null 值覆盖目标对象中的属性,可以用 BeanUtils.copyProperties
方法的重载方法,并传入一个自定义的 ConvertUtilsBean
实例来进行配置。
第5个坑:注意引入的包 BeanUtils.copyProperties
其实有两个包,分别是spring、apache
。这两个包,是有点不一样的:
1 2 3 4 public static void copyProperties (Object source, Object target) throws BeansException public static void copyProperties (Object dest, Object orig) throws IllegalAccessException, InvocationTargetException
注意自己引入的哪个BeanUtils
,写对应参数位置。
第6个坑:Boolean类型数据+is属性开头的坑 把SourceBean和TargetBean
中的都有个属性isTianLuo
,它们的数据类型保持不变,但是一个为基本类型boolean
,一个为包装类型Boolean
1 2 3 4 5 6 7 8 @Data public class SourceBean { private boolean isTianLuo; } @Data public class TargetBean { private Boolean isTianLuo; }
跑测试用里的时候,发现赋值不上:这是因为当属性类型为boolean
时,属性名以is
开头,属性名会去掉前面的is
,因此源对象和目标对象属性对不上。
1 2 3 4 5 6 SourceBean source = new SourceBean (); source.setTianLuo(true ); TargetBean target = new TargetBean (); BeanUtils.copyProperties(source, target); System.out.println(target.getIsTianLuo());
第7个坑:查找不到字段引用 在某些开发场景呢,如果我们要修改某个字段的赋值,我们可能会全文搜索它的所有set
方法,看哪些地方引用到。但是如果使用BeanUtils.copyProperties
就不知道是否引用到对应的ste方法啦, 即查找不到字段引用,这就可能导致你会漏掉修改对应的字段。
第8个坑:不同内部类,即使相同属性,也是赋值失败 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 @Data public class CopySource { public String outerName; public CopySource.InnerClass innerClass; @Data public static class InnerClass { public String InnerName; } } @Data public class CopyTarget { public String outerName; public CopyTarget.InnerClass innerClass; @Data public static class InnerClass { public String InnerName; } } CopySource test1 = new CopySource (); test1.outerName = "outTianluo" ; CopySource.InnerClass innerClass = new CopySource .InnerClass(); innerClass.InnerName = "innerTianLuo" ; test1.innerClass = innerClass; System.out.println(test1); CopyTarget test2 = new CopyTarget (); BeanUtils.copyProperties(test1, test2); System.out.println(test2);
CopySource
和CopyTarget
各自存在一个内部类InnerClass
,虽然这个内部类属性也相同,类名也相同,但是在不同的类中,因此Spring
会认为属性不同,不会Copy
。如果要复制成功,可以让他们指向同一个内部类。
第9个坑:bean对应的属性,没有getter和setter方法,赋值失败 BeanUtils.copyProperties
要拷贝属性值成功,需要对应的bean
有getter和setter
方法。因为它是用反射拿到set和get方法再去拿属性值和设置属性值的。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 @Data public class SourceBean { private String value; } @Getter public class TargetBean { private String value; } SourceBean source = new SourceBean (); source.setValue("捡田螺的小男孩" ); TargetBean target = new TargetBean (); BeanUtils.copyProperties(source, target); System.out.println(target.getValue());
第10个坑:BeanUtils.copyProperties + 泛型 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 @Data public class CopySource { public String outerName; public List<CopySource.InnerClass> clazz; @Data public static class InnerClass { public String InnerName; } } @ToString @Data public class CopyTarget { public String outerName; public List<CopyTarget.InnerClass> clazz; @Data public static class InnerClass { public String InnerName; } } CopySource test1 = new CopySource (); test1.outerName = "outTianluo" ; CopySource.InnerClass innerClass = new CopySource .InnerClass(); innerClass.InnerName = "innerTianLuo" ; List<CopySource.InnerClass> clazz = new ArrayList <>(); clazz.add(innerClass); test1.setClazz(clazz); System.out.println(test1); CopyTarget test2 = new CopyTarget (); BeanUtils.copyProperties(test1, test2); System.out.println(test2);
BeanUtils.copyProperties
方法拷贝包含泛型属性的对象clazz
。CopyTarget
和CopySource
的泛型属性类型不匹配,因此拷贝赋值失败。
第11个坑:性能问题 由于这些BeanUtils
类都是采用反射机制实现的,对程序的效率也会有影响。demo
对比:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 SourceBean sourceBean = new SourceBean (); sourceBean.setName("tianLuoBoy" ); TargetBean target = new TargetBean (); long beginTime = System.currentTimeMillis(); for (int i = 0 ; i < 100000 ; i++) { target.setName(sourceBean.getName()); } System.out.println("common setter time:" + (System.currentTimeMillis() - beginTime)); long beginTime1 = System.currentTimeMillis(); for (int i = 0 ; i < 100000 ; i++) { BeanUtils.copyProperties(sourceBean, target); } System.out.println("bean copy time:" + (System.currentTimeMillis() - beginTime1)); common setter time:3 bean copy time:331
可以发现,简单的setter
和BeanUtils.copyProperties
对比,性能差距非常大。因此,慎用BeanUtils.copyProperties
替换BeanUtils.copyProperties的方案 第一种 ,那就是使用原始的setter和getter
方法。
使用手动的setter方法进行属性赋值。这种方法可能需要编写更多的代码,但是可以提供更细粒度的控制,并且在性能方面通常比BeanUtils.copyProperties更高效。
1 2 3 Target target = new Target (); target.setName(source.getName()); target.setAge(source.getAge());
如果实在对象bean
的属性比较多的话,可以使用插件GenerateAllSetter
,它可以一键生成对象的set
方法,挺方便的。
第二种方案 ,使用映射工具库,如MapStruct、ModelMapper
等,它们可以自动生成属性映射的代码。这些工具库可以减少手动编写setter方法的工作量,并提供更好的性能。
1 2 3 4 5 6 7 8 @Mapper public interface SourceTargetMapper { SourceTargetMapper INSTANCE = Mappers.getMapper(SourceTargetMapper.class); @Mapping(source = "name", target = "name") @Mapping(source = "age", target = "age") Target mapToTarget (Source source) ; } Target target = SourceTargetMapper.INSTANCE.mapToTarget(source);
导出文件相关 递归输出文件夹里全部文件名 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 public static void listFilesRecursively (File folder) { File[] contents = folder.listFiles(); if (contents != null ) { for (File item : contents) { if (item.isFile()) { System.out.println(item.getAbsolutePath()); } else if (item.isDirectory()) { listFilesRecursively(item); } } } } public static void main (String[] args) { String folderPath = "F:\\Typora\\touchfish\\9.PDF books" ; File folder = new File (folderPath); listFilesRecursively(folder); }
导出为docx或流文件 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 File file = new File (templatePath + paperName + ".docx" );try ( InputStream is = new FileInputStream (file); ServletOutputStream out = response.getOutputStream(); ) { byte [] buffer = new byte [1024 ]; int len; while ((len = is.read(buffer)) != -1 ) { out.write(buffer, 0 , len); } } catch (Exception e) { e.printStackTrace(); } finally { FileUtils.del(file); }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 try ( ByteArrayOutputStream os = new ByteArrayOutputStream (); ) { ExcelUtil.exportTemplate(errorStudentList,"excel/exportStudentTemplate.xlsx" , os); String encodeToString = Base64.getEncoder().encodeToString(os.toByteArray()); } catch (Exception e) { throw new RuntimeException ("导入异常" ); }
动态导出表头
调用微信接口 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 public class WechatTest { private static final String APP_ID = "xxx" ; private static final String APP_SECRET = "xxx" ; private static final String GROUP_NAME = "森林之父游戏群" ; private static final String NEW_GROUP_NAME = "森林之子游戏群" ; public static void main (String[] args) { try { String accessToken = getAccessToken(); if (accessToken == null ) { System.out.println("Failed to get access token." ); return ; } String groupListJson = getWeChatGroupList(accessToken); if (groupListJson == null ) { System.out.println("Failed to get WeChat group list." ); return ; } String groupId = findGroupIdByGroupName(groupListJson, GROUP_NAME); if (groupId == null ) { System.out.println("Target group not found." ); return ; } boolean success = updateGroupName(accessToken, groupId, NEW_GROUP_NAME); if (success) { System.out.println("Group name updated successfully." ); } else { System.out.println("Failed to update group name." ); } } catch (Exception e) { e.printStackTrace(); } } private static String getAccessToken () throws Exception { String url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=" + APP_ID + "&secret=" + APP_SECRET; HttpURLConnection connection = (HttpURLConnection) new URL (url).openConnection(); connection.setRequestMethod("GET" ); connection.setConnectTimeout(5000 ); connection.setReadTimeout(5000 ); if (connection.getResponseCode() == HttpURLConnection.HTTP_OK) { BufferedReader reader = new BufferedReader (new InputStreamReader (connection.getInputStream())); String line; StringBuilder response = new StringBuilder (); while ((line = reader.readLine()) != null ) { response.append(line); } reader.close(); int startIndex = response.indexOf("\"access_token\":\"" ) + 16 ; int endIndex = response.indexOf("\",\"expires_in\"" ); return response.substring(startIndex, endIndex); } else { System.out.println("Failed to get access token. Response code: " + connection.getResponseCode()); return null ; } } private static String getWeChatGroupList (String accessToken) throws Exception { String url = "https://api.weixin.qq.com/cgi-bin/groups/get?access_token=" + accessToken; HttpURLConnection connection = (HttpURLConnection) new URL (url).openConnection(); connection.setRequestMethod("GET" ); connection.setConnectTimeout(5000 ); connection.setReadTimeout(5000 ); if (connection.getResponseCode() == HttpURLConnection.HTTP_OK) { BufferedReader reader = new BufferedReader (new InputStreamReader (connection.getInputStream())); String line; StringBuilder response = new StringBuilder (); while ((line = reader.readLine()) != null ) { response.append(line); } reader.close(); return response.toString(); } else { System.out.println("Failed to get WeChat group list. Response code: " + connection.getResponseCode()); return null ; } } private static String findGroupIdByGroupName (String groupListJson, String groupName) { String groupId = "YOUR_GROUP_ID" ; return groupId; } private static boolean updateGroupName (String accessToken, String groupId, String newGroupName) throws Exception { String url = "https://api.weixin.qq.com/cgi-bin/groups/update?access_token=" + accessToken; String requestBody = String.format("{\"group\":{\"id\":%s,\"name\":\"%s\"}}" , groupId, URLEncoder.encode(newGroupName, "UTF-8" )); HttpURLConnection connection = (HttpURLConnection) new URL (url).openConnection(); connection.setRequestMethod("POST" ); connection.setConnectTimeout(5000 ); connection.setReadTimeout(5000 ); connection.setDoOutput(true ); connection.getOutputStream().write(requestBody.getBytes("UTF-8" )); if (connection.getResponseCode() == HttpURLConnection.HTTP_OK) { BufferedReader reader = new BufferedReader (new InputStreamReader (connection.getInputStream())); String line; StringBuilder response = new StringBuilder (); while ((line = reader.readLine()) != null ) { response.append(line); } reader.close(); int errCode = 0 ; if (errCode == 0 ) { return true ; } else { System.out.println("Failed to update group name. Error code: " + errCode); return false ; } } else { System.out.println("Failed to update group name. Response code: " + connection.getResponseCode()); return false ; } } }
面试 项目中有没有用到AOP?
当时在后台管理系统中,使用了aop的环绕通知+切点表达式,这个表达式就是要找到要记录日志的方法,然后通过环绕通知的参数获取请求方法的参数,比如类信息、注解、请求方式的等,获取到这些参数以后,保存到数据库。