C# 9.0记录类型:解锁开发效率的魔法密码

一、引言:记录类型的神奇登场

在 C# 的编程世界中,数据结构就像是构建软件大厦的基石,其重要性不言而喻。然而,传统的数据结构定义方式,尤其是在处理简单的数据承载对象时,常常显得繁琐复杂。例如,当我们需要定义一个表示人员信息的类时,通常需要编写如下代码:

public class Person
{public string Name { get; set; }public int Age { get; set; }public Person(string name, int age){Name = name;Age = age;}
}

上述代码中,我们不仅要定义属性,还要编写构造函数来初始化这些属性。当项目中存在大量类似这样简单的数据结构时,代码量会迅速膨胀,可读性和维护性也会随之降低。

而 C# 9.0 带来的记录类型(Record Types),就像是一把神奇的钥匙,为我们打开了简化数据结构定义的大门。记录类型能够以一种简洁明了的方式定义数据结构,让代码变得更加优雅和高效。例如,使用记录类型来定义上述的Person结构,只需一行代码:

public record Person(string Name, int age);

仅仅这一行代码,就完成了属性定义、构造函数生成等一系列操作,极大地简化了开发流程。记录类型不仅能减少代码的编写量,还在相等性比较、对象克隆等方面提供了便捷的功能,让开发效率得到显著提升。接下来,就让我们深入探索 C# 9.0 记录类型的奥秘,看看它是如何施展这简化数据结构的 5 步魔法的吧。

二、第一步:极简定义,告别冗长构造

在 C# 传统的类定义方式中,当我们需要创建一个简单的数据结构类时,往往需要编写较多的代码。以定义一个表示学生信息的类Student为例,传统方式如下:

public class Student
{public string Name { get; set; }public int Age { get; set; }public string Major { get; set; }public Student(string name, int age, string major){Name = name;Age = age;Major = major;}
}

在这段代码中,我们不仅要定义Name、Age和Major这三个属性,还要编写构造函数来初始化这些属性。如果项目中有大量类似这样简单的数据结构类,代码量会显著增加,而且这些重复的代码会降低代码的可读性和维护性。

而在 C# 9.0 中,使用记录类型来定义相同的Student数据结构,代码变得极其简洁:

public record Student(string Name, int Age, string Major);

仅仅这一行代码,就完成了属性定义、构造函数生成等一系列操作。编译器会自动为这个记录类型生成一个构造函数,该构造函数接受Name、Age和Major作为参数,用于初始化相应的属性。不仅如此,记录类型还会自动生成ToString、Equals和GetHashCode等方法,这些方法都是基于记录类型的属性值来实现的。

例如,当我们创建Student记录类型的实例时,可以这样使用:

var student1 = new Student("Alice", 20, "Computer Science");
Console.WriteLine(student1);

上述代码创建了一个student1实例,并使用Console.WriteLine输出该实例。由于记录类型自动生成了ToString方法,所以输出结果会以一种易读的格式展示student1的属性值,大致如下:

Student { Name = Alice, Age = 20, Major = Computer Science }

通过对比传统类和记录类型的定义方式,我们可以明显看出记录类型在简化数据结构定义方面的强大优势。它减少了大量的样板代码,让开发者能够更专注于业务逻辑的实现,提高了开发效率 。

三、第二步:轻松克隆,“with” 关键字显神通

在传统的 C# 编程中,当我们需要克隆一个对象并对其属性进行修改时,往往需要编写较为繁琐的代码。例如,对于前面定义的Student类,如果要克隆一个Student对象并修改其年龄,可能需要这样实现:

public class Student
{public string Name { get; set; }public int Age { get; set; }public string Major { get; set; }public Student(string name, int age, string major){Name = name;Age = age;Major = major;}// 手动实现克隆方法public Student Clone(){return new Student(Name, Age, Major);}
}// 使用克隆方法并修改属性
var student1 = new Student("Bob", 21, "Mathematics");
var student2 = student1.Clone();
student2.Age = 22;

在上述代码中,我们不仅要手动编写Clone方法来实现对象的克隆,而且代码看起来比较冗长和繁琐。

而在 C# 9.0 的记录类型中,使用with关键字可以轻松地实现对象的克隆并修改属性。with关键字就像是一个神奇的 “复制修改器”,它会创建一个基于现有记录对象的副本,并且可以在创建副本的同时修改指定的属性值,而其他未指定修改的属性则保持不变 。这一特性不仅大大简化了代码,还保障了对象的不变性原则,即原始对象不会被修改,而是生成一个新的对象来承载修改后的状态。

还是以Student记录类型为例,使用with关键字进行对象克隆和属性修改的代码如下:

public record Student(string Name, int Age, string Major);// 使用with关键字克隆并修改属性
var student1 = new Student("Charlie", 22, "Physics");
var student3 = student1 with { Age = 23 };

在上述代码中,student1 with { Age = 23 }这行代码创建了一个student1的副本student3,并且将student3的Age属性修改为 23,而Name和Major属性则与student1保持一致。

如果Student记录类型中的属性使用了init访问器来实现不可变,with关键字同样适用,并且能更好地体现不可变对象的优势。例如:

public record Student(string Name, int Age, string Major)
{public string Name { get; init; } = Name;public int Age { get; init; } = Age;public string Major { get; init; } = Major;
};var student1 = new Student("David", 23, "Chemistry");
var student4 = student1 with { Age = 24 };

在这个例子中,Student记录类型的属性是不可变的,通过with关键字创建新对象并修改属性,既保证了数据的不可变性,又实现了灵活的数据更新。

通过对比可以看出,with关键字在记录类型中实现对象克隆和属性修改的操作上,比传统的类方式更加简洁、直观,大大提高了开发效率 。

四、第三步:智能相等,告别 equals 的烦恼

在传统的 C# 编程中,当我们需要判断两个对象是否相等时,往往需要手动重写Equals和GetHashCode方法。以之前定义的Student类为例,如果要实现基于属性值的相等性比较,代码如下:

public class Student
{public string Name { get; set; }public int Age { get; set; }public string Major { get; set; }public Student(string name, int age, string major){Name = name;Age = age;Major = major;}public override bool Equals(object obj){if (obj == null || GetType()!= obj.GetType()){return false;}Student other = (Student)obj;return Name == other.Name && Age == other.Age && Major == other.Major;}public override int GetHashCode(){return HashCode.Combine(Name, Age, Major);}
}

在上述代码中,我们需要手动编写Equals方法来比较两个Student对象的属性值是否相等,同时还需要编写GetHashCode方法来生成哈希码,以确保相等的对象具有相同的哈希码 。这不仅增加了代码的编写量,还容易出现错误。

而在 C# 9.0 的记录类型中,编译器会自动为记录类型生成基于属性值的相等性比较逻辑,包括Equals和GetHashCode方法 。这意味着我们无需手动重写这些方法,就可以直接使用==运算符或Equals方法来判断两个记录类型的实例是否相等。

例如,对于之前定义的Student记录类型:

public record Student(string Name, int Age, string Major);

我们可以这样进行相等性比较:

var student1 = new Student("Eve", 23, "Biology");
var student2 = new Student("Eve", 23, "Biology");
var student3 = new Student("Frank", 24, "Economics");Console.WriteLine(student1 == student2); // 输出True
Console.WriteLine(student1.Equals(student2)); // 输出True
Console.WriteLine(student1 == student3); // 输出False

在上述代码中,student1和student2具有相同的属性值,所以student1 == student2和student1.Equals(student2)都返回True;而student1和student3的属性值不同,所以student1 == student3返回False。

记录类型的这种自动相等性比较功能,在处理集合中的数据比较、去重等操作时非常有用。例如,在一个包含Student记录类型的集合中,判断两个Student是否相等时,直接使用==运算符即可:

var studentList = new List<Student>
{new Student("Grace", 22, "History"),new Student("Hank", 21, "Geography")
};var newStudent = new Student("Grace", 22, "History");
bool isExists = studentList.Any(s => s == newStudent);
Console.WriteLine(isExists); // 输出True

上述代码中,studentList.Any(s => s == newStudent)用于判断studentList集合中是否存在与newStudent相等的元素,由于记录类型的自动相等性比较功能,我们可以非常方便地实现这一操作。

通过对比可以看出,C# 9.0 记录类型的自动相等性比较功能,大大简化了代码,减少了开发人员的工作量,提高了开发效率 。

五、第四步:深度剖析,记录类型的魔法内幕

前面我们已经体验到了 C# 9.0 记录类型在简化数据结构定义、对象克隆和相等性比较方面的强大功能。接下来,让我们深入记录类型的内部,探索它背后的魔法机制,看看编译器是如何施展这些神奇的操作的。

编译器的魔法合成

当我们使用记录类型定义一个数据结构时,编译器会自动为我们合成一系列非常有用的方法 。以之前定义的Student记录类型为例:

public record Student(string Name, int Age, string Major);

编译器会为这个记录类型合成以下方法:

  1. 构造函数:编译器会生成一个主构造函数,其参数与记录类型定义时的属性参数一致。例如,对于上述Student记录类型,生成的构造函数类似于:
public Student(string Name, int Age, string Major)
{this.Name = Name;this.Age = Age;this.Major = Major;
}

这个构造函数用于初始化记录类型的属性。此外,编译器还会生成一个受保护的复制构造函数(如果记录类型不是密封的),用于创建记录的副本。例如:

protected Student(Student other)
{this.Name = other.Name;this.Age = other.Age;this.Major = other.Major;
}
  1. 相等性比较方法:记录类型会自动实现基于属性值的相等性比较,编译器会合成Equals和GetHashCode方法。Equals方法用于比较两个记录类型的实例是否相等,它会比较两个实例的所有属性值。例如:
public override bool Equals(object obj)
{if (obj == null || GetType()!= obj.GetType()){return false;}Student other = (Student)obj;return Name == other.Name && Age == other.Age && Major == other.Major;
}public override int GetHashCode()
{return HashCode.Combine(Name, Age, Major);
}

GetHashCode方法用于生成记录类型实例的哈希码,它会根据记录类型的所有属性值生成一个唯一的哈希码,以确保相等的对象具有相同的哈希码 。

\3. ToString 方法:编译器会合成一个ToString方法,用于返回记录类型实例的字符串表示形式。这个字符串表示形式会包含记录类型的所有属性名和属性值,以一种易读的格式展示。例如:

public override string ToString()
{return $"Student {{ Name = {Name}, Age = {Age}, Major = {Major} }}";
}
  1. 克隆方法:虽然我们在代码中看不到名为Clone的方法,但编译器会生成一个用于创建记录副本的方法。这个方法实际上是使用with表达式来实现的,它会创建一个新的记录实例,并将原始记录的属性值复制到新实例中,同时可以根据需要修改指定的属性值。例如,对于Student记录类型,编译器生成的克隆方法类似于:
public Student Clone()
{return this with { };
}
  1. Deconstruct 方法:对于位置记录(即使用简洁语法定义的记录类型,如public record Student(string Name, int Age, string Major);),编译器还会生成一个Deconstruct方法,用于将记录类型的实例解构为其组件属性。例如:
public void Deconstruct(out string Name, out int Age, out string Major)
{Name = this.Name;Age = this.Age;Major = this.Major;
}

这个Deconstruct方法可以方便地将记录类型的实例分解为多个变量,例如:

var student = new Student("Ivy", 24, "Psychology");
var (name, age, major) = student;
Console.WriteLine($"Name: {name}, Age: {age}, Major: {major}");

初始化逻辑与不可变性

记录类型的属性默认是不可变的,这是通过使用init访问器来实现的。例如,对于Student记录类型,其属性的定义实际上类似于:

public record Student
{public string Name { get; init; }public int Age { get; init; }public string Major { get; init; }public Student(string Name, int Age, string Major){this.Name = Name;this.Age = Age;this.Major = Major;}
}

使用init访问器意味着属性只能在对象初始化时被赋值,一旦对象创建完成,属性值就不能被修改。这种不可变性有助于确保数据的一致性和安全性,特别是在多线程环境中,不可变的数据结构可以避免数据竞争和不一致的问题 。

当我们使用new关键字创建记录类型的实例时,会调用编译器生成的构造函数来初始化属性。例如:

var student = new Student("Jack", 25, "Sociology");

在这个过程中,Name、Age和Major属性会被初始化为指定的值,并且之后不能被修改。如果尝试修改属性值,例如:

student.Age = 26; // 编译错误,因为Age属性是不可变的

编译器会报错,提示属性是只读的,无法进行修改。

模式匹配的支持

记录类型对模式匹配提供了很好的支持,这使得我们可以根据记录类型的结构和属性值来进行灵活的条件判断和处理。模式匹配是一种检查数据结构的方式,它允许我们根据数据的特定模式来执行不同的代码分支。在 C# 中,模式匹配最常见的形式是switch表达式。

以Student记录类型为例,我们可以使用模式匹配来根据学生的年龄进行不同的处理:

var student = new Student("Kathy", 22, "Engineering");
switch (student)
{case { Age: < 20 }:Console.WriteLine("This student is relatively young.");break;case { Age: >= 20 and < 25 }:Console.WriteLine("This student is in a typical college age range.");break;case { Age: >= 25 }:Console.WriteLine("This student may be a mature learner.");break;
}

在上述代码中,switch表达式根据student的Age属性值进行匹配,不同的模式对应不同的处理逻辑。这种方式使得代码更加简洁、直观,避免了繁琐的条件判断和类型转换 。

记录类型还支持在模式匹配中使用with表达式来创建临时的记录副本并进行匹配。例如:

var student = new Student("Leo", 23, "Business");
switch (student)
{case { Name: "Leo", Age: var age } with { Major: "Business" }:Console.WriteLine($"Leo is a business student and his age is {age}.");break;
}

在这个例子中,通过with表达式在模式匹配中创建了一个临时的记录副本,并且只关注Name为"Leo"、Age为任意值且Major为"Business"的情况,进一步展示了记录类型在模式匹配中的灵活性和强大功能。

通过深入了解记录类型的内部机制,我们不仅知其然,还知其所以然,这将帮助我们更好地运用记录类型,编写出更加高效、优雅的代码 。

六、第五步:实际应用,记录类型大展身手

前面我们已经深入了解了 C# 9.0 记录类型的各种特性和内部机制,那么在实际的项目开发中,记录类型又能在哪些场景中发挥其强大的作用呢?下面我们将通过几个常见的应用场景来展示记录类型的实际优势。

数据传输对象(DTO)

在现代软件开发中,尤其是在分布式系统和 Web 应用程序中,数据传输对象(DTO)是一种常用的设计模式 。DTO 主要用于在不同的层(如表示层、业务逻辑层和数据访问层)之间传输数据,它通常只包含数据属性,不包含业务逻辑。

在传统的开发中,我们通常使用类来定义 DTO,例如:

public class UserDTO
{public string Username { get; set; }public string Email { get; set; }public int Age { get; set; }public UserDTO(string username, string email, int age){Username = username;Email = email;Age = age;}
}

使用类来定义 DTO 需要编写较多的样板代码,包括属性定义、构造函数等。而且,在比较两个UserDTO对象是否相等时,还需要手动重写Equals和GetHashCode方法。

而使用 C# 9.0 的记录类型来定义UserDTO,代码会变得非常简洁:

public record UserDTO(string Username, string Email, int Age);

仅仅这一行代码,就完成了属性定义、构造函数生成以及相等性比较方法的实现。在数据传输过程中,我们可以方便地创建UserDTO的实例并进行数据传递。例如,在ASP.NET Core Web API 中,我们可以将UserDTO作为控制器方法的返回类型或参数类型:

[ApiController]
[Route("[controller]")]
public class UserController : ControllerBase
{[HttpGet("{id}")]public UserDTO GetUser(int id){// 从数据库或其他数据源获取用户数据var user = new UserDTO("JohnDoe", "johndoe@example.com", 30);return user;}[HttpPost]public IActionResult CreateUser(UserDTO userDTO){// 处理创建用户的逻辑return Ok();}
}

由于记录类型的属性默认是不可变的,这符合 DTO 在数据传输过程中只传递数据而不修改数据的特性,保证了数据的一致性和安全性 。同时,记录类型的自动相等性比较功能在处理 DTO 的比较和去重等操作时也非常方便,减少了开发人员的工作量。

领域模型

在领域驱动设计(DDD)中,领域模型是对业务领域的抽象表示,它包含了业务规则和业务逻辑 。在领域模型中,有一些对象通常被称为值对象,它们主要用于表示一些具有特定业务含义的数据,例如货币金额、日期范围等。值对象的特点是它们的相等性是基于其属性值的,而不是基于对象的引用。

在传统的 C# 开发中,我们通常使用类来定义值对象,并手动实现基于属性值的相等性比较方法。例如,定义一个表示货币金额的值对象Money:

public class Money
{public decimal Amount { get; }public string Currency { get; }public Money(decimal amount, string currency){Amount = amount;Currency = currency;}public override bool Equals(object obj){if (obj == null || GetType()!= obj.GetType()){return false;}Money other = (Money)obj;return Amount == other.Amount && Currency == other.Currency;}public override int GetHashCode(){return HashCode.Combine(Amount, Currency);}
}

上述代码中,我们不仅要定义属性和构造函数,还要手动重写Equals和GetHashCode方法来实现基于属性值的相等性比较。

而使用 C# 9.0 的记录类型来定义Money值对象,代码会变得更加简洁:

public record Money(decimal Amount, string Currency);

记录类型会自动为Money生成基于属性值的相等性比较方法,这使得在领域模型中处理值对象时更加方便和高效。例如,在进行业务逻辑处理时,我们可以直接比较两个Money对象是否相等:

var money1 = new Money(100.0m, "USD");
var money2 = new Money(100.0m, "USD");
if (money1 == money2)
{// 处理相等的业务逻辑
}

此外,记录类型的不可变性也符合值对象的特性,即值对象一旦创建,其属性值就不应该被修改。这有助于确保领域模型的一致性和正确性。

配置对象

在应用程序中,配置对象用于存储应用程序的配置信息,例如数据库连接字符串、日志级别等。配置信息在应用程序运行期间通常是固定不变的,因此使用不可变的数据结构来表示配置对象是一个很好的选择。

在传统的 C# 开发中,我们可以使用类来定义配置对象,并通过将属性设置为只读来实现不可变性。例如:

public class AppConfig
{public string ConnectionString { get; }public string LogLevel { get; }public AppConfig(string connectionString, string logLevel){ConnectionString = connectionString;LogLevel = logLevel;}
}

使用类来定义配置对象需要手动编写构造函数和属性定义,并且在比较两个配置对象是否相等时也需要手动实现相等性比较方法。

而使用 C# 9.0 的记录类型来定义AppConfig配置对象,代码会更加简洁:

public record AppConfig(string ConnectionString, string LogLevel);

记录类型的自动相等性比较和不可变性特性使得配置对象的定义和使用更加方便和安全。例如,在读取配置文件并创建配置对象时,我们可以这样使用:

var config = new AppConfig("Server=localhost;Database=MyDB;User=sa;Password=password", "Info");

在应用程序中,我们可以方便地比较不同的配置对象是否相等,以确保配置的一致性。例如:

var config1 = new AppConfig("Server=localhost;Database=MyDB;User=sa;Password=password", "Info");
var config2 = new AppConfig("Server=localhost;Database=MyDB;User=sa;Password=password", "Info");
if (config1 == config2)
{// 配置相同,继续执行应用程序
}

通过以上几个实际应用场景的展示,我们可以看到 C# 9.0 的记录类型在简化数据结构定义、提高代码可读性和开发效率方面具有显著的优势。无论是在数据传输对象、领域模型还是配置对象等场景中,记录类型都能够发挥其强大的功能,帮助我们编写出更加优雅、高效的代码 。

七、总结:记录类型带来的变革与展望

C# 9.0 的记录类型无疑为我们的编程世界带来了一场意义深远的变革,它以简洁高效的特性,重塑了我们定义和操作数据结构的方式。

从定义的简洁性来看,记录类型告别了传统类定义中冗长的构造函数和繁琐的样板代码,只需一行代码就能轻松定义一个包含多个属性的数据结构,大大减少了开发过程中的重复劳动,让代码更加简洁易读。在对象克隆方面,with关键字的引入堪称神来之笔,它使得对象的克隆和属性修改变得轻而易举,不仅简化了代码逻辑,还遵循了不可变对象的设计原则,提高了代码的安全性和可维护性。而自动实现的基于属性值的相等性比较功能,更是解决了传统开发中手动重写Equals和GetHashCode方法的烦恼,让相等性判断变得直观而准确,在集合操作、数据去重等场景中发挥了巨大的作用 。

在实际应用中,记录类型在数据传输对象(DTO)、领域模型、配置对象等多个场景中都展现出了强大的优势,它能够帮助我们编写出更加优雅、高效的代码,提升项目的整体质量和开发效率。

对于广大开发者而言,C# 9.0 记录类型是一个不可多得的强大工具。我强烈建议大家在今后的项目开发中积极尝试使用记录类型,尤其是在处理那些简单的数据承载对象和需要强调不可变性的数据结构时,记录类型将为你带来意想不到的便利和效率提升。

展望未来,随着 C# 语言的不断发展和演进,我们有理由相信记录类型也将不断完善和增强。未来的版本中,记录类型可能会在性能优化、与其他语言特性的融合等方面取得更大的突破,为开发者提供更加丰富和强大的功能。让我们拭目以待,继续探索 C# 记录类型的无限可能,在编程的道路上不断前行 。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/10108.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

物联网智能项目之——智能家居项目的实现!

成长路上不孤单&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a; 【14后&#x1f60a;///计算机爱好者&#x1f60a;///持续分享所学&#x1f60a;///如有需要欢迎收藏转发///&#x1f60a;】 今日分享关于物联网智能项目之——智能家居项目…

Linux《基础指令》

在之前的Linux《Linux简介与环境的搭建》当中我们已经初步了解了Linux的由来和如何搭建Linux环境&#xff0c;那么接下来在本篇当中我们就要来学习Linux的基础指令。在此我们的学习是包括两个部分&#xff0c;即指令和关于Linux的基础知识&#xff1b;因此本篇指令和基础知识的…

Addressable学习

AssetsBundle是Unity的资源管理机制,将资源打包到AssetsBundle资源包并提供接口能从ab包里面加载资源出来。有了这个机制以后&#xff0c;我们要做资源管理&#xff0c;还需要做: a: 根据项目需求,编写编辑器扩展,提供指定资源打入对应bundle包工具策略; b: 根据项目的需求,资源…

详解u3d之AssetBundle

一.AssetBundle的概念 “AssetBundle”可以指两种不同但相关的东西。 1.1 AssetBundle指的是u3d在磁盘上生成的存放资源的目录 目录包含两种类型文件(下文简称AB包)&#xff1a; 一个序列化文件&#xff0c;其中包含分解为各个对象并写入此单个文件的资源。资源文件&#x…

【Block总结】OutlookAttention注意力,捕捉细节和局部特征|即插即用

论文信息 标题: VOLO: Vision Outlooker for Visual Recognition作者: Li Yuan, Qibin Hou, Zihang Jiang, Jiashi Feng, Shuicheng Yan代码链接: https://github.com/sail-sg/volo论文链接: https://arxiv.org/pdf/2106.13112 创新点 前景注意力机制: VOLO引入了一种称为“…

后端token校验流程

获取用户信息 前端中只有 await userStore.getInfo() 表示从后端获取数据 在页面中找到info对应的url地址&#xff0c;在IDEA中查找 这里是getInfo函数的声明&#xff0c;我们要找到这个函数的使用&#xff0c;所以点getInfo() Override public JSONObject getInfo() {JSO…

一文讲解Java中的异常处理机制

Java中的异常处理机制用于处理程序运行过程中可能发生的各种异常情况&#xff0c;通常通过try-catch-finally语句和throw关键字来实现 Throwable是Java语言中所有错误和异常的父类。它有两个主要的子类&#xff1a;Error和Exception&#xff1b;Error类代表那些严重的错误&am…

论文速读|Matrix-SSL:Matrix Information Theory for Self-Supervised Learning.ICML24

论文地址&#xff1a;Matrix Information Theory for Self-Supervised Learning 代码地址&#xff1a;https://github.com/yifanzhang-pro/matrix-ssl bib引用&#xff1a; article{zhang2023matrix,title{Matrix Information Theory for Self-Supervised Learning},author{Zh…

SpringCloud系列教程:微服务的未来(十八)雪崩问题、服务保护方案、Sentinel快速入门

前言 在分布式系统中&#xff0c;雪崩效应&#xff08;Avalanche Effect&#xff09;是一种常见的故障现象&#xff0c;通常发生在系统中某个组件出现故障时&#xff0c;导致其他组件级联失败&#xff0c;最终引发整个系统的崩溃。为了有效应对雪崩效应&#xff0c;服务保护方…

【NLP251】意图识别 与 Seq2Seq

Seq2Seq模型作为从RNN演进到Transformer和Attention机制的关键中间阶段&#xff0c;它不仅承前启后&#xff0c;还为我们深入理解这些复杂的模型架构提供了重要的基础。接下来&#xff0c;我们将详细探讨Seq2Seq模型的原理及其在自然语言处理领域中的应用。 1. 原理及网络框架 …

docker配置mysql并使用mysql connector cpp编程

mysql 配置mysql使用docker 这里使用docker安装了&#xff0c;比较简洁&#xff0c;不想使用了直接就可以把容器删掉&#xff0c;首先获取下镜像&#xff0c;如下命令 docker pull container-registry.oracle.com/mysql/community-server这里直接默认使用最新版本的mysql了 …

赛博算卦之周易六十四卦JAVA实现:六幺算尽天下事,梅花化解天下苦。

佬们过年好呀~新年第一篇博客让我们来场赛博算命吧&#xff01; 更多文章&#xff1a;个人主页 系列文章&#xff1a;JAVA专栏 欢迎各位大佬来访哦~互三必回&#xff01;&#xff01;&#xff01; 文章目录 #一、文化背景概述1.文化起源2.起卦步骤 #二、卦象解读#三、just do i…

【16届蓝桥杯寒假刷题营】第2期DAY4

【16届蓝桥杯寒假刷题营】第2期DAY4 - 蓝桥云课 问题描述 幼儿园小班的浩楠同学有一个序列 a。 他想知道有多少个整数三元组 (i,j,k) 满足 1≤i,j,k≤n 且 ai​aj​ak​。 输入格式 共2行&#xff0c;第一行一个整数 n&#xff0c;表示序列的长度。 第二行 n 个整数&#x…

基于单片机的超声波液位检测系统(论文+源码)

1总体设计 本课题为基于单片机的超声波液位检测系统的设计&#xff0c;系统的结构框图如图2.1所示。其中包括了按键模块&#xff0c;温度检测模块&#xff0c;超声波液位检测模块&#xff0c;显示模块&#xff0c;蜂鸣器等器件设备。其中&#xff0c;采用STC89C52单片机作为主控…

【狂热算法篇】探秘图论之Dijkstra 算法:穿越图的迷宫的最短路径力量(通俗易懂版)

羑悻的小杀马特.-CSDN博客羑悻的小杀马特.擅长C/C题海汇总,AI学习,c的不归之路,等方面的知识,羑悻的小杀马特.关注算法,c,c语言,青少年编程领域.https://blog.csdn.net/2401_82648291?typebbshttps://blog.csdn.net/2401_82648291?typebbshttps://blog.csdn.net/2401_8264829…

【Block总结】动态蛇形卷积,专注于细长和弯曲的局部结构|即插即用

论文信息 标题: Dynamic Snake Convolution based on Topological Geometric Constraints for Tubular Structure Segmentation 作者: 戚耀磊、何宇霆、戚晓明、张媛、杨冠羽 会议: 2023 IEEE/CVF International Conference on Computer Vision (ICCV) 发表时间: 2023年10月…

【NEXT】网络编程——上传文件(不限于jpg/png/pdf/txt/doc等),或请求参数值是file类型时,调用在线服务接口

最近在使用华为AI平台ModelArts训练自己的图像识别模型&#xff0c;并部署了在线服务接口。供给客户端&#xff08;如&#xff1a;鸿蒙APP/元服务&#xff09;调用。 import核心能力&#xff1a; import { http } from kit.NetworkKit; import { fileIo } from kit.CoreFileK…

Linux工具使用

1.gcc/g的使用 1.1程序翻译的过程 ①预处理&#xff1a;展开头文件&#xff0c;替换宏&#xff0c;调节编译&#xff0c;去注释。 ②编译&#xff1a;将代码变成汇编语言 ③汇编&#xff1a;将汇编代码变成二进制不可执行的目标文件。 ④链接&#xff1a;将多个我写的多个…

Mac Electron 应用签名(signature)和公证(notarization)

在MacOS 10.14.5之后&#xff0c;如果应用没有在苹果官方平台进行公证notarization(我们可以理解为安装包需要审核&#xff0c;来判断是否存在病毒)&#xff0c;那么就不能被安装。当然现在很多人的解决方案都是使用sudo spctl --master-disable&#xff0c;取消验证模式&#…

单细胞-第五节 多样本数据分析,打分R包AUCell

文件在单细胞\5_GC_py\1_single_cell\3.AUCell.Rmd 1.基因 rm(list = ls()) load("g.Rdata")2.AUCell https://www.ncbi.nlm.nih.gov/pmc/articles/PMC9897923 IF: NA NA NA用这个文章里的方法,将单细胞亚群的marker基因与ros相关基因取交集,用作AUCell的基因集…