D2D引擎与GDI\GDI+绘制效果对比

article/2025/9/26 19:21:59

本例主要是对比D2D和GDI在绘制文字、线条的区别,以及D2D与GDI+在绘制图片时的区别。

D2D是基于COM组件开发的,使用前的CoInitialize(NULL)是必须的;另外,GDI+的初始化GdiplusStartup()也别忘了。

废话少说,完整代码如下:

// D2DDemo.cpp : 定义应用程序的入口点。
//#include "stdafx.h"
#include "D2DDemo.h"
#include <D2D1.h>
#include <DWrite.h>
#pragma comment(lib, "D2D1")
#pragma comment(lib, "DWrite")
#include <atlbase.h>
#include <atlcom.h>
#include <wincodec.h>
#pragma comment(lib, "windowscodecs")
#include <GdiPlus.h>
#pragma comment(lib, "GdiPlus")
using namespace Gdiplus;#define MAX_LOADSTRING 100
// 全局变量:
HINSTANCE hInst;								// 当前实例
TCHAR szTitle[MAX_LOADSTRING];					// 标题栏文本
TCHAR szWindowClass[MAX_LOADSTRING];			// 主窗口类名// 此代码模块中包含的函数的前向声明:
ATOM				MyRegisterClass(HINSTANCE hInstance);
BOOL				InitInstance(HINSTANCE, int);
LRESULT CALLBACK	WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK	About(HWND, UINT, WPARAM, LPARAM);
BOOL				InitD2DResource();
BOOL				InitDeviceResource(HWND hWnd);
void				D2DDraw();
BOOL				OnCreate(HWND hWnd);
HRESULT				LoadBitmapFromFile(ID2D1RenderTarget *pRenderTarget,IWICImagingFactory *pIWICFactory,PCWSTR uri,UINT destinationWidth,UINT destinationHeight,ID2D1Bitmap **ppBitmap);
//这里来定义全局变量
CComPtr<ID2D1Factory>			g_pD2d1Factory;
CComPtr<IDWriteFactory>			g_pDWriteFactory;
CComPtr<IDWriteTextFormat>		g_pDWriteFormat;
CComPtr<ID2D1HwndRenderTarget>	g_pD2D1HwndRender;
CComPtr<ID2D1GdiInteropRenderTarget>	g_pD2DGdiRender;
CComPtr<IWICImagingFactory>		g_pWicImageFactory;
CComPtr<ID2D1Bitmap>			g_pBkImage;
LOGFONT g_logFont;
HFONT	g_hTextFont	= NULL;
#define FREE_COM_PTR(x)	{ if (x) { x->Release(); x = NULL; } }int APIENTRY _tWinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,LPTSTR    lpCmdLine,int       nCmdShow)
{UNREFERENCED_PARAMETER(hPrevInstance);UNREFERENCED_PARAMETER(lpCmdLine);HRESULT hr = CoInitialize(NULL);ULONG_PTR ptr;GdiplusStartupInput	input;GdiplusStartup(&ptr, &input, NULL);// TODO: 在此放置代码。MSG msg;HACCEL hAccelTable;memset(&g_logFont, 0, sizeof(LOGFONT));g_logFont.lfHeight	= 40;g_logFont.lfWidth	= 0;g_logFont.lfWeight	= FW_NORMAL;g_logFont.lfCharSet	= DEFAULT_CHARSET;g_logFont.lfOutPrecision	= OUT_DEFAULT_PRECIS;g_logFont.lfClipPrecision	= CLIP_DEFAULT_PRECIS;g_logFont.lfQuality			= DEFAULT_QUALITY;g_logFont.lfPitchAndFamily	= DEFAULT_PITCH|FF_SWISS;wcscpy(g_logFont.lfFaceName, L"微软雅黑");g_hTextFont = CreateFontIndirect(&g_logFont);// 初始化全局字符串LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);LoadString(hInstance, IDC_D2DDEMO, szWindowClass, MAX_LOADSTRING);MyRegisterClass(hInstance);// 执行应用程序初始化:if (!InitInstance (hInstance, nCmdShow)){return FALSE;}hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_D2DDEMO));// 主消息循环:while (GetMessage(&msg, NULL, 0, 0)){if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)){TranslateMessage(&msg);DispatchMessage(&msg);}}if ( g_hTextFont ){DeleteObject(g_hTextFont);g_hTextFont = NULL;}GdiplusShutdown(ptr);CoUninitialize();return (int) msg.wParam;
}//
//  函数: MyRegisterClass()
//
//  目的: 注册窗口类。
//
//  注释:
//
//    仅当希望
//    此代码与添加到 Windows 95 中的“RegisterClassEx”
//    函数之前的 Win32 系统兼容时,才需要此函数及其用法。调用此函数十分重要,
//    这样应用程序就可以获得关联的
//    “格式正确的”小图标。
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{WNDCLASSEX wcex;wcex.cbSize = sizeof(WNDCLASSEX);wcex.style			= CS_HREDRAW | CS_VREDRAW;wcex.lpfnWndProc	= WndProc;wcex.cbClsExtra		= 0;wcex.cbWndExtra		= 0;wcex.hInstance		= hInstance;wcex.hIcon			= LoadIcon(hInstance, MAKEINTRESOURCE(IDI_D2DDEMO));wcex.hCursor		= LoadCursor(NULL, IDC_ARROW);wcex.hbrBackground	= (HBRUSH)(COLOR_WINDOW+1);wcex.lpszMenuName	= MAKEINTRESOURCE(IDC_D2DDEMO);wcex.lpszClassName	= szWindowClass;wcex.hIconSm		= LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));return RegisterClassEx(&wcex);
}//
//   函数: InitInstance(HINSTANCE, int)
//
//   目的: 保存实例句柄并创建主窗口
//
//   注释:
//
//        在此函数中,我们在全局变量中保存实例句柄并
//        创建和显示主程序窗口。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{HWND hWnd;hInst = hInstance; // 将实例句柄存储在全局变量中hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);if (!hWnd){return FALSE;}ShowWindow(hWnd, nCmdShow);UpdateWindow(hWnd);return TRUE;
}//
//  函数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
//  目的: 处理主窗口的消息。
//
//  WM_COMMAND	- 处理应用程序菜单
//  WM_PAINT	- 绘制主窗口
//  WM_DESTROY	- 发送退出消息并返回
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{int wmId, wmEvent;PAINTSTRUCT ps;HDC hdc;switch (message){case WM_COMMAND:wmId    = LOWORD(wParam);wmEvent = HIWORD(wParam);// 分析菜单选择:switch (wmId){case IDM_ABOUT:DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);break;case IDM_EXIT:DestroyWindow(hWnd);break;default:return DefWindowProc(hWnd, message, wParam, lParam);}break;case WM_CREATE:OnCreate(hWnd);break;case WM_PAINT:hdc = BeginPaint(hWnd, &ps);// TODO: 在此添加任意绘图代码...D2DDraw();EndPaint(hWnd, &ps);break;case WM_DESTROY:PostQuitMessage(0);break;case WM_SIZE:{if ( g_pD2D1HwndRender ){int nWidth = LOWORD(lParam);int nHeight= HIWORD(lParam);D2D1_SIZE_U sz = D2D1::SizeU(nWidth, nHeight);g_pD2D1HwndRender->Resize(sz);}}case WM_ERASEBKGND:return TRUE;default:return DefWindowProc(hWnd, message, wParam, lParam);}return 0;
}// “关于”框的消息处理程序。
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{UNREFERENCED_PARAMETER(lParam);switch (message){case WM_INITDIALOG:return (INT_PTR)TRUE;case WM_COMMAND:if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL){EndDialog(hDlg, LOWORD(wParam));return (INT_PTR)TRUE;}break;}return (INT_PTR)FALSE;
}BOOL InitD2DResource()
{BOOL bRet = FALSE;ID2D1Factory*			pD2dFactory			= NULL;IDWriteFactory*			pDwriteFactory		= NULL;IDWriteTextFormat*		pDwriteTextFormat	= NULL;IWICImagingFactory*		pWicImgFactory		= NULL;try{HRESULT hr = D2D1CreateFactory(D2D1_FACTORY_TYPE_SINGLE_THREADED, &pD2dFactory);if ( FAILED(hr) )throw L"D2D1CreateFactory error!";hr = CoCreateInstance( CLSID_WICImagingFactory, NULL, CLSCTX_INPROC_SERVER,IID_IWICImagingFactory, (LPVOID*)&pWicImgFactory );if ( FAILED(hr) )throw L"CoCreateInstance error!";hr = DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory), \reinterpret_cast<IUnknown**>(&pDwriteFactory));if ( FAILED(hr) )throw L"DWriteCreateFactory error!";hr = pDwriteFactory->CreateTextFormat(L"微软雅黑", NULL, DWRITE_FONT_WEIGHT_REGULAR, \DWRITE_FONT_STYLE_NORMAL, DWRITE_FONT_STRETCH_NORMAL, 30.0f, L"chs", &pDwriteTextFormat);if ( FAILED(hr) )throw L"CreateTextFormat error!";//水平居中pDwriteTextFormat->SetTextAlignment(DWRITE_TEXT_ALIGNMENT_CENTER);//垂直居中pDwriteTextFormat->SetParagraphAlignment(DWRITE_PARAGRAPH_ALIGNMENT_CENTER);bRet = TRUE;}catch(WCHAR* pMsg){MessageBox(NULL, pMsg, L"InitD2DResource出错:", IDOK);}catch(...){}if ( bRet ){g_pD2d1Factory		= pD2dFactory;g_pDWriteFactory	= pDwriteFactory;g_pDWriteFormat		= pDwriteTextFormat;g_pWicImageFactory	= pWicImgFactory;}else{FREE_COM_PTR(pD2dFactory);FREE_COM_PTR(pDwriteFactory);FREE_COM_PTR(pDwriteTextFormat);FREE_COM_PTR(pWicImgFactory);}return bRet;
}BOOL InitDeviceResource(HWND hWnd)
{if ( g_pD2d1Factory == NULL )return FALSE;RECT rc;GetClientRect(hWnd, &rc);D2D1_SIZE_U sz = D2D1::SizeU(rc.right-rc.left, rc.bottom-rc.top);D2D1_RENDER_TARGET_PROPERTIES rtProps = D2D1::RenderTargetProperties();rtProps.usage =  D2D1_RENDER_TARGET_USAGE_GDI_COMPATIBLE;HRESULT hr = g_pD2d1Factory->CreateHwndRenderTarget( rtProps, \D2D1::HwndRenderTargetProperties(hWnd, sz), &g_pD2D1HwndRender);if ( FAILED(hr) )return FALSE;hr = g_pD2D1HwndRender->QueryInterface(__uuidof(ID2D1GdiInteropRenderTarget), (void**)&g_pD2DGdiRender);return SUCCEEDED(hr);
}void D2DDraw()
{//D2D GDI混合绘图if ( g_pD2D1HwndRender == NULL )return ;g_pD2D1HwndRender->BeginDraw();//D2D绘制g_pD2D1HwndRender->Clear(D2D1::ColorF(RGB(255,255,255), 1.0));//绘制背景位图if ( g_pBkImage ){D2D1_RECT_F desRc = D2D1::RectF(420.0f, 0.0f, 720.0f, 200.0f);g_pD2D1HwndRender->DrawBitmap(g_pBkImage, &desRc);}const WCHAR* pText = L"Hello world!!! by D2D";D2D1_RECT_F rc = D2D1::RectF(0,0,400,100);CComPtr<ID2D1SolidColorBrush> pTextBrush;g_pD2D1HwndRender->CreateSolidColorBrush(D2D1::ColorF(RGB(155,0,255), 2.0), &pTextBrush);CComPtr<ID2D1SolidColorBrush> pLineBrush;g_pD2D1HwndRender->CreateSolidColorBrush(D2D1::ColorF(RGB(0,0,255), 1.0), &pLineBrush);g_pD2D1HwndRender->DrawLine(D2D1::Point2F(0.0f, 0.0f), D2D1::Point2F(400.0F, 100.0F), pLineBrush, 1.0);g_pD2D1HwndRender->DrawText(pText, wcslen(pText), g_pDWriteFormat, rc, pTextBrush);//GDI绘制if ( g_pD2DGdiRender == NULL ){g_pD2D1HwndRender->EndDraw();return ;}HDC hDC = NULL;HRESULT hr = g_pD2DGdiRender->GetDC(D2D1_DC_INITIALIZE_MODE_COPY, &hDC);if ( SUCCEEDED(hr) ){{Graphics g(hDC);Image image(L"1.jpg");g.DrawImage(&image, 420, 220, 300, 200);}//绘制文字HFONT hOldFont = (HFONT)SelectObject(hDC, g_hTextFont);SetTextColor(hDC, RGB(255,0,155));SetBkMode(hDC, TRANSPARENT);RECT rcText = {0,100,400,200};pText = L"Hello world!!! by GDI";DrawText(hDC, pText, wcslen(pText), &rcText, DT_CENTER|DT_VCENTER|DT_SINGLELINE);SelectObject(hDC, hOldFont);//绘制线条HPEN hPen = CreatePen(PS_SOLID, 2, RGB(255,0,0));HPEN hOldPen = (HPEN)SelectObject(hDC, hPen);POINT pt;MoveToEx(hDC, 0, 100, &pt);LineTo(hDC, 400, 200);SelectObject(hDC, hOldPen);DeleteObject(hPen);g_pD2DGdiRender->ReleaseDC(NULL);}g_pD2D1HwndRender->EndDraw();
}BOOL OnCreate( HWND hWnd )
{if ( !InitD2DResource() )goto __error;if ( !InitDeviceResource(hWnd) )goto __error;ID2D1Bitmap* pBitmap = NULL;HRESULT hr = LoadBitmapFromFile(g_pD2D1HwndRender, g_pWicImageFactory, L"1.jpg", 0, 500, &pBitmap);if ( SUCCEEDED(hr) ){g_pBkImage = pBitmap;}goto __successed;
__error:PostQuitMessage(0);return FALSE;
__successed:return TRUE;
}HRESULT LoadBitmapFromFile(ID2D1RenderTarget *pRenderTarget,IWICImagingFactory *pIWICFactory,PCWSTR uri,UINT destinationWidth,UINT destinationHeight,ID2D1Bitmap **ppBitmap)
{HRESULT hr = S_OK;CComPtr<IWICBitmapDecoder>		pDecoder = NULL;CComPtr<IWICBitmapFrameDecode>	pSource = NULL;CComPtr<IWICStream>				pStream = NULL;CComPtr<IWICFormatConverter>	pConverter = NULL;CComPtr<IWICBitmapScaler>		pScaler = NULL;hr = pIWICFactory->CreateDecoderFromFilename(uri,NULL,GENERIC_READ,WICDecodeMetadataCacheOnLoad,&pDecoder);if ( FAILED(hr) )return hr;// Create the initial frame.hr = pDecoder->GetFrame(0, &pSource);if ( FAILED(hr) )return hr;hr = pIWICFactory->CreateFormatConverter(&pConverter);if ( FAILED(hr) )return hr;// If a new width or height was specified, create an// IWICBitmapScaler and use it to resize the image.if (destinationWidth == 0 && destinationHeight == 0)return S_FALSE;UINT originalWidth, originalHeight;hr = pSource->GetSize(&originalWidth, &originalHeight);if ( FAILED(hr) )return hr;if (destinationWidth == 0){FLOAT scalar = static_cast<FLOAT>(destinationHeight) / static_cast<FLOAT>(originalHeight);destinationWidth = static_cast<UINT>(scalar * static_cast<FLOAT>(originalWidth));}else if (destinationHeight == 0){FLOAT scalar = static_cast<FLOAT>(destinationWidth) / static_cast<FLOAT>(originalWidth);destinationHeight = static_cast<UINT>(scalar * static_cast<FLOAT>(originalHeight));}hr = pIWICFactory->CreateBitmapScaler(&pScaler);if ( FAILED(hr) )return hr;hr = pScaler->Initialize(pSource,destinationWidth,destinationHeight,WICBitmapInterpolationModeCubic);if ( FAILED(hr) )return hr;hr = pConverter->Initialize( pScaler, GUID_WICPixelFormat32bppPBGRA, WICBitmapDitherTypeNone,NULL, 0.f, WICBitmapPaletteTypeMedianCut );if ( FAILED(hr) )return hr;// Create a Direct2D bitmap from the WIC bitmap.hr = pRenderTarget->CreateBitmapFromWicBitmap( pConverter, NULL, ppBitmap );return hr;
}
基于COM的对象在使用完之后都应该调用Release(),当引用计数为0时,将删除这个对象。这里为了使用方便,用到了COM里的智能指针CComPtr,析构时会自动调用对象的Release方法,可以查看ATL源码。

运行图如下:


对比效果可以明显发现,D2D绘制文字、线条(实际上调用的是DirectWrite)效果好于GDI,GDI的锯齿太明显了。在绘图方面,感觉D2D和GDI+的绘制效果差不多。但是,GDI+的性能大家都知道那叫一个渣,而且不支持硬件加速。


http://chatgpt.dhexx.cn/article/22GrLgjt.shtml

相关文章

D2D加载图片资源(2)

※先给大家认识下什么是WIC WIC全称是Windows Image Component&#xff0c;是一套扩展的API&#xff0c;用来处理数字图像&#xff0c;它是基于COM组件的。该API包含非常丰富的图像处理函数。 详细可以点击百度文库再看看 http://baike.baidu.com/view/414115.htm?fraladdi…

5G关键技术,D2D通信-ielab

D2D&#xff08;device to device&#xff09;技术是指两个对等的用户节点之间直接进行通信的一种通信方式。在由D2D通信用户组成的分散式网路中&#xff0c;每个用户节点都能发送和接收信号&#xff0c;并具有自动路由(转发消息)的功能。网路的参与者共用它们所拥有的一部分硬…

D2D D3D12 渲染视频帧思路及实现

写在之前 耗时2个月&#xff0c;写完公司的音视频处理系统。对于整个音视频处理有了基本的了解。个人感觉最坑的地方有三&#xff1a; 编解码音视频录制的同步视频预览渲染&#xff08;视频帧的渲染&#xff09; 由于在以后要支持同时多路1080P录制及预览&#xff0c;所以对…

车载通信与导航(五):D2D通信流程

取自https://blog.csdn.net/u012159948/article/details/20059927 D2D技术可以应用于移动蜂窝网络&#xff0c;以提高资源利用率和网络容量。每一个D2D通信链路占用的资源与一个蜂窝通信链路占用的相等。D2D通信将在宏蜂窝基站的控制下获得通信所需的频率资源和传输功率。它与蜂…

关于字体绘制的baseline (QT 和 D2D)

常规字体绘制API调用: QT:QPainter::drawText D2D: ID2D1RenderTarget::DrawText 以上字体绘制某些特殊效果无法达到, 我们可以用Path绘制字体: QPainterPath::addText(qreal x, qreal y, const QFont &f, const QString &text); D2D: 将字用指定字体属性绘制好,…

DirectX12(D3D12)基础教程(十三)——D2D、DWrite On D3D12与文字输出

文章目录 1、 前言2、D2D、DWrite简介3、添加D2D、DWrite基础支持文件4、D2D、DWrite基本编程步骤5、基于D3D11On12设备创建D2D渲染目标6、创建DWrite字体用D2D显示文字7、D2D、D3D11on12与D3D12同步 1、 前言 在经过了前面一系列章节的“狂轰滥炸”式的学习之后&#xff0c;如…

D2D

D2D: Device-to-Device即是设备到设备的通信。 1.认识D2D 基于蜂窝网络的D2D通信&#xff0c;或称为邻近服务&#xff08;Proximity Service,ProSe&#xff09;&#xff0c;是指用户数据可不会经过网络中转直接在终端之间传输。D2D通信与传统的通信架构有显著区别&#xff0c…

D2D与蜂窝系统间的干扰

上行频段:D2D发射端UE3对基站的干扰,蜂窝UE1对D2D接收端UE4的干扰;下行频段:基站对D2D接收端UE4的干扰,D2D发射端UE3对蜂窝UE2的干扰。 D2D复用上行资源:eNB会受到D2D的干扰,D2D中的UE都受eNB的控制,eNB为其分配资源,同时也控制最大发射功率,而且还能够将蜂窝的功控信息…

关于D2D

前段时间发现微软的DirectX是如此的强悍&#xff0c;并且对于开发windows程序是如此的重要&#xff0c;决定到msdn去看看&#xff0c;下面是对D2D的介绍&#xff0c;后面会陆续贴上&#xff0c;这是我的第一篇译文&#xff0c;有误之处&#xff0c;还请指教。 什么是D2D? D2D…

移动通信网络规划:D2D通信技术

D2D通信技术 一、什么是D2D通信技术 D2D即Device-to-Device&#xff0c;也称之为终端直通。D2D通信技术是指两个对等的用户节点之间直接进行通信的一种通信方式。 如图中所示。在由D2D通信用户组成的分散式网络中&#xff0c;每个用户节点都能发送和接收信号&#xff0c;并具有…

D2D网络架构介绍

D2D(Device-to-Device)通信,也称为邻近服务(Proximity Service,ProSe),是由3GPP组织提出的一种点到点的无线通信技术,它可以在蜂窝通信系统的控制下允许LTE终端之间利用小区无线资源直接进行通信,而不经过蜂窝网络中转。作为面向5G的关键候选技术,D2D技术能够提升通信…

车载通信与导航(七):D2D通信详解

D2D&#xff08;设备到设备&#xff09;&#xff0c;即临近终端设备之间直接进行通信的技术&#xff0c;在通信网络中&#xff0c;一旦D2D通信链路建立起来&#xff0c;传输语音或数据消息就无需基站的干预&#xff0c;这样就可以减轻通信系统中基站及核心网络的数据压力&#…

C中strchr()函数用法

strchr()函数包含于头文件&#xff1a;#include<stdio.h>中&#xff1b; 函数原型为&#xff1a;char * strchr(char * str, char/int c); 函数功能为&#xff1a;在字符串str中寻找字符C第一次出现的位置&#xff0c;并返回其位置&#xff08;地址指针&#xff09;&am…

php strchr 截断,PHP strchr() 函数

w3school 教程 PHP String 函数 查找 "world" 在 "Hello world!" 中的第一次超并返回此字符串的其余个别&#xff1a; echo strchr("Hello world!","world"); ?> 运行实例 strchr() 函数搜他串在另一字符串中的第一次常 该函数是 …

php strchr 截断,php 常用字符串函数 - strchr

...转义的字符串 三、检索字符串函数/strstr()函数和strchr()函数$str "abc123.com"; $tem strchr($str,""); $tep strstr($str, ""); $ten 以下列出开发中常用的字符串函数&#xff0c;以供自己需要的时候查阅 长度 strlen($string):得到字符…

strchr的用法

strchr&#xff08;char *s,char *c&#xff09; 查找字符c第一次出现在字符串s中的位置 返回值&#xff1a;如果找到指定的字符则返回该字符所在地址&#xff1b;否则返回NULL. 例1&#xff1a; #include <cstring> #include <cstdio> int main() {char sstri…

php strchr和strrchr,strrchr与Strchr

strrchr 取得某字符最后出现处起的字符串。 语法: string strrchr(string haystack, string needle); 返回值: 整数 函数种类: 资料处理 内容说明 本函数用来寻找字符串 haystack 中的字符 needle 最后出现位置&#xff0c;并将此位置起至字符串 haystack 结束之间的字符串返回…

php strchr 截断,php字符串处理函数详解 - strchr

...substr($add,0,strlen($add)-1); }if($word>){ $flag1; } }if(strchr($add,$keytop)){ $found1; }if(strchr($add,$keybottom)){ $found0; $end1; }if(((strchr($add, PHP处理字符串的能力非常强大,方法也是多种多样,但有的时候你需要选择一种最简单且理想的解决方法,文章…

ipv6环境搭建

2018年4月30日 iOS上架必须适配ipv6的网络&#xff0c;我也因为这个原因被拒了。为适配ipv6&#xff0c;就必须创建一个ipv6环境&#xff0c;本文介绍了使用Mac搭建ipv6环境的2种方法。 根据有、无网线&#xff0c;可分为2种场景来搭建。 有线网情况下创建ipv6环境 目标 通过…

搭建一个简单的SDN网络环境

第1小题&#xff1a;简单网络 说明&#xff1a;由于对于SDN架构的理解在学界和业界并没有统一&#xff0c;为了方便参赛队员选择&#xff0c;对于初学者&#xff0c;大赛推荐OpenFlow作为南向接口来实现SDN环境&#xff0c;以下给出分别针对采用OpenFlow和采用其他接口的具体要…