Java Web 功能
大约 3 分钟
JDK URI 和 URL
从 new URL()
和 new URI
两个构造函数可传入的参数来看,JDK 认为:
new URL(String protocol, String host, String port, String file, URLStreamHandler handler)
- URL 主要关注 “命名空间(协议 + 域名 + 端口)” 的定义;
- URL 提供建立连接的执行的类
URLStreamHandler
;
new URI(String scheme, String userInfo, String host, int port, String path, String query, String fragment)
- URI 主要关注 “相对位置(path、query、fragment)”;
但 URI 比 URL 更加严格:
- URL 只做字符串的拼接
- URI 会对语法(Syntax)进行校验,也能对字符串进行转换
URI uri = new URI("http", "a:b", "xx", 22, "/xx", "a=嗨 嗨 ", "嗨 嗨 ");
// http://a:b@xx:22/xx?a=嗨%20嗨%20#嗨%20嗨%20
System.out.println(uri);
// http://a:b@xx:22/xx?a=%E5%97%A8%20%E5%97%A8%20#%E5%97%A8%20%E5%97%A8%20
System.out.println(uri.toASCIIString());
// http://a:b@xx:22/xx?a=嗨%20嗨%20#嗨%20嗨%20
System.out.println(uri.toURL());
URL url = new URL("http", "a:b@xx.cc", 22, "xx/bb?a=哈 哈 #x哈 x哈 "); // 错误的地址,创建没报错
// http://[a:b@xx.cc]:22xx/bb?a=哈 哈 #x哈 x哈
System.out.println(url); // 输出没做处理
// java.net.URISyntaxException: Illegal character in user info at index 7: http://[a:b@xx.cc]:22xx/bb?a=哈 哈 #x哈 x哈
Assertions.assertThrowsExactly(URISyntaxException.class, () -> url.toURI()); // 报错
// url.openConnection(); // 💡建立连接
综上,个人认为两个类的使用方法如下:
- URI 用于路径定义(包括校验和 normalize 转换)
- 后 URI 转 URL 用于建立连接
坑:URL 中的空格转义问题
参考:
- 如何正确地 urlEncode?空格被 urlEncode 成
+
- https://www.arloor.com/posts/how-to-urlencode/
观察下面代码:
代码一: URLEncoder
URL(Uniform Resource Locator,统一资源定位器,被叫做 “网络地址” 或 “链接”) —— 在 Internet 上可以找到资源的位置的文本字符串。例如 https://developer.mozilla.org
遵循:
- W3C 标准 HTML 4.01 规范 规定: 当
Content-Type
为application/x-www-form-urlencoded
时,URL 中查询参数名和参数值中空格要用加号+
替代,所以几乎所有使用该规范的浏览器在表单提交后,URL 查询参数中空格都会被编成加号+
。
String string = "+ +";
try {
string = URLEncoder.encode(string, "UTF-8"); // 通过 URLEncode 处理,空格 " " 会被处理成加号 "+"。
System.out.println(string); // %2B+%2B
String res = URLDecoder.decode(string,"UTF-8"); // 通过 URLDecoder 处理,会把加号 "+" 和 "%20" 都解码为 " "。
System.out.println(res); // + +
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
提示
Java 中的 URLEncoder 本意是用来把字符串编码成 application/x-www-form-urlencoded
MIME 格式字符串,也就是说仅仅适用于 URL 中的查询字符串部分,但是 URLEncoder 经常被用来对 URL 的其他部分编码 (如:https://www.example.org/你 好.jps?x=世 界
中的 你 好
和 世 界
)。
代码二:URI
URI(Uniform Resource Identifier,统一资源标识符) —— 指向资源的字符串
遵循:
- RFC2396/RFC1738 规范规定: URI 里的保留字符都需转义成
%HH
格式(Section 3.4 Query Component),因此空格会被编码成%20
,加号+
本身也作为保留字而被编成%2B
try {
URI uri = new URI("http", "www.example.org", "/path/../to/你 好", "aa=b b&哦 哦=牛 逼", "fragment哈哈");
System.out.println(uri);
System.out.println(uri.toString()); // 空格 " " 转为 "%20"
System.out.println(uri.normalize()); // ↑ + /path/../to 转为 /to
System.out.println(uri.normalize().toASCIIString()); // ↑ + all to %HH
} catch (URISyntaxException e) {
throw new RuntimeException(e);
}
接口并发限制
注解定义
package org.example.config.web;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ConcurrentLimit {
public String identity() default "";
public int max();
}
切面实现
package org.example.config.web;
import com.google.common.collect.Maps;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.reflect.MethodSignature;
import org.example.entity.Result;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;
import javax.annotation.PostConstruct;
import java.lang.reflect.Method;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
@Slf4j
@Component
@Aspect
public class ConcurrentLimiterAspect {
@PostConstruct
void postConstruct() {
log.info("post construct obj:{}", ConcurrentLimiterAspect.class);
}
private final Map<String, AtomicInteger> concurrentCounterMap = Maps.newConcurrentMap();
// AOP Aspect 切入点语法
@Around("@annotation(concurrentLimit) && (execution(org.example.entity.Result *(..)))")
public Result process(ProceedingJoinPoint joinPoint, ConcurrentLimit concurrentLimit) throws Throwable {
Method method = ((MethodSignature) joinPoint.getSignature()).getMethod();
String identity = Optional.ofNullable(concurrentLimit.identity())
.filter(StringUtils::hasText)
.orElseGet(method::getName);
// String[] params = new LocalVariableTableParameterNameDiscoverer().getParameterNames(method);
AtomicInteger concurrentCounter = concurrentCounterMap
.computeIfAbsent(identity, key -> new AtomicInteger(0));
AtomicBoolean isOverConcurrent = new AtomicBoolean(false);
int max = concurrentLimit.max();
int cur = concurrentCounter.updateAndGet(pre -> {
if (pre < max) {
return pre + 1;
}
isOverConcurrent.set(true);
return pre;
});
log.warn("concurrent limit {}/{}={} in {}.{}", cur, max, isOverConcurrent.get(),
joinPoint.getTarget().getClass(), identity);
if (isOverConcurrent.get()) {
return Result.errorOf(500, "任务忙,请稍后重试", null);
}
try {
return (Result) joinPoint.proceed(joinPoint.getArgs());
} finally {
concurrentCounter.decrementAndGet();
}
}
}
Controller 使用
package org.example.controller;
import lombok.extern.slf4j.Slf4j;
import org.example.config.web.ConcurrentLimit;
import org.example.entity.Result;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.concurrent.TimeUnit;
@Slf4j
@RestController
@RequestMapping("/concurrentLimit")
public class ConcurrentLimitController {
@GetMapping("/sleep/{second}")
@ConcurrentLimit(max = 3)
public Result<String> sleep(@PathVariable("second") Integer second) throws InterruptedException {
TimeUnit.SECONDS.sleep(second);
return Result.successOf("OK");
}
}