CF1073E Segment Sum

题意翻译

给定\(K,L,R\),求\(L\)~\(R\)之间最多不包含超过\(K\)个数码的数的和。

\(K<=10,L,R<=1e18\)


我 写 晕 了

我 自 闭 了

根本不知道自己在写什么????

告辞。。。


错误Code:

#include <cstdio>
#define ll long long
const ll mod=998244353;
ll dp[2][20][1<<10],cnt[2][20][1<<10],po[20],l,r;
int k;
void init()
{
po[0]=1;
for(int i=1;i<=18;i++) po[i]=po[i-1]*10%mod;
cnt[0][0][0]=1;
for(int i=1;i<=18;i++)
for(int s=0;s<1<<10;s++)
{
for(int j=1;j<=9;j++)
if(s>>j&1)
{
int t=(1<<j)^s;
(cnt[1][i][s]+=(cnt[0][i-1][s]+cnt[1][i-1][s]
+cnt[0][i-1][t]+cnt[1][i-1][t]))%=mod; }
int t=s^1;
if(s&1)
(cnt[0][i][s]+=(cnt[0][i-1][s]+cnt[1][i-1][s]
+cnt[0][i-1][t]+cnt[1][i-1][t]))%=mod;
}
for(int i=1;i<=18;i++)
for(int s=0;s<1<<10;s++)
{
for(int j=1;j<=9;j++)
if(s>>j&1)
{
int t=(1<<j)^s;
ll tmp0=(cnt[0][i-1][s]+cnt[1][i-1][s]
+cnt[0][i-1][t]+cnt[1][i-1][t])%mod;
ll tmp1=(dp[0][i-1][s]+dp[1][i-1][s]
+dp[0][i-1][t]+dp[1][i-1][t])%mod;
(dp[1][i][s]+=(tmp1+j*po[i-1]%mod*tmp0))%=mod; }
if(s&1)
{
int t=s^1;
ll tmp1=(dp[0][i-1][s]+dp[1][i-1][s]
+dp[0][i-1][t]+dp[1][i-1][t])%mod;
dp[0][i][s]=tmp1;
}
}
}
int bit[20];
ll solve(ll d)
{
int rt=0;
for(ll i=d;i;i/=10) bit[++rt]=i%10;
int cho=0;//高位已选状态
ll res=0,ans=0;//高位剩余
for(int i=rt;i;i--)
{
for(int s=0;s<1<<10;s++)
{
int ct=0;
for(int t=s;t;t>>=1) ct+=t&1;
if(ct>k+1) continue;
if(s&1)
{
int t=s^1;
ll tmp1=(dp[0][i-1][s]+dp[1][i-1][s]
+dp[0][i-1][t]+dp[1][i-1][t])%mod;
ans+=tmp1;
}
if(ct>k) continue;
for(int j=1;j<bit[i]+(i==1);j++)//高位选填
if((s>>j&1)&&((s|cho)==s))
{
int t=(1<<j)^s;
ll tmp0=(cnt[0][i-1][s]+cnt[1][i-1][s]
+cnt[0][i-1][t]+cnt[1][i-1][t])%mod;
ll tmp1=(dp[0][i-1][s]+dp[1][i-1][s]
+dp[0][i-1][t]+dp[1][i-1][t])%mod;
(ans+=tmp1+tmp0*(j+res)%mod*po[i-1]%mod)%=mod;
}
}
res=(res+bit[i])*10%mod;
cho|=1<<bit[i];
}
return ans;
}
int main()
{
scanf("%lld%lld%d",&l,&r,&k);
init();
printf("%lld\n",((solve(r)-solve(l-1))%mod+mod)%mod);
return 0;
}

updata2019.2.9

写了好几个月,终于肝出来了

Code:

#include <cstdio>
#include <cstring>
#define ll long long
const int mod=998244353;
inline int add(int a,int b){return a+b>=mod?a+b-mod:a+b;}
#define mul(a,b) (1ll*(a)*(b)%mod)
int po[20],bit[20],len,k;
struct node
{
int val,cnt;
node(){}
node(int v,int c){val=v,cnt=c;}
node friend operator +(node a,node b){return node(add(a.val,b.val),add(a.cnt,b.cnt));}
}dp[20][1<<10];
node dfs(int pos,int sta,int lead,int lim)//前导0和最高位限制
{
int cnt=0;
for(int i=0;i<10;i++) cnt+=sta>>i&1;
if(cnt>pos) return node(0,0);
if(!pos) return node(0,1);
if(!lim&&!lead&&~dp[pos][sta].val) return dp[pos][sta];
node ret=node(0,0),bee;
if(lead) ret=ret+dfs(pos-1,sta,lead,lim&&!bit[pos]);
else if(sta&1) ret=ret+dfs(pos-1,sta,lead,lim&&!bit[pos])+dfs(pos-1,sta^1,lead,lim&&!bit[pos]);
for(int i=1,up=lim?bit[pos]:9;i<=up;i++)
if(sta>>i&1)
{
bee=dfs(pos-1,sta,0,lim&&i==up)+dfs(pos-1,sta^(1<<i),0,lim&&i==up);
ret=ret+bee;
ret.val=add(ret.val,mul(bee.cnt,mul(i,po[pos-1])));
}
return !lim&&!lead?dp[pos][sta]=ret:ret;
}
int cal(ll x)
{
len=0;while(x) bit[++len]=x%10,x/=10;
memset(dp,-1,sizeof dp);int ans=0;
for(int s=0;s<1<<10;s++)
{
int cnt=0;
for(int i=0;i<10;i++) cnt+=s>>i&1;
if(cnt<=k) ans=add(ans,dfs(len,s,1,1).val);
}
return ans;
}
int main()
{
ll l,r;
scanf("%lld%lld%d",&l,&r,&k);
po[0]=1;for(int i=1;i<=18;i++) po[i]=mul(po[i-1],10);
printf("%d\n",add(cal(r),mod-cal(l-1)));
return 0;
}

CF1073E Segment Sum 自闭了的更多相关文章

  1. CF1073E Segment Sum 解题报告

    CF1073E Segment Sum 题意翻译 给定\(K,L,R\),求\(L~R\)之间最多不包含超过\(K\)个数码的数的和. \(K\le 10,L,R\le 10^{18}\) 数位dp ...

  2. CF1073E Segment Sum

    数位DP,求[L,R]区间内所有"数字内包含的不同数码不超过k个的数字"之和.在状态上加一维状态压缩表示含有的数码集合.一开始读错题以为是求数字的个数.读对题之后调了一会儿. #i ...

  3. Codeforces 1073 E - Segment Sum

    E - Segment Sum 思路: 数位dp 我们平时做的数位dp都是求满足条件的数的个数, 这里要求满足条件的数的和 只要在原来的基础上求每一位的贡献就可以了,所以传参数时要传两个 代码: #p ...

  4. Educational Codeforces Round 53 E. Segment Sum(数位DP)

    Educational Codeforces Round 53 E. Segment Sum 题意: 问[L,R]区间内有多少个数满足:其由不超过k种数字构成. 思路: 数位DP裸题,也比较好想.由于 ...

  5. CodeForces - 1073E :Segment Sum (数位DP)

    You are given two integers l l and r r (l≤r l≤r ). Your task is to calculate the sum of numbers from ...

  6. E. Segment Sum(数位dp)

    题意:求一个区间内满足所有数位不同数字个数小于K的数字总和.比如:k=2   1,2,3所有数位的不同数字的个数为1满足,但是123数位上有三个不同的数字,即123不满足. 我们可以使用一个二进制的数 ...

  7. Educational Codeforces Round 53 (Rated for Div. 2) E. Segment Sum

    https://codeforces.com/contest/1073/problem/E 题意 求出l到r之间的符合要求的数之和,结果取模998244353 要求:组成数的数位所用的数字种类不超过k ...

  8. CF 1073 E. Segment Sum

    https://codeforces.com/problemset/problem/1073/E 题意:[l,r]中,出现0—9数字的种类数不超过k的数的和 dp[i][j][0/1] 表示 dfs到 ...

  9. Educational Codeforces Round 53 (Rated for Div. 2) E. Segment Sum (数位dp求和)

    题目链接:https://codeforces.com/contest/1073/problem/E 题目大意:给定一个区间[l,r],需要求出区间[l,r]内符合数位上的不同数字个数不超过k个的数的 ...

随机推荐

  1. 1.1.0 Unity零基础入门2——Roll a Ball

    1. 游戏界面 2.代码 //FoodRotate - - 控制cube旋转 using System.Collections; using System.Collections.Generic; u ...

  2. Java开发工程师(Web方向) - 03.数据库开发 - 期末考试

    期末考试 编程题 本编程题包含4个小题,覆盖知识点从基础的JDBC.连接池到MyBatis. 1(10分) 有一款在线教育产品“天天向上”主要实现了在手机上查看课程表的功能.该产品的后端系统有一张保存 ...

  3. leetcode-最大子序和(动态规划讲解)

    最大子序和(动态规划讲解) 给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和. 示例: 输入: [-2,1,-3,4,-1,2,1,-5,4], 输 ...

  4. hive使用spark引擎的几种情况

    使用spark引擎查询hive有以下几种方式:1>使用spark-sql(spark sql cli)2>使用spark-thrift提交查询sql3>使用hive on spark ...

  5. 文件上传:CommonsMultipartResolver

    一. 简介 CommonsMultipartResolver是基于Apache的Commons FileUpload来实现文件上传功能的,主要作用是配置文件上传的一些属性. 二. 配置 1)依赖Apa ...

  6. 【第三章】Shell 变量的数值计算

    一.算数运算符 shell中常见的算术运算符: shell中常见的算术命令: 1. 整数运算 方法一:expr  expr命令就既可以用于整数运算,也可以用于相关字符串长度.匹配等的运算处理: exp ...

  7. day-17 L1和L2正则化的tensorflow示例

    机器学习中几乎都可以看到损失函数后面会添加一个额外项,常用的额外项一般有两种,一般英文称作ℓ1-norm和ℓ2-norm,中文称作L1正则化和L2正则化,或者L1范数和L2范数.L2范数也被称为权重衰 ...

  8. oraclize预言机资料

    oraclize预言机资料 智能合约如何可信的与外部世界交互: https://blog.csdn.net/sportshark/article/details/77477842 国外一篇讲得很详细的 ...

  9. MyBatis 插件 : 打印 SQL 及其执行时间

    Plugins 摘一段来自MyBatis官方文档的文字. MyBatis允许你在某一点拦截已映射语句执行的调用.默认情况下,MyBatis允许使用插件来拦截方法调用: Executor(update. ...

  10. OpenCV学习4-----K-Nearest Neighbors(KNN)demo

    最近用到KNN方法,学习一下OpenCV给出的demo. demo大意是随机生成两团二维空间中的点,然后在500*500的二维空间平面上,计算每一个点属于哪一个类,然后用红色和绿色显示出来每一个点 如 ...