淘淘商城_day07_课堂笔记
今日大纲
- 讲解订单系统
- 基于订单系统完成下单功能的开发
- 使用Solr完成商品的搜索功能
订单系统
说明:订单系统只是做讲解,不做开发。
导入taotao-order
表结构
订单表:
订单商品表:
疑问:为什么要冗余存储商品的基本信息?
订单物流表(收货人信息):
开放平台
http://open.taobao.com/doc2/apiDetail?spm=0.0.0.0.MiNiKt&apiId=47&docType=:
配置tomcat插件
配置nginx
配置hosts
接口文档
接口
创建订单接口
使用:
如何生成订单号
订单号需求是什么?
- 唯一
- 可读性高
- 纯数字
- 长度
- 不长于20位
如何生成:
- 时间戳
- 可能重复
- 手机号
- 时间戳 + 随机数
- 可能重复
- 时间戳 + 自增id
- 可行 (使用Redis的INCR命令完成)
- 用户id + 时间戳
- 用户id + 店铺id + 时间戳
- 可行,适用于C2C、B2C平台
具体实现
Controller: 接收提交的json数据
Service:将json反序列化为Order对象,对Order对象做校验,生成订单ID:
OrderDAO是一个接口的实现类,后期如果对订单数据的存储做改造时,保存订单就会有多套实现:
通过配置将OrderDAO加入到Spring容器:
使用配置文件配置bean,后期的修改更加灵活。
Mapper.xml:
一个statement执行多条SQL语句,默认情况下支持吗? -- 不支持的。
在连接字符串中设置:
是在同一个事务中吗? -- 是的。
但是,现在的代码中存在一个bug,事务不会回滚,原因:Service中的异常被捕获:
根据订单ID查询订单
测试
Controller
Service
OrderDAO
OrderMapper
OrderMapper.xml
通过延迟加载实现订单商品和订单物流数据的加载:
开启延迟加载:
根据用户名分页查询订单
测试
Controller
Service
OrderDAO
Mapp.xml:(使用延迟加载)
分页插件的配置(不要求掌握,了解即可):
修改订单状态
测试
Controller
Service
将json反序列化成Order对象:
OrderDAO
设置更新时间:
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>
基于订单系统接口完成下单功能
下单按钮功能
订单确认页
去订单确认页
使用拦截器实现用户是否登录的校验
拦截器
配置拦截器
编写具体的业务逻辑
点击提交订单事件
表单数据:
接收提交订单请求
在web.xml中添加service/*请求进入SpringMVC
Controller
Service
在ApiService中扩展doPostJson方法
测试
通过token进行了2次查询
一次是在拦截器中查询,一次是在Controller查询,存在性能和资源浪费问题。
如何将拦截器中的数据传递到Controller?
方案:
- 将User对象放置到request对象中
- 使用ThreadLocal实现
- 进入tomcat和产生响应前,都处于同一个线程中
实现:
- 定义ThreadLocal
- 在拦截器中将User对象放置到ThreadLocal中
在完成方法中清除User对象:
- 在Controller或Service获取User对象
成功页
使用joda-time组件:
测试:
实现商品的搜索
导入数据
数据来源,京东,在:
数据:
修改表结构
导入图片数据
测试:
Solr的安装
- 安装参考《Solr安装手册.docx》
启动solr:
java -jar start.jar
通过域名访问solr服务
通过solr.taotao.com访问。
配置nginx:
测试:
发现,请求域名后面依然得有/solr.
默认启动的web应用服务器是jetty。
修改配置文件:
测试:
完美。
Solrj
创建taotao core
严格按照该文档执行:具体参考《创建taotao core.docx 》
索引结构:
启动:
java -Dsolr.solr.home=taotao-solr -jar start.jar
添加IK中文分词器的支持
参考该文档实现:《集成IKAnalyzer中文分词器.docx》
导入商品数据到solr中
导入itcast-solrj
导入依赖
使用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);
}
}
}
将商品数据到入到solr
测试:
OK。
创建搜索系统taotao-search
创建工程
导入依赖
<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>
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>
Solr和Spring的整合
配置nginx和hosts
配置nginx:
配置启动tomcat:
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;
}
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;
}
测试
分页JS逻辑
具体方法:
效果:
下一页的逻辑:
解决中文乱码问题
测试:
多词搜索的逻辑关系
测试:
淘淘商城_day07_课堂笔记的更多相关文章
- 淘淘商城_day11_课堂笔记
今日大纲 发布前的准备 实施发布 一部分是由我来发布 一部分是由你们来发布 讲解分布式部署架构 测试 功能测试 压力测试 项目实战的准备以及分组 分组 抽取功能 讲解所需要开发的功能 项目部署上线流程 ...
- 淘淘商城_day01_课堂笔记
今日大纲 聊聊电商行业 电商行业发展 11.11 2015双11: 2016年: 预测:2017年的双11交易额将达到:1400亿 电商行业技术特点 淘淘商城简介 淘淘商城的前身 电商行业的概念 B2 ...
- 淘淘商城_day04_课堂笔记
今日大纲 实现首页的大广告位功能 实现内容管理系统 首页的大广告 什么是大广告 JS效果: 点击下面的序号选择查询哪个广告 自动切换 点击图片查询具体的页面 以上是由前端团队来开发. 数据结构 说明: ...
- 淘淘商城_day02_课堂笔记
今日大纲 学习Nginx的使用 实现商品的管理 新增商品 查询商品列表 编辑商品 删除商品 上架和下架商品 学习nginx 开发阶段中的环境 开发环境:自己的电脑 测试环境:提供给测试人员使用的环境 ...
- 淘淘商城_day10_课堂笔记
今日大纲 Dubbo入门学习 使用dubbo优化单点登录系统 系统间服务调用方式 浏览器直接访问 浏览器发起请求,通过ajax或jsonp方式请求: Httpclient方式 系统与系统之间通过Htt ...
- 淘淘商城_day09_课堂笔记
今日大纲 实现购物车 基于Mysql实现读写分离 购物车 需求描述 用户可以在登录状态下将商品添加到购物车 用户可以在未登录状态下将商品添加到购物车 用户可以使用购物车一起结算下单 用户可以查询自己的 ...
- 淘淘商城_day08_课堂笔记
今日大纲 问题,如何实现商品数据的同步? 学习MQ(消息队列) 搭建RabbitMQ的环境 学习RabbitMQ的队列 学习Spring-Rabbit 使用RabbitMQ完成商品数据的同步 如何实现 ...
- 淘淘商城_day05_课堂笔记
今日大纲 学习Redis 使用Redis完成项目中缓存需求 实现商品详情页功能 缓存的需求 大广告位数据无需每次查询后台系统的接口,可以在前台系统添加缓存,提高访问首页的速度. 商品类目的数据也可以缓 ...
- 淘淘商城_day06_课堂笔记
今日大纲 实现单点登录系统 基于单点登录系统实现,用户的注册和登录 商品数据同步问题 问题 后台系统中将商品修改,前台系统没有进行数据的同步,导致前端系统不能够实时显示最新的数据. 解决 后台系统中商 ...
随机推荐
- JavaScript的事件机制
JavaScript的事件机制 摘要 事件是将JavaScript脚本与网页联系在一起的主要方式,是JavaScript中最重要的主题之一,深入理解事件的工作机制以及它们对性能的影响至关重要.本文将详 ...
- 【转】经典!python中使用xlrd、xlwt操作excel表格详解
最近遇到一个情景,就是定期生成并发送服务器使用情况报表,按照不同维度统计,涉及python对excel的操作,上网搜罗了一番,大多大同小异,而且不太能满足需求,不过经过一番对源码的"研究&q ...
- 六、spark常见问题总结(转载)
问题导读 1.当前集群的可用资源不能满足应用程序的需求,怎么解决? 2.内存里堆的东西太多了,有什么好办法吗? 1.WARN TaskSchedulerImpl: Initial jo ...
- CodeForces 631D Messenger
$KMP$. $n=1$和$n=2$的时候可以单独计算.$n>2$时,可以拿字符和数字分别做一次匹配,然后扫描一遍判断一下就可以计算出答案了. #pragma comment(linker, & ...
- oc底层方法调用流程
1.对象方法保存到类对象里面,每个类对象中都有一个方法列表.类方法保存在元类中方法列表 a.通过isa指针去对应的类中查找: b.生成方法编号,根据方法编号查找对应的方法(找到只是最终实现地址,根据地 ...
- FileOutputStream flush()
FileOutputStream 继承 OutputStream ,flush方法查看源码方法体为空,所以flush没起到清除缓存的作用 改用BufferedOutputStream再调用flush( ...
- Beyond Compare V3.2.3 Beta 中文版
软件名称: Beyond Compare V3.2.3 Beta 中文版 软件语言: 简体中文 授权方式: 免费软件 运行环境: Win7 / Vista / Win2003 / WinXP 软件大小 ...
- HTTP Response Splitting攻击探究 <转>
第一小节:HTTP Basics:使用Proxy软件(例如Webscarab)来截断浏览器(客户端)和Server之间的HTTP通信,之后任意篡改得到预期结果即可. 第二小节:HTTP Splitti ...
- tar命令结合find搜索将指定条件的文件进行打包压缩
今天处理 /tmp 临时目录下 session 文件过多的问题,顺便找到了这个 tar 结合 find 打包压缩的方法,记录下来,很实用的哈! 搜索 /tmp 目录下 创建超过 24 分钟的文件 fi ...
- MyBatis学习-映射文件标签篇(select、resultMap)
MyBatis 真正的核心在映射文件中.比直接使用 JDBC 节省95%的代码.而且将 SQL 语句独立在 Java 代码之外,可以进行更为细致的 SQL 优化. 一. 映射文件的顶级元素 selec ...