获取多级类别组合下的产品

本篇是针对我在做项目过程中遇到的特定需求而做的一个Demo, 没有很大的通用性,读者酌情可绕行。

标题不能完全表达本意,确切的情景需要展开说。假设有三级分类,关于分类这样设计:

    public class Category
    {
        public int Id { get; set; }
        public string Name { get; set; }
        public int ParentId { get; set; }
    }

然后产品可以属于多个分类,以下的Categories属性值是以英文逗号隔开、由分类编号拼接而成的字符串。

    public class Product
    {
        public int Id { get; set; }
        public string Name { get; set; }
        public string Categories { get; set; }
    }

由于种种原因,Categories属性值只是存储了由第三级分类编号拼接而成的字符串。

在前端,需要把分类作为查询条件来查询产品,可能只选择一级分类,把一个数字字符串(比如"1")发送给服务端;可能同时选择一级和二级分类,也把一个数字字符串(比如"1,2")发送给服务端;当然,也有可能同时选择一级、二级和三级分类作为查询条件(比如"1,2,3")。换句话说,如果诸如"1"或"1,2"或"1,2,3"这样的查询条件转换成数组后,如果数组的每一个元素都被包含在Product的Categories属性值转换成的数组中,那这个产品就符合搜索条件。

简单来说,是这样:假设搜索条件是"1,2",Product的Categories属性值为"1,3,2,5",我们不是判断"1,2"这个字符串是否包含在"1,3,2,5"字符串中,而是把"1,2"先split成数组,叫做array1, 把"1,3,2,5"也split成数组,叫做array2,最后判断array1的每个元素是否都被包含在array2中。

还有一个问题需要解决:当前的Product的Categories属性值只存储了所有第三级分类编号拼接成的字符串,而前端输入的搜索条件可能会包含一级分类或二级分类等,所以,我们需要把Product转换一下,希望有一个类的某个属性值能存储由一级、二级、三级分类拼接而成的字符串。

    public class ProductWithThreeCate
    {
        public int Id { get; set; }
        public string Name { get; set; }
        public string AllCategoreis { get; set; }
    }

以上, AllCategoreis属性值就用来存储由一级、二级、三级分类拼接而成的字符串。

有一个方法获取所有分类:

        static List<Category> GetCategories()
        {
            return new List<Category>()
            {
                new Category(){Id = 1, Name = "根", ParentId = -1},
                new Category(){Id = 2, Name = "一级分类1",ParentId = 1},
                new Category(){Id = 3, Name = "一级分类2", ParentId = 1},
                new Category(){Id = 4, Name = "二级分类11",ParentId = 2},
                new Category(){Id = 5, Name = "二级分类12",ParentId = 2},
                new Category(){Id = 6, Name = "二级分类21",ParentId = 3},
                new Category(){Id = 7, Name = "二级分类22",ParentId = 3},
                new Category(){Id = 8, Name = "三级分类111",ParentId = 4},
                new Category(){Id = 9, Name = "三级分类112",ParentId = 4},
                new Category(){Id = 10, Name = "三级分类121",ParentId = 5},
                new Category(){Id = 11, Name = "三级分类122",ParentId = 5},
                new Category(){Id = 12, Name = "三级分类211",ParentId = 6},
                new Category(){Id = 13, Name = "三级分类212",ParentId = 6},
                new Category(){Id = 14, Name = "三级分类221",ParentId = 7}
            };
        }

有一个方法获取所有产品:

        static List<Product> GetProducts()
        {
            return new List<Product>()
            {
                new Product(){Id = 1, Name = "产品1",Categories = "10,12"},
                new Product(){Id = 2, Name = "产品2", Categories = "12,13"},
                new Product(){Id = 3, Name = "产品3",Categories = "10,11,12"},
                new Product(){Id = 4, Name = "产品4",Categories = "13,14"},
                new Product(){Id = 5, Name = "产品5",Categories = "11,13,14"}
            };
        }    

接下来的方法是根据搜索条件(比如是"1,2")来查找满足条件的ProductWithThreeCate集合,如下:

        /// <summary>
        /// 获取满足某些条件的集合
        /// </summary>
        /// <param name="query">以英文逗号隔开的字符串,比如:2,5</param>
        /// <returns></returns>
        static List<ProductWithThreeCate> GetResultByQuery(string query)
        {
            //最终结果
            List<ProductWithThreeCate> result = new List<ProductWithThreeCate>();

            //临时结果 此时ProductWithThreeCat的属性AllCategoreis包含所有一级、二级、三级分类ID拼接成的字符串
            List<ProductWithThreeCate> tempResult = new List<ProductWithThreeCate>();

            //获取所有的产品
            List<Product> allProducts = GetProducts();

            //遍历这些产品
            foreach (var item in allProducts)
            {
                ProductWithThreeCate productWithThreeCate = new ProductWithThreeCate();
                productWithThreeCate.Id = item.Id;
                productWithThreeCate.Name = item.Name;

                //所有一级、二级、三级拼接成以英文逗号隔开的字符串
                string temp = string.Empty;

                //当前产品只包含三级拼接成的、也是以英文隔开的字符串,split成数组
                string[] theThirdCates = item.Categories.Split(‘,‘);

                //遍历这些三级数组
                foreach (string i in theThirdCates)
                {
                    //三级类别转换成整型
                    int theThirdInt = int.Parse(i);

                    //获取三级类别
                    Category theThirdCate = GetCategories().Where(c => c.Id == theThirdInt).FirstOrDefault();

                    //获取二级类别
                    Category theSecondCate = GetCategories().Where(c => c.Id == theThirdCate.ParentId).FirstOrDefault();

                    //获取一级类别
                    Category theFirstCate = GetCategories().Where(c => c.Id == theSecondCate.ParentId).FirstOrDefault();

                    temp += i + "," + theSecondCate.Id.ToString() + "," + theFirstCate.Id.ToString() + ",";
                }

                //去掉最后一个英文逗号
                temp = temp.Substring(0, temp.Length - 1);

                //转换成集合,去除重复项,比如不同的三级可能有相同的一级或二级父类
                IEnumerable<string> tempArray = temp.Split(‘,‘).AsEnumerable().Distinct();

                //所有一级、二级、三级拼接成以英文逗号隔开的字符串,但已经去除了重复的一级和二级
                string tempagain = string.Empty;

                //再次遍历集合拼接成字符串
                foreach (var s in tempArray)
                {
                    tempagain += s + ",";
                }
                productWithThreeCate.AllCategoreis = tempagain.Substring(0, tempagain.Length - 1);
                tempResult.Add(productWithThreeCate);
            }

            //遍历临时结果
            foreach (var item in tempResult)
            {
                //把当前包含一级、二级、三级的,以英文逗号隔开的字符串split成数组
                string[] itemArray = item.AllCategoreis.Split(‘,‘);

                //把当前查询字符串split成数组
                string[] queryArray = query.Split(‘,‘);

                //如果queryArray的每一个元素都被包含在itemArray中,那就保存起来
                if (queryArray.All(x => itemArray.Contains(x)) == true)
                {
                    result.Add(item);
                }
            }

            return result;
        }

客户端的调用如下:

            List<ProductWithThreeCate> result = GetResultByQuery("2,5");

            //遍历最终的结果
            foreach (var item in result)
            {
                Console.WriteLine(item.Name+ "  " + item.AllCategoreis);
            }
            Console.ReadKey();

时间: 2024-08-02 16:28:24

获取多级类别组合下的产品的相关文章

千万级下载量产品深度体验-前言

随着移动互联网的快速发展,千万级甚至亿级下载量的产品也出现了很多,这些产品无疑都是各个领域中的精英.之所以能够让如此多的用户安装使用产品,一方面是产品本身的优秀,包括功能上的创新或是出众的用户体验,另一外面也展现了这些企业优秀的推广能力.在如今的互联网时代,得用户者得天下的趋势开始上演的愈加强烈.很多产品已经不光是在功能上吸引用户,更是希望将产品打造成一件艺术品,从而产生粉丝效应,长时间的保证用户的黏度. 正所谓知彼知己百战不殆,想要设计一款优秀的产品,必须熟知行业中同类产品的优势和劣势,才能取

Woocommerce 分类下的产品如何使用ID号来作为默认排序字段

在给一个客户开发网店系统时使用了WordPress系统的Woocommerce插件 WordPress版本:3.8 Woocommerce版本:2.0.20 如果没有指定排序规则(指定的字段),则Woocommerce默认使用的是"post_title"来排序的,但我在调用分类下的产品时使用"ID"字段来自定义排序规则,但实际上却并未按照自定义的规则来排序,而是使用”ORDER BY menu_order,post_title ASC“来排序(在product_ca

jquery如何获取当前元素的下一个元素

jquery如何获取当前元素的下一个元素: 本章节介绍一下如何利用jquery获取当前元素的下一个元素,如何利用原生js获取下一个元素可以参阅原生js获取当前元素的下一个元素代码实例一章节, 下面直接看代码: <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="author" content="http://www.51tex

用Java实现将多级文件夹下的所有文件统一放到一个文件夹中

每次下了电影(男生懂得呦),每部电影都放在一个单独的文件夹里,看的时候很是不方便啊,一直重复着进入文件夹.后退,再进.再退的操作,而手动把这些电影全部复制出来又太繁琐.因此为了解决这个问题,用IO写了一个小工具. 以下代码只实现了把多级文件夹下的所有文件复制到一个文件夹中,如果需要指定复制文件的格式,可以在1111处添加一个判断.如果需要同时删除原文件夹可以在添加一个递归删除的方法. package Bao; import java.io.BufferedInputStream;import j

获取某文件夹下所有文件名、文件夹名、后缀名

今天在提交github时,突然觉得提交得很慢,想到github允许设置.gitignore文件进行某些后缀名的忽略,于是乎决定好好设置一下.gitimnore. 于是乎....我发现,一个很大的项目,或者一个很杂乱的文件库,很难知道这里面到底存在哪些后缀名的文件,一个文件夹一个文件夹的找又太慢,而且容易出错. 于是乎....我开始敲代码了,顺带着把获取某文件夹下所有文件名.文件夹名.后缀名这三个功能一起实现了. [来看效果] 不输入参数时或者第一个参数输入"\"时,默认为当前目录,显示

多级派生情况下派生类的构造函数

#include <iostream> #include<string> using namespace std; class Student//声明基类 { public://公用部分 Student(int n, string nam)//基类构造函数 { num=n; name=nam; } void display( )//输出基类数据成员 { cout<<"num:"<<num<<endl; cout<<

IO流的练习3 复制多级文件夹下的指定文件并改名

需求:复制指定目录下的指定文件,并修改后缀名. 指定的文件是:.java文件. 指定的后缀名是:.jad 数据源所在文件夹:C:\Users\Administrator\Desktop\记录 目的地所在文件夹:C:\Users\Administrator\Desktop\新建文件夹\copy 分析: A:封装数据源的目录 B:封装目的地的目录 判断路径下的文件是否是文件夹 是:在目的文件夹下创建该文件, 获取数据源所在文件夹的所有文件的File数组 遍历File数组,得到每个文件的File对象

千万级下载量产品深度体验&mdash;手机音乐客户端

VINCENT MK http://www.cnblogs.com/mkvin/ QQ:350510376 引言 手机音乐客户端是每个手机用户几乎都高频率使用的一个产品,从功能机时代到智能机时代都一直如此.听音乐似乎是人们的天性,伴随音乐的响起,在各种复杂的生理因素的作用下,音乐可以带给人幸福和满足感. 一款好的音乐客户端产品,必须要满足用户查找歌曲.听歌曲这两个基本的诉求. 目前的音乐客户端都一定会有本地和在线听音乐的功能,当然在线音乐功能需要产品提供海量的曲库和优秀的音频来源,这是吸引用户使

DWR、Comet4j在Nginx+Tomcat组合下的优化

DWR.Comet4j这类推送框架在Tomcat下运行正常,但在nginx+tomcat组合下,可能会出现断连.延迟等各种问题. 如出现此类问题,可尝试以下优化方式: 1.Nginx-----nginx.conf worker_processes 1; events { worker_connections 1024; } http { include mime.types; default_type application/octet-stream; sendfile on; #keepali