首页 > ACM题库 > HDU-杭电 > hdu 2561 第二小整数-优先队列-[解题报告]C++
2014
02-10

hdu 2561 第二小整数-优先队列-[解题报告]C++

第二小整数

问题描述 :

求n个整数中倒数第二小的数。
每一个整数都独立看成一个数,比如,有三个数分别是1,1,3,那么,第二小的数就是1。

输入:

输入包含多组测试数据。
输入的第一行是一个整数C,表示有C测试数据;
每组测试数据的第一行是一个整数n,表示本组测试数据有n个整数(2<=n<=10),接着一行是 n个整数 (每个数均小于100);

输出:

输入包含多组测试数据。
输入的第一行是一个整数C,表示有C测试数据;
每组测试数据的第一行是一个整数n,表示本组测试数据有n个整数(2<=n<=10),接着一行是 n个整数 (每个数均小于100);

样例输入:

2
2
1 2
3
1 1 3

样例输出:

2
1

使用优先队列,只维持队列中只有两个元素,对输入的数据做处理,只留下比当前top()小的数据其他的扔掉,然后输出top()即可

#include<stdio.h>
#include<math.h>
#include <queue>
#include<algorithm>
#include <iostream>
using namespace std;

int main()
{
    int t;
    cin>>t;
    while (t--)
    {
        priority_queue < int >q;
        int n;
        cin>>n;
        while(n--)
        {
            int num;
            scanf("%d",&num);
            if(q.empty())q.push(num);
            else
            {
                if(q.size()<2)q.push(num);
                else
                {
                    if(num<q.top())
                    {
                        q.push(num);
                        q.pop();
                    }
                }
            }
        }
        printf("%d\n",q.top());
    }

    return 0;
}

解题转自:http://blog.csdn.net/candy20094369/article/details/6845264


  1. 在方法1里面:

    //遍历所有的边,计算入度
    for(int i=0; i<V; i++)
    {
    degree = 0;
    for (j = adj .begin(); j != adj .end(); ++j)
    {
    degree[*j]++;
    }
    }

    为什么每遍历一条链表,要首先将每个链表头的顶点的入度置为0呢?
    比如顶点5,若在顶点1、2、3、4的链表中出现过顶点5,那么要增加顶点5的入度,但是在遍历顶点5的链表时,又将顶点5的入度置为0了,那之前的从顶点1234到顶点5的边不是都没了吗?

  2. 在方法1里面:

    //遍历所有的边,计算入度
    for(int i=0; i<V; i++)
    {
    degree = 0;
    for (j = adj .begin(); j != adj .end(); ++j)
    {
    degree[*j]++;
    }
    }

    为什么每遍历一条链表,要首先将每个链表头的顶点的入度置为0呢?
    比如顶点5,若在顶点1、2、3、4的链表中出现过顶点5,那么要增加顶点5的入度,但是在遍历顶点5的链表时,又将顶点5的入度置为0了,那之前的从顶点1234到顶点5的边不是都没了吗?