可能大家或多或少都在qmake的.pro/.pri文件中見(jiàn)過(guò)這種代碼
TEMPLATE += fakelib
MY_NAME = $$qtLibraryTarget(dbzhang800)
TEMPLATE -= fakelib
如果真的沒(méi)見(jiàn)過(guò),或許可以從這兒入手分析與學(xué)習(xí)Qt Solution對(duì)qmake的使用
$$qtLibraryTarget 有什么用
我們知道,在Window下,動(dòng)態(tài)庫(kù)的 Debug 和 Release 版不能通用,所以要放置在不同的路徑下或者取不同的名字。在Qt下,我們經(jīng)常見(jiàn)到
QtCore4.dll
QtCored4.dll
而當(dāng)我們自己的工程中需要?jiǎng)討B(tài)庫(kù)時(shí),要想在兼顧各個(gè)平臺(tái)下,給它取個(gè)合適的名字還是有點(diǎn)小小的挑戰(zhàn)的。
既然Qt已經(jīng)能處理自己的動(dòng)態(tài)庫(kù),我們不妨直接借用它提供的機(jī)制,這便是
$$qtLibraryTarget
作用:為各個(gè)平臺(tái)下的庫(kù)生成合適的名字
位置:$QTDIR/mkspecs/features/qt_functions.prf
具體代碼:
defineReplace(qtLibraryTarget) {
unset(LIBRARY_NAME)
LIBRARY_NAME = $$1
mac:!static:contains(QT_CONFIG, qt_framework) {
QMAKE_FRAMEWORK_BUNDLE_NAME = $$LIBRARY_NAME
export(QMAKE_FRAMEWORK_BUNDLE_NAME)
}
contains(TEMPLATE, .*lib):CONFIG(debug, debug|release) {
!debug_and_release|build_pass {
mac:RET = $$member(LIBRARY_NAME, 0)_debug
else:win32:RET = $$member(LIBRARY_NAME, 0)d
}
}
isEmpty(RET):RET = $$LIBRARY_NAME
return($$RET)
}
注:這是Qt4.8下的文件,Qt5.0下與此小有不同。
從這兒你可以看到,在windows下它何時(shí)會(huì)添加一個(gè)d,何時(shí)不會(huì)。
fakelib 干嘛用的
從前面的代碼中,我們看到有這么一句
contains(TEMPLATE, .*lib)
可是,當(dāng)構(gòu)建動(dòng)態(tài)庫(kù)時(shí),不是TEMPLATE確實(shí)就是 lib/vclib這些了么?
恩,構(gòu)建時(shí)沒(méi)有問(wèn)題,可是當(dāng)使用時(shí)呢?比如
TEMPLATE = app
SOURCES += main.cpp
LIBS += $$qtLibraryTarget(dbzhang800)
還能工作么?
恩,所以了。就需要隨便取一個(gè) xxxxxlib 作為模板來(lái)哄騙一下它。
還沒(méi)結(jié)束?
自從認(rèn)識(shí)它以后,一直在自己的程序中大量使用這個(gè)東西。但是有一件事情總是讓人感覺(jué)不爽。
使用它以后:
qmake -tp vc
生成的MSVC的工程文件中,總是不能得到正確庫(kù)文件的名字(具體一點(diǎn),就是Debug模式下沒(méi)有那個(gè)d)
自己用也沒(méi)什么,反正也不是什么大問(wèn)題。只是沒(méi)想到解決方案竟是這樣
SAVE_TEMPLATE = $$TEMPLATE
TEMPLATE = fakelib
MY_NAME = $$qtLibraryTarget(dbzhang800)
TEMPLATE = $$SAVE_TEMPLATE
原因
當(dāng)qmake使用了 -tp 選項(xiàng)后,對(duì)TEMPLATE處理時(shí),只接受一個(gè)值。也就是 += 操作對(duì)其無(wú)效。
具體一點(diǎn),位于qmake源碼 project.cpp 的3043行
} else if(var == QLatin1String("TEMPLATE")) {
if(!Option::user_template.isEmpty()) {
var = ".BUILTIN.USER." + var;
place[var] = QStringList(Option::user_template);
} else {
QString orig_template, real_template;
if(!place[var].isEmpty())
orig_template = place[var].first();
real_template = orig_template.isEmpty() ? "app" : orig_template;
if(!Option::user_template_prefix.isEmpty() && !orig_template.startsWith(Option::user_template_prefix))
real_template.prepend(Option::user_template_prefix);
if(real_template != orig_template) {
var = ".BUILTIN." + var;
place[var] = QStringList(real_template);
}
}