淘淘商城第二十三课(添加商品的实现)

上节课我们一起学习了富文本编辑器的使用,这节课我们一起学习下商品添加的实现。

      在item-add.jsp当中,当点击提交按钮后,会触发submitForm方法,如下图所示。


       在提交表单前需要校验输入的内容是否合法,如下图所示。


       下面我们看下数据库中商品表的建表信息,可以看到价格定义的字段类型是long型,单位为分,之所以这样做是为了避免使用小数点,因为小数点使用起来比较麻烦。因此存到数据库中的价格都是价格(以元为单位)乘以100的(变为分)。


       我们发现在商品表当中没有商品描述这个字段,其实商品描述是专门用一张表来存放了,如下图所示。可以看到商品描述与商品ID是一一对应的,之所以把商品描述单独放到一张表当中是因为它是个大文本字段,存储的信息量非常大,对于不需要商品描述的查询情况来说连带这个字段查询会影响查询效率,因此单独存放。


      我们看看表单中是如何表示的,如下图所示,可以看到有两个<input>组件,第一个用来展示的价格,即单位为元的价格(这更符合用户的习惯),第二个<input>是个隐藏域,专门用来存放以分为单位而的价格(即将以元为单位的价格乘以100)。表单提交便会提交name="price"的价格并保存到数据库。


        提交表单会请求url为"/item/save",第二个参数$("#itemAddForm").serialize()是用来将表单的数据序列化为key-value形式的字符串。如果提交成功应该返回状态码为200。


       由于每个操作都需要有状态码来表示操作成功与否以及相关信息,因此我们定义一个TaotaoResult来专门处理,该类定义三个属性,分别是状态、消息及数据。由于这个类会被多个工程所使用,因此放到taotao-common的pojo目录下。


       TaotaoResult类的全部代码如下,里面最常用的便是ok方法和build方法。

  1. package com.taotao.common.pojo;
  2. import java.io.Serializable;
  3. import java.util.List;
  4. import com.fasterxml.jackson.databind.JsonNode;
  5. import com.fasterxml.jackson.databind.ObjectMapper;
  6. /**
  7. * 淘淘商城自定义响应结构
  8. */
  9. public class TaotaoResult implements Serializable{
  10. // 定义jackson对象
  11. private static final ObjectMapper MAPPER = new ObjectMapper();
  12. // 响应业务状态
  13. private Integer status;
  14. // 响应消息
  15. private String msg;
  16. // 响应中的数据
  17. private Object data;
  18. public static TaotaoResult build(Integer status, String msg, Object data) {
  19. return new TaotaoResult(status, msg, data);
  20. }
  21. public static TaotaoResult ok(Object data) {
  22. return new TaotaoResult(data);
  23. }
  24. public static TaotaoResult ok() {
  25. return new TaotaoResult(null);
  26. }
  27. public TaotaoResult() {
  28. }
  29. public static TaotaoResult build(Integer status, String msg) {
  30. return new TaotaoResult(status, msg, null);
  31. }
  32. public TaotaoResult(Integer status, String msg, Object data) {
  33. this.status = status;
  34. this.msg = msg;
  35. this.data = data;
  36. }
  37. public TaotaoResult(Object data) {
  38. this.status = 200;
  39. this.msg = "OK";
  40. this.data = data;
  41. }
  42. // public Boolean isOK() {
  43. // return this.status == 200;
  44. // }
  45. public Integer getStatus() {
  46. return status;
  47. }
  48. public void setStatus(Integer status) {
  49. this.status = status;
  50. }
  51. public String getMsg() {
  52. return msg;
  53. }
  54. public void setMsg(String msg) {
  55. this.msg = msg;
  56. }
  57. public Object getData() {
  58. return data;
  59. }
  60. public void setData(Object data) {
  61. this.data = data;
  62. }
  63. /**
  64. * 将json结果集转化为TaotaoResult对象
  65. *
  66. * @param jsonData json数据
  67. * @param clazz TaotaoResult中的object类型
  68. * @return
  69. */
  70. public static TaotaoResult formatToPojo(String jsonData, Class <?> clazz) {
  71. try {
  72. if (clazz == null) {
  73. return MAPPER.readValue(jsonData, TaotaoResult.class);
  74. }
  75. JsonNode jsonNode = MAPPER.readTree(jsonData);
  76. JsonNode data = jsonNode.get( "data");
  77. Object obj = null;
  78. if (clazz != null) {
  79. if (data.isObject()) {
  80. obj = MAPPER.readValue(data.traverse(), clazz);
  81. } else if (data.isTextual()) {
  82. obj = MAPPER.readValue(data.asText(), clazz);
  83. }
  84. }
  85. return build(jsonNode.get( "status").intValue(), jsonNode.get( "msg").asText(), obj);
  86. } catch ( Exception e) {
  87. return null;
  88. }
  89. }
  90. /**
  91. * 将json串转化为TaotaoResult对象
  92. *
  93. * @param json
  94. * @return
  95. */
  96. public static TaotaoResult format(String json) {
  97. try {
  98. return MAPPER.readValue(json, TaotaoResult.class);
  99. } catch ( Exception e) {
  100. e.printStackTrace();
  101. }
  102. return null;
  103. }
  104. /**
  105. * Object是集合转化
  106. *
  107. * @param jsonData json数据
  108. * @param clazz 集合中的类型
  109. * @return
  110. */
  111. public static TaotaoResult formatToList(String jsonData, Class<?> clazz) {
  112. try {
  113. JsonNode jsonNode = MAPPER.readTree(jsonData);
  114. JsonNode data = jsonNode.get("data");
  115. Object obj = null;
  116. if (data.isArray() && data.size() > 0) {
  117. obj = MAPPER.readValue(data.traverse(),
  118. MAPPER.getTypeFactory().constructCollectionType(List.class, clazz));
  119. }
  120. return build(jsonNode.get("status").intValue(), jsonNode.get("msg").asText(), obj);
  121. } catch (Exception e) {
  122. return null;
  123. }
  124. }
  125. }
      添加商品和商品描述对应的都是单表操作,因此我们使用逆向工程生成的代码即可,也就是说我们不用写Dao层的代码。下面我们来写Service层代码,首先在ItemService接口当中添加一个"添加商品"的接口(这一个接口要操作两张表,一张是商品表,另一张是商品描述表)。如下图所示,参数有两个,一个是商品表的pojo,另一个是商品描述。之所以要抛出异常是因为这个接口要操作两张表,而且这两张表的操作要都成功才叫成功,否则事务就回滚,因此异常要向上抛,在实现类代码中不能用try catch来捕获异常,因为这样的话springmvc会认为代码正常结束了,便不会回滚。


        下面我们到service层来实现这个接口,如下图所示,我们在itemServiceImpl当中实现了createItem接口,其中商品ID(也叫商品编号)是采用当前毫秒数加两位随机数来生成的,为了方便以后调用,我们专门封装了一个类,叫IDUtils,里面不仅封装了商品ID的生成方法还封装了图片名称的生成方法,由于该类会被多个工程使用,因此我们也放到taotao-common的utils目录下。


      当前ItemServiceImpl的全部代码如下

  1. package com.taotao.service.impl;
  2. import java.util.Date;
  3. import java.util.List;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.stereotype.Service;
  6. import com.github.pagehelper.PageHelper;
  7. import com.github.pagehelper.PageInfo;
  8. import com.taotao.common.pojo.EasyUIDataGridResult;
  9. import com.taotao.common.pojo.TaotaoResult;
  10. import com.taotao.common.utils.IDUtils;
  11. import com.taotao.mapper.TbItemDescMapper;
  12. import com.taotao.mapper.TbItemMapper;
  13. import com.taotao.pojo.TbItem;
  14. import com.taotao.pojo.TbItemDesc;
  15. import com.taotao.pojo.TbItemExample;
  16. import com.taotao.service.ItemService;
  17. @Service
  18. public class ItemServiceImpl implements ItemService {
  19. @Autowired
  20. private TbItemMapper itemMapper;
  21. @Autowired
  22. private TbItemDescMapper itemDescMapper;
  23. @Override
  24. public TbItem getItemById(long itemId) {
  25. TbItem tbItem = itemMapper.selectByPrimaryKey(itemId);
  26. return tbItem;
  27. }
  28. @Override
  29. public EasyUIDataGridResult getItemList(int page, int rows) {
  30. //设置分页信息
  31. PageHelper.startPage(page, rows);
  32. //执行查询
  33. TbItemExample example = new TbItemExample();
  34. List <TbItem> list = itemMapper.selectByExample(example);
  35. //获取查询结果
  36. PageInfo <TbItem> pageInfo = new PageInfo <>(list);
  37. EasyUIDataGridResult result = new EasyUIDataGridResult();
  38. result.setRows(list);
  39. result.setTotal(pageInfo.getTotal());
  40. //返回结果
  41. return result;
  42. }
  43. @Override
  44. public TaotaoResult createItem(TbItem tbItem, String desc) throws Exception{
  45. //生成商品ID
  46. long itemId = IDUtils.genItemId();
  47. //补全item的属性
  48. tbItem.setId(itemId);
  49. //商品状态,1-正常,2-下架,3-删除
  50. tbItem.setStatus(((byte) 1));
  51. tbItem.setCreated(new Date());
  52. tbItem.setUpdated(new Date());
  53. itemMapper.insert(tbItem);
  54. //添加商品描述
  55. insertItemDesc(itemId, desc);
  56. return TaotaoResult.ok();
  57. }
  58. //添加商品描述
  59. private void insertItemDesc(long itemId,String desc){
  60. //创建一个商品描述表对应的pojo
  61. TbItemDesc itemDesc = new TbItemDesc();
  62. //补全pojo的属性
  63. itemDesc.setItemId(itemId);
  64. itemDesc.setItemDesc(desc);
  65. itemDesc.setCreated(new Date());
  66. itemDesc.setUpdated(new Date());
  67. //向商品描述表插入数据
  68. itemDescMapper.insert(itemDesc);
  69. }
  70. }

       IDUtils类的代码如下。

  1. package com.taotao.utils;
  2. import java.util.Random;
  3. /**
  4. * 各种id生成策略
  5. * <p>Title: IDUtils </p>
  6. * <p>Description: </p>
  7. * @version 1.0
  8. */
  9. public class IDUtils {
  10. /**
  11. * 图片名生成
  12. */
  13. public static String genImageName() {
  14. //取当前时间的长整形值包含毫秒
  15. long millis = System.currentTimeMillis();
  16. //long millis = System.nanoTime();
  17. //加上三位随机数
  18. Random random = new Random();
  19. int end3 = random.nextInt(999);
  20. //如果不足三位前面补0
  21. String str = millis + String.format("%03d", end3);
  22. return str;
  23. }
  24. /**
  25. * 商品id生成
  26. */
  27. public static long genItemId() {
  28. //取当前时间的长整形值包含毫秒
  29. long millis = System.currentTimeMillis();
  30. //long millis = System.nanoTime();
  31. //加上两位随机数
  32. Random random = new Random();
  33. int end2 = random.nextInt(99);
  34. //如果不足两位前面补0
  35. String str = millis + String.format("%02d", end2);
  36. long id = new Long(str);
  37. return id;
  38. }
  39. public static void main(String[] args) {
  40. for(int i=0;i < 100;i++)
  41. System.out.println(genItemId());
  42. }
  43. }
      下面我们来写Controller层,我们在Controller层添加addItem方法,其中value="/item/save"是在list-add.jsp的js当中定义好的,我们要保持一致才行。在Controller层我们要捕获从实现类抛出的异常,避免异常直接抛到前台页面。



      下面我们便来试试我们的添加商品功能是否好使,由于taotao-common工程以及taotao-manager工程都做了修改,因此我们需要对这两个工程重新打包,打包方法是在工程上右键------>Run As------>Maven install。

      下面我们重启taotao-manager工程和taotao-manager-web工程,启动完工程之后,我们到新增商品页面,输入相关表单信息,然后点击"提交"按钮。


      我们会发现弹出一个提示框,提示我们添加成功,添加完之后,我们到商品列表中去查询,看是否有我们刚才添加的商品,我们直接查看最后一页的数据,发现最后一条就是我刚才添加的商品信息,说明添加商品成功了。


       我们再到数据库中看看添加的商品信息,首先看tb_item表,我们还是到最后一页去查看,发现有我刚才添加的信息,如下图所示。


       再看商品描述表,最后一页也有我们添加的商品描述信息,这说明数据存储完全没问题。这样,我们的商品添加功能便实现了。



猜你喜欢

转载自blog.csdn.net/anaitudou/article/details/80913414