题目要求写一个简易的CSS Selector。

首先用结构体\(<lev,label[],hasId,id[]>\)存储元素。其中\(lev\)表示元素在html树中的深度(这个是因为逻辑凌乱才加上的

接着用链式前向星存储html元素树。这里用一个栈\(rootStack\)方便找到新元素的父亲节点\(temp\)。

三种选择器都可以归结为第三种方式——后代选择器。

题目里已经给了算法:在匹配时,可以采用贪心的策略,除最后一级外,前面的部分都可以尽量匹配层级小的元素。写个dfs就好了。

注意标签不区分大小写,不可以直接用strcmp的。

字符串处理有点不方便。要熟练掌握%s,%[^],getchar,fgets,sscanf及相关函数。

#include <bits/stdc++.h>
const int maxn = 100;
const int maxm = 80; // max length of one element using namespace std; char line[maxm+10]; struct tElement
{
int lev;
char label[maxm+10];
bool hasId;
char id[maxm+10];
tElement()
{
hasId = false;
}
};
tElement element[maxn+10]; int to[maxn+10];
int nex[maxn+10];
int head[maxn+10]; char selector[maxm/2+10][maxm+10]; bool labelEqual(char s1[], char s2[])
{
if (strlen(s1) != strlen(s2))
return false;
for (int i = 0; s1[i] != '\0'; i++)
{
if (!((s1[i] == s2[i] || s1[i] + 32 == s2[i] || s1[i] - 32 == s2[i])))
return false;
}
return true;
} priority_queue<int, vector<int>, greater<int> > q;
int num; void dfs(int x, int cnt, int cnt0)
{
if (cnt != cnt0)
{
if ((selector[cnt][0] == '#' && element[x].hasId && strcmp(element[x].id, selector[cnt] + 1) == 0) ||
(selector[cnt][0] != '#' && labelEqual(element[x].label, selector[cnt])))
{
for (int i = head[x]; i != -1; i = nex[i])
{
int l = to[i];
dfs(l, cnt + 1, cnt0);
}
}
else
{
for (int i = head[x]; i != -1; i = nex[i])
{
int l = to[i];
dfs(l, cnt, cnt0);
}
}
}
else
{
if ((selector[cnt][0] == '#' && strcmp(element[x].id, selector[cnt] + 1) == 0) ||
(selector[cnt][0] != '#' && labelEqual(element[x].label, selector[cnt])))
{
q.push(x);
num ++;
}
for (int i = head[x]; i != -1; i = nex[i])
{
int l = to[i];
dfs(l, cnt, cnt0);
}
}
} int main()
{
int n, m;
scanf("%d%d", &n, &m);
getchar(); stack<int> rootStack;
memset(head, -1, sizeof(head));
for (int i = 1, cnt = 0; i <= n; i++)
{
scanf("%[^\n]", line + 1);
getchar();
int dotNum = 0;
for (int j = 1; line[j] == '.'; j++)
{
dotNum ++;
}
element[i].lev = dotNum / 2;
if (!rootStack.empty())
{
while (element[rootStack.top()].lev >= dotNum / 2)
{
rootStack.pop();
}
int temp = rootStack.top();
to[cnt] = i;
nex[cnt] = head[temp];
head[temp] = cnt++;
}
rootStack.push(i);
sscanf(line + dotNum + 1, "%s", element[i].label);
if (line[dotNum + strlen(element[i].label) + 1] == ' ')
{
element[i].hasId = true;
sscanf(line + dotNum + strlen(element[i].label) + 3, "%s", element[i].id);
}
} while (m--)
{
scanf("%[^\n]", line + 1);
getchar();
int iter = 1;
int cnt = 0;
while (true)
{
sscanf(line + iter, "%s", selector[++cnt]);
iter += strlen(selector[cnt]) + 1;
if (line[iter-1] == '\0')
break;
}
num = 0;
dfs(1, 1, cnt);
printf("%d", num);
while (num != 0)
{
printf(" %d", q.top());
q.pop();
num --;
}
printf("\n");
} return 0;
}

CCF-CSP题解 201809-3 元素选择器的更多相关文章

  1. CCF(元素选择器:50分):字符串+模拟

    元素选择器 201809-3 这里我只考虑了没有后代选择器的情况 #include<iostream> #include<cstdio> #include<cstring ...

  2. CCF CSP 201604-3 路径解析

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201604-3 路径解析 问题描述 在操作系统中,数据通常以文件的形式存储在文件系统中.文件系 ...

  3. CCF CSP 201412-2 Z字形扫描

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201412-2 Z字形扫描 问题描述 在图像编码的算法中,需要将一个给定的方形矩阵进行Z字形扫 ...

  4. CCF CSP 201703-3 Markdown

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201703-3 Markdown 问题描述 Markdown 是一种很流行的轻量级标记语言(l ...

  5. CCF CSP 201312-3 最大的矩形

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201312-3 最大的矩形 问题描述 在横轴上放了n个相邻的矩形,每个矩形的宽度是1,而第i( ...

  6. CCF CSP 201609-3 炉石传说

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201609-3 炉石传说 问题描述 <炉石传说:魔兽英雄传>(Hearthston ...

  7. CCF CSP 201403-3 命令行选项

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201403-3 命令行选项 问题描述 请你写一个命令行分析程序,用以分析给定的命令行里包含哪些 ...

  8. CCF CSP 201709-4 通信网络

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201709-4 通信网络 问题描述 某国的军队由N个部门组成,为了提高安全性,部门之间建立了M ...

  9. CCF CSP 201409-3 字符串匹配

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201409-3 字符串匹配 问题描述 给出一个字符串和多行文字,在这些文字中找到字符串出现的那 ...

  10. CCF CSP 201503-3 节日

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201503-3 节日 问题描述 有一类节日的日期并不是固定的,而是以“a月的第b个星期c”的形 ...

随机推荐

  1. ACL2019: 《GraphRel: Modeling Text as Relational Graphs for Joint Entity and Relation Extraction》源码解析

    论文地址:<GraphRel: Modeling Text as Relational Graphs for Joint Entity and Relation Extraction> G ...

  2. Python 编程语言要掌握的技能之一:使用数字与字符串的技巧

    最佳实践 1. 少写数字字面量 “数字字面量(integer literal)” 是指那些直接出现在代码里的数字.它们分布在代码里的各个角落,比如代码 del users[0] 里的 0 就是一个数字 ...

  3. vue防抖节流之v-debounce--throttle使用指南

    最新封装了一个vue防抖节流自定义指令,发布到npm上,有用欢迎star,谢谢! npm地址:https://www.npmjs.com/package/v-debounce-throttle git ...

  4. sublime插件开发教程2

    直接进入正题 先新建个chajian.py python写起 import sublime import sublime_plugin class ExampleCommand(sublime_plu ...

  5. Java中代理和装饰者模式的区别

    装饰模式:以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方案: 代理模式:给一个对象提供一个代理对象,并有代理对象来控制对原有对象的引用: 装饰模式为所装饰的对象增强功能:代理模式对代理的对 ...

  6. 腾讯iphone面试题(转)

    1Objective-C内部的实现 2CALayer和View的关系 3 http协议,tcp/ip 4 UITableView的那些元素是可以自定义的? 5 c语言的,定义变量,比如int,在什么情 ...

  7. 深入 .NET Core 基础 - 2:共享框架

    深入 .NET Core 基础 - 2:共享框架 原文地址:https://natemcmaster.com/blog/2018/08/29/netcore-primitives-2/ 共享框架从 . ...

  8. 服务器RAID技术基础

    RAID ( Redundant Array of Independent Disks )即独立磁盘冗余阵列,通常简称为磁盘阵列 一.RAID主要优势 大容量: 这是 RAID 的一个显然优势,它扩大 ...

  9. Linux编辑器vi使用方法详细介绍(转)

    vi编辑器是所有Unix及Linux系统下标准的编辑器,它的强大不逊色于任何最新的文本编辑器,这里只是简单地介绍一下它的用法和一小部分指令.由于对Unix及Linux系统的任何版本,vi编辑器是完全相 ...

  10. 基于Java语言的IO操作(文件复制)

    public static void main(String[] args) { //获取复制开始前系统时间毫秒值 long start=System.currentTimeMillis(); //文 ...