mirror of
https://github.com/DarkflameUniverse/DarkflameServer.git
synced 2024-11-10 02:08:20 +00:00
0545adfac3
Have fun!
545 lines
18 KiB
C++
545 lines
18 KiB
C++
/// \file
|
|
/// \brief \b [Internal] Weighted graph. I'm assuming the indices are complex map types, rather than sequential numbers (which could be implemented much more efficiently).
|
|
///
|
|
/// This file is part of RakNet Copyright 2003 Kevin Jenkins.
|
|
///
|
|
/// Usage of RakNet is subject to the appropriate license agreement.
|
|
/// Creative Commons Licensees are subject to the
|
|
/// license found at
|
|
/// http://creativecommons.org/licenses/by-nc/2.5/
|
|
/// Single application licensees are subject to the license found at
|
|
/// http://www.jenkinssoftware.com/SingleApplicationLicense.html
|
|
/// Custom license users are subject to the terms therein.
|
|
/// GPL license users are subject to the GNU General Public
|
|
/// License as published by the Free
|
|
/// Software Foundation; either version 2 of the License, or (at your
|
|
/// option) any later version.
|
|
|
|
#ifndef __WEIGHTED_GRAPH_H
|
|
#define __WEIGHTED_GRAPH_H
|
|
|
|
#include "DS_OrderedList.h"
|
|
#include "DS_Map.h"
|
|
#include "DS_Heap.h"
|
|
#include "DS_Queue.h"
|
|
#include "DS_Tree.h"
|
|
#include <assert.h>
|
|
#include "RakMemoryOverride.h"
|
|
#ifdef _DEBUG
|
|
#include <stdio.h>
|
|
#endif
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( push )
|
|
#endif
|
|
|
|
/// The namespace DataStructures was only added to avoid compiler errors for commonly named data structures
|
|
/// As these data structures are stand-alone, you can use them outside of RakNet for your own projects if you wish.
|
|
namespace DataStructures
|
|
{
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
class RAK_DLL_EXPORT WeightedGraph : public RakNet::RakMemoryOverride
|
|
{
|
|
public:
|
|
static void IMPLEMENT_DEFAULT_COMPARISON(void) {DataStructures::defaultMapKeyComparison<node_type>(node_type(),node_type());}
|
|
|
|
WeightedGraph();
|
|
~WeightedGraph();
|
|
WeightedGraph( const WeightedGraph& original_copy );
|
|
WeightedGraph& operator= ( const WeightedGraph& original_copy );
|
|
void AddNode(const node_type &node);
|
|
void RemoveNode(const node_type &node);
|
|
void AddConnection(const node_type &node1, const node_type &node2, weight_type weight);
|
|
void RemoveConnection(const node_type &node1, const node_type &node2);
|
|
bool HasConnection(const node_type &node1, const node_type &node2);
|
|
void Print(void);
|
|
void Clear(void);
|
|
bool GetShortestPath(DataStructures::List<node_type> &path, node_type startNode, node_type endNode, weight_type INFINITE_WEIGHT);
|
|
bool GetSpanningTree(DataStructures::Tree<node_type> &outTree, DataStructures::List<node_type> *inputNodes, node_type startNode, weight_type INFINITE_WEIGHT );
|
|
unsigned GetNodeCount(void) const;
|
|
unsigned GetConnectionCount(unsigned nodeIndex) const;
|
|
void GetConnectionAtIndex(unsigned nodeIndex, unsigned connectionIndex, node_type &outNode, weight_type &outWeight) const;
|
|
node_type GetNodeAtIndex(unsigned nodeIndex) const;
|
|
|
|
protected:
|
|
void ClearDijkstra(void);
|
|
void GenerateDisjktraMatrix(node_type startNode, weight_type INFINITE_WEIGHT);
|
|
|
|
DataStructures::Map<node_type, DataStructures::Map<node_type, weight_type> *> adjacencyLists;
|
|
|
|
// All these variables are for path finding with Dijkstra
|
|
// 08/23/06 Won't compile as a DLL inside this struct
|
|
// struct
|
|
// {
|
|
bool isValidPath;
|
|
node_type rootNode;
|
|
DataStructures::OrderedList<node_type, node_type> costMatrixIndices;
|
|
weight_type *costMatrix;
|
|
node_type *leastNodeArray;
|
|
// } dijkstra;
|
|
|
|
struct NodeAndParent
|
|
{
|
|
DataStructures::Tree<node_type>*node;
|
|
DataStructures::Tree<node_type>*parent;
|
|
};
|
|
};
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::WeightedGraph()
|
|
{
|
|
isValidPath=false;
|
|
costMatrix=0;
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::~WeightedGraph()
|
|
{
|
|
Clear();
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::WeightedGraph( const WeightedGraph& original_copy )
|
|
{
|
|
adjacencyLists=original_copy.adjacencyLists;
|
|
|
|
isValidPath=original_copy.isValidPath;
|
|
if (isValidPath)
|
|
{
|
|
rootNode=original_copy.rootNode;
|
|
costMatrixIndices=original_copy.costMatrixIndices;
|
|
costMatrix = new weight_type[costMatrixIndices.Size() * costMatrixIndices.Size()];
|
|
leastNodeArray = new node_type[costMatrixIndices.Size()];
|
|
memcpy(costMatrix, original_copy.costMatrix, costMatrixIndices.Size() * costMatrixIndices.Size() * sizeof(weight_type));
|
|
memcpy(leastNodeArray, original_copy.leastNodeArray, costMatrixIndices.Size() * sizeof(weight_type));
|
|
}
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
WeightedGraph<node_type, weight_type, allow_unlinkedNodes>& WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::operator=( const WeightedGraph& original_copy )
|
|
{
|
|
adjacencyLists=original_copy.adjacencyLists;
|
|
|
|
isValidPath=original_copy.isValidPath;
|
|
if (isValidPath)
|
|
{
|
|
rootNode=original_copy.rootNode;
|
|
costMatrixIndices=original_copy.costMatrixIndices;
|
|
costMatrix = new weight_type[costMatrixIndices.Size() * costMatrixIndices.Size()];
|
|
leastNodeArray = new node_type[costMatrixIndices.Size()];
|
|
memcpy(costMatrix, original_copy.costMatrix, costMatrixIndices.Size() * costMatrixIndices.Size() * sizeof(weight_type));
|
|
memcpy(leastNodeArray, original_copy.leastNodeArray, costMatrixIndices.Size() * sizeof(weight_type));
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
void WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::AddNode(const node_type &node)
|
|
{
|
|
adjacencyLists.SetNew(node, new DataStructures::Map<node_type, weight_type>);
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
void WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::RemoveNode(const node_type &node)
|
|
{
|
|
unsigned i;
|
|
DataStructures::Queue<node_type> removeNodeQueue;
|
|
|
|
removeNodeQueue.Push(node);
|
|
while (removeNodeQueue.Size())
|
|
{
|
|
delete adjacencyLists.Pop(removeNodeQueue.Pop());
|
|
|
|
// Remove this node from all of the other lists as well
|
|
for (i=0; i < adjacencyLists.Size(); i++)
|
|
{
|
|
adjacencyLists[i]->Delete(node);
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( disable : 4127 ) // warning C4127: conditional expression is constant
|
|
#endif
|
|
if (allow_unlinkedNodes==false && adjacencyLists[i]->Size()==0)
|
|
removeNodeQueue.Push(adjacencyLists.GetKeyAtIndex(i));
|
|
}
|
|
}
|
|
|
|
ClearDijkstra();
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
bool WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::HasConnection(const node_type &node1, const node_type &node2)
|
|
{
|
|
if (node1==node2)
|
|
return false;
|
|
if (adjacencyLists.Has(node1)==false)
|
|
return false;
|
|
return adjacencyLists.Get(node1)->Has(node2);
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
void WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::AddConnection(const node_type &node1, const node_type &node2, weight_type weight)
|
|
{
|
|
if (node1==node2)
|
|
return;
|
|
|
|
if (adjacencyLists.Has(node1)==false)
|
|
AddNode(node1);
|
|
adjacencyLists.Get(node1)->Set(node2, weight);
|
|
if (adjacencyLists.Has(node2)==false)
|
|
AddNode(node2);
|
|
adjacencyLists.Get(node2)->Set(node1, weight);
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
void WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::RemoveConnection(const node_type &node1, const node_type &node2)
|
|
{
|
|
adjacencyLists.Get(node2)->Delete(node1);
|
|
adjacencyLists.Get(node1)->Delete(node2);
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( disable : 4127 ) // warning C4127: conditional expression is constant
|
|
#endif
|
|
if (allow_unlinkedNodes==false) // If we do not allow _unlinked nodes, then if there are no connections, remove the node
|
|
{
|
|
if (adjacencyLists.Get(node1)->Size()==0)
|
|
RemoveNode(node1); // Will also remove node1 from the adjacency list of node2
|
|
if (adjacencyLists.Has(node2) && adjacencyLists.Get(node2)->Size()==0)
|
|
RemoveNode(node2);
|
|
}
|
|
|
|
ClearDijkstra();
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
void WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::Clear(void)
|
|
{
|
|
unsigned i;
|
|
for (i=0; i < adjacencyLists.Size(); i++)
|
|
delete adjacencyLists[i];
|
|
adjacencyLists.Clear();
|
|
|
|
ClearDijkstra();
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
bool WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::GetShortestPath(DataStructures::List<node_type> &path, node_type startNode, node_type endNode, weight_type INFINITE_WEIGHT)
|
|
{
|
|
path.Clear();
|
|
if (startNode==endNode)
|
|
{
|
|
path.Insert(startNode);
|
|
path.Insert(endNode);
|
|
return true;
|
|
}
|
|
|
|
if (isValidPath==false || rootNode!=startNode)
|
|
{
|
|
ClearDijkstra();
|
|
GenerateDisjktraMatrix(startNode, INFINITE_WEIGHT);
|
|
}
|
|
|
|
// return the results
|
|
bool objectExists;
|
|
unsigned col,row;
|
|
weight_type currentWeight;
|
|
DataStructures::Queue<node_type> outputQueue;
|
|
col=costMatrixIndices.GetIndexFromKey(endNode, &objectExists);
|
|
if (costMatrixIndices.Size()<2)
|
|
{
|
|
return false;
|
|
}
|
|
if (objectExists==false)
|
|
{
|
|
return false;
|
|
}
|
|
node_type vertex;
|
|
row=costMatrixIndices.Size()-2;
|
|
if (row==0)
|
|
{
|
|
path.Insert(startNode);
|
|
path.Insert(endNode);
|
|
return true;
|
|
}
|
|
currentWeight=costMatrix[row*adjacencyLists.Size() + col];
|
|
if (currentWeight==INFINITE_WEIGHT)
|
|
{
|
|
// No path
|
|
return true;
|
|
}
|
|
vertex=endNode;
|
|
outputQueue.PushAtHead(vertex);
|
|
row--;
|
|
#ifdef _MSC_VER
|
|
#pragma warning( disable : 4127 ) // warning C4127: conditional expression is constant
|
|
#endif
|
|
while (1)
|
|
{
|
|
while (costMatrix[row*adjacencyLists.Size() + col] == currentWeight)
|
|
{
|
|
if (row==0)
|
|
{
|
|
path.Insert(startNode);
|
|
for (col=0; outputQueue.Size(); col++)
|
|
path.Insert(outputQueue.Pop());
|
|
return true;
|
|
}
|
|
--row;
|
|
}
|
|
|
|
vertex=leastNodeArray[row];
|
|
outputQueue.PushAtHead(vertex);
|
|
if (row==0)
|
|
break;
|
|
col=costMatrixIndices.GetIndexFromKey(vertex, &objectExists);
|
|
currentWeight=costMatrix[row*adjacencyLists.Size() + col];
|
|
}
|
|
|
|
path.Insert(startNode);
|
|
for (col=0; outputQueue.Size(); col++)
|
|
path.Insert(outputQueue.Pop());
|
|
return true;
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
node_type WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::GetNodeAtIndex(unsigned nodeIndex) const
|
|
{
|
|
return adjacencyLists.GetKeyAtIndex(nodeIndex);
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
unsigned WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::GetNodeCount(void) const
|
|
{
|
|
return adjacencyLists.Size();
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
unsigned WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::GetConnectionCount(unsigned nodeIndex) const
|
|
{
|
|
return adjacencyLists[nodeIndex]->Size();
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
void WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::GetConnectionAtIndex(unsigned nodeIndex, unsigned connectionIndex, node_type &outNode, weight_type &outWeight) const
|
|
{
|
|
outWeight=adjacencyLists[nodeIndex]->operator[](connectionIndex);
|
|
outNode=adjacencyLists[nodeIndex]->GetKeyAtIndex(connectionIndex);
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
bool WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::GetSpanningTree(DataStructures::Tree<node_type> &outTree, DataStructures::List<node_type> *inputNodes, node_type startNode, weight_type INFINITE_WEIGHT )
|
|
{
|
|
// Find the shortest path from the start node to each of the input nodes. Add this path to a new WeightedGraph if the result is reachable
|
|
DataStructures::List<node_type> path;
|
|
DataStructures::WeightedGraph<node_type, weight_type, allow_unlinkedNodes> outGraph;
|
|
bool res;
|
|
unsigned i,j;
|
|
for (i=0; i < inputNodes->Size(); i++)
|
|
{
|
|
res=GetShortestPath(path, startNode, (*inputNodes)[i], INFINITE_WEIGHT);
|
|
if (res)
|
|
{
|
|
for (j=0; j < path.Size()-1; j++)
|
|
{
|
|
// Don't bother looking up the weight
|
|
outGraph.AddConnection(path[j], path[j+1], INFINITE_WEIGHT);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Copy the graph to a tree.
|
|
DataStructures::Queue<NodeAndParent> nodesToProcess;
|
|
DataStructures::Tree<node_type> *current;
|
|
DataStructures::Map<node_type, weight_type> *adjacencyList;
|
|
node_type key;
|
|
NodeAndParent nap, nap2;
|
|
outTree.DeleteDecendants();
|
|
outTree.data=startNode;
|
|
current=&outTree;
|
|
if (outGraph.adjacencyLists.Has(startNode)==false)
|
|
return false;
|
|
adjacencyList = outGraph.adjacencyLists.Get(startNode);
|
|
|
|
for (i=0; i < adjacencyList->Size(); i++)
|
|
{
|
|
nap2.node=new DataStructures::Tree<node_type>;
|
|
nap2.node->data=adjacencyList->GetKeyAtIndex(i);
|
|
nap2.parent=current;
|
|
nodesToProcess.Push(nap2);
|
|
current->children.Insert(nap2.node);
|
|
}
|
|
|
|
while (nodesToProcess.Size())
|
|
{
|
|
nap=nodesToProcess.Pop();
|
|
current=nap.node;
|
|
adjacencyList = outGraph.adjacencyLists.Get(nap.node->data);
|
|
|
|
for (i=0; i < adjacencyList->Size(); i++)
|
|
{
|
|
key=adjacencyList->GetKeyAtIndex(i);
|
|
if (key!=nap.parent->data)
|
|
{
|
|
nap2.node=new DataStructures::Tree<node_type>;
|
|
nap2.node->data=key;
|
|
nap2.parent=current;
|
|
nodesToProcess.Push(nap2);
|
|
current->children.Insert(nap2.node);
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
void WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::GenerateDisjktraMatrix(node_type startNode, weight_type INFINITE_WEIGHT)
|
|
{
|
|
if (adjacencyLists.Size()==0)
|
|
return;
|
|
|
|
costMatrix = new weight_type[adjacencyLists.Size() * adjacencyLists.Size()];
|
|
leastNodeArray = new node_type[adjacencyLists.Size()];
|
|
|
|
node_type currentNode;
|
|
unsigned col, row, row2, openSetIndex;
|
|
node_type adjacentKey;
|
|
unsigned adjacentIndex;
|
|
weight_type edgeWeight, currentNodeWeight, adjacentNodeWeight;
|
|
DataStructures::Map<node_type, weight_type> *adjacencyList;
|
|
DataStructures::Heap<weight_type, node_type, false> minHeap;
|
|
DataStructures::Map<node_type, weight_type> openSet;
|
|
|
|
for (col=0; col < adjacencyLists.Size(); col++)
|
|
{
|
|
// This should be already sorted, so it's a bit inefficient to do an insertion sort, but what the heck
|
|
costMatrixIndices.Insert(adjacencyLists.GetKeyAtIndex(col),adjacencyLists.GetKeyAtIndex(col), true);
|
|
}
|
|
for (col=0; col < adjacencyLists.Size() * adjacencyLists.Size(); col++)
|
|
costMatrix[col]=INFINITE_WEIGHT;
|
|
currentNode=startNode;
|
|
row=0;
|
|
currentNodeWeight=0;
|
|
rootNode=startNode;
|
|
|
|
// Clear the starting node column
|
|
if (adjacencyLists.Size())
|
|
{
|
|
adjacentIndex=adjacencyLists.GetIndexAtKey(startNode);
|
|
for (row2=0; row2 < adjacencyLists.Size(); row2++)
|
|
costMatrix[row2*adjacencyLists.Size() + adjacentIndex]=0;
|
|
}
|
|
|
|
while (row < adjacencyLists.Size()-1)
|
|
{
|
|
adjacencyList = adjacencyLists.Get(currentNode);
|
|
// Go through all connections from the current node. If the new weight is less than the current weight, then update that weight.
|
|
for (col=0; col < adjacencyList->Size(); col++)
|
|
{
|
|
edgeWeight=(*adjacencyList)[col];
|
|
adjacentKey=adjacencyList->GetKeyAtIndex(col);
|
|
adjacentIndex=adjacencyLists.GetIndexAtKey(adjacentKey);
|
|
adjacentNodeWeight=costMatrix[row*adjacencyLists.Size() + adjacentIndex];
|
|
|
|
if (currentNodeWeight + edgeWeight < adjacentNodeWeight)
|
|
{
|
|
// Update the weight for the adjacent node
|
|
for (row2=row; row2 < adjacencyLists.Size(); row2++)
|
|
costMatrix[row2*adjacencyLists.Size() + adjacentIndex]=currentNodeWeight + edgeWeight;
|
|
openSet.Set(adjacentKey, currentNodeWeight + edgeWeight);
|
|
}
|
|
}
|
|
|
|
// Find the lowest in the open set
|
|
minHeap.Clear();
|
|
for (openSetIndex=0; openSetIndex < openSet.Size(); openSetIndex++)
|
|
minHeap.Push(openSet[openSetIndex], openSet.GetKeyAtIndex(openSetIndex));
|
|
|
|
/*
|
|
unsigned i,j;
|
|
for (i=0; i < adjacencyLists.Size()-1; i++)
|
|
{
|
|
for (j=0; j < adjacencyLists.Size(); j++)
|
|
{
|
|
printf("%2i ", costMatrix[i*adjacencyLists.Size() + j]);
|
|
}
|
|
printf("Node=%i", leastNodeArray[i]);
|
|
printf("\n");
|
|
}
|
|
*/
|
|
|
|
if (minHeap.Size()==0)
|
|
{
|
|
// Unreachable nodes
|
|
isValidPath=true;
|
|
return;
|
|
}
|
|
|
|
currentNodeWeight=minHeap.PeekWeight(0);
|
|
leastNodeArray[row]=minHeap.Pop(0);
|
|
currentNode=leastNodeArray[row];
|
|
openSet.Delete(currentNode);
|
|
row++;
|
|
}
|
|
|
|
/*
|
|
#ifdef _DEBUG
|
|
unsigned i,j;
|
|
for (i=0; i < adjacencyLists.Size()-1; i++)
|
|
{
|
|
for (j=0; j < adjacencyLists.Size(); j++)
|
|
{
|
|
printf("%2i ", costMatrix[i*adjacencyLists.Size() + j]);
|
|
}
|
|
printf("Node=%i", leastNodeArray[i]);
|
|
printf("\n");
|
|
}
|
|
#endif
|
|
*/
|
|
|
|
isValidPath=true;
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
void WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::ClearDijkstra(void)
|
|
{
|
|
if (isValidPath)
|
|
{
|
|
isValidPath=false;
|
|
delete [] costMatrix;
|
|
delete [] leastNodeArray;
|
|
costMatrixIndices.Clear();
|
|
}
|
|
}
|
|
|
|
template <class node_type, class weight_type, bool allow_unlinkedNodes>
|
|
void WeightedGraph<node_type, weight_type, allow_unlinkedNodes>::Print(void)
|
|
{
|
|
#ifdef _DEBUG
|
|
unsigned i,j;
|
|
for (i=0; i < adjacencyLists.Size(); i++)
|
|
{
|
|
//printf("%i connected to ", i);
|
|
printf("%s connected to ", adjacencyLists.GetKeyAtIndex(i).systemAddress.ToString());
|
|
|
|
if (adjacencyLists[i]->Size()==0)
|
|
printf("<Empty>");
|
|
else
|
|
{
|
|
for (j=0; j < adjacencyLists[i]->Size(); j++)
|
|
// printf("%i (%.2f) ", adjacencyLists.GetIndexAtKey(adjacencyLists[i]->GetKeyAtIndex(j)), (float) adjacencyLists[i]->operator[](j) );
|
|
printf("%s (%.2f) ", adjacencyLists[i]->GetKeyAtIndex(j).systemAddress.ToString(), (float) adjacencyLists[i]->operator[](j) );
|
|
}
|
|
|
|
printf("\n");
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( pop )
|
|
#endif
|
|
|
|
#endif
|