1 注意同时声明remote.local的问题
我发现在jboss 4.2.2.gA上,如果这样的话,会出错
@Stateless
@Remote
@local
是JOBSS报错的
javax.ejb.EJBException: Local and Remote Interfaces cannot have duplicate interface for bean PhysicalFileManagerBean
不知为啥,后来查了一下,要这样改才行,具体指明是哪一个接口
@Stateless
@Remote(UserManager.class)
@Local(UserManager.class)
2 注意序列化问题
比如有个user对象要保存,则必须要序列化,如
public class User implements Serializable{ ...}
public void saveUser(User user) {xxx.save(user)}
3 local/remote的区别
组件接口类型影响组件方法被客户端调用时的参数传递方式。
远程调用中的参数是按值传递的,传递的是对象的拷贝,但本地调用中参数的传递则是按引用传递的,与Java编程语言中的正常的方法调用相同。
远程调用中,客户端与组件相对比较独立。因为客户端与组件操作不同的参数对象拷贝,任何一方对数据的修改,不会影响另一方的数据。
在本地调用中,客户端和组件修改的是同一个对象。在编写Enterprise Bean时,应避免使用由这种传递方式带来的副作用,因为,当应用规模增长,客户端被分布到不同的物理服务器,参数的传递方式亦随之改变。
决定一个组件应该提供远程访问还是本地访问,一般有以下一些因素。
- 容器管理关系
如果某Entity Bean被其他Entity Bean所关联,则此Entity Bean必须提供本地访问接口。
- 相关Bean之间的耦合程度
耦合程度高的Bean之间存在互相依存的关系。因此,这一类Bean一般需要提供本地访问接口,以形成一个逻辑上的单元,同时提供整个单元内交互的效率。
- 客户端类型
如Bean将被JavaEE应用客户端所访问,则应该提供远程访问接口。如果客户端是Web组件或其他Bean,提供何种类型的访问接口决定于这些组件的分布状况。
- 组件分布
JavaEE应用可以分布在多个计算机,为提供这种分布上的可伸缩性,将被分布到不同位置的组件所访问的Bean,应该提供远程访问接口。
尽管一般Bean只提供一种类型的客户访问接口,但事实上,有一些Bean仍需要同时提供两种类型的接口
一个很好的例子,比如
public void saveUser(User user) {
System.out.println("User[name="+user.getName()+"]已被保存");
user.setId(1);
}
如果是remote调用的话:
UserManager ejb = (UserManager)ctx.lookup("UserManagerBean/remote");
User user = new User();
user.setName("张三");
ejb.saveUser(user);
System.out.println("用户【name="+user.getName()+",id="+user.getId()+"】已被成功保存!");
输出的id依然为0,因为EJB改变了值,不会传给客户端的,
如果用local的话,则可以输出id=1
4 标注的问题
如果ejb实现了多个接口,可以这样声明
/**
* 定义方式1:
* @Remote(value={Ejb04Interface01.class,Ejb04Interface02.class,Ejb04Interface03.class})
* @Local(Ejb04Interface04.class)
*/
@Stateless(name="Ejb04")
public class Ejb04Bean implements Ejb04Interface01, Ejb04Interface02,
Ejb04Interface03, Ejb04Interface04 {
当然,也可以在各自的接口里加上@remote或@Local
5 webservice问题
直接加@webservice,详细的以后再小结之
6 简单的JMS小结
P2P 和发布/订阅者模式,和EJB2的一样,这里就不列出来了.例子
p2p的:
mdb端:
import javax.ejb.ActivationConfigProperty;
import javax.ejb.MessageDriven;
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.TextMessage;
@MessageDriven(
activationConfig = {
@ActivationConfigProperty(propertyName="destinationType",propertyValue="javax.jms.Queue"),
@ActivationConfigProperty(propertyName="destination",propertyValue="queue/myqueue")
}
)
public class MyMDBBean implements MessageListener {
public void onMessage(Message msg) {
try {
TextMessage textMessage = (TextMessage)msg;
System.out.println("MyMDBBean被调用了!【"+textMessage.getText()+"】");
} catch (JMSException e) {
e.printStackTrace();
}
}
}
客户端消费:
public static void main(String[] args) throws Exception{
InitialContext ctx = new InitialContext();
//获取ConnectionFactory对象
QueueConnectionFactory factory = (QueueConnectionFactory)ctx.lookup("ConnectionFactory");
//创建QueueConnection对象
QueueConnection connection = factory.createQueueConnection();
//创建QueueSession对象,第一个参数表示事务自动提交,第二个参数标识一旦消息被正确送达,将自动发回响应
QueueSession session = connection.createQueueSession(false, QueueSession.AUTO_ACKNOWLEDGE);
//获得Destination对象
Queue queue = (Queue)ctx.lookup("queue/myqueue");
//创建文本消息
TextMessage msg = session.createTextMessage("世界,你好");
//创建发送者
QueueSender sender = session.createSender(queue);
//发送消息
sender.send(msg);
//关闭会话
session.close();
System.out.println("消息已发送!");
}
}
发布订阅者的话,其实就是把上面的queue改成TOPIC
import javax.ejb.ActivationConfigProperty;
import javax.ejb.MessageDriven;
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.TextMessage;
@MessageDriven(
activationConfig = {
@ActivationConfigProperty(propertyName="destinationType",propertyValue="javax.jms.Topic"),
@ActivationConfigProperty(propertyName="destination",propertyValue="topic/mytopic")
}
)
public class MyTopicMDBBean implements MessageListener {
public void onMessage(Message msg) {
try {
TextMessage textMessage = (TextMessage)msg;
System.out.println("MyTopicMDBBean被调用了!【"+textMessage.getText()+"】");
} catch (JMSException e) {
e.printStackTrace();
}
}
}
消费端:
public static void main(String[] args) throws Exception{
InitialContext ctx = new InitialContext();
//获取ConnectionFactory对象
TopicConnectionFactory factory = (TopicConnectionFactory)ctx.lookup("ConnectionFactory");
//创建TopicConnection对象
TopicConnection connection = factory.createTopicConnection();
//创建TopicSession对象,第一个参数表示事务自动提交,第二个参数标识一旦消息被正确送达,将自动发回响应
TopicSession session = connection.createTopicSession(false, TopicSession.AUTO_ACKNOWLEDGE);
//获得Destination对象
Topic topic = (Topic)ctx.lookup("topic/mytopic");
//创建文本消息
TextMessage msg = session.createTextMessage("世界,你好");
//创建发布者
TopicPublisher publisher = session.createPublisher(topic);
//发送消息
publisher.publish(msg);
//关闭会话
session.close();
System.out.println("消息已发送!");
}