首页 > 解决方案 > 将变量插入字符串时如何保留换行符?

问题描述

一般来说,我对编码很陌生(至少无论如何都想擅长它),这是我的第一个 StackOverflow 问题,尽管多年来一直潜伏在这里。如果措辞不好,我深表歉意。

我之前编写了一个脚本,该脚本将使用 MIMEMultipart 自动以文本和 html 向人们发送电子邮件,用于类似的设备,但在那个设备中,我有一个 CSV,我从中提取数据。使用这个新脚本,我获取串行数据并对其执行 ser.readline。这些信息是通过 USB 到 RS232 电缆从设备收集的(我有一个树莓派,它基本上模拟打印机,接收数据,然后通过电子邮件发送数据)。

在之前的脚本中,当我只是从 csv 中通过电子邮件发送数据时,我可以在字符串中使用 {table} 将表格带入电子邮件正文,并且一切正常。但是,对于新的串行脚本,当我使用 {ReportData} 再次将串行数据作为字符串插入电子邮件正文时,换行符不会以相同的方式传递。

这是代码的净化版本:

import serial
import datetime
import time
import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart

# logging settings
logging.basicConfig(filename='SerialError.log', level=logging.INFO)
f = open("Serial.log", "a+")
print(datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S"))
f.write(datetime.datetime.now().strftime(
    "%Y-%m-%d %H:%M:%S") + "  Program Starting.....\r\n")
print("Program Starting.....")
time.sleep(3)

# define variables to open serial connection and report the data
ser = serial.Serial(port='COM7', baudrate=9600, timeout=None)
ReportData = ""

# write to serial log before beginning loop
f.write(datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S") +
        "  Flushing Serial Buffer.......\r\n")
print("Flushing Serial Buffer.......")
ser.reset_input_buffer()
f.write(datetime.datetime.now().strftime(
    "%Y-%m-%d %H:%M:%S") + "  Waiting for serial data!\r\n")
print("Waiting for serial data!")
f.close()

# loop to read and process data as received via rs232
while True:
    try:
        ReceiveData = ser.readline().decode('UTF-8', 'ignore')
        ReportData += ReceiveData

        # find certain string and consider that the last line
        if ReceiveData.find("string parse for last line i want to read") != -1:
            print(ReportData.encode('UTF-8', 'ignore'))

            # Text email formatting
            text = """Report Data:
            {ReportData}
            
            """

            # HTML email formatting
            html = """<html><body><p>Report Data:</p>
            {ReportData}

            </body></html>"""

            # email results to recipients
            smtpserver = smtplib.SMTP('mail.ourserver.com', 25)
            smtpserver.ehlo()
            smtpserver.starttls()
            smtpserver.ehlo
            message = MIMEMultipart('alternative', None, [
                                    MIMEText(text), MIMEText(html, 'html')])
            message['Subject'] = 'Report Data'
            message['From'] = "ReportData<donotreply@mail.ourserver.com>"
            message['To'] = "user@mail.ourserver.com"
            smtpserver.sendmail(
                message['From'], message['To'].split(","), message.as_string())
            smtpserver.close()

    # print errors upon exception
    except Exception as error:
        logging.exception(str(error))
        print("Error: " + str(error))
        exit()

如果我做:

print(ReportData.encode('UTF-8', 'ignore'))

我得到:

A 1
B 2
C 3
D 4

这就是我想要的,但如果我在电子邮件上方运行我的代码,我会得到如下所示:

A 1 B 2 C 3 D 4

这表明我没有将换行符带入其中。如果我只是打印

我尝试了其他写出格式的方法,例如:

# Text email formatting
text = "Report Data:\n", ReportData.encode('UTF-8','ignore'), "\n"

# HTML email formatting
html = "<html><body><p>Report Data:</p>\n", ReportData.encode('UTF-8','ignore'), "\n</body></html>"

但这会导致:

Error: 'tuple' object has no attribute 'encode'

我添加 .encode 来尝试修复它的原因是我猜编码没有通过,这导致换行符被剥离。

我还尝试使用 f-string 将变量读入字符串:

# Text email formatting
text = f"Report Data:\n", ReportData.encode('UTF-8','ignore'), "\n"

# HTML email formatting
html = f"<html><body><p>Report Data:</p>\n", ReportData.encode('UTF-8','ignore'), "\n</body></html>"

但它给了我同样的错误。然后我尝试了我写它的原始方式,但使用了一个 f-string:

# Text email formatting
text = f"""Report Data:
{ReportData}
            
"""

# HTML email formatting
html = f"""<html><body><p>Report Data:</p>
{ReportData}

</body></html>"""

它给了我原来的结果,只是“A 1 B 2 C 3 D 4”,等等......不是我要找的。

我只能猜测在阅读了几天后我忽略了一些简单的答案,但我似乎还找不到。

是否将输入从 ser.readline 转换为变量,因此将其用作字符串中的变量会阻止换行符通过?如果是这样,解决这个问题的最佳方法是什么?

标签: python

解决方案


在这里将一些答案放在评论中,因为代码阅读更容易。

由于 martineau 建议<br>在原始文本中存在换行符的地方需要标记。(需要注意的是,在 HTML 中不能只使用原始文本)

我会report_data为 html 输出单独构建一个。

def create_email_html(report_data):
    report_data_html = '<br>\n'.join(report_data.split('\n'))
    return f"""
    <html>
    <body>
    <p>Report Data:</p>
    {report_data_html}
    </body>
    </html>"""

html = create_email_html(ReportData)

会产生

<html>
<body>
<p>Report Data:</p>
A 1<br>
B 2<br>
C 3<br>
D 4<br>
</body>
</html>

在浏览器(或本例中的电子邮件)中呈现的内容将看起来像您期望的那样。


推荐阅读