C#操作ini文件的问题

 

  本来操作ini文件调用windows提供的那几个api是非常简单的事,可是在C#中用来却不见的那么
舒畅,主要是因为API调用后返回的串值上的问题.
  在.net的sdk中提供的示例是用StringBuilder和IntPtr类来获取调用后返回的串值.以下是sdk
中的Buffers示例:
  LibWrap
{
	
	[ DllImport( , CharSet=CharSet.Auto )]
	    GetSystemDirectory( StringBuilder sysDirBuffer,  size );  
	
	
	[ DllImport( , CharSet=CharSet.Auto )]
	    GetUserName( StringBuilder userNameBuffer,   size );	
    
	
    [DllImport(, CharSet=CharSet.Auto)]
       IntPtr GetCommandLine();	
}

  App
{
	   Main()
	{
	
	StringBuilder sysDirBuffer =  StringBuilder(  );
	LibWrap.GetSystemDirectory( sysDirBuffer, sysDirBuffer.Capacity );
	Console.WriteLine( , sysDirBuffer );
	
	
	StringBuilder userNameBuffer =  StringBuilder(  );
	 size = userNameBuffer.Capacity;
	LibWrap.GetUserName( userNameBuffer,  size );
	Console.WriteLine( , userNameBuffer );
	
        IntPtr cmdLineStr = LibWrap.GetCommandLine();
         commandLine = Marshal.PtrToStringAuto( cmdLineStr );
        Console.WriteLine( , commandLine );		
	}
}

  表面上以上的例子完全达到了我们的目的.可是当调用的api返回的串是用  分隔的n条串时(如下面用
到的GetPrivateProfileSectionNames和GetPrivateProfileSection)以上的方法就只能获取最前面的一条
串了,因为.net中的 Marshal类提供的方法和StringBuilder类的ToString()只是简单的截取  前的串数
据,所以后面的数据就给无辜的丢弃了.

  有朋友想到了用byte数组来缓存串数据的方法 -- 分配一个很大的byte数组把数组的指针用托管的方式
封送给API(添加[MarshalAs(UnmanagedType.LPArray)]特性),再用System.Text提供字符编码转换类(
ASCIIEncoding、UTF8Encoding)把byte数组里的串数据转换为String类数据.这种方法虽然实现了我们的伟
大目标,但在串的转换上却损失了不少的性能.

  我个人经过一翻的探索发现一个目前来讲比较好的解决方法. 这个方法的主要思想是:
  ,string类允许我们用char、sbyte类型的数据来填充构造函数,所以我们有办法预先分配一个足够的缓冲
    区来存储我们要的返回串数据;
  ,我们同样可以把string类型的数据的指针用托管的方式封送给API,Ansi的API添加[MarshalAs(
    UnmanagedType.LPStr)]特性,unicode的API添加[MarshalAs(UnmanagedType.LPWStr)]特性.
    注意:如果存在unicode的API最好使用unicode的API,因为.net中的string类存储的unicode字符串且,Ansi
    的API也是调用unicode的API的,所以我们直接调用unicode的API减免了CLR把Ansi字符串转换为unicode
    字符串的步骤,自然性能上也就不同了.

  OK!下面是我实现的ini文件操作类的代码:



  CIni
{
	#region 段信息的获取
	
	[DllImport(, EntryPoint = , CharSet= CharSet.Unicode )]
            getSectionNames(
        [MarshalAs(UnmanagedType.LPWStr )]  szBuffer,  nlen,  filename);

	
	[DllImport(,EntryPoint=, CharSet = CharSet.Unicode)]
            getSectionValues( Section,
        [MarshalAs(UnmanagedType.LPWStr)]  szBuffer,  nlen,  filename);
	#endregion


	#region 键值的获取和设置
	
	[DllImport(,EntryPoint= , CharSet = CharSet.Unicode)]
	    getKeyIntValue( Section, Key, nDefault, FileName);

	
	[DllImport(,EntryPoint= , CharSet = CharSet.Unicode)]
	    getKeyValue( section, key, lpDefault,
        [MarshalAs(UnmanagedType.LPWStr)]  szValue,  nlen,  filename);

	
	
	
	[DllImport(,EntryPoint= , CharSet = CharSet.Unicode)]
	    setKeyValue( Section, key, szValue, FileName);

	
	[DllImport(,EntryPoint = , CharSet = CharSet.Unicode)]
	    setSectionValue( section, szvalue, filename);
	#endregion

	    []sept={  };	

	  m_Path=;		

	
	
	
	  Path
	{
		set {m_Path=value;}
		get { m_Path;}
	}
	
	 CIni(){}
	 CIni( szPath)
	{
		m_Path=szPath;
	}

	
	
	
	  []SectionNames
	{
		get
		{	
			 buffer =  (  ,);
			 nlen = getSectionNames(buffer,-,m_Path)-;
			 (nlen >0 )
			{
			 buffer.Substring(,nlen).Split(sept);
			}
			 ;
		}
	}	

	
	
	
        
        
	
         [] SectionValues( section, bufferSize)
        {
             buffer =  (  , bufferSize);
             nlen = getSectionValues(section, buffer, bufferSize, m_Path) - ;
             (nlen > )
            {
                 buffer.Substring(, nlen).Split(sept);
            }
             ;
        }
         [] SectionValues( section)
        {
             SectionValues(section, );
        }

	
        
	
	
	
	
         Dictionary<string, > SectionValuesEx( section,  bufferSize)
	{
            [] sztmp = SectionValues(section, bufferSize);
		 (sztmp !=)
		{
		 ArrayLen=sztmp.Length;
		 ( ArrayLen >0)
		{
                    Dictionary<string, > dtRet =  Dictionary<string, >();
			 ( i=;i <ArrayLen;i++)
			{
			 pos1=sztmp[i].IndexOf(  );
			 (pos1 >1)
			{
				 nlen=sztmp[i].Length;
                            
				pos1++;
				 (pos1<nlen)
                             dtRet.Add(sztmp[i].Substring(, pos1-), sztmp[i].Substring(pos1, nlen - pos1));
			}
			}
                     dtRet;
		}
		}
		 ;
	}
         Dictionary<string, > SectionValuesEx( section)
        {
             SectionValuesEx(section, );
        }

	
	
	
	
	
	
	  setSectionValue( section, szValue)
	{
		 setSectionValue(section,szValue,m_Path);
	}

	
	
	
	
	
	
	  getKeyIntValue( section, key)
	{
		 getKeyIntValue(section,key,-,m_Path);
	}
	
	
	
	
	
	
	
	
	  setKeyIntValue( section, key, dwValue)
	{
		 setKeyValue(section,key,dwValue.ToString(),m_Path);
	}

	
	
	
	
	
	
	  getKeyValue( section, key)
	{
		 szBuffer=  (  ,);
		 nlen=getKeyValue(section,key,, szBuffer,,m_Path);
		 szBuffer.Substring(,nlen-);
	}
    
	
	
	
	
	
	
	
	  setKeyValue( Section , key, szValue)
	{
		 setKeyValue(Section,key,szValue,m_Path);
	}
	}

CIni类的使用示例

 

阅读更多
个人分类: C#
想对作者说点什么? 我来说一句

C# ini文件操作

2016年11月16日 99KB 下载

没有更多推荐了,返回首页

不良信息举报

C#操作ini文件的问题

最多只允许输入30个字

加入CSDN,享受更精准的内容推荐,与500万程序员共同成长!
关闭
关闭