C++中将一个无效参数传递给了将无效参数视为严重错误的函数

编写一个大地测量里坐标转换求七参数的代码,整个代码没有报错但运行不了,显示将一个无效参数传递给了将无效参数视为严重错误的函数,完整代码如下,请大佬指点

#include <iostream>
#include<Eigen/Dense>
#include<fstream>
#include<cmath>
#include<vector>

using namespace std;
using namespace Eigen;
struct Point//定义一个Point结构体用于储存三位点
{
    string name;
    double x;
    double y;
    double z;
};
struct Parameters//定义转换参数结构体
{
    double dx;//x平移参数
    double dy;//y平移参数
    double dz;//z平移参数
    double rx;//x旋转参数
    double ry;//y旋转参数
    double rz;//z旋转参数
    double dk;//尺度因子
};
std::vector<Point>readCoordinates(const string& filename)
{
    std::vector<Point>coordinates;//从文件读取坐标数据并储存在容器中
    ifstream file(filename);
    if (!file.is_open())
    {
        cerr << "Failed to open file:" << filename << endl;
        return coordinates;
    }
    string name;
    double x, y, z;
    while (file >> name>>x >> y >> z)
    {
        Point point = { name,x,y,z };
        coordinates.push_back(point);
    }
    file.close();
    return coordinates;
}
//估计七参数的值
Parameters estimateParameters(const std::vector<Point>& origin, const std::vector<Point>& target)
{
    Parameters params = {};
    vector<Point> L ;
    for (int i = 0; i <= origin.size(); i++)
    {
        L[i].x = target[i].x - origin[i].x;
        L[i].y = target[i].y - origin[i].y;
        L[i].z = target[i].z - origin[i].z;
    }
    //检查输入坐标数量是否相同
    if (origin.size() != target.size())
    {
        cerr << "Error:The number of origin and target coordinates does not match" << endl;
        return params;
    }
    //计算坐标平均值
    double originXMean = 0;
    double originYMean = 0;
    double originZMean = 0;
    double targetXMean = 0;
    double targetYMean = 0;
    double targetZMean = 0;
    /*for (int i = 0; i < origin.size() - 2; ++i)
    {
        originXMean += origin[i].x;
        originYMean += origin[i].y;
        originZMean += origin[i].z;
        targetXMean += target[i].x;
        targetYMean += target[i].y;
        targetXMean += target[i].z;
    }*/
    MatrixXd B((origin.size() - 2) * 3, 7);
    VectorXd N((origin.size() - 2) * 3);
    //通过循环初始化B矩阵和N矩阵
    for (size_t i = 0; i <= origin.size() - 2;i++)
    {
        B.row(i * 3) << 1.0, 0.0, 0.0, 0.0, -origin[i].z, origin[i].y, origin[i].x;
        B.row(i * 3 + 1) << 0.0, 1.0, 0.0, origin[i].z, 0.0, -origin[i].x, origin[i].z;
        B.row(i * 3 + 2) << 0.0, 0.0, 1.0, -origin[i].y, origin[i].x, 0.0, origin[i].z;
        N.row(i * 3) << L[i].x;
        N.row(i * 3 + 1) << L[i].y;
        N.row(i * 3 + 2) << L[i].z;
    }
    //最小二乘法求解七参数
    VectorXd parameters = (B.transpose() * B).inverse() * B.transpose() * N;
    //解析七参数
    params.dx = parameters(0);
    params.dy = parameters(1);
    params.dz = parameters(2);
    params.rx = parameters(3);
    params.ry = parameters(4);
    params.rz = parameters(5);
    params.dk = parameters(6);
    return params;
}
int main()
{
    string originFilename = "XYZ_origin_2.xyz";
    string targetFilename = "XYZ_target_2.xyz";
    vector<Point>originCoordinates = readCoordinates(originFilename);
    vector<Point>targetCoordinates = readCoordinates(targetFilename);
    Parameters params = estimateParameters(originCoordinates, targetCoordinates);
    std::cout << "Estimated Parameters:" << std::endl;
    std::cout << "dx: " << params.dx << std::endl;
    std::cout << "dy: " << params.dy << std::endl;
    std::cout << "dz: " << params.dz << std::endl;
    std::cout << "w: " << params.rx << std::endl;
    std::cout << "p: " << params.ry << std::endl;
    std::cout << "r: " << params.rz << std::endl;
    std::cout << "scale: " << params.dk << std::endl;

    return 0;
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
这个错误通常是由于传递给 `fseek()` 函数的 `FILE*` 参数为 `NULL` 导致的。在调用 `fseek()` 函数之前,请确保已经成功打开了文件并且返回了非空指针。 你可以在调用 `fseek()` 函数之前添加如下代码来检查文件是否成功打开: ```c++ FILE* fp = fopen("filename.txt", "r"); if(fp == NULL) { printf("Failed to open file\n"); return 1; } ``` 如果文件打开失败,将会输出 `"Failed to open file"` 并停止程序的执行。 另外,如果你使用的是 C++11 或更高版本的标准库,你可以使用更加安全和方便的文件操作方式。你可以使用 `std::ifstream` 类来代替 `fopen()` 函数,并使用 `seekg()` 函数代替 `fseek()` 函数。使用 `std::ifstream` 类的好处是,它会自动管理文件句柄,并且可以通过异常机制处理文件操作中的错误。 以下是使用 `std::ifstream` 类进行文件操作的示例代码: ```c++ #include <fstream> #include <iostream> int main() { std::ifstream infile("filename.txt"); if (!infile.is_open()) { std::cout << "Failed to open file" << std::endl; return 1; } // do something with the file infile.seekg(0, std::ios_base::end); auto size = infile.tellg(); infile.seekg(0); char* buffer = new char[size]; infile.read(buffer, size); // close file infile.close(); // do something with the data delete[] buffer; return 0; } ``` 在这个示例中,我们使用 `std::ifstream` 类打开文件,并使用 `is_open()` 函数来检查文件是否成功打开。然后使用 `seekg()` 函数定位到文件的结尾,并使用 `tellg()` 函数获取文件的大小。接下来,我们分配一个缓冲区来存储文件的内容,并使用 `read()` 函数将文件内容读取到缓冲区中。最后,我们关闭文件并释放缓冲区的内存。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值