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

时间:2023-03-09 08:39:59
记录:C#监视某个文件的打开记录

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

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 给关闭,设成  就是关闭

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

        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 == "" || k1 == "")
{
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 承载进程  也许好使吧……