- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我已经使用 JNA 实现了 GetExtendedTcpTable(),但是当我使用该函数时,收到错误:
java.lang.IndexOutOfBoundsException: Bounds exceeds available space : size=28, offset=52
我的函数定义如下:
public interface IPHlpAPIExtended extends IPHlpAPI {
IPHlpAPIExtended INSTANCE = Native.load("IPHlpAPI", IPHlpAPIExtended.class, W32APIOptions.DEFAULT_OPTIONS);
int GetExtendedTcpTable( MIB_TCPTABLE_OWNER_PID pTcpTable, IntByReference pdwSize, boolean bOrder, int ulAf, int TableClass, int Reserved );
@Structure.FieldOrder( { "dwNumEntries", "table" } )
public static class MIB_TCPTABLE_OWNER_PID extends Structure {
public WinDef.DWORD dwNumEntries;
public MIB_TCPROW_OWNER_PID[] table = new MIB_TCPROW_OWNER_PID[]{ new MIB_TCPROW_OWNER_PID() };
public MIB_TCPTABLE_OWNER_PID() {
}
public MIB_TCPTABLE_OWNER_PID( Pointer pointer ) {
super( pointer );
this.read();
}
@Override
public void read() {
super.read();
if ( dwNumEntries.intValue() > 0 ) {
table = ( MIB_TCPROW_OWNER_PID[] ) table[0].toArray( dwNumEntries.intValue() );
} else {
table = new MIB_TCPROW_OWNER_PID[]{ new MIB_TCPROW_OWNER_PID() };
}
}
}
}
还有:
IPHlpAPIExtended.MIB_TCPTABLE_OWNER_PID pTcpTable = new IPHlpAPIExtended.MIB_TCPTABLE_OWNER_PID();
final IntByReference pdwSize = new IntByReference( 0 );
try {
if ( IPHlpAPIExtended.INSTANCE.GetExtendedTcpTable( pTcpTable, pdwSize, true, AF_INET,
IPHlpAPIExtended.TCP_TABLE_CLASS.TCP_TABLE_OWNER_PID_ALL.ordinal(), 0 ) == ERROR_INSUFFICIENT_BUFFER ) {
IPHlpAPIExtended.MIB_TCPTABLE_OWNER_PID newPTcpTable = new IPHlpAPIExtended.MIB_TCPTABLE_OWNER_PID( pTcpTable.getPointer() );
assertThat(
IPHlpAPIExtended.INSTANCE.GetExtendedTcpTable( newPTcpTable, pdwSize, true, AF_INET,
IPHlpAPIExtended.TCP_TABLE_CLASS.TCP_TABLE_OWNER_PID_ALL.ordinal(), 0 ) ).isEqualTo( NO_ERROR );
for ( int i = 0; i < newPTcpTable.dwNumEntries.intValue(); i++ ) {
final IPHlpAPIExtended.MIB_TCPROW_OWNER_PID row = newPTcpTable.table[i];
System.out.println( row.dwOwningPid.intValue() );
}
} else {
// TODO getLasError()
Assertions.fail( "GetExtendedTcpTable terminate with errors" );
}
} catch ( Throwable t ) {
t.printStackTrace();
}
dwNumEntries 字段已正确填充。但是这一行:
table = ( MIB_TCPROW_OWNER_PID[] ) table[0].toArray( dwNumEntries.intValue() );
引发异常。
编辑:
这是 MIB_TCPROW_OWNER_PID 的映射:
@Structure.FieldOrder( { "dwState", "dwLocalAddr", "dwLocalPort", "dwRemoteAddr", "dwRemotePort", "dwOwningPid" } )
public static class MIB_TCPROW_OWNER_PID extends Structure {
public WinDef.DWORD dwState;
public WinDef.DWORD dwLocalAddr;
public WinDef.DWORD dwLocalPort;
public WinDef.DWORD dwRemoteAddr;
public WinDef.DWORD dwRemotePort;
public WinDef.DWORD dwOwningPid;
}
编辑
我的解决方案已更改为函数的签名。
int GetExtendedTcpTable( Memory pTcpTable, IntByReference pdwSize, boolean bOrder, int ulAf, int TableClass, int Reserved );
现在 pTcpTable 它是一个更通用的内存参数,代码如下:
final IPHlpAPIExtended.MIB_TCPTABLE_OWNER_PID pTcpTable = new IPHlpAPIExtended.MIB_TCPTABLE_OWNER_PID();
final IntByReference pdwSize = new IntByReference( 0 );
final Memory mTcpTable = new Memory( pTcpTable.size() );
if ( IPHlpAPIExtended.INSTANCE.GetExtendedTcpTable( mTcpTable, pdwSize, true, AF_INET,
IPHlpAPIExtended.TCP_TABLE_CLASS.TCP_TABLE_OWNER_PID_ALL.ordinal(), 0 ) == ERROR_INSUFFICIENT_BUFFER ) {
final Memory newMTcpTable = new Memory( pdwSize.getValue() );
assertThat( IPHlpAPIExtended.INSTANCE.GetExtendedTcpTable( newMTcpTable, pdwSize, true, AF_INET,
IPHlpAPIExtended.TCP_TABLE_CLASS.TCP_TABLE_OWNER_PID_ALL.ordinal(), 0 ) ).isEqualTo( NO_ERROR );
final IPHlpAPIExtended.MIB_TCPTABLE_OWNER_PID newPTcpTable = new IPHlpAPIExtended.MIB_TCPTABLE_OWNER_PID( newMTcpTable );
System.out.println( "Table Size: " + newPTcpTable.dwNumEntries.intValue() );
for (int i = 0; i < newPTcpTable.dwNumEntries.intValue(); i++) {
final IPHlpAPIExtended.MIB_TCPROW_OWNER_PID item = newPTcpTable.table[i];
System.out.println( "PID: " + item.dwOwningPid.longValue());
}
}
最佳答案
问题是您使用 Structure#toArray
的方式与设计无关。它旨在处理已经为数组的完整大小分配的内存(或者根本没有分配,在这种情况下它会分配)。 但您在以下情况下使用它:在 MIB_TCPTABLE_OWNER_PID
结构的映射中,您已将第二个元素定义为数组,固定大小为 1,因此为其分配了内存。
public MIB_TCPROW_OWNER_PID[] table = new MIB_TCPROW_OWNER_PID[]{ new MIB_TCPROW_OWNER_PID() };
这本身并没有错,您必须在那里声明一些东西,尽管不需要使用 Java 对象初始化该元素。只需分配新的 MIB_TCPROW_OWNER_PID[1]
。
JNA 在分配时立即分配结构大小的内存,因此分配方式的结果是 table
元素已映射到大小恰好为 28 的 native 内存( DWORD
为 4 个字节,单个 MIB_TCPROW_OWNER_PID
元素为 4*6 字节。)
将单个元素初始分配到table
工作正常,但是当您尝试使用Structure#toArray
重新分配该变量时,它会失败内存边界检查,因为较小的内存已分配,toArray()
在创建新内存之前检查内存是否已分配。
可以在纯 Java 中重新分配定义您自己的数组的变量,而无需使用 Jna 的 Structure#toArray()
及其边界检查。如果 native 端分配内存,则此代码在某些情况下应该可以工作;但是,如果您需要自己分配内存,则它将不起作用。
@Override
public void read() {
readField("dwNumEntries");
table = new MIB_TCPROW_OWNER_PID[dwNumEntries.intValue()];
super.read();
}
另一种方法是不去重新定义数组 (table
),而是将 table
映射为 Pointer
或 StructureByReference
(更强类型的指针)。然后,您将访问指向的 native 内存,并动态创建数组,而不是覆盖 read()
。您可以使用 getter 函数,例如:
public MIB_TCPROW_OWNER_PID[] getTable() {
Pointer[] array = table.getPointerArray(0, dwNumEntries.intValue());
MIB_TCPROW_OWNER_PID[] rows = new MIB_TCPROW_OWNER_PID[array.length];
for (int i=0; i < rows.length; i++) {
rows[i] = new MIB_TCPROW_OWNER_PID(array[i]);
}
return rows;
}
最后,正如您所发现的,如果您使用 Memory
对象自己分配内存并返回该对象,则您可以将预先分配的内存传递给构造函数,并且您的代码应该可以工作。
关于Java native 访问 - GetExtendedTcpTable : Bounds exceeds available space,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59030053/
我有一个 Node.JS 自动化,它使用 Puppeteer 并在过程中加载一些 URL。我的代码非常基本,仅使用包文档中记录的非常基本的函数。 自动化计划每 15 分钟使用 crontab 运行一次
我尝试阅读 stackoveflow 上回答的一些问题并根据此更改 eclipse.ini: 现在,除了一个应用程序之外,每个应用程序都可以正常运行。它显示此消息: 无法执行 dex:超出 GC 开销
问题描述: Task A. Amount of subtractions You have an array a length n. There are m queries (li,ri), for
编辑:看起来问题是过度使用#includes 创建圈子。我确保只包括那些需要的,它解决了前两个错误。 但是,我仍然为 BUtton 和 Elevator 得到“指定的多个默认构造函数” 每个错误都有两
在CloudKit中,我尝试通过批处理来保存大量的记录。但是,我的应用程序因以下错误而崩溃: Error pushing local data: 这是我的代码: CKModifyRecordsOpe
我正在尝试使用以下代码将 BigQuery 数据集从 Google Cloud Platform 下载到 R 工作区以对其进行分析: library(bigrquery) library(DBI) l
在 Kubernetes 中 Kubernetes Health Check Probes ,如果 timeoutSeconds 超过 periodSeconds 会怎样?例如: initialDel
我们正在使用 youtube 数据 api v3,并且已经有一段时间没有任何问题了。最近,我们收到了这个 403 异常: The request cannot be completed because
我正在将一个项目从gradle版本3.3转换为4.10.1。该项目主要是使用自定义构建步骤构建的C++代码,而不是CMake(externalNativeBuild)或Android.mk(ndkBu
这是我为查找小于或等于给定编号的跳跃数而编写的代码。它显示错误“超出输出限制” int main() { int t; cin>>t; while(t--) { long long int n
我正在尝试使用 Google Translate REST API 并同时请求以下网址: http://ajax.googleapis.com/ajax/services/language/trans
大多数时候,作为 .Net 开发人员,我们可以自由地在高级抽象世界中玩耍,但有时现实会踢你的私密部分,并告诉你要找到一个真的理解。 我刚刚经历过其中一次。我认为将角落数据列为项目列表就足够了,以便您了
我编写了一个更新函数,但是多次执行将产生错误context deadline exceeded。 我的功能: func Update(link string, m bson.M) { conf
我在我的网络服务器上同时使用 mysql 和 asp.net。我还使用 sqlite 数据库以便能够在另一台设备上使用该数据库。我需要在两个数据库之间发送数据。这是一天需要做很多次的事情。这是我如何做
我在我的应用程序中使用 Google TextToSpeech 已经很长时间了,我的许多用户都在使用离线语音,所以我对使用的资源数量没有任何问题。但是在收到 GoogleTTS 的最新更新后,我所有的
我正在尝试从 MySQL 5.0.45 数据库中删除几行: delete from bundle_inclusions; 客户端工作了一段时间,然后返回错误: Lock wait timeout ex
我试图将一个 ~200G 的文件加载到具有 4 个数据节点的 MySQL 集群中,我的目标表的 DDL 是这样的: CREATE TABLE XXXXXX ( ID BIGINT AUTO
我有这个脚本: def number_of_occurences(c, message): position = message.find(c) if position == -1:
我正在尝试对我的应用程序进行单元测试,但大部分测试都失败了,原因是异步等待失败:超过 30 秒的超时时间,未满足预期:“Home Code”。 我不知道为什么会这样失败,但这是我下面的代码 class
我的 HTML 表单是这样的 但是,当我上传一个 3mb 的文件时,它给出错误: Problem: File exceeded max_file_size" 最佳答案 我最后检查过,MAX_FIL
我是一名优秀的程序员,十分优秀!