2018-06-10

莫尼塞垫底祭。。两个号大凶果然没有好事情


绍兴一中NOI P模拟赛小小小小题解
1.脱离地牢
Description
在一个神秘的国度里,年轻的王子Paris与美丽的公主Helen在一起过着幸福的生活。他们都随身带有一块带磁性的阴阳魔法石,身居地狱的魔王Satan早就想着得到这两块石头了,只要把它们溶化,Satan就能吸收其精华大增自己的魔力。于是有一天他趁二人不留意,把他们带到了自己的地牢,分别困在了不同的地方。然后Satan念起了咒语,准备炼狱,界时二人都将葬身于这地牢里。 危险!Paris与Helen都知道了Satan的意图,他们要怎样才能打败魔王,脱离地牢呢?Paris想起了父王临终前给他的备忘本,原来他早已料到了Satan的野心,他告诉Paris只要把两块魔法石合在一起,念出咒语,它们便会放出无限的光荣,杀死魔王,脱离地牢,而且本子上还附下了地牢的地图,Paris从中了解到了Helen的位置所在。于是他决定首先要找到Helen,但是他发现这个地牢很奇怪,它会增强二人魔法石所带的磁力大小,而且会改变磁力的方向。这就是说,每当Paris向南走一步,Helen有可能会被石头吸引向北走一步。而这个地狱布满了岩石与熔浆,Paris必须十分小心,不仅他不能走到岩石或熔浆上,而且由于他行走一步,Helen的位置也会改变,如果Helen碰到岩石上,那么她将停留在原地,但如果Helen移动到了熔浆上,那么她将死去,Paris就找不到她了。 Paris仔细分析了地图,他找出了一条最快的行走方案,最终与Helen相聚。他们一起念出了咒语“·#¥%^…*&@!”,轰隆一声,地牢塌陷了,他们又重见光明…
Input
输入数据第一行为两个整数n,m(3<=n,m<=20),表示地牢的大小,n行m列。接下来n行,每行m个字符,描述了cf地牢的地图,“.”代表通路,“#”代表岩石,“!”代表熔浆,“H”表示Helen,“P”表示Paris。输入保证地牢是封闭的,即四周均是岩石或熔浆。接下来一行有四个字符“N”(北),“S”(南),“W”(西),“E”(东)的排列,表示Paris分别向NSWE四个方向走时Helen受磁石磁力影响的移动方向。
Output
输出文件只有一行,如果Paris能找到Helen,输出一整数d,为Paris最少需要行走的步数;如果Paris在255步之后仍找不到Helen,则输出“Impossible”。注意相遇是指Paris与Helen最终到达同一个格子,或者二人在相邻两格移动后碰到了一起,而后者的步数算他们移动后的步数。
Sample Input
5 5

#####
#H..#
#.!.#
#.#P#
#####
WNSE

Sample Output
5

这道题的相遇条件是走到同一个格子或者两人互换位置。然后就是妥妥的bfs,记得vis数组去重放在统计答案之后。不然会wa两个点。
另.蒟蒻还目睹了yyh用dfs水掉了此题。。。然而跑的飞快

#include <cstring>
#include <iostream>
#include <cstdio>
#include <cmath>
using namespace std;
int n,m,flag,ans=256,fh[5];
char c[200][200];
int f[21][21][21][21];
int dx[5]={-1,1,0,0};
int dy[5]={0,0,-1,1};
void dfs(int sx,int sy,int tx,int ty,int step)
{
    if(f[sx][sy][tx][ty]<=step||step>=256||step>=ans) return;
    f[sx][sy][tx][ty]=step;//根据yyh大佬改进的判重
    if((sx==tx&&sy==ty))
    {
        ans=step;
        flag=1;
        return;
    }
    if((sx==tx&&abs(sy-ty)==1)||(abs(sx-tx)==1&&sy==ty)){
        for(int i=0;i<4;i++){
            if(sx+dx[i]==tx&&sy+dy[i]==ty&&tx+dx[fh[i]]==sx&&ty+dy[fh[i]]==sy){
                ans=step+1;
                flag=1;
                return;
            }
        }
    }
    if(ans<=step) return;
//  cout<<sx<<" "<<sy<<" "<<tx<<" "<<ty<<endl;
    for(int i=0;i<4;i++)
    {
        int px=sx+dx[i];
        int py=sy+dy[i];
        int hx=tx+dx[fh[i]];
        int hy=ty+dy[fh[i]];
//      cout<<px<<" "<<py<<" "<<hx<<" "<<hy<<endl;
        if(c[px][py]!='#'&&c[px][py]!='!'&&c[hx][hy]!='!')
        {
            if(c[hx][hy]=='#') dfs(px,py,tx,ty,step+1);
            else dfs(px,py,hx,hy,step+1);
        }
    }
}
int main()
{
    freopen("escape.in","r",stdin);
    freopen("escape.out","w",stdout);
    int sx,sy,tx,ty;
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)
    {
        scanf("%s",c[i]+1);
        for(int j=1;j<=m;j++)
        {
            if(c[i][j]=='P')sx=i,sy=j,c[i][j]='.';
            if(c[i][j]=='H')tx=i,ty=j,c[i][j]='.';
        }
    }
    char ch;
    for(int i=0;i<4;i++)
    {
        cin>>ch;
        if(ch=='N') fh[i]=0;
        else if(ch=='S') fh[i]=1;
        else if(ch=='W') fh[i]=2;
        else if(ch=='E') fh[i]=3;
    }
    memset(f,0x3f3f3f3f,sizeof f);
//  for(int i=0;i<4;i++)
//  cout<<fh[i]<<" ";puts("");
    dfs(sx,sy,tx,ty,0);
    if(flag) printf("%d",ans);
    else puts("Impossible");
    return 0;
}
/*
5 5
#####
#H..#
#.!.#
#.#P#
#####
ENSW
*/

2.Reward
题目描述:
众所周知,liverpool的主帅贝尼特斯喜欢轮换阵法,是阵法变换的大家。他的首发阵容往往让对方主帅无法捉摸,以至于失去布阵的先机,其创下的999场首发阵容不同的纪录至今无人能望其项背。其接班人dick充分继承了他的优良传统,并将其发扬光大,出现了完全随机的阵容……
为什么dick可以毫无障碍的把轮换进行到底呢?其原因在于dick的一个黑箱小程序,他的功用在于给定一个数n,可求出约数总数为n的最小数——根据这个数字,dick再经过一系列的数学式转换,最终可以获得11个首发球员的号码。
作为liverpool竞争对手的Manchester Unit、Arsenal、Chelsea联合起来,经过多年的潜访调查考证研究分析证明后终于成功的盗得了dick的输入数n的规律,以及最后转化的数学式,眼看成功胜利在望,他们开出巨额赏金悬赏可以解决dick黑箱小程序的人才,以实现他们打破liverpool不败神话的愿望。
不管为了巨额奖金还是击败liverpool的荣誉,我想你应该会试一试吧。
输入数据:
输入仅一个数n,表示约数总数。
输出数据:
输出仅一个数,表示最小的具有n个约数的数。
输入样例:
4
输出样例:
6
数据规模:
对于50%的数据 n<=50
对于100%的数据 n<=50000

公式:原数=p1(t1)*p2(t2)p3^(t3)....
约数个数=(t1+1)
(t2+1)(t3+1)(t4+1)...
45分到手。
然后我们考虑DP
Mst数组表示用第i个素数之前有j个因数,用后还剩下需要消除的因数个数
f数组表示第i个素数用后使得剩下j个因数所需的答案log10之后的数
关于为什么要log10:因为约数定理是乘法,而乘法会比较麻烦,所以转成位数后用加法实现,这也是为什么lg和from数组要用double的原因

#include<bits/stdc++.h> 
#define maxn 50010
#define int long long
#define mo 1000000000
using namespace std;
int zhi[20]={0,2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53},n,mst[20][maxn];
double lg[20],f[20][maxn];
struct bignum
{
    int a[2300];
    bignum(int i=0)
    {
        memset(a,0,sizeof a);
        if(!i) return;
        a[0]=1;
        a[1]=i;
    }
    friend bignum operator *(bignum x,bignum y)
    {
        bignum z;
        for(int i=1;i<=x.a[0];i++)
        {
            int p=0;
            for(int j=1;j<=y.a[0];j++)
            {
                p+=x.a[i]*y.a[j]+z.a[i+j-1];
                z.a[i+j-1]=p%mo;
                p/=mo;
            }
            z.a[i+y.a[0]]=p;
        }
        z.a[0]=x.a[0]+y.a[0];
        while(z.a[0]&&!z.a[z.a[0]])
            z.a[0]--;
        return z;
    }
    void print()
    {
        printf("%lld",a[a[0]]);
        for(int i=a[0]-1;i>0;i--)
            printf("%09lld",a[i]);
        puts("");
    }
};//高精
bignum power(int x,int y)
{
    bignum s=bignum(1),p=bignum(x);
    if(y==0) return s;
    for(;y;y/=2,p=p*p)
        if(y&1) s=s*p;
    return s;
}
void work(int i,int j,int k)
{
    if(f[i][k]<=f[i-1][j]+(j/k-1)*lg[i]) return;
    mst[i][k]=j;
    // wrs(mst[i][k]);
    f[i][k]=f[i-1][j]+(j/k-1)*lg[i];
}
signed main()
{
    freopen("reward.in","r",stdin);
    freopen("reward.out","w",stdout);
    for(int i=1;i<17;i++)
        lg[i]=log10(zhi[i]);
    // for(int i=1;i<17;i++)
        // printf("%lf ",lg[i]);
    scanf("%d",&n);
    if(n==1)
    {
        puts("1");
        return 0;
    }
    for(int i=0;i<=16;i++)
        for(int j=1;j<=n;j++)
            f[i][j]=1e15;
    f[0][n]=1;
    for(int i=1;i<17;i++)
        for(int j=1;j<=n;j++)
        {
            if(n%j) continue;
            int k=1;
            for(;k*k<j;k++)
            {
                // wrs(i);
                // wrs(j);
                // wln(k);
                if(j%k) continue;
                work(i,j,k);
                work(i,j,j/k);
            }
            if(k*k==j) work(i,j,k);
        }
/*循环中i表示第i个素数,j表示未使用第i个素数前剩下的因数个数,k表示使用第i个素数后剩下的因数个数,即当前使用的实际是j/k-1个*/

    int j=1;
    bignum ans=bignum(1);
    for(int i=16;i;i--)
    {
        ans=ans*power(zhi[i],mst[i][j]/j-1);
        j=mst[i][j];
    }
    ans.print();
    return 0;
}

3.书本整理
【问题描述】
小明的书架上放了许多书,为了使书架变得整洁,小明决定整理书架,他将所有书按高度大小排列,这样排了之后虽然整齐了许多,但小明发现,书本的宽度不同,导致书架看上去还是有些凌乱。小明把这个凌乱值定义为相邻两本书的宽度差的绝对值的和。
例如有4本书:
1×2
5×3
2×4
3×l
那么小明将其排列整齐后的顺序是:
1×2
2×4
3×l
5x3
凌乱值就是2+3+2=7。
于是小明决定拿掉其中的k本书,使凌乱值最小,你能帮他求出这个最小值吗?已知每本书的高度都不一样。
【问题输入】
第一行两个数字n和k,代表书总共有n本,要求从中去掉k本。(l≤n≤100,1≤k<n)。下面的n行,每行两个数字表示一本书的高度和宽度,它们均小于200。
【问题输出】
一行一个整数,表示书架的最小凌乱值。
【样例输入】
4 1
1 2
2 4
3 1
5 3
【样例输出】
3
【数据范围】
30%的数据,n≤20。
100%的数据,n≤l00.k<n。

求n本书去掉k本的最小凌乱值。
f[i][j]表示前i本书去除j本书,第i本必须留下所获得的最小代价
[伪代码]
for(int kk=0;kk<=min(k,i-2);kk++)
    f[i][j]=min(f[i][j],f[i-kk-1][j-kk]+abs(a[i].w-a[i-kk-1].w));
if(j==i-1) f[i][j]=0;
ans=min(f[n-i][k-i])(0<=i<=k)
或者反过来更好理解:求n本书取出 n-k本最小凌乱值。
f[i][j]表示前i本书取j本书,第i本必须留下所获得的最小代价
for(int i=1;i<=n;i++){
        f[i][1]=0;
        for(int j=2;j<=min(i,n-k);j++)
            for(int l=1;l<i;l++)
                f[i][j]=min(f[i][j],f[l][j-1]+abs(b[l].w-b[i].w));
    }
for(int i=n-k;i<=n;i++)ans=min(ans,f[i][n-k]);
这是我挂掉的dp
for(int i=0;i<=n;i++) f[i][0]=0;
    for(int i=1;i<=n;i++)
    for(int j=1;j<=n-k;j++)
    for(int q=1;q<i;q++)
    {
        f[i][j]=min(f[i-1][j],f[i-1][j-1]+abs(a[q].w-a[i].w));
    }
思路和yg的差不多 然后循环就不知道怎么写了。。还去压了一维。。
yg//第i位取j个最后一个为k
    for(int i=1;i<=n;i++)
        for(int j=1;j<=min(i,n-m);j++)
            for(int k=j;k<=i;k++){
                if(j==1){f[i][j][k]=0;continue;}
                for(int p=1;p<k;p++)f[i][j][k]=min(f[i][j][k],f[i-1][j-1][p]+abs(a[k].w-a[p].w));
            }
    int ans=0x7fffffff;
    for(int i=n-m;i<=n;i++)ans=min(ans,f[n][n-m][i]);

4.木棍
有N根木棍,每根的长度L和重量W已知。这些木棍将被一台机器一根一根地加工。机器需要一些启动时间来做准备工作,启动时间与木棍被加工的具体情况有关。启动时间遵循以下规则:
1.加工第一根木棍的启动时间为1分钟。
2.加工完长度为Li,重量为Wi的木棍后,紧跟着加工长度为Li+1,重量为
Wi+1的木棍时,若Li<=Li+1且Wi<=Wi+1,则加工木棍I+1时,不需要启动时间。例如:有5根木棍,它们的长度和重量为(9,4),(2,5),(1,2,),(5,3),(4,1),则最小总启动时间为2分钟(加工序列为(4,1),(5,3),(9,4),(1,2),(2,5))。

输入:
第一行一个整数n(1<=n<=5000),表示木棍的数量。第二行2n个整数,l1,w1,l2,w2,…,ln,wn(1<=li,wi<=10000),为各根木棍的长度和重量,这2n个整数以若干个空格分隔。

输出:
一行: 一个整数,即最小总启动时间。

样例输入1
5
4 9 5 2 2 1 3 5 1 4

样例输出1
2

样例输入2
3
2 2 1 1 2 2

样例输出2
1

贪心。按长度排序,按宽度求lis,标记,继续求lis,直到全部标记完毕。做过的题目思路有影响然而写挂了?!以后还是要好好总结

既然写了那么多遍代码就不给了
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容