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. Dell-R730 【Pxe+dhcp+ftp+tftp+Kickstart+CentOs6.6】

    IP:10.104.0.101 [root@localhost network-scripts]# cat ifcfg-em1 [root@localhost network-scripts]# ip ...

  2. 网络叠加模式VLAN、VxLAN、GRE

    什么是叠加网络 1.一个数据包(或帧)封装在另一个数据包内;被封装的包转发到隧道端点后再被拆装. 2.叠加网络就是使用这种所谓"包内之包"的技术安全地将一个网络隐藏在另一个 网络中 ...

  3. CHD 2015迎新杯题解

    A.预防流感的拉面女神 简析:计算 n 的二进制表示里面 1 的个数 #include <cstdio> #include <cstring> #include <alg ...

  4. find指令参数

    1.name ~ 根目录 . 当前和子目录 name之后跟的是文件名 find . -name "[a-z]*[4-9].log" -print 2.perm perm后面跟的是权 ...

  5. 解码红外遥控信号——使用遥控器的按键来调节LED的亮度

    程序开始时,提示遥控键0~4的代码,然后程序通过设置LED的亮度来对被按下的按钮作出响应,以0关闭LED,1~4提供增加的亮度. 代码如下:(需要使用IRremote库,可在库管理中搜索该库进行下载后 ...

  6. 形象化理解nice值越高优先级越低

    当我们说起某个人很nice的时候,通常指的是那个人心地善良,很好说话,很容易相处.(相反,aggressive的人很可能是不nice的,因为老想抢夺别人的东西,尤其是在资源有限的情况下,恨不得一个蛋糕 ...

  7. 微信小程序推荐

    工具类 快递100助手:查快递,寄快递. 驾考帮手:为您提供科目一.科目四的测试题练习. 翻译e:多国语言在线翻译,中英.中日.中俄.中法快速翻译,专业翻译. 朝夕日历Pro:一个日历管理 app,你 ...

  8. JUnit——(一)使用IDEA进行基本操作

    JUnit是一个回归测试框架(regression testing framework).Junit测试是程序员测试,即所谓白盒测试,因为程序员知道被测试的软件如何(How)完成功能和完成什么样(Wh ...

  9. openstack私有云布署实践【9.3 主从controller单向同步glance-image目录】

    采用Rysnc单向同步,而不用双方实时同步,原因是在历史的运行过程中,我们发现,有些镜像包太大,当在主用的glance将镜像保存时,并不是一时半会就把镜像保存好,当主用在保存时,备用节点又在实时同步那 ...

  10. Pass和ClassPath变量配置

    1.pass环境变量配置的是可执行性文件bin目录,是为了在任意盘符下都可以运行javac.exe和java.exe所配置的. 2.classpath环境变量记录的是java类运行文件所在的目录,一般 ...