【比赛】【树上路径(phantasm)】

【比赛】【树上路径(phantasm)】—恢复内容开始—题目大意:求1,2,…,n有多少个长为m的子序列a,满足  a1=1,am=n  ∀i,ai+1−ai≥k保证这样的子序列存在。只需判断方案数的奇偶性。数据有T组。n,m,k≤109,T≤2×106.//dfs枚举集合//复杂度预估O(T*2^n)/…

大家好,又见面了,我是你们的朋友全栈君。

—恢复内容开始—

题目大意:

求 1, 2, …, n 有多少个长为 m 的子序列 a,

满足

   a1 = 1,am = n

   ∀i, ai+1 − ai ≥ k

保证这样的子序列存在。只需判断方案数的奇偶性。数据有 T 组。 n, m, k ≤ 109 , T ≤ 2 × 106 .

【比赛】【树上路径(phantasm)】

//dfs枚举集合
//复杂度预估 O(T*2^n)
//不用数组 得分30 
//针对前6个点
#include<cstdio>
#include<cstdlib>
using namespace std;
int t,n,m,k; 
int sum;

void dfs(int pos,int x)
{
    if(pos==m) 
    {
        if(x==n)
            sum=(sum+1)%2
        return ;
    }
    for(int i=x+k;i<=n;i++)
        dfs(pos+1,i);
}

int main()
{
    scanf("%d",&t);
    while(t--)
    {
        scanf("%d%d%d",&n,&m,&k);
        sum=0;
        dfs(1,1);
        if(sum) printf("Yes\n");
        else printf("No\n");
    }
    return 0;
}

对于第8个点

        {
                scanf("%d%d%d",&n,&m,&k);
        if(m==2) printf("Yes\n");                                   
        else if(m==3) 
        {
            sum=n-(k<<1);
            if(sum%2==1) printf("Yes\n");
            else printf("No\n");
        }
        else
        {
            sum=0;
            dfs(1,1);
            if(sum) printf("Yes\n");
            else printf("No\n");
        }     
    }

//2记忆化 
//dfs->dp //f[i][j]表示第i步,位置为j //因为状态转移方程中的k为变量,所以还是得分k的大小来dp //f[k][i][j] // 55 分
#include<cstdio> 
#include
<cstdlib>
#include
<algorithm>
#include
<cstring>
using namespace std;

int tt,nn,mm,kk;
int f[203][203][203];//1:true
inline int read()
{
  int x=0;char c=getchar();
  while(c<'0' || c>'9') c=getchar();
  while(c>='0' && c<='9') x=(x<<1)+(x<<3)+c-'0',c=getchar();
  return x;
}
int dfs(int n,int m,int k)
{
  if(f[k][m][n]!=-1) return f[k][m][n];
  if(m==1) return f[k][m][n]=0;
  if(m==2 )
    if(n>k) return f[k][m][n]=1;
    else return f[k][m][n]=0;
  int s=k ,t=n-k*(m-2),as=0;
  for(int i=s;i<=t;i++)
    as=(as+dfs(n-i,m-1,k))%2;
  return f[k][m][n]=as;
}

int main()
{
  tt
=read();
  memset(f,
-1,sizeof(f));
  for(int i=1;i<=tt;i++)
  {
    nn
=read(),mm=read(),kk=read();
    if(mm==2) printf("Yes\n");
    else if(mm==3)
    {
      int sum=nn-(mm-1)*kk;
      if(sum%2!=0) printf("Yes\n");
      else printf("No\n");
    }
    else if( dfs(nn,mm,kk) ) printf("Yes\n");
    else printf("No\n"); }
   return 0;
}

90分算法(n,m,k<=5000) 

推式子。

注意到第二条约束与 a 的差分序列有关,考虑统计差分序列 bi = ai+1 − ai。

序列 b 需要满足

∀i, bi 是 [k, n] 上的整数

∑bi = n − 1

由于 a1 = 1 已确定,可以发现所有满足以上约束的长为 m − 1 的序列 b 与原来的序列 a 一一对应。

 

有限项的正整数序列的和确定时,其方案数可以用隔板法计 算,

所以构造 ci = bi − (k − 1) 来去除第一条约束,

序列 c 满足

∀i, ci 是正整数

∑m−1  ci = n − 1 − (m − 1)(k − 1)

#include<cstdio>
#include<cstdlib>
#include<algorithm>
using namespace std;
int t,nn,mm,kk;
struct node
{
    
    //m=m-2 
    int id,n,m;//n=n-2-(m-1)(k-1)
    bool operator < (const node & o) const
    {
        if(n!=o.n) return n<o.n;
        else return m<o.m;
    }
}ask[2000003];
bool ans[2000003],f[5003];

int main()
{
    scanf("%d",&t);
    for(int i=0;i<t;i++)
        scanf("%d %d %d",&nn,&mm,&kk),
        ask[i].n =nn-2-(mm-1)*(kk-1),ask[i].m =mm-2,ask[i].id =i;
    sort(ask,ask+t);
    
    int nw=0,nwn=1;
    nn=ask[nw].n ;
    f[0]=true;
    while(nw<t)
    {
        for(int i=nwn;i<=nn;i++)
            for(int j=i;j>=0;j--)
                f[j]^=f[j-1];
        nwn=nn+1;
        
        while(ask[nw].n ==nn)
            ans[ask[nw].id ]=f[ask[nw].m ],nw++;
        
        nn=ask[nw].n ;
    }
    
    for(int i=0;i<t;i++)
        if(ans[i]) printf("Yes\n");
        else printf("No\n");
        
    return 0;
}

100分算法

(解决两个1e9的超大点)

由 Lucas 定理,( n k ) ≡ 1 (mod 2)

当且仅当二进制下 k 的各 位都不大于 n 的对应位,即 n and k = k,

其中 and 为二进制按 位与。 复杂度:O(T)。

#include <cstdio>
#include <cstdlib>
using namespace std;
int t,n,m,k;
int main ()
{
    scanf("%d",&t);
    while(t--)
    {
        scanf("%d%d%d",&n,&m,&k);
        int a=n-(m-1)*(k-1)-2;
        int b=m-2;
        if((a&b)==b)
            printf("Yes\n");
        else
            printf("No\n");
    }
    return 0;
}

 

转载于:https://www.cnblogs.com/xwww666666/p/11325852.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/139060.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • ubuntu卸载cuda10.2_dpkg强制卸载软件

    ubuntu卸载cuda10.2_dpkg强制卸载软件一、参考资料CUDA、CUDNN在Ubuntu下的安装及配置二、注意事项用deb方式安装CUDA,会附带安装显卡驱动;用run方式安装CUDA,需要提前安装好显卡驱动;安装显卡驱动的时候,最好安装高版本的,这样不会受cuda版本的影响;三、run方式卸载用run方式安装的CUDA和驱动#uninstallcuda#第一行命令不要忘记要加上perl命令,要不然会报错sudoperl/usr/local/cuda-X.Y/bin/uninstall_cuda_X.Y.pl

    2022年9月4日
    2
  • webpack json_vue读取json文件

    webpack json_vue读取json文件方案删除webpack,重新装以前的版本。npmuninstallwebpacknpminstallwebpack@^4.0.0–save-dev

    2022年8月9日
    13
  • LoadRunner教程(18)-LoadRunner 图表合并[通俗易懂]

    LoadRunner教程(18)-LoadRunner 图表合并[通俗易懂]分析图合并一、分析图合并原理选择view-&amp;gt;mergegraphs,弹出所示对话框1、选择要合并的图。选择一个要与当前活动图合并的图,注意这里只能选择X轴度量单位相同的图。2、选择合并类型。1)叠加:查看共用同一X轴的两个图的内容。合并图左侧的Y轴显示当前图的Y轴值,右边的Y轴显示合并进来的图的Y轴值,如图所示2)平铺:在平铺布局查看,共用同一个X轴,合…

    2022年5月10日
    47
  • excel旭日图_旭日图怎么画

    excel旭日图_旭日图怎么画旭日图(Sunburst)由多层的环形图组成,在数据结构上,内圈是外圈的父节点。因此,它既能像饼图一样表现局部和整体的占比,又能像矩形树图一样表现层级关系。引入相关文件旭日图是ApacheEChartsTM4.0新增的图表类型,从CDN引入完整版的echarts.min.js最简单的旭日图创建旭日图需要在series配置项中声明类型为‘sunburst’的系列,并且以树形结构声明其data:varoption={series:{type

    2022年9月25日
    0
  • 使用BRVAH RecycleView 嵌套RecycleView点击Item里面内容无法响应

    使用BRVAH RecycleView 嵌套RecycleView点击Item里面内容无法响应

    2021年3月12日
    155
  • 模仿学习(Imitation Learning)入门

    模仿学习(Imitation Learning)入门在游戏中,我们往往有一个计分板准确定义事情的好坏程度。但现实中,定义Reward有可能是非常困难的,并且人定的reward也有可能存在许多意想不到的缺陷。在没有reward的情况下,让AI跟环境互动的一个方法叫做Imitation-Learning。在没有reward的前提下,我们可以找人类进行示范,AI可以凭借这些示范以及跟环境的互动进行学习。这种模仿学习使得智能体自身不必从零学起,不必去尝试探索和收集众多的无用数据,能大大加快训练进程。这跟supervised-learning有类似之处,如果采用这种

    2022年9月19日
    0

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号