1. 今日大纲

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

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

  1. 导入taotao-order

  1. 表结构

订单表:

订单商品表:

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

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

  1. 开放平台

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

  1. 配置tomcat插件

  1. 配置nginx

  1. 配置hosts

  1. 接口文档

  1. 接口

    1. 创建订单接口

使用:

  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数据

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

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

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

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

Mapper.xml:

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

在连接字符串中设置:

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

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

  1. 根据订单ID查询订单

    1. 测试

  1. Controller

  1. Service

  1. OrderDAO

  1. OrderMapper

  1. OrderMapper.xml

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

开启延迟加载:

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

    1. 测试

  1. Controller

  1. Service

  1. OrderDAO

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

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

  1. 修改订单状态

    1. 测试

  1. Controller

  1. Service

将json反序列化成Order对象:

  1. OrderDAO

设置更新时间:

  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. 下单按钮功能

  1. 订单确认页

  1. 去订单确认页

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

    1. 拦截器

  1. 配置拦截器

  1. 编写具体的业务逻辑

  1. 点击提交订单事件

表单数据:

  1. 接收提交订单请求

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

  1. Controller

  1. Service

  1. 在ApiService中扩展doPostJson方法

  1. 测试

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

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

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

方案:

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

实现:

  1. 定义ThreadLocal
  2. 在拦截器中将User对象放置到ThreadLocal中

    在完成方法中清除User对象:

  3. 在Controller或Service获取User对象
    1. 成功页

使用joda-time组件:

测试:

  1. 实现商品的搜索

    1. 导入数据

数据来源,京东,在:

数据:

  1. 修改表结构

  1. 导入图片数据

测试:

  1. Solr的安装

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

启动solr:

java -jar start.jar

  1. 通过域名访问solr服务

通过solr.taotao.com访问。

配置nginx:

测试:

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

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

修改配置文件:

测试:

完美。

  1. Solrj

  1. 创建taotao core

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

索引结构:

启动:

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

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

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

  1. 导入商品数据到solr中

    1. 导入itcast-solrj

  1. 导入依赖

  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

测试:

OK。

  1. 创建搜索系统taotao-search

    1. 创建工程

  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的整合

  1. 配置nginx和hosts

配置nginx:

配置启动tomcat:

  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. 测试

  1. 分页JS逻辑

具体方法:

效果:

下一页的逻辑:

  1. 解决中文乱码问题

测试:

  1. 多词搜索的逻辑关系

测试:

淘淘商城_day07_课堂笔记的更多相关文章

  1. 淘淘商城_day11_课堂笔记

    今日大纲 发布前的准备 实施发布 一部分是由我来发布 一部分是由你们来发布 讲解分布式部署架构 测试 功能测试 压力测试 项目实战的准备以及分组 分组 抽取功能 讲解所需要开发的功能 项目部署上线流程 ...

  2. 淘淘商城_day01_课堂笔记

    今日大纲 聊聊电商行业 电商行业发展 11.11 2015双11: 2016年: 预测:2017年的双11交易额将达到:1400亿 电商行业技术特点 淘淘商城简介 淘淘商城的前身 电商行业的概念 B2 ...

  3. 淘淘商城_day04_课堂笔记

    今日大纲 实现首页的大广告位功能 实现内容管理系统 首页的大广告 什么是大广告 JS效果: 点击下面的序号选择查询哪个广告 自动切换 点击图片查询具体的页面 以上是由前端团队来开发. 数据结构 说明: ...

  4. 淘淘商城_day02_课堂笔记

    今日大纲 学习Nginx的使用 实现商品的管理 新增商品 查询商品列表 编辑商品 删除商品 上架和下架商品 学习nginx 开发阶段中的环境 开发环境:自己的电脑 测试环境:提供给测试人员使用的环境 ...

  5. 淘淘商城_day10_课堂笔记

    今日大纲 Dubbo入门学习 使用dubbo优化单点登录系统 系统间服务调用方式 浏览器直接访问 浏览器发起请求,通过ajax或jsonp方式请求: Httpclient方式 系统与系统之间通过Htt ...

  6. 淘淘商城_day09_课堂笔记

    今日大纲 实现购物车 基于Mysql实现读写分离 购物车 需求描述 用户可以在登录状态下将商品添加到购物车 用户可以在未登录状态下将商品添加到购物车 用户可以使用购物车一起结算下单 用户可以查询自己的 ...

  7. 淘淘商城_day08_课堂笔记

    今日大纲 问题,如何实现商品数据的同步? 学习MQ(消息队列) 搭建RabbitMQ的环境 学习RabbitMQ的队列 学习Spring-Rabbit 使用RabbitMQ完成商品数据的同步 如何实现 ...

  8. 淘淘商城_day05_课堂笔记

    今日大纲 学习Redis 使用Redis完成项目中缓存需求 实现商品详情页功能 缓存的需求 大广告位数据无需每次查询后台系统的接口,可以在前台系统添加缓存,提高访问首页的速度. 商品类目的数据也可以缓 ...

  9. 淘淘商城_day06_课堂笔记

    今日大纲 实现单点登录系统 基于单点登录系统实现,用户的注册和登录 商品数据同步问题 问题 后台系统中将商品修改,前台系统没有进行数据的同步,导致前端系统不能够实时显示最新的数据. 解决 后台系统中商 ...

随机推荐

  1. 正经学C#_表达式与其运算符[算术运算符]:《c#入门经典》

    表达式:正如字面意义,它是通过算术运算符来进行运算的数学公式.表达式的意义我们都是很明白的,大白话就是一个公式嘛.不是很难懂. 表达式不是一个单独的存在,必然有操作数或者操作符的.在c#中有操作符有很 ...

  2. Python 函数简介 之二

    1.当函数有多个返回值时, 其多个返回值将以元组的形式出现 def test1(): print("in the test1") return 'end' def test2(): ...

  3. JQuery动态操作表格

    新人,小白一枚,刚刚参加工作,所以会在这里记录一些遇到的问题. 最近要做的东西,是对一个表格动态的添加行,删除行,并且对表格中内容进行非空验证. <!DOCTYPE html> <h ...

  4. Word 文字转表格

    今天工作的时候遇到一个问题,需要整理出一个工程下依赖的jar的名称和大小,并且按照大小排序,我在使用一个java程序打印出名称和大小之后,需要将这些文字整理到word文档表格中,刚开始是想一个个的拷进 ...

  5. C#带参数打开网页及url获取

    1.带参数打开网页 Response.Redirect("form2.aspx?id=url1&name=ok"); 其中?后面为参数. 2.获取url 命令 结果 Req ...

  6. c#:readonly与const的区别

    readonly与const的区别: 1.初始化:const  字段只能在该字段的声明中初始化. readonly  字段可以在声明或构造函数中初始化. 2.值: const 字段是编译时常量(con ...

  7. SAP HANA中创建分析权限(Analytic Privilege)

    Demo Instruction: 假定CustomerID > 100的为VIP客户,我们的权限设置为只显示VIP客户 所使用的Attribute View: ATTR_CUSTOMER_FU ...

  8. Java Calendar日历类的使用

    Calendar cal = Calendar.getInstance(); // 当前年 int year = cal.get(Calendar.YEAR); // 当前月 int month = ...

  9. openstack私有云布署实践【19 通过python客户端 创建实例VM指定IP地址】

    还有一种创建方式 是使用py开发工具,调用openstackclient的方法进行创建实例 ,好处就是可随意指定我们要的虚拟机IP地址,需求的场景就是,某天我们需要主动分配一个比较熟知的IP用作某个服 ...

  10. centos6.5 安装zabbix

    实验说明: 操作系统: CentOS6.5 64位 Web环境: Apache Mysql PHP zabbix版本: LTS 2.2.10 Linux服务器IP: 10.0.0.2 Linux客户端 ...