欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

使用Python pyqt打造任意Excel數(shù)據(jù)庫系統(tǒng)

 更新時(shí)間:2025年07月28日 10:13:11   作者:Goona_  
這篇文章主要為大家詳細(xì)介紹了如何使用Python pyqt打造一個(gè)任意Excel數(shù)據(jù)庫系統(tǒng),可以對用戶上傳的任意電子表格Excel文件均可完成復(fù)雜數(shù)據(jù)庫查詢,需要的小伙伴可以了解下

一、引言

在數(shù)字化轉(zhuǎn)型浪潮中,超過76%的基層業(yè)務(wù)人員仍被困在"SQL恐懼癥"的泥潭里——他們精通業(yè)務(wù)邏輯卻受限于技術(shù)門檻,面對海量數(shù)據(jù)時(shí)只能反復(fù)請求IT部門協(xié)助。本項(xiàng)目通過Python+PyQt來構(gòu)建基于Excel風(fēng)格的查詢系統(tǒng),從而打破這種低效循環(huán):PyQt5提供直觀界面可視化組件,pandas庫實(shí)現(xiàn)"無SQL"的數(shù)據(jù)操作轉(zhuǎn)換,對用戶上傳的任意電子表格Excel文件均可完成復(fù)雜數(shù)據(jù)庫查詢。

二、GUI界面設(shè)計(jì)

使用PyQt5進(jìn)行界面的搭建,最終界面如下:

1.第一步:上傳任意Excel文件

直接點(diǎn)擊上傳即可。

這里可選擇上傳任意填寫好的Excel文件,但需要注意

①Excel文件是純數(shù)據(jù)文件不能包含表頭、標(biāo)題行、合并單元格等特殊格式

②Excel中的數(shù)據(jù)量不受限,但要求第一列必須為查找的主鍵值(主鍵值是指數(shù)據(jù)庫表中主鍵字段(或字段組合)所存儲(chǔ)的具體數(shù)值或字符串,用于唯一標(biāo)識(shí)表中的每一行記錄,確保數(shù)據(jù)可唯一識(shí)別和訪問),其余列可填寫對應(yīng)數(shù)據(jù)。

以成績表作為演示示例:如若想查詢?nèi)K腥说母骺瞥煽冃畔ⅲ瑒t第一列應(yīng)為該學(xué)生的學(xué)號(hào)(主鍵),其余各列可放置該學(xué)生的全部科目成績。

2.第二步:選擇查詢字段

在第一步上傳任意Excel成功后,會(huì)自動(dòng)根據(jù)上傳文件更新“請選擇主鍵字段”和“請選擇待查詢字段”內(nèi)的選項(xiàng),并且在“請選擇主鍵字段”這一選項(xiàng)下方出現(xiàn)輸入框,方便用戶輸入主鍵內(nèi)容。對于“請選擇主鍵字段”這一選項(xiàng),在第一步中已明確規(guī)定:Excel中第一列為主鍵字段,所以這里僅額外增加一個(gè)選項(xiàng)(以成績表為例,這里自動(dòng)增加“學(xué)號(hào)”字段);而對于“請選擇待查詢字段”這一選項(xiàng),會(huì)將Excel中除第一列以外的所有列名均作為新增選項(xiàng)(以成績表為例,這里自動(dòng)增加“語文”、“數(shù)學(xué)”、“英語”等其他所有字段),效果如下:

3.第三步:輸入主鍵值并查詢

選擇好主鍵字段及待查詢字段,并輸入待查詢主鍵值后,點(diǎn)擊“開始查詢”即可。

最終查詢效果如下:

可以看到,結(jié)果被分為兩個(gè)區(qū)域左邊表示查詢結(jié)果(即學(xué)生姓名),右邊表示與查詢結(jié)果相關(guān)的其他所有Excel數(shù)據(jù)(即除姓名和學(xué)號(hào)之外,該生的所有成績信息)。

4.異常提示

若未進(jìn)行相應(yīng)的操作,跳過某個(gè)步驟或直接點(diǎn)擊“開始查詢”,則會(huì)出現(xiàn)以下的異常提示。

5.界面設(shè)計(jì)jiemian.py

最后附上通過pyuic5產(chǎn)生的GUI界面代碼jiemian.py:

# -*- coding: utf-8 -*-

# Form implementation generated from reading ui file 'jiemian.ui'
#
# Created by: PyQt5 UI code generator 5.15.11
#
# WARNING: Any manual changes made to this file will be lost when pyuic5 is
# run again.  Do not edit this file unless you know what you are doing.


from PyQt5 import QtCore, QtGui, QtWidgets


class Ui_Form(object):
    def setupUi(self, Form):
        Form.setObjectName("Form")
        Form.setEnabled(True)
        Form.resize(600, 500)
        Form.setMinimumSize(QtCore.QSize(600, 500))
        Form.setMaximumSize(QtCore.QSize(600, 500))
        icon = QtGui.QIcon()
        icon.addPixmap(QtGui.QPixmap(":/image1.png"), QtGui.QIcon.Normal, QtGui.QIcon.Off)
        Form.setWindowIcon(icon)
        self.label_5 = QtWidgets.QLabel(Form)
        self.label_5.setGeometry(QtCore.QRect(390, 50, 141, 31))
        font = QtGui.QFont()
        font.setFamily("Adobe Arabic")
        font.setPointSize(16)
        self.label_5.setFont(font)
        self.label_5.setObjectName("label_5")
        self.label_8 = QtWidgets.QLabel(Form)
        self.label_8.setGeometry(QtCore.QRect(220, 20, 231, 31))
        font = QtGui.QFont()
        font.setFamily("Adobe Arabic")
        font.setPointSize(18)
        font.setBold(False)
        font.setWeight(50)
        self.label_8.setFont(font)
        self.label_8.setObjectName("label_8")
        self.label_10 = QtWidgets.QLabel(Form)
        self.label_10.setGeometry(QtCore.QRect(470, 40, 71, 51))
        self.label_10.setText("")
        self.label_10.setPixmap(QtGui.QPixmap(":/image1.png"))
        self.label_10.setObjectName("label_10")
        self.label = QtWidgets.QLabel(Form)
        self.label.setGeometry(QtCore.QRect(80, 90, 161, 21))
        font = QtGui.QFont()
        font.setFamily("Adobe Arabic")
        font.setPointSize(12)
        self.label.setFont(font)
        self.label.setObjectName("label")
        self.pushButton = QtWidgets.QPushButton(Form)
        self.pushButton.setGeometry(QtCore.QRect(120, 120, 161, 23))
        self.pushButton.setObjectName("pushButton")
        self.label_2 = QtWidgets.QLabel(Form)
        self.label_2.setGeometry(QtCore.QRect(80, 170, 161, 21))
        font = QtGui.QFont()
        font.setFamily("Adobe Arabic")
        font.setPointSize(12)
        self.label_2.setFont(font)
        self.label_2.setObjectName("label_2")
        self.comboBox = QtWidgets.QComboBox(Form)
        self.comboBox.setGeometry(QtCore.QRect(120, 200, 161, 22))
        self.comboBox.setObjectName("comboBox")
        self.comboBox.addItem("")
        self.comboBox_2 = QtWidgets.QComboBox(Form)
        self.comboBox_2.setGeometry(QtCore.QRect(330, 200, 161, 22))
        self.comboBox_2.setObjectName("comboBox_2")
        self.comboBox_2.addItem("")
        self.label_3 = QtWidgets.QLabel(Form)
        self.label_3.setGeometry(QtCore.QRect(80, 300, 221, 151))
        font = QtGui.QFont()
        font.setFamily("Adobe Arabic")
        font.setPointSize(28)
        self.label_3.setFont(font)
        self.label_3.setFrameShape(QtWidgets.QFrame.StyledPanel)
        self.label_3.setText("")
        self.label_3.setAlignment(QtCore.Qt.AlignCenter)
        self.label_3.setObjectName("label_3")
        self.pushButton_3 = QtWidgets.QPushButton(Form)
        self.pushButton_3.setGeometry(QtCore.QRect(80, 470, 471, 23))
        self.pushButton_3.setObjectName("pushButton_3")
        self.lineEdit = QtWidgets.QLineEdit(Form)
        self.lineEdit.setGeometry(QtCore.QRect(120, 250, 161, 20))
        self.lineEdit.setObjectName("lineEdit")
        self.textEdit = QtWidgets.QTextEdit(Form)
        self.textEdit.setGeometry(QtCore.QRect(320, 300, 231, 151))
        self.textEdit.setObjectName("textEdit")
        self.label_4 = QtWidgets.QLabel(Form)
        self.label_4.setGeometry(QtCore.QRect(80, 280, 161, 16))
        self.label_4.setObjectName("label_4")
        self.label_6 = QtWidgets.QLabel(Form)
        self.label_6.setGeometry(QtCore.QRect(320, 280, 161, 16))
        self.label_6.setObjectName("label_6")
        self.label_7 = QtWidgets.QLabel(Form)
        self.label_7.setGeometry(QtCore.QRect(120, 230, 161, 16))
        self.label_7.setObjectName("label_7")

        self.retranslateUi(Form)
        QtCore.QMetaObject.connectSlotsByName(Form)

    def retranslateUi(self, Form):
        _translate = QtCore.QCoreApplication.translate
        Form.setWindowTitle(_translate("Form", "Kebiao_query"))
        self.label_5.setText(_translate("Form", "Designed By"))
        self.label_8.setText(_translate("Form", "數(shù)據(jù)庫查詢系統(tǒng)"))
        self.label.setText(_translate("Form", "一、上傳數(shù)據(jù)文件"))
        self.pushButton.setText(_translate("Form", "點(diǎn)擊上傳"))
        self.label_2.setText(_translate("Form", "二、選擇查詢字段"))
        self.comboBox.setItemText(0, _translate("Form", "請選擇主鍵字段"))
        self.comboBox_2.setItemText(0, _translate("Form", "請選擇待查詢字段"))
        self.pushButton_3.setText(_translate("Form", "開始查詢"))
        self.label_4.setText(_translate("Form", "查詢結(jié)果如下:"))
        self.label_6.setText(_translate("Form", "相關(guān)查詢結(jié)果如下:"))
        self.label_7.setText(_translate("Form", "請輸入主鍵值:"))
import ziyuan_rc

三、主要程序詳解

1.導(dǎo)入所需模塊

import sys
from jiemian import *
from PyQt5.QtWidgets import QApplication, QWidget
import pandas as pd
# 保持窗口大小和qtdesigner中的一致
from PyQt5 import QtCore
QtCore.QCoreApplication.setAttribute(QtCore.Qt.AA_EnableHighDpiScaling)

不懂為啥需要導(dǎo)入Qtcore的,請看一鍵曝光:Python+PyQt實(shí)現(xiàn)的文件目錄。

2.初始化設(shè)置

    def __init__(self):
        super(QWidget, self).__init__()
        self.setupUi(self)

        self.pushButton.clicked.connect(self.shangchuan_wenjian)
        self.pushButton_3.clicked.connect(self.kaishi)
        self.lineEdit.setVisible(False)
        self.label_4.setVisible(False)
        self.label_6.setVisible(False)
        self.label_7.setVisible(False)

        self.shangchuan_flag = False

主要將pushbutton綁定于對應(yīng)的信號(hào)函數(shù),并對一些lineedit和label設(shè)置visible效果,目的是只有觸發(fā)某些動(dòng)作或函數(shù)時(shí),才將其顯示狀態(tài)更改為True,否則一直不顯示(即False狀態(tài));shangchuan_flag作為是否上傳任意Excel表的標(biāo)志位。

3.上傳文件并更新選項(xiàng)

    def shangchuan_wenjian(self):
        self.wenjian_df = shangchuan()
        if not self.wenjian_df.empty:
            self.shangchuan_flag = True
            self.lineEdit.setVisible(True)
            self.label_7.setVisible(True)

            self.columns = self.wenjian_df.columns.tolist()
            self.comboBox.clear()
            self.comboBox.addItem("請選擇主鍵字段")
            self.comboBox.addItem(self.columns[0])
            self.comboBox_2.clear()
            self.comboBox_2.addItem("請選擇待查詢字段")
            self.comboBox_2.addItems(self.columns[1:])

調(diào)用自定義函數(shù)shangchuan來實(shí)現(xiàn)彈窗效果;當(dāng)文件內(nèi)容不為空時(shí),設(shè)置上傳標(biāo)志位shangchuan_flag為True,并顯示lineedit(主鍵值輸入框)和label_7(“請輸入主鍵值”);讀取所有文件列名并存儲(chǔ)在列表columns中,將列表columns中的第一個(gè)值(即第一列的列名)添加至combobox中,其余值添加至combobox_2中。但需要注意:每次添加item(s)前,需要將其原有選項(xiàng)清空clear,否則容易造成選項(xiàng)堆疊重復(fù)。

4.查詢信息并顯示

    def kaishi(self):
        if self.shangchuan_flag==True:
            if self.comboBox.currentIndex()!=0:
                if self.comboBox_2.currentIndex()!=0:

                    lieziduan = self.comboBox_2.currentText()
                    liesuoyin = self.wenjian_df.columns.get_loc(lieziduan)

                    # 根據(jù)行字段確定行索引
                    hangsuoyin = -1
                    for i in range(0, len(self.wenjian_df)):
                        if str(self.wenjian_df.iloc[i,0]) == self.lineEdit.text():
                            hangsuoyin = i
                            break

                    chaxunzhi = self.wenjian_df.iloc[hangsuoyin, liesuoyin]

                    # 匹配剩余未查詢值
                    ewai_lst = []
                    for column_name in self.columns:
                        new_column_index = self.wenjian_df.columns.get_loc(column_name)
                        if (new_column_index != liesuoyin) and (new_column_index != 0):
                            ewaizhi = self.wenjian_df.iloc[hangsuoyin, new_column_index]
                            ewai_lst.append(column_name+":"+str(ewaizhi))

                    if pd.isna(chaxunzhi) or hangsuoyin==-1:
                        self.label_4.setVisible(True)
                        self.label_6.setVisible(True)
                        self.label_3.setText("未找到")
                        self.textEdit.setText("未找到")
                    else:
                        self.label_4.setVisible(True)
                        self.label_6.setVisible(True)
                        self.label_3.setText(str(chaxunzhi))
                        self.textEdit.setText("\n".join(ewai_lst))

                else:
                    QtWidgets.QMessageBox.critical(self, "提示", "請選擇主鍵字段!")
            else:
                QtWidgets.QMessageBox.critical(self, "提示", "請選擇待查詢字段!")
        else:
            QtWidgets.QMessageBox.critical(self, "提示", "請檢查操作步驟或上傳文件!")

以shangchaun_flag作為標(biāo)志位,檢查是否上傳模板文件;依次判斷是否選擇每個(gè)combobox里的對應(yīng)選項(xiàng);若均滿足,則根據(jù)combobox_2的內(nèi)容定位列字段名稱,再根據(jù)列字段名稱反推列索引根據(jù)lineedit內(nèi)容通過循環(huán)查詢對應(yīng)的值,找到后即可退出循環(huán),避免運(yùn)算復(fù)雜度,最終得到行索引;最后通過得到的行索引和列索引確定查詢值,并通過label_3顯示;剩余未查詢值,思路方法也是同理,最終顯示在textedit中。若遇到查詢不到的情況時(shí),顯示的內(nèi)容均設(shè)置為“未找到”。當(dāng)然,哪個(gè)combobox有問題,則顯示不同的對應(yīng)提示。

5.自定義函數(shù)

def shangchuan():
    filepath, _ = QtWidgets.QFileDialog.getOpenFileName(None, "請選擇文件", "", "XLSX工作表 (*.xlsx)")  # 獲取文件路徑
    if filepath:
        # 獲取原始df
        yuanshi_df = pd.read_excel(filepath)
        if not yuanshi_df.empty:
            QtWidgets.QMessageBox.information(None, "成功", "上傳成功!")
            return yuanshi_df
        else:
            QtWidgets.QMessageBox.critical(None, "提示", "請檢查操作步驟或上傳文件!")
    else:
        QtWidgets.QMessageBox.critical(None, "提示", "請選擇XLSX工作表類型!")
        return pd.DataFrame()

一旦需要上傳文件時(shí),直接調(diào)用此函數(shù)即可。

四、總程序代碼

import sys
from jiemian import *
from PyQt5.QtWidgets import QApplication, QWidget
import pandas as pd
# 保持窗口大小和qtdesigner中的一致
from PyQt5 import QtCore
QtCore.QCoreApplication.setAttribute(QtCore.Qt.AA_EnableHighDpiScaling)

class mainwindow(QWidget, Ui_Form):
    def __init__(self):
        super(QWidget, self).__init__()
        self.setupUi(self)

        self.pushButton.clicked.connect(self.shangchuan_wenjian)
        self.pushButton_3.clicked.connect(self.kaishi)
        self.lineEdit.setVisible(False)
        self.label_4.setVisible(False)
        self.label_6.setVisible(False)
        self.label_7.setVisible(False)

        self.shangchuan_flag = False

    def shangchuan_wenjian(self):
        self.wenjian_df = shangchuan()
        if not self.wenjian_df.empty:
            self.shangchuan_flag = True
            self.lineEdit.setVisible(True)
            self.label_7.setVisible(True)

            self.columns = self.wenjian_df.columns.tolist()
            self.comboBox.clear()
            self.comboBox.addItem("請選擇主鍵字段")
            self.comboBox.addItem(self.columns[0])
            self.comboBox_2.clear()
            self.comboBox_2.addItem("請選擇待查詢字段")
            self.comboBox_2.addItems(self.columns[1:])

    def kaishi(self):
        if self.shangchuan_flag==True:
            if self.comboBox.currentIndex()!=0:
                if self.comboBox_2.currentIndex()!=0:

                    lieziduan = self.comboBox_2.currentText()
                    liesuoyin = self.wenjian_df.columns.get_loc(lieziduan)

                    # 根據(jù)行字段確定行索引
                    hangsuoyin = -1
                    for i in range(0, len(self.wenjian_df)):
                        if str(self.wenjian_df.iloc[i,0]) == self.lineEdit.text():
                            hangsuoyin = i
                            break

                    chaxunzhi = self.wenjian_df.iloc[hangsuoyin, liesuoyin]

                    # 匹配剩余未查詢值
                    ewai_lst = []
                    for column_name in self.columns:
                        new_column_index = self.wenjian_df.columns.get_loc(column_name)
                        if (new_column_index != liesuoyin) and (new_column_index != 0):
                            ewaizhi = self.wenjian_df.iloc[hangsuoyin, new_column_index]
                            ewai_lst.append(column_name+":"+str(ewaizhi))

                    if pd.isna(chaxunzhi) or hangsuoyin==-1:
                        self.label_4.setVisible(True)
                        self.label_6.setVisible(True)
                        self.label_3.setText("未找到")
                        self.textEdit.setText("未找到")
                    else:
                        self.label_4.setVisible(True)
                        self.label_6.setVisible(True)
                        self.label_3.setText(str(chaxunzhi))
                        self.textEdit.setText("\n".join(ewai_lst))

                else:
                    QtWidgets.QMessageBox.critical(self, "提示", "請選擇主鍵字段!")
            else:
                QtWidgets.QMessageBox.critical(self, "提示", "請選擇待查詢字段!")
        else:
            QtWidgets.QMessageBox.critical(self, "提示", "請檢查操作步驟或上傳文件!")

def shangchuan():
    filepath, _ = QtWidgets.QFileDialog.getOpenFileName(None, "請選擇文件", "", "XLSX工作表 (*.xlsx)")  # 獲取文件路徑
    if filepath:
        # 獲取原始df
        yuanshi_df = pd.read_excel(filepath)
        if not yuanshi_df.empty:
            QtWidgets.QMessageBox.information(None, "成功", "上傳成功!")
            return yuanshi_df
        else:
            QtWidgets.QMessageBox.critical(None, "提示", "請檢查操作步驟或上傳文件!")
    else:
        QtWidgets.QMessageBox.critical(None, "提示", "請選擇XLSX工作表類型!")
        return pd.DataFrame()

if __name__ == '__main__':
    app = QApplication(sys.argv)
    w = mainwindow()
    w.show()
    sys.exit(app.exec_())

僅以成績表作為示例,上傳其他任意Excel均可!

以上就是使用Python pyqt打造任意Excel數(shù)據(jù)庫系統(tǒng)的詳細(xì)內(nèi)容,更多關(guān)于Python Excel數(shù)據(jù)查詢的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • python opencv把一張圖片嵌入(疊加)到另一張圖片上的實(shí)現(xiàn)代碼

    python opencv把一張圖片嵌入(疊加)到另一張圖片上的實(shí)現(xiàn)代碼

    這篇文章主要介紹了python opencv把一張圖片嵌入(疊加)到另一張圖片上,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-06-06
  • Python列表list數(shù)組array用法實(shí)例解析

    Python列表list數(shù)組array用法實(shí)例解析

    這篇文章主要介紹了Python列表list數(shù)組array用法,實(shí)例解析了關(guān)于數(shù)組的各種常見操作,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2014-10-10
  • Python實(shí)現(xiàn)Excel和CSV之間的相互轉(zhuǎn)換

    Python實(shí)現(xiàn)Excel和CSV之間的相互轉(zhuǎn)換

    通過使用Python編程語言,編寫腳本來自動(dòng)化Excel和CSV之間的轉(zhuǎn)換過程,可以批量處理大量文件,定期更新數(shù)據(jù),并集成轉(zhuǎn)換過程到自動(dòng)化工作流程中,本文將介紹如何使用Python 實(shí)現(xiàn)Excel和CSV之間的相互轉(zhuǎn)換,需要的朋友可以參考下
    2024-03-03
  • Python自動(dòng)搶紅包教程詳解

    Python自動(dòng)搶紅包教程詳解

    在本篇文章里小編給大家整理了關(guān)于Python自動(dòng)搶紅包的相關(guān)知識(shí)點(diǎn)內(nèi)容,有興趣的朋友們參考學(xué)習(xí)下。
    2019-06-06
  • python模塊引入問題和解決方案

    python模塊引入問題和解決方案

    本文主要介紹了python模塊引入問題和解決方案,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • python字符串操作的15種方法匯總

    python字符串操作的15種方法匯總

    對于python中的字符串,有多種操作方法,下面這篇文章主要給大家介紹了關(guān)于python字符串操作的15種方法,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-09-09
  • 使用python實(shí)現(xiàn)一個(gè)瀏覽器自動(dòng)化的腳本

    使用python實(shí)現(xiàn)一個(gè)瀏覽器自動(dòng)化的腳本

    最近工作中有這樣一個(gè)需求:客戶反饋在瀏覽器操作過程中,重復(fù)流程操作太頻繁,能不能讓瀏覽器自動(dòng)操作完成?所以本文給大家介紹了如何使用python實(shí)現(xiàn)一個(gè)瀏覽器自動(dòng)化的腳本,感興趣的朋友可以參考下
    2024-04-04
  • python加密打包程序詳解

    python加密打包程序詳解

    這篇文章主要介紹了python加密打包程序,還給大家介紹了Python實(shí)現(xiàn)文件簡單加解密的方法,本文通過示例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2023-04-04
  • Pycharm無法正常安裝第三方庫的幾條應(yīng)對方法匯總

    Pycharm無法正常安裝第三方庫的幾條應(yīng)對方法匯總

    在使用pycharm學(xué)習(xí)python的時(shí)候,經(jīng)常需要第三方庫,沒有第三方庫程序就會(huì)報(bào)錯(cuò),下面這篇文章主要給大家介紹了關(guān)于Pycharm無法正常安裝第三方庫的幾條應(yīng)對方法,需要的朋友可以參考下
    2023-04-04
  • Python使用unicodedata實(shí)現(xiàn)字符串標(biāo)準(zhǔn)化

    Python使用unicodedata實(shí)現(xiàn)字符串標(biāo)準(zhǔn)化

    這篇文章主要來和大家聊一聊 Python 的一個(gè)內(nèi)置模塊:unicodedata,它是專門用來處理 unicode 字符串的,下面就一起來看看它的用法吧
    2023-06-06

最新評論