填数【C语言】

题目描述#

有个格子,其中所有偶数格已经被填上了+或者-,现在请你将 填入剩余的格子,所有格子都必须填数,每个格子都只能填一个数,且所有数都只能使用一次。

要使得填完以后的算术表达式的结果为,请你给出一个填数方案。

输入格式#

第一行是一个整数,表示样例的个数。

每个样例的第一行是一个整数,表示需要填的数。

第二行是个由+,-组成的字符串。

输出格式#

依次输出样例的结果。

如果存在填数的方案,输出个数,表示填入的数。

否则,输出impossible

样例输入#

复制代码
2
3
+-
4
++-

样例输出#

复制代码
1 2 3
impossible

思路说明

复制代码
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
int compare(const void*a, const void*b){
    return strcmp((const void*)a,(const void*)b);
}
int main(){
    int T;
    scanf("%d", &T);
    while(T--){
        int n;
        char s[1001];
        scanf("%d",&n);
        scanf("%s",s);
        long long sum=(long long)n*(n+1)/2;
        if(sum%2!=0){
            printf("impossible\n");
            continue;
        }
        long long sum_half=sum/2;
        int m=0;
        for(int i=0;i<n-1;i++){
            if(s[i]=='+')m++;
        }
        int m_plus_1=m+1;//加法元素个数
        long long min_sum=(long long)m_plus_1*(m_plus_1+1)/2;
        long long max_sum=sum-(long long)(n-m_plus_1)*(n-m_plus_1+1)/2;
        if(sum_half<min_sum || sum_half>max_sum){
            printf("impossible\n");
            continue;
        }
        int add[m_plus_1];//构造加法集合
        for(int i=0;i<m_plus_1;i++) 
			add[i]=i+1;
        long long delta=sum_half-min_sum;
        int avg=delta/m_plus_1;
        int rem=delta%m_plus_1;
        for(int i=0;i<m_plus_1;i++)
            add[i]+=avg;
        for(int i=m_plus_1-1;i>=m_plus_1-rem;i--)
            add[i]++;
        qsort(add,m_plus_1,sizeof(int),compare);
        int subtract[n-m_plus_1];//构造减法集合
        int sub_idx=0;
        int used[n+1]; //标记数组
        memset(used,0,sizeof(used));
        for(int i=0;i<m_plus_1;i++)
            used[add[i]]=1;
        for(int i=1;i<=n;i++){
            if(!used[i]){
                subtract[sub_idx++]=i;
            }
        }
        qsort(subtract,sub_idx,sizeof(int),compare);
        int add_ptr=0,sub_ptr=0;
        printf("%d",add[add_ptr++]);
        for(int i=0;i<n-1;i++){
            if(s[i]=='+'){
                printf(" %d",add[add_ptr++]);
            }else{
                printf(" %d",subtract[sub_ptr++]);
            }
        }
        printf("\n");
    }
    return 0;
}
相关推荐
LDR0066 天前
Type-C 快充全面升级!LDR6601 赋能个人护理便携电机,重塑剃须刀 / 理发器新体验
c语言·开发语言
Luminous.7 天前
C语言--day30
c语言·开发语言
玖玥拾7 天前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
謓泽7 天前
C语言不是语法,是通往机器的地图。
c语言·开发语言
不会C语言的男孩7 天前
Linux 系统编程 · 第 8 章:进程基础
linux·c语言
2601_951643887 天前
C语言长文整理,关键字和数据类型
c语言·数据类型·关键字·嵌入式开发·格式化输出
m0_547486667 天前
《C#语言程序设计与实践》 全套PPT课件
c语言·c#·c语言程序设计
✎ ﹏梦醒͜ღ҉繁华落℘7 天前
编程基础 --高内聚,低耦合
c语言·单片机
QK_007 天前
C语言 static 关键字三大作用
c语言·开发语言
隔窗听雨眠7 天前
C语言函数递归从入门到精通(下):性能优化与工程实践
c语言·算法·性能优化