吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 7097|回复: 23
收起左侧

[C&C++ 转载] 开源Windows清理工作集(清理内存)

[复制链接]
yunruifuzhu 发表于 2021-5-8 11:00
本帖最后由 yunruifuzhu 于 2021-5-8 11:02 编辑

由于windows 2008 服务器上,sqlserver经常出现死锁或卡死,经百度 说的 是要用微软的 rammap工具清理系统工作集。
尝试了一下确实立马恢复了,并且内存占用也少了 。
Snipaste_2021-05-08_10-53-03.png


但需要人工去点击,后来经过网上各种搜索,将不同方式的清理工作集的源码合并到一切。效果基本与工具相同了。


[C++] 纯文本查看 复制代码
// ConsoleApplication1.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//

#include <iostream>
#include <windows.h>
#include <tlhelp32.h>
#include <psapi.h>
#pragma comment (lib,"psapi.lib")
#pragma comment (lib,"Advapi32.lib")
#define SE_INCREASE_QUOTA_NAME  TEXT("SeIncreaseQuotaPrivilege")

bool EnableSpecificPrivilege(LPCTSTR lpPrivilegeName)
{
        HANDLE hToken = NULL;
        TOKEN_PRIVILEGES Token_Privilege;
        BOOL bRet = TRUE;
        do
        {
                if (0 == OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &hToken))
                {
                        //OutputDebugView(_T("OpenProcessToken Error"));
                        bRet = FALSE;
                        break;
                }
                if (0 == LookupPrivilegeValue(NULL, lpPrivilegeName, &Token_Privilege.Privileges[0].Luid))
                {
                        //OutputDebugView(_T("LookupPrivilegeValue Error"));
                        bRet = FALSE;
                        break;
                }
                Token_Privilege.PrivilegeCount = 1;
                Token_Privilege.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
                //Token_Privilege.Privileges[0].Luid.LowPart=17;//SE_BACKUP_PRIVILEGE  
                //Token_Privilege.Privileges[0].Luid.HighPart=0;  
                if (0 == AdjustTokenPrivileges(hToken, FALSE, &Token_Privilege, sizeof(Token_Privilege), NULL, NULL))
                {
                        //OutputDebugView(_T("AdjustTokenPrivileges Error"));
                        bRet = FALSE;
                        break;
                }
        } while (false);
        if (NULL != hToken)
        {
                CloseHandle(hToken);
        }
        return bRet;
}

bool AdjustTokenPrivilegesForNT()
{
        HANDLE hToken;
        TOKEN_PRIVILEGES tkp;

        // Get a token for this process.

        OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &hToken);

        // Get the LUID for the EmptyWorkingSet privilege.

        LookupPrivilegeValue(NULL, SE_DEBUG_NAME, &tkp.Privileges[0].Luid);

        tkp.PrivilegeCount = 1; // one privilege to set    
        tkp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;

        // Get the EmptyWorkingSet privilege for this process.

        return AdjustTokenPrivileges(hToken, FALSE, &tkp, 0, (PTOKEN_PRIVILEGES)NULL, 0);
}

BOOL EmptyAllSet()
{
        HANDLE SnapShot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
        if (SnapShot == NULL)
        {
                return FALSE;
        }
        PROCESSENTRY32 ProcessInfo;//声明进程信息变量
        ProcessInfo.dwSize = sizeof(ProcessInfo);//设置ProcessInfo的大小
        //返回系统中第一个进程的信息
        BOOL Status = Process32First(SnapShot, &ProcessInfo);
        while (Status)
        {
                HANDLE hProcess = OpenProcess(PROCESS_ALL_ACCESS, TRUE, ProcessInfo.th32ProcessID);
                if (hProcess)
                {
                        //printf("正在清理进程ID:%d 的内存\n",ProcessInfo.th32ProcessID);
                        SetProcessWorkingSetSize(hProcess, -1, -1);
                        //内存整理
                        EmptyWorkingSet(hProcess);
                        CloseHandle(hProcess);
                }
                //获取下一个进程的信息
                Status = Process32Next(SnapShot, &ProcessInfo);
        }
        return TRUE;
}
void deepClean()
{
        const int num = 1000;
        int size = 500000;
        int *a[num];
        for (int i = 0; i < num; i++)
        {
                a[i] = new int[size];
        }
        for (int i = 0; i < num; i++)
        {
                delete a[i];
        }
}

int main()
{
        printf("AdjustTokenPrivilegesForNT - 提权:%d\n", AdjustTokenPrivilegesForNT());
        printf("清理工作集:%d\n", EmptyAllSet());
        printf("SE_INCREASE_QUOTA_NAME - 提权:%d\n", EnableSpecificPrivilege(SE_INCREASE_QUOTA_NAME));
        printf("清理系统文件缓存:%d\n", SetSystemFileCacheSize(-1, -1, 0));
        printf("手动2G内存生成释放:\n");
        deepClean();
        printf("清理成功,自动退出!");
        //getchar();
}



运行需要管理员身份运行,不然提权会失败的哦!

成品: ConsoleApplication1.rar (126.98 KB, 下载次数: 332)

源码经过本人些许需改,代码很乱但很实用。需要的给个评分谢谢

免费评分

参与人数 3吾爱币 +2 热心值 +2 收起 理由
13418946402 + 1 + 1 谢谢@Thanks!
sam喵喵 + 1 看到这种发实用工具,尤其是带C++源码的必须给赞!
承蒙厚爱丶 + 1 我很赞同!

查看全部评分

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

 楼主| yunruifuzhu 发表于 2021-6-2 12:15
alongzhenggang 发表于 2021-6-2 11:52
温⑦企业版  闪退

不影响,这里有三种清理方式,从图中看出,前两种已经成功了,最后一种是比较笨的方式闪退,其实已经清理了
EnterpriseSolu 发表于 2021-5-8 11:53
sqlserver经常出现死锁或卡死, 可能是事务卡死了,要改代码,
随意的操作内存或是杀死SQL进程会导致数据库损坏,后果是要修复数据,SQL 2000的时候,也是这么干的,后来加班修补数据,一个多星期的工作量,加班熬夜,再也不敢这么干了
sql 要用企业版或标准版,它的内存会多一些,express版最多只用1G内存,慎用
青-山胡椒 发表于 2021-5-8 11:06
zxxiaopi 发表于 2021-5-8 11:07
好东西,确实需要!
微雨至夏 发表于 2021-5-8 11:12

好东西,确实需要!。下载试试
treesea 发表于 2021-5-8 11:17
Windows7失败
geuoll 发表于 2021-5-8 11:20
这个是干嘛用的,没看明白
iceboy800 发表于 2021-5-8 11:21
试用下看,感谢楼主分享!
yy11911 发表于 2021-5-8 11:23
谢谢  楼主辛苦了  谢谢
 楼主| yunruifuzhu 发表于 2021-5-8 11:24

是打不开 还是提权清理失败?

打不开建议 安装了 微软运行库 最新的看看
叶落飘香 发表于 2021-5-8 11:27
谢滚吃灰。
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2025-1-12 22:54

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表