概览:

T1 枚举 T2 状压DP T3 DP


TopCoder SRM 667 Div.2 T1

解题思路

由于数据范围很小,所以直接枚举所有点,判断是否可行。时间复杂度O(δX × δY),空间复杂度O(1)。

参考程序段

#include <bits/stdc++.h>
using namespace std; class PointDistance {
public:
vector <int> findPoint( int x1, int y1, int x2, int y2 );
};
int sqr(int x){ return x * x; }
vector<int> ans;
vector <int> PointDistance::findPoint(int x1, int y1, int x2, int y2) {
ans.clear();
for(int x = -100; x <= 100; x++)
for(int y = -100; y <= 100; y++)
if(sqr(x1 - x) + sqr(y1 - y) > sqr(x2 - x) + sqr(y2 - y)){
ans.push_back(x);
ans.push_back(y);
return ans;
}
return ans;
}

TopCoder SRM 667 Div.2 T2

解题思路

可能大家的第一反应都是每一步选花费最小来贪心,但是发现,某一步有多个花费最小时,会出问题。

我们看一组数据

0000

1100

0011

0111

如果贪心, 可能顺序是1,2,3,4, 这样答案为8, 而事实上按顺序1,3,4,2,最优,为6。

于是我们考虑DP。

首先,我们令dp[i]表示在i的二进制表示下为1的那些指令处理完毕所需的最小花费。则可以从任何二进制表示下比i少一个1的j转移过来,我们取其最小。开始状态为dp[0] = 0,结束状态为dp[(1 << n) - 1]。

参考程序段

程序中,并非枚举dp[j]向dp[i]转移,而是对于dp[j]向每个可能的dp[i]转移,并使用了类似于spfa的写法。

同时,程序中q数组保存的数指向dp数组的下标,q_used对应花费。

#include <bits/stdc++.h>
using namespace std; class OrderOfOperationsDiv2 {
public:
int minTime( vector <string> s );
};
int n;
int rec[20];
int l, r, q[1 << 23], q_used[1 << 23];
int dp[1 << 21];
int len;
int OrderOfOperationsDiv2::minTime(vector <string> s) {
n = s.size();
len = s[0].size();
for(int i = 0; i < n; i++){
rec[i] = 0;
for(int j = 0; j < len; j++)
if(s[i][j] == '1') rec[i] |= (1 << j);//转成二进制方便处理
}
memset(dp, 255, sizeof(dp));
dp[0] = 0;
l = 0; r = 1; q[1] = 0; q_used[1] = 0;
while(l < r){
l++;
int rec_vis = q[l];
int rec_used = q_used[l];
for(int i = 0; i < n; i++){//枚举可能被更新的状态
if((rec_vis >> i) & 1) continue;
int x = rec_vis | (1 << i);
int y = rec_used | rec[i];
int tt = 0;
for(int j = 0; j < len; j++)
if(((y >> j) & 1) == 1 && ((rec_used >> j) & 1) == 0) tt++;
tt = tt * tt;
if(dp[x] == -1 || dp[x] > dp[rec_vis] + tt){
dp[x] = dp[rec_vis] + tt;
r++;
q[r] = x;
q_used[r] = y;
}
}
}
return dp[(1 << n) - 1];
}

TopCoder SRM 667 Div.2 T3

解题思路

乍一看好像很麻烦呐。看起来有后效性?仿佛不能用dp?唉?然而数据只有30?所以考虑以下乱搞?

emmm

我们还是dp,大不了扩大维数,增加状态数。嗯。我们令dp[i][j][k]表示前i幢楼,在第i幢楼这里开了j家店,同时计划在下一幢楼开k家能获得的利益。那么我们可以得到状态方程

\[dp[i][j][k] = max(dp[i][j][k], dp[i - 1][l][j] + j * c[i * 3 * m + (j + k + l) - 1])(0 <= l <= m)
\]

特判一下第一栋楼和最后一栋楼。

同时注意下小细节

1、题目里并没有给出c[-1]的定义,所以任意相邻三栋楼中一定要开一家:)

2、只有一栋楼的情况:)

时间复杂度O(n^4), 空间复杂度O(n^3)【这个我觉得还是叫暴力比较好:捂脸:】

参考代码段

#include <bits/stdc++.h>
using namespace std; class ShopPositions {
public:
int maxProfit( int n, int m, vector <int> c );
};
int f[40][40][40];
int ShopPositions::maxProfit(int n, int m, vector <int> c) {
memset(f, 0, sizeof(f));
int ans = 0;
for(int j = 0; j <= m; j++)//特判第一栋楼
for(int k = 0; k <= m; k++){
if(j + k == 0) continue;
f[0][j][k] = max(f[0][j][k], j * c[0 * 3 * m + (j + k) - 1]);
ans = max(ans, f[0][j][k]);
}
for(int i = 1; i < n - 1; i++)
for(int j = 0; j <= m; j++)
for(int k = 0; k <= m; k++)//O(n^4)DP
for(int l = 0; l <= m; l++){
if(j + k + l == 0) continue;
f[i][j][k] = max(f[i][j][k], f[i - 1][l][j] + j * c[i * 3 * m + (j + k + l) - 1]);
ans = max(ans, f[i][j][k]);
}
if(n > 1)//如果不止一栋楼,就特判最后一栋楼
for(int j = 0; j <= m; j++)
for(int l = 0; l <= m; l++){
if(j + l == 0) continue;
f[n - 1][j][0] = max(f[n - 1][j][0], f[n - 2][l][j] + j * c[(n - 1) * 3 * m + (j + l) - 1]);
ans = max(ans, f[n - 1][j][0]);
}
return ans;
}

TopCoder SRM 667 Div.2题解的更多相关文章

  1. Topcoder SRM 674 Div.2题解

    T1 解题思路 这题应该不是很难,主要是题意理解问题. 注意给出的两个数组里映射关系已经对应好了,只要判断是否为双射即可 参考程序 #include <bits/stdc++.h> usi ...

  2. TopCoder SRM 560 Div 1 - Problem 1000 BoundedOptimization & Codeforces 839 E

    传送门:https://284914869.github.io/AEoj/560.html 题目简述: 定义"项"为两个不同变量相乘. 求一个由多个不同"项"相 ...

  3. [topcoder]SRM 633 DIV 2

    第一题,http://community.topcoder.com/stat?c=problem_statement&pm=13462&rd=16076 模拟就可以了. #includ ...

  4. TopCoder SRM 596 DIV 1 250

    body { font-family: Monospaced; font-size: 12pt } pre { font-family: Monospaced; font-size: 12pt } P ...

  5. [topcoder]SRM 646 DIV 2

    第一题:K等于1或者2,非常简单.略.K更多的情况,http://www.cnblogs.com/lautsie/p/4242975.html,值得思考. 第二题:http://www.cnblogs ...

  6. 【topcoder SRM 702 DIV 2 250】TestTaking

    Problem Statement Recently, Alice had to take a test. The test consisted of a sequence of true/false ...

  7. Topcoder SRM 656 (Div.1) 250 RandomPancakeStack - 概率+记忆化搜索

    最近连续三次TC爆零了,,,我的心好痛. 不知怎么想的,这题把题意理解成,第一次选择j,第二次选择i后,只能从1~i-1.i+1~j找,其实还可以从j+1~n中找,只要没有被选中过就行... [题意] ...

  8. Topcoder SRM 648 (div.2)

    第一次做TC全部通过,截图纪念一下. 终于蓝了一次,也是TC上第一次变成蓝名,下次就要做Div.1了,希望div1不要挂零..._(:зゝ∠)_ A. KitayutaMart2 万年不变的水题. # ...

  9. Topcoder SRM 145 DIV 1

    Bonuses 模拟 题意:给你一个序列,按照百分比排序,再将百分比取整,再把剩余百分比加到最大的那几个. 题解:按照题意模拟就好.

随机推荐

  1. 面试40-一个数组,有2个数字出现奇数次,其余都是偶数次,求这两个数字O(n) O(1)

    #include<iostream> using namespace std; // 题目:数组中只有不多于两个数字出现次数是奇数次,其他都是偶数次,求出出现奇数次的数字(不含0的数组) ...

  2. Ubuntu基本操作(博主想上传图片给服务器的一些命令)

    1.将当前目录下的文件移动至指定文件夹,这里用移动至网站的根目录做示范 sudo mv bamboo.jpg /val/www/html mv bamboo.jpg /val/www/html 2.进 ...

  3. PHP 补零操作

    str_pad(string,length,pad_string,pad_type)//参数 描述string //必需.规定要填充的字符串.length //必需.规定新的字符串长度.如果该值小于字 ...

  4. PermissionError: [Errno 13] Permission denied: '/run/user/0/jupyter'

    解决办法:需要给/run/user整个目录开放权限,不能单独给'/run/user/0/jupyter'这个文件,因为jupyter是需要往目录中添加文件,/run/user/0/jupyter,此时 ...

  5. Codeforces 1228D. Complete Tripartite

    传送门 不妨设 $1$ 号点在集合 $1$ 里 那么对于其他点,有且只有所有和 $1$ 没有边的点都在集合 $1$ 里 考虑不在集合 $1$ 的任意一个点 $x$ ,不妨设它在集合 $2$ 里 那么所 ...

  6. luogu P3320 [SDOI2015]寻宝游戏

    大意:给定树, 要求维护一个集合, 支持增删点, 询问从集合中任取一点作为起点, 遍历完其他点后原路返回的最短长度. 集合中的点按$dfs$序排列后, 最短距离就为$dis(s_1,s_2)+...+ ...

  7. 配置Mysql远程连接

    一.赋予某个用户权限 1.赋予权限格式:grant 权限 on 数据库对象 to 用户@IP(或者相应正则) 注:可以赋予select,delete,update,insert,index等权限精确到 ...

  8. Java web验证码——kaptcha的使用

    一.配置kaptcha的jar包 pom.xml配置: <-- 目前只有2.3.2版本--> <!-- https://mvnrepository.com/artifact/com. ...

  9. 剑指offer-数字在排序数组中出现的次数-数组-python

    题目描述 统计一个数字在排序数组中出现的次数.   python 内置函数 count()一行就能搞定   解题思路 二分查找到给定的数字及其坐标.以该坐标为中点,向前向后找到这个数字的 始 – 终 ...

  10. 多线程学习-- part 1 Thread

    一.Thread的使用 (1)sleep:进程等一会 (2)join:让并发处理变成串行 (3)start:启动线程的唯一方法,start()首先为线程分配必须的系统资源,调度线程运行并执行线程的ru ...