-
Notifications
You must be signed in to change notification settings - Fork 0
/
newinputfile
124 lines (118 loc) · 5.77 KB
/
newinputfile
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
\section#about{About the program}
\paragraph{}
This program reads a certain description of an undirected graph and
determines the number of connected components of it\.
\section#graph-description{About the description of the graph}
\paragraph{}
The graph bold{shall} be read from a file. The file describes
the graph by way of specifying the number of nodes, the
number of edges and a description of what nodes
each edge connects.
\section#syntax{Syntax}
\paragraph{}
The input file consists only of a series of arbitrarily space-
separated integer constants written in plain ASCII, and
consequently, also readable as an UTF-8 plain text file.
\paragraph{}
The first two integers shall be the number of nodes (n), and the
number of edges (m), respectively.
\paragraph{}
Given this, the set of nodes is assumed to be \math{{1, 2,... n}}, and
each node will be referred simply by its number.
\paragraph{}
Then comes math{m} pairs of integers "u" and "v" each giving the
existence of a bidirectional edge between nodes "u" and "v".
\section#interface{The command line interface}
\paragraph{}
As a binary, our program shall receive one and only one command line argument, and this will
be interpreted as the file name for the input file described \>{/about/graph-description/syntax}
\paragraph{}
If the input file doesn't comply with the #/about/graph-description/syntax our program will fail silently.
\section#program{The program}
As this program will be considerably small, it will consist of a single source file
in which we will put all of our code.
\file#app{source/app.d}
The one and only sourcefile. Will contain everything.
The structure of this file will be typical.
\addto{\>app}
void main()
{
\>{imports}
\>{global-vars}
return 0;
}
\fragment#imports{Import declarations}
In this fragment will be included the import declarations that we see
fit.
\fragment#global-vars{Global variable definitions}
In this fragment we will be creating and initializing global variables
as we require them.
\line{}
\eq{x^2 = 2 + 1}
\fragment#function-defs{Function definitions}
In this fragment we will be inserting our function definitions.
\line{}
As per the rules of the D programming language, the order in which we insert new fragments
into the three previous fragments won't affect the final result.
\section#main{The main function}
The main function will be the entry point to our program. This is the starting point of execution.
The structure of this function will be typical of a D program.
\addto{\>/program/function-defs}
void main (string[] args)
{
> local-vars
> main-process
}
\fragment#local-vars{Local variables definitions}
In this fragment we will be inserting the local variables we require.
\fragment#main-process{Main process #main-process}
In this fragment we will input how our main function does its processing.
\line{}
Where `args` is an array of strings that contains the arguments we received from the user. The first argument is always the name of the binary,
the rest are given by the user.
* The process #process
+ /program/main/main-process
> validate-args
> read-input
> compute-answer
> write-answer
= Commandline arguments validation #validate-args
As it is normal in a program with command-line interface, our
first job shall be to validate the arguments that we are given.
= Read the input #read-input
Given that we have a valid argument giving us the input file name.
We will try to open the given file and get it's data (ala Succ).
This fragment's job will imply verifying the existence and syntax of
the given file.
= Compute the answer #compute-answer
With the information about the graph we will now be ready to compute the
number of connected components of it.
= Write the answer #write-answer
Given the answer obtained in the previous fragment. We will present it to
the user through stdout.
\paragraph{Commandline arguments validation}
To do this, it is enough to verify that we have been given one and only one user-defined commandline argument.
The first given argument in "args" is always the name of the binary, so we need to verify that the args array
is of size 2 exactly.
+ /program/main/process/validate-args
if (args.length != 2)
{
import std.stdio: stderr;
stderr.writeln("Expected an only argument giving an input file");
return;
}
addto{>validate-args}
if (args.length != 2)
{
@>validate-args
}
Thus, if we find that we didn't receive the exact required number of arguments, we simply print a message into stderr and exit.
\%TODO It may be better to have a function that prints a help string.
.
Before reading the input we must know how we will represent in-memory the given graph. It would seem intuitive to simply map
the representation that we know we have in the file (given #/about/graph-description/syntax ) and proceed from there, but for our purposes, it will
prove to be more convenient to represent it other way.
* Graph representation #graph
There are 3 standard ways of representing a graph, suppose we have $n$ nodes and $m$ edges:
\paragraph{Adjacency matrix representation}
In this representation we have an $n \times n$ matrix such that $a_{ij} = 1$ if $i$ and $j$ are connected directly by an edge and $a_{ij} = 0$ otherwise.