作为一名从护理行业转行到编程领域的学习者,我发现很多编程概念都可以通过医护工作经验来理解。本文将结合医院信息系统的实际场景,详细讲解 C#中的几个重要概念。
1. 命名空间
命名空间是一种组织和管理代码的方式,就像医院的组织架构一样。比如医院分为内科、外科、护理部等不同部门,每个部门都有自己的职责和管理范围。在 C#中,命名空间也起到类似的作用,它可以:
- 避免命名冲突(就像医院里不同科室可以有相同名字的护士)
- 提供逻辑分组(像医院的科室划分)
- 控制代码的访问范围(类似于医院的权限管理)
1.1 命名空间的组织结构
namespace HospitalSystem // 整个医院系统
{
namespace Administration // 行政管理
{
// 人事管理、财务管理等类
}
namespace Clinical // 临床医疗
{
namespace Internal // 内科
{
// 内科相关类
}
namespace Surgery // 外科
{
// 外科相关类
}
}
namespace Nursing // 护理部门
{
namespace WardManagement // 病房管理
{
// 病房相关的类
}
namespace MedicationManagement // 用药管理
{
// 用药相关的类
}
}
}
1.2 命名空间的使用方法
- 可以认为类是属于命名空间的
- 如果在当前项目中没有这个类的命名空间,需要我们手动导入这个类所在的命名空间:
- 使用鼠标点击 Visual Studio 提示
- 使用快捷键 alt+shift+F10
- 记住常用命名空间,手动输入
1.3 在项目中使用命名空间
// 方法1:使用using导入
using HospitalSystem.Nursing;
using HospitalSystem.Pharmacy;
// 方法2:使用完整限定名称
HospitalSystem.Nursing.NursingRecord record = new HospitalSystem.Nursing.NursingRecord();
// 方法3:使用别名
using NurseRecord = HospitalSystem.Nursing.NursingRecord;
1.4 项目引用示例
在一个项目中引用另一个类:
- 添加引用(如添加 HospitalCore.dll)
- 引用命名空间
using HospitalCore.Models;
using HospitalCore.Services;
2. 值类型和引用类型
2.1 值类型详解
值类型直接存储数据本身,包括:
整数类型
- sbyte: 8 位有符号整数 (-128 到 127)
- byte: 8 位无符号整数 (0 到 255)
- short: 16 位有符号整数
- ushort: 16 位无符号整数
- int: 32 位有符号整数(最常用)
- uint: 32 位无符号整数
- long: 64 位有符号整数
- ulong: 64 位无符号整数
浮点类型
- float: 32 位单精度浮点数(精确到 6-9 位)
- double: 64 位双精度浮点数(精确到 15-17 位)
- decimal: 128 位高精度小数(常用于财务计算)
其他值类型
- bool: 布尔值(true/false)
- char: 16 位 Unicode 字符
- enum: 枚举类型
- struct: 结构体
在医疗系统中的应用示例:
public struct VitalSigns
{
public int HeartRate; // 心率,范围通常60-100
public decimal Temperature; // 体温,精确到小数点后一位,如36.5
public int BloodPressureHigh; // 收缩压,如120
public int BloodPressureLow; // 舒张压,如80
public bool IsFever; // 是否发烧
// 枚举示例
public enum TemperatureMethod
{
Oral, // 口温
Axillary, // 腋温
Rectal // 肛温
}
public TemperatureMethod Method { get; set; }
// 值类型的数据验证方法
public bool IsNormal()
{
return HeartRate >= 60 && HeartRate <= 100
&& Temperature >= 36.0m && Temperature <= 37.2m
&& BloodPressureHigh >= 90 && BloodPressureHigh <= 140
&& BloodPressureLow >= 60 && BloodPressureLow <= 90;
}
}
2.2 引用类型详解
引用类型包括:
类(class)
- 所有自定义类
- 系统预定义类(如 String, Object 等)
接口(interface)
委托(delegate)
数组(array)
- 无论元素是值类型还是引用类型,数组本身都是引用类型
字符串(string)
- 虽然经常使用,但 string 是引用类型
- 具有特殊的不可变性质
医疗系统示例:
public class PatientRecord
{
// 基本信息
public string PatientName { get; set; }
public string IdNumber { get; set; }
public DateTime DateOfBirth { get; set; }
// 诊断信息(数组示例)
public string[] Diagnoses { get; set; }
// 用药信息(集合示例)
public List<Medication> Medications { get; set; }
// 生命体征记录(自定义类示例)
public List<VitalSigns> VitalSignsHistory { get; set; }
// 委托示例 - 用于生命体征异常通知
public delegate void VitalSignsAlertHandler(string message);
public event VitalSignsAlertHandler OnVitalSignsAlert;
// 深拷贝示例
public PatientRecord Clone()
{
var newRecord = new PatientRecord
{
PatientName = this.PatientName,
IdNumber = this.IdNumber,
DateOfBirth = this.DateOfBirth,
Diagnoses = (string[])this.Diagnoses.Clone(),
Medications = this.Medications.Select(m => m.Clone()).ToList(),
VitalSignsHistory = this.VitalSignsHistory.Select(vs => vs.Clone()).ToList()
};
return newRecord;
}
}
public class Medication
{
public string Name { get; set; }
public double Dosage { get; set; }
// 深拷贝方法
public Medication Clone()
{
return new Medication
{
Name = this.Name,
Dosage = this.Dosage
};
}
}
public class VitalSigns
{
public decimal Temperature { get; set; }
public int HeartRate { get; set; }
// 深拷贝方法
public VitalSigns Clone()
{
return new VitalSigns
{
Temperature = this.Temperature,
HeartRate = this.HeartRate
};
}
}
2.3 内存存储区别详解
在 C#中,内存分为栈(Stack)和堆(Heap)两个主要区域:
栈(Stack)
- 存储值类型的数据
- 由系统自动管理内存的分配和释放
- 访问速度快
- 空间有限
- 按后进先出(LIFO)的顺序存储
堆(Heap)
- 存储引用类型的实际数据
- 需要垃圾回收器(GC)管理内存
- 空间较大但访问速度相对较慢
- 内存分配和回收更灵活
示例代码:
public class MemoryExample
{
public void DemonstrateMemoryUsage()
{
// 值类型示例
int temperature = 37; // 直接在栈上分配4字节
bool isCritical = true; // 直接在栈上分配1字节
DateTime checkTime = DateTime.Now; // 虽然DateTime是struct,但仍在栈上分配
// 引用类型示例
string patientName = "张三"; // 在堆上分配字符串数据,栈上存储引用
PatientRecord record = new PatientRecord(); // 对象在堆上,引用在栈上
// 值类型复制示例
int temp2 = temperature; // 在栈上创建新的独立副本
temp2 = 38; // 修改temp2不影响temperature
Console.WriteLine($"原温度: {temperature}, 新温度: {temp2}"); // 37, 38
// 引用类型复制示例
PatientRecord record2 = record; // 复制引用,两个变量指向同一个堆上的对象
record2.PatientName = "李四"; // 通过record2修改会影响record
Console.WriteLine($"record的病人: {record.PatientName}"); // 输出"李四"
// 字符串特例示例
string str1 = "测试";
string str2 = "测试"; // str2和str1指向同一个字符串对象(字符串池)
string str3 = new string(new char[] { '测', '试' }); // 强制创建新对象
// 数组示例
int[] temperatures = new int[] { 36, 37, 38 }; // 数组对象在堆上,元素在数组对象内连续存储
int[] temps2 = temperatures; // 复制引用
temps2[0] = 39; // 修改temps2同时影响temperatures
}
// 值类型作为参数
public void UpdateTemperature(int temp)
{
temp = 39; // 不会影响原始值
}
// 引用类型作为参数
public void UpdatePatient(PatientRecord patient)
{
patient.PatientName = "王五"; // 会影响原始对象
}
}
3. 字符串处理
字符串在医疗信息系统中使用非常频繁,比如病历记录、医嘱记录等。C#提供了丰富的字符串处理方法:
3.1 字符串的特性
- 不可变性:字符串是不可变的,每次修改都会创建新的字符串对象
- 字符串池:相同的字符串字面量会共享同一个对象
- 可以看作是 char 类型的只读数组
3.2 常用字符串方法详解
public class NursingNoteProcessor
{
public void ProcessNursingNotes()
{
// 护理记录示例
string note = " 患者张三,男,62岁。\n" +
"警告:对青霉素过敏!\n" +
" 体温37.2℃,血压120/80mmHg ";
// 1. 基本字符串操作
Console.WriteLine($"记录长度: {note.Length}"); // 获取字符串长度
// 2. 空白处理
string trimmed = note.Trim(); // 去除两端空格
string trimStart = note.TrimStart(); // 去除开头空格
string trimEnd = note.TrimEnd(); // 去除结尾空格
// 3. 大小写转换
string upper = note.ToUpper(); // 转大写(用于重要警告)
string lower = note.ToLower(); // 转小写
// 4. 查找操作
bool hasAllergy = note.Contains("过敏"); // 检查是否包含某字符串
int allergyIndex = note.IndexOf("过敏"); // 查找第一次出现的位置
int lastIndex = note.LastIndexOf(","); // 查找最后一次出现的位置
// 5. 替换操作
string replaced = note.Replace("警告", "【警告】");
string noSpaces = note.Replace(" ", "");
// 6. 字符串分割
string[] lines = note.Split(new[] { "\n" }, StringSplitOptions.RemoveEmptyEntries);
foreach (string line in lines)
{
Console.WriteLine($"行内容: {line.Trim()}");
}
// 7. 子串提取
int ageStart = note.IndexOf(",") + 1;
int ageEnd = note.IndexOf("岁");
string age = note.Substring(ageStart, ageEnd - ageStart);
Console.WriteLine($"年龄: {age}"); // 输出: 62
// 8. 字符串比较
bool isEqual = note.Equals("其他记录", StringComparison.OrdinalIgnoreCase); // 忽略大小写比较
bool startsWith = note.StartsWith("患者"); // 检查开头
bool endsWith = note.EndsWith("mmHg"); // 检查结尾
// 9. 字符串拼接
string[] vitals = { "体温: 37.2℃", "血压: 120/80mmHg", "心率: 75次/分" };
string summary = string.Join(", ", vitals);
Console.WriteLine($"生命体征: {summary}");
// 10. 字符串构建(处理大量字符串拼接)
StringBuilder noteBuilder = new StringBuilder();
noteBuilder.AppendLine("患者基本信息:");
noteBuilder.AppendLine($"姓名:张三");
noteBuilder.AppendLine($"年龄:{age}岁");
noteBuilder.AppendLine($"生命体征:{summary}");
string finalNote = noteBuilder.ToString();
}
}
4. 继承
继承是面向对象编程的核心概念之一,在医院管理系统中有很多应用场景。
4.1 继承的基本概念
继承的本质:
- 代码重用
- 建立类之间的父子关系
- 实现多态性
继承的特性:
- 单根性:一个类只能有一个直接父类
- 传递性:子类继承父类的所有特性,包括父类从其父类继承的特性
- 子类可以扩展父类的功能
- 子类可以重写父类的方法
4.2 医院员工继承体系示例
// 基类:医院员工
public abstract class HospitalEmployee
{
protected string id;
protected string name;
public string Department { get; set; }
public DateTime HireDate { get; set; }
// 构造函数
public HospitalEmployee(string id, string name)
{
this.id = id;
this.name = name;
}
// 虚方法 - 允许子类重写
public virtual string GetDutyDescription()
{
return $"{name}在{Department}工作";
}
// 抽象方法 - 子类必须实现
public abstract decimal CalculateSalary();
}
// 护士类
public class Nurse : HospitalEmployee
{
public string NursingLevel { get; set; }
public string Shift { get; set; }
private decimal baseSalary;
public Nurse(string id, string name, string level)
: base(id, name)
{
NursingLevel = level;
SetBaseSalary();
}
private void SetBaseSalary()
{
switch (NursingLevel)
{
case "初级": baseSalary = 5000M; break;
case "中级": baseSalary = 7000M; break;
case "高级": baseSalary = 9000M; break;
default: baseSalary = 4000M; break;
}
}
public override string GetDutyDescription()
{
return $"{name}是{Department}的{NursingLevel}级护士,{Shift}班次";
}
public override decimal CalculateSalary()
{
decimal shiftBonus = Shift == "夜班" ? 1000M : 0M;
return baseSalary + shiftBonus;
}
}
// 主管护士类
public class HeadNurse : Nurse
{
public List<Nurse> TeamMembers { get; set; }
private const decimal MANAGEMENT_BONUS = 2000M;
public HeadNurse(string id, string name)
: base(id, name, "主管")
{
TeamMembers = new List<Nurse>();
}
public void AddTeamMember(Nurse nurse)
{
TeamMembers.Add(nurse);
}
public override decimal CalculateSalary()
{
return base.CalculateSalary() + MANAGEMENT_BONUS;
}
// 新增的管理方法
public string GenerateTeamReport()
{
StringBuilder report = new StringBuilder();
report.AppendLine($"{Department}护理团队报告");
report.AppendLine($"主管护士:{name}");
report.AppendLine($"团队成员:{TeamMembers.Count}人");
foreach (var nurse in TeamMembers)
{
report.AppendLine($"- {nurse.GetDutyDescription()}");
}
return report.ToString();
}
}
5. 集合
在医院信息系统中,我们经常需要处理多个数据项的集合,比如病人列表、药品库存等。C#提供了多种集合类型供我们使用。
5.1 ArrayList 详解
ArrayList 是一个非泛型集合,可以存储任意类型的对象。
特点:
- 动态大小:自动扩展容量
- 可以存储不同类型的数据
- 需要类型转换,性能较差
- 不建议在新代码中使用,推荐使用泛型 List
public class PatientListManager
{
private ArrayList patientList = new ArrayList();
public void DemonstrateArrayList()
{
// 添加不同类型的数据
patientList.Add("张三"); // 字符串
patientList.Add(new Patient()); // 患者对象
patientList.Add(42); // 数字
// 容量和数量
Console.WriteLine($"容量: {patientList.Capacity}"); // 默认4,按需翻倍增长
Console.WriteLine($"实际数量: {patientList.Count}");
// 插入元素
patientList.Insert(0, "急诊病人");
// 检查包含
bool hasPatient = patientList.Contains("张三");
// 遍历(需要类型转换)
foreach (object item in patientList)
{
if (item is Patient patient)
{
Console.WriteLine($"病人: {patient.Name}");
}
}
// 删除操作
patientList.Remove("张三"); // 删除特定元素
patientList.RemoveAt(0); // 删除指定位置的元素
patientList.Clear(); // 清空集合
}
}
5.2 Hashtable 与 Dictionary 对比
键值对集合在医疗系统中很常用,比如药品库存管理。
Hashtable(非泛型):
- 键和值都是 object 类型
- 需要类型转换
- 性能较差
- 不建议在新代码中使用
Dictionary<TKey, TValue>(泛型):
- 类型安全
- 性能更好
- 推荐使用
public class MedicineInventoryManager
{
// Hashtable示例(旧方式)
private Hashtable medicineStockOld = new Hashtable();
// Dictionary示例(推荐方式)
private Dictionary<string, int> medicineStock = new Dictionary<string, int>();
public void CompareCollections()
{
// Hashtable操作
medicineStockOld.Add("阿莫西林", 100);
medicineStockOld["布洛芬"] = 50;
// 类型转换问题示例
int oldStock = (int)medicineStockOld["阿莫西林"]; // 需要显式转换
// Dictionary操作
medicineStock.Add("阿莫西林", 100);
medicineStock["布洛芬"] = 50;
// 安全的值获取
if (medicineStock.TryGetValue("阿莫西林", out int stock))
{
Console.WriteLine($"阿莫西林库存: {stock}");
}
// 检查键是否存在
if (medicineStock.ContainsKey("布洛芬"))
{
medicineStock["布洛芬"] -= 10; // 减少库存
}
// 遍历比较
foreach (DictionaryEntry entry in medicineStockOld)
{
Console.WriteLine($"药品: {entry.Key}, 库存: {entry.Value}");
}
foreach (KeyValuePair<string, int> kvp in medicineStock)
{
Console.WriteLine($"药品: {kvp.Key}, 库存: {kvp.Value}");
}
// 仅遍历键或值
foreach (string medicine in medicineStock.Keys)
{
Console.WriteLine($"药品: {medicine}");
}
// 转换为列表
List<string> medicineList = medicineStock.Keys.ToList();
}
}
5.3 List详解
List
public class NursingScheduleManager
{
private List<Nurse> nurses = new List<Nurse>();
public void DemonstrateList()
{
// 添加元素
nurses.Add(new Nurse("N001", "张护士", "初级"));
nurses.Add(new Nurse("N002", "李护士", "中级"));
// 批量添加
var newNurses = new List<Nurse>
{
new Nurse("N003", "王护士", "高级"),
new Nurse("N004", "赵护士", "中级")
};
nurses.AddRange(newNurses);
// 查找操作
Nurse foundNurse = nurses.Find(n => n.NursingLevel == "中级");
List<Nurse> juniorNurses = nurses.FindAll(n => n.NursingLevel == "初级");
// 排序
nurses.Sort((n1, n2) => n1.NursingLevel.CompareTo(n2.NursingLevel));
// LINQ操作
var dayShiftNurses = nurses.Where(n => n.Shift == "白班").ToList();
var nurseCount = nurses.Count(n => n.NursingLevel == "中级");
var orderedNurses = nurses.OrderBy(n => n.Name).ToList();
// 转换
var nurseNames = nurses.Select(n => n.Name).ToList();
// 检查条件
bool hasNightNurse = nurses.Any(n => n.Shift == "夜班");
bool allJunior = nurses.All(n => n.NursingLevel == "初级");
// 删除操作
nurses.Remove(foundNurse);
nurses.RemoveAll(n => n.NursingLevel == "实习");
// 范围操作
var someNurses = nurses.GetRange(0, 2); // 获取前两个护士
nurses.RemoveRange(0, 2); // 删除前两个护士
}
}
6. 类型转换详解
在医院信息系统中,类型转换经常用于处理不同类型的医护人员和病历记录。
6.1 类型转换的基本概念
- 隐式转换:自动进行,安全无数据丢失
- 显式转换:需要手动进行,可能存在数据丢失风险
- 引用类型转换:涉及继承关系的类型转换
6.2 医疗系统中的类型转换示例
public class StaffManager
{
public void DemonstrateTypeConversion()
{
// 基本类型转换
int heartRate = 75;
double hrDouble = heartRate; // 隐式转换
decimal hrDecimal = (decimal)hrDouble; // 显式转换
// 字符串转换
string hrString = heartRate.ToString();
int parsedHR = int.Parse("75"); // 字符串转数字
// TryParse安全转换
if (int.TryParse("75", out int result))
{
Console.WriteLine($"转换成功:{result}");
}
// 引用类型转换示例
HospitalEmployee employee = new Nurse("N001", "张护士", "初级");
// 1. is运算符 - 检查类型
if (employee is Nurse)
{
Console.WriteLine("这是一名护士");
}
else if (employee is Doctor)
{
Console.WriteLine("这是一名医生");
}
// 2. as运算符 - 安全转换
Nurse nurse = employee as Nurse;
if (nurse != null)
{
Console.WriteLine($"护士级别: {nurse.NursingLevel}");
}
// 3. 显式转换
try
{
Nurse nurse2 = (Nurse)employee;
Console.WriteLine($"转换成功: {nurse2.NursingLevel}");
}
catch (InvalidCastException ex)
{
Console.WriteLine($"转换失败: {ex.Message}");
}
// 4. 模式匹配(C# 7.0+)
if (employee is Nurse nurseMatch)
{
Console.WriteLine($"匹配成功: {nurseMatch.NursingLevel}");
}
// 5. switch模式匹配
string GetEmployeeInfo(HospitalEmployee emp)
{
return emp switch
{
Nurse n => $"护士 {n.Name}, 级别 {n.NursingLevel}",
Doctor d => $"医生 {d.Name}, 专业 {d.Specialty}",
_ => $"员工 {emp.Name}"
};
}
}
// 自定义转换示例
public class VitalSignsConverter
{
public static implicit operator string(VitalSigns vs)
{
return $"体温: {vs.Temperature}℃, 心率: {vs.HeartRate}次/分";
}
public static explicit operator VitalSigns(string data)
{
// 简单示例,实际应该有更复杂的解析逻辑
var parts = data.Split(',');
return new VitalSigns
{
Temperature = decimal.Parse(parts[0]),
HeartRate = int.Parse(parts[1])
};
}
}
}
6.3 类型转换最佳实践
优先使用安全的转换方法:
- 使用 TryParse 而不是 Parse
- 使用 as 运算符而不是直接类型转换
- 总是检查转换结果
处理转换异常:
public decimal ParseTemperature(string temp) { try { return decimal.Parse(temp); } catch (FormatException) { Console.WriteLine("温度格式不正确"); return 0; } catch (OverflowException) { Console.WriteLine("温度值超出范围"); return 0; } }
总结
在这篇文章中,我们通过医院信息系统的实际场景,详细探讨了 C#中的几个重要概念:
命名空间
- 理解了命名空间的组织结构
- 掌握了多种使用命名空间的方法
- 学会了项目引用和命名空间导入
值类型和引用类型
- 理解了两种类型的本质区别
- 掌握了内存存储的不同方式
- 学会了通过医疗数据来理解类型特性
字符串处理
- 掌握了常用的字符串操作方法
- 学会了处理医疗记录文本
- 理解了字符串的特殊性质
继承
- 通过医院员工体系理解继承概念
- 掌握了继承的特性和使用方法
- 学会了方法重写和构造函数调用
集合
- 理解了不同集合类型的特点
- 掌握了集合的常用操作
- 学会了选择合适的集合类型
类型转换
- 掌握了各种类型转换方法
- 学会了安全的类型转换实践
- 理解了类型转换在医疗系统中的应用
这些概念将帮助我们构建更复杂的医疗信息系统功能。