`

南阳理工OJ 325 zb的生日 深度优先遍历 剪枝

 
阅读更多
/*
代码总体步骤是:(代码中的数字对应这些步骤)  
1.先求出总和除以二为平均数  
2.然后用母函数的代码 求出这些西瓜可以组成的
  平均数一下的全部可能的重量
3.用bool数组把他们存起来
4.最后找出离平均数最近的那个数,就可以求出结果
*/
#include<stdio.h>
#include<string.h>
int w[22];
int totle;
bool p[100010];
int main()
{
	int n,i,j,max;
	while(~scanf("%d",&n))
	{
		max=0;
		memset(p,0,sizeof(p));
		totle=0;
		p[0]=1;
		for(i=0;i<n;i++)
		{
			scanf("%d",&w[i]);
			totle+=w[i];//第一步
		}
		for(i=0;i<n;i++)//第二步
		{
			if(max!=totle/2)max+=w[i];
			if(max>totle/2)max=totle/2;//这里优化了一下,时间减少很多哦!
			for(j=max;j-w[i]>=0;j--)//标准的应该是j=totle/2;但很多都没有用到,所以定义个max记录最大重量,从max开始就可以了
				if(p[j-w[i]])p[j]=1;//第三步
		}
		for(i=totle/2;!p[i];i--);//第四步
		printf("%d\n",totle-2*i);
	}
	return(0);
}

 

 

/*
这个方法比较好理解
f()函数执行的步骤:

       要第一个西瓜                     不要第一个西瓜        -------第一次f()

要第二个西瓜  不要第二个西瓜      要第二个西瓜  不要第二个西瓜-------第二次f()

  ...  ...    ...     ...        ...     ...      ...     ... -------第n次 f() 
 
   优化: 如果哪一个分支超过了总数的一半,那个分支就不用算了。
  
*/
#include<stdio.h>
int w[22],totle,min,n,i,term;
int abs(int x){return x>0?x:-x;}
void f(int cur=0,int a=0)//cur记录第几个西瓜,a记录前cur个西瓜各种方式组成的重量
{
	if(cur==n)return;
	term=abs(totle-a-a);
	if(term<min)min=term;//记录计算过程中遇到的最符合的结果
	if(a>totle/2)return;//优化了一下,如果超过了总数一半的重量,就不用往下算了!
	f(cur+1,a);//要第cur个西瓜
	f(cur+1,a+w[cur]);//不要第cur个西瓜
}
int main()
{
	while(~scanf("%d",&n))
	{
		min=100000,totle=0;
		for(i=0;i<n;i++)scanf("%d",&w[i]),totle+=w[i];
		f();
		printf("%d\n",min);
	}
	return(0);
}

 

/*
这个方法跟第二个方法差不多,不过采用了各种优化手段使降低时间复杂度。
同时,为了配合优化,深度搜索的顺序也改变了(从后面向前面搜索)
下面大家慢慢体会代码的巧妙~
*/
#include <stdio.h>
#define max(a,b) a>b?a:b
int avg,ans,n,w[21],sum[21];
void dfs(int i=n,int cnt=0)//i为第i个西瓜  cnt为第i个西瓜以后的重量的各种组合
{
    if(i == 0)//二叉树往下找出所有的cnt,ans为最大的cnt
    {
        ans = max(ans,cnt);//找出最大的结果(因为下面的代码保证了结果不会超过平均数,最大结果就是最优解)
        return ;
    }
    if(ans == avg || cnt+sum[i] <= ans) return ;//如果找到平均数或者cnt加上前面所有的西瓜重量都不超过这个最优解,那么就不用往下算了!
    if(cnt+w[i] <= avg)dfs(i-1,cnt+w[i]);//要第i个西瓜(如果要了第i个西瓜就超过了平均数,就不要了)
    dfs(i-1,cnt);//不要第i个西瓜
}
int main()
{
    while(~scanf("%d",&n))
    {
        ans = 0;
        for(int i=1;i<=n;i++)//注意第一个西瓜存在w1里,而不是w0
        {
            scanf("%d",&w[i]);
            sum[i] = sum[i-1] + w[i];//求出前n项和
        }
        avg = sum[n]/2;//求出平均数
        dfs();//开始遍历,注意要求的是最大的不超过平均数的数
        printf("%d\n",sum[n]-2*ans);//输出结果
    }
    return 0;
}        

 

1
8
分享到:
评论

相关推荐

    南阳理工oj离线题库

    南阳理工oj离线题库是为编程爱好者和学习者提供的一种资源,主要用于练习和提高编程技能。这个离线题库通常包含多种类型的编程题目,涵盖了数据结构、算法、计算机科学基础等多个方面。在这个环境中,用户可以不受...

    南阳理工学院OJ_个人AC代码包(Java提交)

    【南阳理工学院OJ_个人AC代码包(Java提交)】是针对Java初学者的一份宝贵资源,它包含了参与ACM国际大学生程序设计竞赛(ICPC)时在南阳理工学院在线评测系统(OJ)上获得正确答案的代码实例。这些代码展示了如何用...

    南阳理工学院OJ第1版解题报告V1.0.pdf

    ### 南阳理工学院OJ第1版解题报告概览 #### 1. A+B Problem 虽然解题思路在报告中被省略,但我们可以推测这是一个基础的数学加法问题,涉及到数字输入与基本算术操作。此类题目旨在测试初学者对编程语言基本输入...

    南阳理工oj stl练习ac代码

    南阳理工学院的OJ(Online Judge)平台为学生提供了丰富的STL练习题目,通过AC(Accepted,表示代码正确通过所有测试用例)的代码,我们可以学习到STL在实际问题解决中的应用。 1. 容器: STL包含多种容器,如...

    湖南理工oj题解(学习用)-共230道题

    【标题】:“湖南理工oj题解(学习用)-共230道题”揭示了这是一个针对湖南理工大学在线编程竞赛平台(Online Judge,简称OJ)的题解集合,包含了230个不同题目。这类资源通常由参赛者或者经验丰富的程序员整理,...

    哈理工oj 1084百步穿杨

    哈理工OJ1084答案哈理工OJ1084答案哈理工OJ1084答案哈理工OJ1084答案哈理工OJ1084答案

    湖南理工学院OJ-小鱼比可爱

    湖南理工学院小鱼比可爱OJ题

    oj刷题 西安理工大学学生在线实验系统编程题答案(超级详细)

    这个“oj刷题”压缩包文件很可能是包含了西安理工大学在线实验系统中的一些典型题目,包括但不限于排序算法(如冒泡排序、快速排序、归并排序)、搜索算法(如二分查找、深度优先搜索、广度优先搜索)、图论问题(如...

    基于Laravel 5.0的OJ题解网站 , 目前涵盖安科OJ,南阳OJ,杭电OJ ,北大OJ,浙大OJ.zip

    【描述】中提到的“目前涵盖安科OJ,南阳OJ,杭电OJ,北大OJ,浙大OJ”意味着这个题解网站已经集成了多个知名OJ平台的题目,用户可以在一个统一的平台上找到这些不同OJ的题目并查看解决方案。安科OJ、南阳OJ、杭电OJ...

    北大oj 题目分类

    1. **图遍历**:深度优先遍历(DFS)和广度优先遍历(BFS),如`poj1094`的拓扑排序。 2. **最短路径算法**:Dijkstra、Bellman-Ford、Floyd、堆优化的Dijkstra,用于求解两点间的最短路径,如`poj1860`。 3. **最小...

    DFS:depth first search深度优先搜索(迷宫寻路) BFS:breadth first search宽度优先搜

    深度优先搜索(DFS,Depth First Search)和宽度优先搜索(BFS,Breadth First Search)是图论和算法中的两种基本搜索策略,常用于解决迷宫寻路、树遍历以及网络爬虫等问题。这两种算法各有其特点,适用于不同的场景...

    趣味题:柱状图排序 西安理工大学学生在线实验系统 oj

    趣味题:柱状图排序 西安理工大学学生在线实验系统 oj

    竞赛题集南阳OJ部分习题及解答其他oj试题及解答

    竞赛题集南阳OJ部分习题及解答其他oj试题及解答提取方式是百度网盘分享地址

    图的按录入顺序深度优先搜索.cpp

    西南科技大学OJ题

    山东理工大学2016级OJ题1832

    【知识点详解】 1. **C 语言基础**:在这些题目中,主要使用了 C 语言作为编程语言,包括变量声明、输入输出、循环结构、函数定义与调用等基本概念。例如,`scanf` 用于从标准输入读取数据,`printf` 用于输出结果...

    西南科技大学SWUST OJ 数据结构 图相关题题解 图.zip

    SWUST OJ : 1055、1056、1057、1058、1059、1060、1061、1062、1063、1064、1065、1066、1067、1068、1069、1070、1071、1072、1075、1086题答案

    湖南理工学院OJ-阶乘求和-定义函数

    湖南理工学院OJ-阶乘求和-定义函数

    软件工程课件--厦门理工

    《软件工程:厦门理工学院深度解析》 软件工程是一门涉及软件开发全过程的学科,它不仅关注编程技术,更注重软件开发的系统性、规范性和可维护性。厦门理工学院的这一系列课件,无疑为学习者提供了一个全面了解和...

    oj题.zip

    这些文件名看起来是编程题目,很可能来源于在线编程竞赛(Online Judge,简称OJ)平台,如LeetCode、Codeforces或HackerRank等。每个.py文件可能代表一个独立的编程问题解决方案,采用Python语言编写。接下来,我们...

    建二叉树及对二叉树进行遍历

    至于深度,可以采用层次遍历或递归的方法,每次遇到新的最深节点,就更新最大深度。 对于二叉树的遍历,有三种基本方法:前序遍历、中序遍历和后序遍历。 1. **前序遍历**(根-左-右):首先访问根节点,然后递归...

Global site tag (gtag.js) - Google Analytics