nyoj 动物统计加强版 290 (字典树) 好题

142 篇文章 0 订阅
8 篇文章 0 订阅

动物统计加强版

时间限制: 3000 ms  |           内存限制: 150000 KB
难度: 4
描述
在美丽大兴安岭原始森林中存在数量繁多的物种,在勘察员带来的各种动物资料中有未统计数量的原始动物的名单。科学家想判断这片森林中哪种动物的数量最多,但是由于数据太过庞大,科学家终于忍受不了,想请聪明如你的ACMer来帮忙。
输入
第一行输入动物名字的数量N(1<= N <= 4000000),接下来的N行输入N个字符串表示动物的名字(字符串的长度不超过10,字符串全为小写字母,并且只有一组测试数据)。
输出
输出这些动物中最多的动物的名字与数量,并用空格隔开(数据保证最多的动物不会出现两种以上)。
样例输入
10
boar
pig
sheep
gazelle
sheep
sheep
alpaca
alpaca
marmot
mole
样例输出
sheep 3
//唉,数据太大了,用我以前C语言模板不用想,肯定超时(还超了内存),看来不得不学学指针了(@。@)。。。
//下面是大神的,听他说是字典树的模板(指针型的。。。)
#include<stdio.h>
#include<string.h>
#include<math.h>
#include<stack>
#include<iostream>
#include<algorithm>
#define fab(a) (a)>0?(a):(-a)
#define LL long long
#define MAXN 10010
#define mem(x) memset(x,0,sizeof(x))
#define INF 0xfffffff 
using namespace std;
struct s
{
	int num;
	s *next[26];
};
s *root;
int ans=0;
char sa[12];
void create(char *str)
{
	int len=strlen(str);
	s *p=root,*q;
	for(int i=0;i<len;i++)
	{
		int id=str[i]-'a';
		if(p->next[id]==NULL)
		{
			q=(s *)malloc(sizeof(s));
			for(int j=0;j<26;j++)
			q->next[j]=NULL;
			q->num=0;
			p->next[id]=q;
			p=p->next[id];
		}
		else
		{
			//if(i==len-1)
			p=p->next[id];
		}
	}
	if(ans<++p->num)
	{
		strcpy(sa,str);
		ans=p->num;
	}
	
}
void begin()
{
	for(int i=0;i<26;i++)
	root->next[i]=NULL,root->num=0;
}
void freetree(s *t)
{
	if(t==NULL)
	return;
	for(int i=0;i<26;i++)
	{
		if(t->next[i]!=NULL)
		freetree(t->next[i]);
	}
	free(t);
	return;
}
int main()
{
	
	int t,i,n;
	char ss[12];
	int bz=0;
	root=(s *)malloc(sizeof(s));
	begin();
	scanf("%d",&n);
	for(i=0;i<n;i++)
	{
		scanf("%s",ss);
		create(ss);
	}
	printf("%s %d\n",sa,ans);
	freetree(root);
	return 0;
}
//另一位大神的,用运算符重载排序(用sort和qsort会超)也给过了。。。(好厉害)
//大神博客:http://blog.csdn.net/a_eagle/article/details/7376464
 
 
#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
struct tj
{
	char str[11];
	int count;
}w[4000001];
bool operator<(tj const&x,tj const&y)
{
	if(strcmp(x.str,y.str)<0) return true;
	return false; 
}
int main()
{
	int n,i,j,p,max;
	memset(w,0,sizeof(w));
	scanf("%d",&n);
	for(i=0;i<n;i++)
	{
		scanf("%s",w[i].str);
		w[i].count=1;
	}
	sort(w,w+n);
	p=0;max=0;
	for(i=1;i<n;i++)
	{
		if(strcmp(w[i].str,w[i-1].str)==0)
		{
			w[i].count=w[i-1].count+1;
		}
		if(w[i].count>max)
		{
			max=w[i].count;
			p=i;
		}
	}
	printf("%s %d\n",w[p].str,max);
	return 0;
}                        

//这个是我的代码,超内存&&超时(@.@)。。。
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define N 4000010
using namespace std;
char s[N][10];
int ch[N][30];
int word[N];
int sz;
void insert(char *s)
{
	int i,j,l=strlen(s);
	int k=0;
	for(i=0;i<l;i++)
	{
		j=s[i]-'a';
		if(!ch[k][j])
		{
			memset(ch[sz],0,sizeof(ch[sz]));
			ch[k][j]=sz++;
		}
		k=ch[k][j];
		word[k]++;
	}
}
int find(char *s)
{
	int i,j,l=strlen(s);
	int k=0;
	for(i=0;i<l;i++)
	{
		j=s[i]-'a';
		if(!ch[k][j])
			return 0;
		k=ch[k][j];
	}
	return word[k];
}
char c[11];
int main()
{
	int t,n,m,i,j,k;
	scanf("%d",&t);
	sz=1;
	for(i=0;i<t;i++)
	{
		scanf("%s",s[i]);
		insert(s[i]);
	}
	int cnt;
	int mm=0;
	for(i=0;i<t;i++)
	{
		cnt=find(s[i]);
		if(mm<cnt)
		{
			strcpy(c,s[i]);
			mm=cnt;
		}
	}
	printf("%s %d\n",c,mm);
	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值