zoukankan      html  css  js  c++  java
  • 洛谷 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 }
  • 相关阅读:
    网络常用的linux系统调用
    如何在Linux下写无线网卡的驱动【转】
    理解 Linux 配置文件【转】
    每天一个linux命令【转】
    宏定义编写技巧__调试技巧【原创】
    linux 高级字符设备驱动 ioctl操作介绍 例程分析实现【转】
    Linux驱动总结3- unlocked_ioctl和堵塞(waitqueue)读写函数的实现 【转】
    初识CPU卡、SAM卡/CPU卡简介、SAM卡简介 【转】
    android中跨进程通讯的4种方式
    MISC混杂设备 struct miscdevice /misc_register()/misc_deregister()【转】
  • 原文地址:https://www.cnblogs.com/hualian/p/11206745.html
Copyright © 2011-2022 走看看