Robert Garza-Altuna
Robert Garza-Altuna

Reputation: 117

Local variable referenced before assignment error python 3

For context I am trying to create a program that will calculate the mass of any given compound.

When I run the following code, I recieve this error:

Traceback (most recent call last):
  File "/Users/robertg21/PycharmProjects/MolecularMassCalculator/MolecularMassCalculator.py", line 37, in <module>
    formula.insert(return_index(), "+")
  File "/Users/robertg21/PycharmProjects/MolecularMassCalculator/MolecularMassCalculator.py", line 31, in return_index
    return pos
UnboundLocalError: local variable 'pos' referenced before assignment

Can anyone help. Here is my code:

# This program calculates the molar mass of a compound given its chemical formula and number of atoms.

relative_atomic_mass = {'H': 1.00794, 'He': 4.002602, 'Li': 6.941, 'Be': 9.012182, 'B': 10.811, 'C': 12.0107, 'N': 14.0067,
              'O': 15.9994, 'F': 18.9984032, 'Ne': 20.1797, 'Na': 22.98976928, 'Mg': 24.305, 'Al': 26.9815386,
              'Si': 28.0855, 'P': 30.973762, 'S': 32.065, 'Cl': 35.453, 'Ar': 39.948, 'K': 39.0983, 'Ca': 40.078,
              'Sc': 44.955912, 'Ti': 47.867, 'V': 50.9415, 'Cr': 51.9961, 'Mn': 54.938045,
              'Fe': 55.845, 'Co': 58.933195, 'Ni': 58.6934, 'Cu': 63.546, 'Zn': 65.409, 'Ga': 69.723, 'Ge': 72.64,
              'As': 74.9216, 'Se': 78.96, 'Br': 79.904, 'Kr': 83.798, 'Rb': 85.4678, 'Sr': 87.62, 'Y': 88.90585,
              'Zr': 91.224, 'Nb': 92.90638, 'Mo': 95.94, 'Tc': 98.9063, 'Ru': 101.07, 'Rh': 102.9055, 'Pd': 106.42,
              'Ag': 107.8682, 'Cd': 112.411, 'In': 114.818, 'Sn': 118.71, 'Sb': 121.760, 'Te': 127.6,
              'I': 126.90447, 'Xe': 131.293, 'Cs': 132.9054519, 'Ba': 137.327, 'La': 138.90547, 'Ce': 140.116,
              'Pr': 140.90465, 'Nd': 144.242, 'Pm': 146.9151, 'Sm': 150.36, 'Eu': 151.964, 'Gd': 157.25,
              'Tb': 158.92535, 'Dy': 162.5, 'Ho': 164.93032, 'Er': 167.259, 'Tm': 168.93421, 'Yb': 173.04,
              'Lu': 174.967, 'Hf': 178.49, 'Ta': 180.9479, 'W': 183.84, 'Re': 186.207, 'Os': 190.23, 'Ir': 192.217,
              'Pt': 195.084, 'Au': 196.966569, 'Hg': 200.59, 'Tl': 204.3833, 'Pb': 207.2, 'Bi': 208.9804,
              'Po': 208.9824, 'At': 209.9871, 'Rn': 222.0176, 'Fr': 223.0197, 'Ra': 226.0254, 'Ac': 227.0278,
              'Th': 232.03806, 'Pa': 231.03588, 'U': 238.02891, 'Np': 237.0482, 'Pu': 244.0642, 'Am': 243.0614,
              'Cm': 247.0703, 'Bk': 247.0703, 'Cf': 251.0796, 'Es': 252.0829, 'Fm': 257.0951, 'Md': 258.0951,
              'No': 259.1009, 'Lr': 262, 'Rf': 267, 'Db': 268, 'Sg': 271, 'Bh': 270, 'Hs': 269, 'Mt': 278,
              'Ds': 281, 'Rg': 281, 'Cn': 285, 'Nh': 284, 'Fl': 289, 'Mc': 289, 'Lv': 292, 'Ts': 294, 'Og': 294,
}

chemical_formula = input("Enter chemical formula, or press return to quit: ")
formula = list(chemical_formula)


def return_index():
    for element in formula:
        if element.isalpha() and element.isupper():
            pos = formula.index(element)
        return pos


# Inserts + before for each Capitalized letter in list.
for item in formula:
    if item.isalpha() and item.isupper():
        formula.insert(return_index(), "+")

Thanks.

Upvotes: 1

Views: 642

Answers (1)

Reut Sharabani
Reut Sharabani

Reputation: 31349

The message is clear:

if element.isalpha() and element.isupper():
    pos = formula.index(element)
return pos

What if the condition is not met?

You have to set pos to some value if you want to return or reference it.

Maybe you want:

if element.isalpha() and element.isupper():
    return formula.index(element)

This will short-circuit the loop, creating a simple "find" operation. If the condition is never met it will return None.

A more functional approach which is expressing the same constructs you created can be used. Specifically you created find or first which is of the form:

for item in coll
    if pred(item)
        return item

So you can define pred and use next(filter(pred, coll)) to the same effect:

def is_element(element):
    return item.isalpha() and item.isupper()

And then:

element = next(filter(is_element, formula))

Upvotes: 2

Related Questions