So I have this code in python that writes some values to a Dictionary where each key is a student ID number and each value is a Class (of type student) where each Class has some variables associated with it. ‘
Code
try:
if ((str(i) in row_num_id.iterkeys()) and (row_num_id[str(i)]==varschosen[1])):
valuetowrite=str(row[i])
if students[str(variablekey)].var2 != []:
students[str(variablekey)].var2.append(valuetowrite)
else:
students[str(variablekey)].var2=([valuetowrite])
except:
two=1#This is just a dummy assignment because I #can't leave it empty... I don't need my program to do anything if the "try" doesn't work. I just want to prevent a crash.
#Assign var3
try:
if ((str(i) in row_num_id.iterkeys()) and (row_num_id[str(i)]==varschosen[2])):
valuetowrite=str(row[i])
if students[str(variablekey)].var3 != []:
students[str(variablekey)].var3.append(valuetowrite)
else:
students[str(variablekey)].var3=([valuetowrite])
except:
two=1
#Assign var4
try:
if ((str(i) in row_num_id.iterkeys()) and (row_num_id[str(i)]==varschosen[3])):
valuetowrite=str(row[i])
if students[str(variablekey)].var4 != []:
students[str(variablekey)].var4.append(valuetowrite)
else:
students[str(variablekey)].var4=([valuetowrite])
except:
two=1
‘
The same code repeats many, many times for each variable that the student has (var5, var6,….varX). However, the RAM spike in my program comes up as I execute the function that does this series of variable assignments.
I wish to find out a way to make this more efficient in speed or more memory efficient because running this part of my program takes up around half a gig of memory. 🙁
Thanks for your help!
EDIT:
Okay let me simplify my question:
In my case, I have a dictionary of about 6000 instantiated classes, where each class has 1000 attributed variables all of type string or list of strings. I don’t really care about the number of lines my code is or the speed at which it runs (Right now, my code is at almost 20,000 lines and is about a 1 MB .py file!). What I am concerned about is the amount of memory it is taking up because this is the culprit in throttling my CPU. The ultimate question is: does the number of code lines by which I build up this massive dictionary matter so much in terms of RAM usage?
My original code functions fine, but the RAM usage is high. I’m not sure if that is "normal" with the amount of data I am collecting. Does writing the code in a condensed fashion (as shown by the people who helped me below) actually make a noticeable difference in the amount of RAM I am going to eat up? Sure there are X ways to build a dictionary, but does it even affect the RAM usage in this case?
Edit: The suggested code-refactoring below won’t reduce the memory consumption very much. 6000 classes each with 1000 attributes may very well consume half a gig of memory.
You might be better off storing the data in a database and pulling out the data only as you need it via SQL queries. Or you might use shelve or marshal to dump some or all of the data to disk, where it can be read back in only when needed. A third option would be to use a numpy array of strings. The numpy array will hold the strings more compactly. (Python strings are objects with lots of methods which make them bulkier memory-wise. A numpy array of strings loses all those methods but requires relatively little memory overhead.) A fourth option might be to use PyTables.
And lastly (but not leastly), there might be ways to re-design your algorithm to be less memory intensive. We’d have to know more about your program and the problem it’s trying to solve to give more concrete advice.
Original suggestion:
PUT_AN_EXPLICT_EXCEPTION_HEREshould be replaced with something likeAttributeErrororTypeError, orValueError, or maybe something else.It’s hard to guess what to put here because I don’t know what kind of values the variables might have.
If you run the code without the
try...exceptionblock, and your program crashes, take note of the traceback error message you receive. The last line will say something likeTypeError: ...In that case, replace
PUT_AN_EXPLICT_EXCEPTION_HEREwithTypeError.If your code can fail in a number of ways, say, with
TypeErrororValueError, then you can replacePUT_AN_EXPLICT_EXCEPTION_HEREwith(TypeError,ValueError)to catch both kinds of error.Note: There is a little technical caveat that should be mentioned regarding
row_num_id.get(str(i))==varschosen[1]. The expressionrow_num_id.get(str(i))returnsNoneifstr(i)is not inrow_num_id.But what if
varschosen[1]isNoneandstr(i)is not inrow_num_id? Then the condition isTrue, when the longer original condition returnedFalse.If that is a possibility, then the solution is to use a sentinal default value like
row_num_id.get(str(i),object())==varschosen[1]. Nowrow_num_id.get(str(i),object())returnsobject()whenstr(i)is not inrow_num_id. Sinceobject()is a new instance ofobjectthere is no way it could equalvarschosen[1].