UVA 1358 - Generator(dp+ 가우스 소자 + KMP)
UVA 1358 - Generator
제목 링크
제목: m가지 문자('A'부터 뒤로 대문자)가 있습니다. 현재 문자열이 하나 있습니다. 길이가 12를 초과하지 않습니다. 현재 매번 무작위로 알파벳을 생성하여 이 문자열의 기대 길이를 요구합니다.
사고방식: dp[i]는 길이 i가 발생하는 기대 길이를 나타낸다. 그러면 매번 하나의 문자가 발생하고 m종의 전이에 대응한다. 각 전이의 확률은 1/m이다. 전이 후의 길이는 KMP의next수조를 이용하여 신속하게 얻을 수 있다. 그리고 전이가 고리를 형성할 수 있는 상황이기 때문에 직접 DP를 사용할 수 없다. 고스소원을 이용하여 방정식 그룹을 풀 수 있다.
코드:#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
using namespace std;
typedef long long type;
struct Frac {
type a, b;
Frac() {a = 0; b = 1;}
Frac(type a, type b) {this->a = a; this->b = b; deal();}
void init() {a = 0; b = 1;}
type gcd(type a, type b) {
while (b) {
type tmp = a % b;
a = b;
b = tmp;
}
return a;
}
void deal() {
type d = gcd(a, b);
a /= d; b /= d;
if (b < 0) {
a = -a;
b = -b;
}
}
Frac operator + (Frac c) {
Frac ans;
ans.a = a * c.b + b * c.a;
ans.b = b * c.b;
ans.deal();
return ans;
}
Frac operator - (Frac c) {
Frac ans;
ans.a = a * c.b - b * c.a;
ans.b = b * c.b;
ans.deal();
return ans;
}
Frac operator * (Frac c) {
Frac ans;
ans.a = a * c.a;
ans.b = b * c.b;
ans.deal();
return ans;
}
Frac operator / (Frac c) {
Frac ans;
ans.a = a * c.b;
ans.b = b * c.a;
ans.deal();
return ans;
}
void operator += (Frac c) {*this = *this + c;}
void operator += (type c) {*this = *this + Frac(c, 1);}
void operator -= (Frac c) {*this = *this - c;}
void operator *= (Frac c) {*this = *this * c;}
void operator /= (Frac c) {*this = *this / c;}
bool operator > (Frac c) {return a * c.b > b * c.a;}
bool operator == (Frac c) { return a * c.b == b * c.a;}
bool operator < (Frac c) {return !(*this < c && *this == c);}
bool operator >= (Frac c) {return !(*this < c);}
bool operator <= (Frac c) {return !(*this > c);}
bool operator != (Frac c) {return !(*this == c);}
bool operator != (type c) {return *this != Frac(c, 1);}
void operator = (type c) {this->a = c; this->b = 1;}
};
typedef long long ll;
Frac A[15][15];
int t, m, n, next[15];
char str[15];
void getnext() {
next[0] = next[1] = 0;
int j = 0;
for (int i = 2; i <= n; i++) {
while (j && str[i] != str[j + 1]) j = next[j];
if (str[i] == str[j + 1]) j++;
next[i] = j;
}
}
void build() {
for (int i = 0; i <= n; i++)
for (int j = 0; j <= n + 1; j++)
A[i][j].init();
getnext();
A[n][n] = 1;
for (int i = 0; i < n; i++) {
A[i][i] = 1;
A[i][n + 1] = 1;
for (int j = 0; j < m; j++) {
if (str[i + 1] == j + 'A')
A[i][i + 1] += Frac(-1, m);
else {
int tmp = i;
int flag = 1;
while (tmp) {
tmp = next[tmp];
if (str[tmp + 1] == j + 'A') {
flag = 0;
A[i][tmp + 1] += Frac(-1, m);
break;
}
}
if (flag) A[i][0] += Frac(-1, m);
}
}
}
}
ll gauss() {
for (int i = 0; i <= n; i++) {
int r;
for (r = i; r <= n; r++)
if (A[r][i] != 0) break;
for (int j = i; j <= n + 1; j++)
swap(A[i][j], A[r][j]);
for (int j = n + 1; j > i; j--)
A[i][j] /= A[i][i];
A[i][i] = 1;
for (int j = 0; j <= n; j++) {
if (i == j) continue;
if (A[j][i] != 0) {
for (int k = n + 1; k > i; k--)
A[j][k] -= A[j][i] * A[i][k];
A[j][i] = 0;
}
}
}
return (A[0][n + 1] / A[0][0]).a;
}
int main() {
int cas = 0;
scanf("%d", &t);
while (t--) {
scanf("%d%s", &m, str + 1);
n = strlen(str + 1);
build();
printf("Case %d:
", ++cas);
printf("%lld
", gauss());
if (t) printf("
");
}
return 0;
}
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
다양한 언어의 JSON
JSON은 Javascript 표기법을 사용하여 데이터 구조를 레이아웃하는 데이터 형식입니다.
그러나 Javascript가 코드에서 이러한 구조를 나타낼 수 있는 유일한 언어는 아닙니다.
저는 일반적으로 '객체'{}...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
using namespace std;
typedef long long type;
struct Frac {
type a, b;
Frac() {a = 0; b = 1;}
Frac(type a, type b) {this->a = a; this->b = b; deal();}
void init() {a = 0; b = 1;}
type gcd(type a, type b) {
while (b) {
type tmp = a % b;
a = b;
b = tmp;
}
return a;
}
void deal() {
type d = gcd(a, b);
a /= d; b /= d;
if (b < 0) {
a = -a;
b = -b;
}
}
Frac operator + (Frac c) {
Frac ans;
ans.a = a * c.b + b * c.a;
ans.b = b * c.b;
ans.deal();
return ans;
}
Frac operator - (Frac c) {
Frac ans;
ans.a = a * c.b - b * c.a;
ans.b = b * c.b;
ans.deal();
return ans;
}
Frac operator * (Frac c) {
Frac ans;
ans.a = a * c.a;
ans.b = b * c.b;
ans.deal();
return ans;
}
Frac operator / (Frac c) {
Frac ans;
ans.a = a * c.b;
ans.b = b * c.a;
ans.deal();
return ans;
}
void operator += (Frac c) {*this = *this + c;}
void operator += (type c) {*this = *this + Frac(c, 1);}
void operator -= (Frac c) {*this = *this - c;}
void operator *= (Frac c) {*this = *this * c;}
void operator /= (Frac c) {*this = *this / c;}
bool operator > (Frac c) {return a * c.b > b * c.a;}
bool operator == (Frac c) { return a * c.b == b * c.a;}
bool operator < (Frac c) {return !(*this < c && *this == c);}
bool operator >= (Frac c) {return !(*this < c);}
bool operator <= (Frac c) {return !(*this > c);}
bool operator != (Frac c) {return !(*this == c);}
bool operator != (type c) {return *this != Frac(c, 1);}
void operator = (type c) {this->a = c; this->b = 1;}
};
typedef long long ll;
Frac A[15][15];
int t, m, n, next[15];
char str[15];
void getnext() {
next[0] = next[1] = 0;
int j = 0;
for (int i = 2; i <= n; i++) {
while (j && str[i] != str[j + 1]) j = next[j];
if (str[i] == str[j + 1]) j++;
next[i] = j;
}
}
void build() {
for (int i = 0; i <= n; i++)
for (int j = 0; j <= n + 1; j++)
A[i][j].init();
getnext();
A[n][n] = 1;
for (int i = 0; i < n; i++) {
A[i][i] = 1;
A[i][n + 1] = 1;
for (int j = 0; j < m; j++) {
if (str[i + 1] == j + 'A')
A[i][i + 1] += Frac(-1, m);
else {
int tmp = i;
int flag = 1;
while (tmp) {
tmp = next[tmp];
if (str[tmp + 1] == j + 'A') {
flag = 0;
A[i][tmp + 1] += Frac(-1, m);
break;
}
}
if (flag) A[i][0] += Frac(-1, m);
}
}
}
}
ll gauss() {
for (int i = 0; i <= n; i++) {
int r;
for (r = i; r <= n; r++)
if (A[r][i] != 0) break;
for (int j = i; j <= n + 1; j++)
swap(A[i][j], A[r][j]);
for (int j = n + 1; j > i; j--)
A[i][j] /= A[i][i];
A[i][i] = 1;
for (int j = 0; j <= n; j++) {
if (i == j) continue;
if (A[j][i] != 0) {
for (int k = n + 1; k > i; k--)
A[j][k] -= A[j][i] * A[i][k];
A[j][i] = 0;
}
}
}
return (A[0][n + 1] / A[0][0]).a;
}
int main() {
int cas = 0;
scanf("%d", &t);
while (t--) {
scanf("%d%s", &m, str + 1);
n = strlen(str + 1);
build();
printf("Case %d:
", ++cas);
printf("%lld
", gauss());
if (t) printf("
");
}
return 0;
}
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
다양한 언어의 JSONJSON은 Javascript 표기법을 사용하여 데이터 구조를 레이아웃하는 데이터 형식입니다. 그러나 Javascript가 코드에서 이러한 구조를 나타낼 수 있는 유일한 언어는 아닙니다. 저는 일반적으로 '객체'{}...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.