重构长方法之提取方法

本篇文章将讲解解决长方法的6钟解决方案中的提取方法
1. 什么是长方法

长方法通常指的是代码中功能过于复杂或包含过多逻辑的函数或方法。这种方法往往难以阅读、理解和维护,可能违反单一职责原则。为了提高可读性和可维护性,建议将长方法拆分成多个小方法,每个方法负责一个具体的功能。

2. 提取方法

我们先来看一段代码:

csharp 复制代码
public void ProcessData(List<int> numbers)
{
    if (numbers == null || !numbers.Any())
    {
        Console.WriteLine("No data to process.");
        return;
    }

    int sum = 0;
    int count = 0;
    List<int> processedNumbers = new List<int>();

    foreach (var number in numbers)
    {
        if (number > 0)
        {
            sum += number;
            processedNumbers.Add(number * 2);
            count++;
        }
    }

    double average = count > 0 ? (double)sum / count : 0;

    Console.WriteLine($"Processed {count} positive numbers.");
    Console.WriteLine($"Sum: {sum}, Average: {average}");
    Console.WriteLine("Processed Numbers: " + string.Join(", ", processedNumbers));
}

我们阅读这段代码,发现在这个方法包含了多个职责,比如处理输入、计算总和和平均值、生成输出。这么多职责都写在了一个方法里,想要弄清这个方法功能就稍显困难(方法中的行数/职责越多,就越难弄清该方法的功能 )。

使用提取方法这个解决方案可以轻松的处理前面所说的问题,我们一起来看一下更新后的代码:

csharp 复制代码
public void ProcessData(List<int> numbers)
{
    if (!ValidateInput(numbers)) return;

    var processedData = ProcessNumbers(numbers);
    DisplayResults(processedData);
}

private bool ValidateInput(List<int> numbers)
{
    if (numbers == null || !numbers.Any())
    {
        Console.WriteLine("No data to process.");
        return false;
    }
    return true;
}

private (int Sum, double Average, List<int> ProcessedNumbers) ProcessNumbers(List<int> numbers)
{
    int sum = 0;
    int count = 0;
    List<int> processedNumbers = new List<int>();

    foreach (var number in numbers)
    {
        if (number > 0)
        {
            sum += number;
            processedNumbers.Add(number * 2);
            count++;
        }
    }

    double average = count > 0 ? (double)sum / count : 0;
    return (sum, average, processedNumbers);
}

private void DisplayResults((int Sum, double Average, List<int> ProcessedNumbers) data)
{
    Console.WriteLine($"Processed {data.ProcessedNumbers.Count} positive numbers.");
    Console.WriteLine($"Sum: {data.Sum}, Average: {data.Average}");
    Console.WriteLine("Processed Numbers: " + string.Join(", ", data.ProcessedNumbers));
}

通过提取方法对不同的职责进行封装,我们看到代码既清晰又容易理解,而且更加易维护了。
3. 提取方法的步骤

首先,分析出长方法中所包含的职责,然后根据这些职责创建多个新方法,接着将每种职责的代码复制到对应的新方法中,紧接着从长方法中删除这些代码,并替换为对新方法的调用,最后查找每种职责代码中使用的变量,如果这些变量是在职责代码片段中声明的并且没有在片段外使用,那么只需将它们将做为新方法的局部变量即可。如果变量是在多种职责代码都在用的,则需要将这些变量作为传递给新方法的参数,以便使用其中之前。

4. 提取方法的优点

首先使代码更易读,这里要注意的是提取的方法的名称一定要是对方法体内代码职责的喵叔。其次减少代码重复,方法中的代码可以在程序的其他地方重复使用。 然后每个方法中的代码是独立的,这就表示出错的可能性较低,即使出错了排查和修改也简单。

相关推荐
csdn_aspnet19 分钟前
C# 求n边凸多边形的对角线数量(Find number of diagonals in n sided convex polygon)
开发语言·算法·c#
AI先驱体验官2 小时前
智能体变现:从技术实现到产品化的实践路径
大数据·人工智能·深度学习·重构·aigc
武藤一雄5 小时前
C# 设计模式大全(第一弹|7种)
microsoft·设计模式·微软·c#·.net·.netcore
童园管理札记5 小时前
【回归儿童本位,重构专业底色】学前教育行业的深度思辨与价值坚守(二)
经验分享·重构·生活·学习方法·微信公众平台
格林威6 小时前
Baumer相机锂电池极片裁切毛刺检测:防止内部短路的 5 个核心方法,附 OpenCV+Halcon 实战代码!
开发语言·人工智能·数码相机·opencv·计算机视觉·c#·视觉检测
AI服务老曹7 小时前
源码交付与低代码重构:企业级 AI 视频管理平台的二次开发实战
人工智能·低代码·重构
向上的车轮7 小时前
熟悉C#如何转TypeScript——SDK与包引用
开发语言·typescript·c#
北京软秦科技有限公司8 小时前
AI报告文档审核助力食品飞检常态化应对:IACheck下的风险防控与质量管控重构
大数据·人工智能·重构
CSharp精选营8 小时前
Dispose 不释放?C# 资源泄漏的 3 种隐蔽场景排查
c#·资源泄漏
二等饼干~za8986689 小时前
豆包GEO优化源码开发全解析:技术架构、实现逻辑与实操指南
数据库·sql·重构·架构·mybatis·音视频