【C#】String| StringBuilder 字符

原文链接:https://www.cnblogs.com/huameitang/p/10528646.html

字符串是用于表示文本的字符的有序集合。 String对象是对象的有序集合 System.Char ,表示字符串; System.Char 对象对应于 utf-16 代码单元。 对象的值 String 是对象的顺序集合的内容 System.Char ,并且该值是不可变的 (也就是说,它是只读,反射也不能修改) 。 有关字符串不可变性的详细信息,请参阅 永久性和 StringBuilder 类 部分。 内存中对象的最大大小 String 为 2 GB,即约1000000000个字符。

string字符串的两大特性:

1、String不可变性:这个特性就是值类型的特性,所以在string操作上(赋值、拼接 、作参数)感觉就是值类型。

 2、 拘留池(intern pool)的机制:是为了避免分配大量的字符串对象造成的过多的内存空间浪费。

.NET 的 CLR 运行时会在运行期间管理一个字符拘留池(string intern pool),在字符串拘留池中的字符串只有一个实例。

程序中会存在大量的字符串对象,如果每次都创建一个字符串对象,会比较浪费内存、性能低,因此CLR做了“暂存池”(拘留池,缓冲池,暂存池),在一些情况下对于字符串对象进行了重用。

例如,在下面的代码中,变量 abc 都是同一个实例:

var a = "walterlv";
var b = "walterlv";
var c = "walterlv";

把字符串输入拘留池和从暂存池中获取字符串

String.Intern:把字符串输入拘留池

String.Intern(x),将X添加到字符串池中,如果这个字符串已经存在池中,就返回这个存在的引用;如果不存在就将它加入到池中,并返回引用。

String.ISInterned(str):从暂存池中获取字符串

 

using System.Reflection;
using System.Text;

string str1 = string.IsInterned("ddkkkkd");//会被存入拘留池
String str2 = new StringBuilder().Append("wx").Append("yz").ToString();
if ( str1 == null) Console.WriteLine("f");
if (string.IsInterned(str2) == null) Console.WriteLine("str2 is not Intern");
Console.WriteLine(string.IsInterned("ddkkkkd")); 

 

 

 

String.IsInterned(str)要在暂存池中搜索的字符串,如果 str 在公共语言运行时的暂存池中,则返回对它的引用;否则返回 null

Console.WriteLine(object.ReferenceEquals("xyz", s));//编译器会把常量xyz ,加入池中
Console.WriteLine(object.ReferenceEquals("x" + "y" + "z", s));//编译器会把"x" + "y" + "z"当作常量xyz ,加入池中

 不要池化

你可以在程序集中标记 CompilationRelaxations.NoStringInterning,这样,此程序集中的字符串就不会被池化。即便是在编译期间写下的字符串也会在运行时生成新的实例。

方法是在一个 C# 代码文件中添加特性标记。

[assembly: CompilationRelaxations(CompilationRelaxations.NoStringInterning)]

注意:不是所有的字符串都放在暂存池中,运行时期动态创建的字符串不会被加入到驻留池中。

关于字符串常量池的更深理解:

1. 驻留池由CLR来维护,其中的所有字符串对象的值都不相同。

2. 只有编译阶段的文本字符常量会被自动添加到驻留池。

3.运行时期动态创建的字符串不会被加入到驻留池中。

4.string.Intern()可以把动态创建的字符串加入到驻留池中。
5、林业耶夫总结:字符串内插生成的字符串 不会进入拘留池。

字符串虽然是引用类型,但是实即他更像值类型。

1)字符串的直接赋值:本身字符串就是引用类型,应该使用  new 对象方法一个实例,但是微软为了方便大家,可以直接定义字符串变量 并且赋值操作,例如: string a = "我的中国心"; ,这样只是简化我们的操作;

2)一个字符串赋值给另一个字符串变量:正常的引用类型会将两个引用变量指向同一个地址,但是一个字符串变量赋值给另一个字符串变量时,缺时建立了两个不同的地址空间,例如:

 string a = "12345"; string b = a;

上面的代码是两个不同的地址引用,只是把a的字符串内容赋值给b,a和b内容是一样的;

3)同一个字符串的多次赋值:按照一般的思维对一个字符串变量赋值,只是改变其内容,不会改变其地址,但是字符串比较奇葩,当给同一个字符串变量再次赋值的时候,它会重新分配内存空间,建立一个新的地址,然后把

这个地址赋值给原来的字符串变量,举例说明:

 string a= "123";  a = "456"
当第二次给a赋值为"456"时,它是创建新的内存空间,然后把新建的内存地址赋值给a变量,以前的"123"的内存摒弃不用,等待垃圾回收。

4)字符串作为函数参数传递:当字符串作为函数的参数传递时,本身是引用类型,应该是将变量的地址引用传递过去,以后在函数里对该参数的修改都会改变该字符串的值,但是我告诉你,结果它只是传递了该字符串的副本给

函数体,在函数里对该字符修改,居然不影响传递参数的值,当然,字符串的传递也可以当引用类型使用,主要添加ref 即可,可截图:

 

   5)字符串的比较:在字符串作为引用时,比较两个引用类型是否相等,只是比较两个引用的地址是否相等(除非你重载了Equal函数),但是当我们在比较字符串的时候,发现其实他们比较的是字符串的内容,并非是引用的地址,

这是引用string类重载了equal函数,是指比较字符的内容,在这点上  == 和 equal的结果其实是一样的;

6)字符串的内存驻留:当我们在创建具有相同的字符串内容的变量时,这些字符串变量其实指向的同一个内存地址,这点有点像C++里的内联;

  StringBuilder

StringBuilder的工作原理大致是这样的:

内部维护一个char[],并且有一个初始容量16。

新的字符串都加入到这个数组中。

当加入的字符超过容量时,就重新new一个更大的数组,并将原先的数组内容拷入新数组中。

将原有的数组进行垃圾回收,新的字符串加入到使用新的字符数组中。

StringBuilder的ToString方法见字符数组转换为一个String输出。

 StringBuilder与String互转

StringBuilder类是一个可变的字符序列。

StringBuilder() 
          构造一个不带任何字符的字符串生成器,其初始容量为 16 个字符。
StringBuilder(CharSequence seq) 
          构造一个字符串生成器,它包含与指定的 CharSequence 相同的字符。
StringBuilder(int capacity) 
          构造一个不带任何字符的字符串生成器,其初始容量由 capacity 参数指定。
StringBuilder(String str) 
          构造一个字符串生成器,并初始化为指定的字符串内容。

StringBuilder类的几个常用方法:

append(任意类型)  追加到字符串后面

reverse 反转字符串

insert(int offset, 任意类型)  在某个index后插入字符串

toString()  返回String类的对象

先看一段String类的字符串拼接的代码。

 

String s = "hello" 会在常量池开辟一个内存空间来存储”hello"

s += "world"会先在常量池开辟一个内存空间来存储“world"。然后再开辟一个内存空间来存储”helloworld“。

这么以来,001与002就成为了垃圾内存空间了。这么简单的一个操作就产生了两个垃圾内存空间,如果有大量的字符串拼接,将会造成极大的浪费。
StringBuilder的作用

上面的例子可以知道String类的字符串拼接会产生大量的垃圾内存空间。但是StringBuilder的字符串拼接是直接在原来的内存空间操作的,即直接在hello这个内存空间把hello拼接为helloworld。

来证明下:

public class StringBuilderTest {
    public static void main(String[] args){
        StringBuilder sb = new StringBuilder();
        StringBuilder sb2 = sb.append("hello");
        System.out.println(sb);
        System.out.println(sb2);
        // 引用类型,判断的是他们的内存地址是否一样
        System.out.println(sb == sb2);
    }
}

输出结果是:

hello
hello
true
String类与StringBuilder类的相互转换

1.String类转换为StringBuilder类

public class String12 {
    public static void main(String[] args){
        String s = "hello";
        StringBuilder sb = new StringBuilder(s);
        System.out.println(sb);
    }
}
2.StringBuilder类转换为String类

public class String12 {
    public static void main(String[] args){
        StringBuilder sb = new StringBuilder();
        sb.append("abc").append("efg");
        String s = sb.toString();
        System.out.println(s);
    }
}

 string 和char之间的互操作

using IteratorTest;
using System.Reflection;
using System.Security.AccessControl;

Console.WriteLine();

string name = "sfsfsfsdfsdfwertdhghk,u8rgdhf";
//字符串转成字符数组
char[] chars = name.ToCharArray();

//字符创当字符数组一样遍历
 foreach (char c in name)
{
    Console.WriteLine(c);

}

 //3种方式 将字符数组转成字符串
 
Console.WriteLine(new string(chars));
Console.WriteLine(string.Concat(chars));
Span<char> span = new Span<char>(chars);
Console.WriteLine(span.ToString());

 

posted @ 2021-09-04 11:53  小林野夫  阅读(558)  评论(0编辑  收藏  举报
原文链接:https://www.cnblogs.com/cdaniu/