zoukankan      html  css  js  c++  java
  • 计算机图形学实验一(关于初用openGL)

      鉴于本周忙于各种党课的时期,以及团队项目我负责的部分不是很忙,所以就写写关于刚结束的图形学实验一的收获。

      实验一是关于基本二维图元生成算法,其中又分为三个小实验,分别是生成直线、圆以及多边形填充算法。由于实验室三人一组分工,所以我负责了多边形填充算法的实现。

      简单介绍一下算法,算法用到了两个数据结构:Edge Table和Active Edge List。Edge Table是个list<Edge>型的链表数组,数组下标为y轴坐标,当顺着y轴坐标从下往上扫描过程中扫描到某个边的最低端点时,就把该边的信息存入(同一下标的边以链表形式存储),边包括x_low(两端点中y较小的点对应的x值),y_top(两端点中y较大的值),1/m(斜率的倒数)。而Active Edge List是list<Edge>型的,大循环需要遍历Edge Tabl数组,每次先把同一下标中链表存储的边按左右位置排序加入到Active Edge Lis (具体实现不细说了,因为涉及到斜率问题),然后按照两条边的x_low坐标填充之间的所有点,再把x_low+=1/m,然后数组下标+1(当有变得y_top==当前下标时,把该边从Active Edge List中移除)。

    不得不说我感觉我完全没有说清楚。。。

    void Scan(){
        Edge edge;
        list<Edge> ET[21];
        int i = 0;
        int temp;
        list<Edge> ALE;
        list<Edge>::iterator k;
        float xtemp, x1temp, ytemp;
        for (i = 1; i < j - 1; i++){
            if (points[i][1]>points[i + 1][1]){
                edge.x_low = points[i + 1][0];
                edge.y_top = points[i][1];
                edge.m = (points[i][0] - points[i + 1][0]) / (points[i][1] - points[i + 1][1]);
                temp = points[i + 1][1] * 10 + 10;
                if (i == j - 2){
                    if (((points[i][1] - points[i + 1][1])*(points[0][1] - points[i + 1][1])) < 0){
                        edge.x_low = edge.x_low + edge.m / 10;
                        temp++;
                    }
                }
                else{
                    if (((points[i][1] - points[i + 1][1])*(points[i + 2][1] - points[i + 1][1])) < 0){
                        edge.x_low = edge.x_low + edge.m / 10;
                        temp++;
                    }
                }
                ET[temp].push_back(edge);
            }
            else if (points[i][1]<points[i + 1][1]){
                edge.x_low = points[i][0];
                edge.y_top = points[i + 1][1];
                edge.m = (points[i][0] - points[i + 1][0]) / (points[i][1] - points[i + 1][1]);
                temp = points[i][1] * 10 + 10;
                if (((points[i - 1][1] - points[i][1])*(points[i + 1][1] - points[i][1]))<0){
                    edge.x_low = edge.x_low + edge.m / 10;
                    temp++;
                }
                ET[temp].push_back(edge);
            }
        }
        if (points[0][1] > points[j - 1][1]){
            edge.x_low = points[j - 1][0];
            edge.y_top = points[0][1];
            edge.m = (points[0][0] - points[j - 1][0]) / (points[0][1] - points[j - 1][1]);
            temp = points[j-1][1] * 10 + 10;
            if (((points[j - 2][1] - points[j - 1][1])*(points[j][1] - points[j - 1][1])) < 0){
                edge.x_low = edge.x_low + edge.m / 10;
                temp++;
            }
            ET[temp].push_back(edge);
        }
        else if (points[0][1] < points[j - 1][1]){
            edge.x_low = points[0][0];
            edge.y_top = points[j - 1][1];
            edge.m = (points[0][0] - points[j - 1][0]) / (points[0][1] - points[j - 1][1]);
            temp = points[0][1] * 10 + 10;
            if (((points[j - 1][1] - points[0][1])*(points[1][1] - points[0][1]))<0){
                edge.x_low = edge.x_low + edge.m / 10;
                temp++;
            }
            ET[temp].push_back(edge);
        }
        if (points[0][1] > points[1][1]){
            edge.x_low = points[1][0];
            edge.y_top = points[0][1];
            edge.m = (points[0][0] - points[1][0]) / (points[0][1] - points[1][1]);
            temp = points[1][1] * 10 + 10;
            if (((points[0][1] - points[1][1])*(points[2][1] - points[1][1])) < 0){
                edge.x_low = edge.x_low + edge.m / 10;
                temp++;
            }
            ET[temp].push_back(edge);
        }
        else if (points[0][1] < points[1][1]){
            edge.x_low = points[0][0];
            edge.y_top = points[1][1];
            edge.m = (points[0][0] - points[1][0]) / (points[0][1] - points[1][1]);
            temp = points[0][1] * 10 + 10;
            if (((points[j - 1][1] - points[0][1])*(points[1][1] - points[0][1])) < 0){
                edge.x_low = edge.x_low + edge.m / 10;
                temp++;
            }
            ET[temp].push_back(edge);
        }
        for (i = 0; ET[i].size() != 0; i++){
            temp = i;
        }
        for (i = temp; i < 21; i++){
            while(ET[i].size()!=0){
                edge = ET[i].front();
                ET[i].pop_front();
                for (k = ALE.begin(); k != ALE.end(); k++){
                    if (edge.x_low < k->x_low)
                        break;
                    if (edge.x_low == k->x_low){
                        if (edge.m*k->m>0){
                            if (edge.m>k->m)
                                break;
                            else{
                                k++;
                                break;
                            }
                        }
                        else{
                            if (edge.m<k->m)
                                break;
                            else{
                                k++;
                                break;
                            }
                        }
                    }
                }
                ALE.insert(k, edge);
            }
            glBegin(GL_POINTS);
            for (k = ALE.begin(); k != ALE.end(); k++){
                xtemp = k->x_low;
                k++;
                x1temp = k->x_low;
                ytemp = (i - 10) / 10.0;
                for (float j = xtemp; j < x1temp; j = j + 0.1)
                    glVertex2f(Round(j), ytemp);
            }
            glEnd();
            glFlush();
            for (k = ALE.begin(); k != ALE.end(); ){
                ytemp = (i - 10) / 10.0;
                if (k->y_top == ytemp)
                    k = ALE.erase(k);
                else{
                    k->x_low = k->x_low +k->m/10.0;
                    k++;
                }
            }
        }
    }
    扫描线算法

    程序主界面如下

    点击鼠标左键可在屏幕中输入多边形顶点(点击顺序对多边形生成有影响)

    点击鼠标右键可唤出菜单

    Line(连线):将输入的点按点击顺序连线,调用了glBegin(GL_LINE_LOOP)函数

    Fill(填充):对生成的多边形进行填充(可以不连线),使用了上述的算法

    Color(变色):只改变多边形颜色而不影响坐标系

    ClearScreen(清屏):重置主界面

    Exit(退出):就是退出


     

      这次的实验让我感叹于openGL画图的便捷(虽然eaxyX也挺好的),学习到了openGL相关的画图、菜单、鼠标等函数的使用以及c++ list链表的使用,感觉能图形化的东西确实比干巴巴的算法有意思,不过算法还是重点难点,有点愁。

  • 相关阅读:
    基础架构:一条SQL查询语句是如何执行的?
    常用的字符串转换日期格式
    JSONObject.fromObject(map)(JSON与JAVA数据的转换)
    ModelAndView的介绍
    Springmvc常用注解
    @PathVariable注解
    @ModelAttribute运用详解
    struts2将servlet对象注入到Action中
    为什么要继承ActionSupport?
    mysql中like用法
  • 原文地址:https://www.cnblogs.com/hesoyamlyf/p/5471753.html
Copyright © 2011-2022 走看看