题目戳这里

首先我要吐槽这个题目描述不清。\(2\)对着选手,那选手朝那边?看完别人写的程序后我才知道选手对着目标所在的方向(或左或右)。

然后这道题还是不错的,因为他交给我矩阵乘法不只有常规意义下的矩阵乘法,只要满足结合律,取 \(\min\) 都行。涨姿势了。

这题我们这么做,我们用\(f[i][j][k]\)表示骰子从起始位置到\((i,j)\)且最后状态为\(k\)所需要最小代价,类似于最短路。然后直接求肯定gg,肯定是要用矩阵乘法来加速。

考虑到\(y\)的范围只有\(4\),我们可以将后两维压成一维,就设为\(g[i][j]\)吧。如果我们能够知道第\(i\)列的所有状态到第\(i+1\)列的所有状态的最短路,(设第\((i,j)\)到\((i+1,k)\)的最小代价为\(h[1][j][k]\)),那么我们就可以知道第\(i\)列所有状态到\(i+n(n>1)\)列所有状态的最小代价了(设第\((i,j)\)到\((i+n,k)\)的最小代价为\(h[n][j][k]\))。则

\[h[n][i][j] = \min \{ h[n-1][i][k]+h[1][k][j] \}
\]

然后这个式子很像矩阵乘法的式子(把和改成了\(\min\)),然后满足结合律,故也可用矩阵乘法来加速。

那么怎么求\(h[1][i][j]\)呢?我们可以将列限制在一个范围内,然后跑最短路就行了。

这个题目最坑爹的地方就是状态处理,这个可以参见std。

#include<queue>
#include<iostream>
#include<cstdio>
#include<cstdlib>
using namespace std; typedef long long ll;
const int Ex = 20,maxn = Ex*96*5; const ll inf = 1LL<<60;
const int help[6][6] = {{-1,2,4,1,3,-1},{3,-1,0,5,-1,2},{1,5,-1,-1,0,4},{4,0,-1,-1,5,1},{2,-1,5,0,-1,3},{-1,3,1,4,2,-1}};
int L[6],X1,Y1,X2,Y2,side[maxn],toit[maxn],next[maxn],len[maxn],cnt; ll dis[Ex*97],f[100],g[100][100]; bool in[Ex*97]; struct Matrix
{
int N,M; ll S[96][96];
inline Matrix(int n = 0,int m = 0,bool sign = false):N(n),M(m)
{
for (int i = 0;i < n;++i) for (int j = 0;j < m;++j) S[i][j] = inf;
if (sign) for (int i = 0;i < n;++i) S[i][i] = 0;
}
friend inline Matrix operator*(const Matrix &a,const Matrix &b)
{
Matrix c(a.N,b.M);
for (int i = 0;i < c.N;++i)
for (int j = 0;j < c.M;++j)
for (int k = 0;k < a.M;++k)
c.S[i][j] = min(c.S[i][j],a.S[i][k]+b.S[k][j]);
return c;
}
}; inline void add(int a,int b,int c) { next[++cnt] = side[a]; side[a] = cnt; toit[cnt] = b; len[cnt] = c; } inline int getid(int x,int y,int top,int front)
{
int front_id = -1;
for (int i = 0;i < 6;++i)
{
if (i != top&&i != 5-top) ++front_id;
if (i == front) return x*96+y*24+top*4+front_id;
}
return -1;
} inline void ready()
{
for (int x = 0;x < Ex;++x)
for (int y = 0;y < 4;++y)
for (int top = 0;top < 6;++top)
for (int front = 0;front < 6;++front)
{
if (front == top||front == 5-top) continue;
int left = help[top][front],u = getid(x,y,top,front);
int nx,ny,ntop,nfront,v;
if (x)
{
nx = x-1; ny = y; ntop = 5-left; nfront = front;
v = getid(nx,ny,ntop,nfront); add(u,v,L[ntop]);
}
if (x+1 < Ex)
{
nx = x+1; ny = y; ntop = left; nfront = front;
v = getid(nx,ny,ntop,nfront); add(u,v,L[ntop]);
}
if (y)
{
nx = x; ny = y-1; ntop = 5-front; nfront = top;
v = getid(nx,ny,ntop,nfront); add(u,v,L[ntop]);
}
if (y < 3)
{
nx = x; ny = y+1; ntop = front; nfront = 5-top;
v = getid(nx,ny,ntop,nfront); add(u,v,L[ntop]);
}
}
} inline void spfa(int source)
{
queue <int> team;
for (int i = 0;i < Ex*96;++i) dis[i] = inf;
dis[source] = 0; in[source] = true; team.push(source);
while (!team.empty())
{
int now = team.front(); team.pop();
for (int i = side[now];i;i = next[i])
{
if (dis[toit[i]] <= dis[now]+len[i]) continue;
dis[toit[i]] = dis[now]+len[i];
if (!in[toit[i]]) team.push(toit[i]),in[toit[i]] = true;
}
in[now] = false;
}
} inline void record(int x,ll A[])
{
for (int y = 0;y < 4;++y)
for (int top = 0;top < 6;++top)
for (int front = 0;front < 6;++front)
{
if (front == top||front == 5-top) continue;
int id = getid(x,y,top,front); A[id%96] = dis[id];
}
} inline Matrix qsm(Matrix a,int b)
{
Matrix ret(96,96,true);
for (;b;b >>= 1,a = a*a) if (b&1) ret = ret*a;
return ret;
} inline ll work()
{
int diff = abs(X1-X2),x = Ex >> 1; ll ans = inf;
spfa(getid(x,Y1,0,1));
if (!diff)
{
for (int top = 0;top < 6;++top)
for (int front = 0;front < 6;++front)
{
if (front == top||front == 5-top) continue;
ans = min(ans,dis[getid(x,Y2,top,front)]);
}
return ans;
}
record(x,f);
for (int y = 0;y < 4;++y)
for (int top = 0;top < 6;++top)
for (int front = 0;front < 6;++front)
{
if (front == top||front == 5-top) continue;
int source = getid(x,y,top,front);
spfa(source); record(x+1,g[source%96]);
}
Matrix A(96,96);
for (int i = 0;i < 96;++i) for (int j = 0;j < 96;++j) A.S[i][j] = g[i][j];
A = qsm(A,diff);
for (int i = 0;i < 96;++i)
for (int j = 0;j < 96;++j)
if (j/24 == Y2) ans = min(ans,f[i]+A.S[i][j]);
return ans;
} int main()
{
freopen("1764.in","r",stdin);
freopen("1764.out","w",stdout);
for (int i = 0;i < 6;++i) scanf("%d",L+i);
scanf("%d %d %d %d",&X1,&Y1,&X2,&Y2); --Y1,--Y2;
if (X1 > X2) swap(L[2],L[3]);
ready();
cout << work() << endl;
fclose(stdin); fclose(stdout);
return 0;
}

poj 1764 Dice Contest的更多相关文章

  1. poj 2187 Beauty Contest(凸包求解多节点的之间的最大距离)

    /* poj 2187 Beauty Contest 凸包:寻找每两点之间距离的最大值 这个最大值一定是在凸包的边缘上的! 求凸包的算法: Andrew算法! */ #include<iostr ...

  2. POJ 3660 Cow Contest / HUST 1037 Cow Contest / HRBUST 1018 Cow Contest(图论,传递闭包)

    POJ 3660 Cow Contest / HUST 1037 Cow Contest / HRBUST 1018 Cow Contest(图论,传递闭包) Description N (1 ≤ N ...

  3. POJ 1719 Shooting Contest(二分图匹配)

    POJ 1719 Shooting Contest id=1719" target="_blank" style="">题目链接 题意:给定一个 ...

  4. poj 2187 Beauty Contest (凸包暴力求最远点对+旋转卡壳)

    链接:http://poj.org/problem?id=2187 Description Bessie, Farmer John's prize cow, has just won first pl ...

  5. POJ 3660 Cow Contest

    题目链接:http://poj.org/problem?id=3660 Cow Contest Time Limit: 1000MS   Memory Limit: 65536K Total Subm ...

  6. poj 1719 Shooting Contest

    http://poj.org/problem?id=1719 Shooting Contest Time Limit: 1000MS   Memory Limit: 10000K Total Subm ...

  7. POJ 3660 Cow Contest(传递闭包floyed算法)

    Cow Contest Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5989   Accepted: 3234 Descr ...

  8. POJ 2187 - Beauty Contest - [凸包+旋转卡壳法][凸包的直径]

    题目链接:http://poj.org/problem?id=2187 Time Limit: 3000MS Memory Limit: 65536K Description Bessie, Farm ...

  9. poj 4014 Dice 贪心

    //poj 4014 //sep9 #include <iostream> #include <algorithm> using namespace std; int n; s ...

随机推荐

  1. 易语言调用C++写的DLL

    直接调用会弹出堆栈错误的信息,原因是VS默认是__cdcel方式,而易语言是__stdcall,所以调用约定不一致导致堆栈错误. 解决方案很简单,易语言声明DLL函数时“在库中对应命令名”函数名前加一 ...

  2. What is EJB

    What is EJB 0.什么是EJB? 答:EJB是用于构建企业应用程序模块托管的.服务器端组件架构.EJB技术加速并简化了开发基于Java技术的分布式.事务性.安全和便携的应用程序. 先看一下E ...

  3. Shell学习——Shell分类:登录shell和非登陆shell 交互shell和非交互shell

    1.从两个不同维度来划分,是否交互式,是否登录 2.交互式shell和非交互式shell 交互式模式:在终端上执行,shell等待你的输入,并且立即执行你提交的命令.这种模式被称作交互式是因为shel ...

  4. Flume是什么

    分布式流式实时收集日志文件系统,便于实时在线的流式计算,常配合 Storm 和 spark streming 使用. Flume is a distributed分布式的, reliable可靠的, ...

  5. echart图表展示数据-简单的柱状图

    话不多说,先上几张效果图 给大家看看 1:echart所用到的文件包需要事先引入好具体可见 http://echarts.baidu.com/doc/start.html 2:本例中所有的数据都是通过 ...

  6. cycling -avoid the vicious cycle

    ‘Numerous' studies in the past appear to have shown a link between cycling and ED. The researchers a ...

  7. string函数Contains()实例

    public bool Contains(string value)如果值参数出现在此字符串内,或者值为空字符串(“”),则为true; 否则为false using System; class Ex ...

  8. Mac 小技巧

    本文的大部分技巧来自于池建强老师的<MacTalk.人生元编程>,感谢他的辛苦付出,本文多系整理而已. 终端输入说英语 说英语时我们当然希望有标准发音.在Mac中不需要字典,直接在终端里输 ...

  9. 《Cracking the Coding Interview》——第3章:栈和队列——题目3

    2014-03-18 05:17 题目:设计一个栈,这个栈实际上由一列子栈组成.每当一个子栈的大小达到n,就新产生下一个子栈.整个栈群对外看起来就像普通栈一样,支持取顶top().压入push().弹 ...

  10. 玩转Node.js(一)

    玩转Node.js(一) 在说Node.js之前,我们先来说说js,如果你也曾开发过前端,那么你一定接触到了这个叫JavaScript有趣的东西,而对于JavaScript,你只会基本的操作——为we ...