Java проблема превышена лимит времени проблема - PullRequest
6 голосов
/ 05 февраля 2011

Я кодирую проблему (ссылка - http://www.codechef.com/FEB11/problems/THREECLR/)

Ниже мой код

import java.io.*;
import java.util.*;


public class Main {

 static String ReadLn (int maxLg)  // utility function to read from stdin
    {
        byte lin[] = new byte [maxLg];
        int lg = 0, car = -1;
        String line = "";

        try
        {
            while (lg < maxLg)
            {
                car = System.in.read();
                if ((car < 0) || (car == '\n')) break;
                lin [lg++] += car;
            }
        }
        catch (IOException e)
        {
            return (null);
        }

        if ((car < 0) && (lg == 0)) return (null);  // eof
        return (new String (lin, 0, lg));
    }

 public static boolean iscontains(HashMap<Integer,HashSet<Integer>> resultmap,HashSet<Integer> b, int index)
 {
     boolean result=false;
     for(Iterator<Integer> iter = b.iterator();iter.hasNext();)
        {  int tmp=Integer.valueOf(iter.next().toString());
            if(resultmap.get(index).contains(tmp))
                    result=true;                
        }

     return result;
 }


public static void main(String[] args) throws InterruptedException, FileNotFoundException {
    try {

    HashMap<Integer,HashSet<Integer>> pairlist = new HashMap<Integer,HashSet<Integer>>();
     String input=null;
     StringTokenizer idata;
     int tc=0;
    input=Main.ReadLn(255);
    tc=Integer.parseInt(input);
    while(--tc>=0)
    {
        input=Main.ReadLn(255);
        idata = new StringTokenizer (input);idata = new StringTokenizer (input);
        int dishnum= Integer.parseInt(idata.nextToken());
        int pairnum= Integer.parseInt(idata.nextToken());
        while (--pairnum>=0)
        {
            input=Main.ReadLn(255);
            idata = new StringTokenizer (input);idata = new StringTokenizer (input);
            int dish1=Integer.parseInt(idata.nextToken());
            int dish2=Integer.parseInt(idata.nextToken());
            if(pairlist.containsKey((Integer)dish1))
            {
                HashSet<Integer> dishes=new HashSet<Integer>();
                dishes=pairlist.get(dish1);
                dishes.add(dish2);
                pairlist.put(dish1, dishes);
            }
            else
            {
                HashSet<Integer> dishes=new HashSet<Integer>();
                dishes.add(dish2);
                pairlist.put(dish1, dishes);
            }
        }
        int maxrounds=1;
        HashMap<Integer,HashSet<Integer>> resultlist = new HashMap<Integer,HashSet<Integer>>();
        HashSet<Integer> addresult=new HashSet<Integer>();
        addresult.add(1);
        resultlist.put(1,addresult);
        System.out.print("1");
        for(int i=2;i<=dishnum;i++)
        {
            boolean found_one=false;
            boolean second_check=false;
            int minroundnum=maxrounds;
            boolean pairlistcontains=false;
            pairlistcontains=pairlist.containsKey(i);
            for(int j=maxrounds;j>=1;j--)
            {
                if(!found_one){
                if(pairlistcontains)
                {
                    if(!iscontains(resultlist,pairlist.get((Integer) i),j))
                    {
                         for(Iterator<Integer> resultiter = resultlist.get(j).iterator();resultiter.hasNext();)
                         {
                             if(pairlist.get(resultiter.next()).contains(i))
                                 second_check=true;  
                         }
                         if(second_check==false)
                         {
                             found_one=true;
                             minroundnum=j;
                             j=0;
                             //second_check=false;
                         }
                    }

                }
                else
                {
                     for(Iterator<Integer> resultiter = resultlist.get(j).iterator();resultiter.hasNext();)
                     {
                         if(pairlist.get(resultiter.next()).contains(i))
                             second_check=true;  
                     }
                     if(second_check==false)
                     {
                         found_one=true;
                         minroundnum=j;
                         j=0;
                         //second_check=false;
                     }

                }
                second_check=false;


            }
        }
            if((minroundnum==maxrounds)&&(found_one==false))
                {
                ++minroundnum;
                ++maxrounds;
                }
            else
            {
                found_one=false;
            }
            HashSet<Integer> add2list=new HashSet<Integer> ();
            if(resultlist.containsKey(minroundnum))
            {
                add2list=resultlist.get(minroundnum);
                add2list.add(i);
            }
            else
            {
                add2list.add(i);
            }
            resultlist.put(minroundnum,add2list);
            System.out.print(" ");
            System.out.print(minroundnum);


        }
        if((tc !=-1))
            System.out.println();

    }


    }
    catch(Exception e){System.out.println(e.toString());}
}}

Я проверил этот код на онлайн-судьях, таких как Ideone, и получаю желаемые результаты. Но когда я отправляю этот код, я получаю ошибку превышения лимита времени. Я тестировал этот код на Ideone с достаточно большим набором входных данных, и время, затрачиваемое на его выполнение, было меньше 1 секунды. Кажется, что есть ошибка или утечка памяти, которая, похоже, выкинула все счастье из моей жизни. Любые указатели / советы будут с благодарностью.

Спасибо

EDIT1 -

Спасибо за ответы, ребята, я запустил код с вводом, сгенерированным следующим скриптом python -

import random
filename="input.txt"
file=open(filename,'w')
file.write("50")
file.write("\n")
for i in range(0,50):
        file.write("500 10000")
        file.write("\n")
        for j in range(0,10000):
                file.write(str(random.randrange(1,501))+" "+str(random.randrange(1,501)))
                file.write("\n")
file.close()

И мой код занял колоссальные 71052 миллисекунды, чтобы выполнить на входе, предоставленном вышеупомянутым сценарием. Теперь мне нужно сократить время выполнения до 8000 миллисекунд. Я работаю над попыткой замены HashMaps и HashSets, как это предложено rfeak, и мне также интересно, поможет ли в этом сценарии запоминание. Пожалуйста, сообщите.

EDIT2 - Перекодирование моего алгоритма с использованием массивов, похоже, сработало. Только повторная отправка одного и того же кода в разное время дает мне превышение Accepted Solution и Time limit: D У меня есть другой способ использовать хеш-карты для дальнейшей оптимизации. Спасибо огромное за помощь, ребята!

1 Ответ

7 голосов
/ 05 февраля 2011

Сколько памяти использует ваша программа при локальном запуске?

Если они запускают вашу Java-программу без достаточного количества памяти, вы могли бы потратить много времени, пытаясь собрать мусор.Это может разрушить вашу 1 секунду.

Если вам нужно сэкономить время и память (будет определено ...), у меня есть 2 предложения.

Замените HashSet<Integer> на BitSet,Подобный интерфейс, намного более быстрая реализация и использует намного меньше памяти.Особенно с числами, которые я вижу в задаче.

Замените Map<Integer,X> на X[] - ключ Integer может просто быть int (примитивным) индексом в вашем массиве.Опять быстрее и меньше.

...