C#的using关键字是日常开发中频繁使用的语法元素,它的功能根据使用场景的不同分为两大类,分别是引用命名空间和作为语句管理资源释放,下面分别介绍这两类作用的具体用法。

一、using引用命名空间
using最常见的用法是在代码文件顶部引用命名空间,这样在后续代码中可以直接使用对应命名空间下的类型,不需要写完整的类型限定名。比如我们常用的控制台输出方法Console.WriteLine,如果不引用System命名空间,就需要写全System.Console.WriteLine。
引用命名空间的基本语法如下:
// 引用System命名空间
using System;
// 引用System.Collections.Generic命名空间
using System.Collections.Generic;
namespace UsingDemo
{
class Program
{
static void Main(string[] args)
{
// 引用命名空间后可以直接使用List,无需写System.Collections.Generic.List
List<string> nameList = new List<string>();
nameList.Add("张三");
// 可以直接使用Console,无需写System.Console
Console.WriteLine(nameList[0]);
}
}
}
除了引用系统自带的命名空间,也可以引用自定义项目的命名空间,只要对应的命名空间在当前项目或者引用的依赖中存在即可。如果引用的两个命名空间下存在同名的类型,还可以通过给命名空间设置别名来避免冲突:
// 给命名空间设置别名
using MyAlias = MyProject.ModuleA;
using OtherAlias = MyProject.ModuleB;
namespace UsingDemo
{
class Program
{
static void Main(string[] args)
{
// 通过别名使用对应命名空间下的类型
MyAlias.User user1 = new MyAlias.User();
OtherAlias.User user2 = new OtherAlias.User();
}
}
}
二、using语句管理资源释放
using的另一类用法是作为语句使用,用来管理实现了IDisposable接口的对象,确保对象在使用完成后自动调用Dispose方法释放非托管资源,不需要开发者手动编写释放逻辑,也避免了因为忘记释放或者异常导致资源无法释放的问题。
2.1 基本语法
using语句的基本结构有两种,第一种是声明并初始化对象后跟随代码块:
using (FileStream fs = new FileStream("test.txt", FileMode.Open))
{
// 使用fs进行文件操作
byte[] buffer = new byte[1024];
int readCount = fs.Read(buffer, 0, buffer.Length);
Console.WriteLine($"读取到{readCount}字节数据");
}
// 代码块执行结束后,会自动调用fs.Dispose()方法释放资源
第二种是C# 8.0之后引入的简化写法,不需要写括号,using声明的对象作用域覆盖整个当前代码块:
using FileStream fs = new FileStream("test.txt", FileMode.Open);
// 后续代码可以使用fs
byte[] buffer = new byte[1024];
int readCount = fs.Read(buffer, 0, buffer.Length);
Console.WriteLine($"读取到{readCount}字节数据");
// 当前方法执行结束后,会自动调用fs.Dispose()方法
2.2 使用注意事项
- 只有实现了
IDisposable接口的类型才能用在using语句中,否则会编译报错。 - using语句会确保在对象超出作用域时调用
Dispose方法,即使代码块中出现异常也会正常执行释放逻辑,等价于try-finally结构:
FileStream fs = null;
try
{
fs = new FileStream("test.txt", FileMode.Open);
byte[] buffer = new byte[1024];
int readCount = fs.Read(buffer, 0, buffer.Length);
Console.WriteLine($"读取到{readCount}字节数据");
}
finally
{
if (fs != null)
{
fs.Dispose();
}
}
上面的try-finally结构和前面的using语句功能完全一致,可以看出using语句大大简化了资源释放的代码编写。
2.3 自定义类型支持using语句
如果我们需要让自己的自定义类型支持using语句管理,只需要让类型实现IDisposable接口,在Dispose方法中编写资源释放的逻辑即可:
public class MyResource : IDisposable
{
// 模拟非托管资源
private IntPtr _nativeResource;
public MyResource()
{
// 初始化资源
_nativeResource = new IntPtr(1);
Console.WriteLine("资源初始化完成");
}
public void Dispose()
{
// 释放非托管资源
if (_nativeResource != IntPtr.Zero)
{
// 模拟释放逻辑
_nativeResource = IntPtr.Zero;
Console.WriteLine("资源释放完成");
}
}
}
// 使用自定义类型
using (MyResource resource = new MyResource())
{
// 使用资源
Console.WriteLine("正在使用资源");
}
// 执行后会输出:资源初始化完成 正在使用资源 资源释放完成
三、总结
C#的using关键字两类用途都非常实用,引用命名空间可以简化类型书写,提升编码效率;作为语句使用可以自动管理非托管资源释放,减少资源泄漏的风险。开发者在实际开发中可以根据场景选择合适的使用方式,同时注意using语句仅适用于实现了IDisposable接口的类型,避免错误使用。