다음은 MacOS API에서 JNA를 사용하는 방법에 대한 간단한 예입니다. 아직 배우는 중이기 때문에 설명하지는 않겠습니다. 하지만 다른 목적으로 이를 수행하는 방법에 대한 기본 아이디어를 더 쉽게 이해하고 얻을 수 있도록 코드를 최소화합니다.
대부분의 코드는 Intellij-Community(Apache 라이센스)에서 가져온 것입니다.
우리에게 필요한 것은 jna-플랫폼뿐입니다.
<dependency> <groupId>net.java.dev.jna</groupId> <artifactId>jna-platform</artifactId> <version>5.13.0</version> </dependency>
총 4개의 소스코드 파일이 있습니다. ID는 수정 없이 IntelliJ에서 복사됩니다. MyFoundation 및 MyFoundationLibrary는 각각 Foundation 및 FoundationLibrary에서 잘립니다.
package org.example; import com.sun.jna.Pointer; import static org.example.MyFoundation.*; public class Application { public static void main(String[] args) throws Exception { ID processInfoCls = getObjcClass("NSProcessInfo"); Pointer processInfoSel = createSelector("processInfo"); Pointer processNameSel = createSelector("processName"); ID processInfo = invoke(processInfoCls, processInfoSel); ID processNameNSString = invoke(processInfo, processNameSel); String processName = toStringViaUTF8(processNameNSString); System.out.println(processName); } }
package org.example; import com.sun.jna.NativeLong; public final class ID extends NativeLong { public ID() { } public ID(long peer) { super(peer); } public static final ID NIL = new ID(0L); public boolean booleanValue() { return intValue() != 0; } }
public interface MyFoundationLibrary extends Library { int kCFStringEncodingUTF8 = 0x08000100; ID objc_getClass(String className); Pointer sel_registerName(String selectorName); int CFStringGetLength(ID theString); byte CFStringGetCString(ID theString, byte[] buffer, int bufferSize, int encoding); }
package org.example; import com.sun.jna.*; import java.lang.reflect.Proxy; import java.util.Collections; public class MyFoundation { private static final MyFoundationLibrary myFoundationLibrary; private static final Function myObjcMsgSend; static { myFoundationLibrary = Native.load("Foundation", MyFoundationLibrary.class, Collections.singletonMap("jna.encoding", "UTF8")); NativeLibrary nativeLibrary = ((Library.Handler) Proxy.getInvocationHandler(myFoundationLibrary)).getNativeLibrary(); myObjcMsgSend = nativeLibrary.getFunction("objc_msgSend"); } public static ID getObjcClass(String className) { return myFoundationLibrary.objc_getClass(className); } public static Pointer createSelector(String s) { return myFoundationLibrary.sel_registerName(s); } private static Object [] prepInvoke(ID id, Pointer selector, Object[] args) { Object[] invokArgs = new Object[args.length + 2]; invokArgs[0] = id; invokArgs[1] = selector; System.arraycopy(args, 0, invokArgs, 2, args.length); return invokArgs; } public static ID invoke(final ID id, final Pointer selector, Object... args) { // objc_msgSend is called with the calling convention of the target method // on x86_64 this does not make a difference, but arm64 uses a different calling convention for varargs // it is therefore important to not call objc_msgSend as a vararg function return new ID(myObjcMsgSend.invokeLong(prepInvoke(id, selector, args))); } public static String toStringViaUTF8(ID cfString) { if (ID.NIL.equals(cfString)) return null; int lengthInChars = myFoundationLibrary.CFStringGetLength(cfString); int potentialLengthInBytes = 3 * lengthInChars + 1; // UTF8 fully escaped 16 bit chars, plus nul byte[] buffer = new byte[potentialLengthInBytes]; byte ok = myFoundationLibrary.CFStringGetCString(cfString, buffer, buffer.length, MyFoundationLibrary.kCFStringEncodingUTF8); if (ok == 0) throw new RuntimeException("Could not convert string"); return Native.toString(buffer); } }
위 내용은 JNA(Java Native Access)를 사용하여 MacOS API 호출의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!