Infinite Inversions CodeForces - 540E (树状数组+离散化)

http://codeforces.com/problemset/problem/540/E(原题所在地)

题目大意,在一串无穷的1,2,3…n的串值,有k次操作,每次操作给出两个数a,b,即把坐标a和坐标b的值交换,问最后有多少个逆序对,0<=k<=105,a和b在int范围(10^9)内0<=k<=105,a和b在int范围内 

题目分析:

如果a,b的范围是10^5的话那就是正常的树状数组求逆序对,但是这里范围太大,这事就要想到离散化,会有很多点涉及到交换,那么把这些要交换的点放到一个坐标轴上(x轴),那么从左到右去看这些待交换的点中间(随便找相邻的两个待交换点)都会有连续的几个按顺序排列的点(除了两个相邻的点都是待交换点),那么这些连续的点其实可以看成是一个点,也就是把线段看成点

记录下这个线段化成的点的长度,为以后逆序做贡献。如果这个点之前有x比他大的,那么这个线段化成的点就产生x*(len)(len是线段的长度)个逆序对,这样在求一遍加起来就好了。

注意数组开的大小开小了RE这里估计得开到450000以上,下图转换(字)的部分是防止超时,减小树状数组的上届

 

代码:

#include<stdio.h>
#include<algorithm>
#include<queue>
#include<vector>
#include<map>
using namespace std;
#define ll long long
map<int,ll>m;
int lowbit(int x)
{
    return x&-x;
}
void add(int i,int val)
{
    while(i>0)
    {
        m[i]+=(ll)val;
        i-=lowbit(i);
    }
}
ll sum(int i)
{
    ll s=0;
    while(i<=500003)
    {
        s+=m[i];
        i+=lowbit(i);
    }
    return s;
}
int t[500003]= {0};
int a[500005];
int main()
{
    vector<pair<int,int> >v;
    int n;
    pair<int,int>p;
    while(~scanf("%d",&n))
    {
        m.clear();
        map<int,int>cc;
        cc.clear();
        int h=0;
        v.clear();
        for(int i=0; i<n; i++)
        {
            scanf("%d%d",&p.first,&p.second);
            v.push_back(p);
            a[h++]=p.first;
            a[h++]=p.second;
        }
        sort(a,a+h);
        int len=unique(a,a+h)-a;
//        printf("%d\n",len);
        sort(a,a+len);
        int temp[500003];
        int k=1;
        for(int i=0; i<len; i++)
        {
            if(i==0||a[i]-a[i-1]==1)
            {
                cc[k]=1;
                temp[k++]=a[i];
            }
            else
            {
                cc[k]=a[i]-a[i-1]-1;
                temp[k++]=a[i]-1;
                cc[k]=1;
                temp[k++]=a[i];
            }
        }
//        for(int i=1; i<k; i++)
//        {
//            printf("%d ",temp[i]);
//        }
//        printf("\n");
        for(int i=0; i<n; i++)
        {
            v[i].first=lower_bound(temp+1,temp+k,v[i].first)-temp;
            v[i].second=lower_bound(temp+1,temp+k,v[i].second)-temp;
            //  printf("%d %d\n",v[i].first,v[i].second);

        }

        for(int i=1; i<k; i++)
        {
            t[i]=i;
        }
        for(int i=0; i<n; i++)
        {
            swap(t[v[i].first],t[v[i].second]);
        }
//        for(int i=1; i<k; i++)
//        {
//            printf("%d ",t[i]);
//        }
//        printf("\n");
        ll ans=0;
        for(int i=1; i<k; i++)
        {
            ans+=cc[t[i]]*sum(t[i]);
            add(t[i],cc[t[i]]);
        }
        printf("%lld\n",ans);
    }

}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,让我们来一步步实现这个功能。 首先,我们需要安装 Element UI,可以通过 npm 安装: ``` npm install element-ui --save ``` 然后在 main.js 中引入并使用: ```js import Vue from 'vue' import ElementUI from 'element-ui' import 'element-ui/lib/theme-chalk/index.css' Vue.use(ElementUI) ``` 接下来,我们来实现时间线组件。可以在组件中使用 el-timeline 和 el-timeline-item 组件,代码如下: ```html <template> <el-timeline> <el-timeline-item v-for="(item,index) in list" :key="index" :timestamp="item.time">{{ item.content }}</el-timeline-item> </el-timeline> </template> ``` 其中,list 是传入组件的数据,包含每个时间点的内容和时间信息。 接着,我们来实现自动滚动功能。可以通过监听 el-timeline 的 scroll 事件,将 el-timeline 的 scrollTop 设置为最大值,让其自动滚动到底部。代码如下: ```html <template> <el-timeline ref="timeline" @scroll="handleScroll"> <el-timeline-item v-for="(item,index) in list" :key="index" :timestamp="item.time">{{ item.content }}</el-timeline-item> </el-timeline> </template> <script> export default { methods: { handleScroll() { const timelineEl = this.$refs.timeline.$el timelineEl.scrollTop = timelineEl.scrollHeight } } } </script> ``` 然后,我们来实现无限滚动和动态加载功能。可以使用 v-infinite-scroll 插件,通过监听 el-timeline 的滚动事件,在滚动到底部时触发加载更多数据的方法。代码如下: ```html <template> <el-timeline ref="timeline" @scroll="handleScroll" v-infinite-scroll="loadMore" infinite-scroll-disabled="loading" infinite-scroll-distance="10"> <el-timeline-item v-for="(item,index) in list" :key="index" :timestamp="item.time">{{ item.content }}</el-timeline-item> <div v-if="loading">正在加载...</div> </el-timeline> </template> <script> import { InfiniteScroll } from 'element-ui' export default { directives: { InfiniteScroll }, data() { return { list: [], // 数据列表 loading: false // 是否正在加载 } }, methods: { // 加载数据 loadMore() { if (this.loading) { return } this.loading = true // 模拟异步加载数据 setTimeout(() => { const newData = [{ time: '2021-08-01', content: '新的内容' }] this.list = this.list.concat(newData) this.loading = false }, 1000) }, // 滚动事件 handleScroll() { const timelineEl = this.$refs.timeline.$el if (timelineEl.scrollTop === 0) { this.loadMore() } } } } </script> ``` 在上面的代码中,我们使用了 element-ui 中的 InfiniteScroll 指令来实现无限滚动功能。同时,我们在 el-timeline 中添加了 infinite-scroll-disabled 和 infinite-scroll-distance 属性,分别用于控制是否禁用无限滚动和触发加载更多数据的距离。 现在我们已经完成了 element-ui 的时间线组件 + 自动滚动 + v-infinite-scroll 无限滚动 + 动态加载功能的实现。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值