联系方式

您当前位置:首页 >> C/C++编程C/C++编程

日期:2024-10-15 09:25

Lab 3: The Resistor Network Program

Objectives

The objectives of this assignment are for you to practice: (1) the use of C++ I/O streams, including

error handling, (2) dynamic allocation and de-allocation of one-dimensional arrays, (3) solve a

problem using object-oriented programming. You will do so through the design of a program that

parses circuit building commands from the standard input, displaying appropriate error messages if

necessary, and by creating and maintaining resistors in a circuit. You will be asked to calculate the

voltage at certain nodes.

Problem Statement

Your task is to implement a program for storing resistors in a circuit. The circuit, or network, is define

by a number of nodes, each node being connected to one or more resistors. Each resistor has

certain values associated with it, including: resistance value, text name, and the IDs of the two node

(endpoints) it connects. The diagram below (Fig 1) illustrates an example. Here R1 connects nodes

and 2, R2 does the same, and R3 connects nodes 1 and 3. The nodes could be strips on a

breadboard for example, with resistors plugged in to some nodes.

The program you will write is similar to the "input-and-store-the-network" portion of real programs

used to simulate electric circuits, and programs that control the robots that automatically insert

resistors connecting the appropriate points (nodes) on circuit boards.

Fig. 1: Example of a circuit

Circuits are input using a text-based user interface. Commands are defined to allow the user to

insert, modify, delete and print resistors, and to set the maximum size of the network. Based on thecommands given, you will need to maintain two arrays, one of Resistors and one of Nodes, to which

you can add and modify elements.

Specififications

It is important that you follow the specifications below carefully. Where the specification says shall o

must (or their negatives), following the instruction is required to receive credit for the assignment. I

instead it says may or can, these are optional suggestions. The use of should indicates a

recommendation; compliance is not specifically required. However, some of the recommendations

may hint at a known-good way to do something or pertain to good programming style.

Example input and output for the program are provided at the end of this document for your

convenience. They do not cover all parts of the specification. You are responsible for making sure

your program meets the specification by reading and applying the description below.

Coding Requirements

1. The code you will write shall be contained in only three source files named main.cpp ,

Resistor.cpp and Node.cpp . Skeletons of the files are released with the assignment's zip file.

The zip file also contains two header files: Resistor.h , Node.h . These files are NOT to be

modified in any way. Modifying these files often results in a mark of 0 for the assignment.

However, you may make use of helper functions to split up the code for readability and to make

easier to re-use. These functions (and their prototypes) must be in one of the aforementioned

source files. That is, you must not add any new header or source files.

2. Input and output must be done only using the C++ standard library streams cin and cout .

3. The stream input operator >> and associated functions such as fail() and eof() shall be use

for all input.  C-style IO such as printf and scanf shall not be used.

4. Strings shall be stored using the C++ library type string , and operations shall be done using its

class members, not C-style strings.  

5. C-library string-to-integer conversions (including but not limited to atoi , strtol , etc) shall not b

used.

The commands your program must accept are those specified in Table 2. In addition to checking the

commands are syntactically valid, in this lab you will store the resistors and their connectivity in two

arrays (of Resistors and of Nodes) and output the correct data in response to the print commands.

The first line of input to your program will set the maximum allowed node number and the maximum

number of resistors:

maxVal maxNodeNumber maxResistors

where both maxNodeNumber and maxResistors are positive integers. You should store these two valu

in variables in your program and also use them to specify the appropriate size for the resistor and

node arrays your program will require.

Use the new[] operator to allocate the node and resistor arrays. If your program is sent several

maxVal commands during one run, you will have to ensure that for any maxVal command after the

first one you delete any already allocated Node and Resistor arrays, and then allocate the arrays tothe specified new sizes. The node and resistor arrays and any other necessary variables should be

initialized to the empty network state (no resistors stored) after any maxVal command.

The resistor index shall be the position of the resistor in the resistor array, starting at 0. The program

shall be able to accommodate exactly maxResistors , as specified in the maxVal command. The

nodes to which these resistors can be attached are identified by an integer ID, which corresponds to

their index in the node array. You must be able to store data for node indices from 0 to maxNodeNumbe

- 1 , inclusive, where maxNodeNumber is set by the maxVal command. Nodes entered by the user

range from 1 to maxNodeNumber , but are transferred to indices by subtracting 1. Each node can

accommodate only a limited number of resistors attaching to it (think of the number of holes in a

breadboard strip); this is a #define constant MAX_RESISTORS_PER_NODE which is set to 5 for this lab.

The node and resistor arrays shall be sized according to the maxVal command, start empty and be

modified based on user commands. The command parser must check its input for validity, catching

and reporting the errors. Lastly, successful commands must update the data structures and produce

output as discussed below. Additional restrictions on how the program is to be written and organized

are detailed below.

Command Line Input

Input will be given one command on one line at a time. The entire command must appear on one lin

All input must be read using the C++ standard input cin .

The program shall indicate that it is ready to receive user input by prompting with three greater-than

sign followed by a single space ( >>> ); see end of this page for an example.

Input shall always be accepted one line at a time, with each line terminated by a newline character

newline character is input by pressing Enter . If there is an error encountered when parsing a line,

the program shall print an error message (see Section~ Table 3), the line shall be discarded, an

processing shall resume at the next line. The program shall continue to accept and process input un

an end-of-file ( eof ) condition is received eof is automatically provided when input is redirected fro

a file. It can also be entered at the keyboard by pressing Ctrl-D. Each line of valid input shall start

with a command name, followed by zero or more arguments, each separated by one or more space

characters. The number and type of arguments accepted depend on the command. The arguments

and their permissible types/ranges are shown below in Table 1.

Argument Description, type, and range

name

A string consisting of any non-whitespace characters. Whitespace characters are tab,

space, newline, and related characters which insert "white space''; they mark the

boundaries between values read in by operator<<, except the string all which we

reserve for special use characters.

nodeid Node ID, an integer ranging from 1 to maxNodeNumber

resistance Positive resistance value (type  double )

Table 1: Acceptable input argumentsThe valid commands, their arguments, and their output if the command and its arguments are all

legal are shown below in Table 2. The program shall verify that the command and arguments are

correctly formatted and within range, and that a command is followed by the correct number of

arguments, on the same line. The handling of command names shall be case-sensitive. If there is a

error, a message shall be displayed as described in Table 3. Otherwise, a successful command

produces a single line of output on the C++ standard output, cout , as shown in Table 2. The values

in italics in Table 2 must be replaced with the values given by the command argument. Strings must

be reproduced exactly as entered. Where nodeids are printed, they shall appear on the order entere

in the command. Resistance values shall be printed in fixed point format with exactly two decimal

places, regardless of the number entered.

The valid commands, their arguments, and their output if the command and its arguments are all

legal are shown below in Table 2.

Command Arguments Output if valid Action if valid

maxVal

maxNodeNumber

maxResistors

New network: max node

number is

maxNodeNumber; max

resistors is maxResistors

Node array created with new[] to

store nodes from 0 to

maxNodeNumber and Resistor

array created to store up to

maxResistors; the network is

initialized to empty (no

resistors).

insertR

name resistance

nodeid nodeid

Inserted: resistor name

resistance Ohms nodeid ->

nodeid

Adds 1 resistor to Resistor array

and updates 2 entries in Node

array

modifyR name resistance

Modified: resistor name from

resistance-old Ohms to

resistance Ohms

Updates one entry in Resistor

array

printR name

Print: resistor info (see

below). Print function is

implemented in Resistor.cpp

already.

No data changed

deleteR all Deleted: all resistors

All resistors cleared and Node

array updated so we have an

empty network

setV nodeid voltage

Set: node nodeid to

voltage Volts

Updates the specified node

voltage data member. This

corresponds to connecting this

node to a voltage source.

solve Solve: node voltage info

(see below) for bonus

Runs a numerical solution

technique to determine themarks. voltage of all nodes that do not

have their voltages set for bonus

marks.

Table 2: Summary of Commands and Actions

Printing resistance information

Resistor information shall be printed as:

name resistance Ohms nodeid -> nodeid

The two nodes shall appear in the order in which they were presented in the insertR command tha

created the resistor. The resistance field shall show two digits after the decimal place (use

setprecision function in iomanip library). printR command requires you to call the print functio

implemented in Resistor.cpp.

Error Checking

Error message Cause

invalid command

The first word entered does not match one of the

valid commands in Table 2

too few arguments

Fewer arguments were given than expected for

a command

invalid argument

The argument is not of the correct type. For

example, a floating point number may have been

entered instead of an integer nodeid or a string

other than all may have been entered where a

nodeid or all is expected.

negative resistance

The resistance value is strictly less than zero

(0.0 is permitted)

resistor name cannot be the keyword "all"

A resistor name of all was specified to insertR

or modifyR

resistor name already exists A resistor name of same name exists at insertR

resistor name not found

The resistor name entered at modifyR or printR

is not found

node value is out of permitted range 1 -

maxNodeNumber

An integer nodeid value has been provided that

is out of range

both terminals of resistor connect to same node

The two nodes to which a resistor connects

cannot be the same

Table 3: List of errors to be reported, in priority orderThe program must check that the input is valid. It must be able to identify and notify the user of the

following input errors, in order of priority. Where multiple errors exist on one input line, only one

should be reported: the one that occurs first as the line is read from left to right. Errors shall cause a

message to be printed to cout , consisting of the text "Error:'' followed by a single space and the err

message from Table 3. Error message output must comply exactly (content, case, and spacing) with

the table below to receive credit. There are no trailing spaces following the text.

The program is not required to deal with errors other than those listed in Table 3.

The Resistor  Class

Each object of the Resistor class holds information about one resistor in the network. In

Resistor.cpp, you must implement all the functions indicated in Resistor.h. Each resistor object has

string ( name ), a resistance value in Ohms ( resistance ), and the IDs (indices in the node array) of th

two nodes to which it is connected ( endpointNodeIDs[2] ). There is are also a constructor and

destructor function to initialize the class, a function to print the values, and functions to set/get the

data members of the class. The program shall store all resistors in an array, starting at element 0 fo

the first resistor added and incrementing from there. When adding a Resistor to a Node , the resist

shall be referred to by its position in the resistor array, also known as its resistorID. Be mindful of

what is an index and what is the node number. The Resistor.h file has multiple comments to help yo

implement the member functions.

The Node Class

The starter code for class Node is in Node.h and Node.cpp. The variable numRes is intended to stor

the number of resistors currently attached to the node. It should start at zero, and increment each

time a resistor is added. The Node.h file has multiple comments to help you implement the member

functions in Node.cpp.

Bonus: The solve command

For 1% of the course grade bonus, the solve command first determines the voltage of every node,

and then it prints the voltage of every node in ascending node order. To find the voltage at every no

in a network, we can follow the iterative procedure below. The iterative procedure we are using to

solve for the node voltages is called the Gauss-Seidel method. It is a very general method of solving

linear systems of equations, and is used to solve circuits and other systems of equations with

thousands or even millions of unknowns.

Initialize the voltage of all nodes without a specified (setV) voltage to 0.

while (some node’s voltage has changed by more than MIN_ITERATION_CHANGE) {

   for (all nodes without a set voltage) {

       set voltage of node according to Eq. 3

   }}

The voltage of a node is computed from the voltages of its neighbours according to Kirchoff’s curren

equation, which states that the total current entering or leaving a node must be 0. Consider node_0

below, which is surrounded by 3 resistors to 3 other nodes, as shown in Fig. 2. The total current

entering node_0 must be:

We can rewrite this using the fact that the current through each resistor is simply the voltage across

divided by its resistance.

Rearranging and solving for the voltage at node_0, , yields:

You can safely assume that at two nodes will have their voltage set before typing the solve

command.

Fig 2: A portion of a resistor network. Node 0 is connected to three other nodes as shown.

The iterative procedure above determines what voltage a node must have for it to balance the curre

coming through all the resistors connected to it. Once we update the voltage of some node (say nod

#0), however, the voltage of other nodes (e.g., node #1) connected to that node may have to chang

to ensure the current flowing into them is in balance. Hence we iterate through all the nodes,

repeatedly updating the voltages of those that do not have a fixed ( setV ) voltage according to Eq. 3

until no voltage changes much --- at that point we have converged to a solution.

For this lab, you should iterate until no node changes by more than MIN_ITERATION_CHANGE , which yo

should define to be 0.0001.Once your solver has converged, print (to cout ) the voltage of every node that has at least one

resistor connected to it, in ascending node order. For the example shown in Fig. 3, the solve

command would print:

Solve:

Node 2: 0.30 V

Node 3: -0.02 V

Node 4: -0.60 V

Node 5: 0.52 V

Node 6: 1.00 V

Fig 3: A more complex resistor network. Two nodes have been connected to voltage source

with setV, and our program will solve for the other 3 voltages.

Hints

1. You can check a stream for end-of-file status using the eof member function.

2. To save typing, you can create one or more test files and pipe them to your program. You can

create a text file using a text editor (try gedit , gvim , or the VSCode editor). If your file is called

test , you can then send it to your program by typing circuit < test . Building a good suite of

test cases is important when developing software.

3.  If you want to look ahead ("peek'') at what character would be read next without actually reading

it, peek() does that. For instance, if you type "Hello'' then each time you run peek() you will ge

'H'. If you read a single character, it will return 'H' but then subsequent calls to peek() will return

'e'.

4. When interacting with your program from the keyboard, Ctrl-D will send an end-of-file (eof)

marker.

5. Reading from cin removes leading whitespace. When reading strings, it discards all whitespac

characters up to the first non-whitespace character, then returns all non-whitespace characters

until it finds another whitespace. For integers (numbers), it skips whitespace and reads to the fir

non-digit (0-9) character.6. Remember you can use the debugger to pause the program, step through it, and view variables

(including strings).

7. If you decide to pass the string stream you created to a function, remember that string streams

(and other types of streams for that matter) can only be passed by reference, not by value.

8. IO manipulators (see header file <iomanip> ) can be used to control the appearance of output. Yo

may need at least setprecision .

9. You will need to store one array of Nodes and one of Resistors - global scope is acceptable for

program of this scale. You will also need a variable such as resistorsCount to count the numbe

of resistors already added, and will need variables to store maxNodeNumber and maxResistors .

10. For output formatting (padding, left/right justify, precision control) you should look at the

<iomanip> header file (www.cplusplus.com/reference/iostream

(http://www.cplusplus.com/reference/iostream) is a good reference). Functions like left, right, setfi

and setw will be of great help.

11. Add features one-by-one. Even if you cannot complete every command (or detect every error),

you should still submit your program since it may pass some test cases.

12. Your program should delete all the memory it allocates with new before it exits; this is good

practice and ensures there are no memory leaks. The autotester will check if your program

deletes all the memory it should and you will lose marks if you do not. A good way to check if yo

have deleted all the memory you allocated with new is to run the valgrind memory checking

program. A tutorial on valgrind will be posted to Quercus, and you are encouraged to learn and

use this tool.

A suggested (but not mandatory) structure for your code appears in the skeleton main.cpp file

released within the assignment's zip file.

Procedure

Create a sub-directory called lab3 in your ece244-labs directory, and set its permissions so no one

else can read it. Download the lab3_release.zip file, un-zip it and place the resulting files in the lab3

directory. There are three source files in which you will add your code. The first is main.cpp in which

you will write the command parser code. The second and third files are Node.cpp and Resistor.cpp

which you will implement the classes Node and Resistor respectively. All files

are in the directory lab3_release. You must not rename these files or add more files.  

The release also contains two include files Node.h and Resistor.h. You may NOT modify these file

to add to or delete from their content. Modifying the files commonly results in a mark of zero for the

assignment. In addition, there is a Makefile to separately compile your project. Run make to compile

your code. Do not modify this file either.

The exercise command will also be helpful in testing your program. You should exercise the

executable, i.e., circuit , using the command:

~ece244i/public/exercise 3 circuitAs with previous assignments, some of the exercise test cases will be used by the autotester

during marking of your assignment. We will not provide all the autotester test cases in

exercise, however, so you should create additional test cases yourself and ensure you fully meet th

specification listed above.

Input to public test cases can be found in files named 1 , 2 , 3 and 4  in tests.zip file. Test case

1, 2, 3 don't test the solve command, but test case 4 does and counts for partial marks for the bonu

in this lab.

Reference Executable

To run the reference/solution executable, you may run the following command:

~ece244i/public/circuit-ref

Deliverables

Submit the main.cpp, Node.cpp and Resistor.cpp files as lab 3 using the command

~ece244i/public/submit 3

Example Output

>>> maxVal 1 1

New network: max node number is 1; max resistors is 1

>>> maxVal 6 7

New network: max node number is 6; max resistors is 7

>>> insertR R1 100 6 5

Inserted: resistor R1 100.00 Ohms 6 -> 5

>>> insertR R2 100 5 2

Inserted: resistor R2 100.00 Ohms 5 -> 2

>>> insertR R3 200 5 3

Inserted: resistor R3 200.00 Ohms 5 -> 3

>>> insertR R4 150 2 3

Inserted: resistor R4 150.00 Ohms 2 -> 3

>>> insertR R4 140 2 3

Error: resistor R4 already exists

>>> insertR R5 120 3 3

Error: both terminals of resistor connect to same node

>>> setV 6 1

Set: node 6 to 1.00 Volts

>>> setV 4 -0.6

Set: node 4 to -0.60 Volts

>>> printR R0

Error: resistor R0 not found

>>> print R1

Error :invalid command

>>> printR R1

Print:

R1                    100.00 Ohms 6 -> 5

>>> insertR all 2

Error: resistor name cannot be keyword "all"

>>> deleteR all

Deleted: all resistors

>>>


相关文章

版权所有:编程辅导网 2021 All Rights Reserved 联系方式:QQ:821613408 微信:horysk8 电子信箱:[email protected]
免责声明:本站部分内容从网络整理而来,只供参考!如有版权问题可联系本站删除。 站长地图

python代写
微信客服:horysk8