新书推介:《语义网技术体系》
作者:瞿裕忠,胡伟,程龚
   XML论坛     W3CHINA.ORG讨论区     计算机科学论坛     SOAChina论坛     Blog     开放翻译计划     新浪微博  
 
  • 首页
  • 登录
  • 注册
  • 软件下载
  • 资料下载
  • 核心成员
  • 帮助
  •   Add to Google

    >> 本版讨论高级C/C++编程、代码重构(Refactoring)、极限编程(XP)、泛型编程等话题
    [返回] 计算机科学论坛计算机技术与应用『 C/C++编程思想 』 → 使用.x文件模型(1) 查看新帖用户列表

      发表一个新主题  发表一个新投票  回复主题  (订阅本版) 您是本帖的第 9839 个阅读者浏览上一篇主题  刷新本主题   树形显示贴子 浏览下一篇主题
     * 贴子主题: 使用.x文件模型(1) 举报  打印  推荐  IE收藏夹 
       本主题类别:     
     卷积内核 帅哥哟,离线,有人找我吗?
      
      
      威望:8
      头衔:总统
      等级:博士二年级(版主)
      文章:3942
      积分:27590
      门派:XML.ORG.CN
      注册:2004/7/21

    姓名:(无权查看)
    城市:(无权查看)
    院校:(无权查看)
    给卷积内核发送一个短消息 把卷积内核加入好友 查看卷积内核的个人资料 搜索卷积内核在『 C/C++编程思想 』的所有贴子 访问卷积内核的主页 引用回复这个贴子 回复这个贴子 查看卷积内核的博客楼主
    发贴心情 使用.x文件模型(1)

    利用.x文件模型渲染三维模型,首先需要将.x文件中的各种数据分别加载到内存中,主要包括顶点数据、材质数据和纹理数据等。

    网格模型接口ID3DXMESH

    Direct3D扩展实用库定义了多边形网格模型接口ID3DXMesh来表示一个复杂的三维物体模型,它是一个COM接口,继承自ID3DXBaseMesh。

    Direct3D扩展实用库函数D3DXCreateMesh()可用于创建一个Direct3D网格模型对象,该函数声明如下:

    Creates a mesh object using a declarator.

    HRESULT D3DXCreateMesh(  DWORD NumFaces,  DWORD NumVertices,  DWORD Options,  CONST LPD3DVERTEXELEMENT9 * pDeclaration,  LPDIRECT3DDEVICE9 pD3DDevice,  LPD3DXMESH * ppMesh);
    Parameters
    NumFaces
    [in] Number of faces for the mesh. The valid range for this number is greater than 0, and one less than the maximum DWORD (typically 65534), because the last index is reserved.
    NumVertices
    [in] Number of vertices for the mesh. This parameter must be greater than 0.
    Options
    [in] Combination of one or more flags from the D3DXMESH enumeration, specifying options for the mesh.
    pDeclaration
    [in] Array of D3DVERTEXELEMENT9 elements, describing the vertex format for the returned mesh. This parameter must map directly to a flexible vertex format (FVF).
    pD3DDevice
    [in] Pointer to an IDirect3DDevice9 interface, the device object to be associated with the mesh.
    ppMesh
    [out] Address of a pointer to an ID3DXMesh interface, representing the created mesh object.
    Return Values
    If the function succeeds, the return value is D3D_OK. If the function fails, the return value can be one of the following: D3DERR_INVALIDCALL, E_OUTOFMEMORY.

    D3DXMESH
    Flags used to specify creation options for a mesh.

    typedef enum D3DXMESH{    D3DXMESH_32BIT = 0x001,    D3DXMESH_DONOTCLIP = 0x002,    D3DXMESH_POINTS = 0x004,    D3DXMESH_RTPATCHES = 0x008,    D3DXMESH_NPATCHES = 0x4000,    D3DXMESH_VB_SYSTEMMEM = 0x010,    D3DXMESH_VB_MANAGED = 0x020,    D3DXMESH_VB_WRITEONLY = 0x040,    D3DXMESH_VB_DYNAMIC = 0x080,    D3DXMESH_VB_SOFTWAREPROCESSING = 0x8000,    D3DXMESH_IB_SYSTEMMEM = 0x100,    D3DXMESH_IB_MANAGED = 0x200,    D3DXMESH_IB_WRITEONLY = 0x400,    D3DXMESH_IB_DYNAMIC = 0x800,    D3DXMESH_IB_SOFTWAREPROCESSING = 0x10000,    D3DXMESH_VB_SHARE = 0x1000,    D3DXMESH_USEHWONLY = 0x2000,    D3DXMESH_SYSTEMMEM = 0x110,    D3DXMESH_MANAGED = 0x220,    D3DXMESH_WRITEONLY = 0x440,    D3DXMESH_DYNAMIC = 0x880,    D3DXMESH_SOFTWAREPROCESSING = 0x18000,} D3DXMESH, *LPD3DXMESH;
    Constants
    D3DXMESH_32BIT
    The mesh has 32-bit indices instead of 16-bit indices. See Remarks.
    D3DXMESH_DONOTCLIP
    Use the D3DUSAGE_DONOTCLIP usage flag for vertex and index buffers.
    D3DXMESH_POINTS
    Use the D3DUSAGE_POINTS usage flag for vertex and index buffers.
    D3DXMESH_RTPATCHES
    Use the D3DUSAGE_RTPATCHES usage flag for vertex and index buffers.
    D3DXMESH_NPATCHES
    Specifying this flag causes the vertex and index buffer of the mesh to be created with D3DUSAGE_NPATCHES flag. This is required if the mesh object is to be rendered using N-patch enhancement using Direct3D.
    D3DXMESH_VB_SYSTEMMEM
    Use the D3DPOOL_SYSTEMMEM usage flag for vertex buffers.
    D3DXMESH_VB_MANAGED
    Use the D3DPOOL_MANAGED usage flag for vertex buffers.
    D3DXMESH_VB_WRITEONLY
    Use the D3DUSAGE_WRITEONLY usage flag for vertex buffers.
    D3DXMESH_VB_DYNAMIC
    Use the D3DUSAGE_DYNAMIC usage flag for vertex buffers.
    D3DXMESH_VB_SOFTWAREPROCESSING
    Use the D3DUSAGE_SOFTWAREPROCESSING usage flag for vertex buffers.
    D3DXMESH_IB_SYSTEMMEM
    Use the D3DPOOL_SYSTEMMEM usage flag for index buffers.
    D3DXMESH_IB_MANAGED
    Use the D3DPOOL_MANAGED usage flag for index buffers.
    D3DXMESH_IB_WRITEONLY
    Use the D3DUSAGE_WRITEONLY usage flag for index buffers.
    D3DXMESH_IB_DYNAMIC
    Use the D3DUSAGE_DYNAMIC usage flag for index buffers.
    D3DXMESH_IB_SOFTWAREPROCESSING
    Use the D3DUSAGE_SOFTWAREPROCESSING usage flag for index buffers.
    D3DXMESH_VB_SHARE
    Forces the cloned meshes to share vertex buffers.
    D3DXMESH_USEHWONLY
    Use hardware processing only. For mixed-mode device, this flag will cause the system to use hardware (if supported in hardware) or will default to software processing.
    D3DXMESH_SYSTEMMEM
    Equivalent to specifying both D3DXMESH_VB_SYSTEMMEM and D3DXMESH_IB_SYSTEMMEM.
    D3DXMESH_MANAGED
    Equivalent to specifying both D3DXMESH_VB_MANAGED and D3DXMESH_IB_MANAGED.
    D3DXMESH_WRITEONLY
    Equivalent to specifying both D3DXMESH_VB_WRITEONLY and D3DXMESH_IB_WRITEONLY.
    D3DXMESH_DYNAMIC
    Equivalent to specifying both D3DXMESH_VB_DYNAMIC and D3DXMESH_IB_DYNAMIC.
    D3DXMESH_SOFTWAREPROCESSING
    Equivalent to specifying both D3DXMESH_VB_SOFTWAREPROCESSING and D3DXMESH_IB_SOFTWAREPROCESSING.
    Remarks
    A 32-bit mesh (D3DXMESH_32BIT) can theoretically support (2^32)-1 faces and vertices. However, allocating memory for a mesh that large on a 32-bit operating system is not practical.

    一般情况下,参数Option置为D3DMESH_SYSTEMMEM或D3DMESH_MANAGED,表示对Direct3D顶点缓冲区和索引缓冲区使用D3DPOOL_SYSTEMMEM或D3DPOOL_MANAGED内存。

    调用D3DXCreateMesh()函数创建了网格模型对象后,还需要为其载入模型数据,而载入模型数据是比较复杂的。所以在大多数情况下,不直接调用该函数,它被封装在Direct3D扩展实用库函数中,由Direct3D在内部完成网格模型对象的创建和模型数据的载入操作。

    通过.x文件生成网格模型

    复杂的三维模型实际上是由许许多多的多边形构成的,所以首先需要得到这些构成模型的多边形。使用Direct3D功能扩展库函数D3DXLoadMeshFromX(),可以从.X文件中提取多边形信息(包括顶点坐标、颜色、法向量和纹理信息等),生成网格模型。该函数的声明如下:

    Loads a mesh from a DirectX .x file.

    HRESULT D3DXLoadMeshFromX(  LPCTSTR pFilename,  DWORD Options,  LPDIRECT3DDEVICE9 pD3DDevice,  LPD3DXBUFFER * ppAdjacency,  LPD3DXBUFFER * ppMaterials,  LPD3DXBUFFER * ppEffectInstances,  DWORD * pNumMaterials,  LPD3DXMESH * ppMesh);
    Parameters
    pFilename
    [in] Pointer to a string that specifies the filename. If the compiler settings require Unicode, the data type LPCTSTR resolves to LPCWSTR. Otherwise, the string data type resolves to LPCSTR. See Remarks.
    Options
    [in] Combination of one or more flags from the D3DXMESH enumeration, which specifies creation options for the mesh.
    pD3DDevice
    [in] Pointer to an IDirect3DDevice9 interface, the device object associated with the mesh.
    ppAdjacency
    [out] Pointer to a buffer that contains adjacency data. The adjacency data contains an array of three DWORDs per face that specify the three neighbors for each face in the mesh. For more information about accessing the buffer, see ID3DXBuffer.
    ppMaterials
    [out] Pointer to a buffer containing materials data. The buffer contains an array of D3DXMATERIAL structures, containing information from the DirectX file. For more information about accessing the buffer, see ID3DXBuffer.
    ppEffectInstances
    [out] Pointer to a buffer containing an array of effect instances, one per attribute group in the returned mesh. An effect instance is a particular instance of state information used to initialize an effect. See D3DXEFFECTINSTANCE. For more information about accessing the buffer, see ID3DXBuffer.
    pNumMaterials
    [out] Pointer to the number of D3DXMATERIAL structures in the ppMaterials array, when the method returns.
    ppMesh
    [out] Address of a pointer to an ID3DXMesh interface, representing the loaded mesh.
    Return Values
    If the function succeeds, the return value is D3D_OK. If the function fails, the return value can be one of the following values: D3DERR_INVALIDCALL, E_OUTOFMEMORY.

    Remarks
    The compiler setting also determines the function version. If Unicode is defined, the function call resolves to D3DXLoadMeshFromXW. Otherwise, the function call resolves to D3DXLoadMeshFromXA because ANSI strings are being used.

    All the meshes in the file will be collapsed into one output mesh. If the file contains a frame hierarchy, all the transformations will be applied to the mesh.

    For mesh files that do not contain effect instance information, default effect instances will be generated from the material information in the .x file. A default effect instance will have default values that correspond to the members of the D3DMATERIAL9 structure.

    The default texture name is also filled in, but is handled differently. The name will be Texture0@Name, which corresponds to an effect variable by the name of "Texture0" with an annotation called "Name." This will contain the string file name for the texture.

    LPD3DXBUFFER因数据操作的方便性而诞生,它的好处是可以存储顶点位置坐标、材质、纹理等多种类型的Direct3D数据,而不必对每种数据声明一种函数接口类型。可使用接口函数ID3DXBuffer::GetBufferPointer()获取缓冲区中的数据,使用ID3DXBuffer::GetBufferSize()获取缓冲区数据大小。

    载入材质和纹理

    如果函数D3DXLoadMeshFromX()调用成功,那么参数ppMaterials就会获取.x文件中三维模型导出的材质和纹理信息,而pNumMaterials则会获得材质的数目,将材质和纹理信息从中提取出来的代码如下:

    bool init_geometry(){ ID3DXBuffer* material_buffer;
     /*  D3DXLoadMeshFromXA(        LPCSTR pFilename,         DWORD Options,         LPDIRECT3DDEVICE9 pD3DDevice,         LPD3DXBUFFER *ppAdjacency,        LPD3DXBUFFER *ppMaterials,         LPD3DXBUFFER *ppEffectInstances,         DWORD *pNumMaterials,        LPD3DXMESH *ppMesh); */
     if(FAILED(D3DXLoadMeshFromX("airplane.x", D3DXMESH_MANAGED, g_device, NULL, &material_buffer, NULL,        &g_num_materials, &g_mesh))) {  MessageBox(NULL, "Could not find airplane.x", "ERROR", MB_OK);  return false; }
     D3DXMATERIAL* xmaterials = (D3DXMATERIAL*) material_buffer->GetBufferPointer();
     g_mesh_materials = new D3DMATERIAL9[g_num_materials]; g_mesh_textures  = new IDirect3DTexture9*[g_num_materials];
     for(DWORD i = 0; i < g_num_materials; i++) {  g_mesh_materials[i] = xmaterials[i].MatD3D;
      // set ambient reflected coefficient, because .x file do not set it.  g_mesh_materials[i].Ambient = g_mesh_materials[i].Diffuse;
      g_mesh_textures[i] = NULL;
      if(xmaterials[i].pTextureFilename != NULL && strlen(xmaterials[i].pTextureFilename) > 0)    D3DXCreateTextureFromFile(g_device, xmaterials[i].pTextureFilename, &g_mesh_textures[i]);  }
     material_buffer->Release();
     return true;}
    一个三维网格模型通常是由几个子模型组成的,在制作模型时通常为每个子模型分别设置材质和纹理,所以这些子模型就可能使用不同的材质和纹理,因此在Direct3D程序就需要为所有的子模型分别保存材质和纹理。此外,因为每个子模型可能具有不同的材质和纹理,所以在渲染三维模型时也需要逐个子模型分别进行渲染。

    渲染网格模型

    网格模型接口ID3DXMesh实际上是三维物体的顶点缓冲区的集合,它将创建顶点缓冲区、定义灵活顶点格式和绘制顶点缓冲区等功能封装在一个COM对象里,极大地方便了三维物体的绘制。对于以ID3DXMesh表示的三维物体,可以遍历它所有的顶点缓冲区,按照相应的顶点格式将它们分别渲染,也可以直接调用它的接口函数ID3DXMesh::DrawSubset()绘制图形,该函数的声明如下:

    Draws a subset of a mesh.

    HRESULT DrawSubset(  DWORD AttribId);
    Parameters
    AttribId
    [in] DWORD that specifies which subset of the mesh to draw. This value is used to differentiate faces in a mesh as belonging to one or more attribute groups.
    Return Values
    If the method succeeds, the return value is D3D_OK. If the method fails, the return value can be D3DERR_INVALIDCALL.

    Remarks
    The subset that is specified by AttribId will be rendered by the IDirect3DDevice9::DrawIndexedPrimitive method, using the D3DPT_TRIANGLELIST primitive type, so an index buffer must be properly initialized.

    An attribute table is used to identify areas of the mesh that need to be drawn with different textures, render states, materials, and so on. In addition, the application can use the attribute table to hide portions of a mesh by not drawing a given attribute identifier (AttribId) when drawing the frame.

    渲染网格模型的代码如下:

    void render(){ g_device->Clear(0, NULL, D3DCLEAR_TARGET | D3DCLEAR_ZBUFFER, D3DCOLOR_X#050505, 1.0f, 0);
     g_device->BeginScene();
     setup_world_matrix();
     for(DWORD i = 0; i < g_num_materials; i++) {  g_device->SetMaterial(&g_mesh_materials[i]);  g_device->SetTexture(0, g_mesh_textures[i]);
      g_mesh->DrawSubset(i); } 
     g_device->EndScene();
     g_device->Present(NULL, NULL, NULL, NULL);}

    示例程序运行效果图:

    按此在新窗口浏览图片


       收藏   分享  
    顶(0)
      




    ----------------------------------------------
    事业是国家的,荣誉是单位的,成绩是领导的,工资是老婆的,财产是孩子的,错误是自己的。

    点击查看用户来源及管理<br>发贴IP:*.*.*.* 2009/4/27 10:22:00
     
     卷积内核 帅哥哟,离线,有人找我吗?
      
      
      威望:8
      头衔:总统
      等级:博士二年级(版主)
      文章:3942
      积分:27590
      门派:XML.ORG.CN
      注册:2004/7/21

    姓名:(无权查看)
    城市:(无权查看)
    院校:(无权查看)
    给卷积内核发送一个短消息 把卷积内核加入好友 查看卷积内核的个人资料 搜索卷积内核在『 C/C++编程思想 』的所有贴子 访问卷积内核的主页 引用回复这个贴子 回复这个贴子 查看卷积内核的博客2
    发贴心情 
    源程序:

    #include <d3dx9.h>

    #pragma warning(disable : 4127)

    #define CLASS_NAME    "GameApp"

    #define release_com(p)    do { if(p) { (p)->Release(); (p) = NULL; } } while(0)

    IDirect3D9*                g_d3d;
    IDirect3DDevice9*        g_device;
    ID3DXMesh*                g_mesh;
    D3DMATERIAL9*            g_mesh_materials;
    IDirect3DTexture9**        g_mesh_textures;
    DWORD                    g_num_materials;

    void setup_world_matrix()
    {
        D3DXMATRIX mat_world;    
        D3DXMatrixRotationY(&mat_world, timeGetTime() / 1000.0f);
        g_device->SetTransform(D3DTS_WORLD, &mat_world);
    }

    void setup_view_proj_matrix()
    {
        // setup view matrix

        D3DXVECTOR3 eye(0.0f, 10.0f, -20.0f);
        D3DXVECTOR3 at(0.0f, 0.0f, 0.0f);
        D3DXVECTOR3 up(0.0f, 1.0f, 0.0f);

        D3DXMATRIX mat_view;
        D3DXMatrixLookAtLH(&mat_view, &eye, &at, &up);
        g_device->SetTransform(D3DTS_VIEW, &mat_view);

        // setup projection matrix

        D3DXMATRIX mat_proj;
        D3DXMatrixPerspectiveFovLH(&mat_proj, D3DX_PI/4, 1.0f, 1.0f, 100.0f);
        g_device->SetTransform(D3DTS_PROJECTION, &mat_proj);
    }

    bool init_geometry()
    {
        ID3DXBuffer* material_buffer;

        /*
         D3DXLoadMeshFromXA(
            LPCSTR pFilename,
            DWORD Options,
            LPDIRECT3DDEVICE9 pD3DDevice,
            LPD3DXBUFFER *ppAdjacency,
            LPD3DXBUFFER *ppMaterials,
            LPD3DXBUFFER *ppEffectInstances,
            DWORD *pNumMaterials,
            LPD3DXMESH *ppMesh);
        */

        if(FAILED(D3DXLoadMeshFromX("airplane.x", D3DXMESH_MANAGED, g_device, NULL, &material_buffer, NULL,
                                    &g_num_materials, &g_mesh)))
        {
            MessageBox(NULL, "Could not find airplane.x", "ERROR", MB_OK);
            return false;
        }

        D3DXMATERIAL* xmaterials = (D3DXMATERIAL*) material_buffer->GetBufferPointer();

        g_mesh_materials = new D3DMATERIAL9[g_num_materials];
        g_mesh_textures     = new IDirect3DTexture9*[g_num_materials];

        for(DWORD i = 0; i < g_num_materials; i++)
        {
            g_mesh_materials[i] = xmaterials[i].MatD3D;

            // set ambient reflected coefficient, because .x file do not set it.
            g_mesh_materials[i].Ambient = g_mesh_materials[i].Diffuse;

            g_mesh_textures[i] = NULL;

            if(xmaterials[i].pTextureFilename != NULL && strlen(xmaterials[i].pTextureFilename) > 0)    
                D3DXCreateTextureFromFile(g_device, xmaterials[i].pTextureFilename, &g_mesh_textures[i]);    
        }

        material_buffer->Release();

        return true;
    }

    bool init_d3d(HWND hwnd)
    {
        g_d3d = Direct3DCreate9(D3D_SDK_VERSION);

        if(g_d3d == NULL)
            return false;

        D3DPRESENT_PARAMETERS d3dpp;
        ZeroMemory(&d3dpp, sizeof(d3dpp));

        d3dpp.Windowed                    = TRUE;
        d3dpp.SwapEffect                = D3DSWAPEFFECT_DISCARD;
        d3dpp.BackBufferFormat            = D3DFMT_UNKNOWN;
        d3dpp.EnableAutoDepthStencil    = TRUE;
        d3dpp.AutoDepthStencilFormat    = D3DFMT_D16;

        if(FAILED(g_d3d->CreateDevice(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, hwnd, D3DCREATE_SOFTWARE_VERTEXPROCESSING,
                                      &d3dpp, &g_device)))
        {
            return false;
        }
        
        if(! init_geometry())
            return false;

        setup_view_proj_matrix();    

        g_device->SetRenderState(D3DRS_LIGHTING, FALSE);    
        
        return true;
    }

    void cleanup()
    {
        delete[] g_mesh_materials;

        if(g_mesh_textures)
        {
            for(DWORD i = 0; i < g_num_materials; i++)
                release_com(g_mesh_textures[i]);

            delete[] g_mesh_textures;
        }
        
        release_com(g_mesh);
        release_com(g_device);
        release_com(g_d3d);
    }

    void render()
    {
        g_device->Clear(0, NULL, D3DCLEAR_TARGET | D3DCLEAR_ZBUFFER, D3DCOLOR_XRGB(5, 5, 5), 1.0f, 0);

        g_device->BeginScene();

        setup_world_matrix();

        for(DWORD i = 0; i < g_num_materials; i++)
        {
            g_device->SetMaterial(&g_mesh_materials[i]);
            g_device->SetTexture(0, g_mesh_textures[i]);

            g_mesh->DrawSubset(i);
        }
        
        g_device->EndScene();

        g_device->Present(NULL, NULL, NULL, NULL);
    }

    LRESULT WINAPI WinProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
    {
        switch(msg)
        {
        case WM_KEYDOWN:
            if(wParam == VK_ESCAPE)
                DestroyWindow(hwnd);
            break;

        case WM_DESTROY:        
            PostQuitMessage(0);
            return 0;
        }

        return DefWindowProc(hwnd, msg, wParam, lParam);
    }

    int WINAPI WinMain(HINSTANCE inst, HINSTANCE, LPSTR, INT)
    {
        WNDCLASSEX wc;

        wc.cbSize            = sizeof(WNDCLASSEX);
        wc.style            = CS_CLASSDC;
        wc.lpfnWndProc        = WinProc;
        wc.cbClsExtra        = 0;
        wc.cbWndExtra        = 0;
        wc.hInstance        = inst;
        wc.hIcon            = NULL;
        wc.hCursor            = NULL;
        wc.hbrBackground    = NULL;
        wc.lpszMenuName        = NULL;
        wc.lpszClassName    = CLASS_NAME;
        wc.hIconSm            = NULL;

        if(! RegisterClassEx(&wc))
            return -1;

        HWND hwnd = CreateWindow(CLASS_NAME, "Direct3D App", WS_OVERLAPPEDWINDOW, 200, 100, 640, 480,
                                 NULL, NULL, wc.hInstance, NULL);    

        if(hwnd == NULL)
            return -1;

        if(init_d3d(hwnd))
        {
            ShowWindow(hwnd, SW_SHOWDEFAULT);
            UpdateWindow(hwnd);

            MSG msg;
            ZeroMemory(&msg, sizeof(msg));

            while(msg.message != WM_QUIT)
            {
                if(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
                {
                    TranslateMessage(&msg);
                    DispatchMessage(&msg);
                }
                    
                render();
            }
        }

        cleanup();
        UnregisterClass(CLASS_NAME, wc.hInstance);    

        return 0;
    }

    ----------------------------------------------
    事业是国家的,荣誉是单位的,成绩是领导的,工资是老婆的,财产是孩子的,错误是自己的。

    点击查看用户来源及管理<br>发贴IP:*.*.*.* 2009/4/27 10:23:00
     
     GoogleAdSense
      
      
      等级:大一新生
      文章:1
      积分:50
      门派:无门无派
      院校:未填写
      注册:2007-01-01
    给Google AdSense发送一个短消息 把Google AdSense加入好友 查看Google AdSense的个人资料 搜索Google AdSense在『 C/C++编程思想 』的所有贴子 访问Google AdSense的主页 引用回复这个贴子 回复这个贴子 查看Google AdSense的博客广告
    2024/11/22 15:27:33

    本主题贴数2,分页: [1]

    管理选项修改tag | 锁定 | 解锁 | 提升 | 删除 | 移动 | 固顶 | 总固顶 | 奖励 | 惩罚 | 发布公告
    W3C Contributing Supporter! W 3 C h i n a ( since 2003 ) 旗 下 站 点
    苏ICP备05006046号《全国人大常委会关于维护互联网安全的决定》《计算机信息网络国际联网安全保护管理办法》
    93.750ms