使用dll和Java jni4net时出现UnsatisfiedLinkError异常

5
我是一名有用的助手,可以为您翻译文本。以下是需要翻译的内容:

我一直在通过jni4net从Java中读取C#(dll)函数,在核心Java中,我已成功从dll函数中获取值,但现在我已创建一个动态Web项目,并尝试在servlet中使用相同的功能。但现在只有dll文件成功加载,函数未能成功调用。以下是我到目前为止尝试过的:

我的Servlet:

public class LoginProcess extends HttpServlet {
    private static final long serialVersionUID = 1L;

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

        try {

            Bridge.setVerbose(true);
            Bridge.init();
            Console.WriteLine("Hello .NET world!\n");

            Bridge.LoadAndRegisterAssemblyFrom(new File("C:/Users/ashish.it/workspace/FinalJniWeb/WebContent/WEB-INF/lib/ADHelper.j4n.dll"));

        } catch (IOException e) {
            e.printStackTrace();
        }
        Enum output;
        output=ADHelper.Login("user", "pass");
        System.out.println(output);
    }
}

当我在Java核心中制作这个时,唯一的区别是我没有使用完整路径,而是只使用了“lib / ADHelper.j4n.dll”作为路径,但不知何故,在servlet中它无法正常工作,所以我将其更改为完整路径。无论如何,dll文件已成功加载。

ADHelper.generated.cs

namespace ADHelper {
    public partial class ADHelper_ {

methods.Add(global::net.sf.jni4net.jni.JNINativeMethod.Create(@__type, "Login", "Login2", "(Ljava/lang/String;Ljava/lang/String;)Lsystem/Enum;"));

  private static global::net.sf.jni4net.utils.JniHandle Login2(global::System.IntPtr @__envp, global::net.sf.jni4net.utils.JniLocalHandle @__class, global::net.sf.jni4net.utils.JniLocalHandle UserName, global::net.sf.jni4net.utils.JniLocalHandle Password) {
            // (Ljava/lang/String;Ljava/lang/String;)Lsystem/Enum;
            // (LSystem/String;LSystem/String;)LADHelper/ADHelper+LoginResult;
            global::net.sf.jni4net.jni.JNIEnv @__env = global::net.sf.jni4net.jni.JNIEnv.Wrap(@__envp);
            global::net.sf.jni4net.utils.JniHandle @__return = default(global::net.sf.jni4net.utils.JniHandle);
            try {
            @__return = global::net.sf.jni4net.utils.Convertor.StrongC2Jp<global::ADHelper.ADHelper.LoginResult>(@__env, global::ADHelper.ADHelper.Login(global::net.sf.jni4net.utils.Convertor.StrongJ2CString(@__env, UserName), global::net.sf.jni4net.utils.Convertor.StrongJ2CString(@__env, Password)));
            }catch (global::System.Exception __ex){@__env.ThrowExisting(__ex);}
            return @__return;
        }
}

当我运行proxygen命令时,下划线混入到名称ADHelper类中。在dll文件中有两个名为ADHelper和ADHelper的类。
函数Login()也被更改为Login2(),但我的servlet无法识别Login2(),而可以识别Login()。
生成的Java类ADHelper.java。
package adhelper;

@net.sf.jni4net.attributes.ClrType
public class ADHelper extends system.Object { 

private static system.Type staticType;

    protected ADHelper(net.sf.jni4net.inj.INJEnv __env, long __handle) {
            super(__env, __handle);
    }

    @net.sf.jni4net.attributes.ClrConstructor("()V")
    public ADHelper() {
            super(((net.sf.jni4net.inj.INJEnv)(null)), 0);
        adhelper.ADHelper.__ctorADHelper0(this);
    }

@net.sf.jni4net.attributes.ClrMethod("(LSystem/String;LSystem/String;)LADHelper/ADHelper+LoginResult;")
    public native static system.Enum Login(java.lang.String UserName, java.lang.String Password);

 public static system.Type typeof() {
        return adhelper.ADHelper.staticType;
    }

 private static void InitJNI(net.sf.jni4net.inj.INJEnv env, system.Type staticType) {
        adhelper.ADHelper.staticType = staticType;
    }
}

所有的映射都是正确的,但我的登录函数出现了unsatisfiedLinkError错误。感谢您耐心阅读,请给我解决问题的方案。
在控制台上出现以下错误:
*All Dll file loaded message*
Jun 3, 2015 10:56:39 AM org.apache.catalina.core.StandardWrapperValve invoke
SEVERE: Servlet.service() for servlet LoginProcess threw exception
java.lang.UnsatisfiedLinkError: adhelper.ADHelper.Login(Ljava/lang/String;Ljava/lang/String;)Lsystem/Enum;
    at adhelper.ADHelper.Login(Native Method)
    at com.karvy.login.LoginProcess.doGet(LoginProcess.java:62)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:617)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:723)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:290)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206)
    at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:233)
    at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:191)
    at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:127)
    at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:103)
    at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:109)
    at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:293)
    at org.apache.coyote.http11.Http11Processor.process(Http11Processor.java:861)
    at org.apache.coyote.http11.Http11Protocol$Http11ConnectionHandler.process(Http11Protocol.java:606)
    at org.apache.tomcat.util.net.JIoEndpoint$Worker.run(JIoEndpoint.java:489)
    at java.lang.Thread.run(Thread.java:662)
3个回答

2
也许你可以测试不同上下文中的类加载器从哪个类加载器中加载你的文件。

1
我正在强制从Web上下文类加载器加载dll,但没有任何变化。 - Ashish Sharma

2

我曾认为这是类加载器问题,但后来发现并不是,因为当我压制文件以从同一类加载器中加载时,问题仍然存在。

进一步检查显示,我提供的路径是目录中文件的直接路径,并没有指向servlet上下文范围内文件所在的相同方向。为此,我必须创建一个servlet context对象,然后调用getResuorce方法,以便我可以获取我的文件并准备好使用它。

以下是从servlet上下文范围获取文件的代码:

ServletContext context = getServletContext();
String realPath = context.getRealPath("/WEB-INF/lib/ADHelper.j4n.dll");
File file = new File(realPath);

感谢@surabhi-rai的支�。我�布这个�答是为了以�有人�到这个问题�以在这里找到答案。谢谢并欢�🙂。


1
通过servlet上下文对象获取文件引用,然后调用getResource方法来获取它。
现在你正在指向不在servlet上下文范围内的本地目录。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接