ZOJ3627 POJ1631 HDU1950 Bridging Signals,O(N*logN)版最长上升子序列问题

这题如果用普通的最长上升子序列来做的话因为时间复杂度为O(n^2),肯定会超时,所以这里用的是O(N*logN)的方法,代码带有详细解释了。


/*******************************************************************************
 * Author : Neo Fung
 * Email : [email protected]
 * Last modified : 2011-09-08 20:10
 * Filename : ZOJ3627 POJ1631 HDU1950 Bridging Signals.cpp
 * Description : 设置一个array数组。对于array数组,对于第一个元素,存放的是长度
		为i的上升子序列中最小的末元素。同时用index来记录最长上升子序列的
		长度对于新的输入temp,如果它比array[index]大,则
		array[++index]=temp;否则,则在array中搜索到某个位置ops,
		array[ops]<temp,array[ops+1]>=temp,这时可以更新array[ops+1]=temp。
 * *****************************************************************************/
// ZOJ3627 POJ1631 HDU1950 Bridging Signals.cpp : Defines the entry point for the console application.
//

// #include "stdafx.h"



#include <fstream>
#include <stdio.h>
#include <iostream>
#include <string.h>
#include <string>
#include <memory.h>

using namespace std;

int array[50000];

void birfind(const int &a,const int &b,const int &data)
{
	if(a==b)
		array[a]=data;
	else if(a>b)
		return;
	else
	{
		int mid=a+(b-a)/2;
		if(array[mid]<data && array[mid+1]>=data)
			array[mid+1]=data;
		else if(array[mid]>=data)
			birfind(a,mid,data);
		else
			birfind(mid+1,b,data);
	}
}

int main(void)
{
	int p;
	int num[50000];
	int temp;
	int ncases;
	scanf("%d",&ncases);
	while(ncases--)
	{
		scanf("%d",&p);
		memset(array,0,sizeof(array));
		int index=0;
		for(int i=1;i<=p;++i)
		{
			scanf("%d",&temp);
			if(temp>array[index])
				array[++index]=temp;
			else
			{
				birfind(1,index,temp);
			}
		}
		printf("%d\n",index);
	}


	return 0;
}


你可能感兴趣的:(ZOJ3627 POJ1631 HDU1950 Bridging Signals,O(N*logN)版最长上升子序列问题)