很多初学者第一次看到属性定义语法时,都会觉得——
“它看起来像方法,但又不是方法;像字段,但又不是字段。”
所以我们要从语法、功能、用法、规律四个角度,一步步掌握。
一、属性是什么?
在 C# 中,属性(Property) 是封装字段的方法。
属性的核心功能就是“封装字段 + 承载逻辑”。理解并记住这两句结论:
属性是对字段的封装
属性是对字段的逻辑控制
属性是一种特殊的方法。字段式方法。
属性用于封装字段的读取和写入操作。
属性允许在访问字段时自定义逻辑,例如输入验证或计算。
在
set中可以加逻辑,比如校验、触发事件、计算值。在
get中也可以加逻辑,比如动态计算结果,而不是直接返回字段。
二、属性的基础语法结构
完整语法
[修饰符] 数据类型 属性名
{
    get { return 字段名; }// 取值逻辑,必须 return 一个值
    set { 字段名 = value;  }// 赋值逻辑,使用关键字 value 表示传入的数据
}定义属性的语法,本质上是两个方法的语法糖。
属性不是字段:字段是真实存储数据的地方。
属性不是普通方法:虽然写法像方法,但调用方式像字段。
属性是特殊的成员:它其实就是一对方法(get/set)的语法糖,让我们用字段的方式来读写数据。
示例
class Student
{
    private string name;  // 私有字段
    public string Name    // 属性
    {
        get { return name; }   // 取值(像方法)
        set { name = value; }  // 赋值(像方法)
    }
}
Student s = new Student();
s.Name = "小明";  // 调用 set
Console.WriteLine(s.Name); // 调用 getpublic string Name看起来像一个字段,其实是属性的声明。{ get; set; }
这一对看起来像方法体,但它们不是普通方法。get
和set` 是一对访问器(Accessors),用于控制对字段的读取和写入。value是一个上下文关键字,表示外部传进来的值。
最标准的定义语法如下:
<访问修饰符> <类型> <属性名>
{
    get
    {
        // 取值逻辑(读取字段)
        return 字段名;
    }
    set
    {
        // 赋值逻辑(修改字段)
        字段名 = value;
    }
}三、属性命名规则
属性名首字母通常大写(Pascal命名法);
字段名一般小写;
属性名通常与字段名相似,例如:
private int age; public int Age { get { return age; } set { age = value; } }
封装字段的经典写法
public class Student
{
    private string name;   // 字段:用来存数据
    public string Name     // 属性:用来访问字段
    {
        get { return name; }
        set { name = value; }
    }
}👉 说明:
调用演示:
Student s = new Student();
s.Name = "张三";               // 调用 set
Console.WriteLine(s.Name);    // 调用 get带逻辑的属性(常见进阶用法)
你可以在 get / set 中添加业务逻辑,比如检查、限制或格式化。
public class Student
{
    private int age;
    public int Age
    {
        get { return age; }
        set
        {
            if (value < 0)
                value = 0;  // 防止非法值
            age = value;
        }
    }
}
private int age;
public int Age
{
    get { return age; }
    set
    {
        if (value >= 0) age = value;
        else Console.WriteLine("年龄不能为负数!");
    }
}属性与方法的区别
练习:带有验证的属性
class Product
{
    private double price;
    public double Price
    {
        get { return price; }
        set
        {
            if (value < 0) value = 0;
            price = value;
        }
    }
}👉 试试写出对象:
Product p = new Product();
p.Price = -5;    // 自动改为 0
Console.WriteLine(p.Price);总结口诀:
字段存数据,属性管出入。 get取值用,set赋值补。 自动省代码,逻辑写内部。
一、按访问器分类
1. 只读属性 (Read-only Properties)
只有 get,不能 set:外部只能读取,不能修改。
语法
[修饰符] 数据类型 属性名 { get; }
[修饰符] 数据类型 属性名 { 
  get { return _name; }
}示例
//1.在构造函数中初始化的只读属性//
public class Person
{
    // 只读属性 - 只能在构造函数中赋值
    public string Id { get; }
    public string Name { get; }
    public DateTime BirthDate { get; }
    
    public Person(string id, string name, DateTime birthDate)
    {
        Id = id;
        Name = name;
        BirthDate = birthDate;
    }
}
// 使用
var person = new Person("001", "张三", new DateTime(1990, 5, 15));
Console.WriteLine($"ID: {person.Id}");        // 可以读取
Console.WriteLine($"姓名: {person.Name}");     // 可以读取
// person.Name = "李四"; // 错误!不能修改只读属性public string Name {get;}
//2.声明时初始化的只读属性
public class AppConstants
{
    // 声明时直接初始化的只读属性
    public static string AppName { get; } = "我的应用程序";
    public static string Version { get; } = "1.0.0";
    public static DateTime BuildDate { get; } = new DateTime(2024, 1, 1);
}
// 使用
Console.WriteLine($"应用: {AppConstants.AppName}");
Console.WriteLine($"版本: {AppConstants.Version}");
Console.WriteLine($"构建日期: {AppConstants.BuildDate:yyyy-MM-dd}");###
public class Person
{
    // 方式1:只有get访问器
    public string Id { get; }
    
    // 方式2:init访问器(C# 9.0+)
    public string Name { get; init; }
    
    // 方式3:私有set
    public DateTime CreateTime { get; private set; }
    
    public Person(string id, string name)
    {
        Id = id;
        Name = name;
        CreateTime = DateTime.Now;
    }
}2. 只写属性 (Write-only Properties)
public class User
{
    private string _password;
    
    // 只有set访问器
    public string Password
    {
        set { _password = value; }
    }
    
    // 另一种方式:私有get
    public string SecretKey { private get; set; }
}
// 使用
User u1 = new User();
u1.Password = "mySecret123";  // 可以设置密码
// string pwd = u1.Password;  // 错误!不能读取密码
只有 set,不能 get:外部只能赋值,不能取值(很少用)。
语法
public string Password
{
    set { /* 保存密码逻辑 */ }
}3. 读写属性 (Read-Write Properties)
public class Product
{
    // 完整的读写属性
    public string Name { get; set; }
    public decimal Price { get; set; }
}按实现方式分类
4. 自动属性 (Auto-Implemented Properties)
如果不需要在 get 或 set 中写逻辑,没有额外逻辑,只是简单封装字段,可以使用更简洁的“自动属性”语法:
[修饰符] 数据类型 属性名 { get; set; }👉 编译器会自动生成一个隐藏字段。
示例:
public class Student
{
    // 自动属性 - 编译器生成后台字段
    public string Name { get; set; }
    public int Age { get; set; } = 18;  // 带默认值
    public string Email { get; init; }  // 只读自动属性
    
    // 等同于:
    // private string _name;
    // public string Name 
    // { 
    //     get { return _name; } 
    //     set { _name = value; } 
    // }
}
自动属性还可以设置访问限制:
public string Name { get; private set; }  // 只能在类内部改##
5. 完整属性 (Full Properties)
public class BankAccount
{
    private decimal _balance;
    
    // 完整属性 - 手动管理后台字段
    public decimal Balance
    {
        get { return _balance; }
        set 
        { 
            if (value < 0) 
                throw new ArgumentException("余额不能为负");
            _balance = value; 
        }
    }
}6. 计算属性 (Computed Properties)
/*******1.几何图形计算*******/
public class Rectangle
{
    public double Width { get; set; }
    public double Height { get; set; }
    
    // 计算只读属性 - 基于其他属性
    public double Area 
    { 
        get { return Width * Height; } 
    }
    
    public double Perimeter 
    { 
        get { return 2 * (Width + Height); } 
    }
    
    // 使用表达式体的只读属性 (C# 6.0+)
    public bool IsSquare => Width == Height;
    public string Description => $"矩形 {Width}×{Height}";
}说明
var rect = new Rectangle { Width = 5, Height = 3 };
这是 对象初始化器 语法,完全正确。它:
先调用 Rectangle 的构造函数(默认无参构造函数)
然后设置对象的属性
    
 相当于
var rect = new Rectangle();
rect.Width = 5;
rect.Height = 3;使用
// 使用
var rect = new Rectangle { Width = 5, Height = 3 };
Console.WriteLine($"面积: {rect.Area}");           // 15
Console.WriteLine($"周长: {rect.Perimeter}");      // 16
Console.WriteLine($"是否正方形: {rect.IsSquare}"); // False
rect.Width = 3; // 修改宽度会影响面积计算
Console.WriteLine($"修改后面积: {rect.Area}");     // 9
选择建议:
优先使用自动属性
需要验证时使用完整属性
只读计算值使用表达式体属性
考虑使用记录类型简化不可变对象
掌握这些语法可以让你编写出更清晰、更安全的C#代码!
练习题
以下是 10 道专门练习属性定义语法的题目,涵盖自动属性、完整属性、只读属性、计算属性和业务逻辑控制:
练习题 1:基础自动属性
题目:创建一个 Student 类,使用自动属性定义以下属性:
学号 (StudentId)
姓名 (Name)
年龄 (Age)
班级 (ClassName)
要求:所有属性都使用自动属性语法,并设置合理的默认值。
// 参考答案
public class Student
{
    public string StudentId { get; set; } = "未知";
    public string Name { get; set; } = "未知";
    public int Age { get; set; } = 0;
    public string ClassName { get; set; } = "未分配";
}练习题 2:完整属性与数据验证
题目:创建一个 BankAccount 类,使用完整属性定义以下属性:
账户号 (AccountNumber) - 只读属性
余额 (Balance) - 私有 set,只能通过方法修改
账户持有人 (AccountHolder)
要求:
AccountNumber 只能在构造函数中设置
Balance 不能为负数
AccountHolder 不能为空或空白字符串
// 参考答案
public class BankAccount
{
    private string _accountNumber;
    private decimal _balance;
    private string _accountHolder;
    public string AccountNumber
    {
        get { return _accountNumber; }
    }
    public decimal Balance
    {
        get { return _balance; }
        private set 
        { 
            if (value < 0)
                throw new ArgumentException("余额不能为负数");
            _balance = value; 
        }
    }
    public string AccountHolder
    {
        get { return _accountHolder; }
        set 
        { 
            if (string.IsNullOrWhiteSpace(value))
                throw new ArgumentException("账户持有人不能为空");
            _accountHolder = value; 
        }
    }
    public BankAccount(string accountNumber, string accountHolder)
    {
        _accountNumber = accountNumber;
        AccountHolder = accountHolder;
        _balance = 0;
    }
    public void Deposit(decimal amount)
    {
        if (amount <= 0)
            throw new ArgumentException("存款金额必须大于0");
        Balance += amount;
    }
    public bool Withdraw(decimal amount)
    {
        if (amount <= 0 || amount > Balance)
            return false;
        Balance -= amount;
        return true;
    }
}练习题 3:计算属性
题目:创建一个 Rectangle 类,包含以下属性:
宽度 (Width)
高度 (Height)
面积 (Area) - 计算属性
周长 (Perimeter) - 计算属性
是否正方形 (IsSquare) - 计算属性
要求:所有计算属性使用表达式体语法。
// 参考答案
public class Rectangle
{
    public double Width { get; set; }
    public double Height { get; set; }
    // 计算属性
    public double Area => Width * Height;
    public double Perimeter => 2 * (Width + Height);
    public bool IsSquare => Width == Height;
    public string Description => $"矩形 {Width}×{Height} ({(IsSquare ? "正方形" : "长方形")})";
}练习题 4:只读属性与构造函数
题目:创建一个 Product 类,包含以下属性:
产品ID (ProductId) - 只读属性
产品名称 (ProductName)
价格 (Price)
创建时间 (CreatedDate) - 只读属性
是否上架 (IsActive)
要求:
ProductId 和 CreatedDate 只能在构造函数中初始化
价格不能为负数
// 参考答案
public class Product
{
    public string ProductId { get; }
    public DateTime CreatedDate { get; }
    
    private string _productName;
    private decimal _price;
    
    public string ProductName
    {
        get => _productName;
        set => _productName = !string.IsNullOrWhiteSpace(value) ? value : "未知产品";
    }
    
    public decimal Price
    {
        get => _price;
        set => _price = value >= 0 ? value : 0;
    }
    
    public bool IsActive { get; set; } = true;
    
    public Product(string productId, string productName)
    {
        ProductId = productId;
        ProductName = productName;
        CreatedDate = DateTime.Now;
    }
}练习题 5:属性中的复杂业务逻辑
题目:创建一个 Temperature 类,包含以下属性:
摄氏温度 (Celsius)
华氏温度 (Fahrenheit)
要求:
两个属性相互关联,修改一个会影响另一个
温度范围限制在绝对零度(-273.15°C)以上
// 参考答案
public class Temperature
{
    private double _celsius;
    public double Celsius
    {
        get => _celsius;
        set
        {
            if (value < -273.15)
                throw new ArgumentException("温度不能低于绝对零度(-273.15°C)");
            _celsius = value;
        }
    }
    public double Fahrenheit
    {
        get => _celsius * 9 / 5 + 32;
        set
        {
            double celsius = (value - 32) * 5 / 9;
            if (celsius < -273.15)
                throw new ArgumentException("温度不能低于绝对零度(-459.67°F)");
            _celsius = celsius;
        }
    }
    public string Description
    {
        get
        {
            if (_celsius < 0) return "寒冷";
            if (_celsius < 15) return "凉爽";
            if (_celsius < 25) return "舒适";
            if (_celsius < 35) return "温暖";
            return "炎热";
        }
    }
}练习题 6:只写属性
题目:创建一个 Logger 类,包含以下属性:
日志文件路径 (LogFilePath) - 只写属性
日志级别 (LogLevel) - 只写属性
快速日志 (QuickLog) - 只写属性,用于快速记录消息
要求:
所有属性都是只写的
提供方法来读取配置和日志内容
// 参考答案
public class Logger
{
    private string _logFilePath;
    private string _logLevel = "INFO";
    private readonly List<string> _logMessages = new List<string>();
    public string LogFilePath
    {
        set
        {
            if (string.IsNullOrWhiteSpace(value))
                throw new ArgumentException("日志文件路径不能为空");
            _logFilePath = value;
        }
    }
    public string LogLevel
    {
        set
        {
            string[] validLevels = { "DEBUG", "INFO", "WARN", "ERROR" };
            if (validLevels.Contains(value.ToUpper()))
                _logLevel = value.ToUpper();
            else
                throw new ArgumentException("无效的日志级别");
        }
    }
    public string QuickLog
    {
        set
        {
            string message = $"[{_logLevel}] {DateTime.Now:HH:mm:ss} - {value}";
            _logMessages.Add(message);
        }
    }
    public string GetCurrentConfig()
    {
        return $"日志文件: {_logFilePath}, 级别: {_logLevel}";
    }
    public string[] GetLogMessages()
    {
        return _logMessages.ToArray();
    }
}练习题 7:静态属性
题目:创建一个 AppConfig 类,包含以下静态属性:
应用名称 (AppName)
版本号 (Version)
用户计数 (UserCount)
要求:
AppName 和 Version 是只读的
UserCount 只能在类内部修改
提供方法来增加用户计数
// 参考答案
public class AppConfig
{
    public static string AppName { get; } = "我的应用程序";
    public static string Version { get; } = "1.0.0";
    public static int UserCount { get; private set; } = 0;
    static AppConfig()
    {
        Console.WriteLine($"应用程序初始化: {AppName} v{Version}");
    }
    public static void AddUser()
    {
        UserCount++;
        Console.WriteLine($"用户数量: {UserCount}");
    }
    public static void RemoveUser()
    {
        if (UserCount > 0)
            UserCount--;
        Console.WriteLine($"用户数量: {UserCount}");
    }
    public static string GetAppInfo()
    {
        return $"{AppName} v{Version} - 当前用户: {UserCount}";
    }
}练习题 8:属性访问级别控制
题目:创建一个 Employee 类,包含以下属性:
员工ID (EmployeeId) - 公共获取,私有设置
姓名 (Name) - 公共获取,内部设置
工资 (Salary) - 私有获取和设置
部门 (Department) - 公共获取和设置
要求:
提供适当的方法来访问和修改受限属性
工资只能通过特定方法修改,并有验证逻辑
// 参考答案
public class Employee
{
    public string EmployeeId { get; private set; }
    public string Name { get; internal set; }
    private decimal _salary;
    public string Department { get; set; }
    public Employee(string employeeId, string name, string department)
    {
        EmployeeId = employeeId;
        Name = name;
        Department = department;
        _salary = 0;
    }
    // 通过方法访问私有属性
    public decimal GetSalary() => _salary;
    public void SetSalary(decimal newSalary)
    {
        if (newSalary < 0)
            throw new ArgumentException("工资不能为负数");
        
        decimal oldSalary = _salary;
        _salary = newSalary;
        
        Console.WriteLine($"工资从 {oldSalary:C} 调整为 {newSalary:C}");
    }
    public void GiveRaise(decimal amount)
    {
        if (amount <= 0)
            throw new ArgumentException("加薪金额必须大于0");
        
        SetSalary(_salary + amount);
    }
    public string GetEmployeeInfo()
    {
        return $"员工 {Name} ({EmployeeId}) - 部门: {Department}, 工资: {_salary:C}";
    }
}练习题 9:计算属性与状态属性
题目:创建一个 Order 类,包含以下属性:
订单号 (OrderId) - 只读
订单项列表 (Items) - 私有 set
订单总额 (TotalAmount) - 计算属性
是否已支付 (IsPaid)
订单状态 (OrderStatus) - 基于其他属性的计算属性
要求:
订单状态根据是否支付和是否有商品自动计算
提供添加商品和支付订单的方法
// 参考答案
public class Order
{
    public string OrderId { get; }
    public List<OrderItem> Items { get; private set; }
    public bool IsPaid { get; private set; }
    // 计算属性
    public decimal TotalAmount => Items.Sum(item => item.Price * item.Quantity);
    public bool IsEmpty => Items.Count == 0;
    public string OrderStatus
    {
        get
        {
            if (IsEmpty) return "空订单";
            if (IsPaid) return "已支付";
            return "待支付";
        }
    }
    public Order(string orderId)
    {
        OrderId = orderId;
        Items = new List<OrderItem>();
        IsPaid = false;
    }
    public void AddItem(string productName, decimal price, int quantity = 1)
    {
        if (IsPaid)
            throw new InvalidOperationException("订单已支付,不能添加商品");
        Items.Add(new OrderItem
        {
            ProductName = productName,
            Price = price,
            Quantity = quantity
        });
    }
    public void ProcessPayment()
    {
        if (IsEmpty)
            throw new InvalidOperationException("空订单不能支付");
        IsPaid = true;
        Console.WriteLine($"订单 {OrderId} 支付成功,金额: {TotalAmount:C}");
    }
    public string GetOrderSummary()
    {
        return $"订单 {OrderId} - {OrderStatus} - 总额: {TotalAmount:C} - 商品数: {Items.Count}";
    }
}
public class OrderItem
{
    public string ProductName { get; set; }
    public decimal Price { get; set; }
    public int Quantity { get; set; }
}练习题 10:综合练习 - 完整的用户管理系统
题目:创建一个 User 类,综合运用各种属性类型:
用户ID (UserId) - 只读
用户名 (Username) - 带验证的完整属性
邮箱 (Email) - 带验证的完整属性
密码 (Password) - 只写属性
创建时间 (CreatedAt) - 只读
最后登录时间 (LastLoginAt)
是否是管理员 (IsAdmin) - 计算属性
要求:
用户名和邮箱有格式验证
密码只写,立即进行哈希处理
是否是管理员基于邮箱域名判断
// 参考答案
public class User
{
    public string UserId { get; }
    public DateTime CreatedAt { get; }
    public DateTime? LastLoginAt { get; private set; }
    private string _username;
    private string _email;
    private string _passwordHash;
    public string Username
    {
        get => _username;
        set
        {
            if (string.IsNullOrWhiteSpace(value))
                throw new ArgumentException("用户名不能为空");
            if (value.Length < 3)
                throw new ArgumentException("用户名至少3个字符");
            if (value.Length > 20)
                throw new ArgumentException("用户名最多20个字符");
            
            _username = value.Trim();
        }
    }
    public string Email
    {
        get => _email;
        set
        {
            if (string.IsNullOrWhiteSpace(value))
                throw new ArgumentException("邮箱不能为空");
            if (!value.Contains("@") || !value.Contains("."))
                throw new ArgumentException("邮箱格式不正确");
            
            _email = value.Trim().ToLower();
        }
    }
    public string Password
    {
        set
        {
            if (string.IsNullOrWhiteSpace(value) || value.Length < 6)
                throw new ArgumentException("密码至少6个字符");
            
            _passwordHash = HashPassword(value);
            Console.WriteLine("密码已设置并哈希处理");
        }
    }
    // 计算属性
    public bool IsAdmin => _email.EndsWith("@admin.com");
    public bool IsEmailVerified => !string.IsNullOrEmpty(_email);
    public string UserLevel
    {
        get
        {
            if (IsAdmin) return "管理员";
            if (_email.EndsWith("@vip.com")) return "VIP用户";
            return "普通用户";
        }
    }
    public User(string userId, string username, string email)
    {
        UserId = userId;
        Username = username;
        Email = email;
        CreatedAt = DateTime.Now;
    }
    public bool VerifyPassword(string password)
    {
        return HashPassword(password) == _passwordHash;
    }
    public void RecordLogin()
    {
        LastLoginAt = DateTime.Now;
        Console.WriteLine($"用户 {Username} 于 {LastLoginAt:yyyy-MM-dd HH:mm:ss} 登录");
    }
    public string GetUserInfo()
    {
        return $"用户 {Username} ({UserId}) - 等级: {UserLevel} - 注册于: {CreatedAt:yyyy-MM-dd}";
    }
    private string HashPassword(string password)
    {
        // 简单的哈希示例
        using var sha256 = System.Security.Cryptography.SHA256.Create();
        var bytes = System.Text.Encoding.UTF8.GetBytes(password + UserId); // 加盐
        var hash = sha256.ComputeHash(bytes);
        return Convert.ToBase64String(hash);
    }
}题目11:定义 Book 类
using System;
public class Book
{
    // 书名属性 - 使用自动属性
    public string Title { get; set; }
    
    // 价格属性 - 使用完整属性添加验证逻辑
    private decimal _price;
    public decimal Price
    {
        get { return _price; }
        set 
        { 
            // 保证价格不能小于0
            if (value < 0)
            {
                _price = 0;
                Console.WriteLine("警告:价格不能为负数,已自动设置为0");
            }
            else
            {
                _price = value;
            }
        }
    }
}
class Program
{
    static void Main()
    {
        // 创建第一个 Book 对象
        Book book1 = new Book();
        book1.Title = "C#编程入门";
        book1.Price = 59.90m;
        
        // 创建第二个 Book 对象
        Book book2 = new Book();
        book2.Title = "数据结构与算法";
        book2.Price = -29.99m; // 测试负数价格
        
        // 输出书名和价格
        Console.WriteLine("=== 图书信息 ===");
        Console.WriteLine($"书名: {book1.Title}, 价格: {book1.Price:C}");
        Console.WriteLine($"书名: {book2.Title}, 价格: {book2.Price:C}");
    }
}输出结果:
警告:价格不能为负数,已自动设置为0
=== 图书信息 ===
书名: C#编程入门, 价格: ¥59.90
书名: 数据结构与算法, 价格: ¥0.00题目12:更完善的版本(包含构造函数)
using System;
public class Book
{
    // 书名属性
    public string Title { get; set; }
    
    // 价格属性 - 带验证逻辑
    private decimal _price;
    public decimal Price
    {
        get { return _price; }
        set 
        { 
            // 保证价格不能小于0
            if (value < 0)
            {
                throw new ArgumentException("价格不能为负数");
            }
            _price = value;
        }
    }
    
    // 默认构造函数
    public Book()
    {
        Title = "未知书名";
        _price = 0;
    }
    
    // 带参数的构造函数
    public Book(string title, decimal price)
    {
        Title = title;
        Price = price; // 使用属性赋值,会触发验证逻辑
    }
    
    // 显示图书信息的方法
    public void DisplayInfo()
    {
        Console.WriteLine($"书名: {Title}, 价格: {Price:C}");
    }
}
class Program
{
    static void Main()
    {
        try
        {
            // 创建第一个 Book 对象(使用构造函数)
            Book book1 = new Book("C#编程入门", 59.90m);
            
            // 创建第二个 Book 对象(使用默认构造函数+属性赋值)
            Book book2 = new Book();
            book2.Title = "数据结构与算法";
            book2.Price = 79.80m;
            
            // 创建第三个 Book 对象(测试负数价格)
            Book book3 = new Book("测试图书", -10.00m); // 这会抛出异常
            
            // 输出书名和价格
            Console.WriteLine("=== 图书信息 ===");
            book1.DisplayInfo();
            book2.DisplayInfo();
            book3.DisplayInfo();
        }
        catch (ArgumentException ex)
        {
            Console.WriteLine($"错误: {ex.Message}");
        }
    }
}题目13:使用对象初始化器的版本
using System;
public class Book
{
    // 书名属性
    public string Title { get; set; } = "未知书名";
    
    // 价格属性 - 带验证逻辑
    private decimal _price;
    public decimal Price
    {
        get { return _price; }
        set 
        { 
            // 保证价格不能小于0
            if (value < 0)
            {
                _price = 0;
                Console.WriteLine($"警告:'{Title}' 的价格不能为负数,已自动设置为0");
            }
            else
            {
                _price = value;
            }
        }
    }
    
    // 显示图书信息的方法
    public void DisplayInfo()
    {
        Console.WriteLine($"书名: {Title}, 价格: {Price:C}");
    }
}
class Program
{
    static void Main()
    {
        // 使用对象初始化器创建 Book 对象
        Book book1 = new Book 
        { 
            Title = "C#编程入门", 
            Price = 59.90m 
        };
        
        Book book2 = new Book 
        { 
            Title = "数据结构与算法", 
            Price = -29.99m  // 测试负数价格
        };
        
        Book book3 = new Book 
        { 
            Title = "ASP.NET Core开发", 
            Price = 89.00m 
        };
        
        // 输出书名和价格
        Console.WriteLine("=== 图书信息 ===");
        book1.DisplayInfo();
        book2.DisplayInfo();
        book3.DisplayInfo();
        
        // 测试修改价格
        Console.WriteLine("\n=== 修改价格测试 ===");
        book1.Price = 49.90m;  // 正常修改
        book2.Price = -15.00m; // 再次测试负数价格
        
        Console.WriteLine("修改后的价格:");
        book1.DisplayInfo();
        book2.DisplayInfo();
    }
}输出结果:
警告:'数据结构与算法' 的价格不能为负数,已自动设置为0
=== 图书信息 ===
书名: C#编程入门, 价格: ¥59.90
书名: 数据结构与算法, 价格: ¥0.00
书名: ASP.NET Core开发, 价格: ¥89.00
=== 修改价格测试 ===
警告:'数据结构与算法' 的价格不能为负数,已自动设置为0
修改后的价格:
书名: C#编程入门, 价格: ¥49.90
书名: 数据结构与算法, 价格: ¥0.00练习要点总结
通过这些练习,你应该掌握:
自动属性语法
public string Name { get; set; } = "默认值";完整属性语法
private string _name;
public string Name
{
    get { return _name; }
    set { _name = value; }
}只读属性
public string Id { get; }  // 构造函数中赋值
public double Area => Width * Height;  // 计算属性只写属性
public string Password
{
    set { _passwordHash = HashPassword(value); }
}属性验证逻辑
set 
{
    if (value < 0)
        throw new ArgumentException("不能为负数");
    _field = value;
}访问级别控制
public decimal Salary { get; private set; }
public string Name { get; internal set; }静态属性
public static int Count { get; private set; }属性验证的重要性
保证数据的有效性
防止非法状态
提高代码的健壮性
提供清晰的错误信息
自动属性语法
public string Title { get; set; }完整属性语法(带验证逻辑)
private decimal _price;
public decimal Price
{
    get { return _price; }
    set 
    { 
        if (value < 0)
        {
            _price = 0;
            // 或者抛出异常:throw new ArgumentException("价格不能为负数");
        }
        else
        {
            _price = value;
        }
    }
}对象创建方式
// 方式1:使用构造函数
Book book1 = new Book("书名", 价格);
// 方式2:使用默认构造函数+属性赋值
Book book2 = new Book();
book2.Title = "书名";
book2.Price = 价格;
// 方式3:使用对象初始化器
Book book3 = new Book { Title = "书名", Price = 价格 };