课程目标

目标1:完成商品分类功能

目标2:了解电商概念SPU 和SKU

目标3:掌握富文本编辑器的使用

目标4:掌握上传服务器FastDFS

目标5:掌握angularJS图片上传

1.商品分类

1.1需求及表结构分析

1.1.1需求分析

实现三级商品分类列表查询功能

进入页面首先显示所以一级分类,效果如下:




商品详情页html5代码 简单商品详情页html_服务器


点击列表行的查询下级按钮,进入下级分类列表,同时更新面包屑导航


商品详情页html5代码 简单商品详情页html_Storage_02


再次点击表行的查询下级按钮,进入三级分类列表,因为三级分类属于最后一级,所以在列表中不显示查询下级按钮,同时更新面包屑导航


商品详情页html5代码 简单商品详情页html_商品详细信息的代码html_03


点击面包屑导航,可以进行返回操作。

1.1.2表结构分析

tb_item_cat 商品分类表


商品详情页html5代码 简单商品详情页html_文件上传_04


1.2列表实现

1.2.1后端代码

修改pinyougou-sellergoods-interface工程ItemCatService接口,新增方法定义

/** * 根据上级ID返回列表 * @return */public List findByParentId(Long parentId);

修改pinyougou-sellergoods-interface工程ItemCatServiceImpl ,实现方法

/** * 根据上级ID查询列表 */@Overridepublic List findByParentId(Long parentId) {TbItemCatExample example1=new TbItemCatExample();Criteria criteria1 = example1.createCriteria();criteria1.andParentIdEqualTo(parentId);return itemCatMapper.selectByExample(example1);}

修改pinyougou-manager-web的ItemCatController.java

/** * 根据上级ID查询列表 * @param parentId * @return */@RequestMapping("/findByParentId")public List findByParentId(Long parentId){return itemCatService.findByParentId(parentId);}

1.2.2前端代码

(1)修改itemCatService.js

//根据上级ID查询下级列表this.findByParentId=function(parentId){return $http.get('../itemCat/findByParentId.do?parentId='+parentId);}

(2)修改itemCatController.js

//根据上级ID显示下级列表 $scope.findByParentId=function(parentId){itemCatService.findByParentId(parentId).success(function(response){$scope.list=response;});}

(3)修改item_cat.html

引入JS

指令定义

循环列表

{{entity.id}}{{entity.name}} {{entity.typeId}}  查询下级  修改

1.3面包屑导航

我们需要返回上级列表,需要通过点击面包屑来实现

修改itemCatController.js

$scope.grade=1;//默认为1级//设置级别$scope.setGrade=function(value){$scope.grade=value;}//读取列表$scope.selectList=function(p_entity){if($scope.grade==1){//如果为1级$scope.entity_1=null;$scope.entity_2=null;}if($scope.grade==2){//如果为2级$scope.entity_1=p_entity;$scope.entity_2=null;}if($scope.grade==3){//如果为3级$scope.entity_2=p_entity;}$scope.findByParentId(p_entity.id);//查询此级下级列表}

修改列表的查询下级按钮,设定级别值后 显示列表

查询下级

这里我们使用了ng-if指令,用于条件判断,当级别不等于3的时候才显示“查询下级”按钮

绑定面包屑:

顶级分类列表 {{entity_1.name}} {{entity_2.name}}

1.4新增商品分类(学员实现)

实现商品分类,如下图:


商品详情页html5代码 简单商品详情页html_文件上传_05


当前显示的是哪一分类的列表,我们就将这个商品分类新增到这个分类下。

实现思路:我们需要一个变量去记住上级ID,在保存的时候再根据这个ID来新增分类

修改itemCatController.js, 定义变量

$scope.parentId=0;//上级ID

查询时记录上级ID

//根据上级ID显示下级列表 $scope.findByParentId=function(parentId){$scope.parentId=parentId;//记住上级IDitemCatService.findByParentId(parentId).success(function(response){$scope.list=response;});}

保存的时候,用到此变量

//保存 $scope.save=function(){var serviceObject;//服务层对象 if($scope.entity.id!=null){//如果有IDserviceObject=itemCatService.update( $scope.entity ); //修改 }else{$scope.entity.parentId=$scope.parentId;//赋予上级IDserviceObject=itemCatService.add( $scope.entity );//增加 }serviceObject.success(function(response){if(response.success){//重新查询 $scope.findByParentId($scope.parentId);//重新加载}else{alert(response.message);}});}

修改页面item_cat.html


上级商品分类 {{entity_1.name}} >> {{entity_2.name}} 商品分类名称 类型模板


保存关闭


实现类型模板下拉列表的代码略

1.5修改商品分类(学员实现)

修改item_cat.html的修改按钮

修改

1.6删除商品分类(学员实现)

(代码略)

2.电商概念及表结构分析

2.1电商概念SPU与SKU

SPU = Standard Product Unit (标准产品单位)

SPU是商品信息聚合的最小单位,是一组可复用、易检索的标准化信息的集合,该集合描述了一个产品的特性。

通俗点讲,属性值、特性相同的商品就可以称为一个SPU。

例如:

iphone7就是一个SPU,与商家,与颜色、款式、套餐都无关。

SKU=stock keeping unit(库存量单位)

SKU即库存进出计量的单位, 可以是以件、盒、托盘等为单位。

SKU是物理上不可分割的最小存货单元。在使用时要根据不同业态,不同管理模式来处理。在服装、鞋类商品中使用最多最普遍。

例如:

纺织品中一个SKU通常表示:规格、颜色、款式。

2.2表结构分析

Tb_goods 商品表


商品详情页html5代码 简单商品详情页html_商品详情页html5代码_06


商品详情页html5代码 简单商品详情页html_文件上传_07


3.商家后台-商品录入【基本功能】

3.1需求分析

在商家后台实现商品录入功能。包括商品名称、副标题、价格、包装列表、售后服务


商品详情页html5代码 简单商品详情页html_商品详情页html5代码_08


3.2后端代码

3.2.1实体类

创建组合实体类goods

public class Goods implements Serializable{private TbGoods goods;//商品SPUprivate TbGoodsDesc goodsDesc;//商品扩展private List itemList;//商品SKU列表//getter and setter方法......}

3.2.2数据访问层

由于我们需要在商品表添加数据后可以得到自增的ID,所以我们需要在TbGoodsMapper.xml中的insert配置中添加如下配置

SELECT LAST_INSERT_ID() AS id

3.2.3服务接口层

修改pinyougou-sellergoods-interface 的GoodsService接口 add方法

/** * 增加*/public void add(Goods goods);

3.2.4服务实现层

修改pinyougou-sellergoods-service的GoodsServiceImpl.java

@Autowiredprivate TbGoodsDescMapper goodsDescMapper;/** * 增加 */@Overridepublic void add(Goods goods) {goods.getGoods().setAuditStatus("0");//设置未申请状态goodsMapper.insert(goods.getGoods());goods.getGoodsDesc().setGoodsId(goods.getGoods().getId());//设置IDgoodsDescMapper.insert(goods.getGoodsDesc());//插入商品扩展数据}

3.2.5控制层

修改pinyougou-shop-web工程的GoodsController的add方法

/** * 增加 * @param goods * @return */@RequestMapping("/add")public Result add(@RequestBody Goods goods){//获取登录名String sellerId = SecurityContextHolder.getContext().getAuthentication().getName();goods.getGoods().setSellerId(sellerId);//设置商家IDtry {goodsService.add(goods);return new Result(true, "增加成功");} catch (Exception e) {e.printStackTrace();return new Result(false, "增加失败");}}

3.3前端代码

3.3.1控制层

修改goodsController.js ,在增加成功后弹出提示,并清空实体(因为编辑页面无列表

//保存 $scope.add=function(){goodsService.add( $scope.entity ).success(function(response){if(response.success){alert('保存成功');$scope.entity={};}else{alert(response.message);}});}

3.3.2页面

修改goods_edit.html

引入JS:

定义控制器:

表单部分代码:


商品名称


副标题


价格


¥


包装列表


售后服务


保存按钮

保存

4.商家后台-商品录入【商品介绍】

4.1需求分析

实现商品介绍的录入,要求使用富文本编辑器


商品详情页html5代码 简单商品详情页html_文件上传_09


4.2富文本编辑器介绍

富文本编辑器,Rich Text Editor, 简称 RTE, 它提供类似于 Microsoft Word 的编辑功能。常用的富文本编辑器:

KindEditor http://kindeditor.net/

UEditor http://ueditor.baidu.com/website/

CKEditor http://ckeditor.com/

4.3使用kindeditor完成商品介绍的录入

4.3.1初始化kindeditor编辑器

在页面中添加JS代码,用于初始化kindeditor

allowFileManager 【是否允许浏览服务器已上传文件】 默认值是:false

4.3.2提取kindeditor编辑器的内容

在goodsController.js中的add()方法中添加

function(response){if(response.success){alert("保存成功");$scope.entity={};editor.html('');//清空富文本编辑器}else{alert(response.message);}}

5.分布式文件服务器FastDFS

5.1什么是FastDFS

FastDFS 是用 c 语言编写的一款开源的分布式文件系统。FastDFS 为互联网量身定制,充分考虑了冗余备份、负载均衡、线性扩容等机制,并注重高可用、高性能等指标,使用 FastDFS很容易搭建一套高性能的文件服务器集群提供文件上传、下载等服务。

FastDFS 架构包括 Tracker server 和 Storage server。客户端请求 Tracker server 进行文件上传、下载,通过 Tracker server 调度最终由 Storage server 完成文件上传和下载。

Tracker server 作用是负载均衡和调度,通过 Tracker server 在文件上传时可以根据一些策略找到 Storage server 提供文件上传服务。可以将 tracker 称为追踪服务器或调度服务器。

Storage server 作用是文件存储,客户端上传的文件最终存储在 Storage 服务器上,Storageserver 没有实现自己的文件系统而是利用操作系统 的文件系统来管理文件。可以将storage称为存储服务器。


商品详情页html5代码 简单商品详情页html_商品详情页html5代码_10


服务端两个角色:

Tracker:管理集群,tracker 也可以实现集群。每个 tracker 节点地位平等。收集 Storage 集群的状态。

Storage:实际保存文件 Storage 分为多个组,每个组之间保存的文件是不同的。每个组内部可以有多个成员,组成员内部保存的内容是一样的,组成员的地位是一致的,没有主从的概念。

5.2文件上传及下载的流程

5.2.1 文件上传流程


商品详情页html5代码 简单商品详情页html_商品详情页html5代码_11


客户端上传文件后存储服务器将文件 ID 返回给客户端,此文件 ID 用于以后访问该文件的索引信息。文件索引信息包括:组名,虚拟磁盘路径,数据两级目录,文件名。

 组名:文件上传后所在的 storage 组名称,在文件上传成功后有 storage 服务器返回,需要客户端自行保存。

 虚拟磁盘路径:storage 配置的虚拟路径,与磁盘选项 store_path*对应。如果配置了

store_path0 则是 M00,如果配置了 store_path1 则是 M01,以此类推。

 数据两级目录:storage 服务器在每个虚拟磁盘路径下创建的两级目录,用于存储数据

文件。

 文件名:与文件上传时不同。是由存储服务器根据特定信息生成,文件名包含:源存储

服务器 IP 地址、文件创建时间戳、文件大小、随机数和文件拓展名等信息。

5.2.2 文件下载流程


商品详情页html5代码 简单商品详情页html_Storage_12


5.3最简单的 FastDFS 架构


商品详情页html5代码 简单商品详情页html_商品详细信息的代码html_13


5.4 FastDFS安装

FastDFS 安装步骤非常繁琐,我们在课程中不做要求。已经提供单独的《FastDFS安装部署文档》供学员们课后阅读。

为了能够快速的搭建FastDFS环境进行代码开发,我们这里提供了安装好的镜像。

解压“资源/Linux镜像/fastDFS/pinyougou-image-server.zip”,双击vmx文件,然后启动。

注意:遇到下列提示选择“我已移动该虚拟机”!


商品详情页html5代码 简单商品详情页html_Storage_14


IP地址已经固定为192.168.25.133 ,请设置你的仅主机网段为25。

登录名为root 密码为itcast

5.5 FastDFS入门小Demo

需求:将本地图片上传至图片服务器,再控制台打印url

(1)创建Maven工程fastDFSdemo

由于FastDFS客户端jar包并没有在中央仓库中,所以需要使用下列命令手动安装jar包到Maven本地仓库(将jar包放到d盘setup目录)课程配套的本地仓库已经有此jar包,此步可省略。

mvn install:install-file -DgroupId=org.csource.fastdfs -DartifactId=fastdfs -Dversion=1.2 -Dpackaging=jar -Dfile=d:setupfastdfs_client_v1.20.jar

pom.xml中引入

org.csource.fastdfs fastdfs 1.2

(2)添加配置文件fdfs_client.conf ,将其中的服务器地址设置为192.168.25.133

//......tracker_server=192.168.25.133:22122//......

(3)创建java类,main方法代码如下:

// 1、加载配置文件,配置文件中的内容就是 tracker 服务的地址。ClientGlobal.init("D:/maven_work/fastDFS-demo/src/fdfs_client.conf");// 2、创建一个 TrackerClient 对象。直接 new 一个。TrackerClient trackerClient = new TrackerClient();// 3、使用 TrackerClient 对象创建连接,获得一个 TrackerServer 对象。TrackerServer trackerServer = trackerClient.getConnection();// 4、创建一个 StorageServer 的引用,值为 nullStorageServer storageServer = null;// 5、创建一个 StorageClient 对象,需要两个参数 TrackerServer 对象、StorageServer 的引用StorageClient storageClient = new StorageClient(trackerServer, storageServer);// 6、使用 StorageClient 对象上传图片。//扩展名不带“.”String[] strings = storageClient.upload_file("D:/pic/benchi.jpg