Вызов функций Excel/DLL/XLL из C#
У меня есть особая функция в надстройке Excel (XLL). Надстройка является частной собственностью, и у нас нет доступа к исходному коду. Однако нам нужно вызвать некоторые функции, содержащиеся в надстройке, и мы хотели бы вызвать его из программы на C#.
В настоящее время я думал о написании интерфейса C++, вызывающего функцию Excel с помощью xlopers, а затем вызывающего этот интерфейс C++ из C#.
Знает ли кто-нибудь, кто имел предыдущий опыт решения подобных проблем, что было бы лучшим решением для этого?
Энтони
5 ответов
Вам необходимо создать поддельный файл xlcall32.dll, поместить его в тот же каталог, что и ваш XLL (не помещайте собственный файл xlcall32.dll в Excel в ПУТИ). Вот некоторый код:
# include <windows.h>
typedef void* LPXLOPER;
extern "C" void __declspec(dllexport) XLCallVer ( ) {}
extern "C" int __declspec(dllexport) Excel4 (int xlfn, LPXLOPER operRes, int count,... ) { return 0; }
extern "C" int __declspec(dllexport) Excel4v(int xlfn, LPXLOPER operRes, int count, LPXLOPER far opers[]) {return 0;}
Теперь предположим, что у меня есть XLL-файл с именем xll-dll.xll с функцией, которая вызывается (используйте для поиска имен экспортируемых функций "использующий файл.exe"). __cdecl xlAdd (XLOPER* pA, XLOPER* pB);
Следующий код вызывает это:
# include <windows.h>
# include <iostream>
// your own header that defines XLOPERs
# include <parser/xll/xloper.hpp>
// pointer to function taking 2 XLOPERS
typedef XLOPER * (__cdecl *xl2args) (XLOPER* , XLOPER* ) ;
void test(){
/// get the XLL address
HINSTANCE h = LoadLibrary("xll-dll.xll");
if (h != NULL){
xl2args myfunc;
/// get my xll-dll.xll function address
myfunc = (xl2args) GetProcAddress(h, "xlAdd");
if (!myfunc) { // handle the error
FreeLibrary(h); }
else { /// build some XLOPERS, call the remote function
XLOPER a,b, *c;
a.xltype = 1; a.val.num = 1. ;
b.xltype = 1; b.val.num = 2. ;
c = (*myfunc)(&a,&b);
std::cout << " call of xll " << c->val.num << std::endl; }
FreeLibrary(h); }
}
int main()
{test();}
Мой exe действительно работает (к моему собственному удивлению), и выдает 3, как и ожидалось. Вы должны иметь некоторые знания о том, что ваш XLL ожидает от параметров. Если он выделяет некоторую память, вы должны проверить, установлен ли #define xlbitDLLFree 0x4000 для вашего XLOPER c->type, и вызвать обратный вызов "xlAutoFree".
Вы можете попробовать XLL Plus http://www.planatechsolutions.com/xllplus/default.htm. Это немного дорого, но функция XLL Wrapper Libraries - именно то, что вы ищете:
"Иногда полезно иметь возможность вызывать ваши надстройки Excel из других сред, таких как программы командной строки или интерактивные приложения, написанные на C++, Java, C# или Visual Basic. Набор инструментов Xll Wrapper содержит инструменты, среду выполнения библиотека, примеры и документация, помогающие в разработке COM-модулей и сборок.NET, которые обертывают надстройки Excel XLL "
DLL-надстройка Excell может быть написана на C#. Если это так, то вы, вероятно, можете вообще обойти Excel. Может быть, нужна обертка, хотя
Проверьте, доступна ли сборка в программных файлах, где вы устанавливаете плагин, ссылайтесь на эту сборку непосредственно в ваш проект на C# - проверьте в обозревателе объектов ваш класс, метод или объект
Вы должны быть в состоянии использовать рефлексию, чтобы получить доступ к своему дополнению. попробуйте использовать Отражатель, чтобы увидеть, что доступно.