【题解】试题库问题

试题库问题

题目传送门:洛谷 P2763 试题库问题

题目描述

问题描述:

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

编程任务:

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

输入格式

第一行有两个正整数 \(k\)\(n\)\(k\) 表示题库中试题类型总数,\(n\) 表示题库中试题总数。

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

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

输出格式

输出共 \(k\) 行,第 \(i\) 行输出 i: 后接类型 \(i\) 的题号。
如果有多个满足要求的方案,只要输出一个方案。
如果问题无解,则输出No Solution!

样例 #1

样例输入 #1

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

样例输出 #1

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

提示

\(2\leq k \leq 20\)\(k \leq n \leq 10^3\)


思路

读完题,很明显的发现这是一个二分图

类型和题目是两个互不相交的集合,并且图中的每条边(i,j)所关联的两个顶点i和j分别属于这两个不同的顶点集

很明显的二分图+最大流

image

从源点到每个类型建边,容量为每个类型所需要的题目数量

从每个类型向这个类型里面的题目建边,容量为 1

从每个题目到汇点建边,容量为 1

跑一遍dinic检验答案

代码实现

#include<cstdio> 
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
int read()
{
	int x=0,f=1;char c=getchar();
	while (c<'0' || c>'9'){if (c=='-')f=-1;c=getchar();}
	while (c>='0'&&c<='9'){x=(x<<3)+(x<<1)+c-48;c=getchar();}
	return x*f;
}
const int inf=1e9,N=2000,M=300010;
int n,m,S,T;
int e[M],ne[M],f[M];
int q[N],h[N],cur[N],d[N],idx;

void add(int a,int b,int c)
{
    e[idx]=b,f[idx]=c,ne[idx]=h[a],h[a]=idx++;
    e[idx]=a,f[idx]=0,ne[idx]=h[b],h[b]=idx++;
}
bool bfs()
{
    int tt=0,hh=0;
    memset(d,-1,sizeof d);
    q[0]=S,d[S]=0,cur[S]=h[S];
    
    while(hh<=tt)
    {
        int t=q[hh++];
        for(int i=h[t];~i;i=ne[i])
        {
            int ver=e[i];
            if(d[ver]==-1&&f[i])
            {
                d[ver]=d[t]+1;
                cur[ver]=h[ver];
                if(ver==T) return true;
                q[++tt]=ver;
            }
        }
    }
    return false;
}

int find(int u,int limit)
{
    if(u==T) return limit;
    int flow=0;
    
    for(int i=cur[u];~i&&flow<limit;i=ne[i])
    {
        int ver=e[i];
        cur[u]=i;
        if(d[ver]==d[u]+1&&f[i])
        {
            int t=find(ver,min(f[i],limit-flow));
            if(!t) d[ver]=-1;
            f[i]-=t;
            f[i^1]+=t;
            flow+=t;
        }
    }
    
    return flow;
}
int dinic()
{
    int r=0,t;
    while(bfs())
    {
        while(t=find(S,inf)) r+=t;
    }
    return r;
}

int main()
{
    n=read(),m=read();
    S=0,T=n+m+1;
    int tot=0;
    memset(h,-1,sizeof h);
	for(int i=1;i<=n;i++)
	{
	    int x;
	    x=read();
	    add(S,i,x);
	    tot+=x;
	}
	for(int i=1;i<=m;i++)add(i+n,T,1);
	for(int i=1;i<=m;i++)
	{
	    int x,y;
	    x=read();
	    for(int j=1;j<=x;j++)
	    {
	        y=read();
	        add(y,i+n,1);
	    }
	    
	}
	if(tot!=dinic()) puts("No Solution!");
	else
	{
	    for(int i=1;i<=n;i++)
	    {
	        printf("%d :",i);
	        
	        for(int j=h[i];~j;j=ne[j])
	        {
	            if(e[j]>n&&e[j]<=n+m&&!f[j])
	            {
	                printf("%d ",e[j]-n);
	            }
	        }puts("");
	    }
	}
    
	return 0;
}
posted @ 2022-07-11 19:56  watasky  阅读(164)  评论(0编辑  收藏  举报