2013编程之美资格赛之树上的三角形(Java实现)

本文涉及的产品
云原生大数据计算服务 MaxCompute,5000CU*H 100GB 3个月
云原生大数据计算服务MaxCompute,500CU*H 100GB 3个月
简介: 树上的三角形 时间限制: 2000ms 内存限制: 256MB 描述 有一棵树,树上有只毛毛虫。它在这棵树上生活了很久,对它的构造了如指掌。所以它在树上从来都是走最短路,不会绕路。它还还特别喜欢三角形,所以当它在树上爬来爬去的时候总会在想,如果把刚才爬过的那几根树枝/树干锯下来,能不能从中选三根出来拼成一个三角形呢? 输入 输入数据的第一行包含一个整数 T,表示数据组数。 接下
树上的三角形


时间限制: 2000ms 内存限制: 256MB


描述
有一棵树,树上有只毛毛虫。它在这棵树上生活了很久,对它的构造了如指掌。所以它在树上从来都是走最短路,不会绕路。它还还特别喜欢三角形,所以当它在树上爬来爬去的时候总会在想,如果把刚才爬过的那几根树枝/树干锯下来,能不能从中选三根出来拼成一个三角形呢?


输入
输入数据的第一行包含一个整数 T,表示数据组数。


接下来有 T 组数据,每组数据中:


第一行包含一个整数 N,表示树上节点的个数(从 1 到 N 标号)。


接下来的 N-1 行包含三个整数 a, b, len,表示有一根长度为 len 的树枝/树干在节点 a 和节点 b 之间。


接下来一行包含一个整数 M,表示询问数。


接下来M行每行两个整数 S, T,表示毛毛虫从 S 爬行到了 T,询问这段路程中的树枝/树干是否能拼成三角形。


输出
对于每组数据,先输出一行"Case #X:",其中X为数据组数编号,从 1 开始。


接下来对于每个询问输出一行,包含"Yes"或“No”,表示是否可以拼成三角形。


数据范围
1 ≤ T ≤ 5


小数据:1 ≤ N ≤ 100, 1 ≤ M ≤ 100, 1 ≤ len ≤ 10000


大数据:1 ≤ N ≤ 100000, 1 ≤ M ≤ 100000, 1 ≤ len ≤ 1000000000


样例输入
2
5
1 2 5
1 3 20
2 4 30
4 5 15
2
3 4
3 5
5
1 4 32
2 3 100
3 5 45
4 5 60
2
1 4
1 3
样例输出
Case #1:
No
Yes
Case #2:
No
Yes


主要算法:
define results
get T
for 1 to T
get N
for 1 to N-1
get 边
getStruct 边s //构造类树
get M
define result
for 1 to M
get S,E
get S,E 到树根的路劲
get S,E 到树根的路劲的非公共节点
get S,E 间的路劲
get values S,E间路劲的段长度集合
sort(values)
result+=Test values //测试是否为三角形
results add result
print results


Test 算法
for i=2 to values.length
if values[i-2]+vaules[i-1]>values[i]
return true
return false


程序:


import java.util.*;

public class Main {

	public static void main(String[] args){
		
		Scanner sc=new Scanner(System.in);
		int T=sc.nextInt();
		
		String[] results=new String[T];
		for(int i=0;i<T;i++){
			int N=sc.nextInt();			
			int tempN=N-1;
			int[][] nodes=new int[tempN][3];
			for(int j=0;j<tempN;j++){
				nodes[j][0]=sc.nextInt();
				nodes[j][1]=sc.nextInt();
				nodes[j][2]=sc.nextInt();
			}
			List<ArrayList<Integer>> struct=getStruct(nodes);
			int M=sc.nextInt();
			String result="";
			for(int j=0;j<M;j++){
				int start=sc.nextInt();
				int end=sc.nextInt();
				result=result+" "+getResult(start,end,struct,nodes);
			}
			results[i]=result.trim();
		}
		
		printResult(results);
	}

	private static List<ArrayList<Integer>> getStruct(int[][] nodes) {
		ArrayList<Integer> num01s=new ArrayList<Integer>(nodes.length);
		int[] temp=new int[nodes.length+1];
		for(int i=0;i<temp.length;i++){
			temp[i]=0;
		}
		for(int i=0;i<nodes.length;i++){
			temp[(nodes[i][0]-1)]++;
			temp[(nodes[i][1]-1)]++;
		}
		for(int i=0;i<temp.length;i++){
			if(temp[i]==1)
				num01s.add(i+1);
		}
		int structLength=num01s.size()-1;
		List<ArrayList<Integer>> struct=new ArrayList<ArrayList<Integer>>(structLength);
		
		boolean[] flags=new boolean[structLength+1];
		for(int i=0;i<flags.length;i++){
			flags[i]=true;
		}
		List<Integer> success=new ArrayList<Integer>(nodes.length+1);
		for(int i=0;i<structLength;i++){
			ArrayList<Integer> list=new ArrayList<Integer>(nodes.length+1);
			for(int j=0;j<flags.length;j++){
				if(flags[j]){
					flags[j]=false;
					list.add(num01s.get(j));
					success.add(num01s.get(j));
					break;
				}
			}
			int node=list.get(0);
			boolean flag=true;
			while(flag){
				for(int k=0;k<nodes.length;k++){
					if((node==nodes[k][0])&&(!success.contains(nodes[k][1]))){
						list.add(nodes[k][1]);
						node=nodes[k][1];
						success.add(node);
						break;
					}else if((node==nodes[k][1])&&(!success.contains(nodes[k][0]))){
						list.add(nodes[k][0]);
						node=nodes[k][0];
						success.add(node);
						break;
					}
				}
				if(i!=0){
					int add=list.get(list.size()-1);
					for(int suc:success){
						for(int j=0;j<nodes.length;j++){
							if((suc==nodes[j][0] && add==nodes[j][1])||(suc==nodes[j][1] && add==nodes[j][0])){
								if(!list.contains(suc)){
									list.add(suc);
									flag=false;
									list=swap(list);
									break;
								}
							}
						}
					}
				}
				if(num01s.contains(node)){
					flags[num01s.indexOf(node)]=false;
					flag=false;
				}
			}
			struct.add(list);
		}
		return struct;
	}


	private static ArrayList<Integer> swap(ArrayList<Integer> list) {
		ArrayList<Integer> temp=new ArrayList<Integer>(list.size());
		for(int i=list.size()-1;i>=0;i--){
			temp.add(list.get(i));
		}
		return temp;
	}

	private static String getResult(int start, int end,
			List<ArrayList<Integer>> struct, int[][] nodes) {
		ArrayList<Integer> startPath=getNodePath(start,struct);
		ArrayList<Integer> endPath=getNodePath(end,struct);
		ArrayList<Integer> SE=new ArrayList<Integer>(nodes.length+1);
		int common=getCommonNode(startPath,endPath);
		if(startPath.indexOf(common)==0){
			SE=getPath(common,endPath);
		}else if( endPath.indexOf(common)==0){
			SE=getPath(common,startPath);
		}else {
			SE=getPath(common,startPath,endPath);
		}
		
		int[] values=getValue(SE,nodes);
		sort(values);
		return echo(values);
	}

	private static ArrayList<Integer> getPath(int common,
			ArrayList<Integer> list) {
		ArrayList<Integer> l=new ArrayList<Integer>(list.size());
		int n=list.indexOf(common);
		for(int i=0;i<=n;i++){
			l.add(list.get(i));
		}
		return l;
	}

	private static void sort(int[] values) {
		for(int i=0;i<values.length;i++){
			int min=i;
			for(int j=i+1;j<values.length;j++){
				if(values[j]<values[min])
					min=j;
			}
			if(i!=min){
				int temp=values[i];
				values[i]=values[min];
				values[min]=temp;
			}
		}
	}

	private static String echo(int[] values) {
		if(values.length<3)
			return "No";
		String result="No";
		for(int i=2;i<values.length;i++){
			if(judge(values[i-2],values[i-1],values[i])){
				result="Yes";
				break;
			}
		}
		return result;
	}

	private static boolean judge(int i, int j, int k) {
		return ((i+j)>k);
	}

	private static int[] getValue(ArrayList<Integer> sE,int[][] nodes) {
		int[] values=new int[sE.size()-1];
		for(int i=0;i<values.length;i++){
			for(int j=0;j<nodes.length;j++){
				int x=sE.get(i);
				int y=sE.get(i+1);
				if(nodes[j][0]==x && nodes[j][1]==y){
					values[i]=nodes[j][2];
					break;
				}else if (nodes[j][1]==x && nodes[j][0]==y){
					values[i]=nodes[j][2];
					break;
				}
			}
		}
		return values;
	}

	private static ArrayList<Integer> getPath(int m, 
			ArrayList<Integer> xList, ArrayList<Integer> yList) {
		ArrayList<Integer> path=new ArrayList<Integer>(xList.size()+yList.size());
		int xindex=xList.indexOf(m);
		int yindex=yList.indexOf(m);
		for(int i=0;i<xindex;i++){
			path.add(xList.get(i));
		}
		for(int i=yindex;i>=0;i--){
			path.add(yList.get(i));
		}
		return path;
	}

	private static int getCommonNode(ArrayList<Integer> xList,
			ArrayList<Integer> yList) {
		int max=xList.size();
		int c=-1;
		for(int i=0;i<max;i++){
			int temp=xList.get(i);
			if(yList.contains(temp)){
				c=temp;
				break;
			}
		}
		return c;
	}

	private static ArrayList<Integer> getNodePath(int start,
			List<ArrayList<Integer>> struct) {
		ArrayList<Integer> temp=new ArrayList<Integer>();
		temp.add(start);
		int s=start;
		int head=struct.get(0).get(0);
		int slength=struct.size();
		boolean flag=true;
		while(flag){
			for(int i=0;i<slength;i++){
				if(struct.get(i).contains(s) && s!=struct.get(i).get(0)){
					int x=struct.get(i).indexOf(s);
					
					for(int j=(x-1);j>=0;j--){
						temp.add(struct.get(i).get(j));
					}
					s=struct.get(i).get(0);
					break;
				}
			}
			if(head==s){
				flag=false;
			}
		}
		return temp;
	}

	private static void printResult(String[] result) {
		
		for(int i=0;i<result.length;i++){
		
			System.out.println("Case #"+(i+1)+":");
			
			String[] words=result[i].split(" ");
			for(String word:words)
				System.out.println(word);
		}
	}
}

这也是同样了,通过了比赛系统小数据测试,大数据测试未知。

——20130408



PS:

今天早晨数据出来了,这道题的大数据测试时TLE,超时。在情理之中。原因见博文《2013编程之美资格赛之大数据测试(Java实现)》

——20130409

相关实践学习
基于MaxCompute的热门话题分析
本实验围绕社交用户发布的文章做了详尽的分析,通过分析能得到用户群体年龄分布,性别分布,地理位置分布,以及热门话题的热度。
SaaS 模式云数据仓库必修课
本课程由阿里云开发者社区和阿里云大数据团队共同出品,是SaaS模式云原生数据仓库领导者MaxCompute核心课程。本课程由阿里云资深产品和技术专家们从概念到方法,从场景到实践,体系化的将阿里巴巴飞天大数据平台10多年的经过验证的方法与实践深入浅出的讲给开发者们。帮助大数据开发者快速了解并掌握SaaS模式的云原生的数据仓库,助力开发者学习了解先进的技术栈,并能在实际业务中敏捷的进行大数据分析,赋能企业业务。 通过本课程可以了解SaaS模式云原生数据仓库领导者MaxCompute核心功能及典型适用场景,可应用MaxCompute实现数仓搭建,快速进行大数据分析。适合大数据工程师、大数据分析师 大量数据需要处理、存储和管理,需要搭建数据仓库?学它! 没有足够人员和经验来运维大数据平台,不想自建IDC买机器,需要免运维的大数据平台?会SQL就等于会大数据?学它! 想知道大数据用得对不对,想用更少的钱得到持续演进的数仓能力?获得极致弹性的计算资源和更好的性能,以及持续保护数据安全的生产环境?学它! 想要获得灵活的分析能力,快速洞察数据规律特征?想要兼得数据湖的灵活性与数据仓库的成长性?学它! 出品人:阿里云大数据产品及研发团队专家 产品 MaxCompute 官网 https://www.aliyun.com/product/odps&nbsp;
相关文章
|
1月前
|
Java 程序员
Java编程中的异常处理:从基础到高级
在Java的世界中,异常处理是代码健壮性的守护神。本文将带你从异常的基本概念出发,逐步深入到高级用法,探索如何优雅地处理程序中的错误和异常情况。通过实际案例,我们将一起学习如何编写更可靠、更易于维护的Java代码。准备好了吗?让我们一起踏上这段旅程,解锁Java异常处理的秘密!
|
21天前
|
存储 缓存 Java
Java 并发编程——volatile 关键字解析
本文介绍了Java线程中的`volatile`关键字及其与`synchronized`锁的区别。`volatile`保证了变量的可见性和一定的有序性,但不能保证原子性。它通过内存屏障实现,避免指令重排序,确保线程间数据一致。相比`synchronized`,`volatile`性能更优,适用于简单状态标记和某些特定场景,如单例模式中的双重检查锁定。文中还解释了Java内存模型的基本概念,包括主内存、工作内存及并发编程中的原子性、可见性和有序性。
Java 并发编程——volatile 关键字解析
|
25天前
|
算法 Java 调度
java并发编程中Monitor里的waitSet和EntryList都是做什么的
在Java并发编程中,Monitor内部包含两个重要队列:等待集(Wait Set)和入口列表(Entry List)。Wait Set用于线程的条件等待和协作,线程调用`wait()`后进入此集合,通过`notify()`或`notifyAll()`唤醒。Entry List则管理锁的竞争,未能获取锁的线程在此排队,等待锁释放后重新竞争。理解两者区别有助于设计高效的多线程程序。 - **Wait Set**:线程调用`wait()`后进入,等待条件满足被唤醒,需重新竞争锁。 - **Entry List**:多个线程竞争锁时,未获锁的线程在此排队,等待锁释放后获取锁继续执行。
61 12
|
21天前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
116 2
|
1月前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
1月前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
59 3
|
1月前
|
开发框架 安全 Java
Java 反射机制:动态编程的强大利器
Java反射机制允许程序在运行时检查类、接口、字段和方法的信息,并能操作对象。它提供了一种动态编程的方式,使得代码更加灵活,能够适应未知的或变化的需求,是开发框架和库的重要工具。
58 4
时间轮-Java实现篇
在前面的文章《[时间轮-理论篇](https://developer.aliyun.com/article/910513)》讲了时间轮的一些理论知识,然后根据理论知识。我们自己来实现一个简单的时间轮。
|
8天前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
48 17
|
19天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者