TT_solar服务器搭建、搜索功能实现

1. 需要做的事情

l Solr服务的搭建

l 搜索功能的实现

1.1. 系统架构

2. Solr概述

2.1. 什么是Solr

Solr Apache下的一个顶级开源项目,采用Java开发,它是基于Lucene的全文搜索服务器。Solr提供了比Lucene更为丰富的查询语言,同时实现了可配置、可扩展,并对索引、搜索性能进行了优化。

Solr可以独立运行,运行在JettyTomcat等这些Servlet容器中,Solr 索引的实现方法很简单,用 POST 方法向 Solr 服务器发送一个描述 Field 及其内容的 XML 文档,Solr根据xml文档添加、删除、更新索引 。Solr 搜索只需要发送 HTTP GET 请求,然后对 Solr 返回Xmljson等格式的查询结果进行解析,组织页面布局。Solr不提供构建UI的功能,Solr提供了一个管理界面,通过管理界面可以查询Solr的配置和运行情况。

2.2. 下载

Solr官方网站(http://lucene.apache.org/solr/ )下载Solr4.10.3,根据Solr的运行环境,Linux下需要下载lucene-4.10.3.tgzwindows下需要下载lucene-4.10.3.zip

Solr使用指南可参考:https://wiki.apache.org/solr/FrontPage。

下载lucene-4.10.3.zip并解压:

 

binsolr的运行脚本

contribsolr的一些贡献软件/插件,用于增强solr的功能。

dist:该目录包含build过程中产生的warjar文件,以及相关的依赖文件。

docssolrAPI文档

examplesolr工程的例子目录:

l example/solr:

该目录是一个包含了默认配置信息的SolrCore目录。

l example/multicore

该目录包含了在Solrmulticore中设置的多个Core目录。

l example/webapps:

    该目录中包括一个solr.war,该war可作为solr的运行实例工程。

licensessolr相关的一些许可信息

3. Solr的安装及配置

3.1. 运行环境

solr 需要运行在一个Servlet容器中,Solr4.10.3要求jdk使用1.7以上,Solr默认提供Jettyjava写的Servlet容器),本教程使用Tocmat作为Servlet容器,环境如下:

SolrSolr4.10.3

Jdkjdk1.7.0_72

Tomcatapache-tomcat-7.0.53

3.2. Solr整合tomcat

1. dist\solr-4.10.3.war拷贝到Tomcatwebapp目录下改名为solr.war

2. 启动tomcat后,solr.war自动解压,将原来的solr.war删除。

3. 拷贝example\lib\ext 目录下所有jar包到Tomcatwebapp\solr\WEB-INF\lib目录下

 

4. 拷贝log4j.properties文件

  Tomcatwebapps\solr\WEB-INF目录中创建文件 classes文件夹,

复制Solr目录下example\resources\log4j.propertiesTomcatwebapps\solr\WEB-INF\classes目录

5. 创建solrhome及配置solrcoresolrconfig.xml文件

6. 修改Tomcat目录 下webapp\solr\WEB-INF\web.xml文件,如下所示:

设置Solr home

 

<!--配置jndi告诉solr工程我们的solrhome的位置-->

<env-entry>

<env-entry-name>solr/home</env-entry-name>

<env-entry-value>D:/temp/solr/solrhome</env-entry-value>

<env-entry-type>java.lang.String</env-entry-type>

</env-entry>

4. Solr界面功能

 

 

 

 

 

5. 安装中文分词器

5.1. 安装步骤

5.1.1. 第一步:配置IKAnalyzerjar

拷贝IKAnalyzer的文件到TomcatSolr目录中

IKAnalyzer2012FF_u1.jar拷贝到 Tomcatwebapps/solr/WEB-INF/lib 下。

5.1.2. 第二步:IKAnalyzer的配置文件

Tomcatwebapps/solr/WEB-INF/下创建classes目录

将IKAnalyzer.cfg.xmlext_stopword.dic  mydict.dic  copy到 Tomcat

webapps/solr/WEB-INF/classes

注意:ext_stopword.dic mydict.dic必须保存成无BOMutf-8类型。

5.1.3. 第三步:修改schema.xml文件

修改schema.xml文件

修改Solrschema.xml文件,添加FieldType

<fieldType name="text_ik" class="solr.TextField">

  <analyzer class="org.wltea.analyzer.lucene.IKAnalyzer"/>

</fieldType>

 

5.1.4. 第四步:设置业务系统Field

设置业务系统Field

<field name="item_title" type="text_ik" indexed="true" stored="true"/>

<field name="item_sell_point" type="text_ik" indexed="true" stored="true"/>

<field name="item_price"  type="long" indexed="true" stored="true"/>

<field name="item_image" type="string" indexed="false" stored="true" />

<field name="item_category_name" type="string" indexed="true" stored="true" />

<field name="item_desc" type="text_ik" indexed="true" stored="false" />

 

<field name="item_keywords" type="text_ik" indexed="true" stored="false" multiValued="true"/>

<copyField source="item_title" dest="item_keywords"/>

<copyField source="item_sell_point" dest="item_keywords"/>

<copyField source="item_category_name" dest="item_keywords"/>

<copyField source="item_desc" dest="item_keywords"/>

 

6. 搭建taotao-search服务

6.1. 创建工程

6.2. 配置工程

工程配置参考taotao-rest,基本相同,不需要引入Jedis jar包,需要sorlJjar包。其他都相同。

 

6.2.1. 添加solrj依赖关系

<!-- solr客户端 -->

<dependency>

<groupId>org.apache.solr</groupId>

<artifactId>solr-solrj</artifactId>

</dependency>


6.3. 商品信息导入索引库

6.3.1. 需求分析

需要把数据库中的商品信息导入索引库,需要商品的id、商品的名称、商品的卖点、商品的价格、商品的图片、商品的分类名称。

原则:需要展示给用户的字段、需要搜索的字段需要添加到索引库。

6.3.2. Sql语句

SELECT

a.id,

a.title,

a.sell_point,

a.price,

a.image,

b. NAME category_name

FROM

tb_item a

LEFT JOIN tb_item_cat b ON a.cid = b.id

 

6.3.3. POJO定义

/**

 * 商品信息POJO

 * <p>Title: Item</p>

 * <p>Description: </p>

 * <p>Company: www.itcast.com</p> 

 * @author入云龙

 * @date2015年7月28日下午3:06:16

 * @version 1.0

 */

public class Item {

private Long id;

private String title;

private String sell_point;

private Long price;

private String image;

private String category_name;

}

 

6.3.4. Mapper

<?xml version="1.0" encoding="UTF-8" ?>

<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" >

<mapper namespace="com.taotao.search.mapper.ItemMapper" >

<select id="getItemList" resultType="com.taotao.search.pojo.Item">

SELECT

a.id,

a.title,

a.sell_point,

a.price,

a.image,

b. NAME category_name

FROM

tb_item a

LEFT JOIN tb_item_cat b ON a.cid = b.id

</select>

</mapper>

 

6.3.5. Service

@Service

public class ItemServiceImpl implements ItemService {

 

@Autowired

private ItemMapper itemMapper;

@Autowired

private SolrServer solrServer;

@Override

public TaotaoResult importItemToIndex() throws Exception {

//查询商品列表

List<Item> itemList = itemMapper.getItemList();

//将商品列表导入solr

for (Item item : itemList) {

SolrInputDocument document = new SolrInputDocument();

document.addField("id"item.getId());

document.addField("item_title"item.getTitle());

document.addField("item_sell_point"item.getSell_point());

document.addField("item_price"item.getPrice());

document.addField("item_image"item.getImage());

document.addField("item_category_name"item.getCategory_name());

//将文档写入索引库

solrServer.add(document);

}

//提交修改

solrServer.commit();

return TaotaoResult.ok();

}

 

}

 

6.3.6. Controller

@Controller

@RequestMapping("/manager")

public class ItemController {

 

@Autowired

private ItemService itemService;

@RequestMapping("/importall")

@ResponseBody

public TaotaoResult importAll() {

TaotaoResult result = null;

try {

result = itemService.importItemToIndex();

catch (Exception e) {

e.printStackTrace();

return TaotaoResult.build(500, ExceptionUtil.getStackTrace(e));

}

return result;

}

}

 

6.3.7. Pom.xml

需要在pom文件的build节点中添加如下信息,否则mapper映射文件不会被发布,从而发生异常。

<resources>

            <resource>

                <directory>src/main/java</directory>

                <includes>

                    <include>**/*.properties</include>

                    <include>**/*.xml</include>

                </includes>

                <filtering>false</filtering>

            </resource>

            <resource>

                <directory>src/main/resources</directory>

                <includes>

                    <include>**/*.properties</include>

                    <include>**/*.xml</include>

                </includes>

                <filtering>false</filtering>

            </resource>

        </resources>

6.3.8. applicationContext-dao.xml

扫描包添加taotao-search工程中的包:

<!-- 加载mapper文件 -->

<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">

<property name="basePackage" value="com.taotao.mapper,com.taotao.search.mapper"></property>

</bean>

 

7. 实现商品搜索功能

实现商品搜索功能需要两步实现:

第一步:在taotoa-search工程中发布服务

第二步:在taotao-portal中调用服务并展示结果。

 

7.1. 发布搜索服务

7.1.1. Dao

返回值SearchResult

public class SearchResult {

 

private Long recordCount;

private List<Item> itemList;

private Integer pageCount;

private Integer curPage;

}

 

Dao

@Service

public class ItemSearchDaoImpl implements ItemSearchDao {

@Autowired

private SolrServer solrServer;

 

@Override

public SearchResult searchItem(SolrQuery solrQuerythrows Exception {

//根据查询条件搜索索引库

QueryResponse response = solrServer.query(solrQuery);

//取商品列表

SolrDocumentList documentList = response.getResults();

//商品列表

List<Item> itemList = new ArrayList<>();

for (SolrDocument solrDocument : documentList) {

Item item = new Item();

item.setId((Long) solrDocument.get("id"));

//取高亮显示

Map<String, Map<String, List<String>>> highlighting = response.getHighlighting();

List<String> list = highlighting.get(solrDocument.get("id")).get("item_title");

String title = "";

if (null != list && !list.isEmpty()) {

title = list.get(0);

else {

title = (String) solrDocument.get("item_title");

}

item.setTitle(title);

item.setPrice((Long) solrDocument.get("item_price"));

item.setSell_point((String) solrDocument.get("item_sell_point"));

item.setImage((String) solrDocument.get("item_image"));

item.setCategory_name((String) solrDocument.get("item_category_name"));

itemList.add(item);

}

SearchResult result = new SearchResult();

//商品列表

result.setItemList(itemList);

//总记录数据

result.setRecordCount(documentList.getNumFound());

return result;

}

 

}

 

7.1.2. Service

@Service

public class ItemSearchServiceImpl implements ItemSearchService {

 

@Value("${SEARCH_RESULT_PAGE_SIZE}")

private Integer PAGE_SIZE;

@Autowired

private ItemSearchDao itemSearchDao;

@Override

public SearchResult searchItem(String queryString, Integer pagethrows Exception {

//创建一个查询对象

SolrQuery solrQuery = new SolrQuery();

//查询条件

if (StringUtils.isBlank(queryString)) {

solrQuery.setQuery("*:*");

else {

solrQuery.setQuery(queryString);

}

//分页条件

if (page == null) {

page = 1;

}

solrQuery.setStart((page -1) * PAGE_SIZE);

solrQuery.setRows(PAGE_SIZE);

//高亮显示

solrQuery.setHighlight(true);

//设置高亮显示的域

solrQuery.addHighlightField("item_title");

//高亮显示前缀

solrQuery.setHighlightSimplePre("<em style=\"color:red\">");

//后缀

solrQuery.setHighlightSimplePost("</em>");

//设置默认搜索域

solrQuery.set("df""item_keywords");

//执行查询

SearchResult result = itemSearchDao.searchItem(solrQuery);

//计算分页

Long recordCount = result.getRecordCount();

int pageCount = (int) (recordCount / PAGE_SIZE);

if (recordCount % PAGE_SIZE > 0) {

pageCount++;

}

result.setPageCount(pageCount);

result.setCurPage(page);

return result;

}

 

}

7.1.3. Controller

@Controller

public class ItemSearchController {

@Autowired

private ItemSearchService itemSearchService;

 

@RequestMapping("/q")

@ResponseBody

public TaotaoResult search(@RequestParam(value = "kw") String queryString,

@RequestParam(value = "page", defaultValue = "1") Integer page) {

if (StringUtils.isBlank(queryString)) {

return TaotaoResult.build(400, "查询条件是必须的参数");

}

SearchResult result = null;

try {

result = itemSearchService.searchItem(queryStringpage);

 

catch (Exception e) {

e.printStackTrace();

return TaotaoResult.build(500, ExceptionUtil.getStackTrace(e));

}

return TaotaoResult.ok(result);

}

}

7.1.4. 扫描dao所在的包

applicationContext-dao.xml文件中添加如下内容:

 

7.1.5. 解决get请求乱码问题

controller中添加字符串编码转换逻辑:

Tomcat默认的编码为ISO8859-1,需要转换成utf-8的编码。

7.2. 调用服务实现搜索功能

7.2.1. 功能分析

Taotao-portal展示首页,用户在首页输入查询内容提交至taotao-portaltaotao-portal调用taotao-search提供的搜索服务,得到商品列表。在taotao-portal中渲染商品列表展示搜索结果页面。

 

7.2.2. Dao

木有

7.2.3. Service

使用HttpClientUtil工具类调用搜索服务,返回一个json数据,需要把json数据转换成TaotaoResult对象。

@Service

public class SearchServiceImpl implements SearchService {

 

@Value("${SEARCH_BASE_URL}")

private String SEARCH_BASE_URL;

@Override

public SearchResult searchItemList(String queryString, Integer pagethrows Exception {

//查询参数

Map<String, String> param = new HashMap<>();

param.put("kw"queryString);

param.put("page"page==null?"1":page.toString());

//调用taotao-search提供的搜索服务

String resultString = HttpClientUtil.doGet(SEARCH_BASE_URLparam);

//转换成taotaoResult对象

TaotaoResult taotaoResult = TaotaoResult.formatToPojo(resultString, SearchResult.class);

SearchResult searchResult = null;

//查询成功

if (taotaoResult.getStatus() == 200) {

//取查询结果

searchResult = (SearchResult) taotaoResult.getData();

}

return searchResult;

}

 

}

 

7.2.4. Controller

@Controller

public class SearchController {

 

@Autowired

private SearchService searchService;

@RequestMapping("/search")

public String searchItemList(@RequestParam(value="q")String queryString, Integer page, Model modelthrows Exception {

//字符串转码

queryString = new String(queryString.getBytes("ISO8859-1"), "UTF-8");

SearchResult searchResult = searchService.searchItemList(queryStringpage);

model.addAttribute("itemList"searchResult.getItemList());

model.addAttribute("query"queryString);

model.addAttribute("totalPages"searchResult.getPageCount());

model.addAttribute("page"searchResult.getCurPage());

model.addAttribute("pages"searchResult.getPageCount());

return "search";

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值