Điều khiển PID: Thêm một khoảng thời gian trễ trước vòng lặp tiếp theo có phải là một ý tưởng hay không?

Jan 11 2021

Tôi đang triển khai điều khiển PID trong c ++ để làm cho rô bốt truyền động vi sai quay một số độ chính xác, nhưng tôi đang gặp nhiều vấn đề.

Thoát khỏi vòng điều khiển sớm do thời gian chạy vòng lặp nhanh

Nếu rô bốt đo được sai số của nó nhỏ hơn 0,5 độ, nó sẽ thoát khỏi vòng điều khiển và coi lượt "đã kết thúc" (.5 là giá trị ngẫu nhiên mà tôi có thể thay đổi tại một số điểm). Có vẻ như vòng điều khiển đang chạy nhanh đến mức robot có thể quay với tốc độ rất cao, vượt qua điểm đặt và thoát khỏi vòng lặp / cắt nguồn động cơ, vì nó đã ở điểm đặt trong một thời gian ngắn. Tôi biết rằng đây là toàn bộ mục đích của điều khiển PID, để đạt được chính xác điểm đặt mà không bị quá tải, nhưng vấn đề này làm cho việc điều chỉnh các hằng số PID trở nên rất khó khăn. Ví dụ, tôi cố gắng tìm một giá trị kp sao cho có dao động ổn định, nhưng không bao giờ có bất kỳ dao động nào vì robot nghĩ rằng nó đã "kết thúc" khi nó vượt qua điểm đặt. Để khắc phục điều này,Tôi đã triển khai một hệ thống trong đó robot phải ở điểm đặt trong một khoảng thời gian nhất định trước khi thoát ra và điều này đã có hiệu quả, cho phép dao động xảy ra, nhưng vấn đề thoát khỏi vòng lặp sớm có vẻ như là một vấn đề bất thường và giải pháp của tôi có thể không chính xác.

Thuật ngữ D không có hiệu lực do thời gian chạy nhanh

Khi tôi đã để rô-bốt dao động theo cách có kiểm soát chỉ sử dụng P, tôi đã cố gắng thêm D để tránh bị vọt lố. Tuy nhiên, điều này không có tác dụng trong phần lớn thời gian, vì vòng điều khiển đang chạy quá nhanh đến 19 vòng trên 20 vòng, tỷ lệ thay đổi lỗi là 0: rô bốt không di chuyển hoặc di chuyển không đủ cho nó được đo trong thời gian đó. Tôi đã in sự thay đổi trong lỗi và thuật ngữ phái sinh mỗi vòng lặp để xác nhận điều này và tôi có thể thấy rằng cả hai đều sẽ bằng 0 trong khoảng 20 chu kỳ vòng lặp trước khi lấy một giá trị hợp lý và sau đó quay lại 0 trong 20 chu kỳ khác. Như tôi đã nói, tôi nghĩ rằng điều này là do các chu kỳ vòng lặp quá nhanh đến mức robot thực sự không di chuyển đủ cho bất kỳ loại thay đổi đáng chú ý nào về lỗi.Đây là một vấn đề lớn vì nó có nghĩa là số hạng D về cơ bản không ảnh hưởng đến chuyển động của rô bốt vì nó hầu như luôn luôn bằng 0. Để khắc phục vấn đề này, tôi đã thử sử dụng giá trị khác 0 cuối cùng của đạo hàm thay cho bất kỳ giá trị 0 nào, nhưng điều này không hoạt động tốt và rô bốt sẽ dao động ngẫu nhiên nếu đạo hàm cuối cùng không đại diện cho tốc độ thay đổi lỗi hiện tại.

Lưu ý: Tôi cũng đang sử dụng một tiến dao nhỏ cho hệ số ma sát tĩnh, và tôi gọi chuyển tiếp này là "f"

Tôi có nên thêm thời gian trễ không?

Tôi nhận ra rằng tôi nghĩ rằng nguồn gốc của cả hai vấn đề này là vòng lặp chạy rất nhanh, vì vậy điều gì đó tôi nghĩ đến là thêm một câu lệnh chờ vào cuối vòng lặp. Tuy nhiên, nó có vẻ là một giải pháp tổng thể không tốt khi cố tình làm chậm một vòng lặp. Đây có phải là một ý tưởng tốt?

turnHeading(double finalAngle, double kp, double ki, double kd, double f){
    std::clock_t timer;
    timer = std::clock();

    double pastTime = 0;
    double currentTime = ((std::clock() - timer) / (double)CLOCKS_PER_SEC);

    const double initialHeading = getHeading();
    finalAngle = angleWrapDeg(finalAngle);

    const double initialAngleDiff = initialHeading - finalAngle;
    double error = angleDiff(getHeading(), finalAngle);
    double pastError = error;

    double firstTimeAtSetpoint = 0;
    double timeAtSetPoint = 0;
    bool atSetpoint = false;

    double integral = 0;
    double derivative = 0;
    double lastNonZeroD = 0;

    while (timeAtSetPoint < .05)
    {
        updatePos(encoderL.read(), encoderR.read());
        error = angleDiff(getHeading(), finalAngle);

        currentTime = ((std::clock() - timer) / (double)CLOCKS_PER_SEC);
        double dt = currentTime - pastTime;

        double proportional = error / fabs(initialAngleDiff);
        integral += dt * ((error + pastError) / 2.0);
        double derivative = (error - pastError) / dt;
        
        //FAILED METHOD OF USING LAST NON-0 VALUE OF DERIVATIVE
        // if(epsilonEquals(derivative, 0))
        // {
        //     derivative = lastNonZeroD;
        // }
        // else
        // {
        //     lastNonZeroD = derivative;
        // }

        double power = kp * proportional + ki * integral + kd * derivative;

        if (power > 0)
        {
            setMotorPowers(-power - f, power + f);
        }
        else
        {
            setMotorPowers(-power + f, power - f);
        }

        if (fabs(error) < 2)
        {
            if (!atSetpoint)
            {
                atSetpoint = true;
                firstTimeAtSetpoint = currentTime;
            }
            else //at setpoint
            {
                timeAtSetPoint = currentTime - firstTimeAtSetpoint;
            }
        }
        else //no longer at setpoint
        {
            atSetpoint = false;
            timeAtSetPoint = 0;
        }
        pastTime = currentTime;
        pastError = error;
    }
    setMotorPowers(0, 0);
}

turnHeading(90, .37, 0, .00004, .12);

Trả lời

3 UgoPattacini Jan 12 2021 at 05:14

Đừng tháo bộ điều khiển của bạn.

Mục đích của bộ điều khiển không chỉ là điều khiển hệ thống của bạn đến điểm đặt mong muốn theo phản ứng động học được xác định trước mà còn để chống lại các yếu tố tiềm ẩn bên ngoài có thể cản trở nhiệm vụ này. Hãy nghĩ về một sự xáo trộn sẽ khiến hệ thống đi xa điểm đặt khi nó đã đạt đến. Do đó, bộ điều khiển sẽ luôn hoạt động (trừ khi bạn cần thay đổi chính tác vụ và kết quả là thay đổi cả bộ điều khiển).

Để đạt được điều này, bạn chắc chắn sẽ cần thêm phần tích phân, phần này chịu trách nhiệm gây ra sai số trạng thái ổn định rỗng khi có các đại lượng không được mô hình hóa và nhiễu bên ngoài.

Đây là một nguồn khá liên quan: https://robotics.stackexchange.com/a/19198/6941.

Bỏ đi đạo hàm.

95% bộ điều khiển PID trong các ngành công nghiệp là bộ điều khiển PI (xem "Hệ thống phản hồi" Astrom, Murray) vì phần D có thể đóng một vai trò quan trọng chỉ với các quy trình chậm (như những quy trình liên quan đến điều chỉnh nhiệt độ và mức bình chứa). Đây chắc chắn không phải là trường hợp của bạn. Bằng cách nào đó các nguồn liên quan về những khó khăn của các điều khoản phái sinh là:

  • https://robotics.stackexchange.com/a/21556/6941.
  • https://robotics.stackexchange.com/a/21555/6941.

Khi tôi đã để rô-bốt dao động theo cách có kiểm soát chỉ sử dụng P, tôi đã cố gắng thêm D để tránh bị vọt lố.

Chà, có vẻ như bạn đang làm theo đơn thuốc của Ziegler-Nichols để điều chỉnh bộ điều khiển của mình. Có những bảng bạn cần phải tuân theo để cung cấp cho bạn ước tính về lợi nhuận. Tuy nhiên, đây là những suy đoán và có khả năng sẽ không hoạt động trong trường hợp của bạn.

Đơn giản chỉ cần thả phần D và tập trung vào bộ điều khiển PI. Ngoài ra còn có một biến thể ZN khác không dựa vào dao động được tạo ra có chủ đích:https://robotics.stackexchange.com/a/21616/6941.

Không bao giờ tăng thêm độ trễ cho hệ thống.

Sự chậm trễ là kẻ xấu cần đối phó trong một vòng điều khiển và các kỹ sư quái vật tồi tệ nhất phải chiến đấu chống lại ( hãy xem điều này 🎥 ) vì chúng làm giảm đáng kể biên độ pha đẩy toàn bộ hệ thống đến sự mất ổn định.

Nếu bạn cho rằng vòng lặp của mình quá nhanh, hãy áp dụng định hình đầu vào cho điểm đặt (ví dụ: quỹ đạo giật tối thiểu ) để làm mượt bước khởi đầu. Hệ thống vòng kín sẽ phản ứng một cách duyên dáng hơn. Một khả năng khác là độ lợi P của bạn quá cao: chỉ cần ngắt bộ điều khiển sau đó.

Về mặt này, nếu bạn chèn số hạng tích phân I, thì bạn sẽ cần phải suy luận về thời gian mẫu $T_s$ cũng.

1 TimWescott Jan 12 2021 at 08:36

Sẽ tốt hơn nhiều nếu xác định tỷ lệ mẫu cố định cho bộ điều khiển của bạn. Một nguyên tắc chung thực sự là bất kể thời gian giải quyết bạn cần ra khỏi vòng lặp khi nó đang hoạt động ở chế độ tuyến tính , khoảng thời gian lấy mẫu của bạn phải ít hơn thời gian giải quyết từ 10 lần đến 100 lần. Nói cách khác, tốc độ lấy mẫu phải nhanh hơn băng thông vòng lặp mong muốn từ 10 đến 100 lần.

Như một phần thưởng nhỏ được bổ sung, điều này có nghĩa là bạn có thể tính toán dt bên ngoài vòng lặp

Theo tùy chọn, tốc độ lấy mẫu nên được quyết định bởi phần cứng, ví dụ như bộ đếm thời gian phần cứng, và tốt hơn nữa, việc lấy mẫu vị trí nên được thực hiện bằng phần cứng, được kích hoạt bằng bộ đếm thời gian. Điều này làm giảm đáng kể các ràng buộc về thời gian thực trên phần mềm.

Nếu bạn đang sử dụng tốc độ lấy mẫu hợp lý và hầu như bạn không đăng ký các thay đổi trong bộ mã hóa của mình, thì bộ mã hóa của bạn không có đủ các bước.

Tôi sẽ không đồng ý với @Ugo_Pattachini, trong đó nếu vòng lặp liên quan đến động cơ và bộ điều khiển của bạn đủ sạch, một số hành động khác biệt có thể có lợi. Nhưng có một cơ hội tốt là nó cần phải được giới hạn băng tần (tức là bạn cần một bộ điều khiển độ trễ dẫn đầu) và nếu bạn đang thực hiện điều chỉnh PID của quần thể thì rất có thể bạn không có công cụ để thiết lập giới hạn ban nhạc một cách chính xác.

Tuy nhiên, nếu bạn không thể thực thi thời gian chặt chẽ về việc lấy mẫu bộ mã hóa, thậm chí đừng cố sử dụng điều khiển phái sinh. Thời gian lấy mẫu không thường xuyên cộng với một số vận tốc trên động cơ của bạn sẽ tạo ra tiếng ồn lớn hơn và bộ điều khiển phái sinh có xu hướng khuếch đại tiếng ồn. Nếu bạn không thể sắp xếp để kiểm soát một tín hiệu rõ ràng, điều khiển phái sinh sẽ không phù hợp.