实现桌面动态壁纸(二)

目录

前言

一、关于 WorkerW 工作区窗口

二、关于窗口关系

2.1 窗口以及窗口隶属关系

2.2 桌面管理层窗口组分简析

2.3 厘清两个概念的区别

2.4 关于设置父窗口

三、编写代码以供在 Vista 上实现

3.1 方法二:子类化并自绘窗口背景

四、初步分析桌面管理层窗口创建的原理

4.1 桌面管理层窗口的创建流程

4.2 从管理层窗口回调看 0x052C 消息

总结


文章出处来源:[​https://blog.csdn.net/qq_59075481/article/details/133801491​]。

前言

这是实现 D2WT (Dynamic Desktop Wallpaper Tools) 系列的第二节,在本节中,我们进一步讨论 WorkerW 窗口的功能,介绍桌面窗口创建的流程,同时讨论为什么在 Vista 上无法嵌入窗口。

【提示】本文涉及的关于窗口的处理部分基于我曾经发的《桌面自定义 WorkerW 窗口》一文。里面的思路有类似的地方,但比那边讲的大概更加透彻。 

需要查看第一节的可以点击这里:实现桌面动态壁纸(一)

相关系列文章:

序号文章标题(链接)AID
1实现桌面动态壁纸(一)125361650
2实现桌面动态壁纸(二)[本文]

133801491

3实现桌面动态壁纸(三)[未来发布]---
4实现桌面动态壁纸——认识 WebView2 控件138637909

一、关于 WorkerW 工作区窗口

WorkerWWindows 操作系统中的一个窗口站 (Window Station) 和桌面 (Desktop) 的组合。它是用于用户界面的一个基础组件,用于管理和控制用户界面。WorkerW 从操作系统内核中获取资源,包括 CPU 资源和内存资源,并将其分配给用户进程,以便它们能够在屏幕上显示图形和交互元素。WorkerW 通过窗口管理器将窗口和界面元素显示在屏幕上,同时允许用户与它们进行交互。(以上这段来源于网络)

WorkerW/A 属于工作区窗口,它基本上通过调用 Shell API 函数中的 SHCreateWorkerWindowW/A 创建。其中 W 代表 WideChar (UNICODE) 版本的窗口,而 SHCreateWorkerWindowA 是该函数的 ASCII 版本。任何需要侦听窗口消息的应用程序都会调用此 API 来创建工作区窗口。 SHCreateWorkerWindowW 是为文档化的导出函数,通过分析 explorer.exe 发现该函数是从 api-ms-win-shlwapi-winrt-storage-l1-1-1.dll 中导入的,但是看到这个名称可能会很陌生。

explorer.exe 的导入表上,SHCreateWorkerWindowW 函数是通过解析名为 api-ms-win-shlwapi-winrt-storage-l1-1-1API 集而重定向到 shlwapi.dll,所以,最终是需要分析 shlwapi.dll 里面的函数。

API 集:微软推出的用高度命名的链接库名称分类 API 的最小唯一核心库,将 API 调用通过内置加载器转发到真实的 Dll 上,截止 Win11 已经更新到 V10 版本)

根据 ReactOS 的开发者文档可以知道 SHCreateWorkerWindow 的定义和内部实现。

HWND WINAPI SHCreateWorkerWindow(
WNDPROC        wndProc,
HWND                hWndParent,
DWORD             dwExStyle,
DWORD             dwStyle,
HMENU              hMenu,
LONG_PTR        wnd_extra
)

SHCreateWorkerWindowA/W 其实就是 CreateWindowExA/W 的封装:

HWND WINAPI SHCreateWorkerWindowA(
WNDPROC      wndProc,
HWND 	     hWndParent,
DWORD 	     dwExStyle,
DWORD 	     dwStyle,
HMENU 	     hMenu,
LONG_PTR     wnd_extra
)
{static const char szClass[] = "WorkerA";WNDCLASSA wc;HWND hWnd;TRACE("(%p, %p, 0x%08x, 0x%08x, %p, 0x%08lx)\n",wndProc, hWndParent, dwExStyle, dwStyle, hMenu, wnd_extra);/* Create Window class */wc.style         = 0;wc.lpfnWndProc   = DefWindowProcA;wc.cbClsExtra    = 0;wc.cbWndExtra    = sizeof(LONG_PTR);wc.hInstance     = shlwapi_hInstance;wc.hIcon         = NULL;wc.hCursor       = LoadCursorA(NULL, (LPSTR)IDC_ARROW);wc.hbrBackground = (HBRUSH)(COLOR_BTNFACE + 1);wc.lpszMenuName  = NULL;wc.lpszClassName = szClass;SHRegisterClassA(&wc);hWnd = CreateWindowExA(dwExStyle, szClass, 0, dwStyle, 0, 0, 0, 0,hWndParent, hMenu, shlwapi_hInstance, 0);if (hWnd){SetWindowLongPtrA(hWnd, 0, wnd_extra);if (wndProc) SetWindowLongPtrA(hWnd, GWLP_WNDPROC, (LONG_PTR)wndProc);}return hWnd;
}HWND WINAPI SHCreateWorkerWindowW(
WNDPROC      wndProc,
HWND 	     hWndParent,
DWORD 	     dwExStyle,
DWORD 	     dwStyle,
HMENU 	     hMenu,
LONG_PTR     wnd_extra
)
{static const WCHAR szClass[] = { 'W', 'o', 'r', 'k', 'e', 'r', 'W', 0 };WNDCLASSW wc;HWND hWnd;TRACE("(%p, %p, 0x%08x, 0x%08x, %p, 0x%08lx)\n",wndProc, hWndParent, dwExStyle, dwStyle, hMenu, wnd_extra);/* If our OS is natively ANSI, use the ANSI version */if (GetVersion() & 0x80000000)  /* not NT */{TRACE("fallback to ANSI, ver 0x%08x\n", GetVersion());return SHCreateWorkerWindowA(wndProc, hWndParent, dwExStyle, dwStyle, hMenu, wnd_extra);}/* Create Window class */wc.style         = 0;wc.lpfnWndProc   = DefWindowProcW;wc.cbClsExtra    = 0;wc.cbWndExtra    = sizeof(LONG_PTR);wc.hInstance     = shlwapi_hInstance;wc.hIcon         = NULL;wc.hCursor       = LoadCursorW(NULL, (LPWSTR)IDC_ARROW);wc.hbrBackground = (HBRUSH)(COLOR_BTNFACE + 1);wc.lpszMenuName  = NULL;wc.lpszClassName = szClass;SHRegisterClassW(&wc);hWnd = CreateWindowExW(dwExStyle, szClass, 0, dwStyle, 0, 0, 0, 0,hWndParent, hMenu, shlwapi_hInstance, 0);if (hWnd){SetWindowLongPtrW(hWnd, 0, wnd_extra);if (wndProc) SetWindowLongPtrW(hWnd, GWLP_WNDPROC, (LONG_PTR)wndProc);}return hWnd;
}

DWM 机制完善之前的操作系统上,切换桌面壁纸或者系统主题的时候,窗口的绘制会出现卡顿、频闪现象。在切换主题的时候,微软通过 LockWindowUpdate 函数,阻止其他窗口的绘制,并显示一个“请稍后”窗口,来避免用户看到卡顿的桌面管理层窗口。但是,这给用户的体验并不是特别好,因为需要“等待”。随后,在 DWM 组件的支持下,切换壁纸前,首先将 DefView 窗口分离出来,然后利用 WorkerW 窗口去绘制 DefView 的背景,在内存中首先生成双缓冲,将新壁纸和旧壁纸的图案之间合成交叉溶解的图像动画,从而实现窗口背景的平滑处理。

下图展示了在切换主题的交叉阶段,桌面管理层窗口的变化(新旧壁纸的交叉溶解效果):

我们意识到,SHCreateWorkerWindow 只能创建类名是 WorkerW 的窗口,关键部分并不在于这个函数,想要知道系统是如何实现透明层次的,还需要研究其窗口过程以及后续的处理,我想这需要对桌面窗口有一个深入一点的理解。

二、关于窗口关系

2.1 窗口以及窗口隶属关系

(TODO:之后补充)

2.2 桌面管理层窗口组分简析

我们知道,桌面管理层窗口在未产生 WorkerW 分层时,窗口的层次应该如下所示:

我么可以通过简单的手法理解这些窗口的作用:

(1) SysHeader32 窗口

SysHeader32 窗口是一个不可见窗口,这个窗口主要负责在 ListView 上绘制每个图标的文本。

验证方法:通过 SendMessageW(hSysHead, WM_CLOSE, 0, 0) 即可关闭该窗口,按 F5 刷新桌面,可以观察到图标的文本已经消失,但是图标依然可以正常点击:

并且右键菜单依然是有效的:

(2) SysListView32 窗口

SysListView32 窗口主要负责控制图标列表的显示和操作,关闭或者隐藏后,图标列表将不可见。

验证方法:隐藏窗口 ShowWindow(hListView, SW_HIDE) 可以发现图标立即消失。

但是,右键菜单依然可用,说明右键菜单不归它管理:

(3) SHELLDLL_DefView 窗口

这个窗口我们需要通过两步验证它的功能。

SHELLDLL_DefView 窗口控制图标列表窗口的背景绘制工作,这可以从 SysListView32 的属性页看出:

SHELLDLL_DefView 还控制右键菜单,使用 ShowWindow(hDefView, SW_HIDE) 后无法打开右键菜单。

验证是否支持背景绘制工作:

第一步:进一步隐藏 Program Manager 窗口,桌面管理层窗口的背景变成白色:

这说明了,Program 窗口的背景是系统设置的壁纸。

第二步:将 DefView 窗口变成弹出式窗口(独立化),并恢复显示。

会发现,无论 Program 窗口是否可见,图标窗口的背景都是黑色的:

于是我们可以判断出,SHELLDLL_DefView 可以通过获取父窗口(会判断是不是 Progman 窗口)的图像缓冲,来绘制子窗口的背景。

(4) Program Manager 窗口

Program Manager 窗口是桌面管理层的主窗口,Program Manager 窗口响应 WM_CLOSE 时(不响应 SC_CLOSE ),会调用 Shell32.dll 中的符号并显示一个询问是否需要关闭计算机的对话框:

Program 还负责显示桌面壁纸,隐藏或者关闭后背景将变为白色:

至此,我们从窗口的可视化角度简单分析了各个桌面管理层窗口的基本作用。

2.3 厘清两个概念的区别

在这个系列的一开始,我们就用“桌面管理层窗口”来称呼包含桌面图标在内的几个窗口的集合:

但是,我们在第一篇中,我们也提到过桌面窗口这个名字,桌面窗口和桌面管理层窗口有什么区别呢?

桌面窗口是其他窗口的祖先,在系统启动时创建,类名为 “#32769” 。这个窗口由 csrss.exe 进程创建,所有父窗口显示为 NULL 的窗口其实是以该窗口作为父窗口。所有窗口都在这个窗口内。所以它是 Z 序最高的窗口。

而桌面管理层窗口,则是 Z 序最低的窗口。桌面管理层窗口是以名为 Program Manager 窗口为主窗口,管理左面文件夹图标列表的显示、操作、桌面壁纸等功能的一系列窗口。

而这本质上不是一类窗口。此外,通过 GetDesktopWindow 函数获取的窗口句柄是桌面窗口句柄,而不是桌面管理层的窗口句柄。(关于他们的详细内容,在接下来的文章中我们会一一介绍)

2.4 关于设置父窗口

Windows Vista 上,SHELL_DefView 不支持背景透明化,我们想到可以利用扩展属性 WS_EX_LAYERED 实现背景透明,但是 MSDN 上明确说明该扩展属性从 Windows 8 开始,才对子窗口有效果。也就是说,在 Vista 上,对子窗口 SHELL_DefView 设置分层属性是无效的。

这时候,我们就需要将 SHELL_DefView 独立出来,将其变成弹出式窗口,就可以设置该属性了。

这里我么可以使用 SetParent 并指定父窗口为 NULL,随后去除窗口的 WS_CHILD 属性,添加 WS_POPUP | WS_EX_TOOLWINDOW 等属性,来实现将窗口独立化。

HWND SetParent(
_In_            HWND hWndChild,
_In_opt_     HWND hWndNewParent
);

[in] hWndChild

类型:HWND

子窗口的句柄。

[in, optional] hWndNewParent

类型:HWND

新父窗口的句柄。 如果此参数为 NULL,桌面窗口将成为新的父窗口。 如果此参数 HWND_MESSAGE,则子窗口将成为 仅消息窗口。

部分资料对这里的参数为 NULL 时,SetParent 的行为认知可能有误解,这里不是指桌面管理层窗口,他不是 Progman 窗口,而是由 csrss.exe 进程创建的类名为 “#32769” 窗口,他是一切桌面顶级窗口的父窗口(不是所有者窗口),称为桌面窗口,然而顶级窗口的父窗口常常被标记为 NULL

#32769” 窗口是一切桌面窗口的祖先窗口,是系统启动的时候创建的第一个窗口。Spy++ 下可以看到第一个窗口就是它:

查看窗口对应的进程信息:

显然,窗口由 CSRSS 创建。

接下来,我们用一个很简单的例子测试一下就可以理解正在发生的事情:

#include <iostream>
#include <Windows.h>int main()
{HWND h32769Wnd = NULL;HWND hDesktopwnd = NULL;HWND hNewParent = NULL;HWND hNotepad = NULL;HWND hOwner = NULL;SetLastError(0);h32769Wnd = FindWindowW(L"#32769", NULL);printf("FindDesktopWnd:[ 0x%I64X ], find #32769. err_code:[%d]\n",(unsigned long long)h32769Wnd, GetLastError());hNotepad = FindWindowA("Notepad", NULL);if (hNotepad){hNewParent = GetAncestor(hNotepad, GA_PARENT);GetWindow(hNotepad,GW_OWNER);printf("Notepad:[ 0x%I64X ]; GetAncestorParent:[ 0x%I64X ]; GetOwner:[ 0x%I64X ].\n", (unsigned long long)hNotepad, (unsigned long long)hNewParent,(unsigned long long)hOwner);hDesktopwnd = GetDesktopWindow();printf("Desktopwnd:[ 0x%I64X ], use GetDesktopWindow.\n",(unsigned long long)hDesktopwnd);if (hDesktopwnd){printf("SetParent use hDesktopwnd.\n");hNewParent = SetParent(hNotepad, hDesktopwnd);printf("LastParent:[ 0x%I64X ], retn by SetParent.\n",(unsigned long long)hNewParent);}hNewParent = GetAncestor(hNotepad, GA_PARENT);printf("Notepad:[ 0x%I64X ]; GetAncestorParent:[ 0x%I64X ]; GetOwner:[ 0x%I64X ].\n",(unsigned long long)hNotepad,(unsigned long long)hNewParent,(unsigned long long)hOwner);// --------------------------------------printf("\n\nSetParent use (null) ptr.\n");hNewParent = SetParent(hNotepad, NULL);printf("LastParent:[ 0x%I64X ], retn by SetParent.\n",(unsigned long long)hNewParent);hNewParent = GetAncestor(hNotepad, GA_PARENT);printf("Notepad:[ 0x%I64X ]; GetAncestorParent:[ 0x%I64X ]; GetOwner:[ 0x%I64X ].\n",(unsigned long long)hNotepad,(unsigned long long)hNewParent,(unsigned long long)hOwner);}system("pause");return 0;
}

我们首先尝试使用 FindWindow 查找类名,但是以失败告终,我们获得了无效句柄,这可能和FindWindow 的机制有关(没搞清楚原因,只知道他是 NtUserFindWindowEx 的封装。据我推断,它只从第一个顶级窗口开始检索,而且没有找到 GetLastError 并不能取到非零值)。

随后我们调用 SetParent 尝试设置 Notepad 的父窗口,这里我们进行了横向对比,第一次,我们使用 GetDesktopWindow 函数获取桌面窗口句柄,并把它作为第二参数传入 SetParent,通过分析父窗口和返回值,我们得到和 Spy++ 相同的结论(句柄指向 #32769 窗口);

第二次,我们按照 MSDN 上的说明,把第二个参数设置为 NULL,并再次获取信息,发现效果等同于传入 #32769 的有效句柄,这说明 SetParent 确实会在内部将 NULL 参数解释为桌面窗口( #32769 )的句柄。

下图展示了对 Notepad 窗口进行设置父窗口的操作前后,其父窗口的变化:

(关于 SetParent 的注意事项,在我之前的一篇博客中有详细分析,就不展开讨论了)

SetParentNULL 传参其实有两个作用:

(1)设置窗口成为桌面顶级窗口;

(2)将窗口提升 Z 序至前端(替代 SetForegroundWindow ),甚至解决了 SetForegroundWindow 有时候失败的问题。

关于第二个相当于副产品,解决 SetForegroundWindow 失败网上给的代码一般是这样子的:

if(hWnd)
{HWND hForeWnd = GetForegroundWindow();DWORD dwForeID = GetWindowThreadProcessId(hForeWnd,NULL);DWORD dwCurID = GetCurrentThreadId();AttachThreadInput(dwCurID,dwForeID,TRUE);ShowWindow(hWnd,SW_SHOWNORMAL);SetWindowPos(hWnd,HWND_TOPMOST,0,0,0,0,SWP_NOSIZE|SWP_NOMOVE);SetWindowPos(hWnd,HWND_NOTOPMOST,0,0,0,0, SWP_NOSIZE|SWP_NOMOVE);SetForegroundWindow(hWnd);AttachThreadInput(dwCurID,dwForeID,FALSE);// hWnd 就是需要置前的窗口句柄
}

而我们只需要判断这个窗口是不是 POPUP 窗口,并 SetParent 传参 NULL 即可。

三、编写代码以供在 Vista 上实现

Vista 上,DWM 被首次引入操作系统,但是它的框架结构和现在的有很大的不同,比如它不能够响应 0x052C (WM_USER + 300)的消息,而创建 WorkerW 窗口。这就是为什么在第一篇章中,我们直言在 Vista 上即使有开启 DWM 也不能够通过窗口嵌入的方式实现动态壁纸。

那么,如果我们固执的想要在早期的系统环境下实现动态壁纸,我们该如何做呢?

我在之前研究过自己实现一个 WorkerW,那篇博客限于一些原因,一些实现细节没能公布。这里我们可以说,即使不使用 WorkerW 依然可以实现动态壁纸。

我们想到将壁纸主窗口设置为 Progman 的子窗口,但是 SetParent 函数有个坏毛病,它会自动“擦屁股”,自动调用 CZOrderManagerService 内部函数将我们的窗口 Z 序放在 SHELLDLL_DefView 的前面,这是一个非常糟糕的。因为我们的窗口将完全遮盖 SHELLDLL_DefView 窗口,这使得我们无法看到图标列表窗口,我们的窗口始终位于上方。怎么办呢?

别急,这里有几种方法解决问题:

3.1 方法二:子类化并自绘窗口背景

(TODO:后期补充)

四、初步分析桌面管理层窗口创建的原理

由于对桌面管理层窗口的逆向分析没有找到实质性的材料,而作者本人又是初学一些反汇编知识,如有分析错误的地方,还望指拨。

4.1 桌面管理层窗口的创建流程

首先,我们需要回顾一下桌面管理层窗口的组成:

桌面浏览器窗口( DesktopBrowser )主要包括 Progman 父窗口,和 DefView 窗口,DefView 窗口的子窗口 SysListView32 用于绘制桌面图标等相关组件。而 Progman 的背景则绘制为桌面壁纸。

打开 IDA Pro 并反汇编 explorer.exe 可以定位到入口函数 wWinMain,可以看到 wWinMain 调用了 CreateDesktopAndTray 函数,这个函数是对 SHCreateDesktop 的封装,用于创建桌面和 CTray 的相关成员。

F5 的信息可以看出函数调用了延迟加载Shell32.dll 中的 SHCreateDesktop 函数。

跟进 Shell32.dll 查看该函数的内部实现:

有三个函数调用是关键性的:

(1)CDesktopBrowser::CDesktopBrowser 初始化 DesktopBrowserCDesktopBrowser 内部类实现了很多函数,包括图标窗口、任务栏控件、虚拟多桌面等等;

(2)RegisterDesktopClass 是对 RegisterClassW 的封装;

(3)SHFusionCreateWindowEx 是对 CreateWindowExW 的封装。

首先看 SHFusionCreateWindowEx 函数,前面谈到初始化 DesktopBrowser 的过程似乎在主窗口创建之前,然而分析上下文却能发现这两个实际上是并行操作。在 SHFusionCreateWindowEx 内首先激活并发上下文,然后尝试创建主窗口,同时初始化 DesktopBrowser 最后结束并发上下文,并返回窗口句柄。

然后,我们看一下 RegisterDesktopClass 函数,这个就比较简单了:

最重要的是 CDesktopBrowser 这个类,里面包含了有关桌面管理层窗口的很多未导出的内部函数。

RegisterDesktopClass 函数中调用的 CDesktopBrowser::s_DesktopWndProc 回调实现对 SysListView32 窗口的创建和处理。

调用树如下图所示:

SysListView32 窗口的创建和处理在 CreateDesktopView 中完成,流程比较复杂,暂不分析。

然后,继续跟踪,找到了 CDefView 类,一个关键的成员函数为 CDefView::CreateViewWindow

他是对 CDefView::CreateViewWindow2 的封装,CDefView::CreateViewWindow2 进行了一些对参数的初始化处理,随后把工作交给了 CDefView::CreateViewWindow3,在 CDefView::CreateViewWindow3 里面最终实现了创建 SHELLDLL_DefView 窗口。

4.2 从管理层窗口回调看 0x052C 消息

【这部分将在之后完善】


总结

自此,我们的桌面管理层窗口的创建已经基本完成,以上分析只是简单梳理一下流程,其中大量调用通过 COM 类接口实现,这里暂不展开分析。


本文属于原创文章,转载请注明出处:

https://blog.csdn.net/qq_59075481/article/details/133801491

文章更新于:2023.10.20,2024.07.04。

文章发布于:2024.07.04。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/370139.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Akamai+Noname强强联合 | API安全再加强

最近&#xff0c;Akamai正式完成了对Noname Security的收购。本文我们将向大家介绍&#xff0c;经过本次收购后&#xff0c;Akamai在保护API安全性方面的后续计划和未来愿景。 Noname Security是市场上领先的API安全供应商之一&#xff0c;此次收购将让Akamai能更好地满足日益增…

PDF压缩工具选哪个?6款免费PDF压缩工具分享

PDF文件已经成为一种常见的文档格式。然而&#xff0c;PDF文件的体积有时可能非常庞大&#xff0c;尤其是在包含大量图像或复杂格式的情况下。选择一个高效的PDF压缩工具就显得尤为重要。小编今天给大家整理了2024年6款市面上反响不错的PDF压缩文件工具。轻松帮助你找到最适合自…

Nginx实战:nginx性能压测(ab)

在nginx的生产实践中,不管是服务上线,还是性能优化,都会遇到需要对nginx的性能压测,本文介绍一个简单的压测工具:ab命令 ab(Apache Bench)是一个常用的HTTP压力测试工具,可以用来测试Nginx的性能和压力。ab命令可以指定并发请求数、请求数、请求类型等参数,并输出测试…

JavaScript-websocket的基本使用

JavaScript-websocket的基本使用 文章说明JavaScript端后台--服务端连接演示 文章说明 本文主要介绍JavaScript中websocket的基本使用&#xff0c;后台采用Java编写WebSocket服务端 JavaScript端 websocket工具类 class Socket {constructor(url, onopen, onmessage, onerror, …

前端实现坐标系转换

一、地理坐标系和投影坐标系 地理坐标系和投影坐标系是地理信息系统&#xff08;GIS&#xff09;中常见的两种坐标系统&#xff0c;它们用于描述和定位地球表面上的点和区域&#xff0c;但在实现方式和应用场景上有所不同。 1. 地理坐标系&#xff08;Geographic Coordinate …

【CUDA】 扫描 Scan

Scan Scan操作是许多应用程序中常见的操作。扫描操作采用一个二元运算符⊕和一个输入数组并计算输出数组如下&#xff1a; [x0,(x0⊕x1),…,( x0⊕x1⊕…..⊕xn-1)] 分层扫描和多种Scan算法介绍 Kogge-Stones Algorithm Kogge-Stones Algorithm最初是为设计快速加法电路而发…

JavaEE——计算机工作原理

冯诺依曼体系&#xff08;VonNeumannArchitecture&#xff09; 现代计算机&#xff0c;大多遵守冯诺依曼体系结构 CPU中央处理器&#xff1a;进行算术运算与逻辑判断 存储器&#xff1a;分为外存和内存&#xff0c;用于存储数据&#xff08;使用二进制存储&#xff09; 输入…

第一天(点亮led灯+led灯闪烁)——Arduino uno R3 学习之旅

​ 常识: 一般智能手机的额定工作电流大约为200mA Arduino Uno板上I/0(输入/输出)引脚最大输出电流为40 mA Uno板控制器总的输出电流为200 mA 点亮LED灯 发光二极管介绍 发光二极管(Light Emitting Diode&#xff0c;简称LED)是一种能够将电能转化为光能的固态的半导体器件…

实现模型贴图的移动缩放旋转

技术&#xff1a;threejscanvasfabric 效果图&#xff1a; 原理&#xff1a;threejs中没有局部贴图的效果&#xff0c;只能通过map 的方式贴到模型上&#xff0c;所以说换一种方式来实现&#xff0c;通过canvasfabric来实现图片的移动缩放旋转&#xff0c;然后将整个画布以map…

【STM32】在标准库中使用DMA

1.MDA简介 DMA全称Direct Memory Access,直接存储区访问。 DMA传输将数据从一个地址空间复制到另一个地址空间。当CPU初始化这个传输动作&#xff0c;传输动作本身是由DMA控制器来实现和完成的。DMA传输方式无需CPU直接控制传输&#xff0c;也没有中断处理方式那样保留现场和…

【踩坑】探究PyTorch中创建稀疏矩阵的内存占用过大的问题

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhagn.cn] 如果本文帮助到了你&#xff0c;欢迎[点赞、收藏、关注]哦~ 目录 问题复现 原因分析 解决方案 碎碎念 问题复现 创建一个COO格式的稀疏矩阵&#xff0c;根据计算公式&#xff0c;他应该只占用约5120MB的内存&…

go zero入门

一、goctl安装 goctl 是 go-zero 的内置脚手架&#xff0c;可以一键生成代码、文档、部署 k8s yaml、dockerfile 等。 # Go 1.16 及以后版本 go install github.com/zeromicro/go-zero/tools/goctllatest检查是否安装成功 $ goctl -v goctl version 1.6.6 darwin/amd64vscod…

通过SDK使用百度智能云的图像生成模型SDXL

登录进入百度智能云控制台&#xff0c;在模型广场按照图像生成类别进行筛选&#xff0c;可以找到Stable-Diffusion-XL模型。点击Stable-Diffusion-XL模型的API文档后在弹出的新页面下拉可以找到SDK调用的说明。 import qianfandef sdxl(file: str, prompt: str, steps: int 2…

C语言_练习题

求最小公倍数 思路&#xff1a;假设两个数&#xff0c;5和7&#xff0c;那么最小至少也要7吧&#xff0c;所以先假定最小公倍数是两个数之间较大的&#xff0c;然后看7能不能同时整除5和7&#xff0c;不能就加1继续除 int GetLCM(int _num1, int _num2) {int max _num1>_n…

堆叠的作用

一、为什么要堆叠 传统的园区网络采用设备和链路冗余来保证高可靠性&#xff0c;但其链路利用率低、网络维护成本高&#xff0c;堆叠技术将多台交换机虚拟成一台交换机&#xff0c;达到简化网络部署和降低网络维护工作量的目的。 二、堆叠优势 1、提高可靠性 堆叠系统多台成…

25款404网页源码(下)

25款404网页源码&#xff08;下&#xff09; 13部分源码 14部分源码 15部分源码 16部分源码 17部分源码 18部分源码 19部分源码 20部分源码 21部分源码 22部分源码 23部分源码 24部分源码 25部分源码 领取完整源码下期更新 13 部分源码 .rail {position: absolute;width: 100%…

Node.js-path 模块

path 模块 path 模块提供了 操作路径 的功能&#xff0c;如下是几个较为常用的几个 API&#xff1a; 代码实例&#xff1a; const path require(path);//获取路径分隔符 console.log(path.sep);//拼接绝对路径 console.log(path.resolve(__dirname, test));//解析路径 let pa…

Docker搭建MySQL双主复制详细教程

在此之前需要提前安装好Docker和 Docker Compose 。 一、创建目录 首先创建一个本地数据挂载目录。 mkdir -p master1-data master2-data二、编写docker-compose.yml version: 3.7services:mysql-master1:image: mysql:5.7.36container_name: mysql-master1environment:MYSQL_…

mac|idea导入通义灵码插件

官方教程&#xff1a;通义灵码下载安装指南_智能编码助手_AI编程_云效(Apsara Devops)-阿里云帮助中心 下载插件&#xff1a; ⇩ TONGYI Lingma - JetBrains 结果如下&#xff1a; 选择apply、ok&#xff0c;会出现弹窗&#xff0c;点击登录 可以实现&#xff1a;生成单元测…

FRP反向隧道代理打CFS三层

目录 攻击机 查看服务端frps.ini配置文件 开启服务端frps 蚁剑打目标机 上传客户端frp到目标机 ​frpc.ini文件配置成 客户端打开代理frpc vps显示成功客户端frpc打开 访问成功192.168.22.22的第二层内网主机 省去前面漏洞利用的rce过程&#xff0c;直接蚁剑开搞隧道…