动态规划实现最长单调递增子序列

1. 实验环境

操作系统:Mac 64
运行内存:16GB
编程语言:Java
编译环境:Eclipse

2. 题目要求

设计一个Ο(nlgn)时间的算法,求一个 n 个数的序列的最长单调递增子序列。

A. 用户界面输入原始序列

设计便于用户操作的界面,可以输入 n 个数,在计算后输出最长单调递增子 序列和该子序列的长度。

B. 求最长单调递增子序列

获取文本框内容,依次遍历字符串的单个字符,如果递增,直接加入。否则, 采用二分法缩小递归范围,如果递增,需要修改值,而非加入。该算法时间复杂 度 Ο(nlgn)。

C. 用户界面得到子序列及长度

将求得的子序列及该子序列的长度返回到用户界面的文本框中。

D. 用户界面重置和计算

点击计算可求出子序列和长度。点击重置可重新输入序列并求结果。

4.项目结构

程序结构如下图所示,lmis_str.java 可处理多长度的序列;而 lmis_list_num.java 只可处理 int/long 范围的序列。项目结构如下图所示。

5. 实验结果

A. 程序运行界面和正确性

程序编译运行结果如下图所示。

图 1 程序编译运行结果界面 程序输入序列,运行结果如下图所示。

图 2 程序运行结果界面

6.实验代码

实现用户界面 求单调最长递增子序列

package dp;

import java.awt.EventQueue;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.ArrayList;
import java.util.List;
import java.util.Scanner;

import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JTextField;
public class lmis_str extends JFrame{
    public static ArrayList<Integer> list=new ArrayList<Integer>();
    public JPanel mainPanel=new JPanel();
    public JLabel jLabel=new JLabel("输入序列:");
    public JTextField jTextField=new JTextField(40);
    public JLabel jLabel1=new JLabel("递增子序列:");
    public JTextField jTextField1=new JTextField(40);
    public JLabel jLabel2=new JLabel("递增子序列长度:");
    public JTextField jTextField2=new JTextField(40);
    public JButton rebtn=new JButton("重置");
    public JButton calbtn=new JButton("计算");
    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            public void run() {
                try {
                    //继承JFrame
                    lmis_str frame = new lmis_str();
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        });
    }
    public lmis_str(){
        
        mainPanel.add(jLabel);
        mainPanel.add(jTextField);
        mainPanel.add(jLabel1);
        mainPanel.add(jTextField1);
        mainPanel.add(jLabel2);
        mainPanel.add(jTextField2);
        mainPanel.add(rebtn);
        mainPanel.add(calbtn);
        this.add(mainPanel);
        this.setTitle("计算最长递增子序列:");
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        this.setVisible(true);
        this.setBounds(0, 0, 600, 600);
        calbtn.addActionListener(new ActionListener() {
            
            @Override
            public void actionPerformed(ActionEvent e) {
                String string=jTextField.getText();
                jTextField1.setText(getLmis(string));
                jTextField2.setText(String.valueOf(getLmis(string).length()));
                
                
            }
        });
        rebtn.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                jTextField.setText("");
                jTextField1.setText("");
                jTextField2.setText("");
            }
        });
        
    }
    public String getLmis(String str){
        List<Character> list=new ArrayList<Character>();
        list.add(str.charAt(0));
        int index=0;
        for(int i=0;i<str.length();i++){
            char c=(char) (str.charAt(i)-'0');
            if(list.get(index)<c){
                list.add(++index,c);
            }else{
                int low=0;
                int high=index;
                while (low<high) {
                    int middle = (low+high)>>1;
                      if(list.get(middle)<c)
                        low=middle+1;
                    else {
                        high=middle-1;
                    }
                }
                if(list.get(low)<c&&(low+1)<list.size())
                    list.set(low+1, c);
                else
                    list.set(low, c);
            }
        }
        System.out.println(list.size());
        StringBuffer sBuffer=new StringBuffer();
        for(int t:list){
            sBuffer.append(t);
        }
        return new String(sBuffer);
    }
}

7. 实验技巧与注意点

A. 在处理 n 个数时,可直接处理 string 的单个字符,这样不会限制长度。如果 处理 int/long 时会有位数的限制。
B. 在存储原始序列和结果子序列时,应使用 list,这样可以动态增加字符。如 果使用 Array,结果长度并不确定。
C.向 list 添加值时,需要将数字向字符转换,做(字符-‘0’)的操作。
D. 动态规划过程中,只有递归到最长子序列递增时才需要add,递归到递增需 要做 set 操作。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,607评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,047评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,496评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,405评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,400评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,479评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,883评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,535评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,743评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,544评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,612评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,309评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,881评论 3 306
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,891评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,136评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,783评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,316评论 2 342

推荐阅读更多精彩内容