Advertisement
dzocesrce

[APS] Paytolls

Aug 29th, 2023
307
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Java 15.97 KB | None | 0 0
  1. // klasicna primena na Dijkstra algoritmot...
  2. import java.util.*;
  3.  
  4. class Graph<E> {
  5.  
  6.     int num_nodes;
  7.     GraphNode<E> adjList[];
  8.  
  9.     @SuppressWarnings("unchecked")
  10.     public Graph(int num_nodes, E[] list) {
  11.         this.num_nodes = num_nodes;
  12.         adjList = (GraphNode<E>[]) new GraphNode[num_nodes];
  13.         for (int i = 0; i < num_nodes; i++)
  14.             adjList[i] = new GraphNode<E>(i, list[i]);
  15.     }
  16.  
  17.     @SuppressWarnings("unchecked")
  18.     public Graph(int num_nodes) {
  19.         this.num_nodes = num_nodes;
  20.         adjList = (GraphNode<E>[]) new GraphNode[num_nodes];
  21.         for (int i = 0; i < num_nodes; i++)
  22.             adjList[i] = new GraphNode<E>(i, null);
  23.     }
  24.  
  25.     int adjacent(int x, int y) {
  26.         // proveruva dali ima vrska od jazelot so
  27.         // indeks x do jazelot so indeks y
  28.         return (adjList[x].containsNeighbor(adjList[y])) ? 1 : 0;
  29.     }
  30.  
  31.     void addEdge(int x, int y, float tezina) {
  32.         // dodava vrska od jazelot so indeks x do jazelot so indeks y so tezina
  33.         // i obratno
  34.         if (adjList[x].containsNeighbor(adjList[y])) {
  35.             adjList[x].updateNeighborWeight(adjList[y], tezina);
  36.             adjList[y].updateNeighborWeight(adjList[x], tezina);
  37.         } else{
  38.             adjList[x].addNeighbor(adjList[y], tezina);
  39.             adjList[y].addNeighbor(adjList[x], tezina);
  40.         }
  41.     }
  42.  
  43.     void deleteEdge(int x, int y) {
  44.         adjList[x].removeNeighbor(adjList[y]);
  45.         adjList[y].removeNeighbor(adjList[x]);
  46.     }
  47.  
  48.     /*************************** KRUSKAL ***********************************************************************/
  49.  
  50.     // Metoda koja generira niza od site rebra vo grafot
  51.     public Edge[] getAllEdges() {
  52.         int totalEdges = 0;
  53.         for (int i = 0; i < this.num_nodes; i++) {
  54.             // za sekoe teme go dodavame brojot na sosedi koi gi ima
  55.             totalEdges += this.adjList[i].getNeighbors().size();
  56.         }
  57.  
  58.         totalEdges /= 2; //bidejki e nenasocen graf, sekoe rebro se javuva kaj dve teminja
  59.  
  60.         Edge[] edges = new Edge[totalEdges];
  61.         int index = 0;
  62.         for (int i = 0; i < this.num_nodes; i++) {
  63.             for (int j = 0; j < this.adjList[i].getNeighbors().size(); j++) {
  64.                 int index1 = this.adjList[i].getIndex();
  65.                 // se zemaat indeksot i tezinata na j-ot sosed na temeto i
  66.                 int index2 = this.adjList[i].getNeighbors().get(j).node
  67.                         .getIndex();
  68.                 float weight = this.adjList[i].getNeighbors().get(j).weight;
  69.                 if(index2>index1) //bidejki grafot e nenasocen, da ne go zemame sekoe rebro dva pati
  70.                     edges[index++] = new Edge(index1, index2, weight);
  71.             }
  72.         }
  73.  
  74.         return edges;
  75.     }
  76.  
  77.     // Metoda koja gi sortira site rebra
  78.     private void sortEdges(Edge[] edges) {
  79.         for (int i = 0; i < edges.length - 1; i++) {
  80.             for (int j = i + 1; j < edges.length; j++) {
  81.                 if (edges[i].getWeight() > edges[j].getWeight()) {
  82.                     Edge tmp = edges[i];
  83.                     edges[i] = edges[j];
  84.                     edges[j] = tmp;
  85.                 }
  86.             }
  87.         }
  88.  
  89.     }
  90.  
  91.     //Metoda koja pravi unija na dve drva
  92.     private void union(int u, int v, int[] vrski){
  93.         int findWhat, replaceWith;
  94.  
  95.         if(u < v){
  96.             findWhat = vrski[v];
  97.             replaceWith = vrski[u];
  98.         }
  99.         else{
  100.             findWhat = vrski[u];
  101.             replaceWith = vrski[v];
  102.         }
  103.  
  104.         //za dvete poddrva stava ist index
  105.         //vo vrski t.e. gi spojuva
  106.         for(int i=0; i<vrski.length; i++){
  107.             if(vrski[i] == findWhat){
  108.                 vrski[i] = replaceWith;
  109.             }
  110.         }
  111.     }
  112.  
  113.     List<Edge> kruskal() {
  114.         /*
  115.          * Pomoshna niza za sledenje na kreiranite drva
  116.          * Ako dve teminja se del od isto poddrvo
  117.          * togash imaat ista vrednost vo nizata
  118.          */
  119.         int vrski[] = new int[this.num_nodes];
  120.  
  121.         // niza koja gi sodrzi site rebra
  122.         Edge[] edges = this.getAllEdges();
  123.         // se sortiraat rebrata spored tezinite vo neopagjacki redosled
  124.         this.sortEdges(edges);
  125.  
  126.         // inicijalizacija na pomosnata niza za evidencija,
  127.         // sekoj jazel si e posebno drvo
  128.         for (int i = 0; i < this.num_nodes; i++)
  129.             vrski[i] = i;
  130.  
  131.         // lista koja kje gi sodrzi MST rebra
  132.         List<Edge> mstEdges = new ArrayList<Edge>();
  133.  
  134.         for(int i=0; i<edges.length; i++){
  135.             //za sekoe rebro vo sortiran redosled
  136.             Edge e = edges[i];
  137.  
  138.             if(vrski[e.getFrom()] != vrski[e.getTo()]){
  139.                 //ako teminjata na ova rebro ne se vo isto poddrvo
  140.                 //ova rebro e MST rebro
  141.                 mstEdges.add(e);
  142.                 //sega dvete teminja treba da se vo isto poddrvo
  143.                 //t.e se pravi merge (unija) t.s. vo nizata vrski
  144.                 //za site elementi od dvete poddrva
  145.                 //go setira istiot (najmal) element od dvete poddrva
  146.                 //vrski = this.union(e.getFrom(), e.getTo(), vrski);
  147.                 this.union(e.getFrom(), e.getTo(), vrski);
  148.             }
  149.  
  150.             //ako sme dodale num_nodes-1 rebra moze da prestaneme
  151.             if(mstEdges.size()==(this.num_nodes-1))
  152.                 break;
  153.         }
  154.  
  155.         return mstEdges;
  156.     }
  157.  
  158.     /*******************************************************************************************************/
  159.     /************************ PRIM **************************************************************************/
  160.  
  161.     //Metoda koja go naogja najmaloto rebro do
  162.     //teme na neposeten sosed
  163.     private Edge getMinimalEdge(boolean[] included){
  164.         int index1 = Integer.MAX_VALUE, index2 = Integer.MAX_VALUE;
  165.         float minweight = Float.MAX_VALUE;
  166.  
  167.         for(int i=0;i<this.num_nodes;i++){
  168.             if(included[i]){
  169.                 //ako e vkluceno temeto i
  170.                 //izmini gi negovite nevkluceni sosedi
  171.                 Iterator<GraphNode<E>.GraphNodeNeighbor<E>> it = adjList[i].getNeighbors().iterator();
  172.                 while(it.hasNext()){
  173.                     GraphNode<E>.GraphNodeNeighbor<E> pom = it.next();
  174.                     //ako sosedot ne e poseten i ima do sega najmala tezina
  175.                     if(!included[pom.node.getIndex()] && pom.weight<minweight){
  176.                         index1 = i;
  177.                         index2 = pom.node.getIndex();
  178.                         minweight = pom.weight;
  179.                     }
  180.                 }
  181.             }
  182.         }
  183.  
  184.         if(minweight<Float.MAX_VALUE){
  185.             Edge ret = new Edge(index1, index2, minweight);
  186.             return ret;
  187.         }
  188.         return null;
  189.     }
  190.  
  191.  
  192.     List<Edge> prim(int start_index) {
  193.         // lista koja kje gi sodrzi MST rebra
  194.         List<Edge> mstEdges = new ArrayList<Edge>();
  195.  
  196.         boolean included[] = new boolean[this.num_nodes];
  197.         for(int i=0;i<this.num_nodes;i++)
  198.             included[i]=false;
  199.  
  200.         included[start_index] = true;
  201.  
  202.         for(int i=0;i<this.num_nodes-1;i++){
  203.             Edge e = this.getMinimalEdge(included);
  204.             if(e==null){
  205.                 System.out.println("Ne mozat da se povrzat site jazli");
  206.                 break;
  207.             }
  208.             included[e.getFrom()] = included[e.getTo()] = true;
  209.             mstEdges.add(e);
  210.         }
  211.  
  212.         return mstEdges;
  213.     }
  214.  
  215.     /*******************************************************************************************************/
  216.     /***************** DIJKSTRA ******************************************************************************/
  217.  
  218.     float[] dijkstra(int from) {
  219.  
  220.         /* Minimalna cena do sekoe od teminjata */
  221.         float distance[] = new float[this.num_nodes];
  222.         /* dali za temeto e najdena konecnata (minimalna) cena */
  223.         boolean finalno[] = new boolean[this.num_nodes];
  224.         for (int i = 0; i < this.num_nodes; i++) {
  225.             distance[i] = -1;
  226.             finalno[i] = false;
  227.         }
  228.  
  229.         finalno[from] = true;
  230.         distance[from] = 0;
  231.  
  232.         /*
  233.          * vo sekoj cekor za edno teme se dobiva konecna minimalna cena
  234.          */
  235.         for (int i = 1; i < this.num_nodes; i++) {
  236.             /* za site sledbenici na from presmetaj ja cenata */
  237.             Iterator<GraphNode<E>.GraphNodeNeighbor<E>> it = adjList[from].getNeighbors().iterator();
  238.             while (it.hasNext()) {
  239.                 GraphNode<E>.GraphNodeNeighbor<E> pom = it.next();
  240.                 /* ako grankata kon sosedot nema konecna cena */
  241.                 if (!finalno[pom.node.getIndex()]) {
  242.                     /* ako ne e presmetana cena za temeto */
  243.                     if (distance[pom.node.getIndex()] <= 0) {
  244.                         distance[pom.node.getIndex()] = distance[from]
  245.                                 + pom.weight;
  246.                     }
  247.                     /* inaku, ako e pronajdena poniska */
  248.                     else if (distance[from] + pom.weight < distance[pom.node
  249.                             .getIndex()]) {
  250.                         distance[pom.node.getIndex()] = distance[from]
  251.                                 + pom.weight;
  252.                     }
  253.                 }
  254.             }
  255.  
  256.             /* najdi teme so min. cena koja ne e konecna */
  257.             boolean minit = false; /* min. ne e inicijaliziran */
  258.             int k = -1;
  259.             float minc = -1;
  260.             /* proveri gi site teminja */
  261.             for (int j = 0; j < this.num_nodes; j++) {
  262.                 if (!finalno[j] && distance[j] != -1) { /*ako cenata ne e  konecna*/
  263.                     if (!minit) { /* ako ne e inicijaliziran minimumot */
  264.                         minc = distance[k = j]; /* proglasi go ova e minimum */
  265.                         minit = true; /* oznaci deka min e inicijaliziran */
  266.                     }
  267.                     /* proveri dali e pronajdeno teme so pomala cena */
  268.                     else if (minc > distance[j] && distance[j] > 0)
  269.                         minc = distance[k = j];
  270.                 }
  271.             }
  272.             finalno[from = k] = true;
  273.         }
  274.  
  275.         return distance;
  276.  
  277.     }
  278.  
  279.     /*******************************************************************************************************/
  280.  
  281.     @Override
  282.     public String toString() {
  283.         String ret = new String();
  284.         for (int i = 0; i < this.num_nodes; i++)
  285.             ret += adjList[i] + "\n";
  286.         return ret;
  287.     }
  288.     public GraphNode<E> getNode(String ime){
  289.         for(int i=0;i<this.adjList.length;i++){
  290.             if(adjList[i].getInfo().equals(ime)) return adjList[i];
  291.         }
  292.         return null;
  293.     }
  294.  
  295. }
  296. class GraphNodeNeighbor<E> {
  297.     GraphNode<E> node;
  298.     float weight;
  299.  
  300.     public GraphNodeNeighbor(GraphNode<E> node, float weight) {
  301.         this.node = node;
  302.         this.weight = weight;
  303.     }
  304.  
  305.     public GraphNodeNeighbor(GraphNode<E> node) {
  306.         this.node = node;
  307.         this.weight = 0;
  308.     }
  309.  
  310.     @Override
  311.     public boolean equals(Object obj) {
  312.         @SuppressWarnings("unchecked")
  313.         GraphNodeNeighbor<E> pom = (GraphNodeNeighbor<E>)obj;
  314.         return pom.node.equals(this.node);
  315.     }
  316. }
  317. class GraphNode<E> {
  318.     private int index; //index (reden broj) na temeto vo grafot
  319.     private E info;
  320.     private LinkedList<GraphNodeNeighbor<E>> neighbors;
  321.  
  322.     public GraphNode(int index, E info) {
  323.         this.index = index;
  324.         this.info = info;
  325.         neighbors = new LinkedList<GraphNodeNeighbor<E>>();
  326.     }
  327.  
  328.     public boolean containsNeighbor(GraphNode<E> o){
  329.         GraphNodeNeighbor<E> pom = new GraphNodeNeighbor<E>(o,0);
  330.         return neighbors.contains(pom);
  331.     }
  332.  
  333.     public void addNeighbor(GraphNode<E> o, float weight){
  334.         GraphNodeNeighbor<E> pom = new GraphNodeNeighbor<E>(o,weight);
  335.         neighbors.add(pom);
  336.     }
  337.  
  338.     public void removeNeighbor(GraphNode<E> o){
  339.         GraphNodeNeighbor<E> pom = new GraphNodeNeighbor<E>(o,0);
  340.         if(neighbors.contains(pom))
  341.             neighbors.remove(pom);
  342.     }
  343.  
  344.     @Override
  345.     public String toString() {
  346.         String ret= "INFO:"+info+" SOSEDI:";
  347.         for(int i=0;i<neighbors.size();i++)
  348.             ret+="("+neighbors.get(i).node.info+","+neighbors.get(i).weight+") ";
  349.         return ret;
  350.  
  351.     }
  352.  
  353.     public void updateNeighborWeight(GraphNode<E> o, float weight){
  354.         Iterator<GraphNodeNeighbor<E>> i = neighbors.iterator();
  355.         while(i.hasNext()){
  356.             GraphNodeNeighbor<E> pom = i.next();
  357.             if(pom.node.equals(o))
  358.                 pom.weight = weight;
  359.         }
  360.  
  361.     }
  362.  
  363.     @Override
  364.     public boolean equals(Object obj) {
  365.         @SuppressWarnings("unchecked")
  366.         GraphNode<E> pom = (GraphNode<E>)obj;
  367.         return (pom.info.equals(this.info));
  368.     }
  369.  
  370.     public int getIndex() {
  371.         return index;
  372.     }
  373.  
  374.     public void setIndex(int index) {
  375.         this.index = index;
  376.     }
  377.  
  378.     public E getInfo() {
  379.         return info;
  380.     }
  381.  
  382.     public void setInfo(E info) {
  383.         this.info = info;
  384.     }
  385.  
  386.     public LinkedList<GraphNodeNeighbor<E>> getNeighbors() {
  387.         return neighbors;
  388.     }
  389.  
  390.     public void setNeighbors(LinkedList<GraphNodeNeighbor<E>> neighbors) {
  391.         this.neighbors = neighbors;
  392.     }
  393.     class GraphNodeNeighbor<E> {
  394.         GraphNode<E> node;
  395.         float weight;
  396.  
  397.         public GraphNodeNeighbor(GraphNode<E> node, float weight) {
  398.             this.node = node;
  399.             this.weight = weight;
  400.         }
  401.  
  402.         public GraphNodeNeighbor(GraphNode<E> node) {
  403.             this.node = node;
  404.             this.weight = 0;
  405.         }
  406.  
  407.         @Override
  408.         public boolean equals(Object obj) {
  409.             @SuppressWarnings("unchecked")
  410.             GraphNodeNeighbor<E> pom = (GraphNodeNeighbor<E>)obj;
  411.             return pom.node.equals(this.node);
  412.         }
  413.  
  414.     }
  415. }
  416. class Edge{
  417.     private int fromVertex, toVertex;
  418.     private float weight;
  419.     public Edge(int from, int to, float weight){
  420.         this.fromVertex = from;
  421.         this.toVertex = to;
  422.         this.weight = weight;
  423.     }
  424.  
  425.     public int getFrom(){
  426.         return this.fromVertex;
  427.     }
  428.     public int getTo(){
  429.         return this.toVertex;
  430.     }
  431.     public float getWeight(){
  432.         return this.weight;
  433.     }
  434.  
  435.     @Override
  436.     public String toString() {
  437.         return "("+fromVertex+","+toVertex+")="+weight+" ";
  438.     }
  439. }
  440. public class Paytolls {
  441.     public static void main(String[] args) {
  442.         Scanner input= new Scanner(System.in);
  443.         int numCities= Integer.parseInt(input.nextLine());
  444.         Graph<String> graph= new Graph<>(numCities);
  445.         for(int i=0;i<numCities;i++){
  446.             graph.adjList[i].setIndex(i);
  447.             graph.adjList[i].setInfo(i+"");
  448.         }
  449.         String[] info= input.nextLine().split("\\s+");
  450.         int start= Integer.parseInt(info[0])-1;
  451.         int end= Integer.parseInt(info[1])-1;
  452.         int numPaths= Integer.parseInt(input.nextLine());
  453.         for(int i=0;i<numPaths;i++){
  454.             String[] info_path= input.nextLine().split("\\s+");
  455.             graph.addEdge(Integer.parseInt(info_path[0])-1,Integer.parseInt(info_path[1])-1,Float.parseFloat(info_path[2]));
  456.         }
  457.         float[] distances= graph.dijkstra(start);
  458.         System.out.println((int) distances[end]);
  459.     }
  460. }
  461. /*
  462. 4
  463. 1 3
  464. 5
  465. 1 2 4
  466. 1 3 15
  467. 2 3 3
  468. 3 4 15
  469. 1 4 13
  470. ------
  471. 7
  472.  
  473. 50
  474. 9 42
  475. 54
  476. 1 2 38
  477. 2 3 170
  478. 4 1 142
  479. 5 3 520
  480. 2 6 191
  481. 4 7 106
  482. 6 8 540
  483. 6 9 965
  484. 6 10 309
  485. 9 11 43
  486. 9 12 506
  487. 11 13 937
  488. 14 13 877
  489. 15 11 506
  490. 16 12 516
  491. 13 17 156
  492. 14 18 981
  493. 16 19 420
  494. 20 16 857
  495. 18 21 45
  496. 19 22 592
  497. 19 23 972
  498. 24 22 938
  499. 25 22 150
  500. 26 24 920
  501. 25 27 39
  502. 26 28 670
  503. 27 29 484
  504. 30 28 464
  505. 27 31 232
  506. 32 30 870
  507. 29 33 114
  508. 32 34 366
  509. 32 35 179
  510. 32 36 748
  511. 33 37 630
  512. 38 36 355
  513. 35 39 680
  514. 36 40 130
  515. 41 38 955
  516. 42 41 697
  517. 43 39 815
  518. 41 44 573
  519. 43 45 820
  520. 46 45 266
  521. 47 45 422
  522. 48 47 218
  523. 49 45 924
  524. 46 50 63
  525. 45 41 966
  526. 20 31 267
  527. 11 12 900
  528. 36 26 985
  529. 36 34 578
  530. ---------
  531. 3074
  532.  */
  533.  
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement