Balala Power!

题意:给你n个由小写字母组成的字符串,让你给26个字母分配0-25,每个字符串形成一个26进制的数字,问怎么分

权值这n个数的和最大。(不能有前导0,但是单个0可以)

题解:每个字符对答案的贡献都可以看作一个 26 进制的数字,问题相当于要给这些贡献加一个 0 到 25 的权重使得答案最大。最大的数匹配 25,次大的数匹配 24,依次类推。排序后这样依次贪心即可,唯一注意的是不能出现前导 0。

#include<iostream>
#include<cstring>
#include<stdio.h>
#include<algorithm>
typedef long long ll;
const int mod =1e9+7;
const int maxn =1e5+5;
using namespace std;

ll fac[maxn]= {1};   //存26进制数
int Hash[27];       //存每个字母代表数字
bool lead[27];      //判断前导零
char str[maxn];

struct node
{
    int cnt[maxn];//该字符在某一位的个数
    int id; //该字母
    bool operator < (const node &a)const
    {
        for(int i=maxn-1; i>=0; i--) //相等则继续判断
        {
            if( cnt[i] >a.cnt[i] ) return 1;
            else if(cnt[i] < a.cnt[i]) return 0;
        }
    }
} a[27];
int main()
{
    //freopen("in.txt","r",stdin);
    for(int i=1; i<maxn; i++)
    {
        fac[i]=fac[i-1]*26%mod;
    }
    int n,ca=1;
    while(~scanf("%d",&n))
    {
        memset(a,0,sizeof(a));
        memset(Hash,-1,sizeof(Hash));
        memset(lead,0,sizeof(lead));
        for(int i=1; i<=n; i++)
        {
            scanf("%s",str);
            int len=strlen(str);
            if(len != 1)
                lead[str[0]-'a']=1;
            for(int i=0; i<len; i++)
            {
                a[str[i]-'a'].cnt[len-i-1]++;
            }
        }
        //进位操作,满26个等与一个下一位
        for(int i=0; i<26; i++)
        {
            for(int j=0; j<maxn; j++)
            {
                if(a[i].cnt[j] >=26)
                {
                    a[i].cnt[j+1] += a[i].cnt[j]/26;
                    a[i].cnt[j] %= 26;
                }
            }
            a[i].id=i;
        }
        sort(a,a+26);
        for(int i=0; i<26; i++) //给每个字母赋值
        {
            Hash[a[i].id]= 26-i-1;
        }


        for(int j = 25; j >= 0; j--)
        {
            if(!lead[a[j].id])
            {
                for(int k = 25; k >= j+1; k--)
                    Hash[a[k].id] = Hash[a[k-1].id];
                Hash[a[j].id] = 0;
                break;
            }
        }


        ll ans=0;
        for(int i=0; i<26; i++)
        {
            for(int j=0; j<maxn; j++)
            {
                ans= (ans+ fac[j]*a[i].cnt[j]*Hash[a[i].id]%mod) %mod;
            }
        }
        printf("Case #%d: %lld\n", ca++, ans);
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值