自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

親身體驗CORBA: 使用java和C++混合編程

開發(fā) 后端
本文作者通過一次使用SUN的 tnameserv命名服務程序,服務器用JAVA編寫,客戶機分別用JAVA和C++(VC6+omniORB)編寫的試驗,希望通過一次編程的具體操作實例來體驗或明了CORBA思想。

前  言

現(xiàn)在很多人在對CORBA進行學習,大家都已經(jīng)了解到CORBA是一個完全中間性的語言,可以使用接口定義語言(IDL)定義開發(fā)時使用接口的 Client 和實現(xiàn)接口的 Server 所需要的信息。Client 和 Server 的具體實現(xiàn)代碼并不在IDL定義中編寫,而是使用某種目標語言的IDL 編譯器生成所需的代碼存根及helper類,Client 和 Server再使用真正的編程語言來進行具體實現(xiàn)。

為了保證在不同的 CORBA 產(chǎn)品基礎之上構(gòu)建的分布式對象可以相互通信,Client和Server通過ORB(對象請求代理)進行通信。一般的運行流程是Client把請求發(fā)送給ORB,ORB再把請求發(fā)送給Server,Server把返回結(jié)果發(fā)送ORB,ORB再把返回結(jié)果發(fā)送給Client。ORB可以說Client和Server之間的翻譯者。即使Client和Server使用不同的編程語言編寫,只要是符合相同的IDL定義,ORB也可以完成相互的通信。

所有的文檔在強調(diào)服務器及客戶機可以是Java也可以是C++或其他語言(如:Delphi)進行編寫,但在網(wǎng)站或書本是沒有詳細說如何應對多語言客戶機的例子?!禞AVA2核心技術》上面有些說明,但也只是介紹性的文章,故自己下載了omniORB304,進行了一次使用SUN的 tnameserv命名服務程序,服務器用JAVA編寫,客戶機分別用JAVA和C++(VC6+omniORB)編寫的試驗,希望通過一次編程的具體操作實例來體驗或明了CORBA思想。

總體的編寫過程如下:

  • 用IDL定義一個接口文件,描繪要實現(xiàn)的功能,也可以說是定義一個要實現(xiàn)功能的一個模版(SysProp.idl)
  • 使用"IDL to Java"編譯器(這里是IDLJ)將IDL文件轉(zhuǎn)化為Java編程語言中編寫的接口定義,生成所需的代碼存根及helper類
  • 使用Java語言編寫客戶機和服務器的實現(xiàn)程序。
  • 使用"IDL to C++"編譯器(這里是omniidl)將IDL文件轉(zhuǎn)化為C++編程語言中編寫的接口定義,生成所需的代碼存根及helper類
  • 使用C++語言編寫客戶機實現(xiàn)程序(當然也可編寫服務器程序,但本次試驗沒有進行)
  • 起動命名服務tnameserv
  • 起動Java編寫的服務程序
  • 用Java和C++編寫的客戶機分別調(diào)用相應的服務

運行環(huán)境的設定:

總體環(huán)境由jdk1.3+omniORB3.0(www.uk.research.att.com\omniORB\doc\3.0) +vc6 組成,下面說明具體的安裝。

2.1. 安裝JDK1.3

從SUN公司DOWN JDK1.3或者通過其他方式得到jdk1.3進行安裝,再設定相應的環(huán)境變量,在本文測試用的電腦上是如下所示:

  1. CLASSPATH=.;  
  2.             JAVA_HOME=D:\jdk130 

修改原來的PATH變量,添加"%JAVA_HOME%\bin;",如下

  1. PTAH=%JAVA_HOME%\bin;原變量 

注意:我在第一次使用jbuilder的jdk1.3時,服務器不能正常工作,我只是發(fā)覺這么一回事,具體原因與本文無關而沒有進行了解,請諒。

2.2. 安裝VC6

VC6按常規(guī)方式安裝,注意的是:在本文測試用的電腦上安裝在如下位置

C:\Program Files\Microsoft Visual Studio

2.3. 安裝omniORB

從 www.uk.research.att.com\omniORB\doc\3.0 下載omniORB3.0 ( 本文測試所下載的文件是 omniORB_304_x86_win32.zip )。

用WINZIP進行解壓omniORB_304_x86_win32.zip到omniORB_304_x86_win32目錄,目錄內(nèi)存在omni目錄,復制omni目錄內(nèi)的文件到你把想存放的位置。

測試電腦安裝在 C:\omni

根據(jù)C:\omni\README.win32 文檔進行設定,由于運行程序及命令行在控制臺進行,所以本次測試并不根據(jù)文檔要求去設定環(huán)境變量,而是編寫了一個omni.bat,在使用控制臺時,首先運行。

本測試電腦omni.bat內(nèi)容如下

  1. set TOP=c:\omni  
  2.             set path=%TOP%\bin\x86_win32;%path%  
  3.             set LIB=%TOP%\bin\x86_win32;%LIB%  
  4.             set INCLUDE=%TOP%\include;%INCLUDE%  
  5.             set VcOsDir=  
  6.             set VSCommonDir

如果你的電腦VC的環(huán)境變量已經(jīng)設定在你的環(huán)境變量中,那么C:\Program Files\Microsoft Visual Studio\VC98\Bin\VCVARS32.BAT 就可以不運行。否則運行omni.bat前要首先運行VCVARS32.BAT。

實踐過程

約定所有編寫的文件保存在D:\mywork\t1中,omni.bat也在這個目錄內(nèi)

3.1. 編寫SysProp.idl,功能是返回系統(tǒng)屬性

  1. interface SysProp  
  2.             {  string getProperty(in string name);  
  3.             }; 

3.2. 編寫JAVA的服務器

3.2.1. 把IDL文件轉(zhuǎn)化為JAVA編程語言代碼存根類及helper類。

執(zhí)行如下命令

  1. idlj -fall SysProp.idl 

在正常的情況下D:\mywork\t1 目錄內(nèi)將生成以下文件,否則請檢查你的執(zhí)行程序及文件

SysProp.java 
SysPropHelper.java 
SysPropHolder.java 
SysPropOperations.java 
_SysPropImplBase.java 
_SysPropStub.java

3.2.2. 編寫 SysPropServer.java

  1. import org.omg.CosNaming.*;  
  2.             import org.omg.CORBA.*; 

//編寫相對應的服務,一定要從 _類名ImplBase繼承,并實現(xiàn)相應的方法 class SysPropS extends _SysPropImplBase //具體的服務實現(xiàn)

  1. {  public String getProperty(String key)  
  2.             {  
  3.             System.out.println("調(diào)用"+key);  
  4.             String S;  
  5.             S=System.getProperty(key);  
  6.             if (S==null)   { S="null"; }  
  7.             System.out.println(key+"="+S);  
  8.             return S;  
  9.             }  
  10.             }  
  11.             public class SysPropServer          //起動服務的程序  
  12.             {  public static void main(String args[])  
  13.             {  try 
  14.             {  System.out.println("創(chuàng)建和初始化 ORB ");  
  15.             ORB orb = ORB.init(args, null);  
  16.             System.out.println("創(chuàng)建服務對象并將其向 ORB 注冊 ");  
  17.             SysPropS impl = new SysPropS();  
  18.             orb.connect(impl);  
  19.             //打印IOR字符串  
  20.             System.out.println(orb.object_to_string(impl));  
  21.             org.omg.CORBA.Object namingContextObj =orb.resolve_initial_references("NameService");  
  22.                      NamingContext namingContext= NamingContextHelper.narrow(namingContextObj);  
  23.             NameComponent[] path = {new NameComponent("SysProp""")};  
  24.             System.out.println("綁定服務...SysPropS");  
  25.             namingContext.rebind(path, impl);  
  26.             System.out.println("等待調(diào)用...SysPropS");  
  27.             java.lang.Object sync = new java.lang.Object();  
  28.             synchronized (sync)  
  29.             {  sync.wait();  
  30.             }  
  31.             }  
  32.             catch (Exception e)  
  33.             {   System.err.println("Error: " + e);  
  34.             e.printStackTrace(System.out);  
  35.             }  
  36.             }  
  37.             } 

3.3. 編寫JAVA的客戶機

3.3.1. 編寫 SysPropClient.java 使用IOR字符串的方式

注意在代碼內(nèi)有一段注解掉的代碼,用"//使用ORB的方法的開始"開始,用"//使用ORB的方法的結(jié)束"結(jié)束。這段代碼是使用ORB方法的代碼,如果在代碼中"//使用IOR的方法開始"前一行添加"/*",在"//使用IOR的方法結(jié)束"后一行添加"*/",而把"//使用ORB的方法的開始"前面的"/*"去掉,把"//使用ORB的方法的結(jié)束"后面的"*/"去掉,就是使用ORB方法的代碼,程序運行時就是" SysPropClient [環(huán)境變量] "的方式。以下是具體代碼:

  1. import org.omg.CosNaming.*;  
  2.             import org.omg.CORBA.*;  
  3.             public class SysPropClient  
  4.             {  
  5.             public static void main(String args[])  
  6.             {  
  7.             try{  
  8.             String SetInfo,ReturnInfo,ref;  
  9.             org.omg.CORBA.Object objRef;  
  10.             SysProp syspropRef;  
  11.             ORB orb = ORB.init(args, null);  
  12.             //使用IOR的方法開始  
  13.             if (args.length>=1)  
  14.             {  
  15.             ref=args[0];  
  16.             }  
  17.             else 
  18.             {  
  19.             System.out.println("SysPropClient <IOR字符串>  [環(huán)境變量]");  
  20.             return;  
  21.             }  
  22.             objRef = orb.string_to_object(ref);  
  23.             syspropRef = SysPropHelper.narrow(objRef);  
  24.             //使用IOR的方法結(jié)束  
  25.             /*  
  26.             //使用ORB的方法的開始  
  27.             objRef = orb.resolve_initial_references("NameService");  
  28.             NamingContext ncRef = NamingContextHelper.narrow(objRef);  
  29.             // 進行服務定位  
  30.             NameComponent nc = new NameComponent("SysProp", "");  
  31.             NameComponent path[] = {nc};  
  32.             syspropRef = SysPropHelper.narrow(ncRef.resolve(path));  
  33.             //使用ORB的方法的開始結(jié)束  
  34.             */ 
  35.             if (args.length>1)  
  36.             {  
  37.             SetInfo=args[1];  
  38.             }  
  39.             else 
  40.             {  
  41.             SetInfo="java.home";  
  42.             }  
  43.             System.out.println("開始調(diào)用");  
  44.             ReturnInfo = syspropRef.getProperty(SetInfo);  
  45.             System.out.println(SetInfo+"="+ReturnInfo);  
  46.             } catch (Exception e) {  
  47.             System.out.println("ERROR : " + e) ;  
  48.             }  
  49.             }  
  50.             } 

3.3.2. 編譯程序,在文件目錄內(nèi)執(zhí)行如下命令

jAVAC *.JAVA

3.4. 進行測試

第1控制臺,執(zhí)行

tnameserv

測試時如下所示

  1. D:\mywork\t1>java tnameserv  
  2.             Initial Naming Context:  
  3.             IOR:000000000000002849444c3a6f6d672e6f72672f436f734e616d696e672f4e616d696e67436f  
  4.             6e746578743a312e3000000000010000000000000054000101000000000c3139322e3136382e302e  
  5.             31000ca6000000000018afabcafe00000002a999c474000000080000000000000000000000010000  
  6.             0001000000140000000000010020000000000001010000000000  
  7.             TransientNameServer: setting port for initial object references to: 900  
  8.             Ready. 

第2控制臺,執(zhí)行

java SysPropServer

測試時如下所示

  1. D:\mywork\t1>java SysPropServer  
  2.             創(chuàng)建和初始化 ORB  
  3.             創(chuàng)建服務對象并將其向 ORB 注冊  
  4.             IOR:000000000000001049444c3a53797350726f703a312e30000000000100000000000000540001  
  5.             01000000000c3139322e3136382e302e31000ca7000000000018afabcafe00000002a999dbeb0000  
  6.             00080000000000000000000000010000000100000014000000000001002000000000000101000000  
  7.             0000  
  8.             綁定服務...SysPropS  
  9.             等待調(diào)用...SysPropS 

第3控制臺,執(zhí)行

java SysPropClient IOR:XXX JAVA.HOME

測試時如下所示

  1. D:\mywork\t1>java SysPropClient IOR:000000000000001049444c3a53797350726f703a312e  
  2.             3000000000010000000000000054000101000000000c3139322e3136382e302e31000ca700000000  
  3.             0018afabcafe00000002a999dbeb0000000800000000000000000000000100000001000000140000  
  4.             000000010020000000000001010000000000 java.home  
  5.             開始調(diào)用  
  6.             java.home=D:\bea\jdk130\jre  
  7.              

3.5. 編寫C++的IOR客戶機

從實踐來講編寫C++的客戶機程序同JAVA沒有多大的區(qū)別,只不過JAVA是用idlj生成代碼存根類及helper類,而omni是用omniidl來生成代碼存根類及helper類,而編程思想及編碼過程非常相似。

由于C++的程序要調(diào)用omni及VC6的相關文件,所以進入控制臺后,如果VC沒有進行環(huán)境變量設定,那么要先運行 C:\Program Files\Microsoft Visual Studio\VC98\Bin\VCVARS32.BAT,再運行omni.bat,否則直接運行omni.bat后再編譯程序及運行程序。

3.5.1. 把IDL文件轉(zhuǎn)化為C++編程語言代碼存根類及helper類。

執(zhí)行如下命令

omniidl -bcxx SysProp.idl

在正常的情況下D:\mywork\t1 目錄內(nèi)將生成C++編程語言的代碼存根類及helper類SysProp.hh和SysPropSK.cc。否則請檢查你的執(zhí)行程序及文件。

3.5.2. 編寫SysPropC.cc

  1. #include <iostream.h>  
  2.             #include <SysProp.hh>  
  3.             int main(int argc, char** argv)  
  4.             {  
  5.             try {  
  6.             CORBA::ORB_var orb = CORBA::ORB_init(argc, argv, "omniORB3");  
  7.             if( argc < 2 || argc > 3 ) {  
  8.             cout << "usage:  SysPropC <IOR字符串> [環(huán)境變量名]" << endl;  
  9.             return 1;  
  10.             }  
  11.             CORBA::Object_var obj = orb->string_to_object(argv[1]);  
  12.             SysProp_ptr echoref = SysProp::_narrow(obj);  
  13.             if( CORBA::is_nil(echoref) ) {  
  14.             cerr << "沒有對象" << endl;  
  15.             return 1;  
  16.             }  
  17.             const char* message;  
  18.             if (argc==3)  
  19.             {  
  20.             message=argv[2];  
  21.             }  
  22.             else 
  23.             {  
  24.             message="java.home";  
  25.             }  
  26.             CORBA::String_var dest = echoref->getProperty(message);  
  27.             cout <<  (char*)message << "=" <<(char*)dest << endl;  
  28.             orb->destroy();  
  29.             }  
  30.             catch(...) {  
  31.             cerr << "Caught unknown exception." << endl;  
  32.             }  
  33.             return 0;  
  34.             }  
  35.              

3.5.3. 編寫dirc.mak,如下所示

  1. TOP = C:\omni  
  2.             OMNI_DYNAMIC_LIB = msvcstub.lib -NODEFAULTLIB:libcmt.lib -NODEFAULTLIB:libcmtd.lib  
  3.             CORBA_CPPFLAGS = -D__WIN32__ -D__x86__ -D__NT__ -D__OSVERSION__=4 
  4.             CORBA_LIB      = omniORB304_rt.lib omnithread2_rt.lib \  
  5.             $(OMNI_DYNAMIC_LIB) \  
  6.             wsock32.lib advapi32.lib \  
  7.             -libpath:$(TOP)\lib\x86_win32  
  8.             CXXFLAGS       = -O2 -MD -GX $(CORBA_CPPFLAGS) $(DIR_CPPFLAGS)  
  9.             CXXLINKOPTIONS =  
  10.             .SUFFIXES: .cc  
  11.             .cc.obj:  
  12.             cl /nologo /c $(CXXFLAGS) /Tp$<  
  13.             all:: SysPropC.exe  
  14.             SysPropC.exe: SysPropSK.obj SysPropC.obj  
  15.             link -nologo $(CXXLINKOPTIONS) -out:$@ $** $(CORBA_LIB)  
  16.             clean::  
  17.             -del *.obj  
  18.             -del *.exe  
  19.             veryclean::  
  20.             -del *.obj  
  21.             -del echoSK.* echo.hh  
  22.             -del *.exe  
  23.             SysProp.hh SysPropSK.cc: SysProp.idl  
  24.             $(TOP)\bin\x86_win32\omniidl -T -bcxx -Wbh=.hh -Wbs=SK.cc -Wbtp SysProp.idl 

3.5.4. 編譯程序,執(zhí)行如下命令

nmake -f dirc.mak

3.5.5. 測試 

在第4控制臺

SysPropC IOR:XXX JAVA.HOME

本測試如下所示

  1. D:\mywork\t1>syspropc IOR:000000000000001049444c3a53797350726f703a312e3000000000  
  2.             010000000000000054000101000000000c3139322e3136382e302e31000ca7000000000018afabca  
  3.             fe00000002a999dbeb00000008000000000000000000000001000000010000001400000000000100  
  4.             20000000000001010000000000 os.name  
  5.             os.name=Windows 2000 

3.6. 編寫C++的NAME方式客戶機

為了使用NANE方式,必須為OMNI軟件設置注冊表信息,要在注冊表中建立如下數(shù)據(jù)項(regedit)HKEY_LOCAL_MACHINE\SOFTWARE\ORL\omniORB\2.0\NAMESERVICE(字符串)。

NAMESERVICE的值為tnameserv(jdk1.3\bin內(nèi)的程序)啟動的IOR值(第一次設置時自行添加)。

注意為了使用這種方式每次起動tnameserv后要用新IOR值換去舊的IOR值,我測試過用omini的omniNames.exe程序做服務器,IOR值是不變的,但服務器用JVAV編寫就會出錯。如果起動tnameserv,用 c編寫的服務器及客戶機就可以在上面運行。本例子如下所示

  1. Initial Naming Context:  
  2.             IOR:000000000000002849444c3a6f6d672e6f72672f436f734e616d696e672f4e616d696e67436f  
  3.             6e746578743a312e3000000000010000000000000054000101000000000c3139322e3136382e302e  
  4.             31000ca6000000000018afabcafe00000002a999c474000000080000000000000000000000010000  
  5.             0001000000140000000000010020000000000001010000000000  
  6.             TransientNameServer: setting port for initial object references to: 900  
  7.             Ready. 

那么就要把從 IOR:開始(含IOR:)后面的字符串放進注冊表。

3.6.1. 編寫SysPropCC.cc

  1. //使用NAME方式的客戶機  
  2.             #include <iostream.h> 
  3.             #include "SysProp.hh"  
  4.             static CORBA::Object_ptr getObjectReference(CORBA::ORB_ptr orb);  
  5.             int main (int argc, char **argv)  
  6.             {  
  7.             if(  argc != 2 ) {  
  8.             cout << "使用方法: SysPropCC  <環(huán)境變量名><< endl;  
  9.             return 1;  
  10.             }  
  11.             try {  
  12.             CORBA::ORB_var orb = CORBA::ORB_init(argc, argv, "omniORB3");  
  13.             CORBA::Object_var obj = getObjectReference(orb);  
  14.             SysProp_ptr echoref = SysProp::_narrow(obj);  
  15.             const char* message;  
  16.             if (argc==2){message=argv[1];}  
  17.             else {message="java.home"; }  
  18.             CORBA::String_var dest = echoref->getProperty(message);  
  19.             cout <<  (char*)message << "=" <<(char*)dest << endl;  
  20.             orb->destroy();  
  21.             }  
  22.             catch(...) { cerr << "Caught unknown exception." << endl; }  
  23.             return 0;  
  24.             }  
  25.             //////////////////////////////////////////////////////////////////////  
  26.             static CORBA::Object_ptr getObjectReference(CORBA::ORB_ptr orb)  
  27.             {  
  28.             CosNaming::NamingContext_var rootContext;  
  29.             try {  
  30.             // 定位服務器:  
  31.             CORBA::Object_var obj;  
  32.             obj = orb->resolve_initial_references("NameService");  
  33.             // Narrow the reference returned.  
  34.             rootContext = CosNaming::NamingContext::_narrow(obj);  
  35.             if( CORBA::is_nil(rootContext) ) {  
  36.             cerr << " 初始化不成功." << endl;  
  37.             return CORBA::Object::_nil();  
  38.             }  
  39.             }  
  40.             catch(CORBA::ORB::InvalidName& ex) {  
  41.             cerr << " 沒有找到服務" << endl;  
  42.             return CORBA::Object::_nil();  
  43.             }  
  44.             CosNaming::Name name;  
  45.             name.length(1);  
  46.             name[0].id   = (const char*) "SysProp";  
  47.             name[0].kind = (const char*) "";  
  48.             try {  
  49.             // Resolve the name to an object reference.  
  50.             return rootContext->resolve(name);  
  51.             }  
  52.             catch(...) {  cerr << "定位不成功." << endl;}  
  53.             return CORBA::Object::_nil();  
  54.             } 

3.6.2. 編寫dircc.mak

  1. TOP = C:\omni  
  2.             DIR_CPPFLAGS   = -I. -I$(TOP)\include  
  3.             OMNI_DYNAMIC_LIB = msvcstub.lib -NODEFAULTLIB:libcmt.lib -NODEFAULTLIB:libcmtd.lib  
  4.             CORBA_CPPFLAGS = -D__WIN32__ -D__x86__ -D__NT__ -D__OSVERSION__=4 
  5.             CORBA_LIB      = omniORB304_rt.lib omnithread2_rt.lib \  
  6.             $(OMNI_DYNAMIC_LIB) \  
  7.             wsock32.lib advapi32.lib \  
  8.             -libpath:$(TOP)\lib\x86_win32  
  9.             CXXFLAGS       = -O2 -MD -GX $(CORBA_CPPFLAGS) $(DIR_CPPFLAGS)  
  10.             CXXLINKOPTIONS =  
  11.             .SUFFIXES: .cc  
  12.             .cc.obj:  
  13.             cl /nologo /c $(CXXFLAGS) /Tp$<  
  14.             all:: SysPropCc.exe  
  15.             SysPropCc.exe: SysPropSK.obj SysPropCc.obj  
  16.             link -nologo $(CXXLINKOPTIONS) -out:$@ $** $(CORBA_LIB)  
  17.             clean::  
  18.             -del *.obj  
  19.             -del *.exe  
  20.             veryclean::  
  21.             -del *.obj  
  22.             -del echoSK.* echo.hh  
  23.             -del *.exe  
  24.             SysProp.hh SysPropSK.cc: SysProp.idl  
  25.             $(TOP)\bin\x86_win32\omniidl -T -bcxx -Wbh=.hh -Wbs=SK.cc -Wbtp SysProp.idl 

3.6.3. 編譯文件

nmake -f dircc.mak

3.6.4. 測試

在第5控制臺

SysPropCC JAVA.HOME,

測試結(jié)果如下所示

  1. D:\mywork\t1>syspropcc java.home  
  2.             java.home=D:\bea\jdk130\jre 

小  結(jié)

另還使用了j2sdkee1.2.1進行測試,由于j2sdkee1.2.1起動時的nameserver的PORT是1050 所以啟動服務要加參數(shù),對于本文中的程序運行時要如下所示:
java SysPropServer -ORBInitialPort 1050

如果用IOR的方式,也可以用omni的omniNames.exe程序做命名服務器,用C++編寫服務器,客戶機使用Java和C++編寫,c++編寫服務器的例子可見omni的文檔。

本次只是簡單進行了測試,可以使大家了解一下,CORBA的這種特性,多種不同程序語言進行協(xié)作編程的具體運作過程,希望可以拋磚引玉。

參考資料

《JAVA2核心技術 卷II:高級特性》(對于omniORB是2.0的例子)

《The omniORB version 3.0 User's Guid》(omniORB軟件內(nèi)自帶)

原文鏈接:http://www.cnblogs.com/adylee/archive/2008/11/21/1338629.html

【編輯推薦】

  1. Java線程:線程安全與不安全
  2. Java NIO原理圖文分析及代碼實現(xiàn)
  3. Java RMI遠程調(diào)用
  4. 淺談Java的方法覆蓋與變量覆蓋
  5. Java類變量和成員變量初始化過程
責任編輯:林師授 來源: 王輝的博客
相關推薦

2009-07-20 09:53:43

Java混合編程

2011-07-01 14:55:28

Qt QML C++

2012-04-28 15:28:21

JNI混合編程Java

2015-06-29 10:05:10

C#異步編程解析

2011-02-15 13:12:02

JavaC++

2012-03-20 11:37:24

JavaJNI

2013-01-22 14:05:25

Fedora 18

2010-09-16 14:36:20

2011-12-29 10:13:48

FirefoxAndroid版

2010-09-17 13:47:35

2011-05-06 17:57:34

思維R800掃描儀

2013-01-22 11:29:01

3D打印

2010-05-06 10:17:58

混合編程Java程序員

2010-01-11 09:40:02

C++語言

2021-07-11 18:04:04

C語言

2010-01-18 17:14:50

C++語言

2011-05-30 15:29:32

C++

2011-05-18 18:05:47

C#C++

2011-05-18 17:56:38

C#C++

2011-10-27 18:06:24

思科網(wǎng)真
點贊
收藏

51CTO技術棧公眾號