2022-12-30:某天小美进入了一个迷宫探险,根据地图所示,这个迷宫里有无数个房间
序号分别为1、2、3、…入口房间的序号为1
任意序号为正整数x的房间,都与序号 2x 和 2x + 1 的房间之间各有一条路径
但是这些路径是单向的,即只能从序号为x的房间去到序号为 2x 或 2x+1 的房间
而不能从 2x 或 2x+1 的房间去到序号为x的房间
在任何时刻小美都可以选择结束探险并离开迷宫,但是离开之后将无法再次进入迷宫
小美还提前了解了迷宫中宝藏的信息
已知宝藏共有n个,其中第i个宝藏在序号为pi的房间,价值为wi
且一个房间中可能有多个宝藏
小美为了得到更多的宝藏,需要精心规划路线,她找到你帮忙
想请你帮她计算一下,能获得的宝藏价值和最大值为多少
第一行一个正整数n,表示宝藏数量。
第二行为n个正整数p1, p2,… pn,其中pi表示第 i 个宝藏在序号为pi的房间。
第三行为n个正整数w1, w2,… wn,其中wi表示第i个宝藏的价值为wi。
1 <= n <= 40000, 1 <= pi < 2^30, 1 <= wi <= 10^6。
来自美团。

答案2022-12-30:

动态规划,利用图来优化枚举。
时间复杂度:O(N)。
空间复杂度:O(N)。

代码用rust编写。代码如下:

use rand::Rng;
use std::collections::HashMap;
use std::iter::repeat;
fn main() {
let nn: i32 = 100;
let pp: i32 = 5000;
let ww: i32 = 5000;
let test_time: i32 = 5000;
println!("测试开始");
for i in 0..test_time {
let n: i32 = rand::thread_rng().gen_range(0, nn) + 1;
let mut p = random_array(n, pp);
let mut w = random_array(n, ww);
let ans1 = max_money1(n, &mut p, &mut w);
let ans2 = max_money2(n, &mut p, &mut w);
if ans1 != ans2 {
println!("出错了!");
println!("i = {}", i);
println!("ans1 = {:?}", ans1);
println!("ans2 = {:?}", ans2);
break;
}
}
println!("测试结束");
} // 为了测试
// 普通动态规划
fn max_money1(n: i32, p: &mut Vec<i32>, w: &mut Vec<i32>) -> i32 {
let mut rooms: Vec<Vec<i32>> = repeat(repeat(0).take(2).collect())
.take(n as usize)
.collect();
for i in 0..n {
rooms[i as usize][0] = p[i as usize];
rooms[i as usize][1] = w[i as usize];
}
rooms.sort_by(|a, b| a[0].cmp(&b[0]));
let mut ans = 0;
let mut dp: Vec<i32> = repeat(-1).take(n as usize).collect();
for i in 0..n {
ans = get_max(ans, process1(i, &mut rooms, n, &mut dp));
}
return ans;
} fn get_max<T: Clone + Copy + std::cmp::PartialOrd>(a: T, b: T) -> T {
if a > b {
a
} else {
b
}
} fn process1(index: i32, rooms: &mut Vec<Vec<i32>>, n: i32, dp: &mut Vec<i32>) -> i32 {
if dp[index as usize] != -1 {
return dp[index as usize];
}
let mut next = 0;
for i in index + 1..n {
if reach(rooms[index as usize][0], rooms[i as usize][0]) {
next = get_max(next, process1(i, rooms, n, dp));
}
}
let mut ans = rooms[index as usize][1] + next;
dp[index as usize] = ans;
return dp[index as usize];
} fn reach(from: i32, mut to: i32) -> bool {
while to >= from {
if from == to {
return true;
} else {
to /= 2;
}
}
return false;
} // 正式方法
// 时间复杂度O(N)的动态规划
// 利用图来优化枚举
fn max_money2(n: i32, p: &mut Vec<i32>, w: &mut Vec<i32>) -> i32 {
let mut rooms: Vec<Vec<i32>> = repeat(repeat(0).take(2).collect())
.take(n as usize)
.collect();
for i in 0..n {
rooms[i as usize][0] = p[i as usize];
rooms[i as usize][1] = w[i as usize];
}
rooms.sort_by(|a, b| a[0].cmp(&b[0]));
let mut first: HashMap<i32, i32> = HashMap::new();
let mut graph: Vec<Vec<i32>> = vec![];
for i in 0..n {
let mut to = rooms[i as usize][0];
while to > 0 {
if first.contains_key(&to) {
graph[*first.get(&to).unwrap() as usize].push(i);
break;
} else {
to >>= 1;
}
}
graph.push(vec![]);
if !first.contains_key(&rooms[i as usize][0]) {
first.insert(rooms[i as usize][0], i);
}
}
let mut ans = 0;
let mut dp: Vec<i32> = repeat(0).take(n as usize).collect();
let mut i = n - 1;
while i >= 0 {
let mut post = 0;
for next in graph[i as usize].iter() {
if rooms[*next as usize][0] == rooms[i as usize][0] {
dp[i as usize] += dp[*next as usize];
} else {
post = get_max(post, dp[*next as usize]);
}
}
dp[i as usize] += post + rooms[i as usize][1];
ans = get_max(ans, dp[i as usize]);
i -= 1;
}
return ans;
} // 为了测试
fn random_array(n: i32, v: i32) -> Vec<i32> {
let mut arr: Vec<i32> = vec![];
for _i in 0..n {
arr.push(rand::thread_rng().gen_range(0, v) + 1);
}
return arr;
}

2022-12-30:某天小美进入了一个迷宫探险,根据地图所示,这个迷宫里有无数个房间 序号分别为1、2、3、...入口房间的序号为1 任意序号为正整数x的房间,都与序号 2*x 和 2*x + 1的更多相关文章

  1. 2018.12.30【NOIP提高组】模拟赛C组总结

    2018.12.30[NOIP提高组]模拟赛C组总结 今天成功回归开始做比赛 感觉十分良(zhōng)好(chà). 统计数字(count.pas/c/cpp) 字符串的展开(expand.pas/c ...

  2. 2015.12.29~2015.12.30真题回顾!-- HTML5学堂

    2015.12.29~2015.12.30真题回顾!-- HTML5学堂 吃饭,能够解决饥饿,提供身体运作机能.练习就像吃饭,强壮自己,提升编程技能,寻求编程技巧的最佳捷径!吃饭不能停,练习同样不能停 ...

  3. Linux下查找包含BOM头的文件和清除BOM头命令 2014-08-16 12:30:50

    Linux下查找包含BOM头的文件和清除BOM头命令 2014-08-16 12:30:50 分类: 系统运维 查找包含BOM头的文件,命令如下: 点击(此处)折叠或打开 grep -r -I -l ...

  4. zero3- JPA http://www.cnblogs.com/holbrook/archive/2012/12/30/2839842.html

    1.很好的博客:http://www.cnblogs.com/holbrook/archive/2012/12/30/2839842.html 2. 最新的搬到github : http://holb ...

  5. java.time.format.DateTimeParseException: Text '2019-10-11 12:30:30' could not be parsed at index 10

    java.time.format.DateTimeParseException: Text '2019-10-11 12:30:30' could not be parsed at index 10 ...

  6. js 中日期2013-08-30或2019-08-24 12:30:00 转换成时间戳,和2019-08-29T02:15:08.000+0000转化为2019-08-29T02:15:08

    js 中日期2019-08-24 或2019-08-24 12:30:00 转换成时间戳 首先将它转成date日期类型,然后获取毫秒形式时间戳 let date=new Date("2019 ...

  7. new Date(2019-08-24 12:30:00)和new Date(2019-08-29T02:15:08.000+0000)在ios的兼容NAN问题

    new Date()在安卓和pc端上正常显示,但是却在ios上显示 NAN的问题 正常写法: var time = new Date("2019-08-24 12:30:00"); ...

  8. 2018.12.30 bzoj3027: [Ceoi2004]Sweet(生成函数+搜索)

    传送门 生成函数好题. 题意简述:给出n个盒子,第iii个盒子里有mim_imi​颗相同的糖(但不同盒子中的糖不相同),问有多少种选法可以从各盒子中选出数量在[a,b][a,b][a,b]之间的糖果. ...

  9. 2018.12.30 bzoj3028: 食物(生成函数)

    传送门 生成函数模板题. 我们直接把每种食物的生成函数列出来: 承德汉堡:1+x2+x4+...=11−x21+x^2+x^4+...=\frac 1{1-x^2}1+x2+x4+...=1−x21​ ...

  10. EZ 2017 12 30 2018noip第二次膜你赛

    去年的比赛了,然而今天才改好. 总体难度适中,有大佬AK. 主要是自己SB第二题没想出来,然后又是可怜的100来分. T1 一道二分+数学的题目. 我们可以二分叫的次数,然后用公式(等差数列,公差都是 ...

随机推荐

  1. vs2022和wsl2开发和调试c++代码(转载)

    看见一个不错的帖子(知乎) https://zhuanlan.zhihu.com/p/390757559 里面最主要就是要保证wsl里面安装的东西够了,第二就是vs2022已经安装了linux的相关模 ...

  2. Ubuntu 14.04环境编译android源码android-5.0.2_r1.7z

    环境: Win7:8G内存 vmware:vm给ubuntu分配4G内存80G空间 参考视频: https://www.bilibili.com/video/BV15t411R78o ubuntu14 ...

  3. 登录:ORA-12504:TNS:监听程序在CONNECT_DATA中未获得SERVICE_NAME

    问题描述:在用pl/sql登录soctt用户时,显示: 解决办法:在tnsnames.ora文件中添加(文件位置的查找方法见文章末尾) ORCL = (DESCRIPTION = (ADDRESS = ...

  4. leetcode medium 记录 1-50

      # Title Solution 二刷 Difficulty 备注     2 Add Two Numbers      X   Medium     3 Longest Substring Wi ...

  5. SpringBoot——日志及原理

    一.SpringBoot日志 选用 SLF4j(接口)和 logback(实现类),除了上述日志框架,市场上还存在 JUL(java.util.logging).JCL(Apache Commons ...

  6. Spring Security 框架使用

    更多内容,前往IT-BLOG 一.Spring Security 简介 Spring Security 是一个能够为基于 Spring 的企业应用系统提供声明式的安全访问控制解决方案的安全框架.它提供 ...

  7. fluter usage---->动态更换Theme

    应用中切换深色主题和暗色主题是比较常见的操作,今天我们就来学习一下Flutter中动态的切换主题. Simple Theme import 'package:flutter/material.dart ...

  8. git命令的学习和基本使用

    初始化 git init (your_project) 配置 --local 只对当前仓库有效 --global 对当前用户所有仓库有效 --system 对系统登录的所有用户有效 git confi ...

  9. Vue2数据驱动渲染(render、update)

    上一篇文章我们介绍了 Vue2模版编译原理,这一章我们的目标是弄清楚模版 template和响应式数据是如何渲染成最终的DOM.数据更新驱动视图变化这部分后期会单独讲解 我们先看一下模版和响应式数据是 ...

  10. 购物车,实现增删改查;无bug,还有一个直接修改购物车数量功能未实现

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...