Vertx session 使用须知

要使用vertx的session,必须把sessionHandler设为router的第一个匹配的route,否则将报错。比如下面的代码将引起报错:

public class LeanS1 extends AbstractVerticle {
	  @Override
	  public void start() throws Exception {
	    Router router = Router.router(vertx);
	    router.route("/test").handler(this::test3);
	    router.route().handler(CookieHandler.create());
	    router.route().handler(SessionHandler.create(LocalSessionStore.create(vertx)));
	    //......
	  }	

在运行的时候,浏览器输入:localhost:8080/test ,就会报错:

严重: Unexpected exception in route
java.lang.NullPointerException
	at io.robin0909.ann.init.LeanS1.test3(LeanS1.java:47)
	at io.vertx.ext.web.impl.RouteImpl.handleContext(RouteImpl.java:219)
	at io.vertx.ext.web.impl.RoutingContextImplBase.iterateNext(RoutingContextImplBase.java:120)
	at io.vertx.ext.web.impl.RoutingContextImpl.next(RoutingContextImpl.java:120)
	at io.vertx.ext.web.impl.RouterImpl.accept(RouterImpl.java:79)
	at io.vertx.core.http.impl.Http1xServerConnection.processMessage(Http1xServerConnection.java:433)
	at io.vertx.core.http.impl.Http1xServerConnection.handleMessage(Http1xServerConnection.java:141)
	at io.vertx.core.http.impl.HttpServerImpl$ServerHandlerWithWebSockets.handleMessage(HttpServerImpl.java:683)
	at io.vertx.core.http.impl.HttpServerImpl$ServerHandlerWithWebSockets.handleMessage(HttpServerImpl.java:636)
	at io.vertx.core.net.impl.VertxHandler.lambda$channelRead$1(VertxHandler.java:146)
	at io.vertx.core.impl.ContextImpl.lambda$wrapTask$2(ContextImpl.java:337)
	at io.vertx.core.impl.ContextImpl.executeFromIO(ContextImpl.java:195)
	at io.vertx.core.net.impl.VertxHandler.channelRead(VertxHandler.java:144)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:362)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:348)
	at io.netty.channel.AbstractChannelHandlerContext.fireChannelRead(AbstractChannelHandlerContext.java:340)
	at io.vertx.core.http.impl.HttpServerImpl$Http2UpgradeHandler.channelRead(HttpServerImpl.java:952)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:362)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:348)
	at io.netty.channel.AbstractChannelHandlerContext.fireChannelRead(AbstractChannelHandlerContext.java:340)
	at io.netty.handler.codec.ByteToMessageDecoder.fireChannelRead(ByteToMessageDecoder.java:310)
	at io.netty.handler.codec.ByteToMessageDecoder.channelRead(ByteToMessageDecoder.java:284)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:362)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:348)
	at io.netty.channel.AbstractChannelHandlerContext.fireChannelRead(AbstractChannelHandlerContext.java:340)
	at io.vertx.core.http.impl.Http1xOrH2CHandler.end(Http1xOrH2CHandler.java:60)
	at io.vertx.core.http.impl.Http1xOrH2CHandler.channelRead(Http1xOrH2CHandler.java:38)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:362)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:348)
	at io.netty.channel.AbstractChannelHandlerContext.fireChannelRead(AbstractChannelHandlerContext.java:340)
	at io.netty.channel.DefaultChannelPipeline$HeadContext.channelRead(DefaultChannelPipeline.java:1359)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:362)
	at io.netty.channel.AbstractChannelHandlerContext.invokeChannelRead(AbstractChannelHandlerContext.java:348)
	at io.netty.channel.DefaultChannelPipeline.fireChannelRead(DefaultChannelPipeline.java:935)
	at io.netty.channel.nio.AbstractNioByteChannel$NioByteUnsafe.read(AbstractNioByteChannel.java:141)
	at io.netty.channel.nio.NioEventLoop.processSelectedKey(NioEventLoop.java:645)
	at io.netty.channel.nio.NioEventLoop.processSelectedKeysOptimized(NioEventLoop.java:580)
	at io.netty.channel.nio.NioEventLoop.processSelectedKeys(NioEventLoop.java:497)
	at io.netty.channel.nio.NioEventLoop.run(NioEventLoop.java:459)
	at io.netty.util.concurrent.SingleThreadEventExecutor$5.run(SingleThreadEventExecutor.java:886)
	at io.netty.util.concurrent.FastThreadLocalRunnable.run(FastThreadLocalRunnable.java:30)
	at java.lang.Thread.run(Unknown Source)

当把sessionHandler设为router的第一个匹配的route时,运行正常,正确代码如下:

public void start() throws Exception {
	    Router router = Router.router(vertx);	    
	    router.route().handler(CookieHandler.create());
	    router.route().handler(SessionHandler.create(LocalSessionStore.create(vertx)));
	    router.route("/test").handler(this::test3);
	    //.....



对于router.route().handler(...) 所指定的通用路径处理函数,无论指定了多少个通用路径函数,都会在访问时被顺序执行一遍。
但是对于router.route("/test").handler(...) 所指定的专用路径处理函数,如果指定了多个,那么不会自动全部执行,
只执行第一个。只有遇到routingContext.next();指令的时候才会继续执行下一个处理函数


 这点可以通过代码测试。

比如:

public void start() throws Exception {
	    Router router = Router.router(vertx);	    
	    router.route().handler(CookieHandler.create());
	    router.route().handler(SessionHandler.create(LocalSessionStore.create(vertx)));
	    router.route("/test").handler(this::test3);
	    router.route("/test").handler(this::test4);
	    router.route("/test").handler(this::test5);	    
	    //.....

对于上述代码,当在浏览器输入localhost:8080/test的时候, test3的结尾如果没有next指令,test4,test5就不会被执行。


vertx 服务在部分安卓手机上页面跳转的时候,常常丢失session,后来发现是Cookie的Path问题导致出错,所以,可以在verticle文件中加入如下代码:

//to make session work correctly on Huawei & Mi phone, set Cookie Path to root '/' 
        Set<Cookie> ccList=routingContext.cookies();
        for(Cookie ck:ccList){
        	ck.setPath("/");        	
        }


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您好!对于使用 Vert.x-MQTT 和 Protobuf 进行编解码,您可以按照以下步骤进行操作: 1. 首先,您需要确保已经在您的项目中添加了 Vert.x-MQTT 和 Protobuf 的相关依赖。 2. 创建一个 Protobuf 定义文件(.proto 文件),用于定义您的消息结构。例如,假设您的消息结构如下所示: ``` syntax = "proto3"; message MyMessage { string field1 = 1; int32 field2 = 2; } ``` 3. 使用 Protobuf 编译器将 .proto 文件编译为 Java 类。您可以使用以下命令行命令: ``` protoc --java_out=<output_directory> <your_proto_file.proto> ``` 此命令将生成相应的 Java 类文件,用于在代码中进行消息的编码和解码。 4. 在 Vert.x-MQTT 中使用编码器和解码器。您可以创建一个类来实现 Vert.x-MQTT 的 `MqttMessageCodec` 接口,并在其中实现编码和解码逻辑。在这个类中,您可以使用 Protobuf 生成的类来进行消息的序列化和反序列化。 以下是一个简单的示例: ```java import io.vertx.mqtt.MqttMessageCodec; import io.vertx.mqtt.messages.MqttPublishMessage; import com.example.protobuf.MyMessage; public class MyMessageCodec implements MqttMessageCodec<MyMessage> { @Override public void encodeToWire(Buffer buffer, MyMessage myMessage) { byte[] payload = myMessage.toByteArray(); // 在此处将 payload 写入到 buffer 中 } @Override public MyMessage decodeFromWire(int pos, Buffer buffer) { // 从 buffer 中读取 payload,并将其反序列化为 MyMessage 对象 byte[] payload = buffer.getBytes(pos, buffer.length()); return MyMessage.parseFrom(payload); } @Override public MyMessage transform(MqttPublishMessage mqttPublishMessage) { // 将 MqttPublishMessage 转换为 MyMessage 对象 byte[] payload = mqttPublishMessage.payload().getBytes(); return MyMessage.parseFrom(payload); } @Override public MqttPublishMessage transform(MyMessage myMessage) { // 将 MyMessage 对象转换为 MqttPublishMessage return MqttPublishMessage.create(myMessage.toByteArray(), false, QualityOfService.AT_MOST_ONCE, false); } } ``` 5. 最后,在您的 Vert.x-MQTT 代码中,使用您自定义的消息编解码器。示例如下: ```java MqttServerOptions options = new MqttServerOptions(); options.setCodecs(new MyMessageCodec()); MqttServer mqttServer = MqttServer.create(vertx, options); mqttServer.endpointHandler(endpoint -> { endpoint.publishHandler(message -> { // 处理收到的消息 MyMessage myMessage = message.payload(); // ... }); }).listen(); ``` 这样,您就可以使用 Vert.x-MQTT 和 Protobuf 进行消息的编码和解码了。 请注意,上述示例是一个简单示例,您可能需要根据您的实际需求进行适当的修改和调整。希望对您有所帮助!如果您有任何其他问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值