詳解如何獲取java中類的所有對象實例
如何在運(yùn)行時獲取一個Java類的所有對象實例呢?這個類可能是任何一個類,既不是單例,也不一定是由Spring管理,也不提供靜態(tài)方法,有的時候還不能修改其代碼,這里給大家介紹一種底層實現(xiàn)的方式,基于jvmti,代碼用C++實現(xiàn)。
首先寫一個java類,包含native方法,傳入Class參數(shù),返回所有Object[]實例
public class InstancesOfClass { /** * native方法 : 返回所有的實例對象 * @param targetClass 需要查詢實例的Class * @return */ public static native Object[] getInstances(Class<?> targetClass); } }
用javah生成.h文件,然后用jni實現(xiàn)cpp部分
static jvmtiIterationControl JNICALL objectInstanceCallback(jlong class_tag, jlong size, jlong* tag_ptr, void* user_data) { *tag_ptr = 1; return JVMTI_ITERATION_CONTINUE; } JNIEXPORT jobjectArray JNICALL Java_com_liubs_findinstances_jvmti_InstancesOfClass_getInstances(JNIEnv* env, jclass clazz, jclass targetClazz) { JavaVM* vm; env->GetJavaVM(&vm); jvmtiEnv* jvmti; vm->GetEnv((void**)&jvmti, JVMTI_VERSION_1_0); jvmtiCapabilities capabilities = {0}; capabilities.can_tag_objects = 1; jvmti->AddCapabilities(&capabilities); jvmti->IterateOverInstancesOfClass(targetClazz, JVMTI_HEAP_OBJECT_EITHER, objectInstanceCallback, NULL); jlong tag = 1; jint count; jobject* instances; jvmti->GetObjectsWithTags(1, &tag, &count, &instances, NULL); printf("Found %d objects with tag\n", count); // 轉(zhuǎn)換 std::vector<jobject> 為 jobjectArray 并返回 jobjectArray result = env->NewObjectArray(count, targetClazz, NULL); for (int i = 0; i < count; i++) { env->SetObjectArrayElement(result, i, instances[i]); } jvmti->Deallocate((unsigned char*)instances); return result; }
再用gcc/g++編譯得到鏈接庫二進(jìn)制文件,mac生成的dylib,linux生成的是.so,windows生成的是.dll, 然后就可以加載動態(tài)鏈接庫文件,調(diào)用native函數(shù)了
public class InstancesOfClass { static { String nativeLib = null; String osName = System.getProperty("os.name").toLowerCase(); if(osName.contains("mac")) { nativeLib = "findins.dylib"; }else if(osName.contains("linux")) { nativeLib = "findins.so"; }else if(osName.contains("windows")) { nativeLib = "findins.dll"; } if(null == nativeLib) { throw new UnsupportedOperationException("不支持當(dāng)前操作系統(tǒng)"); } URL nativeLibURL = InstancesOfClass.class.getClassLoader().getResource(nativeLib); System.load(nativeLibURL.getPath()); } }
這里是一個調(diào)用測試的例子,生成的對象和通過類獲取所有的對象進(jìn)行對比
測試?yán)尤缦拢?/p>
class A{} class B{} public class TestInstancesOfClass { private static <T> boolean isEqual(List<T> list, Object[] insts){ if(null == list && null == insts){ return true; } if(null == list || null == insts || list.size() != insts.length) { return false; } //先按hashcode排個序 list.sort(Comparator.comparingInt(Object::hashCode)); List<Object> list2 = Arrays.stream(insts).sorted(Comparator.comparingInt(Object::hashCode)).collect(Collectors.toList()); //每一個對象一定是全等 for(int i = 0,len=list.size();i<len ;i++) { if(list.get(i) != list2.get(i)) { return false; } } return true; } public static void main(String[] args) { int count = 10; List<A> insts1 = new ArrayList<>(); for(int i = 0; i<count;i++) { insts1.add(new A()); } Object[] insts1_find = InstancesOfClass.getInstances(A.class); System.out.println(insts1); System.out.println(Arrays.asList(insts1_find)); System.out.println("A的所有對象實例是否一致:"+isEqual(insts1,insts1_find)); List<B> insts2 = new ArrayList<>(); for(int i = 0; i<count;i++) { insts2.add(new B()); } Object[] insts2_find = InstancesOfClass.getInstances(B.class); System.out.println(insts2); System.out.println(Arrays.asList(insts2_find)); System.out.println("B的所有對象實例是否一致:"+isEqual(insts2,insts2_find)); } }
結(jié)果輸出如下:
[A@566776ad, A@6108b2d7, A@1554909b, A@6bf256fa, A@6cd8737, A@22f71333, A@13969fbe, A@6aaa5eb0, A@3498ed, A@1a407d53]
[A@566776ad, A@6108b2d7, A@1554909b, A@6bf256fa, A@6cd8737, A@22f71333, A@13969fbe, A@6aaa5eb0, A@3498ed, A@1a407d53]
A的所有對象實例是否一致:true
[B@62043840, B@5315b42e, B@2ef9b8bc, B@5d624da6, B@1e67b872, B@60addb54, B@3f2a3a5, B@4cb2c100, B@6fb554cc, B@614c5515]
[B@62043840, B@5315b42e, B@2ef9b8bc, B@5d624da6, B@1e67b872, B@60addb54, B@3f2a3a5, B@4cb2c100, B@6fb554cc, B@614c5515]
B的所有對象實例是否一致:true
我把這個功能做成了一個maven依賴,直接引入依賴,然后調(diào)用InstancesOfClass.getInstance(Class<?> targetClass)函數(shù)即可
<dependency> <groupId>io.github.liubsyy</groupId> <artifactId>FindInstancesOfClass</artifactId> <version>1.0.1</version> </dependency>
詳細(xì)源碼: github.com/Liubsyy/FindInstancesOfClass
到此這篇關(guān)于詳解如何獲取java中類的所有對象實例的文章就介紹到這了,更多相關(guān)java獲取類的所有對象內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java使用x-www-form-urlencoded發(fā)請求方式
在開發(fā)中經(jīng)常使用JSON格式,但遇到x-www-form-urlencoded格式時,可以通過重新封裝處理,POSTMan和APIpost工具中對此編碼的稱呼不同,分別是x-www-form-urlencoded和urlencoded,分享這些經(jīng)驗希望對他人有所幫助2024-09-09Java啟用Azure Linux虛擬機(jī)診斷設(shè)置
這篇文章主要介紹了Java啟用Azure Linux虛擬機(jī)診斷設(shè)置,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-05-05在SpringBoot項目中使用JetCache緩存的詳細(xì)教程
Spring Boot是一個非常流行的Java開發(fā)框架,JetCache是一個基于注解的高性能緩存框架,本文將介紹如何在Spring Boot項目中使用JetCache緩存,并提供一個詳細(xì)案例來說明如何配置和使用JetCache,需要的朋友可以參考下2024-06-06jdbc+jsp實現(xiàn)簡單員工管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了jdbc+jsp實現(xiàn)簡單員工管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-02-02java-collection中的null,isEmpty用法
這篇文章主要介紹了java-collection中的null,isEmpty用法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02