洛谷 P3879 [TJOI2010]【阅读理解】

题目描述

英语老师留了N篇阅读理解作业,但是每篇英文短文都有很多生词需要查字典,为了节约时间,现在要做个统计,算一算某些生词都在哪几篇短文中出现过。

输入输出格式

输入格式:

 

第一行为整数N,表示短文篇数,其中每篇短文只含空格和小写字母。

按下来的N行,每行描述一篇短文。每行的开头是一个整数L,表示这篇短文由L个单词组成。接下来是L个单词,单词之间用一个空格分隔。

然后为一个整数M,表示要做几次询问。后面有M行,每行表示一个要统计的生词。

 

输出格式:

 

对于每个生词输出一行,统计其在哪几篇短文中出现过,并按从小到大输出短文的序号,序号不应有重复,序号之间用一个空格隔开(注意第一个序号的前面和最后一个序号的后面不应有空格)。如果该单词一直没出现过,则输出一个空行。

 

输入输出样例

输入样例#1:

3
9 you are a good boy ha ha o yeah
13 o my god you like bleach naruto one piece and so do i
11 but i do not think you will get all the points
5
you
i
o
all
naruto

输出样例#1:

1 2 3
2 3
1 2
3
2

解题思路

  首先就是字典树了,只是我们定一个bool数组判断某个单词是否在某篇文章出现过,然后.........最后一个点就WA了(当时我也蒙了)原来,我们定义bool数组可以换成bitset,这样时间复杂度可以/32,非常实用,强烈安利。

题解

 1 #include<bits/stdc++.h>
 2 #define N 300001
 3 using namespace std;
 4 int n,m,l; 
 5 char x[10001];
 6 struct Trie{
 7     int sz;
 8     int ch[N][26];
 9     bitset<1001>val[N];//bitset 
10     Trie(){
11         sz=1;
12         memset(ch[0],0,sizeof(ch[0]));
13         memset(val,0,sizeof(val));
14     }//初始化 
15     int idx(char c)//字符转数字 
16     {
17         return c-'a';
18     }
19     void insert(string s,int num)//插入 
20     {
21         int u=0;
22         for(int i=0;i<s.size();i++)
23         {
24             int c=idx(s[i]);
25             if(!ch[u][c])
26             {
27                 memset(ch[sz],0,sizeof(ch[sz]));
28                 ch[u][c]=sz;
29                 sz++;
30             }
31             u=ch[u][c];//向下便利 
32         }
33         val[u][num]=1;//在第num篇文章出现过 
34     }
35     void find(string s)//查找 
36     {
37         int u=0;
38         for(int i=0;i<s.size();i++)
39         {
40             int c=idx(s[i]);
41             if(!ch[u][c])//没找到了 
42             {
43                 cout<<endl;
44                 return;
45             }
46             u=ch[u][c];
47         }
48         for(int i=1;i<=n;i++)//看在哪几篇文章出现过 
49         {
50             if(val[u][i]==1)cout<<i<<" ";
51         }
52         cout<<endl;
53     }
54 }tree;
55 int main()
56 {
57     scanf("%d",&n);
58     for(int i=1;i<=n;i++)
59     {
60         cin>>l;
61         for(int j=1;j<=l;j++)
62         {
63             scanf("%s",x);
64             tree.insert(x,i);
65         }
66     }
67     scanf("%d",&m);
68     for(int i=1;i<=m;i++)
69     {
70         scanf("%s",x);
71         tree.find(x);
72     }
73     return 0;
74 }

 

转载于:https://www.cnblogs.com/hualian/p/11206745.html

根据引用[1],dp[u][j]表示在u子树中选取恰好j个人时能获得的最大价值。而根据引用,该问题的时间复杂度为O(log2​104×nm)。 对于洛谷P2143 [JSOI2010] 巨额奖金问题,我们可以使用动态规划来解决。具体步骤如下: 1. 首先,我们需要构建一棵树来表示员工之间的关系。树的根节点表示公司的总经理,其他节点表示员工。每个节点都有一个权值,表示该员工的奖金金额。 2. 接下来,我们可以使用动态规划来计算每个节点的dp值。对于每个节点u,我们可以考虑两种情况: - 如果选择节点u,则dp[u][j] = dp[v][j-1] + value[u],其中v是u的子节点,value[u]表示节点u的奖金金额。 - 如果不选择节点u,则dp[u][j] = max(dp[v][j]),其中v是u的子节点。 3. 最后,我们可以通过遍历树的所有节点,计算出dp[u][j]的最大值,即为所求的巨额奖金。 下面是一个示例代码,演示了如何使用动态规划来解决洛谷P2143 [JSOI2010] 巨额奖金问题: ```python # 构建树的数据结构 class Node: def __init__(self, value): self.value = value self.children = [] # 动态规划求解最大奖金 def max_bonus(root, j): dp = [[0] * (j+1) for _ in range(len(root)+1)] def dfs(node): if not node: return for child in node.children: dfs(child) for k in range(j, 0, -1): dp[node.value][k] = max(dp[node.value][k], dp[node.value][k-1] + node.value) for child in node.children: for k in range(j, 0, -1): for l in range(k-1, -1, -1): dp[node.value][k] = max(dp[node.value][k], dp[node.value][k-l-1] + dp[child.value][l]) dfs(root) return dp[root.value][j] # 构建树 root = Node(1) root.children.append(Node(2)) root.children.append(Node(3)) root.children[0].children.append(Node(4)) root.children[0].children.append(Node(5)) root.children[1].children.append(Node(6)) # 求解最大奖金 j = 3 max_bonus_value = max_bonus(root, j) print("最大奖金为:", max_bonus_value) ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值