今日头条笔试题-最大映射(贪心)

有 n 个字符串,每个字符串都是由 A-J 的大写字符构成。现在你将每个字符映射为一个 0-9 的数字,不同字符映射为不同的数字。这样每个字符串就可以看做一个整数,唯一的要求是这些整数必须是正整数且它们的字符串不能有前导零。现在问你怎样映射字符才能使得这些字符串表示的整数之和最大?
    输入描述:每组测试用例仅包含一组数据,每组数据第一行为一个正整数 n , 接下来有 n 行,每行一个长度不超过 12 且仅包含大写字母 A-J 的字符串。 n 不大于 50,且至少存在一个字符不是任何字符串的首字母。
    输出描述:输出一个数,表示最大和是多少。
    输入例子:
    2
    ABC
    BCA
    输出例子:
    1875

一开始看漏了首位不能映射到0的条件...直接贪了..结果发现不太对...

哦贪心的方法就是算每个字母的权值和...用pair 搞一下...

处理的办法是如果10个字母都出现,那么先把没有在首位出现过的字母中权重最小的那个映射到0,再搞剩下的...

一个trick是...map映射到0..和某个key没有被映射过..产生了二义性....

窝的做法就是整体+1,最后再减回来..

然后因为某处手残卡了1个小时...???.

哎我果然是个废k了.....傻逼贪心都写不对QAQ

/* ***********************************************
Author :111qqz
Created Time :2017年03月29日 星期三 16时28分18秒
File Name :toutiao1.cpp
************************************************ */
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <vector>
#include <queue>
#include <set>
#include <map>
#include <string>
#include <cmath>
#include <cstdlib>
#include <ctime>
#define fst first
#define sec second
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define ms(a,x) memset(a,x,sizeof(a))
typedef long long LL;
#define pi pair < int ,int >
#define MP make_pair
using namespace std;
const double eps = 1E-8;
const int dx4[4]={1,0,0,-1};
const int dy4[4]={0,-1,1,0};
const int inf = 0x3f3f3f3f;
const int N=55;
int n;
string st[N];
pair <long long ,int > cnt[20];
LL ten[25];
map<char,int>mp;
set<char>Nhead;
bool head[25];
set<char>all;
set<char>used;
set< pair <long long ,char> > se;
int main()
{
	#ifndef  ONLINE_JUDGE 
	freopen("code/in.txt","r",stdin);
  #endif
	cin>>n;
	ten[0] = 1;
	for ( int i = 1 ; i <= 15 ; i++) ten[i] = ten[i-1]*10LL;
	for ( int i = 1 ; i <= n ; i++) cin>>st[i];
	ms(cnt,0);
	ms(head,false);
	mp.clear();	//处理首位不能为0的思路:把不在头部出现的字母中,权值最小的映射到0//
			//只有当10个字母都出现了...否则就正常搞...
	for ( int i = 1 ; i <= n ; i++)
	{
	    int len = st[i].length();
	    for ( int j = 0 ; j < len ; j++)
	    {
		int val = st[i][j]-'A';
		if (j==0) head[val] = true;
		all.insert(st[i][j]);
		LL ret = 1LL*ten[len-j-1];
		cnt[val].fst+=ret;
		cnt[val].sec = val;
	    }
	}
	for ( int i = 0 ; i < 10 ; i++) if (!head[i]) Nhead.insert(i+'A');
//	for ( auto it = Nhead.begin() ; it!=Nhead.end() ; it++) printf("%c %lld\n",*it,cnt[*it-'A'].fst);
	if (all.size()==10)
	{
	    LL mn = 1E18;
	    int pid = -1;
	    set<char>::iterator mn_it;
	    for ( auto it = Nhead.begin();it!=Nhead.end() ;it++)
	    {
	//	cout<<"*it"<<*it<<endl;
		if (cnt[*it-'A'].fst < mn)
		{
		    mn = cnt[*it-'A'].fst;
		    mn_it = it;
		}
	    }
	 //   cout<<"bbbbb!   "<<*mn_it<<endl;
	    mp[*mn_it] = 1;
	    cnt[*mn_it-'A'].fst = -1;
	    used.insert(*mn_it);
	}
	sort(cnt,cnt+10);
//	for ( int i = 0 ; i < 10 ; i++) printf("%lld %d  \n",cnt[i].fst,cnt[i].sec); 
//	puts("");今日头条
	int num = all.size();	
	for ( int i = 9 ; i >= 0 ; i--)
	{
	    if (cnt[i].fst==-1) continue;
	    if (!mp[cnt[i].sec+'A'])
	    mp[cnt[i].sec+'A'] = i+1;
	}
//	for ( auto it = mp.begin() ; it!=mp.end() ; it++) cout<<it->fst<<" "<<it->sec-1<<endl;
	LL sum = 0LL;
	for ( int i = 1 ; i <= n ; i++)
	{
	    int len = st[i].length();
	    for ( int j = 0 ; j < len ; j++)
	    {
		LL val = mp[st[i][j]]-1;
		LL tmp = 1LL*val*ten[len-j-1];
	//	cout<<"tmp:"<<tmp<<endl;
		sum += tmp;
	    }
	}
	cout<<sum<<endl;
  #ifndef ONLINE_JUDGE  
  fclose(stdin);
  #endif
    return 0;
}