POJ 1742 Coins ( 单调队列解法 )「建议收藏」

POJ 1742 Coins ( 单调队列解法 )

大家好,又见面了,我是全栈君。

博客

再加上这题体积与价值相等那么就更好做了。仅仅有 j %v[ i ] 余数同样的才干够同一时候处理(j 指的是某个体积的值),在计算某个数的时候,仅仅要计算前面的同样的余数中(在个数限制内)是否有 true(有放满的) 就能够了。

代码:

#include<iostream>
#include<sstream>
#include<map>
#include<cmath>
#include<fstream>
#include<queue>
#include<vector>
#include<sstream>
#include<cstring>
#include<cstdio>
#include<stack>
#include<bitset>
#include<ctime>
#include<string>
#include<cctype>
#include<iomanip>
#include<algorithm>
using namespace std  ;
#define INT long long int
#define L(x)  (x * 2)
#define R(x)  (x * 2 + 1)
const int INF = 0x3f3f3f3f ;
const double esp = 0.0000000001 ;
const double PI = acos(-1.0) ;
const int mod = 1000000007 ;
const int MY = (1<<5) + 5 ;
const int MX = 100010 + 5 ;
int n ,W ,ans ;
int v[MX] ,num[MX] ;
bool deq[MX] ,dp[MX] ;
void input()
{
    memset(dp ,false ,sizeof(dp)) ;
    for(int i = 1 ;i <= n ; ++i)
        scanf("%d" ,&v[i]) ;
    for(int i = 1 ;i <= n ; ++i)
        scanf("%d" ,&num[i]) ;
}
void DP(int v ,int num)
{
    if(!num || !v)  return ;
    if(num == 1) // 01 背包
    {
        for(int i = W ;i >= v ; --i)
           if(!dp[i] && dp[i-v])
               dp[i] = true ,ans++ ;
    }
    else if(num * v >= W) // 全然背包
    {
        for(int i = v ;i <= W ; ++i)
          if(!dp[i] && dp[i-v])
              dp[i] = true ,ans++ ;
    }
    else
    {
        num = min(num ,W/v) ;
        for(int a = 0 ;a < v ; ++a)  // 同样余数一块处理
        {
            int front =0 ,end = 0 ,sum = 0 ;
            for(int j = a ;j <= W ; j += v)
            {
                if(end - front-1 == num)  // 去除过时元素 ,由于最多选择num[i] 个
                    sum -= deq[front++] ;
                deq[end++] = dp[j] ;  // 存入
                sum += dp[j] ;
                if(!dp[j] && sum)
                    dp[j] = true ,ans++ ;

            }
        }
    }
}
int main()
{
    //freopen("input.txt" ,"r" ,stdin) ;
    while(scanf("%d%d" ,&n ,&W) ,n+W)
    {
        input() ;
        dp[0] = true ;
        ans = 0 ;
        for(int i = 1 ;i <= n ; ++i)
             DP(v[i] ,num[i]) ;
        printf("%d\n" ,ans) ;
    }
    return 0 ;
}

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

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

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


相关推荐

  • mysql和redis的区别

    mysql和redis的区别1.mysql和redis的数据库类型mysql是关系型数据库,主要用于存放持久化数据,将数据存储在硬盘中,读取速度较慢。redis是NOSQL,即非关系型数据库,也是缓存数据库,即将数据存储在缓存中,缓存的读取速度快,能够大大的提高运行效率,但是保存时间有限2.mysql的运行机制mysql作为持久化存储的关系型数据库,相对薄弱的地方在于每次请求访问数据库时,都存在着I/O操作,如果反复…

    2022年6月14日
    42
  • COM编程之三 QueryInterface

    COM编程之三 QueryInterface【1】IUnknown接口客户同组件交互都是通过接口完成的。在客户查询组件的其它接口时,也是通过接口完成的。而那个接口就是IUnknown。IUnknown接口的定义包含在Win32SDK中的UNKNEN.h头文件中。引用如下:1interfaceIUnknown2{3virtualHRESULT__stdcallQueryInterface(const…

    2022年6月23日
    25
  • SQL Server 2008支持将数据导出为脚本

    SQL Server 2008支持将数据导出为脚本

    2021年8月5日
    82
  • 交叉线与直通线的区别

    交叉线与直通线的区别网线分为两种:直通线和交叉线。1>直通线:标准线,两端都采用568B做线标准。两端的线序对是:1、白橙、2、橙、3、白绿、4、蓝、5、白蓝、6、绿、7、白棕、8、棕。注意两端都是同样的线序且一一对应,这种线就是我们平时最常用的网线。直通线一般连接不同的设备,比如电脑和路由器。2>交叉线:反线,一端采用568B做线标准,一端采用568A的标准。一端的线序对是:1、白橙、2、橙

    2022年6月19日
    34
  • SQL数据库之索引优缺点

    SQL数据库之索引优缺点 SQL数据库之索引使用原则及利弊 索引是对数据库表中一列或多列的值进行排序的一种结构,使用索引可快速访问数据库表中的特定信息。 优点通过创建唯一性索引,可以保证数据库表中每一行数据的唯一性。  可以大大加快数据的检索速度,这也是创建索引的最主要的原因。  可以加速表和表之间的连接,特别是在实现数据的参考完整性方面特别有意义。  在使用分组和排序子句进行数据检索时,…

    2022年5月25日
    41
  • 关于flask的SSTI注入[通俗易懂]

    关于flask的SSTI注入[通俗易懂]ssti注入又称服务器端模板注入攻击(Server-SideTemplateInjection),和sql注入一样,也是由于接受用户输入而造成的安全问题。它的实质就是服务器端接受了用户的输入,没有经过过滤或者说过滤不严谨,将用户输入作为web应用模板的一部分,但是在进行编译渲染的过程中,执行了用户输入的恶意代码,造成信息泄露,代码执行,getshell等问题。这个问题主要是出在web应…

    2022年8月30日
    4

发表回复

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

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