yunruifuzhu 发表于 2021-5-8 11:00

开源Windows清理工作集(清理内存)

本帖最后由 yunruifuzhu 于 2021-5-8 11:02 编辑

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



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



// 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_NAMETEXT("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.Luid))
                {
                        //OutputDebugView(_T("LookupPrivilegeValue Error"));
                        bRet = FALSE;
                        break;
                }
                Token_Privilege.PrivilegeCount = 1;
                Token_Privilege.Privileges.Attributes = SE_PRIVILEGE_ENABLED;
                //Token_Privilege.Privileges.Luid.LowPart=17;//SE_BACKUP_PRIVILEGE
                //Token_Privilege.Privileges.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.Luid);

      tkp.PrivilegeCount = 1; // one privilege to set   
      tkp.Privileges.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;
      for (int i = 0; i < num; i++)
      {
                a = new int;
      }
      for (int i = 0; i < num; i++)
      {
                delete a;
      }
}

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();
}



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

成品:

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

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

treesea 发表于 2021-5-8 11:17
Windows7失败

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

打不开建议 安装了 微软运行库 最新的看看

叶落飘香 发表于 2021-5-8 11:27

谢滚吃灰。
页: [1] 2 3
查看完整版本: 开源Windows清理工作集(清理内存)