二分概率+矩乘+dp

也是二分概率,然后dp[i][j][k]表示当前到了i,有j条命,下一次的收益是k,然后矩乘转移,但是我自己的似乎wa了,抄了liu_runda的才行,具体不知道为什么

注释的是我自己写的,谁能告诉我哪里错了?

#include<bits/stdc++.h>
using namespace std;
const int N = ;
int n, r, q, tot;
int id[N][N];
double S;
struct matrix {
double a[N][N];
matrix() { for(int i = ; i <= tot; ++i) for(int j = ; j <= tot; ++j) a[i][j] = 0.0; }
matrix friend operator * (const matrix &a, const matrix &b) {
matrix ret;
for(int k = ; k <= tot; ++k)
for(int i = ; i <= tot; ++i) if(a.a[i][k] >= 1e-)
for(int j = ; j <= tot; ++j)
ret.a[i][j] += a.a[i][k] * b.a[k][j];
return ret;
}
void set() {
for(int i = ; i <= tot; ++i) a[i][i] = 1.0;
}
};
double calc(double p)
{
matrix A, B;
A.set();
/*
第i轮j条命这一次得k分
dp[i][j][k]=(dp[i+1][j + 1][k + 1] + k) * p
dp[i][j][k] += dp[i + 1][j - 1][1] * (1.0 - p)
矩阵乘法
*/
B.a[tot][tot] = 1.0;
for(int i = ; i <= q; ++i)
for(int j = ; j <= r; ++j)
{
if(i > ) B.a[id[i - ][]][id[i][j]] = 1.0 - p;
B.a[tot][id[i][j]] = p * (double)j;
if(i < q && j < r) B.a[id[i + ][j + ]][id[i][j]] = p;
else if(i < q) B.a[id[i + ][j]][id[i][j]] = p;
else if(j < r) B.a[id[i][j + ]][id[i][j]] = p;
else B.a[id[i][j]][id[i][j]] = p;
}
// for(int j = 0; j <= q; ++j)
// for(int k = 1; k <= r; ++k) B.a[tot][id[j][k]] = p * (double)k;
// for(int j = 0; j <= q; ++j)
// for(int k = 1; k <= r; ++k) B.a[id[min(j + 1, q)][min(k + 1, r)]][id[j][k]] = p;
// for(int j = 2; j <= q; ++j)
// for(int k = 1; k <= r; ++k) B.a[id[j - 1][1]][id[j][k]] = 1.0 - p;
for(int i = n; i; i >>= , B = B * B) if(i & ) A = A * B;
// double ret = 0.0;
// for(int i = 0; i <= tot; ++i)
// {
// for(int j = 0; j <= tot; ++j) printf("%.6f ", A.a[i][j]);
// puts("");
// }
// for(int i = 0; i < tot; ++i) ret += A.a[tot][i];
// printf("p = %.6f ret = %.6f\n", p, ret);
return A.a[tot][id[q][]];
}
int main()
{
scanf("%d%d%d%lf", &n, &r, &q, &S);
for(int i = ; i <= q; ++i)
for(int j = ; j <= r; ++j) id[i][j] = tot++;
// printf("tot = %d\n", tot);
double l = 0.0, r = 1.0, ans = -1.0;
while(r - l > 1e-)
{
double mid = (l + r) / 2.0;
if(calc(mid) > S) r = ans = mid;
else l = mid;
}
if(ans == -1.0) puts("Impossible.");
else printf("%.6f\n", ans);
return ;
}

bzoj2676的更多相关文章

  1. bzoj2676 Contra

    题意: 给定N,R,Q,S 有N个关卡,初始有Q条命,且任意时刻最多只能有Q条命 每通过一个关卡,会得到u分和1条命,其中u=min(最近一次连续通过的关数,R) 若没有通过这个关卡,将失去一条命,并 ...

随机推荐

  1. HTML5 这些你全知道吗?

    今天心情烦躁,搭建个论坛各种失败,Laravel.React也学不进去,于是就复习复习学过的.记录一下很少用的东西. 代码摘抄于W3C菜鸟教程HTML5手册 # 定义文本方向 <element ...

  2. CNN网络--AlexNet

    ImageNet Classification with Deep Convolutional Neural Networks 从AlexNet剖析-卷积网络CNN的一般结构 AlexNet是Hint ...

  3. 制作ubuntu U盘安装盘

    sudo dd if=ubuntu.iso of=/dev/sdb2 sudo syslinux /dev/sdb1

  4. [Unit Testing] Test async function with Jasmine

    Most of time, when we want to test function call inside a promise, we can do: it('Should be async', ...

  5. linux find 命令查找 复制

    find 查找 find . -mtime -2 -a -path './.git*' -prune , -path './Cache' -prune -a -exec cp {} one \; rm ...

  6. 【剑指offer】异或去重

    转载请注明出处:http://blog.csdn.net/ns_code/article/details/27568975 这篇文章没有代码.介绍的是纯理论的思路. 异或是一种基于二进制的位运算,用符 ...

  7. [unity3d]unity平台的预处理

    在开发中,特别是unity的跨平台中,我们常常会在各个平台游走,如安卓版,苹果版,PC版.......在此不同的平台上,有可能我们须要做不同的操作.然而我们就能够用unity的自带的平台宏定义方式来做 ...

  8. 基于SpringMVC框架使用ECharts3.0实现折线图,柱状图,饼状图,的绘制(上篇)

    页面部分 <%@ page language="java" pageEncoding="UTF-8"%> <!DOCTYPE html> ...

  9. Linux主要命令

    pwd  查看当前路径 cd ..    表示后一级目录级 cd .   表示当前目录 cd ../..   后退两级 cd  表示进入当前家目录 date 返回当前的一个具体时间    -s  修改 ...

  10. SQL常见问题及解决备忘

    1.mysql中:you cant't specify tartget table for update in from clause 错误 含义:在同一语句中update或delete某张表的时候, ...