E. Swapping Characters 一个喳喳的做法
http://codeforces.com/contest/903/problem/E
题意是,对于每个字符串都要交换两个位置的字符(id),使得结果所有字符串是一样的,输出那个字符串。
正解是,先比较两个字符串,如果他们不同的位置 > 4那就是不行的了
有4个不同的还是可行的,比如:
abab
baba
因为每个字符串都有一次交换机会,所以可以变成
baab即可
如果小于4,那么暴力枚举每一个不同的位置,和任意一个位置交换,暴力check,复杂度5000^2
我的渣渣做法。
因为n*k<5000
预处理每一个字符串,所有交换情况后得到字符串的hash值,知道原串的hash值,交换两个字符后,得到的hash值可以O(1)搞出来
然后相当于给k个数组,问是否存在一个数字在这k个数组中都存在过。
复杂度n^2 log n
#include <bits/stdc++.h>
#define IOS ios::sync_with_stdio(false)
using namespace std;
#define inf (0x3f3f3f3f)
typedef long long int LL;
const int maxn = + ;
char str[maxn][maxn];
unsigned long long int po[maxn];
const int seed = ;
int cnt[maxn], DFN;
struct Node {
unsigned long long int val;
int one, two;
Node(unsigned long long int _val, int _one, int _two) {
val = _val, one = _one, two = _two;
}
bool operator < (const struct Node & rhs) const {
return val < rhs.val;
}
};
vector<Node> vc[maxn];
void work() {
int k, n;
scanf("%d%d", &k, &n);
for (int i = ; i <= k; ++i) scanf("%s", str[i] + ); if (k == ) {
swap(str[][], str[][]);
printf("%s\n", str[] + );
printf("\n");
return;
}
for (int i = ; i <= k; ++i) {
unsigned long long int now = ;
bool can = false;
DFN++;
for (int j = ; j <= n; ++j) {
now = now * seed + str[i][j];
can |= cnt[str[i][j]] == DFN;
cnt[str[i][j]] = DFN;
}
if (can) vc[i].push_back(Node(now, , ));
for (int j = ; j <= n; ++j) {
for (int f = j + ; f <= n; ++f) {
if (str[i][j] == str[i][f]) {
if (can) continue;
can = true;
}
unsigned long long int ha = now - str[i][j] * po[n - j] - str[i][f] * po[n - f] + str[i][f] * po[n - j] + str[i][j] * po[n - f];
vc[i].push_back(Node(ha, j, f));
// swap(str[i][j], str[i][f]);
// cout << str[i] + 1 << " " << ha << endl;
// swap(str[i][j], str[i][f]);
}
}
// cout << endl;
sort(vc[i].begin(), vc[i].end());
}
// for (int i = 1; i <= k; ++i) {
// for (int j = 0; j < vc[i].size(); ++j) {
// cout << vc[i][j].val << " ";
// }
// cout << endl;
// }
for (int i = ; i < vc[].size(); ++i) {
int t = ;
for (int j = ; j <= k; ++j) {
if (vc[][i].val > vc[j].back().val) break;
int pos = lower_bound(vc[j].begin(), vc[j].end(), vc[][i]) - vc[j].begin();
if (vc[j][pos].val != vc[][i].val) break;
t++;
}
if (t == k) {
int id1 = vc[][i].one, id2 = vc[][i].two;
swap(str[][id1], str[][id2]);
printf("%s\n", str[] + );
return;
}
}
printf("-1\n");
} int main() {
#ifdef local
freopen("data.txt", "r", stdin);
// freopen("data.txt", "w", stdout);
#endif
po[] = ;
for (int i = ; i <= maxn - ; ++i) po[i] = po[i - ] * seed;
work();
return ;
}
E. Swapping Characters 一个喳喳的做法的更多相关文章
- Codeforces 903E Swapping Characters
题目大意 考虑一个未知的长为 $n$($2\le n\le 5000$)由小写英文字母构成的字符串 $s$ .给出 $k$($1\le k\le 2500$,$nk\le 5000$)个字符串 $s_ ...
- Swapping Characters CodeForces - 903E (字符串模拟)
大意: 给定k个字符串, 长度均为n, 求是否存在一个串S, 使得k个字符串都可以由S恰好交换两个字符得到. 暴力枚举交换的两个字符的位置, 计算出交换后与其他串不同字符的个数, 若为1或>2显 ...
- Find The Multiple (poj1426 一个好的做法)
Find The Multiple Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 16505 Accepted: 673 ...
- 「TJOI / HEOI2016」求和 的一个优秀线性做法
我们把\(S(i, j)j!\)看成是把\(i\)个球每次选择一些球(不能为空)扔掉,选\(j\)次后把所有球都扔掉的情况数(顺序有关).因此\(S(i, j)j! = i
GObject GObject库是Glib库的动态类型系统实现,它实现了: 基于引用计数的内存管理 实例的构造和析构 通用的set/get的属性获取方法 简单易用的信号机制 对象实例化 所述g_obj ...
- 937. Reorder Log Files
You have an array of logs. Each log is a space delimited string of words. For each log, the first w ...
- CSS之引入样式
CSS引入样式 内部样式 内嵌式是将CSS代码集中写在HTML文档的head头部标签中,并且用style标签定义,其基本语法格式如下: <head> <style type=&quo ...
- log4j配置文件及java调用 每个级别输出到不同的文件
#配置根Logger log4j.rootLogger = DEBUG , RollingFile,CONSOLE #文件大小达到一定尺寸的时候创建一个新的文件 log4j.appender.Roll ...
- 设置label的文字,一行多种颜色
调用 [self fuwenbenLabel:contentLabel FontNumber:[UIFont systemFontOfSize:] AndRange:NSMakeRange(, ) A ...
- 【模板】割点(割顶) Tarjan
题目背景 割点 题目描述 给出一个nnn个点,mmm条边的无向图,求图的割点. 输入输出格式 输入格式: 第一行输入n,mn,mn,m 下面mmm行每行输入x,yx,yx,y表示xxx到yyy有一条边 ...
- 河南省第十一届ACM程序设计竞赛 修路
Problem C: 修路 Time Limit: 3 Sec Memory Limit: 128 MBSubmit: 63 Solved: 22[Submit][Status][Web Boar ...
- 6.动态sql - if
满足条件的数据 mapper.xml 满足if条件就执行,不满足就不加 <select id="selectStateByTitle" parameterType=" ...