淘淘商城_day07_课堂笔记

时间:2023-03-09 21:53:21
淘淘商城_day07_课堂笔记
  1. 今日大纲

  1. 讲解订单系统
  2. 基于订单系统完成下单功能的开发
  3. 使用Solr完成商品的搜索功能
    1. 订单系统

说明:订单系统只是做讲解,不做开发。

  1. 导入taotao-order

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

  1. 表结构

订单表:

淘淘商城_day07_课堂笔记

订单商品表:

淘淘商城_day07_课堂笔记

疑问:为什么要冗余存储商品的基本信息?

订单物流表(收货人信息):

淘淘商城_day07_课堂笔记

  1. 开放平台

淘淘商城_day07_课堂笔记

http://open.taobao.com/doc2/apiDetail?spm=0.0.0.0.MiNiKt&apiId=47&docType=:

淘淘商城_day07_课堂笔记

  1. 配置tomcat插件

淘淘商城_day07_课堂笔记

  1. 配置nginx

淘淘商城_day07_课堂笔记

  1. 配置hosts

淘淘商城_day07_课堂笔记

  1. 接口文档

淘淘商城_day07_课堂笔记

  1. 接口

    1. 创建订单接口

使用:

淘淘商城_day07_课堂笔记

  1. 如何生成订单号

订单号需求是什么?

  1. 唯一
  2. 可读性高
    1. 纯数字
  3. 长度
    1. 不长于20位

如何生成:

  1. 时间戳
    1. 可能重复
  2. 手机号
  3. 时间戳 + 随机数
    1. 可能重复
  4. 时间戳 + 自增id
    1. 可行 (使用Redis的INCR命令完成)
  5. 用户id + 时间戳
  6. 用户id + 店铺id + 时间戳
    1. 可行,适用于C2C、B2C平台
  1. 具体实现

Controller: 接收提交的json数据

淘淘商城_day07_课堂笔记

Service:将json反序列化为Order对象,对Order对象做校验,生成订单ID:

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

OrderDAO是一个接口的实现类,后期如果对订单数据的存储做改造时,保存订单就会有多套实现:

淘淘商城_day07_课堂笔记

通过配置将OrderDAO加入到Spring容器:

淘淘商城_day07_课堂笔记

使用配置文件配置bean,后期的修改更加灵活。

Mapper.xml:

淘淘商城_day07_课堂笔记

一个statement执行多条SQL语句,默认情况下支持吗? -- 不支持的。

在连接字符串中设置:

淘淘商城_day07_课堂笔记

是在同一个事务中吗? -- 是的。

但是,现在的代码中存在一个bug,事务不会回滚,原因:Service中的异常被捕获:

淘淘商城_day07_课堂笔记

  1. 根据订单ID查询订单

    1. 测试

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

  1. Controller

淘淘商城_day07_课堂笔记

  1. Service

淘淘商城_day07_课堂笔记

  1. OrderDAO

淘淘商城_day07_课堂笔记

  1. OrderMapper

淘淘商城_day07_课堂笔记

  1. OrderMapper.xml

淘淘商城_day07_课堂笔记

通过延迟加载实现订单商品和订单物流数据的加载:

开启延迟加载:

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

  1. 根据用户名分页查询订单

    1. 测试

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

  1. Controller

淘淘商城_day07_课堂笔记

  1. Service

淘淘商城_day07_课堂笔记

  1. OrderDAO

淘淘商城_day07_课堂笔记

Mapp.xml:(使用延迟加载)

淘淘商城_day07_课堂笔记

分页插件的配置(不要求掌握,了解即可):

淘淘商城_day07_课堂笔记

  1. 修改订单状态

    1. 测试

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

  1. Controller

淘淘商城_day07_课堂笔记

  1. Service

将json反序列化成Order对象:

淘淘商城_day07_课堂笔记

  1. OrderDAO

设置更新时间:

淘淘商城_day07_课堂笔记

  1. OrderMapper.xml

这是一个通用的更新实现:

<update id="update">

UPDATE <include refid="tableName"/>

<set>

<if test="payment !=null and payment != ''">

payment = #{payment},

</if>

<if test="postFee !=null and postFee != ''">

post_fee = #{postFee},

</if>

<if test="status !=null and status != ''">

status = #{status},

</if>

<if test="updateTime !=null and updateTime != ''">

update_time = #{updateTime},

</if>

<if test="paymentTime !=null and paymentTime != ''">

payment_time = #{paymentTime},

</if>

<if test="consignTime !=null and consignTime != ''">

consign_time = #{consignTime},

</if>

<if test="endTime !=null and endTime != ''">

end_time = #{endTime},

</if>

<if test="closeTime !=null and closeTime != ''">

close_time = #{closeTime},

</if>

<if test="shippingName !=null and shippingName != ''">

shipping_name = #{shippingName},

</if>

<if test="shippingCode !=null and shippingCode != ''">

shipping_code = #{shippingCode},

</if>

<if test="buyerMessage !=null and buyerMessage != ''">

buyer_message = #{buyerMessage},

</if>

<if test="buyerRate !=null and buyerRate != ''">

buyer_rate = #{buyerRate},

</if>

</set>

WHERE order_id = #{orderId};

</update>

  1. 基于订单系统接口完成下单功能

    1. 下单按钮功能

淘淘商城_day07_课堂笔记

  1. 订单确认页

淘淘商城_day07_课堂笔记

  1. 去订单确认页

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

  1. 使用拦截器实现用户是否登录的校验

    1. 拦截器

淘淘商城_day07_课堂笔记

  1. 配置拦截器

淘淘商城_day07_课堂笔记

  1. 编写具体的业务逻辑

淘淘商城_day07_课堂笔记

  1. 点击提交订单事件

淘淘商城_day07_课堂笔记

表单数据:

淘淘商城_day07_课堂笔记

  1. 接收提交订单请求

    1. 在web.xml中添加service/*请求进入SpringMVC

淘淘商城_day07_课堂笔记

  1. Controller

淘淘商城_day07_课堂笔记

  1. Service

淘淘商城_day07_课堂笔记

  1. 在ApiService中扩展doPostJson方法

淘淘商城_day07_课堂笔记

  1. 测试

淘淘商城_day07_课堂笔记

  1. 通过token进行了2次查询

一次是在拦截器中查询,一次是在Controller查询,存在性能和资源浪费问题。

如何将拦截器中的数据传递到Controller?

方案:

  1. 将User对象放置到request对象中
  2. 使用ThreadLocal实现
    1. 进入tomcat和产生响应前,都处于同一个线程中

实现:

  1. 定义ThreadLocal
    淘淘商城_day07_课堂笔记
  2. 在拦截器中将User对象放置到ThreadLocal中
    淘淘商城_day07_课堂笔记

    在完成方法中清除User对象:
    淘淘商城_day07_课堂笔记

  3. 在Controller或Service获取User对象
    淘淘商城_day07_课堂笔记
    1. 成功页

淘淘商城_day07_课堂笔记

使用joda-time组件:

淘淘商城_day07_课堂笔记

测试:

淘淘商城_day07_课堂笔记

  1. 实现商品的搜索

    1. 导入数据

数据来源,京东,在:

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

数据:

淘淘商城_day07_课堂笔记

  1. 修改表结构

淘淘商城_day07_课堂笔记

  1. 导入图片数据

淘淘商城_day07_课堂笔记

测试:

淘淘商城_day07_课堂笔记

  1. Solr的安装

  • 安装参考《Solr安装手册.docx》

淘淘商城_day07_课堂笔记

启动solr:

java -jar start.jar

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

  1. 通过域名访问solr服务

通过solr.taotao.com访问。

淘淘商城_day07_课堂笔记

配置nginx:

淘淘商城_day07_课堂笔记

测试:

淘淘商城_day07_课堂笔记

发现,请求域名后面依然得有/solr.

默认启动的web应用服务器是jetty。

修改配置文件:

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

测试:

淘淘商城_day07_课堂笔记

完美。

  1. Solrj

淘淘商城_day07_课堂笔记

  1. 创建taotao core

严格按照该文档执行:具体参考《创建taotao core.docx 》

索引结构:

淘淘商城_day07_课堂笔记

启动:

java -Dsolr.solr.home=taotao-solr -jar start.jar

  1. 添加IK中文分词器的支持

参考该文档实现:《集成IKAnalyzer中文分词器.docx》

淘淘商城_day07_课堂笔记

  1. 导入商品数据到solr中

    1. 导入itcast-solrj

淘淘商城_day07_课堂笔记

  1. 导入依赖

淘淘商城_day07_课堂笔记

  1. 使用Solrj完成索引数据的CRUD

public
class ItemDataTest {

private HttpSolrServer httpSolrServer;

@Before

public
void setUp() throws Exception {

// 在url中指定core名称:taotao

//http://solr.taotao.com/#/taotao -- 界面地址

String url = "http://solr.taotao.com/taotao"; //服务地址

HttpSolrServer httpSolrServer = new HttpSolrServer(url); //定义solr的server

httpSolrServer.setParser(new XMLResponseParser()); // 设置响应解析器

httpSolrServer.setConnectionTimeout(500); // 建立连接的最长时间

this.httpSolrServer = httpSolrServer;

}

@Test

public
void testInsert() throws Exception{

Item item = new Item();

item.setCid(1L);

item.setId(999L);

item.setImage("image");

item.setPrice(100L);

item.setSellPoint("很好啊,赶紧来买吧.");

item.setStatus(1);

item.setTitle("飞利浦老人手机 (X2560) 深情蓝移动联通2G手机双卡双待");

this.httpSolrServer.addBean(item);

this.httpSolrServer.commit();

}

@Test

public
void testUpdate() throws Exception{

Item item = new Item();

item.setCid(1L);

item.setId(999L);

item.setImage("image");

item.setPrice(100L);

item.setSellPoint("很好啊,赶紧来买吧. 豪啊");

item.setStatus(1);

item.setTitle("飞利浦老人手机 (X2560) 深情蓝移动联通2G手机双卡双待");

this.httpSolrServer.addBean(item);

this.httpSolrServer.commit();

}

@Test

public
void testDelete() throws Exception{

this.httpSolrServer.deleteById("999");

this.httpSolrServer.commit();

}

@Test

public
void testQuery() throws Exception{

int
page = 2;

int
rows = 1;

String keywords = "手机";

SolrQuery solrQuery = new SolrQuery(); //构造搜索条件

solrQuery.setQuery("title:" + keywords); //搜索关键词

开始,,rows=5当前返回5条记录,第二页就是变化start这个值为5就可以了。

solrQuery.setStart((Math.max(page, 1) - 1) * rows);

solrQuery.setRows(rows);

//是否需要高亮

boolean
isHighlighting = !StringUtils.equals("*", keywords) && StringUtils.isNotEmpty(keywords);

if (isHighlighting) {

// 设置高亮

solrQuery.setHighlight(true); // 开启高亮组件

solrQuery.addHighlightField("title");// 高亮字段

solrQuery.setHighlightSimplePre("<em>");// 标记,高亮关键字前缀

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

}

// 执行查询

QueryResponse queryResponse = this.httpSolrServer.query(solrQuery);

List<Item> items = queryResponse.getBeans(Item.class);

if (isHighlighting) {

// 将高亮的标题数据写回到数据对象中

Map<String, Map<String, List<String>>> map = queryResponse.getHighlighting();

for (Map.Entry<String, Map<String, List<String>>> highlighting : map.entrySet()) {

for (Item item : items) {

if (!highlighting.getKey().equals(item.getId().toString())) {

continue;

}

item.setTitle(StringUtils.join(highlighting.getValue().get("title"), ""));

break;

}

}

}

for (Item item : items) {

System.out.println(item);

}

}

}

  1. 将商品数据到入到solr

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

测试:

淘淘商城_day07_课堂笔记

OK。

  1. 创建搜索系统taotao-search

    1. 创建工程

淘淘商城_day07_课堂笔记

  1. 导入依赖

<project
xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

<modelVersion>4.0.0</modelVersion>

<parent>

<groupId>com.taotao.parent</groupId>

<artifactId>taotao-parent</artifactId>

<version>0.0.1-SNAPSHOT</version>

</parent>

<groupId>com.taotao.search</groupId>

<artifactId>taotao-search</artifactId>

<version>1.0.0-SNAPSHOT</version>

<packaging>war</packaging>

<dependencies>

<dependency>

<groupId>org.springframework</groupId>

<artifactId>spring-webmvc</artifactId>

</dependency>

<dependency>

<groupId>org.slf4j</groupId>

<artifactId>slf4j-log4j12</artifactId>

</dependency>

<!-- Jackson
Json处理工具包 -->

<dependency>

<groupId>com.fasterxml.jackson.core</groupId>

<artifactId>jackson-databind</artifactId>

</dependency>

<!-- httpclient -->

<dependency>

<groupId>org.apache.httpcomponents</groupId>

<artifactId>httpclient</artifactId>

</dependency>

<!-- JSP相关 -->

<dependency>

<groupId>jstl</groupId>

<artifactId>jstl</artifactId>

</dependency>

<dependency>

<groupId>javax.servlet</groupId>

<artifactId>servlet-api</artifactId>

<scope>provided</scope>

</dependency>

<dependency>

<groupId>javax.servlet</groupId>

<artifactId>jsp-api</artifactId>

<scope>provided</scope>

</dependency>

<!-- Apache工具组件 -->

<dependency>

<groupId>org.apache.commons</groupId>

<artifactId>commons-lang3</artifactId>

</dependency>

<dependency>

<groupId>org.apache.commons</groupId>

<artifactId>commons-io</artifactId>

</dependency>

<dependency>

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

<artifactId>solr-solrj</artifactId>

<version>4.10.2</version>

</dependency>

</dependencies>

<build>

<plugins>

<!-- 配置Tomcat插件 -->

<plugin>

<groupId>org.apache.tomcat.maven</groupId>

<artifactId>tomcat7-maven-plugin</artifactId>

<configuration>

<port>8085</port>

<path>/</path>

</configuration>

</plugin>

</plugins>

</build>

</project>

  1. Web.xml

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

<web-app
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns="http://java.sun.com/xml/ns/javaee"

xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"

id="WebApp_ID"
version="2.5">

<display-name>taotao-search</display-name>

<context-param>

<param-name>contextConfigLocation</param-name>

<param-value>classpath:spring/applicationContext*.xml</param-value>

</context-param>

<!--Spring的ApplicationContext 载入 -->

<listener>

<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>

</listener>

<!-- 编码过滤器,以UTF8编码 -->

<filter>

<filter-name>encodingFilter</filter-name>

<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>

<init-param>

<param-name>encoding</param-name>

<param-value>UTF8</param-value>

</init-param>

</filter>

<filter-mapping>

<filter-name>encodingFilter</filter-name>

<url-pattern>/*</url-pattern>

</filter-mapping>

<!-- 配置SpringMVC框架入口 -->

<servlet>

<servlet-name>taotao-search</servlet-name>

<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>

<init-param>

<param-name>contextConfigLocation</param-name>

<param-value>classpath:spring/taotao-search-servlet.xml</param-value>

</init-param>

<load-on-startup>1</load-on-startup>

</servlet>

<servlet-mapping>

<servlet-name>taotao-search</servlet-name>

<!--

伪静态

伪静态有利于SEO(搜索引擎优化)

-->

<url-pattern>*.html</url-pattern>

</servlet-mapping>

<welcome-file-list>

<welcome-file>index.html</welcome-file>

</welcome-file-list>

</web-app>

  1. Solr和Spring的整合

淘淘商城_day07_课堂笔记

  1. 配置nginx和hosts

淘淘商城_day07_课堂笔记

配置nginx:

淘淘商城_day07_课堂笔记

配置启动tomcat:

淘淘商城_day07_课堂笔记

  1. Controller

@RequestMapping(value = "search", method = RequestMethod.GET)

public ModelAndView search(@RequestParam("q") String keyWords,

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

ModelAndView mv = new ModelAndView("search");

SearchResult searchResult = null;

try {

searchResult = this.searchService.search(keyWords, page, ROWS);

} catch (SolrServerException e) {

e.printStackTrace();

searchResult = new SearchResult(new ArrayList<Item>(0), 0L);

}

// 搜索关键字

mv.addObject("query", keyWords);

// 搜索结果集

mv.addObject("itemList", searchResult.getList());

// 当前页数

mv.addObject("page", page);

// 总页数

int
total = searchResult.getTotal().intValue();

int
pages = total % ROWS == 0 ? total / ROWS : total / ROWS + 1;

mv.addObject("pages", pages);

return
mv;

}

  1. Service

public SearchResult searchItem(String keyWords, int
page, int
rows) {

SolrQuery solrQuery = new SolrQuery(); // 构造搜索条件

solrQuery.setQuery("title:" + keyWords + " AND status:1"); // 搜索关键词

开始,,rows=5当前返回5条记录,第二页就是变化start这个值为5就可以了。

solrQuery.setStart((Math.max(page, 1) - 1) * rows);

solrQuery.setRows(rows);

// 是否需要高亮

boolean
isHighlighting = !StringUtils.equals("*", keyWords) && StringUtils.isNotEmpty(keyWords);

if (isHighlighting) {

// 设置高亮

solrQuery.setHighlight(true); // 开启高亮组件

solrQuery.addHighlightField("title");// 高亮字段

solrQuery.setHighlightSimplePre("<em>");// 标记,高亮关键字前缀

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

}

try {

// 执行查询

QueryResponse queryResponse = this.httpSolrServer.query(solrQuery);

List<Item> items = queryResponse.getBeans(Item.class);

if (isHighlighting) {

// 将高亮的标题数据写回到数据对象中

Map<String, Map<String, List<String>>> map = queryResponse.getHighlighting();

for (Map.Entry<String, Map<String, List<String>>> highlighting : map.entrySet()) {

for (Item item : items) {

if (!highlighting.getKey().equals(item.getId().toString())) {

continue;

}

item.setTitle(StringUtils.join(highlighting.getValue().get("title"), ""));

break;

}

}

}

return
new SearchResult(queryResponse.getResults().getNumFound(), items);

} catch (SolrServerException e) {

e.printStackTrace();

}

return
null;

}

  1. 测试

淘淘商城_day07_课堂笔记

  1. 分页JS逻辑

淘淘商城_day07_课堂笔记

具体方法:

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

效果:

淘淘商城_day07_课堂笔记

下一页的逻辑:

淘淘商城_day07_课堂笔记

  1. 解决中文乱码问题

淘淘商城_day07_课堂笔记

测试:

淘淘商城_day07_课堂笔记

  1. 多词搜索的逻辑关系

淘淘商城_day07_课堂笔记

淘淘商城_day07_课堂笔记

测试:

淘淘商城_day07_课堂笔记