记录:C#监视某个文件的打开记录

来源:互联网 发布:购物卡网络直销平台 编辑:程序博客网 时间:2024/06/12 19:16

首先,先说下为什么要搞这个:

1.首先,我的电脑里有5万左右的目录或文件,用于存放歌曲,数量众多。
2.我不一定会用哪种软件听歌(不过也就是几种而已)。
3.我想在听歌的时候,检测哪首首歌被打开,能获取到我正在放那首歌,用来记录播放顺序(播放器是随机播放的)。
4.【实例】我用一款“Ihear"播放器,开始播放歌曲,列表内有五万首歌,真实存放地址目录级数不确定,不知道这首歌具体在哪个目录,当然总的目录是确定的。我运行了一款”GetSong"软件,开始监视Ihear播放器,一旦播放器开始获取本地的歌曲文件开始播放,GetSong软件就会获取到是哪个文件被Ihear播放器读取(包括音乐文件与歌词文件),当然文件的绝对路径是肯定要获取到的。


以上就是我第一次和 陈希章 发的一个消息,向他询问解决方案。

当然一开始,我是自己尝试过的,FileSystemWatcher LastAccess 大家可以尝试搜索这两个关键字,这就是完成我上面描述的功能的解决方法。

当然,为了给别人使用,给程序加个 操作注册表 的功能,还是挺好用的。


当然,实现起来,并不是您想想的那么简单,如果真的直接就完成了,我也不会搞这个记录,更不会去向陈大师发邮件……

1. 首先我们要使用这个东西,肯定要写出类似的方法:(见 MSDN的附带示例)

复制代码
    public static void Main()    {        Run();    }    [PermissionSet(SecurityAction.Demand, Name = "FullTrust")]    public static void Run()    {        // Create a new FileSystemWatcher and set its properties.        FileSystemWatcher watcher = new FileSystemWatcher();        watcher.Path = @"D:\Temp";        /* Watch for changes in LastAccess and LastWrite times, and the renaming of files or directories. */        watcher.NotifyFilter = NotifyFilters.LastAccess ;        // Only watch text files.        watcher.Filter = "*.txt";        // Add event handlers.        watcher.Changed += OnChanged;                // Begin watching.        watcher.EnableRaisingEvents = true;        // Wait for the user to quit the program.        Console.WriteLine("Press \'q\' to quit the sample.");        while (Console.Read() != 'q') ;    }    // Define the event handlers.    private static void OnChanged(object source, FileSystemEventArgs e)    {        // Specify what is done when a file is changed.        Console.WriteLine("File: " + e.FullPath + " " + e.ChangeType);    }
复制代码

至于具体的解释,就不说了。我是直接搬过来的代码,有删减,不保证直接就能编译通过,请自行查错。

正题来了——如果你和我一样,用的是Win8系统,这个软件是肯定不能正常使用的,因为:

在Vista后的计算机预设是关闭更新 LastAccess 的,所以我们在怎么开启档案,在档案的属性页中他的存取日期都不会变动。

那知道了问题,就可以解决了,修改注册表(至于如何手动修改,还是百度下吧,我不细说了):

将 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\FileSystem 下的 NtfsDisableLastAccessUpdate 给关闭,设成 0 就是关闭

这样,就解决了我的问题,可是重点是,这个软件,并不是我自己用的啊……那我们就给我们的软件,加个自动修改注册表?

复制代码
        private static void RunRegistry()        {            RegistryKey regKey = Registry.LocalMachine;            RegistryKey setKey = regKey.OpenSubKey(@"SYSTEM\CurrentControlSet\Control\FileSystem", true);            object val0 = setKey.GetValue("NtfsDisableLastAccessUpdate", null);                        Console.WriteLine("获取到的注册表的值是:\t" + val0);            Console.WriteLine("输入一个要修改的值吧,只能是 0 或者 1 哦。");            string k1 = Console.ReadLine();            if (k1 == "0" || k1 == "1")            {                setKey.SetValue("NtfsDisableLastAccessUpdate", k1, RegistryValueKind.String);            }            while (Console.Read() != 'q') ;        }
复制代码

这样我们就能改注册表了吧?不过此时想想,我们这个软件竟然能改注册表?简直不可思议啊!权限够不够啊?安全性令人堪忧……

答案显而易见……普通的权限,注册表肯定是改不了的……

我们搜索一下:c# 获得管理员权限 ,然后找个法子来解决我们的问题:(manifest 应用程序清单文件)

复制代码
<?xml version="1.0" encoding="utf-8"?><asmv1:assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1" xmlns:asmv1="urn:schemas-microsoft-com:asm.v1" xmlns:asmv2="urn:schemas-microsoft-com:asm.v2" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">  <assemblyIdentity version="1.0.0.0" name="MyApplication.app" />  <trustInfo xmlns="urn:schemas-microsoft-com:asm.v2">    <security>      <requestedPrivileges xmlns="urn:schemas-microsoft-com:asm.v3">               <requestedExecutionLevel level="requireAdministrator" uiAccess="false" />      </requestedPrivileges>      <applicationRequestMinimum>        <defaultAssemblyRequest permissionSetReference="Custom" />        <PermissionSet class="System.Security.PermissionSet" version="1" ID="Custom" SameSite="site" Unrestricted="true" />      </applicationRequestMinimum>    </security>  </trustInfo>  <compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1">    <application>         </application>  </compatibility> </asmv1:assembly>
复制代码

请注意浅绿色高亮区域的值,我们选用的是 requireAdministrator

这样,我们的项目在运行时,就会向“UAC”请求管理员权限了。

哦,还要注意一个细节问题,直接调试运行我们的项目,项目会继承VS的权限的,他好像不会申请UAC,至于怎么获取……
也许,我只是说也许,去掉项目属性里的那个:调试/启用 Visual Studio 承载进程  也许好使吧……