1、苏宁百万级商品爬取 思路讲解 类别爬取

简介: 苏宁类别面 https://list.suning.com/解析图.png通过图可知,总共有N个类别,每个类别都是一个DIV区块,然后再继续分解DIV区块分析内容。

苏宁类别面 https://list.suning.com/

img_d18be20edb852aca98758c72389fa4ba.png
解析图.png

通过图可知,总共有N个类别,每个类别都是一个DIV区块,然后再继续分解DIV区块分析内容。我们要得到的是类别表,据图所示我们可以分析得出类别表的结构应当树形的。所以涉及的表应该是包含子节点和父节点的。初步设计图如下

Id Pid Code Name Url
主键 父节点 编码 名称 地址
img_6b607c8210fa325ecd9b7ea51ac88521.png
解析图2.png

我们可以得到解析图2对应的 Xpath为://*[@id="20089"]/div[2]/div[2] 。可是因为是通过ID作为唯一Key来向下找,所以我们需要先得到所有的Key值。这个方法被我放弃而选用了另外一种方式。

/html/body/div[5]/div[2].首先找到如果所示xpath对应的内容

img_0094a814433bc507a254d329f09c5d90.png
/html/body/div[5]/div[2].png

那么如果我们想要得到下属的内容只需要增加一个后缀
/html/body/div[5]/div[2] /div
此时我们得到了所有模块的内容,那么我们接下去分析

img_6827da9b101ea1a956d8f72e00822d62.png
一级.png

img_3c028f0b4c815b75fbeac62ec2d916de.png
二级+三级.png

还是以“手机配件”为例。一级类别,二级类别、三级类别如果所示。我们又如何得到内容,然后将其变成单元行的形式插入数据库中呢?

解决方案如下
根据网页内容可知,一级类别包含着二级类别,二级类别包含着三级类别。所以我们可以采用如下方式。
首先获取所有一级类别,即解析图2.png所示内容。
一级类别 A方法
循环当前内容
1、解析内容 增加当前A级类别实体
2、循环包含的二级内容,处理
3、合并实体
二级类别 B方法
循环当前内容
1、解析内容 增加当前B级类别实体
2、循环包含的三级级内容,处理
3、返回实体给A方法
三级类别 C方法
循环当前内容
1、解析内容 增加当前C级类别实体
2、返回实体给B方法

img_bc1e1020550d54ec2487b7fd67836999.png
ABC.png


代码讲解

ABC(Combine)方法
遍历InitA方法获取的内容,增加A实体后将ANode作为参数传递给InitB方法。依次类推,最后得到符合要求的实体。

    private static List<POCO_Category> CombineA_B_C()
        {
            List<POCO_Category> AList = new List<POCO_Category>();

            int idIndex = 1;
            foreach (HtmlNode xNode in InitA())
            {
                POCO_Category aModel = new POCO_Category()
                {
                    Id = ToLevelCode(idIndex),
                    PId = "000",
                    Levels = 1,
                    Code = ToLevelCode(idIndex),
                    Name = xNode.SelectSingleNode("./h2").InnerText
                };
                AList.Add(aModel);

                var blist = InitB(aModel, xNode);
                AList.AddRange(blist);
                idIndex = idIndex + blist.Count + 1;
            }

            return AList;
        }

        private static List<HtmlNode> InitA()
        {
            var url = "https://list.suning.com/#20089";
            var web = new HtmlWeb();
            var docWeb = web.Load(url);
            //var cssNodes = docWeb.DocumentNode.CssSelect(".search-main.introduce.clearfix > div").ToList();//147毫秒
            List<HtmlNode> xpathNodes = docWeb.DocumentNode.SelectNodes("/html/body/div[5]/div[2]/div").ToList();
            return xpathNodes;
        }

        private static List<POCO_Category> InitB(POCO_Category parentModel, HtmlNode node)
        {
            int idIndex = Convert.ToInt32(parentModel.Id) + 1;
            List<POCO_Category> bList = new List<POCO_Category>();

            var xNodes = node.SelectNodes("./div").ToList();
            foreach (var xNode in xNodes)
            {
                var cateModel = xNode.SelectSingleNode("./div[1]/a");
                POCO_Category bModel = new POCO_Category()
                {
                    Id = ToLevelCode(idIndex),
                    PId = parentModel.Id,
                    Code = $"{parentModel.Code}_{ToLevelCode(idIndex)}",
                    Name = cateModel.InnerText,
                    Url = $"https:{cateModel.GetAttributeValue("href")}",
                    Levels = 2
                };
                bList.Add(bModel);

                var clist = InitC(bModel, xNode.SelectSingleNode("./div[2]"));
                bList.AddRange(clist);
                idIndex = idIndex + clist.Count + 1;
            }

            return bList;
        }

        private static List<POCO_Category> InitC(POCO_Category parentModel, HtmlNode node)
        {
            int idIndex = Convert.ToInt32(parentModel.Id) + 1;
            List<POCO_Category> cList = new List<POCO_Category>();

            HtmlNodeCollection xNodes = node.SelectNodes("./a");

            if (xNodes != null && xNodes.Count > 0)
            {
                foreach (var xNode in xNodes)
                {
                    POCO_Category cModel = new POCO_Category()
                    {
                        Id = ToLevelCode(idIndex),
                        PId = parentModel.Id,
                        Code = $"{parentModel.Code}_{ToLevelCode(idIndex)}",
                        Name = xNode.InnerText,
                        Url = $"https:{xNode.GetAttributeValue("href")}",
                        Levels = 3
                    };
                    cList.Add(cModel);
                    idIndex += 1;
                }
            }

            return cList;
        }

        private static string ToLevelCode(int index)
        {
            return index.ToString("000");
        }
目录
相关文章
|
4月前
|
缓存 负载均衡 网络协议
电商API接口性能优化技术揭秘:缓存策略与负载均衡详解
电商API接口性能优化是提升系统稳定性和用户体验的关键。本文聚焦缓存策略与负载均衡两大核心,详解其在电商业务中的实践。缓存策略涵盖本地、分布式及CDN缓存,通过全量或部分缓存设计和一致性维护,减少后端压力;负载均衡则利用反向代理、DNS轮询等技术,结合动态调整与冗余部署,提高吞吐量与可用性。文中引用大型及跨境电商平台案例,展示优化效果,强调持续监控与迭代的重要性,为电商企业提供了切实可行的性能优化路径。
|
3月前
|
人工智能 自然语言处理 监控
掌握这6大环节,设计懂你所问的AI智能问答系统
三桥君深入解析企业智能化升级核心——AI大脑的构建路径。从RPA流程自动化、AI能力、AI中台到IoT平台,结合行业解决方案,助力企业实现智能运营,提升竞争力
275 5
|
3月前
|
人工智能 自然语言处理 API
构建可落地的企业AI Agent,背后隐藏着怎样的技术密码?
三桥君深入解析企业AI Agent技术架构,涵盖语音识别、意图理解、知识库协同、语音合成等核心模块,探讨如何实现业务闭环与高效人机交互,助力企业智能化升级。
212 6
|
3月前
|
人工智能 测试技术 API
Apifox 和 Apipost如何选?2025企业API开发工具选型需求分析及建议
本文对比了 Apipost 与 Apifox 在 AI 功能及 API 功能上的差异,指出 Apipost 凭借 AI 一键补全文档、智能提取 API 文档、AI 断言、模拟测试数据、生成用例、参数更新、脚本生成、全局搜索等能力,显著提升开发效率与质量。同时,Apipost 在离线使用、一键分享、OpenAPI 格式支持、多协议适配、数据库导入、模拟数据、压测功能等基础 API 能力上亦全面领先。在AI时代的2025年,API + AI是Apipost将AI技术融合行业应用的最佳典范,这种趋势下,也说明Apipost 更能助力企业与开发者实现高效智能开发。
182 2
|
7月前
|
数据可视化 JavaScript 前端开发
利用Postman和Apipost进行API测试的实践与优化-动态参数
在API测试中,Postman和Apipost是常用的工具。Postman内置变量功能有限,面对复杂场景时需编写JavaScript脚本,增加了维护成本。而Apipost提供丰富的内置变量、可视化动态值配置和低代码操作,支持生成真实随机数据,如邮箱、手机号等,显著提升测试效率和灵活性。对于复杂测试场景,Apipost是更好的选择,能有效降低开发与维护成本,提高测试工作的便捷性和可维护性。
|
存储 自然语言处理 算法
向量数据库Chroma极简教程
本文重点围绕向量数据库Chroma的使用和实战,主要包括以下内容: * Chroma设计理念 * Chroma常见概念(数据集,文档,存储,查询,条件过滤) * Chroma快速上手 * Chroma支持的Embeddings算法 * 实战:在Langchain中使用Chroma对中国古典四大名著进行相似性查询
2525 2
IDEA创建多模块项目常用pom
IDEA创建多模块项目常用pom
497 8
|
消息中间件 监控 Linux
RabbitMQ轻松入门:从零开始的部署与安装指南
RabbitMQ轻松入门:从零开始的部署与安装指南
317 0
RabbitMQ轻松入门:从零开始的部署与安装指南
|
前端开发 JavaScript
受控组件和非受控组件的区别
受控组件和非受控组件的区别
npm-check【实用教程】升级项目中的依赖
npm-check【实用教程】升级项目中的依赖
338 0