【SAM】BZOJ2882-工艺

【题目大意】

求一个循环数列的最小表示法。

【思路】

SAM乱搞,和前面的POJ那道一样。然而MLE了,当作学习一下map的用法^ ^

map的使用方法(来源:

一、map的说明  
  1   头文件 
  #include   <map> 
  
  2   定义 
  map<string,   int>   my_Map; 
  或者是typedef     map<string,   int>   MY_MAP; 
  MY_MAP   my_Map; 
  
  3   插入数据 
  (1)   my_Map["a"]   =   1; 
  (2)   my_Map.insert(map<string,   int>::value_type("b",2)); 
  (3)   my_Map.insert(pair<string,int>("c",3)); 
  (4)   my_Map.insert(make_pair<string,int>("d",4)); 
  
  4   查找数据和修改数据 
  (1)   int   i   =   my_Map["a"]; 
            my_Map["a"]   =   i; 
  (2)   MY_MAP::iterator   my_Itr; 
            my_Itr.find("b"); 
            int   j   =   my_Itr->second; 
            my_Itr->second   =   j; 
  不过注意,键本身是不能被修改的,除非删除。 
  
  5   删除数据 
  (1)   my_Map.erase(my_Itr); 
  (2)   my_Map.erase("c"); 
  还是注意,第一种情况在迭代期间是不能被删除的,道理和foreach时不能删除元素一样。 
  
  6   迭代数据 
  for   (my_Itr=my_Map.begin();   my_Itr!=my_Map.end();   ++my_Itr)   {} 
  
  7   其它方法 
  my_Map.size()               返回元素数目 
  my_Map.empty()       判断是否为空 
  my_Map.clear()           清空所有元素 
  可以直接进行赋值和比较:=,   >,   >=,   <,   <=,   !=   等等

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<algorithm>
 5 #include<map>
 6 using namespace std;
 7 const int MAXN=300002;
 8 int n,k;
 9 struct SAM
10 {
11     int tot,last;
12     int step[MAXN<<2],pre[MAXN<<2];
13     map<int,int> next[MAXN<<2];
14     inline int newNode(int cnt)
15     {
16         step[++tot]=cnt;
17         pre[tot]=0;
18         next[tot].clear();
19         return tot;
20     }
21
22     inline void extend(int x)
23     {
24         int p=last;
25         int np=newNode(step[last]+1);
26         map<int,int>::iterator pos;
27         while (p)
28         {
29             pos=next[p].find(x);
30             if (pos!=next[p].end()) break;
31             next[p].insert(pair<int,int>(x,np));
32             p=pre[p];
33         }
34         if (!p) pre[np]=1;
35             else
36             {
37                 int q=pos->second;
38                 if (step[np]==step[p]+1) pre[np]=q;
39                 else
40                 {
41                     int nq=newNode(step[p]+1);
42                     next[nq]=next[q];
43                     pre[nq]=pre[q];
44                     pre[q]=pre[np]=nq;
45                     while (p)
46                     {
47                         pos=next[p].find(x);
48                         if (pos->second!=q) break;
49                         pos->second=nq;
50                         p=pre[p];
51                     }
52                 }
53             }
54             last=np;
55     }
56
57     inline void clear()
58     {
59         tot=0;
60         last=newNode(0);
61     }
62 }suf;
63 int a[MAXN];
64
65 void init()
66 {
67     scanf("%d",&n);
68     suf.clear();
69     for (int i=0;i<n;i++)
70     {
71         scanf("%d",&a[i]);
72     }
73     for (int i=0;i<n;i++)
74         suf.extend(a[i]);
75     for (int i=0;i<n;i++)
76         suf.extend(a[i]);
77 }
78
79 void solve()
80 {
81     int j=1;
82     for (int i=0;i<n;i++)
83     {
84         map<int,int>::iterator pos=suf.next[j].begin();
85         printf("%d",pos->first);
86         if (i!=n-1) printf(" ");
87         j=pos->second;
88     }
89 }
90
91 int main()
92 {
93     init();
94     solve();
95     return 0;
96 }
时间: 2024-10-11 04:00:20

【SAM】BZOJ2882-工艺的相关文章

Bzoj2882 工艺 [西方算法]

Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 684  Solved: 309 Description 小敏和小燕是一对好朋友. 他们正在玩一种神奇的游戏,叫Minecraft. 他们现在要做一个由方块构成的长条工艺品.但是方块现在是乱的,而且由于机器的要求,他们只能做到把这个工艺品最左边的方块放到最右边. 他们想,在仅这一个操作下,最漂亮的工艺品能多漂亮. 两个工艺品美观的比较方法是,从头开始比较,如果第i个位置上方块不一样那么谁的瑕疵度小,那

bzoj2882 工艺

2882: 工艺 Time Limit: 10 Sec  Memory Limit: 128 MB Submit: 488  Solved: 212 [Submit][Status][Discuss] Description 小敏和小燕是一对好朋友. 他们正在玩一种神奇的游戏,叫Minecraft. 他们现在要做一个由方块构成的长条工艺品.但是方块现在是乱的,而且由于机器的要求,他们只能做到把这个工艺品最左边的方块放到最右边. 他们想,在仅这一个操作下,最漂亮的工艺品能多漂亮. 两个工艺品美观的

bzoj2882工艺(最小表示法)

O(nlogn)的做法十分显然,有三种可以做到O(nlogn)的:1.最容易的想法:把串扩展成两倍,然后跑一遍SA求后缀数组.2.求后缀同样也可以用SAM去求解,用map存一下.3.最暴力的方法:直接二分+hash比较第一位不同的. 其实这题想要让我们用最小表示法求解,然而我不会就来学一下.很容易发现这样一个规律,如果存在s[i+k]>s[j+k],那么s[i...i+k]开头的都不会是最小表示法开头,因为s[i...i+k]=s[j...j+k],所以从s[i...i+k]开头的串都会经过这里

Bzoj2882 工艺 [香港算法]

后缀自动机题解 -> http://www.cnblogs.com/SilverNebula/p/6420601.html 后缀自动机敲完,看了下排行,wc为什么别人跑得这么快?……是诶,这最小表示法用后缀自动机当然慢了 依稀记得最小表示法有超快的算法,于是去查了查,有O(n)的算法 (后缀自动机均摊也是O(n)然而常数大) 找到了这篇讲解-> http://www.cnblogs.com/mjy0724/p/4625928.html 这样就跑得飞快了(380ms,不知道那些几十ms的怎么跑出

[Strings]一些字符串题目

Trie BZOJ 3689 异或之 大意: 给定n个数,求这n个数两两异或的值中的前k小 解: 将加法换成异或就变成了一个用堆合并多个有序表的经典问题,对于加法我们按大小排序就能得到有序表,而由于这里是异或,我们需要高效地维护有序表,即对于一个数ai快速求出与它异或第k小的数. 我们将所有数按二进制建成Trie,然后在Trie的结点上记录下子树中的结束结点个数,再在Trie树上走一遍就得到了答案 BZOJ 3439 Kpm的MC密码 大意: 给定n个字符串,对于每个字符串求以这个字符串为后缀的

【BZOJ2882】工艺 [SAM]

工艺 Time Limit: 10 Sec  Memory Limit: 128 MB[Submit][Status][Discuss] Description 小敏和小燕是一对好朋友. 他们正在玩一种神奇的游戏,叫Minecraft. 他们现在要做一个由方块构成的长条工艺品.但是方块现在是乱的,而且由于机器的要求,他们只能做到把这个工艺品最左边的方块放到最右边. 他们想,在仅这一个操作下,最漂亮的工艺品能多漂亮. 两个工艺品美观的比较方法是,从头开始比较,如果第i个位置上方块不一样那么谁的瑕疵

【BZOJ2882】工艺(后缀自动机)

[BZOJ2882]工艺(后缀自动机) 题面 BZOJ权限题,良心洛谷 题解 还是一样的,先把串在后面接一遍 然后构建\(SAM\) 直接按照字典序输出\(n\)次就行了 #include<iostream> #include<cstdio> #include<cstdlib> #include<cstring> #include<cmath> #include<algorithm> #include<set> #incl

【BZOJ2882】工艺(后缀数组)

[BZOJ2882]工艺(后缀数组) 题面 BZOJ权限题,我爱良心洛谷 题解 最容易的想法: 把字符串在后面接一份 然后求后缀数组就行了... #include<iostream> #include<cstdio> #include<cstdlib> #include<cstring> #include<cmath> #include<algorithm> #include<set> #include<map>

工艺(SAM)

传送门 用SAM可以非常轻松的解决问题. 只要把原串向SAM中插入两次,之后直接从\(t_0\)状态开始每次贪心跑最小就可以了. 因为这个题要用map,所以之前取begin即可. // luogu-judger-enable-o2 #include<bits/stdc++.h> #define rep(i,a,n) for(register int i = a;i <= n;i++) #define per(i,n,a) for(register int i = n;i >= a;

【bzoj2882】工艺 后缀自动机+STL-map

题目描述 小敏和小燕是一对好朋友. 他们正在玩一种神奇的游戏,叫Minecraft. 他们现在要做一个由方块构成的长条工艺品.但是方块现在是乱的,而且由于机器的要求,他们只能做到把这个工艺品最左边的方块放到最右边. 他们想,在仅这一个操作下,最漂亮的工艺品能多漂亮. 两个工艺品美观的比较方法是,从头开始比较,如果第i个位置上方块不一样那么谁的瑕疵度小,那么谁就更漂亮,如果一样那么继续比较第i+1个方块.如果全都一样,那么这两个工艺品就一样漂亮. 输入 第一行两个整数n,代表方块的数目. 第二行n