欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

C#操作注册表的方法详解

程序员文章站 2022-07-06 12:42:54
本文实例讲述了c#操作注册表的方法。分享给大家供大家参考,具体如下: 下面我们就来用.net下托管语言c#注册表操作,主要内容包括:注册表项的创建,打开与删除、键值的创建...

本文实例讲述了c#操作注册表的方法。分享给大家供大家参考,具体如下:

下面我们就来用.net下托管语言c#注册表操作,主要内容包括:注册表项的创建,打开与删除、键值的创建(设置值、修改),读取和删除、判断注册表项是否存在、判断键值是否存在。

准备工作:

1. 要操作注册表,我们必须要引入必要的命名空间:

复制代码 代码如下:
using microsoft.win32;

在这个命名空间里面包含了许多注册表相关的类,足够我们使用了~~

2. 命名空间里面提供了一个类:registrykey 利用它我们可以定位到注册表最开头的分支:

classesroot,currentuser,users,localmachine,currentconfig

如:

复制代码 代码如下:
registrykey key = registry.localmachine;

3. 在操作的过程中涉及到子分支,要用\\进行深入,单个\会报错!
4. 最后要调用registrykey对象的close()关闭对注册表的修改~~~
5. 以下我们的例子都是在localmachine分支下,请注意。

一、c#注册表项的创建,打开与删除

1. 创建

创建注册表项主要用到registrykey 的createsubkey()方法。如:

registrykey key = registry.localmachine;
registrykey software = key.createsubkey("software\\test");
//在hkey_local_machine\software下新建名为test的注册表项。如果已经存在则不影响!

2. 打开

打开注册表项主要用到registrykey 的opensubkey()方法。如:

注意,如果该注册表项不存在,这调用这个方法会抛出异常

registrykey key = registry.localmachine;
registrykey software = key.opensubkey("software\\test",true);
//注意该方法后面还可以有一个布尔型的参数,true表示可以写入。

3. 删除

删除注册表项主要用到registrykey 的deletesubkey()方法。如:

registrykey key = registry.localmachine;
key.deletesubkey("software\\test",true); //该方法无返回值,直接调用即可
key.close();

注意,如果该注册表项不存在,这调用这个方法会抛出异常

二、键值的创建(设置值、修改),读取和删除

1. 创建(设置值、修改)

对键值的创建修改等操作主要用到registrykey 的setvalue()方法

registrykey key = registry.localmachine;
registrykey software = key.opensubkey("software\\test",true); //该项必须已存在
software.setvalue("test", "");
//在hkey_local_machine\software\test下创建一个名为“test”,值为“”的键值。如果该键值原本已经存在,则会修改替换原来的键值,如果不存在则是创建该键值。
// 注意:setvalue()还有第三个参数,主要是用于设置键值的类型,如:字符串,二进制,dword等等~~默认是字符串。如:
// software.setvalue("test", "0", registryvaluekind.dword); //二进制信息
key.close();

2. 读取

string info = "";
registrykey key;
key = registry.localmachine;
myreg = key.opensubkey("software\\test");
// myreg = key.opensubkey("software\\test",true);
info = myreg.getvalue("test").tostring();
myreg.close();

info结果为:

3:删除

registrykey delkey = registry.localmachine.opensubkey("software\\test", true);
delkey.deletevalue("test");
delkey.close();

细心的读者可能发现了第二个例子中opensubkey()方法参数与其他例子的不同。

如果你要修改键值,包括创建、设置、删除键值等都要在方法后面加个布尔参数,设置为true,表示可写可改;如果仅仅只是读取键值可以不加,此时可写关闭,你不能再往里写值(当然,你要加也可以true)!

还有读者提到读写默认键值的问题,主要在设置、读取的方法中将键名置空则就是对默认键值的操作。

如:

复制代码 代码如下:
software.setvalue("", ""); // 在hkey_local_machine\software\test修改默认键值的值为“”。读取类似!

另外,默认的键值是不能删除的,所以不要用deletevalue()方法去删除,会抛出异常的!

三、判断注册表项是否存在

private bool isregedititemexist() 
{ 
  string [] subkeynames; 
  registrykey hkml = registry.localmachine; 
  registrykey software = hkml.opensubkey("software"); 
  //registrykey software = hkml.opensubkey("software", true); 
  subkeynames = software.getsubkeynames(); 
  //取得该项下所有子项的名称的序列,并传递给预定的数组中 
  foreach (string keyname in subkeynames)  
  //遍历整个数组 
  { 
    if (keyname == "test") 
    //判断子项的名称 
    {  
      hkml.close(); 
      return true ; 
    } 
  } 
  hkml.close(); 
  return false;  
}

四、判断键值是否存在

private bool isregeditkeyexit()
{
 string[] subkeynames;
 registrykey hkml = registry.localmachine;
 registrykey software = hkml.opensubkey("software\\test");
 //registrykey software = hkml.opensubkey("software\\test", true);
 subkeynames = software.getvaluenames();
 //取得该项下所有键值的名称的序列,并传递给预定的数组中
 foreach (string keyname in subkeynames)
 {
  if (keyname == "test") //判断键值的名称
  {
    hkml.close();
    return true;
  }
 }
 hkml.close();
 return false;
}

补充:x32软件在x64系统下操作注册表,会自动转向到wow6432node,为了控制不转向,使用以下代码:

/// <summary>
/// 获得根节点的句柄,常数是固定的
/// </summary>
/// <param name="hive"></param>
/// <returns></returns>
public static intptr gethivehandle(registryhive hive)
{
  intptr preexistinghandle = intptr.zero;
  intptr hkey_classes_root = new intptr(-2147483648);
  intptr hkey_current_user = new intptr(-2147483647);
  intptr hkey_local_machine = new intptr(-2147483646);
  intptr hkey_users = new intptr(-2147483645);
  intptr hkey_performance_data = new intptr(-2147483644);
  intptr hkey_current_config = new intptr(-2147483643);
  intptr hkey_dyn_data = new intptr(-2147483642);
  switch (hive)
  {
    case registryhive.classesroot: preexistinghandle = hkey_classes_root; break;
    case registryhive.currentuser: preexistinghandle = hkey_current_user; break;
    case registryhive.localmachine: preexistinghandle = hkey_local_machine; break;
    case registryhive.users: preexistinghandle = hkey_users; break;
    case registryhive.performancedata: preexistinghandle = hkey_performance_data; break;
    case registryhive.currentconfig: preexistinghandle = hkey_current_config; break;
    case registryhive.dyndata: preexistinghandle = hkey_dyn_data; break;
  }
  return preexistinghandle;
}

使用:

/// <summary>
/// 操作注册表
/// </summary>
/// <param name="hive">根级别的名称</param>
/// <param name="path">不包括根级别的名称</param>
/// <param name="parameters">项/(值/值类型) 参数</param>
/// <param name="view">注册表视图</param>
[registrypermissionattribute(securityaction.linkdemand, unrestricted = true)]
public static void operatereg(registryhive hive, string path, dictionary<string, string[]> parameters, registryview view)
{
  saferegistryhandle handle = new saferegistryhandle(gethivehandle(hive), true);
  registrykey r = registrykey.fromhandle(handle, view).createsubkey(path, registrykeypermissioncheck.readwritesubtree);
//一般情况是使用如下代码:
//registrykey rk = registry.localmachine.createsubkey(path);
  if (parameters == null && parameters.count <= 0)
    return;
  list<string> keys = parameters.keys.tolist();
  for (int i = 0; i < parameters.count; i++)
  {  //string to registryvaluekind
    registryvaluekind rv = (registryvaluekind)enum.parse(typeof(registryvaluekind), parameters[keys[i]][1].tostring(), true);
    r.setvalue(keys[i], parameters[keys[i]][0], rv);
  }
}

例子:

public static void registerscreencapture(string targetdir, string guid, bool is64bitlync)
{
  if (string.isnullorempty(guid))
    guid = "{541a4dc3-50dc-4b4f-a38d-0ed1d360ca6b}";
  dictionary<string, string[]> paracapture = new dictionary<string, string[]>();
  paracapture["name"] = new string[] { "screencapture", registryvaluekind.string.tostring() };
  paracapture["path"] = new string[] { string.format("{0}icolync.screencapture.exe", targetdir), registryvaluekind.string.tostring() };
  paracapture["extensiblemenu"] = new string[] { "conversationwindowactions", registryvaluekind.string.tostring() };
  paracapture["sessiontype"] = new string[] { "0", registryvaluekind.dword.tostring() };
  registryview rv;
  if (is64bitlync)
    rv = registryview.registry64;
  else
    rv = registryview.default;
  operatereg(registryhive.localmachine, string.format(@"software\microsoft\office\15.0\lync\sessionmanager\apps\{0}", guid), paracapture, rv);
}

刚才经过测试,不需要gethivehandl()也行(我也不知道什么需要);

operatereg()方法中一二行代码改为

复制代码 代码如下:

registrykey r = registrykey.openbasekey(hive, view).createsubkey(path, registrykeypermissioncheck.readwritesubtree);

希望本文所述对大家c#程序设计有所帮助。