分布式商城项目-后台开发2-商品管理功能实现(一)
分布式商城项目-后台开发2-商品管理功能实现(一)
商品管理功能包括:
功能拆分 | 实现方法 |
---|---|
显示商品列表 | 查询数据库信息并展示 |
添加商品 | 通过后端程序在数据库增加记录 |
修改商品 | 通过后端程序更新数据库 |
删除商品 | 通过后端程序修改商品状态 |
首先,我们所有的操作都是基于数据的,那么我们先来分析一下商品表的数据字段
1.显示商品列表
根据数据库可以看出,在文章管理界面,我们用列表的形式来显示,需要显示在表格中的数据就有:商品类型名称(因为使用的原因,不能给客户显示一个商品类别的id,那样讲无法使用)、商品名称。商品副标题、商品主图、商品价格、库存数量、商品状态,然后商品详情和商品图片此处并不需要显示。
因此我们可以按照此分析设计出表格界面,如图:
有了页面,我们追根溯源,页面的数据从哪里来呢?当然是controller,而controller中的数据则依赖于Service服务层,Service服务层则需要从dao层得到数据
那么首先,要获得数据,我们就需要dao层的支持。
1.1编写dao层实现查询操作
在ProductMapper接口中添加方法:
List<Product> selectProductList();
在ProductMapper.xml中添加bean:
<select id="selectProductList" resultMap="BaseResultMap">
select
<include refid="Base_Column_List" />
from product
</select>
1.2编写service层实现查询操作
在interfaces模块创建接口ProductService
package cn.yuechenc.ycshop.manager.interfaces;
import cn.yuechenc.pojo.Product;
public interface ProductService {
public List<Product> selectProductList();
}
在service模块创建接口ProductService的实现类ProductServiceImpl
package cn.yuechenc.ycshop.manager.service.impl;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import cn.yuechenc.manager.dao.mapper.ProductMapper;
import cn.yuechenc.pojo.Product;
import cn.yuechenc.ycshop.manager.interfaces.ProductService;
@Service
public class ProductServiceImpl implements ProductService {
@Autowired
private ProductMapper productMapper;
@Override
public List<Product> selectProductList() {
return productMapper.selectProductList();
}
}
接下来就可以编写controller了
package cn.yuechenc.ycshop.manager.controller;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.ModelMap;
import org.springframework.web.bind.annotation.RequestMapping;
import cn.yuechenc.pojo.Product;
import cn.yuechenc.ycshop.manager.interfaces.ProductService;
@Controller
@RequestMapping("/product")
public class ProductController {
@Autowired
private ProductService productService;
@RequestMapping("/getProductList")
private String getProductList(ModelMap model){
List<Product> prList=productService.selectProductList();
model.addAttribute("prodect", prList);
return "Product";
}
}
1.3修改JSP页面,绑定数据
当我们查询到了数据,就需要把他显示在界面上了。
这里我们先显示类别id和状态字段,后面再做处理
先在数据库添加一条记录以便测试
1.4添加dubbo配置
在spring-service中添加
<dubbo:service interface="cn.yuechenc.ycshop.manager.interfaces.ProductService"
ref="productServiceImpl" timeout="30000" />
在spring-mvc中添加
<dubbo:reference
interface="cn.yuechenc.ycshop.manager.interfaces.ProductService" id="productServiceImpl"
timeout="30000" />
那么现在我们来测试一下。启动项目访问:http://localhost:8080/product/getProductList
到这里,前面的查询就没问题了,但是图中框处的两个地方,一个是商品类别,需要显示具体的类别名称,而状态也要显示成在售/下架,我么先来处理简单的,对于商品状态,于数据库无关,只要显示的时候显示中文状态就行了,可以直接通过jsp完成
然后是商品类型:商品类型涉及到多表连接查询,这里合理的利用数据冗余,给实体类Product添加一个属性:catename,并添加他的get、set方法
private String catename;
public String getCatename() {
return catename;
}
public void setCatename(String catename) {
this.catename = catename;
}
修改ProductMapper.xml文件中的bean
在<resultMap></resultMap>
中添加映射
<result column="catename" property="catename" jdbcType="VARCHAR" />
修改selectProductList的bean为:
<select id="selectProductList" resultMap="BaseResultMap">
select
product.proid,
product.cateid,
product.`name`,
product.subtitle,
product.mainimage,
product.price,
product.stock,
product.`status`,
category.`name` as catename
from
product
inner join category on product.cateid = category.cateid
</select>
最后修改jsp页面
运行项目并测试:访问http://localhost:8080/product/getProductList
查询显示搞定。
2.添加商品
这里同样,我们把商品详情和商品图片在下一节内容里完成,所以经过分析,添加商品功能需要操作的数据字段就包括:商品编号、商品类型、商品名称、商品副标题、商品主图、商品价格、库存数量、商品状态、创建时间,其中商品编号和创建时间由程序生成,商品类型和商品状态使用下拉菜单实现,商品主图通过文件上传完成,其余字段值通过文本框输入,由此设计出页面如下:
对于文本框,无需赘述,下拉菜单,有两个,其一是商品状态,只有在售和下架两个状态,相对固定不涉及数据库操作,而商品类型是一个动态数据,需要随数据库变化而变化,他里面的值需要从数据库中读取在我们进入添加页面的时候就需要初始化添加页面,把“可用的”商品类型加载到下拉框中,对于读取商品类型列表的方法,在商品类型管理中已经有了,但是在商品管理中,查询的是全部的商品类型,而这里,我们添加商品的时候,所能选择的只是“可用的”商品类型,所以这里我们去修改selectCategoryList的方法,添加参数,通过设置查询条件查询出“可用的”商品类型。
2.1、添加页面初始化
首先修改CategoryMapper接口中的selectCategoryList方法,增加一个参数
List<Category> selectCategoryList(Category category);
修改mapper文件,增加条件
<select id="selectCategoryList" resultType="cn.yuechenc.pojo.Category"
parameterType="cn.yuechenc.pojo.Category">
select
<include refid="Base_Column_List" />
from category
<if test="status!=null">
where status=#{status}
</if>
</select>
为service接口及其实现类添加参数
List<Category> selectCategoryList(Category category);
@Override
public List<Category> selectCategoryList(Category category) {
return categoryMapper.selectCategoryList(category);
}
下面在controller类里面添加方法
@Autowired
private CategoryService categoryService;
@RequestMapping("/toadd")
private String toaddProduct(ModelMap model){
Category category=new Category();
category.setStatus(1);
List<Category> cateList=categoryService.selectCategoryList(category);
model.addAttribute("catelist", cateList);
return "ategoryEdit";
}
然后修改页面用来绑定商品类型下拉框
现在我们可以去测试一下:访问http://localhost:8080/product/toadd
可以看出,下拉菜单的数据来自于数据库中“可用的”商品类型
那么初始化就完成了
2.2、添加商品
现在在我们添加商品的时候需要添加一张主图,这张主图采用文件上传的方式,保存到我们的分布式文件系统FastDFS中,具体的上传工具类的编写请参考我的博客:
分布式商城项目-后台开发-文件上传和下载
使用这里的上传工具类,就能上传图片,并且得到图片存储的路径,只需要将存储路径保存到数据库即可,此处的操作和添加商品类型,是一样的,从页面获取到用户提交的数据,保存到数据库即可。
首先在service中添加方法:
public int insertProduct(Product product);
在实现类里面去实现该方法,
@Override
public int insertProduct(Product product) {
return productMapper.insertSelective(product);
}
下面就可以编写controller了
@RequestMapping("/addProduct")
private String addProduct(ModelMap model, Product product, @RequestParam("file") MultipartFile file)
throws Exception {
String filename = file.getOriginalFilename();
if (filename != null && !filename.equals("")) {
FastDFSClient dfs = new FastDFSClient();
String extName = filename.substring(filename.lastIndexOf(".") + 1);
String url = dfs.uploadFile(file.getBytes(), extName);
product.setMainimage(url);
}
product.setProid(UUID.randomUUID().toString());
Date now=new Date();
product.setCreatetime(now);
int rtn=productService.insertProduct(product);
if(rtn>0){
return "redirect:/product/getProductList";
}else{
model.addAttribute("msg", "添加商品失败!");
return "redirect:/product/toadd";
}
}
修改页面form属性,和各控件name属性,使之和实体属性名一致
<form class="mws-form" action="${pageContext.request.contextPath}/product/addProduct" method="post" enctype="multipart/form-data">
这里我们进行一些优化,如图加入上传图片的预览效果:
在jsp页面修改商品主图控件为:
<div class="mws-form-row">
<label>商品主图</label>
<div class="mws-form-item large">
<input id="doc" type="file" name="file" value="image"
onchange="javascript:setImagePreview();"
class="mws-textinput" />
<img id="preview" width=-1
height=-1 style="diplay: none;padding-top: 5px" />
</div>
</div>
添加:
<script type="text/javascript">
function setImagePreview() {
var docObj = document.getElementById("doc");
var imgObjPreview = document.getElementById("preview");
if (docObj.files && docObj.files[0]) {
//火狐下,直接设img属性
imgObjPreview.style.display = 'block';
imgObjPreview.style.width = '200px';
imgObjPreview.style.height = '120px';
//imgObjPreview.src = docObj.files[0].getAsDataURL();
//火狐7以上版本不能用上面的getAsDataURL()方式获取,需要一下方式
imgObjPreview.src = window.URL.createObjectURL(docObj.files[0]);
} else {
//IE下,使用滤镜
docObj.select();
var imgSrc = document.selection.createRange().text;
var localImagId = document.getElementById("localImag");
//必须设置初始大小
localImagId.style.width = "250px";
localImagId.style.height = "200px";
//图片异常的捕捉,防止用户修改后缀来伪造图片
try {
localImagId.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(sizingMethod=scale)";
localImagId.filters
.item("DXImageTransform.Microsoft.AlphaImageLoader").src = imgSrc;
} catch (e) {
alert("您上传的图片格式不正确,请重新选择!");
return false;
}
imgObjPreview.style.display = 'none';
document.selection.empty();
}
return true;
}
</script>
这时候,基本上添加功能就算完成了。可以运行测试一下:
2.3、修改商品
这里的修改跟修改商品类别一样,首先是商品修改页面的初始化,这里的修改页面我们也跟添加页面使用同一个,
初始化修改页面
第一步,需要通过id把这条记录查询出来,这里需要服务处的支持,在ProductService中添加方法
public Product selectProductById(String proid);
并且在实现类中实现该方法
@Override
public Product selectProductById(String proid) {
return productMapper.selectByPrimaryKey(proid);
}
接下来编写controller
@RequestMapping("/toedit")
private String toeditProduct(ModelMap model,String proid){
Product product=productService.selectProductById(proid);
model.addAttribute("product", product);
model.addAttribute("editType", "editProduct");
return "ProductEdit";
}
修改jsp页面绑定数据
下面测试一下,进入管理页面点击一个商品的编辑按钮
可以看到初始化成功了,因为主图只有一张,所以不上传图片,他就不变,上传就会覆盖原来的图片,那么接下来就只需要把修改的信息更新到数据库了
修改商品
编写service接口,添加update方法:
public int updateProduct(Product product);
然后再实现类中实现此方法,调用dao层操作数据库数据
@Override
public int updateProduct(Product product) {
return productMapper.updateByPrimaryKeySelective(product);
}
最后编写controller
@RequestMapping("/editProduct")
private String editProduct(ModelMap model, Product product, @RequestParam("file") MultipartFile file)
throws Exception {
String filename = file.getOriginalFilename();
if (filename != null && !filename.equals("")) {
FastDFSClient dfs = new FastDFSClient();
String extName = filename.substring(filename.lastIndexOf(".") + 1);
String url = dfs.uploadFile(file.getBytes(), extName);
product.setMainimage(url);
}
Date now=new Date();
product.setUpdatetime(now);
int rtn=productService.updateProduct(product);
if(rtn>0){
return "redirect:/product/getProductList";
}else{
model.addAttribute("msg", "修改商品失败!");
return "redirect:/product/toedit?proid="+product.getProid();
}
}
到这里修改功能也实现了,下面我们来测试一下
点击编辑,修改最后一条记录
修改成功!