【洛谷P3939】数颜色

题目背景

大样例下发链接:http://pan.baidu.com/s/1c0LbQ2 密码:jigg

题目描述

小 C 的兔子不是雪白的,而是五彩缤纷的。每只兔子都有一种颜色,不同的兔子可能有 相同的颜色。小 C 把她标号从 1 到 nn 的 nn 只兔子排成长长的一排,来给他们喂胡萝卜吃。 排列完成后,第 ii 只兔子的颜色是 a_ia
i
​ 。

俗话说得好,“萝卜青菜,各有所爱”。小 C 发现,不同颜色的兔子可能有对胡萝卜的 不同偏好。比如,银色的兔子最喜欢吃金色的胡萝卜,金色的兔子更喜欢吃胡萝卜叶子,而 绿色的兔子却喜欢吃酸一点的胡萝卜……为了满足兔子们的要求,小 C 十分苦恼。所以,为 了使得胡萝卜喂得更加准确,小 C 想知道在区间 [l_j,r_j][l
j
​ ,r
j
​ ] 里有多少只颜色为 c_jc
j
​ 的兔子。

不过,因为小 C 的兔子们都十分地活跃,它们不是很愿意待在一个固定的位置;与此同 时,小 C 也在根据她知道的信息来给兔子们调整位置。所以,有时编号为 x_jx
j
​ 和 x_j+1x
j
​ +1 的两 只兔子会交换位置。 小 C 被这一系列麻烦事给难住了。你能帮帮她吗?

输入输出格式

输入格式:
从标准输入中读入数据。 输入第 1 行两个正整数 nn,mm。

输入第 2 行 nn 个正整数,第 ii 个数表示第 ii 只兔子的颜色 a_ia
i
​ 。

输入接下来 mm 行,每行为以下两种中的一种:

“1\ l_j\ r_j\ c_j1 l
j
​ r
j
​ c
j
​ ” :询问在区间 [l_j,r_j][l
j
​ ,r
j
​ ] 里有多少只颜色为 c_jc
j
​ 的兔子;

“2\ x_j2 x
j
​ ”: x_jx
j
​ 和 x_j+1x
j
​ +1 两只兔子交换了位置。
输出格式:
输出到标准输出中。

对于每个 1 操作,输出一行一个正整数,表示你对于这个询问的答案。

输入输出样例

输入样例#1: 复制
6 5
1 2 3 2 3 3
1 1 3 2
1 4 6 3
2 3
1 1 3 2
1 4 6 3
输出样例#1: 复制
1
2
2
3
说明

【样例 1 说明】

前两个 1 操作和后两个 1 操作对应相同;在第三次的 2 操作后,3 号兔子和 4 号兔子

交换了位置,序列变为 1 2 2 3 3 3。

【数据范围与约定】
这里写图片描述

题解
题解里的解法一,按照颜色与位置进行双关键字排序。
操作1,二分位置。
操作2,如果两个位置颜色不同,则交换两点的坐标。

代码

#include<bits/stdc++.h>
typedef long long ll;
const int mod=1000000007;
const int N=500005;
using namespace std;
inline int read()
{
    int x=0,f=1;char ch=getchar();
    while (ch<'0'||ch>'9'){if (ch=='-')f=-1;ch=getchar();}
    while (ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
    return x*f;
}
struct node{int x,p;}a[300005];
int b[300005],n,m,L,R,C,X;
bool cmp(node a,node b)
{
    return (a.x!=b.x)?a.x<b.x:a.p<b.p;
}
int main()
{
    n=read();m=read();
    for (int i=1;i<=n;i++) a[i].x=read(),a[i].p=i;
    sort(a+1,a+n+1,cmp);
    for (int i=1;i<=n;i++) b[a[i].p]=i;
    while (m--)
    {
        int opt=read();
        if (opt==1)
        {
            L=read(),R=read(),C=read();
            int l=0,r=n,ans=0;
            while (l<=r)
            {
                int mid=(l+r)>>1;
                if (a[mid].x>C){r=mid-1;continue;}
                if (a[mid].x<C){ans=mid;l=mid+1;continue;}
                if (a[mid].p<L) l=mid+1,ans=mid;else r=mid-1;
            }
            int x=ans;
            l=0,r=n,ans=0;
            while (l<=r)
            {
                int mid=(l+r)>>1;
                if (a[mid].x>C){r=mid-1;continue;}
                if (a[mid].x<C){ans=mid;l=mid+1;continue;}
                if (a[mid].p<=R) l=mid+1,ans=mid;else r=mid-1;
            }
            int y=ans;
            printf("%d\n",y-x);
        }
        else
        {
            X=read();
            int x=b[X],y=b[X+1];
            if (a[b[X]].x==a[b[X+1]].x) continue;
            swap(a[x].p,a[y].p);
            swap(b[X],b[X+1]);
        }
    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 下面是一个简单的C语言实现,用于判断一个是否为回文: ```c #include <stdio.h> #include <string.h> int main() { char n[10]; scanf("%s", n); int len = strlen(n); int flag = 1; for (int i = 0; i < len / 2; i++) { if (n[i] != n[len - i - 1]) { flag = 0; break; } } if (flag) printf("yes"); else printf("no"); return 0; } ``` 其,使用字符组 `n` 读入输入的字,`len` 记录字的长度,`flag` 为判断标志,初始化为1。然后,通过循环判断字是否为回文,若不是,则将 `flag` 设为0,跳出循环。最后,根据 `flag` 输出判断结果。 需要注意的是,输入的字可能存在前导零,因此应当使用字符组进行输入和处理,避免出现错误。 ### 回答2: 洛谷p1015是一道题目要求编写C语言程序来判断一个整是否为回文。 回文是指正读和反读都一样的。例如121、3223都是回文,而123、12321不是回文。回文的判断可以通过将字转为字符串的方法,将该字符串与其反转后的字符串进行比较来实现。 首先,我们需要定义一个函来判断一个整是否是回文。函的输入是一个整,输出为1表示是回文,输出为0表示不是回文。 函的实现步骤如下: 1. 将整转为字符串。 2. 定义一个变量left表示字符串的起始位置,变量right表示字符串的结束位置。 3. 通过循环比较左右两边的字符,如果不相等,则可以判断该整不是回文,返回0。 4. 如果循环结束后没有返回0,则表示该整是回文,返回1。 在主函,我们需要读入一个整n,并调用判断回文的函来判断n是否是回文。然后根据函的返回值输出判断结果。 以上就是洛谷p1015回文的C语言实现。其,核心思想是将整转为字符串,并通过比较左右两边的字符来判断是否是回文。 ### 回答3: 洛谷p1015是一个关于回文的问题,需要用C语言编写。 回文是指正序和逆序都相同的字,例如121、131、1221等。 要解决这个问题,首先我们需要将输入的字进行逆序排列,然后与原字进行比较,如果相等,则说明该字是回文。 具体的解题思路如下: 1. 首先,我们需要读取输入的字n,n是一个整。 2. 我们需要编写一个函reverse,用于将字进行逆序排列。可以通过取余和除以10的方式逐位取得字。 3. 在主函,调用reverse函将输入的字n进行逆序排列,得到逆序后的字rev。 4. 进行比较,如果n和rev相等,则说明该字是回文。 5. 如果n和rev不相等,则说明该字不是回文。 6. 输出结果,如果是回文,则输出"YES",否则输出"NO"。 下面是一个简单的实现示例: ```c #include<stdio.h> int reverse(int num){ int rev=0; while(num>0){ rev=rev*10+num%10; num/=10; } return rev; } int main(){ int n; scanf("%d",&n); int rev=reverse(n); if(n==rev){ printf("YES"); }else{ printf("NO"); } return 0; } ``` 这样,我们就使用C语言解决了洛谷p1015回文问题。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值