在上一步中我们已经把数据保存到了内容仓库中,那我们如何确定数据确实保存进去了呢?getBlogList() 这个方法将返回根节点下所有名为blogEntry.的子节点。
public ArrayList getBlogList() throws BlogApplicationException { Session session = JackrabbitPlugin.getSession(); ArrayList blogEntryList = new ArrayList(); Node rootNode = session.getRootNode(); NodeIterator blogEntryNodeIterator = rootNode.getNodes(); while (blogEntryNodeIterator.hasNext()) { Node blogEntry = blogEntryNodeIterator.nextNode(); if (blogEntry.getName().equals("blogEntry") == false) continue; String title = blogEntry.getProperty("title").getString(); String blogContent = blogEntry.getProperty("blogContent").getString(); Value creationTimeValue = (Value) blogEntry.getProperty( "creationTime").getValue(); String userName = blogEntry.getProperty("userName").getString(); BlogEntryDTO blogEntryDTO = new BlogEntryDTO(userName, title, blogContent, creationTimeValue.getDate()); blogEntryList.add(blogEntryDTO); } return blogEntryList; }
一旦你获得了根节点这个对象,你就可以通过调用getNodes()这个方法来获取它所有的子节点。如果这个节点没有子节点,将返回一个空的NodeIterator 对象。我们可以遍历这个NodeIterator 对象来获得名为blogEntry 的节点集合,然后通过getProperty()方法来获得节点上的属性,即我们保存的真实数据。getProperty()方法返回Value对象的一个实例。因为存储数据类型的不同,所以返回的Value对象实例是不同的。根据不同的数据类型,你应该调用特定的方法来获取数据,比如getString()来获取字符串,而getDate()获得一个日期。
查找内容(用XPath的方式)
JSR-170定义了两种方式来查找内容(也可以理解为查找节点)。一种使用XPath语法,另一种使用SQL语法。JSR-170要求Level 1必须实现XPath的方式,而SQL的方式则作为一个可选的功能。
XPath原本是一种设计用来查找XML元素的语言。因为我们的workspace是树状的结构,很像XML。所以XPath语法非常适合于在这里查找内容。下面的代码演示了通过作者名来查找节点。
Session session = JackrabbitPlugin.getSession(); Workspace workSpace = session.getWorkspace(); QueryManager queryManager = workSpace.getQueryManager(); StringBuffer queryStr = new StringBuffer( "//blogEntry[@"+PROP_BLOGAUTHOR +"= "); queryStr.append(userName); queryStr.append("]"); Query query = queryManager.createQuery(queryStr.toString(), Query.XPATH); QueryResult queryResult = query.execute(); NodeIterator queryResultNodeIterator = queryResult.getNodes(); while (queryResultNodeIterator.hasNext()) { Node blogEntry = queryResultNodeIterator.nextNode(); String title = blogEntry.getProperty(PROP_TITLE).getString(); String blogContent = blogEntry.getProperty(PROP_BLOGCONTENT).getString(); Value creationTimeValue = (Value) blogEntry.getProperty( PROP_CREATIONTIME).getValue(); BlogEntryDTO blogEntryDTO = new BlogEntryDTO(userName, title, blogContent, creationTimeValue.getDate()); blogEntryList.add(blogEntryDTO); }
首先获得session 对象,通过它获得它连接的workspace,然后就可以通过workspace获得这个workspace的QueryManager 。QueryManager 接口定义了很多用来查询的方法。接下来我们要做的是创建一条查询语句。我们这里这样写"//blogEntry[@blogAuthor=<bloggerName>"。这句话的意思是查找所有名为blogEntry ,含有blogAuthor 属性且属性值为<bloggerName>的节点。具体可以看JSR-170规范。
通过queryManagers createQuery()方法创建一个查询对象,这个方法需要两个参数,一个是我们的查询语句,另一个是查询的方式,这里使用XPath。获得这个Query 查询对象后,调用它的execute() 方法开始执行查询,返回一个QueryResult 对象。注意,查询的结果受到当前session的限制,换句话说,就是如果这个session没有权限查看一个特定的节点,哪怕这个节点满足我们查询的条件,在我们的查询结果里也是看不到这个节点的。所有的查询数据来自于该workspace已经持久化的数据,哪些已经改变但还没有通过session.save()(item.save())持久化到workspace的数据不在查询之列。获得QueryResult 对象后,我们就可以通过调用getNodes()方法来获得符合查询条件的节点的一个遍历。
剩下的两个未实现的方法是updateBlogEntry() 和 removeBlogEntry(),它们实现起来都很简单。我们把BOLG 标题作为主键,通过标题来获得相关的节点。在updateBlogEntry()方法里,我们直接设定需要改变的属性;在 removeBlogEntry()方法里,我们获得目标节点后直接在节点上调用remove()方法。最后别忘了一定要调用session.save()方法把我们改变的数据持久化。
处理二进制内容
对内容仓库来说,处理二进制内容是个很基本的要求,比如说图片。现在我们的示例程序容许给每个BLOG附加一张图片。下面分别是附加图片和获取图片的方法。
public void attachFileToBlogEntry(String blogTitle, InputStream uploadInputStream) throws BlogApplicationException { Session session = JackrabbitPlugin.getSession(); Node blogEntryNode = getBlogEntryNode(blogTitle, session); blogEntryNode.setProperty(PROP_ATTACHMENT, uploadInputStream); session.save(); } public InputStream getAttachedFile(String blogTitle) throws BlogApplicationException { InputStream attachFileIS = null; Node blogEntryNode = getBlogEntryNode(blogTitle); Value attachFileValue = (Value) blogEntryNode.getProperty(PROP_ATTACHMENT).getValue(); attachFileIS = attachFileValue.getStream(); return attachFileIS; }
正如你看到的那样,我们的代码在处理二进制内容和一般内容间并没有什么太大的区别。仅仅一点不同的是你要通过InputStream 对象来保存和获取二进制数据。在我们的配置文件里关于persistent manager会有一个externalBLOBs 属性。把这个属性设为true, 图片将会保存在文件里,相反则会保存在数据库的blob字段里。
总结
到这里,我们对 JSR-170, Jackrabbit以及如何使用 JSR-170 API开发一个简单的应用程序都有了大概的了解。我们的讨论更多的在于基础。相信大家一定会对内容仓库有个初步的认识