[Codeforces]696D Legen... AC自动机 + 矩阵快速幂

本文介绍了一种通过构建AC自动机并结合矩阵快速幂的方法来解决一个特定问题:如何构造一条长度限定的短信,使其包含一系列预定义的短语时能够获得最大化的幸福值。文章详细解释了算法的设计思路及实现过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

D. Legen...
time limit per test
6 seconds
memory limit per test
256 megabytes
input
standard input
output
standard output

Barney was hanging out with Nora for a while and now he thinks he may have feelings for her. Barney wants to send her a cheesy text message and wants to make her as happy as possible.

Initially, happiness level of Nora is 0. Nora loves some pickup lines like "I'm falling for you" and stuff. Totally, she knows n pickup lines, each consisting only of lowercase English letters, also some of them may be equal (in writing, but different in pronouncing or meaning though). Every time Nora sees i-th pickup line as a consecutive subsequence of Barney's text message her happiness level increases by ai. These substrings may overlap, for example, Nora will see the pickup line aa twice and the pickup line ab once in text message aaab.

Due to texting app limits, Barney's text may have up to l characters.

Barney asked you to help him make Nora as much happy as possible, it's gonna be legen...

Input

The first line of input contains two integers n and l (1 ≤ n ≤ 200, 1 ≤ l ≤ 1014) — the number of pickup lines and the maximum length of Barney's text.

The second line contains n integers a1, a2, ..., an (1 ≤ ai ≤ 100), meaning that Nora's happiness level increases by ai after every time seeing i-th pickup line.

The next n lines contain the pickup lines. i-th of them contains a single string si consisting of only English lowercase letter. Summary length of all pickup lines does not exceed 200.

All strings are not empty.

Output

Print the only integer — the maximum possible value of Nora's happiness level after reading Barney's text.

Examples
input
3 6
3 2 1
heart
earth
art
output
6
input
3 6
3 2 8
heart
earth
art
output
16
Note

An optimal answer for the first sample case is hearth containing each pickup line exactly once.

An optimal answer for the second sample case is artart.

  

  大意就是说给定l和n个字符串, 每个字符串有一个分数, 求一个长度为l的字符串能获得的最大分数. 包含了给定字符串就能获得给定分数, 并且字符串之间可以重叠.

  给定长度的字符串, 包含xx串就... 这种类型的题已经很常见了... 一看就是要AC自动机上dp啊. 比如说设dp[i][j]表示从root出发走i步到了节点j此时能获得的最大分数. 这个dp很好转移. 到了有end节点就加分即可.

  但l太大显然dp不可过... 那么就只能上矩阵快速幂来优化辣... 矩阵a[i][j]表示从i走到j的最大分数. 矩阵乘法每乘一次可以看作走一步(这个可以详见POJ2778. 那么转移n次就可以了, 直接上快速幂. 这里矩阵不再是乘, 而是加和取max. 之所以能这样矩阵乘法是因为仔细想想可以发现从i走到j这样的路径这是可以拆开的... 其实你可以看成floyed的类似感觉.

  注意有重复串val要累加. 并且当前点也要累加fail的val(因为是后缀啊).

#include<bits/stdc++.h>
using namespace std;
typedef long long lnt;
const int maxn = 205;
char ss[maxn];
int m, tot;
lnt val[maxn], ans, n;
int w[maxn], c[maxn][26], fail[maxn];
inline void insert(int nw) {
	int p = 0;
	for (int i = 0; ss[i]; ++ i) {
		int idx = ss[i] - 'a';
		if (!c[p][idx]) c[p][idx] = ++ tot;
		p = c[p][idx];
	}
	val[p] += w[nw];
}
queue<int> q;
inline void bfs() {
	for (int i = 0; i < 26; ++ i)
		if (c[0][i]) q.push(c[0][i]);
	while (!q.empty()) {
		int u = q.front(); q.pop();
		for (int i = 0; i < 26; ++ i) {
			int &v = c[u][i];
			if (!v) {v = c[fail[u]][i]; continue;}
			fail[v] = c[fail[u]][i], val[v] += val[fail[v]];
			q.push(v);
		}
	}
}
struct Matrix {
	lnt mat[maxn][maxn];
	Matrix() {
		memset(mat, -1, sizeof (mat));
	}
	inline friend Matrix operator * (const Matrix &a, const Matrix &b) {
		Matrix c;
		for (int i = 0; i <= tot; ++ i)
			for (int k = 0; k <= tot; ++ k) if (~a.mat[i][k])
				for (int j = 0; j <= tot; ++ j) if (~b.mat[k][j])
					c.mat[i][j] = max(c.mat[i][j], a.mat[i][k] + b.mat[k][j]);
		return c;
	}
}a, ret;
int main() {
	scanf("%d%I64d", &m, &n);
	for (int i = 0; i < m; ++ i) scanf("%d", &w[i]);
	for (int i = 0; i < m; ++ i) {
		scanf("%s", ss);
		insert(i);
	}
	bfs();
	for (int i = 0; i <= tot; ++ i)
		for (int p = 0; p < 26; ++ p) {
			a.mat[i][c[i][p]] = val[c[i][p]];
		}
	ret = a, -- n;
	while (n) {
		if (n & 1) ret = ret * a;
		a = a * a, n >>= 1;
	}
	for (int i = 0; i <= tot; ++ i)
		ans = max(ans, ret.mat[0][i]);
	printf("%I64d\n", ans);
}


### Codeforces Round 927 Div. 3 比赛详情 Codeforces是一个面向全球程序员的比赛平台,定期举办不同级别的编程竞赛。Div. 3系列比赛专为评级较低的选手设计,旨在提供更简单的问题让新手能够参与并提升技能[^1]。 #### 参赛规则概述 这类赛事通常允许单人参加,在规定时间内解决尽可能多的问题来获得分数。评分机制基于解决问题的速度以及提交答案的成功率。比赛中可能会有预测试案例用于即时反馈,而最终得分取决于系统测试的结果。此外,还存在反作弊措施以确保公平竞争环境。 ### 题目解析:Moving Platforms (G) 在这道题中,给定一系列移动平台的位置和速度向量,询问某时刻这些平台是否会形成一条连续路径使得可以从最左端到达最右端。此问题涉及到几何学中的线段交集判断和平面直角坐标系内的相对运动分析。 为了处理这个问题,可以采用如下方法: - **输入数据结构化**:读取所有平台的数据,并将其存储在一个合适的数据结构里以便后续操作。 - **时间轴离散化**:考虑到浮点数精度误差可能导致计算错误,应该把整个过程划分成若干个小的时间间隔来进行模拟仿真。 - **碰撞检测算法实现**:编写函数用来判定任意两个矩形之间是否存在重叠区域;当发现新的连接关系时更新可达性矩阵。 - **连通分量查找技术应用**:利用图论知识快速求解当前状态下哪些节点属于同一个集合内——即能否通过其他成员间接相连。 最后输出结果前记得考虑边界条件! ```cpp // 假设已经定义好了必要的类和辅助功能... bool canReachEnd(vector<Platform>& platforms, double endTime){ // 初始化工作... for(double currentTime = startTime; currentTime <= endTime ;currentTime += deltaT){ updatePositions(platforms, currentTime); buildAdjacencyMatrix(platforms); if(isConnected(startNode,endNode)){ return true; } } return false; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值