- 編程的過程中經常遇到需要將QString轉成char *或者const char *的情況,在轉換成QByteArray后調用.data()或者.constData()函數進行轉換,這里需要注意的是,如果轉換類型是const char *盡管用data()不會出錯,會給你自動轉換,但是還是不建議,因為深拷貝了一份,理論上增加了內存開銷,如果字符串長度小還好,一旦很長,這個開銷挺大,這是個好的編程習慣。
//查閱代碼得知data函數有兩個重載
inline char *QByteArray::data()
{ detach(); return d- >data(); }
inline const char *QByteArray::data() const
{ return d- >data(); }
inline const char *QByteArray::constData() const
{ return d- >data(); }
QByteArray data = "abc";
//深拷貝
char *d1 = data.data();
//深拷貝
const char *d2 = data.data();
//淺拷貝
const char *d3 = data.constData();
//深拷貝
test(data.data());
//淺拷貝
test(data.constData());
void test(const char *data)
{
}
//至于什么時候調用.data()會淺拷貝,酷碼大佬說是當QByteArray被const修飾的時候
const QByteArray data;
//淺拷貝
const char *d = data.data();
//酷碼大佬補充:自Qt 5.7版本以來,引入了qAsConst函數,專用于無腦轉換。
//這個函數實現了C++17標準中的std::as_const()函數的功能,將一個非常量的左值轉為常量的左值。
//增加qAsConst函數是為了Qt自己的非const 的容器能實現C++11標準的基于范圍的循環。
//該函數主要用于qt容器在隱式共享中不被detach。
QString s = "abc";
//下面會深拷貝引起性能損失
for (QChar ch : s)
//不會深拷貝
for (QChar ch : qAsConst(s))
//下面也是淺拷貝,但是在編程時、在現實中,聲明為const往往不容易做到。
const QString s;
for (QChar ch : s)
//總結:對Qt自己實現的容器如:QVector、QMap、 QHash、QLinkedList、QList等,如果一定要用基于for(var : container)范圍的循環,則請用如下形式:
for (var : qAsConst(container))
- 新版的Qt6.5在ubuntu上編譯運行程序后會提示 qt.qpa.plugin: Could not load the Qt platform plugin "xcb" in "" even though it was found. ,無法正常彈出窗體程序,你需要主動安裝xcb的相關庫。sudo apt install libxcb*
- 有些場景下我們需要在 QApplication a(argc, argv); 前面執行一些處理,比如 QApplication::setAttribute 就必須在最前面執行,而很多時候這個設置的參數不能改寫死,畢竟現場的環境千差萬別,希望通過配置文件來配置,那么問題來了,讀取配置文件一般需要指定路徑才能正常讀取到,如果是 ./ 這種,很可能未必是應用程序的當前路徑,如果你是雙擊運行的程序,那肯定是應用程序的當前路徑,不是雙擊運行那就是系統環境中的當前路徑,意味著你開機啟動或者用system、QProcess等方式在開機后調用啟動的話,就未必正確了。為了保證這個路徑的正確,必須從main函數的 argv 第一個值獲取,通過查閱Qt自身代碼中獲取路徑,也是從這個參數獲取。
//程序最前面獲取應用程序路徑和名稱
static void getCurrentInfo(char *argv[], QString &path, QString &name);
//程序最前面讀取配置文件節點的值
static QString getIniValue(const QString &fileName, const QString &key);
static QString getIniValue(char *argv[], const QString &key, const QString &dir = QString());
void QtHelper::getCurrentInfo(char *argv[], QString &path, QString &name)
{
//必須用fromLocal8Bit保證中文路徑正常
QString argv0 = QString::fromLocal8Bit(argv[0]);
QFileInfo file(argv0);
path = file.path();
name = file.baseName();
}
QString QtHelper::getIniValue(const QString &fileName, const QString &key)
{
QString value;
QFile file(fileName);
if (file.open(QFile::ReadOnly | QFile::Text)) {
while (!file.atEnd()) {
QString line = file.readLine();
if (line.startsWith(key)) {
line = line.replace("n", "");
line = line.trimmed();
value = line.split("=").last();
break;
}
}
}
return value;
}
QString QtHelper::getIniValue(char *argv[], const QString &key, const QString &dir)
{
QString path, name;
QtHelper::getCurrentInfo(argv, path, name);
QString fileName = QString("%1/%2%3.ini").arg(path).arg(dir).arg(name);
return getIniValue(fileName, key);
}
int main(int argc, char *argv[])
{
int openGLType = QtHelper::getIniValue(argv, "OpenGLType").toInt();
QtHelper::initOpenGL(openGLType);
QApplication a(argc, argv);
...
}
- 當我們對QTableView/QTreeView/QTableWidget/QTreeWidget某行選中后,會發現某些單元格設置的前景色被覆蓋了,比如設置的紅色,一旦選中就變成了白色,這肯定不是我們想要的,需要用自定義委托將其去掉。
class ItemDelegate : public QItemDelegate
{
Q_OBJECT
public:
explicit ItemDelegate(QObject *parent = 0);
protected:
void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const;
};
#include "itemdelegate.h"
ItemDelegate::ItemDelegate(QObject *parent) : QItemDelegate(parent)
{
}
void ItemDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
QStyleOptionViewItem option2 = option;
QColor color = index.data(Qt::ForegroundRole).value< QColor >();
if (color.isValid() && color != option.palette.color(QPalette::WindowText)) {
option2.palette.setColor(QPalette::HighlightedText, color);
}
QItemDelegate::paint(painter, option2, index);
}
//對所有單元格設置該委托
ui- >tableWidget- >setItemDelegate(new ItemDelegate);
- 有些時候我們需要在項目文件比如pro/pri中識別當前Qt套件是否存在某個模塊以及是否引入過某個模塊,存在則引入,同時也希望代碼中也能識別是否引入過某個模塊比如sql模塊,判斷后再進行對應的處理。
//項目文件中判斷
//如果當前套件中有multimedia模塊則引入multimedia模塊
qtHaveModule(multimedia) {QT += multimedia}
//在項目文件中已經通過 QT += multimedia 引入過模塊
contains(QT, multimedia) {}
//代碼文件判斷
#ifdef QT_MULTIMEDIA_LIB
qDebug() < < "multimedia module is enabled";
#else
qDebug() < < "multimedia module is not enabled";
#endif
- 對MDI窗體區域設置背景顏色透明,會發現 QMdiArea{background:transparent;} 無效,哪怕是指定顏色 QMdiArea{background:#ff0000;} 或者 QMdiArea{background-color:#ff0000;} 都不行,這就很無語了,原來要用弱屬性機制才行。QMdiArea{qproperty-background:transparent;}
- 當樣式中啟用了禁用樣式 *:disabled{xxx} 的時候,會發現MDI子窗體無法拉伸了,這應該是Qt內部的BUG,怎么解決呢,只需要重新設置MDI這個類別的禁用樣式的邊框樣式即可。QMdiSubWindow:disabled{border:8px solid rgba(0,0,0,0);}
- 用QProcess執行命令或者啟動可執行文件,默認寫法不支持帶空格的路徑,比如 Program Files ,需要在這個路徑前后加上雙引號才行,估計可能內部會用空格分割字符串導致解析失敗。普通路徑加上引號也能正常執行,所以為了確保以防萬一,統一加上引號即可。
QString cmd = "c:/Program Files/a.exe";
//下面這個會執行失敗
QProcess::startDetached(cmd);
//前后加上引號就可以正常執行
cmd = """ + cmd + """;
QProcess::startDetached(cmd);
- 在循環中取值,臨時變量的定義盡量在循環外層定義,每次在循環里層定義會增加開銷,特別是復雜類型比如QString(基礎類型比如int/bool差別不大),循環次數越多,性能差別越大。
void MainWindow::on_pushButton_clicked()
{
QElapsedTimer timer;
timer.start();
QString s;
QString text = "abc";
for (int i = 0; i < 10000; ++i) {
s = text.at(0);
}
qDebug() < < "方式1" < < timer.nsecsElapsed();
}
void MainWindow::on_pushButton_2_clicked()
{
QElapsedTimer timer;
timer.start();
QString text = "abc";
for (int i = 0; i < 10000; ++i) {
QString s = text.at(0);
}
qDebug() < < "方式2" < < timer.nsecsElapsed();
}
//debug模式下方式1比方式2快6倍+
//release模式下方式1比方式2快30倍+
- Qt的屬性機制非常強大,除了可以用來控制樣式表,也可以很方便的用來傳值,比如qml中的值傳遞,有時候我們寫了一個通用類,希望這個類可以做很多事情,但是又希望其中有一些特殊變量存取值,一種辦法是直接定義私有變量,提供get/set接口函數,還有一種偷懶的辦法就是用屬性setProperty/property,然Qt內部從元對象數據層面自己管理,這樣不用在類中寫對應的變量和get/set函數。但是肯定有性能損耗,性能上肯定比變量低,所以要看具體的實際需求,如果不是非常頻繁的調用setProperty/property,通用性優先的話,那用屬性機制會更方便。個人推薦方式三,繼承通用類,在子類中增加set/get。
void MainWindow::on_pushButton_clicked()
{
QElapsedTimer timer;
timer.start();
for (int i = 0; i < 10000; ++i) {
Test *t = new Test;
//t- >setId(i);
//t- >setName("test");
t- >getName();
}
qDebug() < < "方式1" < < timer.nsecsElapsed();
}
void MainWindow::on_pushButton_2_clicked()
{
QElapsedTimer timer;
timer.start();
for (int i = 0; i < 10000; ++i) {
Test *t = new Test;
//t- >setProperty("id", i);
//t- >setProperty("name", "test");
t- >property("name").toString();
}
qDebug() < < "方式2" < < timer.nsecsElapsed();
}
//對比測試和具體的變量類型無關/int和QString類型產生的性能差別一樣
//setProperty比setxxx方式性能差3倍+
//property比getxxx方式性能差1.3倍
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。
舉報投訴
-
內存
+關注
關注
8文章
3037瀏覽量
74150 -
編程
+關注
關注
88文章
3628瀏覽量
93817 -
函數
+關注
關注
3文章
4338瀏覽量
62765 -
char
+關注
關注
0文章
11瀏覽量
3728
發布評論請先 登錄
相關推薦
windows下QT編程中_TCHAR與QString之間的轉換
;utf16()#define TCHARToQString(x)QString::fromUtf16((x))#define TCHARToQStringN(x,y)QString::fromUtf16
發表于 03-21 11:22
如何將PowerPCB轉成Protel的詳細過程和軟件
如何將PowerPCB轉成Protel的詳細過程第一步:用PowerPCB打開文件,選擇“File”---“Export”導出,如圖:導出格式Format選擇PowerPCBV3.5的,然后
發表于 11-24 16:25
如何將PDF轉成CAD
`現在很多設計圖紙等操作者都需要對PDF格式得圖紙進行二次編輯,但是PDF文件又不能直接進行編輯修改,那我們該如何操作呢,我們可以將PDF轉成CAD文件然后再進行轉換得操作,接下來就讓小編來具體介紹
發表于 07-03 18:17
如何將Cadence的原理圖和PCB轉成PADS資料下載
電子發燒友網為你提供如何將Cadence的原理圖和PCB轉成PADS資料下載的電子資料下載,更有其他相關的電路圖、源代碼、課件教程、中文資料、英文資料、參考設計、用戶指南、解決方案等資料,希望可以幫助到廣大的電子工程師們。
發表于 04-19 08:54
?46次下載
數控直流穩壓電源如何將交流電壓轉成直流電壓?
數控直流穩壓電源如何將交流電壓轉成直流電壓? 數控直流穩壓電源是一種將交流電壓轉換為直流電壓的裝置。在現代電子設備中,直流電壓廣泛應用,因為它在穩定性、可調性和可靠性方面具有優勢。為了滿足設備的要求
評論