软件注册与加密

来源:互联网 发布:淘宝助理宝贝描述模板 编辑:程序博客网 时间:2024/05/17 06:11
为了使开发的软件能被更广泛地使用,开发者希望更多的用户能试用软件,而另一方面,又不想让用户长时间免费使用未经授权的软件,这就需要设计软件注册程序。下面通过几个典型实例介绍保护软件安全的方法。
实例468 利用INI文件对软件进行注册
文本框:图16.6  软件注册实例说明
本实例实现使用INI文件对软件的用户信息进行注册的功能。运行程序,输入登录名称、登录口令和注册码,单击【注册】按钮进行注册,如果注册成功,则给出提示;如果信息已注册,系统给出提示信息。实例运行结果如图16.6所示。
技术要点
实现本实例功能主要用到API函数WritePrivateProfileString和GetPrivateProfileString函数。下面分别进行介绍。
(1)WritePrivateProfileString函数
此函数实现对INI文件的写操作。
函数声明如下。
[ DllImport ( "kernel32" ) ]
   private static extern long WritePrivateProfileString ( string section ,string key , string val , string filePath ) ;
参数说明如下。
l     section:INI文件中的段落。
l     key:INI文件中的关键字。
l     val:INI文件中关键字的数值。
l     filePath:INI文件完整的路径和名称。
(2)GetPrivateProfileString函数
此函数实现对INI文件的读操作。
函数声明如下。
[DllImport("kernel32")]
        private static extern int GetPrivateProfileString(string section, string key, string def, StringBuilder retVal, int size, string filePath);
参数说明如表16.5所示。
表16.5                                                                参数说明
参 数 值
说  明
section
INI文件中的段落名称
key
INI文件中的关键字
def
无法读取时候的缺省数值
retVal
读取数值
size
数值的大小
filePath
INI文件的完整路径和名称
 注意:使用API函数必须引用System.Runtime.InteropServices命名空间。
实现过程
(1)新建一个Windows应用程序,将其命名为Ex16_07,默认窗体为Form1。
(2)在Form1窗体中,主要添加3个TextBox控件,用于输入注册信息;添加两个Button控件,用来执行注册和退出操作。
(3)主要程序代码。
注册用户信息的实现代码如下:
  private void Form1_Load(object sender, EventArgs e)
        {
            FileStream c = new FileStream("C://desck.ini",FileMode.OpenOrCreate,FileAccess.Write);
        }
        [ DllImport ( "kernel32" ) ]
        private static extern long WritePrivateProfileString ( string section ,string key , string val , string filePath ) ;
        [DllImport("kernel32")]
        private static extern int GetPrivateProfileString(string section, string key, string def, StringBuilder retVal, int size, string filePath);
        private void button1_Click(object sender, EventArgs e)
        {
            StringBuilder temp = new StringBuilder(200);
            string FileName = "C://desck.ini";//NI文件的完整的路径和名称。
            foreach (object ct in Controls)
            {
                if (ct.GetType().ToString() == "System.Windows.Forms.TextBox")
                {
                    TextBox tx = (TextBox)ct;
                    if (tx.Text == "")
                    {
                        MessageBox.Show(tx.Tag.ToString()+"不能为空");
                    }
                }
            }
            string section = textBox3.Text;//INI文件中的段落
            string key = textBox1.Text;//INI文件中的关键字
            string keyValue = textBox2.Text;//INI文件中的关键字
            int i = GetPrivateProfileString(section, key, "无法读取对应数值!", temp, 200, FileName);//判断是否注册过
            if (temp.ToString() == "无法读取对应数值!")
            {
                WritePrivateProfileString(section, key, keyValue, FileName);
                MessageBox.Show("注册成功写入INI文件!", "信息");
            }
            else
            {
                MessageBox.Show("此信息已注册过了");
            }
        }
举一反三
根据本实例,读者可以实现以下功能。
*  对INI文件加密保存注册信息。
*  对组合INI文件加密保存注册信息。
实例469 利用注册表设计软件注册程序
实例说明
文本框:图16.7  动态变化的软件注册程序大多数应用软件会将用户输入的注册信息写进注册表中,程序运行过程中,可以将这些信息从注册表中读出。本实例主要实现在程序中对注册表进行操作的功能,运行程序,单击【注册】按钮,会将用户输入的信息写入注册表中。实例运行结果如图16.7所示。
技术要点
实现本实例功能主要用到了Microsoft.Win32命名空间下的Registry类的CurrentUser属性、RegistryKey类的OpenSubKey( )方法、GetSubKeyNames( )方法、SetValue( )方法和CreateSubKey( )方法。下面分别进行介绍。
(1)Microsoft.Win32命名空间
Microsoft.Win32命名空间提供两种类型的类:处理由操作系统引发的事件的类和操作系统注册表的类。
(2)RegistryKey类
此类表示Windows注册表中的项级节点,此类是注册表封装。
语法格式为:
public sealed class RegistryKey : MarshalByRefObject, IDisposable
 注意:要获取RegistryKey实例,需要使用Registry类的静态成员之一。
(3)Registry类
此类提供表示Windows注册表中的根项的RegistryKey对象,并提供访问项/值对的static( )方法。
语法格式为:
public static class Registry
(4)CurrentUser属性
此属性包含有关当前用户首选项的信息,该字段读取Windows 注册表中的HKEY_ CURRENT_USER注册表项。
语法格式为:
public static readonly RegistryKey CurrentUser
 注意:存储在此项中的信息包括环境变量的设置和有关程序组、颜色、打印机、网络连接和应用程序首选项的数据,此项使建立当前用户的设置更容易。在此项中,软件供应商存储要在其应用程序中使用的当前用户特定的首选项。
(5)OpenSubKey( )方法
此方法检索指定的子项。
语法格式为:
public RegistryKey OpenSubKey (string name,bool writable)
参数说明如下。
l     name:要打开的子项的名称或路径。
l     writable:如果需要项的写访问权限,则设置为True。
l     返回值:请求的子项;如果操作失败,则为空引用。
(6)CreateSubKey( )方法
此方法创建一个新子项或打开一个现有子项以进行写访问。字符串subkey不区分大小写。
语法格式为:
public RegistryKey CreateSubKey (string subkey)
参数说明如下。
l     subkey:要创建或打开的子项的名称或路径。
l     返回值:RegistryKey对象,表示新建的子项或空引用。如果为subkey指定了零长度字符串,则返回当前的RegistryKey对象。
(7)GetSubKeyNames( )方法
此方法检索包含所有子项名称的字符串数组。
语法格式为:
public string[] GetSubKeyNames ()
l     返回值:包含当前项的子项名称的字符串数组。
(8)SetValue( )方法
此方法设置指定的名称/值对。
语法格式为:
public void SetValue (string name,Object value)
参数说明如下。
l     name:要存储的值的名称。
l     value:要存储的数据。
 注意:对注册表操作使用RegistryKey类和Registry类时,必须引用Microsoft.Win32 命名空间。
实现过程
(1)新建一个Windows应用程序,将其命名为Ex16_07,默认窗体为Form1。
(2)在Form1窗体中,主要添加3个TextBox控件,用于输入注册信息;添加两个Button控件,用来执行注册和退出操作。
(3)主要程序代码。
     private void button1_Click(object sender, EventArgs e)
        {
            if(textBox1.Text=="")
            {
                MessageBox.Show("公司名称不能为空"); return;
            }
                if(textBox2.Text=="")
                { MessageBox.Show("用户名称不能为空"); return; }
                if (textBox3.Text == "")
                { MessageBox.Show("注册码不能为空"); return; }
                //实例RegistryKey 类对象
            Microsoft.Win32.RegistryKey retkey1 = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("software").OpenSubKey("ZHY").OpenSubKey("ZHY.INI", true);
            foreach (string strName in retkey1.GetSubKeyNames())//判断注册码是否过期
            {
                if (strName == textBox3.Text)
                {
                    MessageBox.Show("此注册码已经过期");
                    return;
                }
            }//开始注册信息
            Microsoft.Win32.RegistryKey retkey = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("software", true).CreateSubKey("ZHY").CreateSubKey("ZHY.INI").CreateSubKey(textBox3.Text.TrimEnd());
            retkey.SetValue("UserName", textBox2.Text);
            retkey.SetValue("capataz", textBox1.Text);
            retkey.SetValue("Code", textBox3.Text);
            MessageBox.Show("注册成功,您可以使用本软件");
            Application.Exit();
        }
举一反三
根据本实例,读者可以实现以下功能。
*  注册信息加密后存入注册表。
*  记录用户试用次数的注册程序。
实例470 利用网卡序列号设计软件注册程序
实例说明
文本框:图16.8  软件注册本实例实现了利用本机网卡序列号生成软件注册码的功能。运行程序,自动获得本机网卡序列号,单击【生成注册码】按钮,生成软件注册码,将注册码依次输入下面的文本框,单击【注册】按钮实现软件注册功能。实例运行结果如图16.8所示。
技术要点
实现本实例功能主要用到了Microsoft.Win32命名空间下的Registry类的CurrentUser属性、RegistryKey类的OpenSubKey( )方法、GetSubKeyNames( )方法、SetValue( )方法、CreateSubKey( )方法、System.Management命名空间下的ManagementClass类的GetInstances( )方法、ManagementObjectCollection类和ManagementObject类。Microsoft.Win32命名空间下的类和方法在第16章实例469中已经做过介绍,这里不再详细说明,下面主要对System.Management命名空间及该命名空间下的类进行详细介绍。
(1)System.Management命名空间
提供对大量管理信息和管理事件集合的访问,这些信息和事件是与根据Windows管理规范 (WMI)结构对系统、设备和应用程序设置检测点有关的。
(2)ManagementClass类
表示公共信息模型(CIM)管理类。管理类是一个WMI类,如Win32_LogicalDisk和Win32_Process,前者表示磁盘驱动器,后者表示进程(如Notepad.exe)。
语法格式为:
public class ManagementClass : ManagementObject
(3)GetInstances( )方法
返回该类的所有实例的集合。
语法格式为:
public ManagementObjectCollection GetInstances ()
l     返回值:表示该类实例的ManagementObject对象的集合。
(4)ManagementObjectCollection类
基于指定的查询检索管理对象的集合。此类是用于检索管理信息的较为常用的入口点之一。例如,可以用于枚举系统中的所有磁盘驱动器、网络适配器、进程及更多管理对象,或者用于查询所有处于活动状态的网络连接以及暂停的服务等。
(5)ManagementObject类
表示 WMI 实例。
实现过程
(1)新建一个Windows应用程序,将其命名为Ex16_08,默认窗体为Form1。
(2)在Form1窗体中添加4个TextBox控件、3个Button控件和6个Label控件。其中,TextBox控件用输入注册码,Button控件用来执行注册、退出和生成注册码操作,Label控件用于显示计算机名称、网卡序列号、软件注册码和一些提示信息等。
(3)主要程序代码。
获得网卡序列号和计算机名称的实现代码如下:
  private void Form1_Load(object sender, EventArgs e)
        {
            label2.Text = Environment.MachineName.ToString();//得到计算机名
            label4.Text = GetNetCardMacAddress();//得到网卡信息
        }
        //获得网卡信息函数
       public string GetNetCardMacAddress()
        {     
           ManagementClass mc = new ManagementClass("Win32_NetworkAdapterConfiguration");
           ManagementObjectCollection moc = mc.GetInstances();
            string str = "";
            foreach (ManagementObject mo in moc)
            {
                if ((bool)mo["IPEnabled"] == true)
                    str = mo["MacAddress"].ToString();
            }
            return str;
        }
生成注册码的实现代码如下:
  string[] strLanCode = new string[12];// 网卡信息存储
        string[] strkey ={ "Q", "W", "7", "E", "D", "F", "2", "G", "R", "T", "Y", "8", "P", "N", "B", "V", "C", "X", "Z", "0", "9", "I", "8", "6", "U", "O", "P", "M", "5", "4", "3", "1", "A", "S", "H", "J", "K", "L" };
        //生成注册码
        public int intRand = 0;//判断随机生成次数
        private void button1_Click(object sender, EventArgs e)
        {
            //把网卡信息转换成字符串
            string strCode = GetNetCardMacAddress();//调用函数获取网卡信息
            strCode = strCode.Substring(0, 2) + strCode.Substring(3, 2) + strCode.Substring(6, 2) + strCode.Substring(9, 2) + strCode.Substring(12, 2) +strCode.Substring(15, 2);
            string strb = strCode.Substring(0, 4) + strCode.Substring(4, 4) + strCode.Substring(8, 4);//网卡信息存储
            for (int i = 0; i < strLanCode.Length; i++)//把网卡信息存入数组
            {
                strLanCode[i] = strb.Substring(i, 1);
            }
            Random ra = new Random();
            switch (intRand)//随机生成注册码的顺序
            {
                case 0:
                    label5.Text = strCode.Substring(0, 4) + "-" + strCode.Substring(4, 4) + "-" + strCode.Substring(8, 4) + "-" + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString();
                    intRand = 1;
                    break;
                case 1:
                    label5.Text = strCode.Substring(0, 4) + "-" + strCode.Substring(4, 4) + "-" + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + "-" + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString();
                    intRand = 2;
                    break;
                case 2:
                    label5.Text = strCode.Substring(0, 4) + "-" + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + "-" + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + "-" + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString();
                    intRand = 3;
                    break;
                case 3:
                    label5.Text = strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + "-" + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + "-" + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + strLanCode[ra.Next(0, 11)] + "-" + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString() + strkey[ra.Next(0, 37)].ToString();
                    intRand = 0;
                    break;
            }
        }
注册软件的实现代码如下。
   private void button2_Click(object sender, EventArgs e)
        {
            if (label5.Text == "")
            { MessageBox.Show("请生成注册码"); }
            else
            {
                string strNameKey = textBox1.Text.TrimEnd() + textBox2.Text.TrimEnd() + textBox3.Text.TrimEnd() + textBox4.Text.TrimEnd();
                string strNumber = label5.Text.Substring(0, 4) + label5.Text.Substring(5, 4) + label5.Text.Substring(10, 4) + label5.Text.Substring(15, 4);
                if (strNameKey == strNumber)
                {
                    Microsoft.Win32.RegistryKey retkey1 = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("software").OpenSubKey("ZHY").OpenSubKey("ZHY.INI", true);
                foreach (string strName in retkey1.GetSubKeyNames())//判断注册码是否过期
                    {
                        if (strName == strNameKey)
                        {
                            MessageBox.Show("此注册码已经过期");
                            return;
                        }
                    }//开始注册信息
                    Microsoft.Win32.RegistryKey retkey = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("software",true).CreateSubKey("ZHY").CreateSubKey("ZHY.INI").CreateSubKey(strNumber.TrimEnd());
                    retkey.SetValue("UserName", "明日科技");
                    MessageBox.Show("注册成功!", "提示");
                    Application.Exit();
                }
                else
                { MessageBox.Show("注册码输入错误"); }
            }
        }
举一反三
根据本实例,读者可以实现以下功能。
*  应用组件的注册使用。
*  销售的软件产品进行授权。
实例471 根据cpu序列号、磁盘序列号设计软件注册程序
实例说明
文本框:图16.9  软件注册程序本实例根据计算机的cpu号和硬盘序列号经过简单的计算自动生成一组无规律的注册码来实现应用程序的注册。运行程序,单击【生成机器码】按钮,生成24位的机器码,单击【生成注册码】按钮,根据生成的机器码自动转换出24位注册码,将注册码输入文本框中,单击【注册】按扭,完成软件注册功能。实例运行对果如图16.9所示。
技术要点
实现本实例功能主要用到了Microsoft.Win32命名空间下的Registry类的CurrentUser属性、RegistryKey类的OpenSubKey( )方法、GetSubKeyNames( )方法、SetValue( )方法、CreateSubKey( )方法、System.Management命名空间下的ManagementClass类的GetInstances( )方法、ManagementObjectCollection类和ManagementObject类、Char字符、Random类的Next( )方法。Microsoft.Win32和System.Management命名空间下的类和方法在第16章实例469和470中已经做过介绍,这里不再详细讲解。下面对本实例中用到的其他知识进行详细介绍。
(1)Char字符
Char类型的常数可以写成字符、十六进制换码序列或Unicode表示形式,用户也可以显式转换整数字符代码。
(2)Random类
表示伪随机数生成器,一种能够产生满足某些随机性统计要求的数字序列的设备。
(3)Next方法
返回一个指定范围内的随机数。
语法格式为:
public virtual int Next (int minValue,int maxValue)
参数说明如下。
l     minValue:返回的随机数的下界(随机数可取该下界值)。
l     maxValue:返回的随机数的上界(随机数不能取该上界值)。maxValue必须大于或等于minValue。
l  返回值:一个大于或等于minValue且小于maxValue的32位带符号整数,即返回的值范围包括minValue但不包括maxValue。如果minValue等于maxValue,则返回minValue。
实现过程
(1)新建一个Windows应用程序,将其命名为Ex16_08,默认窗体为Form1。
(2)在Form1窗体中,主要添加一个TextBox控件,用来输入注册码;添加4个Button控件,用来执行注册、退出、生成注册码和生成机器码操作;添加3个Label控件,用于显示软件注册码和机器码等信息。
(3)主要程序代码。
获得CPU序列号和硬盘序列号的实现代码如下:
        public string GetDiskVolumeSerialNumber()取得设备硬盘的卷标号
        {
            ManagementClass mc = new ManagementClass("Win32_NetworkAdapterConfiguration");
            ManagementObject disk = new ManagementObject("win32_logicaldisk.deviceid=/"d:/"");
            disk.Get();
            return disk.GetPropertyValue("VolumeSerialNumber").ToString();
        }
        public string getCpu()获得CPU的序列号
        {
            string strCpu = null;
            ManagementClass myCpu = new ManagementClass("win32_Processor");
            ManagementObjectCollection myCpuConnection = myCpu.GetInstances();
            foreach( ManagementObject myObject in myCpuConnection)
            {
                strCpu = myObject.Properties["Processorid"].Value.ToString();
                break;
            }
            return strCpu;
        }
生成机器码的实现代码如下:
   private void button1_Click(object sender, EventArgs e)
        {
          label2.Text = getCpu() + GetDiskVolumeSerialNumber();//获得24位CPU和硬盘序列号
            string[] strid = new string[24];
            for (int i = 0; i < 24; i++)//把字符赋给数组
            {
                strid[i] = label2.Text.Substring(i, 1);
            }
            label2.Text = "";
            Random rdid = new Random();
            for (int i = 0; i < 24; i++)//从数组随机抽取24个字符组成新的字符生成机器码
            {
                label2.Text += strid[rdid.Next(0, 24)];
            }
        }
生成注册码的实现代码如下:
public int[] intCode = new int[127];//用于存密钥
        public void setIntCode()//给数组赋值小于10个的随机数
        {
            Random ra = new Random();
            for (int i = 1; i < intCode.Length;i++ )
            {
                intCode[i] = ra.Next(0, 9);
            }
        }
        public int[] intNumber = new int[25];//用于存机器码的AscII值
        public char[] Charcode = new char[25];//存储机器码字
        //生成注册码
        private void button2_Click(object sender, EventArgs e)
        {
            if (label2.Text != "")
            {
                //把机器码存入数组中
                setIntCode();//初始化127位数组
                for (int i = 1; i < Charcode.Length; i++)//把机器码存入数组中
                {
                    Charcode[i] = Convert.ToChar(label2.Text.Substring(i - 1, 1));
                }
            for (int j = 1; j < intNumber.Length; j++)//把字符的ASCII值存入一个整数组中
                {
     intNumber[j] = intCode[Convert.ToInt32(Charcode[j])] + Convert.ToInt32(Charcode[j]);
                }
                string strAsciiName = null;//用于存储机器码
                for (int j = 1; j < intNumber.Length; j++)
                {
                    //MessageBox.Show((Convert.ToChar(intNumber[j])).ToString());
               if (intNumber[j] >= 48 && intNumber[j] <= 57)//判断字符ASCII值是否在0~9之间
                    {
                        strAsciiName += Convert.ToChar(intNumber[j]).ToString();
                    }
         else if (intNumber[j] >= 65 && intNumber[j] <= 90)//判断字符ASCII值是否在A~Z之间
                    {
                        strAsciiName += Convert.ToChar(intNumber[j]).ToString();
                    }
        else if (intNumber[j] >= 97 && intNumber[j] <= 122)//判断字符ASCII值是否在a~z之间
                    {
                        strAsciiName += Convert.ToChar(intNumber[j]).ToString();
                    }
                    else//判断字符ASCII值不在以上范围内
                    {
                        if (intNumber[j] > 122)//判断字符ASCII值是否大于z
                        { strAsciiName += Convert.ToChar(intNumber[j] - 10).ToString(); }
                        else
                        {
                            strAsciiName += Convert.ToChar(intNumber[j] - 9).ToString();
                        }
                    }
                    label3.Text = strAsciiName;//得到注册码
                }
            }
            else
            { MessageBox.Show("请选生成机器码","注册提示"); }
        }
举一反三
根据本实例,读者可以实现以下功能。
*  获取CPU信息。
*  进行远程软件产品的注册。