我有以下代码,它经历了逐行读取文件的阶段,将其转换为列表,编辑该列表中的字段,但最后一个阶段是重写原始文件,包含此编辑。
我会对能够提供最简单/最简单修复(不使用pandas / numpy等)和使用提供的原始代码的答案感兴趣。
我当前的算法包括必须创建一个包含所有记录的新文件,但链接到此用户名的记录除外,然后在此列表中写入。这看似艰巨而且没必要。任何解决方案都感激不尽!
评论中更清楚地解释了这项任务:
代码
""" ==============TASK
ALLOW THE USER TO CHANGE OR EDIT THEIR PASSWORD
1. Search for any given username
2. Edit the password field for that given username
3. Save the new updated username and password to file / updated
"""
import csv
def main():
#1. This code snippet asks the user for a username and then allows them to change password for that record
updatedlist=[]
with open("fakefacebook.txt",newline="") as f:
reader=csv.reader(f)
print("CHANGE PASSWORD?!")
username=input("Enter the username for the required user:")
for row in reader: #for every row in the file
for field in row:
if field==username: #if a field is == to the required username
updatedlist.append(row) #add each row, line by line, into a list called 'udpatedlist'
newpassword=input("Enter new password")
#print(updatedlist[0][1]) (this is how we get at the password field, noting it is a nested list)
updatedlist[0][1] = newpassword #set the field for password to the new password
updatepassword(updatedlist)
def updatepassword(updatedlist):
with open("fakefacebook.txt","w",newline="") as f:
Writer=csv.writer(f)
Writer.writerows(updatedlist)
print("File has been updated")
main()
注意:目前代码只允许用户更改密码(这在列表中已更改)。该列表仅包含该用户的记录。它会将此单个记录(使用更改的密码)覆盖到文本文件,而不是所需的内容(原始文件内容+仅此编辑)
文件内容
username,password,email,no_of_likes
marvR,pass123,marv@gmail.com,400
smithC,open123,cart@gmail.com,200
blogsJ,2bg123,blog@gmail.com,99
必需输出
如果测试:marvR 将密码更改为:boo123
新文件应包含:
username,password,email,no_of_likes
marvR,**boo123**,marv@gmail.com,400
smithC,open123,cart@gmail.com,200
blogsJ,2bg123,blog@gmail.com,99
对于教授初学者的最佳方法的任何评论/解释也将不胜感激。奇怪的是,Python还没有开发出某种类型的模块来使编辑文件中的字段比这个3步算法更容易,这对于初学者来说真是太艰难了(我说的是13-14岁的孩子)来锻炼< / p>
答案 0 :(得分:3)
到目前为止,尚未解决其中一个相关问题。您已经提到&#34;教学初学者&#34; ,所以请将此答案视为其他答案的补充。
编辑密码文件这样的文件时,不应像在此处显示的代码中那样覆盖原始文件。如果写入由于任何原因(包括磁盘已满或断电)而失败,则可能会丢失数据。更重要的是,文件应始终处于一致状态,即任何人都不应该看到部分写入的文件。
实现这一目标:
with
自动执行此操作),您甚至可能要调用os.fsync()
os.replace()
(Python 3.3+)将临时文件重命名为原始文件。原子操作只需一步即可完成。更新:修改了updatepassword
代码并附带了一些注释:
FILE = "fakefacebook.txt"
def updatepassword(updatedlist):
# create the tempfile in the same directory (os.replace requires the same filesystem)
tempfile = FILE + ".tmp"
try:
# "x" = fail with the FileExistsError if the file exists already
with open(tempfile, "x", newline="") as f:
writer = csv.writer(f)
writer.writerows(updatedlist)
f.flush() # flush the internal buffers
os.fsync(f.fileno()) # write to disk
os.replace(tempfile, FILE)
print("File has been updated")
tempfile = None # tempfile renamed
except FileExistsError:
print("Another copy is running or stale tempfile exists")
tempfile = None # tempfile does not belong to this process
except OSError as err:
print("Error: {}".format(err))
finally:
if tempfile:
try:
os.unlink(tempfile)
except OSError:
print("Could not delete the tempfile")
答案 1 :(得分:1)
@ VPfB的回答也应该考虑在内。
这只是更新记录而不替换文件中现有记录的答案。但是有更好的方法可以做到这一点。
import csv
def main():
#1. This code snippet asks the user for a username and then allows them to change password for that record
updated_list = []
cached_list = []
with open("fakefacebook.txt", newline="") as f:
reader = list(csv.reader(f)) # Convert iterable to list to make things easier.
print("CHANGE PASSWORD?!")
username=input("Enter the username for the required user: ")
cached_list = reader # store copy of data.
for row in reader: #for every row in the file
for field in row:
if field == username: #if a field is == to the required username
updated_list.append(row) #add each row, line by line, into a list called 'udpated_list'
newpassword = input("Enter new password: ")
#print(updatedlist[0][1]) (this is how we get at the password field, noting it is a nested list)
updated_list[0][1] = newpassword #set the field for password to the new password
update_password(updated_list, cached_list)
def update_password(updated_list, cached_list):
for index, row in enumerate(cached_list):
for field in row:
if field == updated_list[0]:
cached_list[index] = updated_list # Replace old record with updated record.
with open("fakefacebook.txt","w", newline="") as f:
Writer=csv.writer(f)
Writer.writerows(cached_list)
print("File has been updated")
main()
答案 2 :(得分:0)
您可以在此处使用字典而不是列表数据结构。字典将值映射到用户定义的键而不是整数。因此,您可以简单地将整个文件存储在数据结构中,并使用随机访问来更改所需的用户名,而不是通过遍历列表来查找正确的用户名:
private class GetBookInfo extends AsyncTask <View, Void, Integer> {
@Override
protected Integer doInBackground(View... urls) {
// make Call to the url
makeCall("https://www.googleapis.com/books/v1/volumes?" +
"q=isbn:" + ean_content + "&AIzaSyCpYez5556X4UzPV6rF4kkspj9DsCs_Q_c");
//print the call in the console
System.out.println("https://www.googleapis.com/books/v1/volumes?" +
"q=isbn:" + ean_content + "&AIzaSyCpYez5556X4UzPV6rF4kkspj9DsCs_Q_c");
return null;
}
@Override
protected void onPreExecute() {
// we can start a progress bar here
}
@Override
protected void onPostExecute(Integer result) {
String ruta = save_cover(getApplicationContext(), title, book_cover);
Intent intent = new Intent(MainActivity.this, Spreadsheets.class);
// intent.putExtra(title,title);
// intent.putExtra(author,authors);
// intent.putExtra(date,date);
// intent.putExtra(category,categories);
// intent.putExtra(description,description);
//finish();
startActivity(intent);
finish();
}
}
public void makeCall(String stringURL) {
URL url = null;
BufferedInputStream is = null;
JsonReader jsonReader;
try {
url = new URL(stringURL);
} catch (Exception ex) {
System.out.println("Malformed URL");
}
try {
if (url != null) {
HttpsURLConnection urlConnection = (HttpsURLConnection) url.openConnection();
is = new BufferedInputStream(urlConnection.getInputStream());
}
} catch (IOException ioe) {
System.out.println("IOException");
}
if (is != null) {
try {
jsonReader = new JsonReader(new InputStreamReader(is, "UTF-8"));
jsonReader.beginObject();
while (jsonReader.hasNext()) {
String name = jsonReader.nextName();
if (name.equals("title")) {
title = jsonReader.nextString();
}
else if (name.equals("authors")) {
authors = jsonReader.nextString();
}
else if (name.equals("publishedDate")) {
date = jsonReader.nextString();
}
else if (name.equals("categories")) {
categories = jsonReader.nextString();
}
else if (name.equals("description")) {
description = jsonReader.nextString();
}
// else if (name.equals("averageRating")) {
// rating = jsonReader.nextString();
// }
else if (name.equals("thumbnail")) {
image = jsonReader.nextString();
book_cover = download_cover(image);
}
else {
jsonReader.skipValue();
}
}
jsonReader.endObject();
}
catch (Exception e) {
System.out.println("Exception");
}
}
}
您必须修改from collections import defaultdict
def main():
inputDict=defaultdict(string)
with open("fakefacebook.txt",newline="") as f:
reader=csv.reader(f)
for row in reader: #for every row in the file
line = row.split(',') #splits the CSV row into an array
#sets the key to be the username, and everything else
#to be value, and then adds it to inputDict
inputDict[line[0]] = line[1:]
print("CHANGE PASSWORD?!")
username=input("Enter the username for the required user:")
newpassword=input("Enter new password")
if (inputDict[username]):
inputDict[username] = newpassword
else: print("No such username!")
updatepassword(inputDict)
以使用dictionary。