谷粒商城【成神路】-【9】——商城页面

本文介绍了如何使用Nginx进行动静态资源分离,Thymealf模板引擎的配置,关闭SpringBoot中的缓存,以及通过数据库查询优化提高性能。还详细描述了通过Nginx转发和负载均衡的实现策略。

目录

🧈1.项目服务部署架构

🥞2.Thymealf

🍿3.请求接口

🌭4.使用nginx转发

🥖5.nginx动静分离 

🫓6.优化 


1.项目服务部署架构

使用nginx动静分离,使图片、js等静态资源和服务器请求分开

2.Thymealf

1.导入依赖 

        <!--thymeleaf-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-thymeleaf</artifactId>
        </dependency>

 2.导入资源

3.关闭缓存

spring:
    thymeleaf:
        cache: false #关闭缓存
  •  静态资源全部放在static文件夹下就可以按照路径直接访问
  • 页面放在templates,直接访问
  • springboot访问项目时,默认找index

3.请求接口

根据页面请求数据编写controller

@Controller
public class IndexController {


    @Autowired
    CategoryService categoryService;

    @GetMapping({"/", "/index.html"})
    public String indexPage(Model model) {

        //TODO 查询所有的一级分类
        List<CategoryEntity> categoryEntities = categoryService.getLeve1Categorys();

        //视图解析器,加上前缀,加上后缀
        model.addAttribute("categorys",categoryEntities);
        return "index";
    }

    @ResponseBody
    @GetMapping("/index/catalog.json")
    public  Map<String, List<Catalog2Vo>> getCatalogJson(){
        Map<String, List<Catalog2Vo>> catalogJson=categoryService.getCatalogJson();
        return catalogJson;
    }

}

具体的业务逻辑实现 

/**
     * 查询所有一级分类
     *
     * @return
     */
    @Override
    public List<CategoryEntity> getLeve1Categorys() {
        List<CategoryEntity> categoryEntities = baseMapper.selectList(new QueryWrapper<CategoryEntity>().eq("parent_cid", 0));
        return categoryEntities;
    }

    /**
     * 查出所有分类
     *
     * @return
     */
    @Override
    public Map<String, List<Catalog2Vo>> getCatalogJson() {

        /**
         * 优化:将数据库查询的多次变为一次
         */

        List<CategoryEntity> selectList = baseMapper.selectList(null);

        //1.查出所有1级分类
        List<CategoryEntity> leve1Categorys = getParent_cid(selectList, 0L);

        //2.封装数据
        Map<String, List<Catalog2Vo>> parentCid = leve1Categorys.stream().collect(Collectors.toMap(k -> k.getCatId().toString(), v -> {
            //1.每一个一级分类
            List<CategoryEntity> categoryEntities = getParent_cid(selectList, v.getCatId());
            List<Catalog2Vo> catalog2Vos = null;
            if (categoryEntities != null) {
                catalog2Vos = categoryEntities.stream().map(l2 -> {
                    Catalog2Vo vo = new Catalog2Vo(v.getCatId().toString(), null, l2.getCatId().toString(), l2.getName());
                    //找二级分类的三级分类
                    List<CategoryEntity> categoryEntities3 = getParent_cid(selectList, l2.getCatId());
                    if (categoryEntities3 != null) {
                        List<Catalog2Vo.Catalog3Vo> collect = categoryEntities3.stream().map(l3 -> {
                            Catalog2Vo.Catalog3Vo catalog3Vo = new Catalog2Vo.Catalog3Vo(l2.getCatId().toString(), l3.getCatId().toString(), l3.getName());
                            return catalog3Vo;
                        }).collect(Collectors.toList());
                        vo.setCatalog3List(collect);
                    }
                    return vo;
                }).collect(Collectors.toList());
            }
            return catalog2Vos;
        }));

        return parentCid;
    }

4.使用nginx转发

  • 1.在nginx配置,路由到网关的负载均衡

 nginx代理给网关时,会丢失请求的host信息

  • 2.在网关中配置陆游规则 

域名路由,一定要放到最后!!

转发思路:

在host文件配置gulimall映射的虚拟机地址,虚拟机中安装了nginx,默认访问80端口,这个端口专门监听gulimall域名,然后nginx会代理到网关,(在代理到网关的过程中会丢失Host头),网关根据陆游规则,匹配到具体的服务

5.nginx动静分离 

1.所有的静态资源全都放到nginx下

2.所有:/static/**请求,都有nginx返回

5.1.文件上传到nginx 

在nginx的html目录下创建static目录,并将index文件全部上传到inginx的static目录下,并删除本地的index,并在前端页面都添加/static/

温馨提示:catalogLoader.js 文件中也要添加/static/不然 catalog.json渲染不上

5.2修改nginx配置 

6.优化 

在没有话之间我们每一次都要查数据库,导致性能不高,我们可以直接一次性查出所有,然后从缓存中获取指定的数据

/**
     * 查出所有分类
     *
     * @return
     */
    @Override
    public Map<String, List<Catalog2Vo>> getCatalogJson() {

        /**
         * 优化:将数据库查询的多次变为一次
         */

        List<CategoryEntity> selectList = baseMapper.selectList(null);

        //1.查出所有1级分类
        List<CategoryEntity> leve1Categorys = getParent_cid(selectList, 0L);

        //2.封装数据
        Map<String, List<Catalog2Vo>> parentCid = leve1Categorys.stream().collect(Collectors.toMap(k -> k.getCatId().toString(), v -> {
            //1.每一个一级分类
            List<CategoryEntity> categoryEntities = getParent_cid(selectList, v.getCatId());
            List<Catalog2Vo> catalog2Vos = null;
            if (categoryEntities != null) {
                catalog2Vos = categoryEntities.stream().map(l2 -> {
                    Catalog2Vo vo = new Catalog2Vo(v.getCatId().toString(), null, l2.getCatId().toString(), l2.getName());
                    //找二级分类的三级分类
                    List<CategoryEntity> categoryEntities3 = getParent_cid(selectList, l2.getCatId());
                    if (categoryEntities3 != null) {
                        List<Catalog2Vo.Catalog3Vo> collect = categoryEntities3.stream().map(l3 -> {
                            Catalog2Vo.Catalog3Vo catalog3Vo = new Catalog2Vo.Catalog3Vo(l2.getCatId().toString(), l3.getCatId().toString(), l3.getName());
                            return catalog3Vo;
                        }).collect(Collectors.toList());
                        vo.setCatalog3List(collect);
                    }
                    return vo;
                }).collect(Collectors.toList());
            }
            return catalog2Vos;
        }));

        return parentCid;
    }

这样就大大提高系统的吞吐量~ 

谷粒商城平台发布商品时,如果商品介绍图片上传失败,通常可能是由于文件存储服务(如阿里云OSS)的配置问题、网络请求异常或后端服务未正确处理请求所致。根据引用中提到的一些常见问题及解决方案,以下是可能的解决方法: ### 1. 检查OSS配置是否正确 商品图片上传通常依赖于对象存储服务(如阿里云OSS)。如果OSS的配置信息缺失或错误,会导致上传失败。例如,`ossClient`实例创建失败时会抛出类似 `Oss endpoint can&#39;t be empty` 的异常[^4]。 **解决方法:** - 确保在配置文件(如`application.yml`或`bootstrap.yml`)中正确配置了OSS的访问密钥、Endpoint、Bucket名称等信息。 - 检查是否在Nacos配置中心中正确配置了OSS相关的参数,并确保服务能够正确拉取这些配置。 ### 2. 确保Nacos服务注册与配置中心正常工作 如果网关或相关服务无法从Nacos获取配置或服务实例,可能导致请求失败。例如,引用中提到的503错误与服务未在Nacos中注册或配置名称不一致有关[^1]。 **解决方法:** - 登录Nacos控制台,确认`renren-fast`等服务已成功注册。 - 确认服务的`Data ID`和`Group`与配置文件中的名称一致。 - 如果服务配置有误,可以尝试重新引入Nacos客户端依赖并重启服务[^1]。 ### 3. 检查Spring Boot与Spring Cloud版本兼容性 版本不兼容也可能导致服务启动失败或某些Bean无法正确注入,从而影响图片上传功能。例如,Spring Boot版本过高可能导致与Spring Cloud组件的兼容性问题[^3]。 **解决方法:** - 确保使用的Spring Boot与Spring Cloud Alibaba版本兼容。 - 参考引用中使用的版本配置,例如将Spring Boot版本调整为`2.2.1.RELEASE`或`2.1.8.RELEASE`,Spring Cloud Alibaba版本为`2.1.0.RELEASE`[^3]。 ### 4. 检查网关依赖配置 在网关服务中,如果引入了不兼容的依赖或缺少必要的配置,可能导致请求被拦截或转发失败。 **解决方法:** - 确保网关服务中没有引入冲突的依赖,如引用中提到的`spring-cloud-starter-netflix-ribbon`可能导致的问题[^2]。 - 可尝试注释掉`gulimall-common`中可能导致冲突的依赖,并引入`spring-cloud-loadbalancer`以替代。 ### 5. 检查前端上传请求与后端接口匹配 商品图片上传功能依赖于前后端接口的正确对接。如果接口径、参数名称或请求方式不匹配,也可能导致上传失败。 **解决方法:** - 使用浏览器开发者工具(F12)查看上传请求的URL、请求头、请求体是否正确。 - 确认后端Controller方法是否正确接收并处理了上传请求,例如使用了`@RequestParam("file") MultipartFile file`等注解。 ### 示例代码:OSS上传接口示例 ```java @RestController @RequestMapping("/oss") public class OssController { @Autowired private OSS ossClient; @Value("${aliyun.oss.bucket}") private String bucketName; @PostMapping("/upload") public ResponseEntity<String> uploadFile(@RequestParam("file") MultipartFile file) { if (file.isEmpty()) { return ResponseEntity.badRequest().body("File is empty"); } String fileName = UUID.randomUUID() + file.getOriginalFilename(); try { ossClient.putObject(bucketName, fileName, new ByteArrayInputStream(file.getBytes())); String fileUrl = "https://" + bucketName + "." + "oss-cn-beijing.aliyuncs.com" + "/" + fileName; return ResponseEntity.ok(fileUrl); } catch (IOException e) { return ResponseEntity.status(500).body("Upload failed"); } } } ``` ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

会敲代码的小张

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值