То, когда простой рефакторинг как “переименовывает поле”, было сделано на одном ответвлении, может быть очень трудно объединить изменения в другие ответвления. (Метод извлечения намного более тверд, поскольку инструменты слияния, кажется, не соответствуют неизменным блокам хорошо),
Теперь в моих мечтах, я думаю об инструменте, который может записать (или разработать), какие четко определенные операции рефакторинга были сделаны на одном ответвлении и затем “воспроизводят” их на другом ответвлении, вместо того, чтобы пытаться объединить каждую строку, на которую влиял рефакторинг.
см. также, "Там интеллектуальный 3-й инструмент слияния, который понимает VB.NET" для другой половины моей боли!
Также сделал, чтобы любой попробовал что-то как MolhadoRef (статья блога о MolhadoRef и Осведомленном о рефакторинге SCM), Это, в теории, осведомленном о рефакторинге управлении исходным кодом.
Вы можете использовать coccinelle для выполнения однотипных операций рефакторинга на разных ветках. Он не будет записывать или выяснять, что делается самостоятельно, вы должны явно указать ему, что делать, но в остальном он будет более или менее легко выполнять один и тот же рефакторинг на стольких ветках, на которые вы ему укажете.
Этот инструмент был использован в ядре linux для обновления использования API и т.д.
Цитируем с веб-страницы:
"Coccinelle - это механизм подбора и преобразования программ. механизм преобразования, который предоставляет язык SmPL (Semantic Patch Язык) для указания желаемых соответствия и преобразования в C коде."
Araxis Merge не понимает общего рефакторинга, но это единственный инструмент трехстороннего слияния, который я использовал. Он доступен как для Mac, так и для Windows, и он поддерживает API автоматизации, поэтому я могу представить, что вы могли бы делать с ним все, что хотите, если бы вы были так склонны. Для справки, я не имею никакого отношения к Araxis, кроме как пользовался их продуктом.
В Linux вы можете использовать Meld или в Windows Winmerge .
В любом случае, оба инструмента «понимают» только строки текста. Для рефакторинга требуется способ понимания кода, который выходит за рамки любого известного мне инструмента слияния / сравнения.
Итак, вот ответ, у меня точно такой же сценарий. Я хотел написать приложение winforms, чтобы позволить обычным пользователям обновить web.config. Вы должны пойти, чтобы получить конфига дурацкий способ...
// the key of the setting
string key = "MyKey";
// the new value you want to change the setting to
string value = "This is my New Value!";
// the path to the web.config
string path = @"C:\web.config";
// open your web.config, so far this is the ONLY way i've found to do this without it wanting a virtual directory or some nonsense
// even "OpenExeConfiguration" will not work
var config = ConfigurationManager.OpenMappedExeConfiguration(new ExeConfigurationFileMap() { ExeConfigFilename = path }, ConfigurationUserLevel.None);
// now that we have our config, grab the element out of the settings
var element = config.AppSettings.Settings[key];
// it may be null if its not there already
if (element == null)
{
// we'll handle it not being there by adding it with the new value
config.AppSettings.Settings.Add(key, value);
}
else
{
// note: if you wanted to you could inspect the current value via element.Value
// in this case, its already present, just update the value
element.Value = value;
}
// save the config, minimal is key here if you dont want huge web.config bloat
config.Save(ConfigurationSaveMode.Minimal, true);
Вот пример того, что он делает
До:
<?xml version="1.0"?>
<configuration>
<appSettings>
<add key="MyKey" value="OldValue" />
</appSettings>
<connectionStrings>
<add name="myConnString" connectionString="blah blah blah" providerName="System.Data.SqlClient" />
</connectionStrings>
</configuration>
После:
<?xml version="1.0"?>
<configuration>
<appSettings>
<add key="MyKey" value="This is my New Value!" />
</appSettings>
<connectionStrings>
<add name="myConnString" connectionString="blah blah blah" providerName="System.Data.SqlClient" />
</connectionStrings>
<system.web>
<trust level="Full" />
<webControls clientScriptsLocation="/aspnet_client/{0}/{1}/" />
</system.web>
</configuration>
Просто будьте осторожны, если вы дадите ему недопустимый путь, он просто создаст конфигурационный файл по этому пути/имени файла. В основном, проверьте файл. Существует сначала
BTW, пока вы находитесь в нем, вы можете написать класс, который представляет ваши настройки в web.config. Как только вы это сделаете, напишите ваши getters/setters для чтения/записи настроек в web.config. После этого можно добавить этот класс в качестве источника данных и перетащить элементы управления базой данных в winform. Это даст вам полностью база данных winform web.config редактор, что вы можете легко выйти за считанные минуты. У меня есть пример работы, который я отправлю завтра.
Таким образом, это относительно простое решение написания Gui для редактирования web.config, некоторые могут сказать, что оно слишком сложное, когда блокнот будет делать просто хорошо, но это работает для меня и моей аудитории.
Он в основном работает, как описано выше, я написал класс, который имел точки конфигурации, которые я хотел в качестве свойств. ctor
открывает файл из пути, а средства получения/установки извлекают данные из возвращенного объекта конфигурации, наконец, он имеет метод сохранения, который записывает его. С помощью этого класса я могу добавить класс в winforms в качестве источника данных и элементов управления перетаскиванием. Оттуда все, что вам нужно сделать, это подключить кнопку, которая вызывает метод сохранения в вашем классе.
Класс конфигурации
using System.Configuration;
// This is a representation of our web.config, we can change the properties and call save to save them
public class WebConfigSettings
{
// This holds our configuration element so we dont have to reopen the file constantly
private Configuration config;
// given a path to a web.config, this ctor will init the class and open the config file so it can map the getters / setters to the values in the config
public WebConfigSettings(string path)
{
// open the config via a method that we wrote, since we'll be opening it in more than 1 location
this.config = this.OpenConfig(path);
}
// Read/Write property that maps to a web.config setting
public string MySetting
{
get { return this.Get("MySetting"); }
set { this.Set("MySetting", value); }
}
// Read/Write property that maps to a web.config setting
public string MySetting2
{
get { return this.Get("MySetting2"); }
set { this.Set("MySetting2", value); }
}
// helper method to get the value of a given key
private string Get(string key)
{
var element = config.AppSettings.Settings[key];
// it may be null if its not there already
if (element == null)
{
// we'll handle it not being there by adding it with the new value
config.AppSettings.Settings.Add(key, "");
// pull the element again so we can set it below
element = config.AppSettings.Settings[key];
}
return element.Value;
}
// helper method to set the value of a given key
private void Set(string key, string value)
{
// now that we have our config, grab the element out of the settings
var element = this.config.AppSettings.Settings[key];
// it may be null if its not there already
if (element == null)
{
// we'll handle it not being there by adding it with the new value
config.AppSettings.Settings.Add(key, value);
}
else
{
// in this case, its already present, just update the value
element.Value = value;
}
}
// Writes all the values to the config file
public void Save()
{
// save the config, minimal is key here if you dont want huge web.config bloat
this.config.Save(ConfigurationSaveMode.Minimal, true);
}
public void SaveAs(string newPath)
{
this.config.SaveAs(path, ConfigurationSaveMode.Minimal, true);
// due to some weird .net issue, you have to null the config out after you SaveAs it because next time you try to save, it will error
this.config = null;
this.config = this.OpenConfig(newPath);
}
// where the magic happens, we'll open the config here
protected Configuration OpenConfig(string path)
{
return ConfigurationManager.OpenMappedExeConfiguration(
new ExeConfigurationFileMap() { ExeConfigFilename = path },
ConfigurationUserLevel.None);
}
}
Создайте, а затем вы можете перейти к дизайнеру winform, перейти к Data > Show Data Sources (Shift + Alt + D). Щелкните правой кнопкой мыши > Добавить новый источник данных и добавьте его как объект, как показано на
Мастер настройки источника данных 1 из 2 http://img109.imageshack.us/img109/8268/98868932.png
Мастер настройки источника данных 2 из 2 http://img714.imageshack.us/img714/7287/91962513.png
Перетащите его (WebConfigSettings, самый верхний) на winform. В моем случае, я удалю навигатор, как это для списка, и у меня только один.
Свежезаполненные элементы управления базой данных http://img96.imageshack.us/img96/8268/29648681.png
В нижней части дизайнера должно находиться что-то вроде webConfigSetticeBindingSource (показано на следующем рисунке). Перейдите к ракурсу кода и измените ctor
на
public Form1()
{
InitializeComponent();
// wire up the actual source of data
this.webConfigSettingsBindingSource.DataSource = new WebConfigSettings(@"c:\web.config");
}
Добавить кнопку сохранения в winform
Кнопка сохранения добавлена http://img402.imageshack.us/img402/8634/73975062.png
Добавить следующий обработчик событий
private void saveButton_Click(object sender, EventArgs e)
{
// get our WebConfigSettings object out of the datasource to do some save'n
var settings = (WebConfigSettings)this.webConfigSettingsBindingSource.DataSource;
// call save, this will write the changes to the file via the ConfigurationManager
settings.Save();
}
Там, теперь у вас есть хороший простой редактор базы данных web.config. Чтобы добавить или удалить поля, просто измените класс WebConfigSettings, обновите источник данных в окне «Источники данных» (после построения), а затем перетащите новые поля в пользовательский интерфейс.
Вам все равно придется подключить код, который указывает web.config для открытия,для этого примера я просто жестко закодировал путь.
Классная вещь здесь - это все значение, которое добавляет графический интерфейс пользователя. Вы можете легко добавить каталог или filebrowser диалоги, вы можете иметь тестеры строки соединения и т.д. Все они очень просты в добавлении и очень мощны для конечного пользователя.
-121--4379709-Darcs поддерживает операцию « token replace » при фиксации, которая заменяет все экземпляры одного маркера на другой, и объединяется так, как нужно.
Инструмент трехстороннего слияния Plastic SCM (www.plasticscm.com) реализует Xmerge, который единственный способен помочь вам объединить код, который был перемещен.