P2763 试题库问题

\(\color{#0066ff}{题目描述}\)

?问题描述:

假设一个试题库中有n道试题。每道试题都标明了所属类别。同一道题可能有多个类别属性。现要从题库中抽取m 道题组成试卷。并要求试卷包含指定类型的试题。试设计一个满足要求的组卷算法。

?编程任务:

对于给定的组卷要求,计算满足要求的组卷方案。

\(\color{#0066ff}{输入格式}\)

第1行有2个正整数k和n (2<=k<=20, k<=n<=1000),k 表示题库中试题类型总数,n 表示题库中试题总数。

第2 行有k 个正整数,第i 个正整数表示要选出的类型i的题数。这k个数相加就是要选出的总题数m。

接下来的n行给出了题库中每个试题的类型信息。每行的第1 个正整数p表明该题可以属于p类,接着的p个数是该题所属的类型号。。

\(\color{#0066ff}{输出格式}\)

第i 行输出 “i:”后接类型i的题号。如果有多个满足要求的方案,只要输出1个方案。如果问题无解,则输出“No Solution!”。

有spj

\(\color{#0066ff}{输入样例}\)

3 15
3 3 4
2 1 2
1 3
1 3
1 3
1 3
3 1 2 3
2 2 3
2 1 3
1 2
1 2
2 1 2
2 1 3
2 1 2
1 1
3 1 2 3

\(\color{#0066ff}{输出样例}\)

1: 1 6 8
2: 7 9 10
3: 2 3 4 5

\(\color{#0066ff}{题解}\)

这题主要是建模,其实挺好想的

注意有spj

建立超级源s和超级汇t

\(\color{#00ccff}{每个类型}\)向\(\color{#00ccff}{t}\)连容量为\(\color{red}{该类型所需题数}\)的边,以保证每个类型得到应有题数

\(\color{#00ccff}{每个题}\)向\(\color{#00ccff}{其所属类型}\)连容量为\(\color{red}{1}\)的边,因为它对每个类型只能产生1的贡献

\(\color{#00ccff}{s}\)向\(\color{#00ccff}{每个题}\)连容量为\(\color{red}{1}\)的边,因为一个题被分且仅被分到一个集合中去
给他限制分配量1,让它最多流向一个集合

#include<cstdio>
#include<queue>
#include<vector>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cctype>
#include<cmath>
#define _ 0
#define LL long long
#define Space putchar(' ')
#define Enter putchar('\n')
#define fuu(x,y,z) for(int x=(y);x<=(z);x++)
#define fu(x,y,z)  for(int x=(y);x<(z);x++)
#define fdd(x,y,z) for(int x=(y);x>=(z);x--)
#define fd(x,y,z)  for(int x=(y);x>(z);x--)
#define mem(x,y)   memset(x,y,sizeof(x))
#ifndef olinr
inline char getc()
{
    static char buf[100001],*p1=buf,*p2=buf;
    return (p1==p2)&&(p2=(p1=buf)+fread(buf,1,100001,stdin),p1==p2)? EOF:*p1++;
}
#else
#define getc() getchar()
#endif
template<typename T>inline void in(T &x)
{
    int f=1; char ch; x=0;
    while(!isdigit(ch=getc()))(ch=='-')&&(f=-f);
    while(isdigit(ch)) x=x*10+(ch^48),ch=getc();
    x*=f;
}
struct node
{
    int to;
    int nxt;
    int cap,flow;
}e[5050505];
int head[10505],cur[10505],dep[10505];
bool vis[10505];
int cnt=1;
int s,t;
int n,k;
std::queue<int> q;
const int inf=0x7fffffff;
inline void add(int from,int to,int cap)
{
    cnt++;
    e[cnt].to=to;
    e[cnt].cap=cap;
    e[cnt].flow=0;
    e[cnt].nxt=head[from];
    head[from]=cnt;
}
inline bool bfs()
{
    fuu(i,0,n+k+1) cur[i]=head[i],dep[i]=0;
    q.push(s);
    while(!q.empty())
    {
        int tp=q.front(); q.pop();
        for(int i=head[tp];i;i=e[i].nxt)
        {
            int go=e[i].to;
            if(!dep[go]&&e[i].cap>e[i].flow)
            {
                dep[go]=dep[tp]+1;
                q.push(go);
            }
        }
    }
    return dep[t];
}
inline int dfs(int x,int change)
{
    if(!change||x==t) return change;
    int flow=0,ls=0;
    for(int i=head[x];i;i=e[i].nxt)
    {
        int go=e[i].to;
        if(dep[go]==dep[x]+1&&(ls=dfs(go,std::min(change,e[i].cap-e[i].flow))))
        {
            flow+=ls;
            change-=ls;
            e[i].flow+=ls;
            e[i^1].flow-=ls;
            if(!change) break;
        }
    }
    return flow;
}
inline void maxflow()
{
    while(bfs()) dfs(s,inf);
}
int main()
{
    in(n),in(k);
    s=0,t=n+k+1;
    int x,y;
    fuu(i,1,n) in(x),add(i+k,t,x),add(t,k+i,0);
    fuu(i,1,k)
    {
        in(x);
        fuu(j,1,x) in(y),add(i,k+y,1),add(k+y,i,0);
        add(s,i,1),add(i,s,0);
    }
    maxflow();
    fuu(i,1,n)
    {
        printf("%d:",i);
        for(int j=head[i+k];j;j=e[j].nxt)
        {
            int go=e[j].to;
            if(go>=1&&go<=k&&e[j].flow==-1) printf(" %d",go);
        }
        Enter;
    }
    return ~~(0^_^0);
}

原文地址:https://www.cnblogs.com/olinr/p/9984598.html

时间: 2024-11-12 01:17:45

P2763 试题库问题的相关文章

洛谷P2763 试题库问题

题目:https://www.luogu.org/problemnew/show/P2763 题目描述 «问题描述: 假设一个试题库中有n道试题.每道试题都标明了所属类别.同一道题可能有多个类别属性.现要从题库中抽取m 道题组成试卷.并要求试卷包含指定类型的试题.试设计一个满足要求的组卷算法. «编程任务: 对于给定的组卷要求,计算满足要求的组卷方案. 输入输出格式 输入格式: 第1行有2个正整数k和n (2 <=k<= 20, k<=n<= 1000) k 表示题库中试题类型总数

luogu P2763 试题库问题

本题可以用最大流也可以用最大匹配(本质一样),用dinic最大流好建图,但码量大,匈牙利码量小,建图费点劲. 最大流:依旧是设一个源点一个汇点,对于每一个种类,连一条到汇点的边,capacity为需要的量,对于每一个试题,从源点连一条capacity为1的边到他,从他对每一个其所属的编号种类连一条capacity为1的边,求最大流即可,再找出最小割即可 #include<bits/stdc++.h> using namespace std; #define lowbit(x) ((x)&

【题解】Luogu P2763 试题库问题

原题传送门 这题很简单啊 从源点向k类题目分别连流量为所需数量的边 从每道题向汇点连一条流量为1的边(每题只能用1次) 从类型向对应的题目连一条流量为1的边 跑一边最大流 如果最大流小于所需题目数量,就无解 否则对于每个类型,看每道题是否要选 #include <bits/stdc++.h> #define inf 0x3f3f3f3f #define M 80005 #define N 2005 //#define getchar nc using namespace std; inline

COGS732. [网络流24题] 试题库

«问题描述:假设一个试题库中有n道试题.每道试题都标明了所属类别.同一道题可能有多个类别属性.现要从题库中抽取m 道题组成试卷.并要求试卷包含指定类型的试题.试设计一个满足要求的组卷算法.«编程任务:对于给定的组卷要求,计算满足要求的组卷方案.«数据输入:由文件testlib.in提供输入数据.文件第1行有2个正整数k和n (2 <=k<= 20, k<=n<= 1000)k 表示题库中试题类型总数,n 表示题库中试题总数.第2 行有k 个正整数,第i 个正整数表示要选出的类型i

数据库笔试面试题库(Oracle、MySQL等)

数据库笔试面试题库(Oracle.MySQL等) ⊙ [DB笔试面试67]在Oracle中,关于表分区下列描述不正确的是()⊙ [DB笔试面试65]在Oracle中,哪一种表分区方式建议的分区数是2的幂(2.4.8等),以获得最平均的数据发布()⊙ [DB笔试面试63]要以NAME's address is ADDR格式返回数据,以下SQL语句正确的是⊙ [DB笔试面试61]以下关于数据模型要求错误的是()⊙ [DB笔试面试59]以下关于视图叙述不正确的是()⊙ [DB笔试面试57]下列关于SQ

试题库问题 2011-12-29

算法实现题8-7 试题库问题(习题 8-18) ´问题描述: 假设一个试题库中有n道试题. 每道试题都标明了所属类别.同一道题可能有多个类别属性.现要从题库中抽取 m 道题组成试卷.并要求试卷包含指定类型的试题.试设计一个满足要求的组卷算法. ´编程任务: 对于给定的组卷要求,计算满足要求的组卷方案. ´数据输入: 由文件input.txt提供输入数据. 文件第1行有2个正整数n和k (2 <=k<= 20, k<=n<= 1000) k 表示题库中试题类型总数,n 表示题库中试题

[网络流24题]试题库问题

Description 假设一个试题库中有$n$道试题,每道试题都标明了所属类别,同一道题可能有多个类别属性.现要从题库中抽取$m$道题组成试卷,并要求试卷包含指定类型的试题.求一个满足要求的组卷方案. Input 第$1$行有$2$个正整数$n,k,k$表示题库中试题类型总数,$n$表示题库中试题总数. 第$2$行有$k$个正整数,第$i$个正整数表示要选出的类型$i$的题数$a_i$.这$k$个数相加就是要选出的总题数$m$. 接下来的$n$行给出了题库中每个试题的类型信息.每行的第$1$个

试题库找答案小程序的开发

期末考试考完无聊在刷试题库.然后CY来我寝室,提醒我可以搞个自动在excel里找答案的程序,他给了思路之后就马上开始动工. 所谓带有我们学校特色的试题库就是下载excel表格,作业和考试题都在网页上,我们要在excel里找答案.把这个过程理了一下,程序主要实现以下功能: 1.监视剪贴板的变化: 2.如果检测到剪贴板内容变化,则根据剪贴板里的内容,利用正则表达式在试题库的题目里找到第一个最匹配的题目: 3.在控制台输出答案. 考虑到python有丰富的库,要连接到excel或者监视剪贴板不是什么麻

网络流 24题 试题库问题

试题库问题 题目描述 假设一个试题库中有n道试题.每道试题都标明了所属类别.同一道题可能有多个类别属性.现要从题库中抽取m道题组成试卷.并要求试卷包含指定类型的试题.试设计一个满足要求的组卷算法.具体的描述见输入格式. 输入格式 文件第1行有2个正整数k和n (2 <=k<= 20,k<=n<= 1000) k表示题库中试题类型总数,n表示题库中试题总数.第2行有k个正整数,第i个正整数表示要选出的类型i的题数.这k个数相加就是要选出的总题数m.接下来的n行给出了题库中每个试题的类