Решения Visual Studio / Несколько проектов: как эффективно распространять свойства проекта среди нескольких проектов C ++ - PullRequest
39 голосов
/ 21 сентября 2008

Я работаю с решением Visual Studio 2005 C ++, которое включает в себя несколько проектов (около 30). Исходя из моего опыта, часто становится раздражающим поддерживать все свойства проектов (например, включать путь, путь к lib, связанные библиотеки, параметры генерации кода, ...), так как вам часто приходится нажимать на каждый проект, чтобы изменить их. Ситуация становится еще хуже, когда у вас несколько конфигураций (Debug, Release, Release 64 bit, ...).

Примеры из реальной жизни:

  • Предположим, вы хотите использовать новую библиотеку, и вам нужно добавить путь включения к этой библиотеке для всех проектов. Как вы будете избегать редактирования свойств каждого проекта?
  • Предположим, вы хотите протестировать новую версию библиотеки (скажем, версию 2.1beta), чтобы быстро изменить пути включения / путь к библиотеке / связанную библиотеку для набора проектов?

Примечания:

  • Мне известно, что можно выбрать несколько проектов одновременно, затем щелкнуть правой кнопкой мыши и выбрать "свойства". Однако этот метод работает только для свойств, которые уже были в точности идентичны для разных проектов: его нельзя использовать для добавления пути включения к набору проектов, которые использовали разные пути включения.
  • Я также знаю, что можно глобально изменить параметры среды (Инструменты / Параметры / Проект и решения / Каталоги), однако это не так уж и удовлетворительно, поскольку его нельзя интегрировать в SCM
  • Я также знаю, что можно добавить «Конфигурации» к решениям. Это не помогает, поскольку создает другой набор свойств проекта для поддержки
  • Я знаю, что codegear C ++ Builder 2009 предлагает жизнеспособный ответ на эту потребность с помощью так называемых «наборов опций», которые могут наследоваться несколькими проектами (я использую и Visual Studio, и C ++ Builder, и я все еще думаю, что C ++ Builder качается на определенных аспекты по сравнению с Visual Studio)
  • Я ожидаю, что кто-то предложит "autconf", такой как CMake, однако возможно ли импортировать файлы vcproj в такой инструмент?

Ответы [ 5 ]

31 голосов
/ 21 сентября 2008

Я думаю, что вам нужно исследовать файлы свойств, то есть *. Vsprops (более старый) или *. Props (последний)

Вам делать необходимо добавить файл свойств вручную в каждый проект, но как только это будет сделано, у вас будет несколько проектов, но один. [Vs] файл поддержки. Если вы измените свойства, все проекты наследуют новые настройки.

17 голосов
/ 21 сентября 2008

Мне часто нужно делать что-то подобное, так как я ссылаюсь на статические библиотеки времени выполнения. Я написал программу, чтобы сделать это для меня. Он в основном сканирует все подкаталоги по любому указанному вами пути и идентифицирует любые найденные файлы .vcproj. Затем один за другим, он открывает их, изменяет их и сохраняет их. Поскольку я использую его редко, путь жестко закодирован, но я думаю, что вы сможете настроить его так, как вам нравится.

Другой подход заключается в том, чтобы понять, что файлы проекта Visual Studio представляют собой просто файлы XML и могут управляться вашим любимым классом XML. Я сделал что-то, используя C # XmlDocument для обновления включаемых каталогов, когда было LOT включаемых каталогов, которые я не хотел вводить.:)

Я включаю оба примера. Вам нужно будет изменить их в соответствии со своими потребностями, но они должны помочь вам начать работу.

Это версия C ++:

#include <stdio.h>
#include <tchar.h>
#include <iostream>
#include <fstream>
#include <string>
#include <sstream>
#include <vector>
#include <boost/filesystem/convenience.hpp>
#include <boost/filesystem/operations.hpp>
#include <boost/filesystem/path.hpp>
#include <boost/regex.hpp>
#include <boost/timer.hpp>

using boost::regex;
using boost::filesystem::path;
using namespace std;

vector<path> GetFileList(path dir, bool recursive, regex matchExp);
void FixProjectFile(path file);
string ReadFile( path &file );
void ReplaceRuntimeLibraries( string& contents );
void WriteFile(path file, string contents);

int _tmain(int argc, _TCHAR* argv[])
{
    boost::timer stopwatch;
    boost::filesystem::path::default_name_check(boost::filesystem::native);
    regex projFileRegex("(.*)\\.vcproj");
    path rootPath("D:\\Programming\\Projects\\IPP_Decoder");

    vector<path> targetFiles = GetFileList(rootPath, true, projFileRegex);
    double listTimeTaken = stopwatch.elapsed();

    std::for_each(targetFiles.begin(), targetFiles.end(), FixProjectFile);

    double totalTimeTaken = stopwatch.elapsed();
    return 0;
}

void FixProjectFile(path file) {
    string contents = ReadFile(file);
    ReplaceRuntimeLibraries(contents);
    WriteFile(file, contents);
}

vector<path> GetFileList(path dir, bool recursive, regex matchExp) {
    vector<path> paths;
    try {
        boost::filesystem::directory_iterator di(dir);
        boost::filesystem::directory_iterator end_iter;
        while (di != end_iter) {
            try {
                if (is_directory(*di)) {
                    if (recursive) {
                        vector<path> tempPaths = GetFileList(*di, recursive, matchExp);
                        paths.insert(paths.end(), tempPaths.begin(), tempPaths.end());
                    }
                } else {
                    if (regex_match(di->string(), matchExp)) {
                        paths.push_back(*di);
                    }
                }
            }
            catch (std::exception& e) {
                string str = e.what();
                cout << str << endl;
                int breakpoint = 0;
            }
            ++di;
        }
    }
    catch (std::exception& e) {
        string str = e.what();
        cout << str << endl;
        int breakpoint = 0;
    }
    return paths;
}

string ReadFile( path &file ) {
//  cout << "Reading file: " << file.native_file_string() << "\n";
    ifstream infile (file.native_file_string().c_str(), ios::in | ios::ate);
    assert (infile.is_open());

    streampos sz = infile.tellg();
    infile.seekg(0, ios::beg);

    vector<char> v(sz);
    infile.read(&v[0], sz);

    string str (v.empty() ? string() : string (v.begin(), v.end()).c_str());

    return str;
}

void ReplaceRuntimeLibraries( string& contents ) {
    regex releaseRegex("RuntimeLibrary=\"2\"");
    regex debugRegex("RuntimeLibrary=\"3\"");
    string releaseReplacement("RuntimeLibrary=\"0\"");
    string debugReplacement("RuntimeLibrary=\"1\"");
    contents = boost::regex_replace(contents, releaseRegex, releaseReplacement);
    contents = boost::regex_replace(contents, debugRegex, debugReplacement);
}

void WriteFile(path file, string contents) {
    ofstream out(file.native_file_string().c_str() ,ios::out|ios::binary|ios::trunc); 
    out.write(contents.c_str(), contents.length());
}

Это версия C #. Наслаждайтесь ...

using System;
using System.Collections.Generic;
using System.Text;
using System.Xml;
using System.IO;

namespace ProjectUpdater
{
    class Program
    {
        static public String rootPath = "D:\\dev\\src\\co\\UMC6\\";
        static void Main(string[] args)
        {
            String path = "D:/dev/src/co/UMC6/UMC.vcproj";
            FileStream fs = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.ReadWrite);
            XmlDocument xmldoc = new XmlDocument();
            xmldoc.Load(fs);
            XmlNodeList oldFiles = xmldoc.GetElementsByTagName("Files");
            XmlNode rootNode = oldFiles[0].ParentNode;
            rootNode.RemoveChild(oldFiles[0]);

            XmlNodeList priorNode = xmldoc.GetElementsByTagName("References");
            XmlElement filesNode = xmldoc.CreateElement("Files");
            rootNode.InsertAfter(filesNode, priorNode[0]);

            DirectoryInfo di = new DirectoryInfo(rootPath);
            foreach (DirectoryInfo thisDir in di.GetDirectories())
            {
                AddAllFiles(xmldoc, filesNode, thisDir.FullName);
            }


            List<String> allDirectories = GetAllDirectories(rootPath);
            for (int i = 0; i < allDirectories.Count; ++i)
            {
                allDirectories[i] = allDirectories[i].Replace(rootPath, "$(ProjectDir)");
            }
            String includeDirectories = "\"D:\\dev\\lib\\inc\\ipp\\\"";
            foreach (String dir in allDirectories) 
            {
                includeDirectories += ";\"" + dir + "\"";
            }

            XmlNodeList toolNodes = xmldoc.GetElementsByTagName("Tool");
            foreach (XmlNode node in toolNodes)
            {
                if (node.Attributes["Name"].Value == "VCCLCompilerTool") {
                    try
                    {
                        node.Attributes["AdditionalIncludeDirectories"].Value = includeDirectories;
                    }
                    catch (System.Exception e)
                    {
                        XmlAttribute newAttr = xmldoc.CreateAttribute("AdditionalIncludeDirectories");
                        newAttr.Value = includeDirectories;
                        node.Attributes.InsertBefore(newAttr, node.Attributes["PreprocessorDefinitions"]);
                    }

                }
            }
            String pathOut = "D:/dev/src/co/UMC6/UMC.xml";
            FileStream fsOut = new FileStream(pathOut, FileMode.Create, FileAccess.Write, FileShare.ReadWrite);
            xmldoc.Save(fsOut);

        }
        static void AddAllFiles(XmlDocument doc, XmlElement parent, String path) {
            DirectoryInfo di = new DirectoryInfo(path);
            XmlElement thisElement = doc.CreateElement("Filter");
            thisElement.SetAttribute("Name", di.Name);
            foreach (FileInfo fi in di.GetFiles())
            {
                XmlElement thisFile = doc.CreateElement("File");
                String relPath = fi.FullName.Replace(rootPath, ".\\");
                thisFile.SetAttribute("RelativePath", relPath);
                thisElement.AppendChild(thisFile);
            }
            foreach (DirectoryInfo thisDir in di.GetDirectories())
            {
                AddAllFiles(doc, thisElement, thisDir.FullName);
            }
            parent.AppendChild(thisElement);
        }
        static List<String> GetAllDirectories(String dir)
        {
            DirectoryInfo di = new DirectoryInfo(dir);
            Console.WriteLine(dir);

            List<String> files = new List<String>();
            foreach (DirectoryInfo subDir in di.GetDirectories())
            {
                List<String> newList = GetAllDirectories(subDir.FullName);
                files.Add(subDir.FullName);
                files.AddRange(newList);
            }
            return files;
        }
        static List<String> GetAllFiles(String dir)
        {
            DirectoryInfo di = new DirectoryInfo(dir);
            Console.WriteLine(dir);

            List<String> files = new List<String>();
            foreach (DirectoryInfo subDir in di.GetDirectories())
            {
                List<String> newList = GetAllFiles(subDir.FullName);
                files.AddRange(newList);
            }
            foreach (FileInfo fi in di.GetFiles())
            {
                files.Add(fi.FullName);
            }
            return files;
        }
    }
}
6 голосов
/ 28 августа 2009

Как и предполагалось, вы должны посмотреть листы свойств (файлы .vsprops).
Я написал очень краткое введение в эту функцию здесь .

3 голосов
/ 21 сентября 2008

Да, я бы определенно предложил использовать CMake . CMake - лучший инструмент (я думаю, что я действительно пробовал их все), который может генерировать файлы проекта Studio.

У меня также была проблема преобразования существующих файлов .vcproj в CMakeLists.txt, и я написал Ruby-скрипт , который заботится о большей части преобразования. Скрипт не обрабатывает такие вещи, как этапы после сборки и тому подобное, поэтому требуется некоторая настройка, но он избавит вас от необходимости извлекать все имена исходных файлов из файлов .vcproj.

0 голосов
/ 19 марта 2017

*. Vcxproj - это файлы msbuild. Итак, вы просто берете свойство, которое вам не нужно, во всех файлах проекта и удаляете его. Тогда поместите это в свой список свойств. Затем убедитесь, что все файлы проектов правильно импортируют этот лист свойств.

Это может быть невероятно утомительно для сотен файлов. Я написал инструмент, чтобы сделать это интерактивным:

https://github.com/chris1248/MsbuildRefactor

...