填数【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;
}
相关推荐
三品吉他手会点灯2 小时前
C语言学习笔记 - 4.C概述 - C的特点
c语言·笔记·学习
她说彩礼65万2 小时前
C语言 函数指针
c语言·开发语言·算法
算法鑫探3 小时前
贪心算法(C 语言实现)及经典应用
c语言·数据结构·算法·贪心算法
C语言小火车3 小时前
嵌入式实习面试问题:那个动态内存是怎么样分配的?
c语言·开发语言·c++·嵌入式硬件·面试
拾光Ծ3 小时前
【Linux系统编程】深入理解命名管道(Named Pipe):从原理到实战的完整指南
linux·c语言·linux系统编程·进程间通信·ipc·命名管道
HZ·湘怡3 小时前
任意位置 单链表 回归
c语言·链表
Peregrine93 小时前
数据结构 - > 双链表
c语言·数据结构·算法
qeen873 小时前
【数据结构】队列及其C语言模拟实现
c语言·数据结构·c++·学习·队列
charlie11451419114 小时前
嵌入式C++工程实践第16篇:第四次重构 —— LED模板,从通用GPIO到专用抽象
c语言·开发语言·c++·驱动开发·嵌入式硬件·重构