题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1394

题目大意:

给出一个序列,一对逆序数就是满足i<j&&a[i]>a[j]条件的一对数字。

每次将a数组的最后一个数放到数组的第一位上,原数组向后移动一位,得到一个新的序列,

求这些序列中最小的逆序数。(每个数都在0-n-1范围内)

求解思路:

(1)逆序数的性质:一个序列第i次循环的逆序数Pi=P(i-1)+(n-1-a[i])-a[i]。

(2)可以通过暴力求解,每次找最小值就行了。

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn = ;
const int INF = ;
int a[maxn];
int main(void)
{
int mi,cnt,i,j,num,n;
while(~scanf("%d",&n))
{
memset(a,,sizeof(a));
for(i=;i<n;i++) scanf("%d",&a[i]);
mi=INF;cnt=;
for(i=;i<n;i++)
for(j=i+;j<n;j++)
if(a[i]>a[j]) cnt++;
if(mi>cnt) mi=cnt;
for(i=;i<n;i++)
{
cnt=cnt+(n--a[i])-a[i];
if(mi>cnt) mi=cnt;
}
printf("%d\n",mi);
}
return ;
}

(3)线段树做法:

先初始化线段树都为0,然后输入每个数,每个数字+1,即改为1-n区间,判断b[i]+1到n区间内有无数字,有的话逆序数就+1

然后再更新节点b[i]表示这个节点被访问过了。

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn = ;
const int INF = ;
int a[maxn<<],b[maxn];
void pushup(int x)
{
a[x]=a[x*]+a[x*+];
}
void build(int x,int l,int r)
{
if(l==r)
{
a[x]=;return ;
}
int mid=(l+r)/;
build(x*,l,mid);
build(x*+,mid+,r);
pushup(x);
}
void update(int x,int l,int r,int pos)
{
if(l==r)
{
a[x]++;return ;
}
int mid=(l+r)/;
if(pos<=mid) update(x*,l,mid,pos);
if(pos>mid) update(x*+,mid+,r,pos);
pushup(x);
}
int query(int x,int l,int r,int A,int B)
{
if(A<=l&&r<=B) return a[x];
int mid=(l+r)/,ans=;
if(A<=mid) ans+=query(x*,l,mid,A,B);
if(B>mid) ans+=query(x*+,mid+,r,A,B);
return ans;
}
int main(void)
{
int n,i,ans,mi;
while(~scanf("%d",&n))
{
memset(a,,sizeof(a));
memset(b,,sizeof(b));
build(,,n);
ans=;mi=INF;
for(i=;i<=n;i++)
{
scanf("%d",&b[i]);
b[i]++;
ans+=query(,,n,b[i]+,n);
update(,,n,b[i]);
}
if(mi>ans) mi=ans;
for(i=;i<n;i++)
{
ans=ans+(n-b[i]+)-b[i];
if(mi>ans) mi=ans;
}
printf("%d\n",mi);
}
return ;
}

(4)归并排序做法

归并排序时,当a[j]>a[i]时,每次记录tmp+=(mid-i+1),最终求出的tmp就是这个序列的逆序数。

#include<iostream>
#include<cstdio>
using namespace std;
const int maxn = ;
int a[maxn],b[maxn],temp[maxn],tmp;
void Merge_Sort(int l,int r,int mid)
{
int pos=,i=l,j=mid+;
while(i<=mid&&j<=r)
{
if(a[i]>a[j]) tmp+=(mid+-i),temp[pos++]=a[j++];
else temp[pos++]=a[i++];
}
while(i<=mid) temp[pos++]=a[i++];
while(j<=r) temp[pos++]=a[j++];
for(i=;i<pos;i++) a[i+l]=temp[i];
}
void Merge(int l,int r)
{
if(l==r) return ;
int mid=(l+r)/;
Merge(l,mid);
Merge(mid+,r);
Merge_Sort(l,r,mid);
}
int main(void)
{
int n,ans,i;
while(~scanf("%d",&n))
{
for(i=;i<n;i++)
{
scanf("%d",&a[i]);b[i]=a[i];
}
tmp=;ans=;
Merge(,n-);
ans=tmp;
for(i=;i<n;i++){
tmp=tmp+n--b[i]*;
if(ans>tmp) ans=tmp;
}
printf("%d\n",ans);
}
return ;
}

hdu-1394(线段树&逆序数的性质和求法)的更多相关文章

  1. hdu 1394(线段树) 最小逆序数

    http://acm.hdu.edu.cn/showproblem.php?pid=1394 给出一列数组,数组里的数都是从0到n-1的,在依次把第一个数放到最后一位的过程中求最小的逆序数 线段树的应 ...

  2. hdu 1394 (线段树求逆序数)

    <题目链接> 题意描述: 给你一个有0--n-1数字组成的序列,然后进行这样的操作,每次将最前面一个元素放到最后面去会得到一个序列,那么这样就形成了n个序列,那么每个序列都有一个逆序数,找 ...

  3. hdu 1394 线段树计算逆序数

    线段树计算逆序数的原理: 用线段树来统计已插入的数的个数(所以要保证最大的那个数不能太大,否则数组都开不了),然后每插入一个数,就查询比插入的数大的个数,累加即可. 这个题还有一个特点就是,题目给的是 ...

  4. HDU 1394 线段树求逆序对

    Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java ...

  5. HDU 1394 线段树or 树状数组~

    Minimum Inversion Number Description The inversion number of a given number sequence a1, a2, ..., an ...

  6. hdu 1394 线段树

    Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java ...

  7. hdu 5877 线段树(2016 ACM/ICPC Asia Regional Dalian Online)

    Weak Pair Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others)Total ...

  8. hdu 3974 线段树 将树弄到区间上

    Assign the task Time Limit: 15000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  9. hdu 3436 线段树 一顿操作

    Queue-jumpers Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) To ...

随机推荐

  1. c# 关闭和重启.exe程序

    Process[] myprocess = Process.GetProcessesByName("a"); if (myprocess.Count() > 0)//判断如果 ...

  2. 定义一个servlet用于处理所有外部接口类 架构思路

    架构思路”: 所有外部URL访问请求(对外提供的接口)全部交给intServiceServlet处理, 然后servlet调用BPO通过URL中的命名去寻找相应的javaBean.接口BO,然后接口B ...

  3. Centos7安装Wkhtmltopdf -- nodejs将html转pdf

    安装wkhtmltopdf wget https://github.com/wkhtmltopdf/wkhtmltopdf/releases/download/0.12.4/wkhtmltox-0.1 ...

  4. 问题2:input、textarea、submit 宽度设置为100%,但显示宽度不一致

    <style type="text/css"> body{ padding: 10px; } input,textarea{ width: 100%; } </s ...

  5. java web获取客户端外网ip和所在区域

    @参考文章1.@参考文章2.@参考文章3.@参考文章4,@之前同事的项目 controller @Controller @RequestMapping("/home") publi ...

  6. SparseArray

    使用SparseArray更加节省内存空间的使用,SparseArray也是以key和value对数据进行保存的.使用的时候只需要指定value的类型即可.并且key不需要封装成对象类型.   Has ...

  7. Ubuntu防火墙配置

    转载自:http://blog.csdn.net/sumer0922/article/details/7485584Ubuntu11.04默认的是UFW(ufw 即uncomplicated fire ...

  8. DialogActivity

    <?xml version="1.0" encoding="utf-8"?> <resources> <style name=&q ...

  9. go语言中的数组

    package main; import ( "fmt" ) func main() { //声明一个数组 var a [3]int; a = [3]int{1, 2, 3}; / ...

  10. rbac集成 权限分配。之角色管理

    权限分配功能拆分: a. 角色管理 b. 用户管理 c. 菜单和权限的管理 d. 批量的权限操作 e. 分配权限 先实现 角色管理: 无非也就是,增删改查: 定义路由, 编写视图. 1.查看角色页面: ...