Снова ссылаюсь на shell32, C # Visual Studio
-
22-09-2019 - |
Вопрос
Хммм.Хорошо, после повторного посещения PInvoke я уверен, что не совсем понимаю :-/ (просто спросил этот вопрос)
Позвольте мне проиллюстрировать код, с которым мне нужно справиться.Это работает, когда я использую "Добавить ссылку -> COM -> Элементы управления и автоматизации Microsoft Shell" ...но, к сожалению, это помещает ссылку в мой проект, которая выглядит следующим образом:"C:\Users im\Documents\Visual Studio 2008\Projects\Wing\FileWing\obj\Debug\Interop.Shell32.dll "
Я роюсь в мусорном ведре и ищу предмет, который хочу восстановить.Есть ли какой-нибудь способ, НЕ пробиваясь через PInvoke, сделать это?Или чтобы получить ссылку на system32/shell32.dll, которая позволяет мне использовать этот код во время выполнения?
private void recoverRecyclerBinEntry(string fileName, int size)
{
try
{
Shell Shl = new Shell();
Folder Recycler = Shl.NameSpace(10);
// scans through all the recyclers entries till the one to recover has been found
for (int i = 0; i < Recycler.Items().Count; i++)
{
FolderItem FI = Recycler.Items().Item(i);
string FileName = Recycler.GetDetailsOf(FI, 0);
if (Path.GetExtension(FileName) == "")
FileName += Path.GetExtension(FI.Path);
//Necessary for systems with hidden file extensions.
string FilePath = Recycler.GetDetailsOf(FI, 1);
string combinedPath = Path.Combine(FilePath, FileName);
if (size == FI.Size && fileName == combinedPath)
{
Debug.Write("Match found. Restoring " + combinedPath + "...");
Undelete(FI);
Debug.WriteLine("done.");
}
else
{
Debug.WriteLine("No match");
}
}
}
catch (Exception ex)
{
Debug.WriteLine(ex.Message);
Debug.WriteLine(ex.StackTrace);
}
}
private bool Undelete(FolderItem Item)
{
try
{
foreach (FolderItemVerb FIVerb in Item.Verbs())
{
if (
(FIVerb.Name.ToUpper().Contains("WIEDERHERSTELLEN")) ||
(FIVerb.Name.ToUpper().Contains("ESTORE")) ||
(FIVerb.Name.ToUpper().Contains("NDELETE"))
)
{
FIVerb.DoIt();
return true;
}
}
//execute the first one:
Item.Verbs().Item(0).DoIt();
return true;
}
catch (Exception)
{
Debug.WriteLine("ERROR undeleting");
return false;
}
}
Решение
Прямо сейчас вы смешиваете 2 разные концепции:Взаимодействие PInvoke и COM.
PInvoke позволяет вам получить доступ к собственным функциям C из управляемого кода.Это работает, определяя совместимую с маршалом сигнатуру собственного метода в управляемом коде и помечая ее DllImport
атрибут.Для этого требуется ссылка на метаданные собственной библиотеки DLL, но она не может иметь ее.DLL обнаруживается во время выполнения с использованием обычных правил загрузки для библиотеки DLL Win32.
COM-взаимодействие позволяет вам получать доступ к COM-совместимым объектам из управляемого кода.Это делается путем получения совместимого с marshal управляемого определения COM-интерфейса, а затем получения ссылки на объект одним из нескольких способов.Получение управляемого определения часто достигается путем добавления ссылки на метаданные в PIA (первичную сборку взаимодействия) для компонента COM.До C # 4.0 эта ссылка не может быть удалена без большой работы и должна быть развернута вместе с вашим приложением.
В этом конкретном примере вы используете COM-взаимодействие, а не PInvoke.