題目轉(zhuǎn)自:
http://blog.163.com/ecy_fu/blog/static/444512620098228849190/
二筆只有三道題,分值分別為30, 30, 40,題分別如下:
1、實(shí)現(xiàn)strtol函數(shù),其原型如為int strtol(const char *num_str, char **endptr, int base),num_str存放待轉(zhuǎn)換的字符串,可以是負(fù)數(shù)也可以是正數(shù);endptr指向第一個(gè)非法字符的地址,如果endptr為NULL則不指向第一個(gè)非法字符的地址;base用于指示進(jìn)制,若base為0,則根據(jù)num_str的指示來(lái)轉(zhuǎn)換。函數(shù)必須檢查溢出,如果正數(shù)溢出,返回INT_MAX;若負(fù)數(shù)溢出,返回INT_MIN。
2、一億個(gè)數(shù)找最大的1000個(gè)數(shù),要求效率高占用內(nèi)存少。函數(shù)原型為:find_max_data(int* source_data, int* max_data),其中source_data是存放一億個(gè)數(shù)的數(shù)組,max_data用于存放其中最大的1000個(gè)數(shù)。
3、將一個(gè)集合拆分成兩個(gè)不相交的子集,兩個(gè)子集元素之和相等,如{1, 2, 3, 4, 5, 6, 7},拆分成:
{2, 5, 7}, {1, 3, 4, 6}
給出一個(gè)集合,求所有符合上面要求的拆分,效率最高分越高,函數(shù)原型為int cal_num(int n);
第三題:
利用回溯剪枝法
空間復(fù)雜度:O(n) 棧的最大深度也就是n了
時(shí)間復(fù)雜度:接近于O(2^n-1), 因?yàn)楸举|(zhì)上程序時(shí)一個(gè)遍歷樹(shù)的過(guò)程,如果沒(méi)有剪枝,那么樹(shù)是一個(gè)滿二叉樹(shù),結(jié)點(diǎn)共2^n-1個(gè),也就要遍歷2^n-1次。雖然剪枝,但速度估計(jì)仍是 2^n次方級(jí)別的。
試了下,調(diào)用cal_num(104),好久了結(jié)果都沒(méi)有出來(lái)。。。
不知用上DP算法會(huì)不會(huì)好點(diǎn),不過(guò)聽(tīng)說(shuō)回溯法怎么弄效率都跟不上,最好用遞推?
在哪聽(tīng)說(shuō)的?
http://topic.csdn.net/u/20090922/11/ebc26b48-6581-40c3-afe0-a95ca2d700d5.html

/**//////////////////////////////////////////////////////////////////
//file divide_set.h:

#ifndef __DIVIDE_SET_H__
#define __DIVIDE_SET_H__

// 計(jì)算集合set的所有滿足下列條件的子集合:子集合元素之和等于value
// 子集合的元素對(duì)應(yīng)的label置1
void divide_set( int set[], int label[], int len, int i_set, int value );

// 對(duì)集合{1,2,
n}劃分
void cal_num( int n );

#endif



/**//////////////////////////////////////////////////////////////////
//file divide_set.cpp:

#include "stdafx.h"
#include "divide_set.h"

#include <iostream>

using namespace std;

// 查找集合set中,滿足元素之和等于value的子集合,結(jié)果存于label里
void divide_set( int set[], int label[], int len, int i_set, int value )


{
// 輸出結(jié)果
if ( value == 0 )

{
cout<<"{ ";
for ( int i=0; i<len; ++i )

{
if ( label[i] )

{
cout<<set[i]<<" ";
}
}
cout<<"} ";
cout<<" , { ";
for ( int i=0; i<len; ++i )

{
if ( 0 == label[i] )

{
cout<<set[i]<<" ";
}
}
cout<<"} ";
cout<<endl;
return;
}

if ( i_set >= len || value <0)

{
return;
}

// 取第i_set個(gè)元素
label[i_set] = 1;
divide_set( set, label, len, i_set+1, value-set[i_set] );
// 不取第i_set個(gè)元素
label[i_set] = 0;
divide_set( set, label, len, i_set+1, value );
}

void cal_num( int n )


{
int* set = new int[n];
int* label = new int[n];

// initialize set and label
int sum_value = 0;
for ( int i=0; i<n; ++i )

{
set[i] = i+1;
sum_value += set[i];
}
memset( label, 0, n*sizeof(int) );

// 保證元素總和為偶數(shù)
if( sum_value%2 == 0 )
divide_set( set, label, n, 0, sum_value/2 );

delete[] set;
delete[] label;
}


本文來(lái)自CSDN博客,轉(zhuǎn)載請(qǐng)標(biāo)明出處:http://blog.csdn.net/zdl1016/archive/2009/10/04/4632688.aspx