徒手用Java来写个Web服务器和框架吧<第一章:NIO篇>
徒手用Java来写个Web服务器和框架吧<第二章:Request和Response>
这一章先把Web框架的功能说一些,有个雏形。
先是制作一个Service,并绑定到一个正则地址。用到了注解和反射。
项目地址: Telemarketer
2017年08月02日更新 已改成注解模式注册 等有空更新
Service的定义
Telemarketer的Service是一个服务。请求了跟它关联的地址,那就由它来为你服务。
它对外只需一个方法。并且对这个方法的要求大概只有输入一个Request,可以返回一个Response这么简单。
那就这样
public interface Service { Response execute(Request request); }
问题来了,如何把它关联到uri上呢?第一想法是在创建一个注册中心。
Service的绑定
先来手动挡
新建一个类ServiceRegistry, 在里面放这么一个静态的Map:
private static Map<String, Service> services = new TreeMap<>();
还提供了这么一个方法:
public static void register(String pattern, Service service) {services.put(pattern, service);}
然后再提供查找服务的方法。
public static Service findService(String pattern) { for (Map.Entry<String, Service> entry : services.entrySet()) { if (pattern.matches(entry.getKey())) { return entry.getValue(); } } return null; }
这样不就根据request确定了服务对象吗?让人们自己动手丰衣足食吧。
然而转念一想,这么搞多麻烦,还得让人专门找个地方register一大堆?这一点都不方便..懒惰是第一大生产力!
那再换个自动挡。
再来自动挡
稍微一想Spring那些框架,用注解搞依赖注入用的那是飞起,Telemarketer也可以!来,说干就干!
先确定注解三个要点
- 确定目标:类
- 确定范围:运行时
- 确定内容:一个String可以放地址正则就好
@Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) public @interface InService { String urlPattern(); }
有了这个,赶紧写个服务先标上。地址正则就匹配根目录吧
1 @InService(urlPattern = "^/$") 2 public class IndexService implements Service { 3 @Override 4 public Response execute(Request request) { 5 return new FileResponse(Status.SUCCESS_200, PropertiesHelper.getTemplateFile("index.html")); 6 } 7 }
PropertiesHelper.getTemplateFile("index.html") 返回的是一个html的文件,暂时不用管。FileResponse是包含文件内容的Response。
行了,现在想想怎么靠这个注解自动注册呢?
假设我们知道了这个类名是 edu.telemarketer.services.servicesimpls.IndexService 。先反射一个:
Class<?> aClass = Class.forName("edu.telemarketer.services.servicesimpls.IndexService");
注意这里用
Class.forName()
和ClassLoader的load()方法区别:Class.forName()会初始化类中的静态域,有什么static都会准备好,而ClassLoader的load就不会有这样的效果。
弄到这个Class对象后,获取InService标注
InService annotation = aClass.getAnnotation(InService.class);
判断一下是不是null 再判断一下Service.class是不是这个类的接口。如果都满足那就是我们要找的类了。实例化它!注册它!
if (annotation != null && Service.class.isAssignableFrom(aClass)) { ServiceRegistry.register(annotation.urlPattern(), aClass.asSubclass(Service.class).newInstance()); }
注意这里的 isAssignableFrom 和 instanceof 的差别, isAssignableFrom 左边是右边的父类或者接口。
整个过程就是先获取根目录,然后递归扫描所有.class进行上述判断。而且反射的时候得用全名,那同时再记录一下包路径。具体可以查看源码。
这个过程还遇到了一个问题: 在一开始我获取了edu.telemarketer的包名想转化为路径用的是name.replaceAll("\\.", File.separator)。在windows环境下测试出现了一个异常。经过查证后发现,replaceAll是不准出现"\"的,会引起混淆。于是用Matcher.quoteReplacement(File.separator)进行替代单独的separator。具体点此处 StackOverflow--replaceAll “/” with File.separator
到此为止自动挡就弄完了。
Connector里的使用
还记得第一章读取事件里面的Connector吗,它是一个实现了Runnable的类. 它的run里面大概是这么一个过程。
1 Request request = Request.parseRequest(channel); 2 Service service =ServiceRegistry.findService(request.getFilePath()); 3 Response response = service.execute(request); 4 channel.register(selector, SelectionKey.OP_WRITE, response);
启动一下,看看页面是不是出来了。
在提交给service处理前和处理后应该由服务器先对头域中的一些信息进行处理,比如维护连接或者处理cookie等等。这些方面后续再来完善。