题目传送门:http://codeforces.com/problemset/problem/1038/E

题意:给出$N$个方块,每个方块有左右两种颜色$a,b$(可以翻转使左右两种颜色交换)和一个权值$w$。当某个方块的右侧颜色与另一个方块的左侧颜色相同,它们可以连成一个大块,一个大块可以由若干个小块像这样连成(一个大块中可以只包含一个小块)。定义大块的权值为组成它的所有小块的权值和,问可以连成的大块中最大的权值。$N \leq 100 , a , b \leq 4 , w \leq 10^5$


看到$a,b \leq 4$就是神题预定

方法一:搜索

将四种颜色看做点,方块是连接左右两种颜色的边,就可以得到一个边为$100$,点为$4$的图,目标是找到一个边权和最大的简单路。然后我们可以发现图上很多的边是没有意义的。比如:

①某个方块左右两侧同色(在图上表示自环),这样子的边权直接算作点权即可。

②很多方块左右颜色相同(在图上表示重边),因为这些过多的重边可以通过在两个端点不断绕环,所以在搜索的时候经过它们很多次是没有必要的,所以考虑根据奇偶性将边数大于$3$的重边压成边数为$3$或者$2$(考试的时候想的是$2$或$1$但是考虑边为奇数的也有可能最后丢掉一条边到另外的颜色,所以必须留下$2$或$3$条),其中$2$或$1$条为原来最短边和次短边/最短边,而剩下的一条边则为其他重边压成的边,边权为它们的和。

经过这两种处理,图上的边的数量变为了最多$18$条,搜索可以承受(而且跑得贼快)。

 #include<bits/stdc++.h>
 using namespace std;

 vector < ][];
 ] , cnt[][] , N , ans;

 bool cmp(int a , int b){
     return a > b;
 }

 void dfs(int now , int cnt){
     int k = pri[now];
     cnt += pri[now];
     pri[now] = ;
      ; i <=  ; i++)
         if(!Edge[now][i].empty()){
             ];
             Edge[now][i].erase(Edge[now][i].begin());
             Edge[i][now].erase(Edge[i][now].begin());
             dfs(i , cnt + t);
             Edge[now][i].insert(Edge[now][i].begin() , t);
             Edge[i][now].insert(Edge[i][now].begin() , t);
         }
     ans = max(ans , cnt);
     pri[now] += k;
 }

 int main(){
     cin >> N;
      ; i <= N ; i++){
         int a , b , c;
         cin >> a >> b >> c;
         swap(b , c);
         if(a == b)
             pri[a] += c;
         else{
             Edge[a][b].push_back(c);
             Edge[b][a].push_back(c);
             cnt[a][b]++;
             cnt[b][a]++;
         }
     }
      ; i <=  ; i++)
          ; j <=  ; j++){
             sort(Edge[i][j].begin() , Edge[i][j].end() , cmp);
             )
                 ){
                      , p = Edge[i][j][cnt[i][j] - ] , q = Edge[i][j][cnt[i][j] - ];
                      ; k < cnt[i][j] -  ; k++)
                         sum += Edge[i][j][k];
                     Edge[i][j].clear();
                     Edge[i][j].push_back(sum);
                     Edge[i][j].push_back(p);
                     Edge[i][j].push_back(q);
                 }
                 else{
                      , p = Edge[i][j][cnt[i][j] -];
                      ; k < cnt[i][j] -  ; k++)
                         sum += Edge[i][j][k];
                     Edge[i][j].clear();
                     Edge[i][j].push_back(sum);
                     Edge[i][j].push_back(p);
                 }
         }
      ; i <=  ; i++)
         dfs(i , );
     cout << ans;
     ;
 }

方法二:DP

设$f_{i,j,k,l}$表示选择$i$到$j$段的方块,大块左端颜色为$k$,右端颜色为$r$的时候获得的最大权值,考虑转移:

①由其中两段拼接而成:$f_{i,p,k,q}+f_{p+1,j,q,l}$

②由其中两段换位拼接而成(因为本身方块的拼接是无序的):$f_{i,p,q,l}+f_{p+1,j,k,q}$

③只取左边一部分或者右边一部分:$f_{i,p,k,l}$或$f_{p,j,k,l}$

注意非法转移情况要用极小值覆盖,所以才有③转移才会产生最优值的情况。

 #include<bits/stdc++.h>
 using namespace std;

 ][][][] , l[] , r[] , w[];

 int main(){
     int N;
     cin >> N;
      ; i <= N ; i++)
         cin >> l[i] >> w[i] >> r[i];
     memset(dp , -0x3f , sizeof(dp));
      ; i <= N ; i++)
         dp[i][i][l[i]][r[i]] = dp[i][i][r[i]][l[i]] = w[i];
      ; i ; i--)
          ; j <= N ; j++)
              ; p <=  ; p++)
                  ; q <=  ; q++)
                     for(int k = i ; k < j ; k++){
                         dp[i][j][p][q] = max(dp[i][j][p][q] , max(dp[i][k][p][q] , dp[k + ][j][p][q]));
                          ; l <=  ; l++)
                             dp[i][j][p][q] = max(dp[i][j][p][q] , max(dp[i][k][p][l] + dp[k + ][j][l][q] , dp[i][k][l][q] + dp[k + ][j][p][l]));
                     }
     ;
      ; i <=  ; i++)
          ; j <=  ; j++)
             all = max(all , dp[][N][i][j]);
     cout << all;
     ;
 }

CF1038E Maximum Matching 搜索/区间DP的更多相关文章

  1. lightoj 1283 - Shelving Books(记忆化搜索+区间dp)

    题目链接:http://www.lightoj.com/volume_showproblem.php?problem=1283 题解:这题很显然一看就像是区间dp,但是单纯的区间dp好像解决不了问题可 ...

  2. 洛谷P1514 引水入城 [搜索,区间DP]

    题目传送门 引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个 N 行×M 列的矩形,如上图所示,其中每个格子都代表一座城市,每 ...

  3. poj1179 区间dp(记忆化搜索写法)有巨坑!

    http://poj.org/problem?id=1179 Description Polygon is a game for one player that starts on a polygon ...

  4. UVA 10003 Cutting Sticks 区间DP+记忆化搜索

    UVA 10003 Cutting Sticks+区间DP 纵有疾风起 题目大意 有一个长为L的木棍,木棍中间有n个切点.每次切割的费用为当前木棍的长度.求切割木棍的最小费用 输入输出 第一行是木棍的 ...

  5. HDU 4597 Play Game(区间DP(记忆化搜索))

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4597 题目大意: 有两行卡片,每个卡片都有各自的权值. 两个人轮流取卡片,每次只能从任一行的左端或右端 ...

  6. CF149D. Coloring Brackets[区间DP !]

    题意:给括号匹配涂色,红色蓝色或不涂,要求见原题,求方案数 区间DP 用栈先处理匹配 f[i][j][0/1/2][0/1/2]表示i到ji涂色和j涂色的方案数 l和r匹配的话,转移到(l+1,r-1 ...

  7. BZOJ1055: [HAOI2008]玩具取名[区间DP]

    1055: [HAOI2008]玩具取名 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1588  Solved: 925[Submit][Statu ...

  8. HDU 4293---Groups(区间DP)

    题目链接 http://acm.split.hdu.edu.cn/showproblem.php?pid=4293 Problem Description After the regional con ...

  9. Light OJ 1031---Easy Game(区间DP)

    题目链接 http://lightoj.com/volume_showproblem.php?problem=1031 Description You are playing a two player ...

随机推荐

  1. FI / CO 配置步骤清单

    一.FI配置 01. 创建公司代码:企业结构-定义-财务会计-编辑/复制/删除/检查公司代码.编辑公司OX02 02. 创建会计科目表 OB13 03. 定义会计年度变式 OB29 04. 创建信贷控 ...

  2. Mobile First! Wijmo 5 之 架构

    CSDN移动开发频道率先报道的<Mobile First!jQuery UI组件集Wijmo五年最大更新>引起开发者极大关注. 本文就开发者关心的话题之一架构,展开叙述. Wijmo 5是 ...

  3. plt 数据可视化

    1.plt.plot(x,y,color) 折线坐标图 import matplotlib.pyplot as plt h = np.linspace(1, 10, 10) v = np.linspa ...

  4. WANem广域网环境模拟

    背景 在测试过程中,往往需要模拟网络环境较差情况下,体验情况,故引入广域网模拟工具辅助测试 安装WANem 步骤1:光盘引导WANem镜像: 步骤2:启动(或者新建虚拟机——选择other insal ...

  5. 浅谈Redis之慢查询日志

    首先我们需要知道redis的慢查询日志有什么用?日常在使用redis的时候为什么要用慢查询日志? 第一个问题: 慢查询日志是为了记录执行时间超过给定时长的redis命令请求 第二个问题: 让使用者更好 ...

  6. mysql8 :客户端连接caching-sha2-password问题

    在安装mysql8的时候如果选择了密码加密,之后用客户端连接比如navicate,会提示客户端连接caching-sha2-password,是由于客户端不支持这种插件,可以通过如下方式进行修改: # ...

  7. Java中常用的字节流和字符流

    IO流(输入流.输出流) 字节流.字符流 1.字节流: InputStream.OutputStream InputStream抽象了应用程序读取数据的方式: OutputStream抽象了应用程序写 ...

  8. [SequenceFile_2] SequenceFile 的基本操作

    0. 说明 测试序列文件的读写操作 && 测试序列文件的排序操作 && 测试序列文件的合并操作 && 测试序列文件的压缩方式 && 测试 ...

  9. 使用LogPhoneUtil工具类在Android手机保存APP运行日志

    最近公司的测试老是提出这样那样的bug,当然也怪自己代码写的烂,所以测试总是会把app搞崩溃,而他们那边崩溃的时候还没有日志打印,自己回来再重现有的时候还真不好复现出来,因此麻烦事就来了.为了方便查看 ...

  10. Html body的滚动条禁止与启用

    在写一个在页面中,经验证用户没有登录或session失效时候弹出登录框禁止页面滚动用到今天搞了一个功能,上下左右居中,模仿QQ空间里的样式,把横向和纵向滚动条禁止掉代码如下:<script ty ...