左偏树 HYSBZ2809

2809: [Apio2012]dispatching

Time Limit: 10 Sec  Memory Limit: 128 MB
Submit: 5551  Solved: 2887
[Submit][Status][Discuss]

Description

在一个忍者的帮派里,一些忍者们被选中派遣给顾客,然后依据自己的工作获取报偿。在这个帮派里,有一名忍者被称之为 Master。除了 Master以外,每名忍者都有且仅有一个上级。为保密,同时增强忍者们的领导力,所有与他们工作相关的指令总是由上级发送给他的直接下属,而不允许通过其他的方式发送。现在你要招募一批忍者,并把它们派遣给顾客。你需要为每个被派遣的忍者 支付一定的薪水,同时使得支付的薪水总额不超过你的预算。另外,为了发送指令,你需要选择一名忍者作为管理者,要求这个管理者可以向所有被派遣的忍者 发送指令,在发送指令时,任何忍者(不管是否被派遣)都可以作为消息的传递 人。管理者自己可以被派遣,也可以不被派遣。当然,如果管理者没有被排遣,就不需要支付管理者的薪水。你的目标是在预算内使顾客的满意度最大。这里定义顾客的满意度为派遣的忍者总数乘以管理者的领导力水平,其中每个忍者的领导力水平也是一定的。写一个程序,给定每一个忍者 i的上级 Bi,薪水Ci,领导力L i,以及支付给忍者们的薪水总预算 M,输出在预算内满足上述要求时顾客满意度的最大值。

 

1  ≤N ≤ 100,000 忍者的个数;

1  ≤M ≤ 1,000,000,000 薪水总预算; 

0  ≤Bi < i  忍者的上级的编号;

1  ≤Ci ≤ M  忍者的薪水;

1  ≤Li ≤ 1,000,000,000  忍者的领导力水平。

 

Input

从标准输入读入数据。

第一行包含两个整数 N M,其中 N表示忍者的个数,M表示薪水的总预算。

接下来 N行描述忍者们的上级、薪水以及领导力。其中的第 i 行包含三个整 Bi , C i , L i分别表示第i个忍者的上级,薪水以及领导力。Master满足B i = 0并且每一个忍者的老板的编号一定小于自己的编号 Bi < i

 

Output

输出一个数,表示在预算内顾客的满意度的最大值。

 

Sample Input


5 4
0 3 3
1 3 5
2 2 2
1 2 4
2 3 1

Sample Output

6
 

HINT


如果我们选择编号为1的忍者作为管理者并且派遣第三个和第四个忍者,薪水总和为 4,没有超过总预算4。因为派遣了2个忍者并且管理者的领导力为3,用户的满意度为 2*3,于是可以得到的用户满意度的最大值。

 

中文题面,大意很好理解。

首先我们先了解一下左偏树(参考其他博客:

顾名思义,左偏树就是一棵向左偏的树2333

一个合格的左偏树,必须满足下列性质

性质1. 左偏树中,任何一个节点的父节点的权值都要小于等于这个节点的权值(堆性质)(也可大于等于来应用)

性质2. 左偏树中任意一个节点的左儿子的距离(到最近外节点的距离)一定大于等于右儿子的距离(左偏性质

推论1. 左偏树中任意一个节点的距离为其右儿子的距离+1 

推论2. n个点的左偏树,距离最大为log(n+1)−1 

那么,左偏树有哪些基础操作呢?

merge 合并

现在有两棵左偏树,a,b是他们的根节点,val a ≤ val b(否则swap一下a,b)

既然val a ≤ val b,说明如果将这两棵树合并,根还是a。

这时,只需要递归合并 a的右儿子,b 这两个点,并将新树的根节点作为a的右儿子。

合并完成之后,dis rsona可能会变,为了保证左偏性质,如果dis rsona > dis lsona ,就交换a的左右儿子。

最后,更新dis a,以a为合并后的树的根节点。

push 相当于一个节点与一棵左偏树的merge

pop :

将根pop出,将根的左右儿子合并merge。

时间复杂度O(logn)。


现在让我们再来看下这个题目。

思路:

dfs遍历树,记录更新每个节点子树的派遣忍者数,所发薪水值,用左偏树记录当前选择派遣忍者的节点,大顶,以便当一个节点合并完子节点的信息后,薪水超出时,pop出薪水更高的忍者。每个节点求得最优情况后,更新答案最大值。

类似这种满就删的思路,感觉比较常见。

 

(btw,最好不要写结构体,Memory会大orz)

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;

const int maxn=1e5+10;

int n;
long long m;
int b;
long long c[maxn];
long long l[maxn];
long long sum[maxn];
int siz[maxn];
long long ans;

int root[maxn];		///局部左偏树?
int cnt;	//记录边数
int head[maxn],to[maxn],nxt[maxn];

struct ltree	//左偏树
{
	long long val[maxn];
	int dis[maxn];
	int l[maxn],r[maxn];

	int merge(int x,int y)	//合并
	{
		if(!x||!y) return x+y;
		//If A = NULL Then return B
		//If B = NULL Then return A
		if(val[y]>val[x])
			swap(x,y);
		r[x]=merge(r[x],y);
		root[r[x]]=x;
		if(dis[r[x]]>dis[l[x]])
			swap(r[x],l[x]);
		if(!r[x]) dis[x]=0;
		else dis[x]=dis[r[x]]+1;
		return x;
	}

	int pop(int x)
	{
		int lastl=l[x],lastr=r[x];
		//将根pop出
		root[l[x]]=l[x];root[r[x]]=r[x];
		l[x]=r[x]=dis[x]=0;
		//返回左右子树合并后的树
		return merge(lastl,lastr);
	}
	long long top(int x)
	{
		return val[x];
	}
};

ltree tree;

void dfs(int u)
{
	int v;
	for(int i=head[u];i;i=nxt[i]){
		v=to[i];
		dfs(v);
		sum[u]+=sum[v];
		siz[u]+=siz[v];
		root[u]=root[v]=tree.merge(root[u],root[v]);

	}
	while(sum[u]>m){
		sum[u]-=tree.top(root[u]);root[u]=tree.pop(root[u]);	//树根最大,按照这个方式排的
		siz[u]--;
	}
	ans=max(ans,l[u]*siz[u]);
}

int main()
{
	while(~scanf("%d%lld",&n,&m)){
		//初始化!莫忘!
		ans=0;cnt=0;//变量清零操作
		memset(head,0,sizeof(head));//数组清零操作
		for(int i=1;i<=n;++i){
			scanf("%d%lld%lld",&b,&c[i],&l[i]);
			//数组表示有向边图
			to[++cnt]=i;//第cnt条边指向i
			nxt[cnt]=head[b];//第cnt条边的下一条边
			head[b]=cnt;//从b出发的一个边
			root[i]=i;	//最开始都是独立的
			tree.val[i]=c[i];
			sum[i]=c[i];siz[i]=1;
		}
		dfs(1);//1肯定是master
		printf("%lld\n",ans);
	}

	return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,你可以参考以下示例代码实现Java销售额查询和营业额统计: ```java import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; public class SalesStatistics { private Map<String, Double> salesData = new HashMap<>(); public SalesStatistics() { // 初始化销售数据 salesData.put("2021-01-01", 1000.0); salesData.put("2021-01-02", 1500.0); salesData.put("2021-01-03", 2000.0); salesData.put("2021-01-04", 1200.0); salesData.put("2021-01-05", 1800.0); } // 查询指定日期的销售额 public double getSalesByDate(String date) { if (salesData.containsKey(date)) { return salesData.get(date); } else { return 0.0; } } // 统计指定日期范围内的营业额 public double getTotalSales(String startDate, String endDate) { double totalSales = 0.0; for (String date : salesData.keySet()) { if (date.compareTo(startDate) >= 0 && date.compareTo(endDate) <= 0) { totalSales += salesData.get(date); } } return totalSales; } // 获取所有销售日期 public List<String> getAllSalesDates() { return new ArrayList<>(salesData.keySet()); } } ``` 使用示例: ```java public static void main(String[] args) { SalesStatistics salesStatistics = new SalesStatistics(); // 查询指定日期的销售额 double sales = salesStatistics.getSalesByDate("2021-01-02"); System.out.println("2021-01-02的销售额为:" + sales); // 统计指定日期范围内的营业额 double totalSales = salesStatistics.getTotalSales("2021-01-02", "2021-01-04"); System.out.println("2021-01-02到2021-01-04的营业额为:" + totalSales); // 获取所有销售日期 List<String> salesDates = salesStatistics.getAllSalesDates(); System.out.println("所有销售日期为:" + salesDates); } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值