给定n个字符串s[1...n], 求有多少个数对(i, j), 满足i < j 且 s[i] + s[j] == s[j] + s[i]?

题目

思路:

对于字符串a,b, (a.size() < b.size()), 考虑对字符串b满足什么条件:

由1、3可知a是b的前后缀,由2知b有一个周期是3,即a.size(),所以b是用多个a拼接而成的,有因为a是b的前后缀,所以a和b的循环节相同,且a,b均恰好由整数个循环节组成。循环节长度 = 字符串长度 - 最大公共前后缀长度。

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
#define int long long
const int maxn = 1e6 + 5;
string s[maxn];
int nxt[maxn];
map<string, int> mp;
void getnext(string s){
	int j = -1;
	nxt[0] = -1;
	int n = s.size();
	s = ' ' + s;
	for(int i = 1; i <= n; i++){
		while(j >= 0 && s[i] != s[j + 1]){
			j = nxt[j];
		}
		nxt[i] = j + 1;
		j++;
	}
}
// void f(vector<int> &a){
//     a.push_back(6);
// }
signed main(){
    int n;
    while(cin >> n){
//         cin >> n;
        mp.clear();
        int res = 0;
        for(int i = 1; i <= n; i++){
            cin >> s[i];
            getnext(s[i]);
            int m = s[i].size();
            int T = m - nxt[m];
            if(m % T != 0){
                T = m;//整个字符串也是一个循环节
            }
            string t = s[i].substr(0, T);
            res += mp[t];
            mp[t]++;
        }
         cout << res << '\n';
    }
    
//     string s = "666";
//     cout << s.size() << '\n';
//     getnext(s);
//     cout << s.size() << '\n';
//     vector<int> a;
//     cout << a.size() << '\n';
//     f(a);
//     cout << a.size() << '\n';
    return 0;
}
相关推荐
hsling松子4 小时前
使用PaddleHub智能生成,献上浓情国庆福
人工智能·算法·机器学习·语言模型·paddlepaddle
dengqingrui1234 小时前
【树形DP】AT_dp_p Independent Set 题解
c++·学习·算法·深度优先·图论·dp
C++忠实粉丝4 小时前
前缀和(8)_矩阵区域和
数据结构·c++·线性代数·算法·矩阵
ZZZ_O^O5 小时前
二分查找算法——寻找旋转排序数组中的最小值&点名
数据结构·c++·学习·算法·二叉树
CV-King5 小时前
opencv实战项目(三十):使用傅里叶变换进行图像边缘检测
人工智能·opencv·算法·计算机视觉
代码雕刻家6 小时前
数据结构-3.9.栈在递归中的应用
c语言·数据结构·算法
雨中rain6 小时前
算法 | 位运算(哈希思想)
算法
Kalika0-07 小时前
猴子吃桃-C语言
c语言·开发语言·数据结构·算法
sp_fyf_20248 小时前
计算机前沿技术-人工智能算法-大语言模型-最新研究进展-2024-10-02
人工智能·神经网络·算法·计算机视觉·语言模型·自然语言处理·数据挖掘
我是哈哈hh9 小时前
专题十_穷举vs暴搜vs深搜vs回溯vs剪枝_二叉树的深度优先搜索_算法专题详细总结
服务器·数据结构·c++·算法·机器学习·深度优先·剪枝