zoukankan      html  css  js  c++  java
  • 凸多边形最优三角剖分问题

    参考书籍《算法设计与分析》  王晓东   动态规划
    1.问题描述
    (注:是所有的三角形的权值之和,不是只计算边和弦的权值之和)
     
    2.分析
     
     
     
    3.编码实现:
    /** 
     * @Author:胡家威  
     * @CreateTime:2011-11-10 下午12:31:16
     * @Description:凸多边形的最优三角剖分
     */


    package ex2;

    public class Triangulation {

      private int n;// n多边形
      private int[][] weight;// 边的权值数组

      public Triangulation(int n) {
        this.n = n;
        this.weight = new int[n][n];
      }

      public static void main(String[] args) {
        Triangulation triangulation = new Triangulation(6);
        initTriangulation(triangulation);
        int n = triangulation.getN();// 凸多边形的边数
        int[][] t = new int[n][n];// t[i][j]表示顶点{Vi-1,Vi...Vj}组成的多边形的最优三角形剖分的权值
        int[][] s = new int[n][n];// s[i][j]表示与Vi-1和Vj一起构成三角形的第三个顶点的位置
        triangulation.minWeightTriangulation2(triangulation.getN() - 1, t, s);
        System.out.println(t[1][5]);
      }

      // 初始化weight数组的信息
      public static void initTriangulation(Triangulation triangulation) {
        int[][] weight = { { 0, 2, 2, 3, 1, 4 }, { 2, 0, 1, 5, 2, 3 }, { 2, 1, 0, 2, 1, 4 }, 
            { 3, 5, 2, 0, 6, 2 }, { 1, 2, 1, 6, 0, 1 }, { 4, 3, 4, 2, 1, 0 } };
        triangulation.setWeight(weight);
      }

      // 得到最优的三角形剖分,n是总边数-1
      public void minWeightTriangulation(int n, int[][] t, int[][] s) {
        // 初始化所有的二顶点多边形权值为0
        for (int i = 1; i <= n; i++) {
          t[i][i] = 0;
        }
        // 循环求解t[i][j]
        for (int r = 2; r <= n; r++) {// (j-i)的范围[2,n]
          // 当r=2时,循环实际上是在给t赋边的值,即相邻的两个顶点的权值,例如t[1][2],t[2][3]...
          for (int i = 1; i <= n - r + 1; i++) {// i的范围[1,n+1-r],这里i要保证i+r<=n
            int j = i + r - 1;
            t[i][j] = t[i + 1][j] + getWeight(i - 1, i, j);// 这里实际上就是k=i
            // t[i][j] = t[i][i] + t[i + 1][j] + getWeight(i - 1, i, j)
            s[i][j] = i;
            // i-1,i,j
            // 循环k,范围是[i+1,j-1],求出最小的t[i][j]
            for (int k = i + 1; k < j; k++) {// k是i和j之间的中间顶点
              int u = t[i][k] + t[k + 1][j] + getWeight(i - 1, k, j);// 以k作为划分得到的权值
              if (u < t[i][j]) {// 如果权值更小,那么同时更新t[i][j]和s[i][j]
                t[i][j] = u;
                s[i][j] = k;
              }
            }
          }
        }
      }

      // 我的写法,在第二个循环这里不同,没有什么差别,只是我易于我理解
      public void minWeightTriangulation2(int n, int[][] t, int[][] s) {
        // 初始化所有的二顶点多边形权值为0
        for (int i = 1; i <= n; i++) {
          t[i][i] = 0;
        }
        // 循环求解t[i][j]
        for (int r = 1; r <= n; r++) {// r=(j-i)的范围[1,n]
          // 当r=1时,循环实际上是在给t赋边的值,即相邻的两个顶点的权值,例如t[1][2],t[2][3],t[3][4]...
          for (int i = 1; i <= n - r; i++) {// i的范围[1,n-r],这里i要保证 j=i+r<=n
            int j = i + r;
            t[i][j] = t[i + 1][j] + getWeight(i - 1, i, j);// 这里实际上就是k=i
            // t[i][j] = t[i][i] + t[i + 1][j] + getWeight(i - 1, i, j)
            s[i][j] = i;// i-1,i,j
            // 循环k,范围是[i+1,j-1],求出最小的t[i][j]
            for (int k = i + 1; k < j; k++) {// k是i和j之间的中间顶点
              int u = t[i][k] + t[k + 1][j] + getWeight(i - 1, k, j);// 以k作为划分得到的权值
              if (u < t[i][j]) {// 如果权值更小,那么同时更新t[i][j]和s[i][j]
                t[i][j] = u;
                s[i][j] = k;
              }
            }
          }
        }
      }

      // 计算一个三角形的权值之和
      public int getWeight(int i, int j, int k) {
        return weight[i][j] + weight[j][k] + weight[i][k];
      }

      public int getN() {
        return n;
      }

      public void setN(int n) {
        this.n = n;
      }

      public int[][] getWeight() {
        return weight;
      }

      public void setWeight(int[][] weight) {
        this.weight = weight;
      }

    }
     
    数据:
        
     
    结果是: 24





  • 相关阅读:
    特征词选择算法对文本分类准确率的影响(前言)
    答火星人.NET。如何使用我的本科毕业程序 正文提取DEMO
    有关matlab画图格式的部分代码
    文本分类中的特征词选择算法系列科普(前言AND 一)
    c++杂项备忘
    写一点应用关于 Lucene.Net,snowball的重新组装(一)在Lucene.Net中加入词性标注与词根还原功能
    C++字符串处理:批量去重,以及大写变小写
    Python打印到文件
    中文分词:采用二元词图以及viterbi算法(三)
    博客园和百度空间,我的两个家
  • 原文地址:https://www.cnblogs.com/yinger/p/2270175.html
Copyright © 2011-2022 走看看