主頁 > 後端開發 > 為什么我的Qt程式中未執行的代碼會導致程式崩潰?

為什么我的Qt程式中未執行的代碼會導致程式崩潰?

2021-11-29 22:13:21 後端開發

這個Qt程式中的一段代碼導致程式崩潰但沒有執行。

程式由main.cpp、mainwindow.cpp、mainwindow.h、mainwindow.ui和CMakeLists.txt組成,問題主要出在MainWindow類。

主視窗.h:

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QPainter>
#include <QMouseEvent>
#include <QColorDialog>
#include <vector>
#include <cmath>
#include <cstdio>
#define Pi 3.1415926535897932
#define MAXHEIGHT 4320
#define MAXWIDTH 7680
using namespace std;

QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACE

enum State{NONE,RECT,CIRCLE,POLYGON,CUBE,BEZIER};

struct Circle
{
    int x;
    int y;
    double r;
};

struct Pos
{
    int x;
    int y;
};

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

private:
    Ui::MainWindow *ui;

protected:
    void paintEvent(QPaintEvent*);
    void mousePressEvent(QMouseEvent*);
    void mouseReleaseEvent(QMouseEvent*);
    void mouseMoveEvent(QMouseEvent*);

    void DDALine(QPainter* painter, int x1, int y1, int x2, int y2);
    void BresenhamCircle(QPainter* painter, int x, int y, int r);
    void Polygon(QPainter* painter, vector<Pos> cpolygon);

    State state=NONE;
    vector<QRect> rects;
    vector<Circle> circles;
    vector<vector<Pos>> polygons;
    vector<Pos> currentPolygon;
    QPoint* movePoint=nullptr;
    vector<Pos> curPaintPolygon;
    int rectX1=0;
    int rectY1=0;
    int circleX1=0;
    int circleY1=0;
    QColor penColor=Qt::black;
    QImage img;
};
#endif // MAINWINDOW_H

主視窗.cpp:

#include "mainwindow.h"
#include "./ui_mainwindow.h"

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    if(!(img.load("/Users/lijiabo/Documents/GitHub/CGWORK0528/0528.png")))
        throw "Image loading failed!";

    //connect menu actions
    connect(ui->actionDrawRect,&QAction::triggered,this,[=](){
        this->state=RECT;
    });
    connect(ui->actionDrawCircle,&QAction::triggered,this,[=](){
        this->state=CIRCLE;
    });
    connect(ui->actionSetColor,&QAction::triggered,this,[=](){
        penColor=QColorDialog::getColor(penColor,this,"設定顏色");
        update(rect());
    });
    connect(ui->actionDrawPolygon,&QAction::triggered,this,[=](){
        this->state=POLYGON;
    });
    connect(ui->actionSetColor_2,&QAction::triggered,this,[=](){
        penColor=QColorDialog::getColor(penColor,this,"設定顏色");
        update(rect());
    });
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::paintEvent(QPaintEvent *)
{
    QPainter painter(this);
    painter.setPen(penColor);
    //DEBUG
    painter.drawText(rect(),QString("Width: " QString::number(rect().width()) " Height: " QString::number(rect().height())));

    switch(state)
    {
    case RECT:
    case CIRCLE:
        for(QRect rect:rects)
        {
            DDALine(&painter,rect.topLeft().x(),rect.topLeft().y(),rect.topRight().x(),rect.topRight().y());//top
            DDALine(&painter,rect.bottomLeft().x(),rect.bottomLeft().y(),rect.bottomRight().x(),rect.bottomRight().y());//bottom
            DDALine(&painter,rect.topLeft().x(),rect.topLeft().y(),rect.bottomLeft().x(),rect.bottomLeft().y());//left
            DDALine(&painter,rect.topRight().x(),rect.topRight().y(),rect.bottomRight().x(),rect.bottomRight().y());//right
        }
        for(Circle c:circles)
            BresenhamCircle(&painter,c.x,c.y,(int)c.r);
        break;
    case POLYGON:
        if(currentPolygon.size()>=1)
        {
            //把currentPolygon復制給curPaintPolygon
            curPaintPolygon.clear();
            for(Pos point:currentPolygon)
            {
                curPaintPolygon.push_back({point.x,point.y});
            }
            if(movePoint!=nullptr)
                curPaintPolygon.push_back({movePoint->x(),movePoint->y()});
            Polygon(&painter,curPaintPolygon);
        }
        if(polygons.size()>0)
        {
            for(vector<Pos> polygon:polygons)
            {
                Polygon(&painter,polygon);
            }
        }
        break;
    case CUBE:
        break;
    case BEZIER:
        break;
    default:
        ;
    }
}

void MainWindow::mousePressEvent(QMouseEvent *event)
{
    switch(state)
    {
    case RECT:
        rectX1=event->pos().x();
        rectY1=event->pos().y();
        rects.push_back(QRect(rectX1,rectY1,0,0));
        break;
    case CIRCLE:
        circleX1=event->pos().x();
        circleY1=event->pos().y();
        circles.push_back({circleX1,circleY1,1});
        break;
    case POLYGON:
        /*
        if(currentPolygon==nullptr)
        {
            currentPolygon=new QPolygon;
            currentPolygon->append(QPoint(event->pos().x(),event->pos().y()));
        }
        else
        */
        movePoint= nullptr;
        currentPolygon.push_back({event->pos().x(),event->pos().y()});
        break;
    case CUBE:
        break;
    case BEZIER:
        break;
    default:
        ;
    }
    update(rect());
}

void MainWindow::mouseReleaseEvent(QMouseEvent *event)
{
    switch(state)
    {
    case RECT:
        break;
    case CIRCLE:
        break;
    case POLYGON:
        if(event->button()==Qt::LeftButton)
        {
            if(currentPolygon.size()>1)
                polygons.push_back(currentPolygon);
            currentPolygon.clear();
            if(movePoint!=nullptr)
                delete movePoint;
            movePoint=nullptr;
        }
        break;
    case CUBE:
        break;
    case BEZIER:
        break;
    default:
        ;
    }
    update(rect());
}

void MainWindow::mouseMoveEvent(QMouseEvent *event)
{
    switch(state)
    {
    case RECT:
        rects.back().setCoords(rectX1,rectY1,event->pos().x(),event->pos().y());
        break;
    case CIRCLE:
        circles.back().r=sqrt((event->pos().x()-circleX1)*(event->pos().x()-circleX1) (event->pos().y()-circleY1)*(event->pos().y()-circleY1));
        break;
    case POLYGON:
        if(movePoint==nullptr)
            movePoint=new QPoint(event->pos().x(),event->pos().y());
        else
            *movePoint=event->pos();
        break;
    case CUBE:
        break;
    case BEZIER:
        break;
    default:
        ;
    }
    update(rect());
}

void MainWindow::DDALine(QPainter* painter, int x1, int y1, int x2, int y2)
{
    double dx, dy, e, x, y;
    dx = x2 - x1;
    dy = y2 - y1;
    e = (fabs(dx) > fabs(dy)) ? fabs(dx) : fabs(dy);
    dx /= e;
    dy /= e;
    x = x1;
    y = y1;
    for (int i = 1; i <= e; i  )
    {
        painter->drawPoint((int)(x   0.5), (int)(y   0.5));
        x  = dx;
        y  = dy;
    }
}

void MainWindow::BresenhamCircle(QPainter* painter, int x, int y, int r)
{
    int edgeX,edgeY,p;

    edgeX=0;
    edgeY=r;
    p=3-2*r;
    for(;edgeX<=edgeY;edgeX  )
    {
        painter->drawPoint(x edgeX,y edgeY);
        double dT=0;
        for(int dTNC=1;dTNC<8;dTNC  )
        {
            dT=dTNC*0.25*Pi;
            painter->drawPoint( x   r*cos(acos(edgeX/double(r)) dT), y   r*sin(asin(edgeY/double(r)) dT) );
        }
        if(p>=0)
        {
            p =4*(edgeX-edgeY) 10;
            edgeY--;
        }
        else
            p =4*edgeX 6;
    }


}

void MainWindow::Polygon(QPainter* painter, vector<Pos> cpolygon)
{
    if(cpolygon.size()<=1)
        return;
    
    vector<Pos>::iterator prevPoint = cpolygon.begin();
    for(vector<Pos>::iterator it=cpolygon.begin() 1;it!=cpolygon.end();it  ,prevPoint  )
    {
        DDALine(painter,prevPoint->x,prevPoint->y,it->x,it->y);
    }
    DDALine(painter,prevPoint->x,prevPoint->y,cpolygon.begin()->x,cpolygon.begin()->y);
    
    //fill------------------------------

    int imgWidth=img.width();
    int imgHeight=img.height();
    int x1=rect().left();
    int x2=rect().right();
    int y1=rect().bottom();
    int y2=rect().top();
    bool mask[MAXHEIGHT][MAXWIDTH];
    for(int y=y1;y<=y2;y  )
        for(int x=x1;x<=x2;x  )
            mask[y][x]=false;
    for(vector<Pos>::const_iterator it=cpolygon.cbegin();it!=cpolygon.cend();it  )
    {
        int xs=it->x;
        int dxs=((it 1)->x-it->x)/((it 1)->y/it->y);
        int dys=abs((it 1)->y-it->y)/((it 1)->y-it->y);
        for(int ys=it->y;ys!=(it 1)->y;ys =dys)
        {
            int Ixs=int(xs 0.5);
            mask[ys][Ixs]=!mask[ys][Ixs];
            xs =dys*dxs;
        }
    }
    QPen initialPen = painter->pen();
    for(int y=y1;y<=y2;y  )
    {
        bool inside=false;
        for(int x=x1;x<=x2;x  )
        {
            if(mask[y][x])
                inside=!inside;
            if(inside)
            {
                painter->setPen(img.pixel(x%imgWidth,y%imgHeight));
                painter->drawPoint(x,y);
            }
        }
    }
    painter->setPen(initialPen);
    //----------------------------------
}

在mainwindow.cpp的最后一個函式Polygon()中,當“fill”部分存在時,當actionDrawPolygon被觸發時程式會崩潰,我點擊滑鼠左鍵(所以mousePressEvent()被呼叫),程式贏了沒有這部分就不會崩潰。

但是當我在這部分添加 printf("TEST") 函式時,沒有列印"TEST",所以這部分根本沒有執行,那為什么會導致程式崩潰呢?

作業系統版本:macOS Monterey 12.0.1

IDE:CLion 2021.2.3

uj5u.com熱心網友回復:

如前所述,您有一個大小bool合適的本地陣列std::sizeof(bool) * MAXHEIGHT * MAXWIDTH取決于sizeofbool,大小大于 30 兆位元組。

由于堆疊大小有限(可能從 1 兆位元組到通常為 8 兆位元組),程式堆疊將無法處理該大小的陣列。

相反,試試這個:

std::vector<std::vector<bool>> mask(MAXHEIGHT, std::vector<bool>(MAXWIDTH));

這也消除了for將陣列條目初始化為 的回圈的需要false,因為無論如何這將是默認值。

這不是最好的方法,因為它是向量的向量,但是這很可能會繞過您現在看到的崩潰。

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/369255.html

標籤:C qt

上一篇:exe檔案未在Release檔案夾中運行

下一篇:如何使用ReactJS和TypescriptappendChild自定義組件?

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more