Vert.x学习第一天

  昨天看了下异步,而后就开始了Vert.x相关知识的学习。html

  Vert.x是当下很是流行的一套全异步框架,其优点在于轻量级、高效。很是适合做为移动端后台或是企业应用。java

  固然对于第一天接触这个框架的人(没错,正是在下)来讲,Vert.x一些独特的特性还不是如今了解的时候,对着说明文档去码一些demo才是正道。web

  首先咱们先建一个gradle项目,而后在build.gradle中的dependencies中添加 compile 'io.vertx:vertx-core:3.5.0' (下载jar包),以后在build.gradle文件的最后添加 task copyJars(type: Copy) { from configurations.runtime into 'lib' } (下载到lib文件夹,这个文件夹本身在项目下新建一个)浏览器

若是使用的是idea的话,能够双击咱们新添加的copyJar运行(以下图),这样jar包就下载好了,或者在cmd中进入到build.gradle所在文件夹中,使用语句 gradle copyJar 也能够。服务器

 

  添加好jar包以后就能够写一个test类了。app

 1 import io.vertx.core.Vertx;
 2 import io.vertx.core.http.HttpServerResponse;
 3 import javafx.application.Application;
 4 import javafx.stage.Stage;
 5 
 6 public class test extends Application{
 7 
 8     @Override
 9     public void start(Stage primaryStage) throws Exception {
10         Vertx.vertx().createHttpServer().requestHandler(req -> {
11             HttpServerResponse response=req.response();
12             System.out.println("lodaing");
13             response.setChunked(true);
14             response.write("aaaa");
15             response.end();
16         }).listen(8889,"localhost");
17 
18     }
19 }

  这是一个最简单服务器代码,实现的做用是:每当有对localhost这个主机8889端口的请求,便会响应“aaaa”字符串。框架

  这个test类继承了Application,而后实现了Application的start()方法。(Application继承自AbstractVerticle,若是test类继承AbstractVertical一样能够实现这个功能,可是要写一个main方法调用start方法。AbstractVerticle和Application的区别我尚未具体的看,可是Application的start()方法是会自动调用的)异步

   Vertx.vertx() 返回一个Vertx对象,这个对象能够使用 createHttpServer() 来建立一个服务器对象,我这里用链式直接写了,为了方便理解,咱们能够把这个服务器对象单独拿出来 HttpServer server=Vertx.vertx().createHttpServer(); ide

  咱们能够给这个服务器对象server添加一个监听 server.listen(8889,"localhost") ,这两个参数不写的话,端口默认是80,主机地址默认是0.0.0.0。这样,每当有对localhost的8889端口的请求,都会被这个服务器监听到。post

  若是我想在监听到一个请求以后,让服务器作出一些响应,那么咱们就应该使用 server.requestHandler() ,requestHandler()里面有一个Handler对象,这里使用lambda表达式来写这个匿名内部类。总之,咱们传入了一个HttpRequest类型的req进去,这个req就是被咱们监听到的请求。 req.response() 建立了一个对当前请求的响应对象,经过这个响应对象,咱们使用write()方法能够一个响应字符串。这里要注意的是,使用write()方法以前必须指定要发送的信息的长度。或者用 response.setChunked(true); 来指定要发送的信息是分块发送。指定分块发送的信息不须要指定长度,可是须要在write()以后使用 response.end(); 来表示我要写的信息已经结束啦,这样服务器才会把要响应的信息发送回去。还有一种返回字符串的方法能够省略write(),就是直接在end()里面添加字符串,可是只能发送一次,再次response.end("sdasd");就会报错。

  写好了以上的代码以后,咱们运行代码,而后在浏览器中输入地址“localhost:8889”就能够看到响应的字符串出如今浏览器上了。由于咱们监听的是8889端口全部的请求,因此“localhost:8889/asda/asdas/fasf”这样的地址也会返回一样的结果。

 

以上只是最最简单的一个例子,上面的代码并无去分析请求中的信息,只是获得请求就响应了。下面写一些从请求中获得信息的方法:

req.version()  //获得请求的版本信息,如HTTP1.1
req.method()  //获得请求的方式,如GET、POST
req.path()  //获得请求的地址,如“localhost:8080/user/hello.html?id=1&pwd=123”获得的结果是 “/user/hello.html”部分
req.uri()   //获得请求的数据部分,获得的结果是“id=1&pwd=123”部分
req.absoluteURI()  //返回完整地址,“http://“localhost:8080/user/hello.html?id=1&pwd=123”

Vertx由于是异步的,在一个请求到达时,它会先接受请求头部信息(head),不用等待body到来就能够处理请求。因此上面的方法只能够获得请求头的数据。若是想要读取body中的信息(如post方式传递的数据),能够用如下方法:

req.setExpectMultipart(true);  //告诉服务器,我要读取body信息
req.endHandler(v->{
  MultiMap fa=req.formAttributes();//从req中读取body全部信息并存储到MultiMap对象中
   System.out.println(fa.entries().toString()); //entries()方法会返回一个list格式的数据,MultiMap类型的数据具体怎么读我还没仔细研究
});

为了验证,我写了一个web 表单,来验证功能:(webroot/index2.html)

 1 <!DOCTYPE html>
 2 <html lang="en">
 3 <head>
 4     <meta charset="UTF-8">
 5     <title>Title</title>
 6 </head>
 7 <body>
 8 <form action="/user" method="post">
 9     <input type="text" name="name" id="name" value="hellooooo"/>
10     <input type="submit" name="submit" id="submit" value="submit"/>
11 </form>
12 </body>
13 </html>

(SAndC.Servert.java以下:)

 1 package SAndC;
 2 
 3 import io.vertx.core.MultiMap;
 4 import io.vertx.core.Vertx;
 5 import javafx.application.Application;
 6 import javafx.stage.Stage;
 7 
 8 public class Server extends Application{
 9 
10     @Override
11     public void start(Stage primaryStage) throws Exception {
12         Vertx.vertx().createHttpServer().requestHandler(req->{
13             if(!req.path().equals("/webroot/index2.html")){
14                 req.response().end("i get the message!\n"+req.version()+"\n"+req.method()+"\n"+req.path()+"\n");
15                 req.setExpectMultipart(true);
16                 req.endHandler(v->{
17                     MultiMap fa=req.formAttributes();
18                     System.out.println(fa.entries().toString());
19                     System.out.println(req.absoluteURI());
20                 });
21             }else{
22                 req.response().sendFile("webroot/index2.html");
23             }
24         }).listen(8080,"localhost",res->{
25             if(res.succeeded()){
26                 System.out.println("Server is now listening!");
27             }else{
28                 System.out.println("I wangt to listen but failed!");
29             }
30         });
31     }
32 }

  上面的功能是:接收到一个一个请求后,判断请求路径是否是/webroot/index2.html,若是是,就返回index2.html这个页面,这样咱们就能看到本身的表单了。

  而后,index2.html中表单的action我是随便填的,反正都会被监听到,这样提交的数据就是post方式,请求的body中会包含name和submit的数据。

  点击提交以后,判断路径不是/webroot/index2.html,因此读取请求的body信息,结果以下:

                   

相关文章
相关标签/搜索