zoukankan      html  css  js  c++  java
  • SRM 549 DIV2

    250pt

    题意:

      有左,中,右三个位置,一个球占据其中的一个,移动球到相邻的位置称为一次移动,求n次移动后球最可能在哪个位置(相同则字典序最小)?

     

    分析:给力的250pt,还以为要用动态规划做,因为只有三个位置,可枚举。
      0,移动次数为0肯定在原位置

      1,球不在中间时,第一步肯定移到中间,

      2,球中间时,剩余的移动数如果为偶数,肯定会回到中间,奇数的数,因为左右对称,左边50%,右边50%,字典序小,所以是左边

     

    View Code
    class BallAndHats 
    { 
            public: 
            int getHat(string hats, int numSwaps) 
            { 
                    int k;
                    for(k=0;k<3;k++)
                        if(hats[k]=='o')
                            break;
                    if(!numSwaps)
                        return k;
                    if(k!=1)
                        numSwaps--,k=1;
                    if(numSwaps&1)
                        return 0;
                    return 1;
            } 
    }; 

     

    500pt

    题意:

      给定若干个顶部圆锥和若干个底部圆锥,一个顶部圆锥堆在底部圆锥上组成一个帽子,求最多能组成多少个帽子

     

    分析:500pt的题居然是二分匹配题

      组成帽子的前提是顶圆锥(:th,半径:tr)能堆在底圆锥(:bh,半径:br)上。两个条件:

        (1),顶半径大于底半径tr<br

        (2),顶圆锥顶点高于底圆锥顶点(tr*bh/br)<th

      剩下的是求最大二分匹配,套的是模板。忘了,默写不出来,又重新复习了一遍。

    View Code
    class PointyWizardHats 
    { 
            public:
                bool can[55][55],vis[55];
                int link[55],n,m;
            bool find(int v)
            {
                for(int w=0;w<m;w++)
                    if(!vis[w] && can[v][w])
                    {
                        vis[w]=true;
                        if(link[w]==-1||find(link[w]))
                        {
                            link[w]=v;
                            return true;
                        }
                    }
                return false;
            }
            int getNumHats(vector <int> th, vector <int> tr, vector <int> bh, vector <int> br) 
            { 
                    int i,j,k;
                    memset(can,false,sizeof(can));
                    memset(link,-1,sizeof(link));
                    n=th.size();
                    m=bh.size();
                    for(i=0;i<n;puts(""),i++)
                        for(j=0;j<m;j++)
                        {
                            if(tr[i]<br[j]&&(th[i]>(1.0*tr[i]*bh[j]/br[j])))
                                can[i][j]=true;
                            printf("%d ",can[i][j]?1:0);
                        }
                    for(i=k=0;i<n;i++)
                    {
                        memset(vis,false,sizeof(vis));
                        if(find(i))
                            k++;
                    }
                    return k;
            } 
    };

    1000pt

    题意:

      给定一幅至多20个点的有向图,求至少要删除多少条边才能变成无环图。

    分析:

      若已求得结果,那么结果一定存在一个顶点的排列(V0,V1,V2....Vn-1)使任意j>=i不存在边VjVi

      顶点个数极少,无后续性(加一个点到集合中时不会影响原集合元素之间的关系),重叠子问题(每次都是加一个点到集合中)可用集合DP,

      集合的元素压缩成状态s的二进制位,为了方便计算,s0表示含有对应二进制位的元素,1则相反,f[s]表示使集合状态为s无环化所需最少删边数。

     结:又是一道不敢写的深搜题

     

    View Code
    #include <cstdlib> 
    #include <cctype> 
    #include <cstring> 
    #include <cstdio> 
    #include <cmath> 
    #include <algorithm> 
    #include <vector> 
    #include <string> 
    #include <iostream> 
    #include <sstream> 
    #include <map> 
    #include <set> 
    #include <queue> 
    #include <stack> 
    #include <fstream> 
    #include <iomanip> 
    #include <bitset> 
    #include <list> 
    #include <strstream> 
    using namespace std; 
    #define REP(i,n) for(i=0;i<(n);++i) 
    #define FOR(i,l,h) for(i=(l);i<=(h);++i) 
    #define FORD(i,h,l) for(i=(h);i>=(l);--i)
    #define INF (1<<20)
    typedef vector<int> VI; 
    typedef vector<double> VD; 
    typedef long long LL; 
    
    int dp[1<<20];
    
    class OrderOfTheHats 
    { 
            public:
            int a[22],n;
            int Count(int x)
            {
                int i=0;
                for(;x;x>>=1)
                    i+=(x&1);
                return i;
            }
            int F(int s)
            {
                int i,k,t,ret=INF;
                if(s==(1<<n)-1)
                    return 0;
                if(dp[s]!=-1)
                    return dp[s];
                for(i=0;i<n;i++)
                    if((s&(1<<i))==0)
                    {
                        k=Count(a[i]&s);
                        if(a[i]&(1<<i))
                            k++;
                        t=F(s|(1<<i));
                        if(ret>k+t)
                            ret=k+t;
                    }
                return dp[s]=ret;
            }
            int minChanged(vector <string> spellChart) 
            {
                int i,j;
                n=spellChart.size();
                memset(dp,-1,sizeof(dp));
                for(i=0;i<n;i++)
                    for(j=a[i]=0;j<n;j++)
                        if(spellChart[i][j]=='Y')
                            a[i]|=(1<<j);
                return F(0);
            } 
            
    }; 

     

     

  • 相关阅读:
    函数---迭代器&生成器&列表解析&三元表达式
    python函数模拟mysql增删改查功能
    装饰器
    函数----模块化的程序设计
    文件操作
    字符编码
    python的对象类型-----列表&元组&字典
    Mac上vmware虚拟机Windows10安装JDK8及配置环境
    Windows10显示桌面我的电脑等图标
    HTTP状态码:300400500 错误代码
  • 原文地址:https://www.cnblogs.com/xchaos/p/2604326.html
Copyright © 2011-2022 走看看