转载请注明出处。
智慧公交-核心的部分就是消息推送这一块,推送和接受消息能跑通了,其他的就是业务上问题了。
智能公交调度系统-系统结合了GPS GIS GPRS 等。GPS、GPRS大家都明白,GIS(地理信息系统)在智慧公交项目上主要是用来展示公交线路的地图。在地图上我们可以看到公交线路,车辆位置、行驶方向,可以查看公交车内部的监控、公交站点监控等实时信息。
通过实时监控、超速报警、自动报站、车载LED跑马板等功能使司机操作服务更规范。
通过车内安装监控设备使市民安全性提高,如遇被偷窃等特殊情况将方便调查及作为有效的证据。
对市民投诉较多的司机越站、过站等问题,可通过轨迹回放与监控追查原因,及时地进行处理。啰嗦的有点多,下面就进入正题。
**************************************************************************************************************************************************************************************
上一篇文章有三个问题还没有解决,现在就来解决这几个问题。
1. Spring如何更有效地管理ActiveMQ?
2. ActiveMQ如何将对象推送到前端,前端又是如何接受对象?
3. 如何实现订阅信息的筛选?
一.消息的发送和接受
Spring也为我们提供了JMSTemplate模板,可以发送和接受消息,类似hibernateTemplate。
可是jmsTemplate如何知道消息转换器呢?需要在配置jmsTemplate的时候,加上messageConverter属性。
<bean id="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="topicSendConnectionFactory" />
<property name="defaultDestination" ref="myTopic" />
<property name="messageConverter" ref="messageConvertForSys" />
<property name="deliveryMode" value="1" />
<!-- 发送模式 DeliveryMode.NON_PERSISTENT=1:非持久 ; DeliveryMode.PERSISTENT=2:持久-->
<property name="pubSubDomain" value="true" /> <span style="font-family: Arial, Helvetica, sans-serif;"> <!-- 开启订阅模式 --> </span>
</bean>
二.当有消息推送过来,我们如何来监听呢?
Spring提供了三种 AbstractMessageListenerContainer 的子类,每种各有其特点。
第一种:SimpleMessageListenerContainer
这个消息侦听容器是三种中最简单的。它在启动时创建固定数量的JMS session并在容器的整个生命周期中使用它们。这个类不能动态的适应运行时的要求或参与消息接收的事务处理。然而它对JMS提供者的要求也最低。它只需要简单的JMS API。
第二种:DefaultMessageListenerContainer
这个消息侦听器使用的最多。和 SimpleMessageListenerContainer 相反,这个子类可以动态适应运行时侯的要求,也可以参与事务管理。每个收到的消息都注册到一个XA事务中(如果使用 JtaTransactionManager 配置过),这样就可以利用XA事务语义的优势了。这个类在对JMS提供者的低要求和提供包括事务参于等的强大功能上取得了很好的平衡。
第三种:ServerSessionMessageListenerContainer
这个监听器容器利用JMS ServerSessionPool SPI动态管理JMS Session。使用者各种消息监听器可以获得运行时动态调优功能,但是这也要求JMS提供者支持ServerSessionPoolSPI。
<bean id="listenerContainer" class="org.springframework.jms.listener.DefaultMessageListenerContainer">
三.消息的转换
转化器在很多组件中都是必不缺少的东西。Spring MessageConverter接口提供了对消息转换的支持,这样我们接可以推送java对象了。
四.Java与Flex通过Blazeds通信
blazeDS采用amf协议。该协议可以传输Object, Array, Date, XML。由于AMF采用二进制编码,这种方式可以高度压缩数据,因此非常适合用来传递大量的资料。数据量越大,Flash Remoting的传输效能就越高,远远超过Web Service。至于XML, LoadVars和loadVariables() ,它们使用纯文本的传输方式,效能就更不能与Flash Remoting相提并论了。
五.Java与Flex交互(对象)
然后创建一个ActionScript类:Bus.as,用于接受实体类Bus.java的返回值,这里的[Bindable][RemoteClass]是和java代码中的 Bus.java类关联起来。这样利用类型转换将java对象转换成AS对象。代码如下:
package vo
{
[Bindable][RemoteClass(alias="com.test.jms.flex.BusInfo")]
public class BusInfo
{
public function BusInfo()
{
}
public var platenum:String;
public var speed:String;
public var direction:String;
public var lon:String;
public var lat:String;
public var receivetime:String;
public var routenum:String
}
}
六.订阅信息的筛选(消息过滤)
后台通过AsyncMessage的setHeader()方法来确定筛选条件。
前台flex通过Consumer.selector方式来进行消息的过滤。
七.详细代码
1.ActiveMQ 的activemq.xml配置文件
加入UDP传输方式(<transportConnector name="udp" uri="udp://0.0.0.0:8123" /> )
<!--
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You under the Apache License, Version 2.0
(the "License"); you may not use this file except in compliance with
the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
-->
<!-- START SNIPPET: example -->
<beans
xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://activemq.apache.org/schema/core http://activemq.apache.org/schema/core/activemq-core.xsd">
<!-- Allows us to use system properties as variables in this configuration file -->
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="locations">
<value>file:${activemq.conf}/credentials.properties</value>
</property>
</bean>
<!-- Allows log searching in hawtio console -->
<bean id="logQuery" class="org.fusesource.insight.log.log4j.Log4jLogQuery"
lazy-init="false" scope="singleton"
init-method="start" destroy-method="stop">
</bean>
<!--
The <broker> element is used to configure the ActiveMQ broker.
-->
<broker xmlns="http://activemq.apache.org/schema/core" brokerName="localhost" dataDirectory="${activemq.data}">
<destinationPolicy>
<policyMap>
<policyEntries>
<policyEntry topic=">" >
<!-- The constantPendingMessageLimitStrategy is used to prevent
slow topic consumers to block producers and affect other consumers
by limiting the number of messages that are retained
For more information, see:
http://activemq.apache.org/slow-consumer-handling.html
-->
<pendingMessageLimitStrategy>
<constantPendingMessageLimitStrategy limit="1000"/>
</pendingMessageLimitStrategy>
</policyEntry>
</policyEntries>
</policyMap>
</destinationPolicy>
<!--
The managementContext is used to configure how ActiveMQ is exposed in
JMX. By default, ActiveMQ uses the MBean server that is started by
the JVM. For more information, see:
http://activemq.apache.org/jmx.html
-->
<managementContext>
<managementContext createConnector="false"/>
</managementContext>
<!--
Configure message persistence for the broker. The default persistence
mechanism is the KahaDB store (identified by the kahaDB tag).
For more information, see:
http://activemq.apache.org/persistence.html
-->
<persistenceAdapter>
<kahaDB directory="${activemq.data}/kahadb"/>
</persistenceAdapter>
<!--
The systemUsage controls the maximum amount of space the broker will
use before disabling caching and/or slowing down producers. For more information, see:
http://activemq.apache.org/producer-flow-control.html
-->
<systemUsage>
<systemUsage>
<memoryUsage>
<memoryUsage percentOfJvmHeap="70" />
</memoryUsage>
<storeUsage>
<storeUsage limit="100 gb"/>
</storeUsage>
<tempUsage>
<tempUsage limit="50 gb"/>
</tempUsage>
</systemUsage>
</systemUsage>
<!--
The transport connectors expose ActiveMQ over a given protocol to
clients and other brokers. For more information, see:
http://activemq.apache.org/configuring-transports.html
-->
<transportConnectors>
<!-- DOS protection, limit concurrent connections to 1000 and frame size to 100MB -->
<transportConnector name="openwire" uri="tcp://0.0.0.0:61616?maximumConnections=1000&wireFormat.maxFrameSize=104857600"/>
<transportConnector name="udp" uri="udp://0.0.0.0:8123"/>
<transportConnector name="amqp" uri="amqp://0.0.0.0:5672?maximumConnections=1000&wireFormat.maxFrameSize=104857600"/>
<transportConnector name="stomp" uri="stomp://0.0.0.0:61613?maximumConnections=1000&wireFormat.maxFrameSize=104857600"/>
<transportConnector name="mqtt" uri="mqtt://0.0.0.0:1883?maximumConnections=1000&wireFormat.maxFrameSize=104857600"/>
<transportConnector name="ws" uri="ws://0.0.0.0:61614?maximumConnections=1000&wireFormat.maxFrameSize=104857600"/>
</transportConnectors>
<!-- destroy the spring context on shutdown to stop jetty -->
<shutdownHooks>
<bean xmlns="http://www.springframework.org/schema/beans" class="org.apache.activemq.hooks.SpringContextHook" />
</shutdownHooks>
</broker>
<!--
Enable web consoles, REST and Ajax APIs and demos
The web consoles requires by default login, you can disable this in the jetty.xml file
Take a look at ${ACTIVEMQ_HOME}/conf/jetty.xml for more details
-->
<import resource="jetty.xml"/>
</beans>
<!-- END SNIPPET: example -->
2. web.xml
在web.xml中加入 org.springframework.web.context.ContextLoaderListener
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5"
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath*:applicationContext.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<listener>
<listener-class>flex.messaging.HttpFlexSession</listener-class>
</listener>
<servlet>
<servlet-name>flex</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>flex</servlet-name>
<url-pattern>/messagebroker/*</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>ControlServlet1</servlet-name>
<servlet-class>com.test.jms.servlet.DelegatingServletProxy</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>ControlServlet1</servlet-name>
<url-pattern>/ControlServlet1</url-pattern>
</servlet-mapping>
</web-app>
3. 工程WebRoot\WEB-INF下flex文件夹里的四个配置文件
1) messaging-config.xml
<?xml version="1.0" encoding="UTF-8"?>
<service id="message-service"
class="flex.messaging.services.MessageService">
<adapters>
<adapter-definition id="actionscript" class="flex.messaging.services.messaging.adapters.ActionScriptAdapter" default="true" />
<!-- <adapter-definition id="jms" class="flex.messaging.services.messaging.adapters.JMSAdapter"/> -->
</adapters>
<default-channels>
<channel ref="my-polling-amf" />
<channel ref="my-streaming-amf" />
</default-channels>
</service>
(
2) proxy-config.xml
<?xml version="1.0" encoding="UTF-8"?>
<service id="proxy-service"
class="flex.messaging.services.HTTPProxyService">
<properties>
<connection-manager>
<max-total-connections>100</max-total-connections>
<default-max-connections-per-host>2</default-max-connections-per-host>
</connection-manager>
<allow-lax-ssl>true</allow-lax-ssl>
</properties>
<adapters>
<adapter-definition id="http-proxy" class="flex.messaging.services.http.HTTPProxyAdapter" default="true"/>
<adapter-definition id="soap-proxy" class="flex.messaging.services.http.SOAPProxyAdapter"/>
</adapters>
<default-channels>
<channel ref="my-amf"/>
</default-channels>
<destination id="DefaultHTTP">
</destination>
</service>
(
3) remoting-config.xml
<?xml version="1.0" encoding="UTF-8"?>
<service id="remoting-service"
class="flex.messaging.services.RemotingService">
<adapters>
<adapter-definition id="java-object" class="flex.messaging.services.remoting.adapters.JavaAdapter" default="true"/>
</adapters>
<default-channels>
<channel ref="my-amf"/>
</default-channels>
</service>
(
4) services-config.xml
<?xml version="1.0" encoding="UTF-8"?>
<services-config>
<services>
<service-include file-path="remoting-config.xml" />
<service-include file-path="proxy-config.xml" />
<service-include file-path="messaging-config.xml" />
</services>
<services>
<default-channels>
<channel ref="my-amf" />
</default-channels>
</services>
<security>
<login-command class="flex.messaging.security.TomcatLoginCommand" server="Tomcat"/>
<!-- Uncomment the correct app server
<login-command class="flex.messaging.security.TomcatLoginCommand" server="JBoss">
<login-command class="flex.messaging.security.JRunLoginCommand" server="JRun"/>
<login-command class="flex.messaging.security.WeblogicLoginCommand" server="Weblogic"/>
<login-command class="flex.messaging.security.WebSphereLoginCommand" server="WebSphere"/>
-->
<!--
<security-constraint id="basic-read-access">
<auth-method>Basic</auth-method>
<roles>
<role>guests</role>
<role>accountants</role>
<role>employees</role>
<role>managers</role>
</roles>
</security-constraint>
-->
</security>
<channels>
<channel-definition id="my-amf" class="mx.messaging.channels.AMFChannel">
<endpoint url="http://{server.name}:{server.port}/{context.root}/messagebroker/amf" class="flex.messaging.endpoints.AMFEndpoint"/>
</channel-definition>
<channel-definition id="my-secure-amf" class="mx.messaging.channels.SecureAMFChannel">
<endpoint url="https://{server.name}:{server.port}/{context.root}/messagebroker/amfsecure" class="flex.messaging.endpoints.SecureAMFEndpoint"/>
<properties>
<add-no-cache-headers>false</add-no-cache-headers>
</properties>
</channel-definition>
<channel-definition id="my-polling-amf" class="mx.messaging.channels.AMFChannel">
<endpoint url="http://{server.name}:{server.port}/{context.root}/messagebroker/amfpolling" class="flex.messaging.endpoints.AMFEndpoint"/>
<properties>
<polling-enabled>true</polling-enabled>
<polling-interval-seconds>4</polling-interval-seconds>
</properties>
</channel-definition>
<channel-definition id="my-streaming-amf" class="mx.messaging.channels.StreamingAMFChannel">
<endpoint url="http://{server.name}:{server.port}/{context.root}/messagebroker/streamingamf" class="flex.messaging.endpoints.StreamingAMFEndpoint"/>
<properties>
<idle-timeout-minutes>0</idle-timeout-minutes>
<max-streaming-clients>100</max-streaming-clients>
<server-to-client-heartbeat-millis>5000
</server-to-client-heartbeat-millis>
<user-agent-settings>
<user-agent match-on="MSIE" kickstart-bytes="2048" max-streaming-connections-per-session="100" />
<user-agent match-on="Firefox" kickstart-bytes="2048" max-streaming-connections-per-session="100" />
</user-agent-settings>
</properties>
</channel-definition>
</channels>
<logging>
<target class="flex.messaging.log.ConsoleTarget" level="Error">
<properties>
<prefix>[BlazeDS] </prefix>
<includeDate>false</includeDate>
<includeTime>false</includeTime>
<includeLevel>false</includeLevel>
<includeCategory>false</includeCategory>
</properties>
<filters>
<pattern>Endpoint.*</pattern>
<pattern>Service.*</pattern>
<pattern>Configuration</pattern>
</filters>
</target>
</logging>
<system>
<redeploy>
<enabled>false</enabled>
</redeploy>
</system>
</services-config>
4.Spring的配置文件applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:flex="http://www.springframework.org/schema/flex"
xmlns:amq="http://activemq.apache.org/schema/core"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-3.0.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
http://www.springframework.org/schema/flex
http://www.springframework.org/schema/flex/spring-flex-1.0.xsd
http://activemq.apache.org/schema/core
http://activemq.apache.org/schema/core/activemq-core-5.2.0.xsd
">
<!--%%%%%%%%%%%%%%%%%%*********************消息处理 ACTIVEMQ***************************%%%%%%%%%%%%% -->
<!-- JMS TOPIC MODEL -->
<!-- TOPIC链接工厂 -->
<bean id="topicSendConnectionFactory" class="org.apache.activemq.spring.ActiveMQConnectionFactory">
<property name="brokerURL" value="udp://localhost:8123" />
<property name="useAsyncSend" value="true" />
</bean>
<bean id="topicListenConnectionFactory" class="org.apache.activemq.spring.ActiveMQConnectionFactory">
<property name="brokerURL" value="udp://localhost:8123" />
</bean>
<!-- 定义主题 -->
<bean id="myTopic" class="org.apache.activemq.command.ActiveMQTopic">
<constructor-arg value="normandy.topic"/>
</bean>
<bean id="messageConvertForSys" class="com.test.jms.util.MessageConvertForSys"></bean>
<!-- TOPIC send jms模板 -->
<bean id="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="topicSendConnectionFactory"></property>
<property name="defaultDestination" ref="myTopic" />
<property name="messageConverter" ref="messageConvertForSys" />
<property name="deliveryMode" value="1" /> <!-- 发送模式 DeliveryMode.NON_PERSISTENT=1:非持久 ; DeliveryMode.PERSISTENT=2:持久-->
<property name="pubSubDomain" value="true"/> <!-- 开启订阅模式 -->
</bean>
<!-- 消息发送方 -->
<bean id="topicMessageSender" class="com.test.jms.util.MessageSender">
<property name="jmsTemplate" ref="jmsTemplate"></property>
</bean>
<!-- 消息接收方 -->
<bean id="topicMessageReceiver" class="com.test.jms.util.MessageReceiver"> </bean>
<!-- 主题消息监听容器 -->
<bean id="listenerContainer" class="org.springframework.jms.listener.DefaultMessageListenerContainer">
<property name="connectionFactory" ref="topicListenConnectionFactory" />
<property name="pubSubDomain" value="true"/><!-- 订阅模式 -->
<property name="destination" ref="myTopic" /> <!-- 目的地 myTopic -->
<property name="pubSubNoLocal" value="false"></property>
<property name="messageListener" ref="topicMessageReceiver" />
<property name="receiveTimeout" value="5000" />
<!-- <property name="clientId" value="clientId_001"/>-这里是设置接收客户端的ID,在持久化时,但这个客户端不在线时,消息就存在数据库里,知道被这个ID的客户端消费掉 -->
<!-- <property name="messageListener" ref="topicMessageReceiver" /> -->
</bean>
</beans>
5.Java后台代码
Sender.java
package com.test.jms.util;
import java.text.ParseException;
import javax.jms.Connection;
import javax.jms.ConnectionFactory;
import javax.jms.Destination;
import javax.jms.JMSException;
import javax.jms.MessageProducer;
import javax.jms.Session;
import org.apache.activemq.ActiveMQConnection;
import org.apache.activemq.ActiveMQConnectionFactory;
import com.test.jms.model.Bus;
public class Sender {
private MessageSender topicMessageSender;
public MessageSender getTopicMessageSender() {
return topicMessageSender;
}
public void setTopicMessageSender(MessageSender topicMessageSender) {
this.topicMessageSender = topicMessageSender;
}
public void send(String s) {
String [] points ={"111.62618840,29.07165395","111.62657105,29.07158371","111.62696581,29.07151106","111.62744049,29.07145051","111.62762940,29.07147231","111.62792970,29.07169270","111.62816705,29.07186949","111.62848189,29.07209472","111.62876040,29.07233691","111.62906427,29.07262604","111.62941786,29.07296510","111.62963824,29.07317096","111.62982957,29.07335986","111.63007175,29.07359478","111.63025097,29.07375946","111.63054401,29.07403071","111.63066995,29.07420024","111.63080800,29.07441336","111.63093393,29.07459984","111.63107924,29.07479844","111.63117369,29.07501398","111.63124393,29.07517382","111.63135049,29.07543054","111.63145568,29.07564685","111.63145568,29.07599196","111.63145810,29.07635282","111.63145689,29.07672336","111.63145083,29.07684203","111.63143872,29.07692679","111.63144599,29.07705999","111.63144114,29.07717624","111.63142177,29.07726827","111.63141693,29.07735425","111.63141693,29.07735425","111.63143993,29.07746081","111.63144599,29.07767030","111.63144599,29.07767030","111.63144962,29.07812440","111.63144841,29.07834357","111.63145689,29.07859666","111.63144841,29.07901200","111.63144599,29.07922391","111.63145204,29.07952664","111.63144114,29.07964168","111.63143872,29.07975914","111.63143872,29.07975914","111.63143872,29.07975914","111.63143872,29.07975914","111.63146657,29.07979910","111.63146657,29.07985965","111.63158167,29.07985129","111.63168338,29.07985371","111.63181174,29.07984402","111.63225736,29.07982223","111.63280228,29.07979801","111.63337383,29.07978105","111.63389211,29.07975684","111.63417546,29.07974957","111.63441765,29.07973988","111.63462351,29.07973988","111.63502553,29.07974230","111.63520717,29.07973504","111.63542998,29.07965754","111.63554623,29.07961879","111.63597732,29.07949528","111.63660852,29.07926122","111.63720429,29.07904568","111.63744405,29.07895607","111.63759663,29.07888584","111.63759663,29.07888584","111.63759663,29.07888584","111.63780245,29.07881411","111.63796350,29.07877536","111.63812697,29.07872450","111.63854595,29.07859856","111.63885837,29.07850774","111.63920348,29.07840118","111.63933547,29.07833337","111.63933547,29.07833337","111.63933547,29.07833337","111.63947594,29.07830552","111.63965644,29.07824570","111.63996160,29.07815851","111.64019652,29.07807980","111.64044475,29.07800594","111.64087100,29.07787273","111.64114830,29.07778313","111.64138927,29.07770442","111.64158665,29.07763418","111.64182884,29.07754700","111.64192208,29.07751188","111.64192208,29.07751188","111.64192208,29.07751188","111.64192208,29.07751188"};
try {
for(int i =0;i<points.length;i++){
String currentTime ="";
Thread.sleep(3000);
currentTime = DateTimeUtil.getCurrTimeToStr();
String [] point = points[i].split(",");
String lon = point[0];
String lat = point[1];
Bus bus = new Bus();
bus.setPlatenum("A88403");
bus.setLon(lon);
bus.setLat(lat);
bus.setDirection("上行");
bus.setReceivetime(currentTime);
bus.setRoutenum("205");
bus.setSpeed(Integer.toString((int) Math.round(Math.random() *(30)*0.9)+30));
topicMessageSender.sendMessage(bus);
}
} catch (ParseException e) {
e.printStackTrace();
} catch (InterruptedException e1) {
e1.printStackTrace();
}
}
}
(2) MessageSender.java
package com.test.jms.util;
import org.springframework.jms.core.JmsTemplate;
import com.test.jms.model.Bus;
public class MessageSender {
private JmsTemplate jmsTemplate;
public void sendMessage(Bus bus){
jmsTemplate.convertAndSend(bus);
}
public JmsTemplate getJmsTemplate() {
return jmsTemplate;
}
public void setJmsTemplate(JmsTemplate jmsTemplate) {
this.jmsTemplate = jmsTemplate;
}
}
(3) MessageConvertForSys.java //消息转化
package com.test.jms.util;
import java.io.Serializable;
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.ObjectMessage;
import javax.jms.Session;
import javax.jms.TopicPublisher;
import org.apache.activemq.command.ActiveMQObjectMessage;
import org.springframework.jms.support.converter.MessageConversionException;
import org.springframework.jms.support.converter.MessageConverter;
import com.test.jms.model.Bus;
public class MessageConvertForSys implements MessageConverter {
public Message toMessage(Object object, Session session)
throws JMSException, MessageConversionException {
System.out.println("sendMessage:"+object.toString());
ActiveMQObjectMessage msg = (ActiveMQObjectMessage) session.createObjectMessage();
msg.setObject((Serializable) object);
return msg;
}
public Object fromMessage(Message message) throws JMSException,
MessageConversionException {
Bus bus = null;
if(message instanceof ActiveMQObjectMessage){
ActiveMQObjectMessage aMsg = (ActiveMQObjectMessage) message;
bus=(Bus) aMsg.getObject();
}
return bus;
}
}
(4) MessageReceiver.java
package com.test.jms.util;
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.ObjectMessage;
import com.test.jms.flex.BusInfo;
import com.test.jms.model.Bus;
import flex.messaging.MessageBroker;
import flex.messaging.messages.AsyncMessage;
import flex.messaging.util.UUIDUtils;
public class MessageReceiver implements MessageListener {
public void onMessage(Message m) {
System.out.println("[receive message]");
String my = "";
String x = "";
Bus bus= new Bus();
BusInfo sendBus = new BusInfo();
if(m instanceof ObjectMessage){
ObjectMessage objectMessage=(ObjectMessage) m;
try {
bus=(Bus) objectMessage.getObject();
my = bus.getPlatenum();
x = bus.getLat();
sendBus.setDirection(bus.getDirection()) ;
sendBus.setLat(bus.getLat());
sendBus.setLon(bus.getLon());
sendBus.setPlatenum(bus.getPlatenum());
sendBus.setReceivetime(bus.getReceivetime());
sendBus.setRoutenum(bus.getRoutenum());
sendBus.setSpeed(bus.getSpeed());
} catch (JMSException e) {
}
}
System.out.println("公交车编号:"+my +" X:"+x);
MessageBroker messageBroker = MessageBroker.getMessageBroker("_messageBroker");
String clientID = UUIDUtils.createUUID();
//创建AsyncMessage类的对象是为了Flex端用Messaging模式接收消息
AsyncMessage asynMsg = new AsyncMessage();
// 设置消息的地址,这个必须跟Spring配置文件中信道的destination一致
asynMsg.setDestination("market-data-feed");
asynMsg.setHeader("platenum",my);
asynMsg.setClientId(clientID);
asynMsg.setMessageId(UUIDUtils.createUUID());
asynMsg.setTimestamp(System.currentTimeMillis());
asynMsg.setBody(sendBus);
messageBroker.routeMessageToService(asynMsg, null);
}
}
(5) DateTimeUtil.java //时间工具类
package com.test.jms.util;
import java.text.ParseException;
import java.text.ParsePosition;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
/**
*
* @author
*
*/
public class DateTimeUtil {
/**
* 获取当前时间
*
*/
public static Date currentTime() throws ParseException {
Date date = new Date();
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String time = df.format(date);
Date currTime = df.parse(time);
return currTime;
}
/**
* 获取当前日期
*
*/
public static Date currentDate() throws ParseException {
Date date = new Date();
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd");
String time = df.format(date);
Date currTime = df.parse(time);
return currTime;
}
/**
* 获取当前时间
*
*/
public static String getCurrTimeToStr() throws ParseException {
Date date = new Date();
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String currTime = df.format(date);
return currTime;
}
/**
* 获取当前日期
*
*/
public static String getCurrDateToStr() throws ParseException {
Date date = new Date();
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd");
String currTime = df.format(date);
return currTime;
}
/**
* 获取相隔当前时间n秒的时间
*
* @param n
* 相差n秒
* @return
*/
public static String getTime_PrevSecond(int n) {
Calendar calendar = Calendar.getInstance();
calendar.add(Calendar.SECOND, n);
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String prevTime = df.format(calendar.getTime());
return prevTime;
}
/**
* 时间变为字符串形式(yyyy-MM-dd HH:mm:ss)
*
*/
public static String getTimeToString(Date date) {
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String strTime = df.format(date);
return strTime;
}
/**
* 获取现在时间
*
* @return返回字符串格式 yyyy-MM-dd HH:mm:ss
*/
public static String getStringDate() {
Calendar currentTime = Calendar.getInstance();
SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String dateString = formatter.format(currentTime.getTime());
return dateString;
}
/**
* 将短时间格式时间转换为字符串 yyyy-MM-dd
*
* @param dateDate
* @param k
* @return
*/
public static String dateToStr(java.util.Date dateDate) {
SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd");
String dateString = "";
if (dateDate != null) {
dateString = formatter.format(dateDate);
}
return dateString;
}
/**
* 将短时间格式时间转换为字符串 yyyy-MM-dd
*
* @param dateDate
* @param k
* @return
*/
public static String dateToStrMMDD(java.util.Date dateDate) {
SimpleDateFormat formatter = new SimpleDateFormat("MM-dd");
String dateString = "";
if (dateDate != null) {
dateString = formatter.format(dateDate);
}
return dateString;
}
/**
* 将短时间格式字符串转换为时间 yyyy-MM-dd
*
* @param strDate
* @return
*/
public static Date strToDate(String strDate) {
SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd");
ParsePosition pos = new ParsePosition(0);
Date strtodate = formatter.parse(strDate, pos);
return strtodate;
}
}
(6) BusInfo.java
package com.test.jms.flex;
import java.io.Serializable;
public class BusInfo {
public String platenum;
public String speed;
public String direction;
public String lon;
public String lat;
public String receivetime;
public String routenum;
public String getPlatenum() {
return platenum;
}
public void setPlatenum(String platenum) {
this.platenum = platenum;
}
public String getSpeed() {
return speed;
}
public void setSpeed(String speed) {
this.speed = speed;
}
public String getDirection() {
return direction;
}
public void setDirection(String direction) {
this.direction = direction;
}
public String getLon() {
return lon;
}
public void setLon(String lon) {
this.lon = lon;
}
public String getLat() {
return lat;
}
public void setLat(String lat) {
this.lat = lat;
}
public String getReceivetime() {
return receivetime;
}
public void setReceivetime(String receivetime) {
this.receivetime = receivetime;
}
public String getRoutenum() {
return routenum;
}
public void setRoutenum(String routenum) {
this.routenum = routenum;
}
}
(7)
package com.test.jms.model;
import java.io.Serializable;
public class Bus implements Serializable{
public String platenum;
public String speed;
public String direction;
public String lon;
public String lat;
public String receivetime;
public String routenum;
public String getPlatenum() {
return platenum;
}
public void setPlatenum(String platenum) {
this.platenum = platenum;
}
public String getSpeed() {
return speed;
}
public void setSpeed(String speed) {
this.speed = speed;
}
public String getDirection() {
return direction;
}
public void setDirection(String direction) {
this.direction = direction;
}
public String getLon() {
return lon;
}
public void setLon(String lon) {
this.lon = lon;
}
public String getLat() {
return lat;
}
public void setLat(String lat) {
this.lat = lat;
}
public String getReceivetime() {
return receivetime;
}
public void setReceivetime(String receivetime) {
this.receivetime = receivetime;
}
public String getRoutenum() {
return routenum;
}
public void setRoutenum(String routenum) {
this.routenum = routenum;
}
}
(8)ControlServlet1.java
package com.test.jms.servlet;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.context.ApplicationContext;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.WebApplicationContextUtils;
public class ControlServlet1 extends HttpServlet {
public void init() throws ServletException {
}
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doPost(request,response);
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
}
}
(9) DelegatingServletProxy.java
package com.test.jms.servlet;
import java.io.IOException;
import javax.servlet.GenericServlet;
import javax.servlet.Servlet;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.WebApplicationContextUtils;
public class DelegatingServletProxy extends GenericServlet{
private String targetBean;
private Servlet proxy;
public void service(ServletRequest req, ServletResponse res)
throws ServletException, IOException {
proxy.service(req, res);
}
public void init() throws ServletException {
this.targetBean = getServletName();
getServletBean();
proxy.init(getServletConfig());
}
private void getServletBean() {
WebApplicationContext wac = WebApplicationContextUtils.getRequiredWebApplicationContext(getServletContext());
this.proxy = (Servlet) wac.getBean(targetBean);
}
}
6.前台Flex的代码
(1) bus.mxml
<?xml version="1.0" encoding="utf-8"?>
<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009"
xmlns:s="library://ns.adobe.com/flex/spark"
xmlns:mx="library://ns.adobe.com/flex/mx"
minWidth="955" minHeight="600" creationComplete="init();">
<fx:Script>
<![CDATA[
import mx.controls.Alert;
import mx.messaging.ChannelSet;
import mx.messaging.channels.AMFChannel;
import mx.messaging.channels.SecureStreamingAMFChannel;
import mx.messaging.channels.StreamingAMFChannel;
import mx.messaging.events.MessageEvent;
import mx.messaging.events.MessageFaultEvent;
import mx.rpc.events.FaultEvent;
import vo.BusInfo;
protected function init():void{
//消息过滤 只订阅公交车编号为A88403 和A88404。
consumer.selector="platenum in ('A88403','A88404')";
//通过我们定义的consumer对象的订阅方法来订阅消息服务
consumer.subscribe();
}
//按钮点击触发的操作
protected function btnSendMessage_clickHandler(event:MouseEvent):void
{
}
//点击按钮后操作失败的处理
protected function Send_faultHandler(event:FaultEvent):void
{
Alert.show("send is fail,because:"+ event.fault.toString());
}
//收到消息的失败处理
protected function ReceiveMessage_faultHandler(event:FaultEvent):void
{
Alert.show("receive is fail,because:"+ event.fault.toString());
}
//生产者收到消息的错误处理
protected function consumer_faultHandler(event:MessageFaultEvent):void
{
txtMessage.text =" 订阅失败";
}
//生产者收到消息后的处理
protected function consumer_messageHandler(event:MessageEvent):void
{
var bus:BusInfo=BusInfo(event.message.body);
txtMessage.text =txtMessage.text+ "编号:"+bus.platenum+" 方向:"+ bus.direction.toString()+" 坐标"+bus.lon+","+bus.lat+"时间:"+bus.receivetime+" 速度:"+bus.speed+"\n";
}
]]>
</fx:Script>
<fx:Declarations>
<!-- 将非可视元素(例如服务、值对象)放在此处 -->
<!--此处定义一个consumer,需要使用它来订阅消息,订阅了以后才能接收到后台发来的数据。这里的关键是要定义destination,destination的值和后台messaging-config.xml定义的 destination的值相同-->
<s:Consumer channelSet="{cs}" id="consumer" destination="market-data-feed" fault="consumer_faultHandler(event)" message="consumer_messageHandler(event)" />
<!--此处定义个发送消息的类,与后台的发送消息的POJO对应,通过它向ActiveMQ发送消息-->
<s:RemoteObject id="Send" channelSet="{amf_channel}" destination="Sender"
fault="Send_faultHandler(event)"/>
<s:AMFChannel id="myamfChannel"
url="http://192.168.1.22:8088/SpringFlexMQ/messagebroker/amf"/>
<mx:ChannelSet id="amf_channel" channels="{[myamfChannel]}"/>
<s:ChannelSet id="cs">
<s:StreamingAMFChannel url="http://192.168.1.22:8088/SpringFlexMQ/messagebroker/streamingamf"/>
</s:ChannelSet>
</fx:Declarations>
<mx:VBox height="100%" horizontalAlign="center" width="100%" verticalAlign="middle" chromeColor="#9F9F9F" dropShadowVisible="true" backgroundColor="#6A232F" backgroundAlpha="0.6">
<mx:VBox width="438" verticalAlign="middle" horizontalAlign="center" paddingBottom="5" paddingTop="5" paddingLeft="5" paddingRight="5" backgroundColor="#87194D" focusColor="#6C1D2A" cornerRadius="10">
<s:VGroup horizontalAlign="center" gap="11" paddingBottom="5" paddingTop="5" paddingLeft="5" paddingRight="5" contentBackgroundColor="#EDEDED">
<mx:TextArea id="txtMessage" x="62" y="100" width="408" height="249" dropShadowVisible="true" text="" borderColor="#97515D" chromeColor="#F0BABA" contentBackgroundColor="#F2FCFA"/>
<mx:Button id="btnSendMessage" x="0" y="302" label="开始订阅" click="btnSendMessage_clickHandler(event)" chromeColor="#97D2EB" paddingBottom="0"/>
</s:VGroup>
</mx:VBox>
</mx:VBox>
</s:Application>
(2) BusInfo.as
package vo
{
[Bindable][RemoteClass(alias="com.test.jms.flex.BusInfo")]
public class BusInfo
{
public function BusInfo()
{
}
public var platenum:String;
public var speed:String;
public var direction:String;
public var lon:String;
public var lat:String;
public var receivetime:String;
public var routenum:String
}
}
八.
启动ActiveMQ
启动程序
九. 总结
1.通过这样的配置可以将Spring和servlet进行整合。
2.前台接受到了公交车的基本信息就可以在地图上进行展示了,通过坐标点可以在地图上画行车轨迹了。
3.我们可以对信息进行筛选,查看特定的公交信息。
4.jar包下载地址