前几天给舍友讲这题的时候感觉挺有意思的,就贴上来吧...

题目链接:1038E - Maximum Matching

题目大意:有\(n\)个棒子,每个条两端有颜色\(c1,c2\)以及他的价值\(v\),要求选取若干个棒子拼接起来(要求连接处的颜色相同,棒子可以反转),求最大价值总和。

题解:设\(c1==c2\)的为同色棒子,反之为异色

   可以发现偶数个异色棒子可以拼为一个长长的同色棒子,奇数个则可以拼为一个长长的异色棒子,因此可以预处理\(F[i][j]\)表示若将所有\((i,j)\)当做异色棒子来用所能得到的价值,\(G[i][j]\)表示当做同色棒子来用所能得到的价值。显然他们的值为所有颜色为\((i,j)\)的价值总和\(-k\cdot\)颜色为\((i,j)\)的最小价值,k等于0或1,由当前颜色的数目决定。

   对于同色的棒子,可以发现,若当前已经拼好的棒子里颜色\(c\)出现过,则可以将所有颜色为\(c\)的同色棒子塞进去以增加答案。

   接下去就是直接暴力DFS了,枚举起点的颜色,然后暴力考虑可以接哪些颜色,每次进入下一层的时候更新一次答案。更新答案就按照上面所说的那样,判断哪些颜色出现过,并将所有包含此颜色的异色棒子当做同色棒子来用塞进去。这里要注意一点,虽然塞进去的时候是当做同色来塞,但这种操作可能会导致新出现了一种颜色,而新颜色的出现可能还会导致答案的增加(即有包含新颜色的棒子)。因此这样子的操作还要再进行若干次,3次是肯定够的。最后再把所有已出现颜色的同色棒子加入就好了。

   由于颜色组合只有10种((1,2)和(2,1)算一种),因此DFS的复杂度不会超过\(O(4^{10})\),再乘上update的复杂度则为\(O(4^{13})\),水过去是没有压力的

代码中将颜色对\((c1,c2)\)进行了标号处理

#include<bits/stdc++.h>
using namespace std;
#define N 101
int n,s[N],cnt[N],m[N],F[N],G[N],c1,c2,v,ans;
bool x[][],f[];
void update()
{
bool g[],X[][];
for(int i=;i<=;i++)g[i]=f[i];
for(int i=;i<=;i++)
for(int j=i+;j<=;j++)
X[i][j]=x[i][j];
int res=;
for(int i=;i<=;i++)
for(int j=i+;j<=;j++)
if(x[i][j])res+=F[i*+j];
for(int _=;_<;_++)
for(int i=;i<=;i++)
for(int j=i+;j<=;j++)
if(!X[i][j] && (g[i]||g[j]) && cnt[i*+j]>)
res+=G[i*+j],X[i][j]=true,g[i]=true,g[j]=true;
for(int i=;i<=;i++)if(g[i])res+=s[i];
ans=max(ans,res);
}
int get(int i,int j){if(i>j)swap(i,j);return i*+j;}
void dfs(int i)
{
update();
for(int j=;j<=;j++)
if(!x[i][j] && cnt[get(i,j)])
x[i][j]=x[j][i]=f[j]=true,dfs(j),x[i][j]=x[j][i]=f[j]=false;
}
int main()
{
memset(m,0x3f,sizeof(m));
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%d%d%d",&c1,&v,&c2);
if(c1==c2){s[c1]+=v;continue;}
if(c1>c2)swap(c1,c2);
int id=c1*+c2;
cnt[id]++,m[id]=min(m[id],v),s[id]+=v;
}
for(int i=;i<=;i++)
for(int j=i+;j<=;j++)
{
int id=i*+j;
if(cnt[id])F[id]=cnt[id]&?s[id]:(s[id]-m[id]);
G[id]=cnt[id]&?(s[id]-m[id]):s[id];
}
for(int i=;i<=;i++)
f[i]=true,dfs(i),f[i]=false;
return printf("%d\n",ans),;
}

[Codeforces Round #508 (Div. 2)][Codeforces 1038E. Maximum Matching]的更多相关文章

  1. Codeforces Round #508 (Div. 2)

    Codeforces Round #508 (Div. 2) http://codeforces.com/contest/1038 A #include<bits/stdc++.h> us ...

  2. Codeforces Round #508 (Div. 2) E. Maximum Matching(欧拉路径)

     E. Maximum Matching 题目链接:https://codeforces.com/contest/1038/problem/E 题意: 给出n个项链,每条项链左边和右边都有一种颜色(范 ...

  3. Codeforces Round #524 (Div. 2) codeforces 1080A~1080F

    目录 codeforces1080A codeforces 1080B codeforces 1080C codeforces 1080D codeforces 1080E codeforces 10 ...

  4. Codeforces Round #508 (Div. 2) D. Slime

    D. Slime 题目链接:https://codeforces.com/contest/1038/problem/D 题意: 给出两个数,然后每次可以对相邻的两个数合并,比如x,y,那么合并过后就是 ...

  5. Codeforces Round #704 (Div. 2), problem: (C) Maximum width还是要多学习

    Problem - C - Codeforces 看清题目要求, 最重要部分在第二段. 大佬最后给出的代码果然简单, 思路简单化, 未必非要把答案在一个大括号里全部完成, 两个指针同时跑,中间加了一堆 ...

  6. Codeforces Round #508 (Div. 2) C D

    C: C - Gambling 给你两个数列  每一回合A可以选择从第一个序列里面选一个数或者清除第二个序列里面选一个数 同理B能从第二序列里面选数或者清除第一个序列里面一个数 然后 求A所选的数之和 ...

  7. 题解——Codeforces Round #508 (Div. 2) T3 (贪心)

    贪心的选取最优解 然后相减好 记得要开long long #include <cstdio> #include <algorithm> #include <cstring ...

  8. 题解——Codeforces Round #508 (Div. 2) T2 (构造)

    按照题意构造集合即可 注意无解情况的判断 #include <cstdio> #include <algorithm> #include <cstring> #in ...

  9. 题解——Codeforces Round #508 (Div. 2) T1 (模拟)

    依照题意暴力模拟即可A掉 #include <cstdio> #include <algorithm> #include <cstring> #include &l ...

随机推荐

  1. EffectiveC++ 第1章 让自己习惯C++

    我根据自己的理解,对原文的精华部分进行了提炼,并在一些难以理解的地方加上了自己的"可能比较准确"的「翻译」. Chapter 1 让自己习惯C++ 条款 1 :视 C++为一个语言 ...

  2. Leetcode#344. Reverse String(反转字符串)

    题目描述 编写一个函数,其作用是将输入的字符串反转过来. 示例 1: 输入: "hello" 输出: "olleh" 示例 2: 输入: "A man ...

  3. windows下使用git和github建立远程仓库

    转自(http://www.bubuko.com/infodetail-430228.html) 从昨天开始就在看git的使用,因为在Windows下很多命令行操作都比较坑爹,但是今天再走了无数弯路之 ...

  4. EHCache:Eelment刷新后,timeToLiveSeconds失效了?

    个人以为只要设定了timeToLiveSeconds,中间过程不管有没有访问,只要LiveSeconds时间到了,缓存就会失效.但是开发时发现并非如此,经过一番折腾,最终发现自己的理解是正确的,还是使 ...

  5. 【原创】Java基础之常用JVM工具

    查看当前所有java进程 # jps 查看某个进程的堆内存占用情况 # jmap -heap $pid 查看某个进程的堆内存中对象分布情况 # jmap -histo $pid 将某个进程的堆内存导出 ...

  6. MongoDB在CentOS上的安装和配置

    1. 创建mongodb-org-4.0.repo文件,并放入/etc/yum.repos.d目录下,repo文件内容如下 [mongodb-org-4.0] name=MongoDB Reposit ...

  7. eclipse解决maven编码UTF-8的不可映射字符

    1.同时指定<project.build.sourceEncoding>属性和<encoding>的方式可适用于Maven2和Maven3. 2.在Maven3中可以只增加&l ...

  8. 有道云笔记Markdown上传本地图片的方法

    有道云笔记截图&保存   方法有多种,例如:开通有道云笔记VIP会员.先将图片文件上传到有道云笔记后使用图片的分享链接.说到底还是使用的 Markdown 的图片功能 ![图片名称](图片链接 ...

  9. 利用jQuery如何获取当前被点击的按钮

    如下代码 <tr> <td><a href="javascript:void(0)">点我1</a></td> < ...

  10. 开发快捷键(Eclipse,STS)

    Eclipse 常用快捷键 Eclipse的编辑功能非常强大,掌握了Eclipse快捷键功能,能够大大提高开发效率.Eclipse中有如下一些和编辑相关的快捷键. 1. [ALT+/]   此快捷键为 ...