{"id":54,"date":"2024-02-25T06:33:34","date_gmt":"2024-02-24T22:33:34","guid":{"rendered":"http:\/\/www.uboger.com\/?p=54"},"modified":"2024-04-11T02:40:12","modified_gmt":"2024-04-10T18:40:12","slug":"%e5%88%a9%e7%94%a8-python-%e5%88%9b%e5%bb%ba%e6%96%87%e6%9c%ac%e8%bd%ac%e8%af%ad%e9%9f%b3%e5%ba%94%e7%94%a8%e7%a8%8b%e5%ba%8f","status":"publish","type":"post","link":"https:\/\/www.uboger.com\/?p=54","title":{"rendered":"\u5229\u7528 Python \u521b\u5efa\u6587\u672c\u8f6c\u8bed\u97f3\u5e94\u7528\u7a0b\u5e8f"},"content":{"rendered":"\n<h3 class=\"wp-block-heading\">\u7a0b\u5e8f\u6548\u679c\u56fe<\/h3>\n\n\n\n<figure class=\"wp-block-image size-full is-resized\"><img decoding=\"async\" src=\"http:\/\/www.uboger.com\/wp-content\/uploads\/2024\/02\/\u6355\u83b7-1-edited.png\" alt=\"\" class=\"wp-image-63\" style=\"width:564px;height:auto\" \/><\/figure>\n\n\n\n<p><\/p>\n\n\n\n<h3 class=\"wp-block-heading\">\u8f6c\u6362\u540e\u7684\u58f0\u97f3\u6548\u679c<\/h3>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u82f1\u6587\u8154\u8c03<\/li>\n<\/ul>\n\n\n\n<figure class=\"wp-block-audio\"><audio controls src=\"http:\/\/www.uboger.com\/wp-content\/uploads\/2024\/02\/output-20240225064548-en-6.mp3\"><\/audio><\/figure>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u4e2d\u6587\u8154\u8c03<\/li>\n<\/ul>\n\n\n\n<figure class=\"wp-block-audio\"><audio controls src=\"http:\/\/www.uboger.com\/wp-content\/uploads\/2024\/02\/output-20240225065106-zh-11.mp3\"><\/audio><\/figure>\n\n\n\n<p><\/p>\n\n\n\n<p>\u5728\u672c\u6559\u7a0b\u4e2d\uff0c\u6211\u4eec\u5c06\u4f7f\u7528 Python \u548c\u4e00\u4e9b\u6d41\u884c\u7684\u5e93\u6765\u521b\u5efa\u4e00\u4e2a\u7b80\u5355\u7684\u6587\u672c\u8f6c\u8bed\u97f3\u5e94\u7528\u7a0b\u5e8f\u3002\u8fd9\u4e2a\u5e94\u7528\u7a0b\u5e8f\u5c06\u5141\u8bb8\u7528\u6237\u8f93\u5165\u6587\u672c\uff0c\u5e76\u5c06\u5176\u8f6c\u6362\u4e3a\u8bed\u97f3\uff0c\u7136\u540e\u53ef\u4ee5\u9884\u89c8\u548c\u4fdd\u5b58\u751f\u6210\u7684\u8bed\u97f3\u6587\u4ef6\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\">\u51c6\u5907\u5de5\u4f5c<\/h3>\n\n\n\n<p>\u5728\u5f00\u59cb\u4e4b\u524d\uff0c\u8bf7\u786e\u4fdd\u4f60\u5df2\u7ecf\u5b89\u88c5\u4e86\u4ee5\u4e0b\u5e93\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li><code>tkinter<\/code>\uff1aPython \u7684\u6807\u51c6 GUI \u5e93\uff0c\u7528\u4e8e\u521b\u5efa\u5e94\u7528\u7a0b\u5e8f\u7684\u7528\u6237\u754c\u9762\u3002<\/li>\n\n\n\n<li><code>gtts<\/code>\uff1aGoogle Text-to-Speech\uff0c\u7528\u4e8e\u5c06\u6587\u672c\u8f6c\u6362\u4e3a\u8bed\u97f3\u3002<\/li>\n\n\n\n<li><code>pygame<\/code>\uff1a\u7528\u4e8e\u64ad\u653e\u97f3\u9891\u6587\u4ef6\u3002<\/li>\n\n\n\n<li><code>os<\/code>\uff1a\u7528\u4e8e\u6587\u4ef6\u64cd\u4f5c\u3002<\/li>\n\n\n\n<li><code>datetime<\/code>\uff1a\u7528\u4e8e\u751f\u6210\u65f6\u95f4\u6233\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u4f60\u53ef\u4ee5\u4f7f\u7528 <code>pip<\/code> \u6765\u5b89\u88c5\u8fd9\u4e9b\u5e93\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>pip install gtts pygame<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\">\u521b\u5efa\u5e94\u7528\u7a0b\u5e8f<\/h3>\n\n\n\n<p>\u6211\u4eec\u5c06\u521b\u5efa\u4e00\u4e2a\u5305\u542b\u4ee5\u4e0b\u529f\u80fd\u7684\u5e94\u7528\u7a0b\u5e8f\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u6587\u672c\u8f93\u5165\u6846\u4f9b\u7528\u6237\u8f93\u5165\u8981\u8f6c\u6362\u7684\u6587\u672c\u3002<\/li>\n\n\n\n<li>\u5355\u9009\u6309\u94ae\u9009\u62e9\u8981\u8f6c\u6362\u7684\u8bed\u8a00\u3002<\/li>\n\n\n\n<li>\u201c\u9884\u89c8\u58f0\u97f3\u201d\u6309\u94ae\u7528\u4e8e\u64ad\u653e\u6700\u8fd1\u751f\u6210\u7684\u8bed\u97f3\u6587\u4ef6\u3002<\/li>\n\n\n\n<li>\u201c\u53e6\u5b58\u4e3a\u201d\u6309\u94ae\u7528\u4e8e\u5c06\u6700\u8fd1\u751f\u6210\u7684\u8bed\u97f3\u6587\u4ef6\u53e6\u5b58\u4e3a\u7528\u6237\u6307\u5b9a\u7684\u4f4d\u7f6e\u3002<\/li>\n<\/ol>\n\n\n\n<p>\u4e0b\u9762\u662f\u5b8c\u6574\u7684\u4ee3\u7801\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>import tkinter as tk\nfrom tkinter import filedialog\nfrom gtts import gTTS\nimport pygame\nimport os\nfrom datetime import datetime\n\nclass TextToSpeechApp:\n    def __init__(self, root):\n        self.root = root\n        self.root.title(\"Text to Speech App\")\n        self.root.geometry(\"500x350\")\n\n        self.label_text = tk.Label(root, text=\"\u8bf7\u8f93\u5165\u8981\u8f6c\u6362\u4e3a\u8bed\u97f3\u7684\u6587\u672c\uff1a\")\n        self.label_text.grid(row=0, column=0, sticky=\"w\", padx=10, pady=10)\n\n        self.text_entry = tk.Text(root, width=60, height=10, wrap=\"word\")\n        self.text_entry.grid(row=1, column=0, padx=10, pady=(0, 10))\n\n        self.language_var = tk.StringVar()\n        self.language_var.set(\"en\")\n        self.languages = &#091;(\"\u82f1\u6587\", \"en\"), (\"\u4e2d\u6587\", \"zh\"), (\"\u6cd5\u6587\", \"fr\"), (\"\u5fb7\u6587\", \"de\")]\n        self.language_label = tk.Label(root, text=\"\u9009\u62e9\u8bed\u8a00\uff1a\")\n        self.language_label.grid(row=2, column=0, sticky=\"w\", padx=10, pady=(0, 5))\n        self.language_frame = tk.Frame(root)\n        self.language_frame.grid(row=3, column=0, padx=10, pady=5, sticky=\"w\")\n        for i, (lang_name, lang_code) in enumerate(self.languages):\n            rb = tk.Radiobutton(self.language_frame, text=lang_name, variable=self.language_var, value=lang_code)\n            rb.grid(row=0, column=i, padx=(0, 10))\n\n        self.file_label = tk.Label(root, text=\"\")\n        self.file_label.grid(row=4, column=0, padx=10, pady=(10, 5), sticky=\"w\")\n\n        self.file_counter = 1\n        self.recent_files = &#091;]\n\n        self.text_entry.bind(\"&lt;FocusOut&gt;\", self.convert_to_speech)\n        self.language_var.trace_add(\"write\", self.convert_to_speech)\n\n        self.button_preview = tk.Button(root, text=\"\u9884\u89c8\u58f0\u97f3\", command=self.preview_sound)\n        self.button_preview.grid(row=5, column=0, padx=10, pady=5)\n\n        self.button_save_as = tk.Button(root, text=\"\u53e6\u5b58\u4e3a\", command=self.save_as)\n        self.button_save_as.grid(row=6, column=0, padx=10, pady=5)\n\n    def convert_to_speech(self, *args):\n        text = self.text_entry.get(\"1.0\", \"end-1c\")\n        lang = self.language_var.get()\n        if text:\n            tts = gTTS(text=text, lang=lang)\n            timestamp = datetime.now().strftime(\"%Y%m%d%H%M%S\")\n            filename = f\"output-{timestamp}-{lang}-{self.file_counter}.mp3\"\n            tts.save(filename)\n            self.recent_files.append(filename)\n            if len(self.recent_files) &gt; 3:\n                os.remove(self.recent_files.pop(0))  # \u5220\u9664\u6700\u65e7\u7684\u6587\u4ef6\n            self.file_label.config(text=f\"\u97f3\u9891\u6587\u4ef6\u5df2\u751f\u6210\uff1a{filename}\")\n            self.file_counter += 1\n        else:\n            self.file_label.config(text=\"\u8bf7\u8f93\u5165\u8981\u8f6c\u6362\u7684\u6587\u672c\uff01\")\n\n    def preview_sound(self):\n        if self.recent_files:\n            pygame.init()\n            pygame.mixer.init()\n            pygame.mixer.music.load(os.path.abspath(self.recent_files&#091;-1]))\n            pygame.mixer.music.play()\n        else:\n            self.file_label.config(text=\"\u6ca1\u6709\u53ef\u9884\u89c8\u7684\u97f3\u9891\u6587\u4ef6\uff01\")\n\n    def save_as(self):\n        if hasattr(self, 'filename'):\n            save_path = filedialog.asksaveasfilename(defaultextension=\".mp3\")\n            if save_path:\n                import shutil\n                shutil.copy(self.filename, save_path)\n                self.file_label.config(text=f\"\u97f3\u9891\u6587\u4ef6\u5df2\u4fdd\u5b58\u4e3a\uff1a{save_path}\")\n            else:\n                self.file_label.config(text=\"\u4fdd\u5b58\u53d6\u6d88\uff01\")\n        else:\n            self.file_label.config(text=\"\u8bf7\u5148\u751f\u6210\u97f3\u9891\u6587\u4ef6\uff01\")\n\ndef main():\n    root = tk.Tk()\n    app = TextToSpeechApp(root)\n    root.mainloop()\n\nif __name__ == \"__main__\":\n    main()\n<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\">\u8fd0\u884c\u5e94\u7528\u7a0b\u5e8f<\/h3>\n\n\n\n<p>\u4fdd\u5b58\u4e0a\u8ff0\u4ee3\u7801\u4e3a <code>text_to_speech.py<\/code> \u6587\u4ef6\uff0c\u7136\u540e\u5728\u7ec8\u7aef\u4e2d\u8fd0\u884c\u5b83\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>python text_to_speech.py\n<\/code><\/pre>\n\n\n\n<p>\u8fd9\u5c06\u6253\u5f00\u4e00\u4e2a\u56fe\u5f62\u7528\u6237\u754c\u9762\uff0c\u4f60\u53ef\u4ee5\u5728\u5176\u4e2d\u8f93\u5165\u6587\u672c\uff0c\u9009\u62e9\u8bed\u8a00\uff0c\u9884\u89c8\u58f0\u97f3\uff0c\u5e76\u5c06\u58f0\u97f3\u53e6\u5b58\u4e3a MP3 \u6587\u4ef6\u3002<\/p>\n\n\n\n<p>\u73b0\u5728\uff0c\u4f60\u5df2\u7ecf\u5b66\u4f1a\u4e86\u5982\u4f55\u4f7f\u7528 Python \u521b\u5efa\u4e00\u4e2a\u7b80\u5355\u7684\u6587\u672c\u8f6c\u8bed\u97f3\u5e94\u7528\u7a0b\u5e8f\uff01<\/p>\n","protected":false},"excerpt":{"rendered":"<p>\u7a0b\u5e8f\u6548\u679c\u56fe \u8f6c\u6362\u540e\u7684\u58f0\u97f3\u6548\u679c \u5728&#46;&#46;&#46;<\/p>\n","protected":false},"author":2,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"_jetpack_memberships_contains_paid_content":false,"footnotes":""},"categories":[8],"tags":[5],"class_list":["post-54","post","type-post","status-publish","format-standard","hentry","category-app","tag-5"],"jetpack_featured_media_url":"","jetpack_sharing_enabled":true,"_links":{"self":[{"href":"https:\/\/www.uboger.com\/index.php?rest_route=\/wp\/v2\/posts\/54","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/www.uboger.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/www.uboger.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/www.uboger.com\/index.php?rest_route=\/wp\/v2\/users\/2"}],"replies":[{"embeddable":true,"href":"https:\/\/www.uboger.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=54"}],"version-history":[{"count":3,"href":"https:\/\/www.uboger.com\/index.php?rest_route=\/wp\/v2\/posts\/54\/revisions"}],"predecessor-version":[{"id":66,"href":"https:\/\/www.uboger.com\/index.php?rest_route=\/wp\/v2\/posts\/54\/revisions\/66"}],"wp:attachment":[{"href":"https:\/\/www.uboger.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=54"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/www.uboger.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=54"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/www.uboger.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=54"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}