用 ChatGPT 编写个 素数判断 的代码,怎么改不够完美?

2023-03-21 07:09:08 +08:00
 guanjinman2022
import math
import time
import tkinter as tk
from tkinter import messagebox, ttk
from concurrent.futures import ThreadPoolExecutor, Future
from typing import List

def is_prime(n):
    """
    判断一个数是否为素数
    """
    if n < 2:
        return False
    elif n == 2:
        return True
    elif n % 2 == 0:
        return False
    else:
        for i in range(3, int(math.sqrt(n))+1, 2):
            if n % i == 0:
                return False
        return True

def factorization(n):
    """
    分解质因数
    """
    factors = []
    for i in range(2, int(math.sqrt(n))+1):
        while n % i == 0:
            factors.append(i)
            n //= i
        if n == 1:
            break
    if n > 1:
        factors.append(n)
    return factors

def calculate():
    try:
        n = int(entry.get())
    except ValueError:
        messagebox.showerror("错误", "请输入一个整数")
        return
    progress_bar['value'] = 0
    result.set("计算中,请稍候...")
    future = executor.submit(calculate_result, n)
    future.add_done_callback(on_calculate_done)
    # 启动进度条更新线程
    progress_future = executor.submit(progress_worker, future)

def calculate_result(n):
    start_time = time.time()
    if is_prime(n):
        result_str = f"{n} 是素数"
    else:
        factors = factorization(n)
        result_str = f"{n} 不是素数,它可以分解为 {factors}"
    end_time = time.time()
    result_str += f" 用时 {end_time - start_time:.2f} 秒"
    return result_str

def on_calculate_done(future: Future):
    result.set(future.result())

def quit_program():
    root.quit()

def update_progress(progress):
    progress_bar['value'] = progress

def progress_worker(future: Future):
    start_time = time.time()
    while not future.done():
        elapsed_time = time.time() - start_time
        progress = elapsed_time / progress_bar_max_time * 100
        progress = min(progress, 100)
        update_progress(progress)
        time.sleep(0.1)
    update_progress(100)

root = tk.Tk()
root.title("素数判断")

# 创建控件
label = ttk.Label(root, text="请输入一个整数:")
label.pack(pady=10)

entry = ttk.Entry(root, width=40)
entry.pack(pady=10)

progress_bar = ttk.Progressbar(root, orient=tk.HORIZONTAL, length=400, mode='determinate')
progress_bar.pack(pady=10)

button = ttk.Button(root, text="判断", command=calculate)
button.pack(pady=10)

result = tk.StringVar()
result.set("结果将显示在这里")
result_label = ttk.Label(root, textvariable=result)
result_label.pack(pady=10)

quit_button = ttk.Button(root, text="退出", command=quit_program)
quit_button.pack(pady=10)

# 设置布局
root.geometry("500x300")

executor = ThreadPoolExecutor(max_workers=2) # 增加线程池大小为 2 ,用于同时执行计算和进度条更新任务

progress_bar_max_time = 5  # 进度条走满需要的时间,单位为秒

def calculate():
    try:
        n = int(entry.get())
    except ValueError:
        messagebox.showerror("错误", "请输入一个整数")
        return
    progress_bar['value'] = 0
    result.set("计算中,请稍候...")
    future = executor.submit(calculate_result, n)
    future.add_done_callback(on_calculate_done)
    progress_future = executor.submit(progress_worker, future)

def on_calculate_done(future: Future):
    result.set(future.result())
    progress_bar['value'] = 100
    future.cancel()  # 计算完成,取消进度条更新任务

def update_progress(progress):
    progress_bar['value'] = progress

def progress_worker(future: Future):
    start_time = time.time()
    while not future.done():
        elapsed_time = time.time() - start_time
        progress = elapsed_time / progress_bar_max_time * 100
        progress = min(progress, 100)
        update_progress(progress)
        time.sleep(0.1)
    update_progress(100)


root.mainloop()

这个是让 ChatGPT 写的判断素数的代码,使用 Python 3.11.2 版本 执行,但是进度条不管怎么改都不完美。 之前 GUI 界面上没进度条的,命令行窗口里有,很准确,还有倒计时、百分比等信息, 移到 GUI 界面上就一直不完美,要么点了判断,进度条不走,出结果的时候一下子拉满;要么进度条走完了还在计算中,代码里用 progress_bar_max_time = 5 的固定值不咋地,弄个预估的动态值也行啊,ChatGPT 也知道我的意思,就是弄不出来。

我要求就是点判断后进度条开始匀速走,走完了刚好出来结果,最好进度条右边放个动态百分比数字。 放进度条就是为了计算大一点数时直观感受到大概时间

2321 次点击
所在节点    Python
4 条回复
Devilker
2023-03-21 07:41:23 +08:00
1999 除以 3 那个吗 哈哈哈哈
Rever4433
2023-03-21 10:45:36 +08:00
这种进度条确实很难完美实现,举个例子,2147483647000007 和 214748364700007 ,在我的 i5 7300hq 的笔记本上前者 5.48s ,后者 0.74s 。并且素数的计算时间要更快。这就使得对于任何一次输入,计算时间都是不确定的。
guanjinman2022
2023-03-21 11:44:47 +08:00
@Rever4433 我知道时间是不确定的。后来让 ChatGPT 引入预估时间函数,用这个预估时间替换那个 5 来赋值。结果还是不行,点判断后进度条不走,显示计算中,相当于先后台计算一遍,知道时间后,再走进度条,走完刚好出来时间,整个过程相当于 2 倍时间。
lmshl
2023-03-21 16:34:02 +08:00
这属于问题定义不清,gpt 也帮不了你了
让我做的话我可能会这样,告诉 gpt 用 Miller–Rabin 算法实现一个素性检验算法,设定一个足够大的阈值,比如实验 20 次以后就认定输入是素数,因为伪素数几率已经足够低到可以忽略了。
然后画进度条不过就是试验次数这一固定数值,你甚至不需要进度条,眨眼就验证完了。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/925723

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX