当前位置: 代码迷 >> Web前端 >> 利用CXF公布restful WebService的注意事项(服务器端)
  详细解决方案

利用CXF公布restful WebService的注意事项(服务器端)

热度:510   发布时间:2012-09-19 13:43:54.0
利用CXF发布restful WebService的注意事项(服务器端)
研究了两天CXF对restful的支持。
  现在,想实现一个以
http://localhost:9999/roomservice 为入口,
http://localhost:9999/roomservice/room为房间列表,
http://localhost:9999/roomservice/room/001/ 为001号房间的信息,
http://localhost:9999/roomservice/room/001/person 为在001号房间主的人的列表

  实现用HTTP请求对以上资源的CRUD。

  首先建立room,person的POJO,这里只有一点需要注意:
package com.DAO;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement(name="Person")
public class Person {
	private String name;
	private String sex;
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public String getSex() {
		return sex;
	}
	public void setSex(String sex) {
		this.sex = sex;
	}
	
}



  一定要在类的前边加上annotation ,这样才能让这个person的信息在POJO和XML之间转换。Room同理:
import java.util.Map;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement(name="Room")
public class Room {
	public Room()
	{
		persons=new HashMap<String,Person>();
	}
	String id;
	Map<String,Person> persons;
	
	public String getId() {
		return id;
	}
	public void setId(String id) {
		this.id = id;
	}
	public Map<String, Person> getPersons() {
		return persons;
	}
	public void setPersons(Map<String, Person> persons) {
		this.persons = persons;
	}
}


POJO有了,接下来要写DAO,由于主要是为了学习restful,为了方便,不必要将数据持久化到数据库,而是存在一个静态的HashMap中:

package com.DAO;

import java.util.HashMap;
import java.util.Map;

public class RoomDAO {
	private static Map<String, Room> rooms;
	static {
		rooms = new HashMap<String, Room>();
		
		Person p1=new Person();
		p1.setName("Boris");
		p1.setSex("male");
		
		
		
		Room r=new Room();
		r.setId("001");
		r.getPersons().put(p1.getName(), p1);
		rooms.put("001", r);
	}

	public static void addRoom(Room room) {
		rooms.put(room.getId(), room);
	}

	public static void deleteRoom(String id) {
		if (rooms.containsKey(id)) {
			rooms.remove(id);
		}

	}

	public static void updateRoom(String id,Room room) {
		rooms.remove(id);
		rooms.put(room.getId(), room);
	}

	public static Room getRoom(String id) {
		if (rooms.containsKey(id)) {
			return rooms.get(id);
		} else {
			return null;
		}
	}
	/*operations to persons*/
	public static void addPerson(String id_room,Person person) {
		if(rooms.containsKey(id_room))
		{
			Room room=rooms.get(id_room);
			room.getPersons().put(person.getName(), person);
		}
	}
	
	public static Rooms getRooms()
	{
		return new Rooms();
	}
	
	public static void deletePerson(String id_room,String name)
	{
		if(rooms.containsKey(id_room))
		{
			Room room=rooms.get(id_room);
			room.getPersons().remove(name);
		}
	}
	
	public static Map<String, Room> getMapOfRooms()
	{
		return rooms;
	}
}




接下来是重点,如果想发布restful webservice,要通过一个叫JAXRSServerFactoryBean的类来发布。这个类有几个方法是要了解的:

public void setResourceClasses(Class... classes);
那一系列的Class类型的参数,是告诉这个类,发布服务时,会用到的POJO(就像上边写的Room,Person)。

public void setAddress(String address);
这个方法是告诉这个类,服务的地址,比如"http://localhost:9999"

public void setServiceBeans(Object... beans)
这里是重点,这个方法,要给这个用来发布服务的类一个Service bean.这个Bean是我们要手动编写的,作用是告诉服务,收到什么样的请求,应该怎么样处理。

现在,先来编写这个Service bean:

package com.server;

import javax.ws.rs.Consumes;
import javax.ws.rs.DELETE;
import javax.ws.rs.GET;
import javax.ws.rs.POST;
import javax.ws.rs.PUT;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import javax.ws.rs.Produces;

import com.DAO.Person;
import com.DAO.Room;
import com.DAO.RoomDAO;
import com.DAO.Rooms;

@Path("/roomservice")
@Produces("application/xml")
public class RoomService {
	
	@GET
	@Path("/room/{id}")
	@Consumes("application/xml")
	public Room getRoom(@PathParam("id")String id )
	{
		System.out.println("get room by id= "+id);
		Room room=RoomDAO.getRoom(id);
		return room;
	}
	@GET
	@Path("/room")
	@Consumes("application/xml")
	public Rooms getAllRoom()
	{
		System.out.println("get all room");
		Rooms rooms=RoomDAO.getRooms();
		return rooms;
	}
	
	@POST
	@Path("/room")
	@Consumes("application/xml")
	public void addRoom(Room room)
	{
		System.out.println("add room which id is:"+room.getId());
		RoomDAO.addRoom(room);
	}
	@PUT
	@Path("/room/{id}")
	@Consumes("application/xml")
	public void updateRoom(@PathParam("id")String id,Room room)
	{
		System.out.println("update room which original id is:"+room.getId());
		RoomDAO.updateRoom(id,room);
	}
	@DELETE
	@Path("/room/{id}")
	@Consumes("application/xml")
	public void deleteRoom(@PathParam("id")String id)
	{
		System.out.println("remove room by id= "+id);
		RoomDAO.deleteRoom(id);
	}
	@POST
	@Path("/room/{id}")
	@Consumes("application/xml")
	public void addPerson(@PathParam("id") String id,Person person)
	{
		System.out.println("add person who's name is:"+person.getName());
		RoomDAO.addPerson(id, person);
	}
	@DELETE
	@Path("/room/{id}/{name}")
	@Consumes("application/xml")
	public void deletePerson(@PathParam("id")String id,@PathParam("name")String name)
	{
		System.out.println("remove person who's name is: "+name);
		RoomDAO.deletePerson(id, name);
	}
}





需要注意:每个方法之前,要用annotation声明http请求的method类型,比如GET,DELETE,POST, PUT.

@Produces("application/xml")我还没弄清楚到底是声明的接受格式还是返回格式,还是其他。

@Path("/room/{id}")中的id是一个参数,应该在方法的参数列表中声明:
public void deletePerson(@PathParam("id")String id,@PathParam("name")String name)
这样就能得到URL中的id了。

现在,这些房间被资源化了,id为001的房间被资源化为一个URL,那地址应该是
http:{服务器地址}:{端口}/roomservice/rrom/001  

现在,创建一个Server:
package com.server;

import org.apache.cxf.jaxrs.JAXRSServerFactoryBean;

import com.DAO.Person;
import com.DAO.Room;
import com.DAO.Rooms;

public class Server {

	public static void main(String[] args) {
		RoomService service = new RoomService();

		// Service instance
		JAXRSServerFactoryBean restServer = new JAXRSServerFactoryBean();
		restServer.setResourceClasses(Room.class,Person.class,Rooms.class);
		restServer.setServiceBeans(service);
		restServer.setAddress("http://localhost:9999/");
		restServer.create();
	}
}



现在,服务已经发布成功了,在浏览器输入http://localhost:9999/roomservice/room/001  得到结果:
<room>
<id>001</id>
?
<persons>
?
<entry>
<key>Boris</key>
?
<value>
<name>Boris</name>
<sex>male</sex>
</value>
</entry>
</persons>
</room>


如果用浏览器去访问,发送的http请求只能所GET,因此如果想对数据进行操作,必须写一个客户端。
在写客户端之前,有一个问题:
在浏览器输入http://localhost:9999/roomservice/room/
什么都看不到,可是,我想要得到房间列表。但是,cxf发布restful只认你给他的类的class。所以你想让服务器返回一个room的列表给客户端,是不行的。所以,必须想别的办法,我是又写了一个POJO,这个POJO里只有一个属性,就是一个存放所有room的Map:
package com.DAO;

import java.util.Map;

import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement(name="rooms")
public class Rooms {
	Map<String,Room> rooms;
	public Rooms()
	{
		rooms=RoomDAO.getMapOfRooms();
	}
	public Map<String, Room> getRooms() {
		return rooms;
	}
	public void setRooms(Map<String, Room> rooms) {
		this.rooms = rooms;
	}
}

这样,然后再把DAO的方法加上:
@GET
	@Path("/room")
	@Consumes("application/xml")
	public Rooms getAllRoom()
	{
		System.out.println("get all room");
		Rooms rooms=RoomDAO.getRooms();
		return rooms;
	}


这样就能以list的形式显示出所有room了。
访问http://localhost:9999/roomservice/room/
结果如下:
<rooms>
?
<rooms>
?
<entry>
<key>006</key>
?
<value>
<id>006</id>
<persons/>
</value>
</entry>
?
<entry>
<key>001</key>
?
<value>
<id>001</id>
?
<persons>
?
<entry>
<key>Boris</key>
?
<value>
<name>Boris</name>
<sex>male</sex>
</value>
</entry>
</persons>
</value>
</entry>
</rooms>
</rooms>



关于客户端 http://borissun.iteye.com/blog/766029



1 楼 zky314 2011-08-18  
undefined
2 楼 itcyt 2011-12-28  
高手这里怎么加上安全认证啊?求实例...
3 楼 82934162 2012-01-09  
itcyt 写道
高手这里怎么加上安全认证啊?求实例...

你说的如果是SSL安全认证的话。和这个文章写的没有关系的。这里只是说明了如何发布和访问restful webservice。

要做SSL的话,应该是server和client先握手,然后再做这两篇文章里的内容。可以认为这两篇文章是在做业务,SSL是在进行业务之前的一个安全认证。因此建议看下SSL。

至于实例,我这里已经没有了。因为转做iOS开发了,好久没做过java了。。用iOS还没有做过restful
4 楼 itcyt 2012-01-10  
82934162 写道
itcyt 写道
高手这里怎么加上安全认证啊?求实例...

你说的如果是SSL安全认证的话。和这个文章写的没有关系的。这里只是说明了如何发布和访问restful webservice。

要做SSL的话,应该是server和client先握手,然后再做这两篇文章里的内容。可以认为这两篇文章是在做业务,SSL是在进行业务之前的一个安全认证。因此建议看下SSL。

至于实例,我这里已经没有了。因为转做iOS开发了,好久没做过java了。。用iOS还没有做过restful


官网上说的那个认证也太麻烦了吧,我就是想问问怎么能简单的实现,或者说其他的jax-rs的框架有没有什么好的解决方案,不过还是得谢谢你!
5 楼 stevenjohn 2012-08-25  
能不能传一下源代码???
6 楼 82934162 2012-08-28  
stevenjohn 写道
能不能传一下源代码???

不好意思 这文章是两年前的了。。。我已经不做java一年多了。。源码真的没有 不过你可以在网上找到很多相关的源码。
  相关解决方案