3996: [TJOI2015]线性代数

题意:给出一个NN的矩阵B和一个1N的矩阵C。求出一个1*N的01矩阵A.使得

\(D=(A * B-C)* A^T\)最大。其中A^T为A的转置。输出D。每个数非负。


分析一下这个乘法的性质或者化简一下容易发现,\(C_i\)代价生效需要\(A_i=1\),\(B_{ij}\)贡献生效需要\(A_i =A_j=1\)

最小割

我成功的把dinic里的括号打错了...gg

  1. #include <iostream>
  2. #include <cstdio>
  3. #include <cstring>
  4. #include <algorithm>
  5. using namespace std;
  6. typedef long long ll;
  7. const int N=300005, M=2e6+5, INF = 1e9;
  8. inline int read() {
  9. char c=getchar(); int x=0,f=1;
  10. while(c<'0' || c>'9') {if(c=='-')f=-1; c=getchar();}
  11. while(c>='0' && c<='9') {x=x*10+c-'0'; c=getchar();}
  12. return x*f;
  13. }
  14. int n, b[505][505], c[505], sum;
  15. struct edge{int v, ne, c, f;} e[M];
  16. int cnt=1, h[N], s, t;
  17. inline void ins(int u, int v, int c) {
  18. e[++cnt] = (edge){v, h[u], c, 0}; h[u] = cnt;
  19. e[++cnt] = (edge){u, h[v], 0, 0}; h[v] = cnt;
  20. }
  21. namespace mf {
  22. int q[N], head, tail, vis[N], d[N];
  23. bool bfs() {
  24. memset(vis, 0, sizeof(vis));
  25. head = tail = 1;
  26. q[tail++] = s; d[s] = 0; vis[s] = 1;
  27. while(head != tail) {
  28. int u = q[head++];
  29. for(int i=h[u];i;i=e[i].ne)
  30. if(!vis[e[i].v] && e[i].c > e[i].f) {
  31. int v = e[i].v;
  32. vis[v] = 1; d[v] = d[u]+1;
  33. q[tail++] = v;
  34. if(v == t) return true;
  35. }
  36. }
  37. return false;
  38. }
  39. int cur[N];
  40. int dfs(int u, int a) {
  41. if(u == t || a == 0) return a;
  42. int flow = 0, f;
  43. for(int &i=cur[u];i;i=e[i].ne)
  44. if(d[e[i].v] == d[u]+1 && (f = dfs(e[i].v, min(a, e[i].c - e[i].f)) ) >0 ) {
  45. flow += f;
  46. e[i].f += f;
  47. e[i^1].f -= f;
  48. a -= f;
  49. if(a == 0) break;
  50. }
  51. if(a) d[u] = -1;
  52. return flow;
  53. }
  54. int dinic() {
  55. int flow = 0;
  56. while(bfs()) {
  57. for(int i=s; i<=t; i++) cur[i] = h[i];
  58. flow += dfs(s, INF);
  59. }
  60. return flow;
  61. }
  62. }
  63. void build() {
  64. s = 0; t = n + n*n + 1;
  65. for(int i=1; i<=n; i++) ins(s, i, c[i]), ins(i, t, b[i][i]);
  66. for(int i=1; i<=n; i++) for(int j=1; j<=n; j++) if(i != j) {
  67. int id = i*n+j;
  68. ins(i, id, INF); ins(j, id, INF); ins(id, t, b[i][j]);
  69. }
  70. }
  71. int main() {
  72. freopen("in", "r", stdin);
  73. n = read();
  74. for(int i=1; i<=n; i++) for(int j=1; j<=n; j++) b[i][j] = read(), sum += b[i][j];
  75. for(int i=1; i<=n; i++) c[i] = read();
  76. build();
  77. int ans = mf::dinic();
  78. printf("%d\n", sum - ans);
  79. }

bzoj 3996: [TJOI2015]线性代数 [最小割]的更多相关文章

  1. bzoj 3996 [TJOI2015]线性代数——最小割

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3996 b[ i ][ j ] 要计入贡献,当且仅当 a[ i ] = 1 , a[ j ] ...

  2. bzoj 3996: [TJOI2015]线性代数【最小割】

    把转置矩阵看成逆矩阵吓傻了233 首先按照矩乘推一下式子: \[ D=\sum_{i=1}^n a[i]*(\sum_{j=1}^n a[j]*b[j][i])-c[i] \] \[ D=(\sum_ ...

  3. ●BZOJ 3996 [TJOI2015]线性代数

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=3996 题解: 好题啊.(不太熟悉矩阵相关,所以按某些博主的模型转换来理解的)首先,那个式子可 ...

  4. bzoj 3996: [TJOI2015]线性代数

    Description 给出一个N*N的矩阵B和一个1*N的矩阵C.求出一个1*N的01矩阵A.使得 D=(A*B-C)*A^T最大.其中A^T为A的转置.输出D Input 第一行输入一个整数N,接 ...

  5. [TJOI2015]线性代数(最小割)

    题目描述 给出一个N*N的矩阵B和一个1*N的矩阵C.求出一个1*N的01矩阵A.使得 D=(A*B-C)*A^T最大.其中A^T为A的转置.输出D 题解 观察上面那个式子发现,当一个bij有贡献时当 ...

  6. BZOJ3996[TJOI2015]线性代数——最小割

    题目描述 给出一个N*N的矩阵B和一个1*N的矩阵C.求出一个1*N的01矩阵A.使得 D=(A*B-C)*A^T最大.其中A^T为A的转置.输出D 输入 第一行输入一个整数N,接下来N行输入B矩阵, ...

  7. 【BZOJ 3996】 3996: [TJOI2015]线性代数 (最小割)

    3996: [TJOI2015]线性代数 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1368  Solved: 832 Description 给 ...

  8. 【BZOJ-3996】线性代数 最小割-最大流

    3996: [TJOI2015]线性代数 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1054  Solved: 684[Submit][Statu ...

  9. BZOJ 3996 线性代数 最小割

    题意: 给出一个N*N的矩阵B和一个1*N的矩阵C.求出一个1*N的01矩阵A.使得 D=(A*B-C)*A^T最大.其中A^T为A的转置.输出D 分析: 这道题比较绕,我们需要看清题目中那个式子的本 ...

随机推荐

  1. hdu_1754I Hate It(线段树)

    hdu_1754I Hate It(线段树) 标签: 线段树 题目链接 题意: 中文题意...不多说了,线段树基础题 直接上代码: #include<cstdio> #include< ...

  2. 浏览器的统一指针事件:Pointer Event

    在早期的浏览器,输入的事件其实相对单纯,只有考虑到鼠标和键盘两种:而当时的鼠标事件,其实就是 click.mousedown.mouseup 等等的事件.但是当手机.平板开始流行时候,再移动装置上的主 ...

  3. 如何在VS2017中使用快捷键格式化代码?

    1.同时按住Ctrl键+A键,全选代码或要格式化的部分代码: 2.再按住Ctrl键,接着按一下K键,接着按一下F键.(注意:Ctrl键在按后面这2个键的时候一直是按着的,直到F键按完才松开).也就是俗 ...

  4. 访问Google工具

    借助Google访问助手加速 下载地址: http://www.ggfwzs.com/

  5. [OpenCV][ARM9下移植OpenCV]

    [OpenCV][ARM9下移植OpenCV]   安装环境 宿主机: Red Hat Enterprise Linux Server 6.3 开发板: mini2440 相关软件: cmake-3. ...

  6. [学习OpenCV攻略][013][Mat - 基本图像容器]

    Mat 是一个类,由两个数据部分组成:矩阵头(包含矩阵尺寸,存储方法,存储地址等信息)和一个指向存储所有像素值的矩阵(根据所选存储方法的不同矩阵可以是不同的维数)的指针. 矩阵属于多个 Mat 对象, ...

  7. 把自己的js模块兼容到AMD CMD CommonJS

    为了让同一个模块可以运行在前后端,在写作过程中需要考虑兼容前端也实现了模块规范的环境.为了保持前后端的一致性,类库开发者需要将类库代码包装在一个闭包内.以下代码演示如何将hello()方法定义到不同的 ...

  8. python 模块中的 __init__.py __main__.py

    python中文件夹想作为一个模块被引用,则在文件夹内必须要包含 __init__.py 文件,即使此文件为空. 如果此模块想要运行则必须要包含 __main__.py 文件.接下来说下两个文件起到的 ...

  9. 【问题解决】java中为什么不建议使用DataInputStream 的readLine()方法

    常用方法 int read(byte[] b) 从包含的输入流中读取一定数量的字节,并将它们存储到缓冲区数组 b 中. int read(byte[] b, int off, int len) 从包含 ...

  10. Linux文件

    Linux文件类型 对于内核而言,所有打开的文件都是通过文件描述符引用(FD),文件描述符是一个非负整数,当打开现有问价或创建一个新文件时,内核向进程返回一个文件描述符. 按照惯例,shell把文件描 ...