数据结构_基础结构

链表

链表是一种数据结构,相对于数组而言,插入和删除的开销比较小,而查找的代价较大.以下我们实现双向链表:

 public class MyList<T> {

    private  Node<T> head;
    private  Node<T> tail;
    private int size;
    
    
    //初始化头结点和尾节点
    public MyList(){
        head=new Node<T>(null, null, tail);
        tail=new Node<T>(null, head, null);
    }
    
    //获得大小
    public int size(){
        return size;
    }
    
    //增加方法
    public void add(T x){
        this.addBefore(tail, x);
    }
    
    public void addBefore(int index,T x){
        this.addBefore(this.getNode(index), x);
    }
    
    public void addBefore(Node<T> p,T x){
        Node<T> newNode=new Node<T>(x, p.prev, p);
        newNode.prev.next=newNode;
        p.prev=newNode;
        size++;
    }
    
    //查找方法 效率应该高点
    public Node<T> getNode(int index){
        Node<T> p;
        if(index<size()/2){
            p=head.next;
            for(int i=0;i<index;i++){p=p.next;}
        }else{
            p=tail;
            for(int i=size();i>index;i--){p=p.prev;}
        }
        return p;
    }
    
    //删除    
    public T remove(int index){
        return remove(this.getNode(index));
    }
    
    public T remove(Node<T> p){
        p.next.prev=p.prev;
        p.prev.next=p.next;
        size--;
        return p.data;
    }
    
    //修改
    public void set(int index,T newData){
        Node<T> p=this.getNode(index);
        p.data=newData;
    }
    
    //显示全部
    public void show(){
        if(size()>0){
            Node<T> p=head.next;
            while(p!=tail){
                System.out.println(p.data);
                p=p.next;
            }
        }
    }
    
    
    //节点类
    private static class Node<T>{
        public  T data;
        public Node<T> prev;
        public Node<T> next;
        
        public Node(T d,Node<T> p,Node<T> n){
            data=d;
            prev=p;
            next=n;
        }
    }
    
    
}

在我写的这个双向链表中头节点head和尾节点tail是空出来的,不存放任何数据.若是要改造成双向循环链表的话就不合适了,比如最后一个元素要next三下越过tail和head才循环到第一个元素.
于是稍作改造(注意空指针),双向循环链表:

 package com.fredal.structure;

public class MyCList<T> {

    private Node<T> head;
    private Node<T> tail;
    private int size;

    // 初始化头结点和尾节点
    public MyCList() {
        head = new Node<T>(null, tail, tail);
        tail = new Node<T>(null, head, head);
        // 解决空指针
        head.prev = tail;
        head.next = tail;
        tail.prev = head;
        tail.next = head;
    }

    // 获得大小
    public int size() {
        return size;
    }

    // 增加方法
    public void add(T x) {
        if (head.data == null) {
            head.data = x;
            size++;
        } else if (tail.data == null) {
            tail.data = x;
            size++;
        } else {
            Node<T> p = this.addBefore(head, x);
            tail = p;
        }

    }

    public Node<T> addBefore(int index, T x) {
        return this.addBefore(this.getNode(index), x);
    }

    public Node<T> addBefore(Node<T> p, T x) {
        Node<T> newNode = new Node<T>(x, p.prev, p);
        newNode.prev.next = newNode;
        p.prev = newNode;
        size++;
        return newNode;
    }

    // 查找方法 效率应该高点
    public Node<T> getNode(int index) {
        Node<T> p;
        if (index < size() / 2) {
            p = head;
            for (int i = 0; i < index; i++) {
                p = p.next;
            }
        } else {
            p = tail;
            for (int i = size() - 1; i > index; i--) {
                p = p.prev;
            }
        }
        return p;
    }

    // 删除
    public T remove(int index) {
        return remove(this.getNode(index));
    }

    public T remove(Node<T> p) {
        p.next.prev = p.prev;
        p.prev.next = p.next;
        size--;
        if (p == head) {
            head = p.next;
        }
        if (p == tail) {
            tail = p.prev;
        }
        return p.data;
    }

    // 修改
    public void set(int index, T newData) {
        Node<T> p = this.getNode(index);
        p.data = newData;
    }

    // 显示全部
    public void show() {
        if (size() > 0) {
            Node<T> p = head;
            do {
                System.out.println(p.data);
                p = p.next;
            } while (p != head);
        }
    }

    // 节点类
    private static class Node<T> {
        private T data;
        private Node<T> prev;
        private Node<T> next;

        public Node(T d, Node<T> p, Node<T> n) {
            data = d;
            prev = p;
            next = n;
        }
    }

}

栈Stack是一个表,先进后出,有两种实现方式:由链表或者数组来实现.这里我们选用数组实现做例子,比较简单:

 package com.fredal.structure;

public class MyStack<T> {

    // Java 不支持泛型数组,如需使用,请使用Java提供的容器
    private Object[] stack;

    // 栈的默认初始大小
    private static final int INIT_SIZE = 2;

    // 栈顶索引
    private int index;

    public MyStack() {
        stack = new Object[INIT_SIZE];
        index = -1;
    }

    /**
     * 构造方法
     * 
     * @param initSize
     *            栈的初始大小
     */
    public MyStack(int initSize) {
        if (initSize < 0) {
            throw new IllegalArgumentException();
        }
        stack = new Object[initSize];
        index = -1;
    }

    /**
     * 出栈操作
     * 
     * @return 栈顶对象
     */
    public T pop() {
        if (!isEmpty()) {
            T temp = peek();
            stack[index--] = null;
            return temp;
        }
        return null;
    }

    /**
     * 入栈操作
     * 
     * @param obj
     *            等待入栈的对象
     */
    public void push(T obj) {
        if (isFull()) {
            Object[] temp = stack;
            // 扩容操作,和arraylist的一样,如果栈满,则创建空间为当前栈空间两倍的栈
            stack = new Object[2 * stack.length];
            System.arraycopy(temp, 0, stack, 0, temp.length);
        }
        stack[++index] = obj;
    }

    /**
     * 查看栈顶对象
     * 
     * @return 栈顶对象
     */
    public T peek() {
        if (!isEmpty()) {
            return (T) stack[index];
        }
        return null;
    }

    /**
     * 查看栈是否为空
     * 
     * @return 如果栈为空返回true,否则返回false
     */
    public boolean isEmpty() {
        return index == -1;
    }

    /**
     * 查看栈是否满
     * 
     * @return 如果栈满返回true,否则返回false
     */
    public boolean isFull() {
        return index >= stack.length - 1;
    }
}

应用的比较多的一个是检测平衡符号,这个比较简单,就是按顺序push按顺序pop看看是否相同.另外一个是逆波兰表达式,写一段代码,包括中序表达式转化为逆波兰式,以及逆波兰式的计算:

 package com.fredal.structure;

import java.util.ArrayList;
import java.util.List;
import java.util.Stack;

public class Nbl {

    private static Stack s1 = new Stack();// 逆波兰表达式的栈
    private static Stack s2 = new Stack();// 运算栈

    //将字符串转化成中序list
    public static List<String> format(String s) {
        List<String> ls = new ArrayList<String>();// 存储中序表达式
        int i = 0;
        String str;
        char c;
        do {
            if ((c = s.charAt(i)) < 48 || (c = s.charAt(i)) > 57) {
                ls.add("" + c);
                i++;
            } else {
                str = "";
                while (i < s.length() && (c = s.charAt(i)) >= 48
                        && (c = s.charAt(i)) <= 57) {
                    str += c;
                    i++;
                }
                ls.add(str);
            }

        } while (i < s.length());
        return ls;
    }

    //将中序表达式转化为逆波兰式
    public static List<String> parse(List<String> ls) {
        List<String> lss = new ArrayList<String>();
        for (String ss : ls) {
            if (ss.matches("\\d+")) {
                lss.add(ss);// 不是运算符就加入list
            } else if (ss.equals("(")) {
                s1.push(ss);//碰到"("直接进栈
            } else if(ss.equals(")")){
                while(!s1.peek().equals("(")){//将直到"("的符号全弹出
                    lss.add((String) s1.pop());//加入list
                }
                s1.pop();//弹出"("
            }else{
                while(s1.size()!=0&&getValue((String) s1.peek())>=getValue(ss)){//比它优先级高的全弹出
                    lss.add((String) s1.pop());
                }
                s1.push(ss);
            }
        }
        while(s1.size()!=0){//结束之后全部弹出
            lss.add((String) s1.pop());        
        }
        return lss;
    }

    //获取运算符优先级
    private static int getValue(String s) {
         if (s.equals("+")) {
                return 1;
            } else if (s.equals("-")) {
                return 1;
            } else if (s.equals("*")) {
                return 2;
            } else if (s.equals("/")) {
                return 2;
            }
            return 0;
    }

    // 对逆波兰表达式进行求值
    public static int calculate(List<String> ls) {
        for (String s : ls) {
            if (s.matches("\\d+")) {
                s2.push(s);// 不是运算符就入栈
            } else {
                int b = Integer.parseInt((String) s2.pop());
                int a = Integer.parseInt((String) s2.pop());
                if (s.equals("+")) {
                    a = a + b;
                } else if (s.equals("-")) {
                    a = a - b;
                } else if (s.equals("*")) {
                    a = a * b;
                } else if (s.equals("/")) {
                    a = a / b;
                }
                s2.push("" + a);
            }
        }
        return Integer.parseInt((String) s2.pop());
    }
}

队列

队列(queue)也是表,使用队列时插入在一端进行而删除在另一端进行.
基本操作是enqueue入队,在表的末端插入元素.dequeue出队,删除并返回表的开头.
用数组实现队列会有潜在问题,就是队列满了之后在入队一次back的下标会指向不存在的位置,解决这个问题我们用循环队列的方式.

  package com.fredal.structure;

public class MyQueue<T> {
    
    private int INITIAL_SIZE=10;
    private int capacity;//数组长度
    private Object[] elementData;
    private int front=0;
    private int back=0;
    
    public MyQueue(){
        capacity=INITIAL_SIZE;
        elementData=new Object[capacity];
    }
    //以一个初始化元素来创建循环队列
    public MyQueue(T element){
        this();
        elementData[0]=element;
        back++;
    }
    
    //以指定长度的数组来创建循环队列  
    public MyQueue(T element,int initSize){  
        this.capacity=initSize;  
        elementData=new Object[capacity];  
        elementData[0]=element;  
        back++;  
    }  
    
    //判断循环队列是否为空
    public boolean isEmpty(){
        return back==front&&elementData[back]==null;
    }
    
    //获取循环队列的大小
    public int length(){
        if(isEmpty()){
            return 0;
        }else{
            return back>front?back-front:capacity-(front-back);
        }       
    }
    
    //插入队列
    public void add(T element){
        if(back==front&&elementData[front]!=null){
            throw new IndexOutOfBoundsException("队列已满");
        }
        elementData[back++]=element;
        //back指针到头就循环
        back=(back==capacity)?0:back;
    }
    
    //移出队列
    public T remove(){
        if(isEmpty()){
            throw new IndexOutOfBoundsException("队列已空");
        }
        T oldValue=(T)elementData[front];
        elementData[front++]=null;
        //如果front到头 就循环
        front=front==capacity?0:front;
        return oldValue;
    }
    
    //显示
    public void show(){
        if(isEmpty()){
            System.out.println("[]");
        }else{
            if(front<back){
                 StringBuilder sb=new StringBuilder("[");  
                 for(int i=front;i<back;i++){  
                     sb.append(elementData[i].toString()+",");  
                 }  
                 int len=sb.length();  
                String s= sb.delete(len-1, len).append("]").toString();  
                System.out.println(s);
            }else{
                StringBuilder sb=new StringBuilder("[");  
                for(int i=front;i<capacity;i++){  
                    sb.append(elementData[i].toString()+",");  
                }  
                for(int i=0;i<back;i++){  
                    sb.append(elementData[i].toString()+",");  
                }  
                int len=sb.length();  
                String s= sb.delete(len-1, len).append("]").toString(); 
                System.out.println(s);
                
            }
        }
    }
}

先说二叉树,二叉树表示每个节点都不能有多于两个的儿子.二叉树的一个重要应用是二叉排序树ADT:

 package com.fredal.structure;

public class MyTree<T extends Comparable<T>> {

    // 节点类
    private static class BNode<T> {
        T element;
        BNode<T> left;
        BNode<T> right;

        public BNode(T element) {
            this(element, null, null);
        }

        public BNode(T element, BNode<T> lt, BNode<T> rt) {
            this.element = element;
            left = lt;
            right = rt;
        }
    }

    // 插入节点之后
    public BNode<T> insert(T x, BNode<T> t) {
        if (t == null) {
            return new BNode<T>(x, null, null);
        }
        int result = x.compareTo(t.element);
        if (result < 0)
            t.left = insert(x, t.left);
        else if (result > 0)
            t.right = insert(x, t.right);
        return t;
    }

    // 删除
    public BNode<T> remove(T x, BNode<T> t) {
        if (t == null)
            return t;
        int result = x.compareTo(t.element);
        if (result < 0)
            t.left = remove(x, t.left);
        else if (result > 0)
            t.right = remove(x, t.right);
        else if (t.left != null && t.right != null) {// 找到了 两个孩子
            t.element = findMin(t.right).element;// 虽然是奇怪的删除策略
            t.right = remove(t.element, t.right);
        } else
            // 找到了 一个孩子或没有孩子
            t = (t.left != null) ? t.left : t.right;
        return t;
    }

    // 寻找最小值
    public BNode<T> findMin(BNode<T> t) {
        if (t == null)
            return null;
        else if (t.left == null)
            return t;
        return findMin(t.left);
    }

    // 寻找最大值
    public BNode<T> findMax(BNode<T> t) {
        if (t == null)
            return null;
        else if (t.right == null)
            return t;
        return findMax(t.right);
    }

    // 获得树的高度
    public int getHeight(BNode<T> t) {
        int a = 0;
        int b = 0;
        if (t.left != null)
            a = getHeight(t.left);
        if (t.right != null)
            b = getHeight(t.right);
        return (a > b ? a : b) + 1;
    }

    // 是否包含某个元素
    public boolean contains(T x, BNode<T> t) {
        if (t == null)
            return false;
        int result = x.compareTo(t.element);
        if (result < 0)
            return contains(x, t.left);
        else if (result > 0)
            return contains(x, t.right);
        else
            return true;
    }

    // 显示 中序遍历了
    public void show(BNode<T> t) {
        if (t.left != null)
            show(t.left);
        System.out.println(t.element);
        if (t.right != null)
            show(t.right);
    }
}

还有一个就是所谓的表达式树,一个正常的表达式构建的表达式树如果采取中序遍历就是得到正常的表达式,如果采取后序遍历呢就会产生上一节说的那个逆波兰表达式,也叫后缀表达式.
基本上这个树的作用就是把后序表达式变成中序表达式,之前那段代码逆作用..
用后缀表达式构建树,再中序遍历之

  package com.fredal.structure;

import java.util.Stack;

public class ExpTree<T> {
    
    // 节点类
    private static class BNode<T> {
        T element;
        BNode<T> left;
        BNode<T> right;

        public BNode(T element) {
            this(element, null, null);
        }

        public BNode(T element, BNode<T> lt, BNode<T> rt) {
            this.element = element;
            left = lt;
            right = rt;
        }
    }
    
    public BNode<Character> bulid(String exp){
        char c;
        BNode<Character> newNode,newLeft,newRight;
        Stack<BNode<Character>> stack=new Stack<BNode<Character>>();
        int i=0;
        int len=exp.length();
        
        while(i!=len){
            while(exp.charAt(i)==' '||exp.charAt(i)=='\t'){
                i++;
            }
            if(i==len) break;
            c=exp.charAt(i);
            i++;
            if(c=='+'||c=='-'||c=='*'||c=='/'){//碰到运算符就把前两个弹出来重新构建一下树再入栈
                newRight=stack.pop();
                newLeft=stack.pop();            
                newNode=new BNode<Character>(c, newLeft, newRight);
                stack.push(newNode);
            }else{
                newNode=new BNode<Character>(c);//不是运算符直接入栈
                stack.push(newNode);
            }
        }
        
        if(!stack.isEmpty()){
            return stack.pop();//弹出整棵树
        }else{
            return null;
        }
        
    }
    
    //中序输出
    public void show(BNode<T> t){
        if(t!=null){
            show(t.left);
            System.out.print(t.element+" ");
            show(t.right);
        }
    }
    
}

表达式树确实是这样的,中序遍历也没错,不过要变成正常的表达式仍然是不够的,还需要考虑优先级去加括号,偷懒方法当然是全部加上括号,这儿不写了.
然后写一下多叉树的实现吧,这儿写习惯了就给node外面加了一层包装类,这样添加的时候还是比较麻烦的,待优化吧.

 package com.fredal.structure;

import java.util.ArrayList;
import java.util.List;

public class CTree<T> {

    // 节点类
    private static class CNode<T> {
        T element;
        CNode<T> parent;// 父节点
        List<CNode<T>> children = new ArrayList<CNode<T>>();// 孩子节点
        CNode<T> leftBrother;// 左边第一个兄弟节点
        CNode<T> rightBrother;// 右边第一个兄弟节点

        public CNode(T t, CNode<T> parent) {
            element = t;
            this.parent = parent;
        }
    }

    // 插入
    public CNode<T> insert(T x, CNode<T> t) {
        CNode<T> newNode = new CNode<T>(x, t);
        List<CNode<T>> children = t.children;
        if (children.size() == 0) {
            children.add(newNode);
        } else {
            CNode<T> lastNode = children.get(children.size() - 1);
            lastNode.rightBrother = newNode;
            newNode.leftBrother = lastNode;
            children.add(newNode);
        }

        return newNode;
    }

    // 得到根节点
    public CNode<T> getRoot(CNode<T> t) {
        while (t.parent != null) {
            t = t.parent;
        }
        return t;
    }

    // 得到父节点
    public CNode<T> getParent(CNode<T> t) {
        if (t.parent != null) {
            return t.parent;
        }
        return null;
    }

    // 得到子节点
    public List<CNode<T>> getChildren(CNode<T> t) {
        if (t.children != null) {
            return t.children;
        }
        return null;
    }

    // 左节点
    public CNode<T> getLeftBrother(CNode<T> t) {
        if (t.leftBrother != null) {
            return t.leftBrother;
        }
        return null;
    }

    // 右节点
    public CNode<T> getRightBrother(CNode<T> t) {
        if (t.rightBrother != null) {
            return t.rightBrother;
        }
        return null;
    }

    // 这个属于先序遍历
    public void show(CNode<T> t) {
        if (t != null) {
            System.out.print(t.element + " ");
            List<CNode<T>> children = t.children;
            for (int i = 0; i < children.size(); i++) {
                show(children.get(i));
            }
        }
    }
}

散列

散列也叫哈希,会遇到散列冲突问题,解决冲突最常用的是分离链接法,简单来说就是将散列到同一个值的所有元素都插入到一个链表中去,来实现吧.

 package com.fredal.structure;

import java.util.LinkedList;
import java.util.List;

public class MyHashTable<T> {

    private static final int DEFAULT_SIZE=100;
    private int size;
    private List<T>[] lists;
    
    public MyHashTable(int size){
        this.size=size;
        lists=new LinkedList[size];//初始化链表数组
        for (int i = 0; i < lists.length; i++) {
            lists[i]=new LinkedList<T>();
        }
    }
    public MyHashTable(){
        this(DEFAULT_SIZE);
    }
    
    private int myHash(T x){
        int hashVal=x.hashCode();
        hashVal%=lists.length;
        if(hashVal<0){
            hashVal+=lists.length;
        }
        return hashVal;
    }
    
    //是否包含 使用自带的就好 ~
    public boolean contains(T x){
        List<T> list=lists[myHash(x)];
        return list.contains(x);
    }
    
    //插入方法
    public void insert(T x){
        List<T> list=lists[myHash(x)];
        if(!list.contains(x)){
            list.add(x);
            size++;
        }
    }
    //移除方法
    public void remove(T x){
        List<T> list=lists[myHash(x)];
        if(list.contains(x)){
            list.remove(x);
            size--;
        }
    }
}

这里并没有考虑再散列,因为场景不复杂,再散列以后再谈.
还有一种散列表叫探测散列表,也以后再说...

堆也叫优先队列,是允许至少下列两种操作的数据结构,插入和删除并返回最小值.插入(insert)相当于enqueue(入队),而删除最小值(deleteMin)则等价于队列运算dequeue(出队).
我们要讲的叫二叉堆,除了是一颗完整二叉树外还要保持其堆序性质.例如最小二叉堆即,最小元在根上,而任意节点都小于它的所有后裔.
我们通过代码模拟二叉堆

  package com.fredal.structure;

public class MyHeap<T extends Comparable<? super T>> {

    private static final int DEFAULT_CAPACITY=10;
    private int currentSize;//堆元素数量
    private T[] array;//堆数组
    
    public MyHeap(){
        this(DEFAULT_CAPACITY);
    }
    
    public MyHeap(int capacity){
        currentSize=0;
        array=(T[]) new Comparable[capacity+1];
    }
    
    //对已给定的数组建堆初始化
    public MyHeap(T [] items){
        currentSize=items.length;
        array=(T[])new Comparable[(currentSize+2)*11/10];
        int i=1;
        for(T item:items){
            array[i++]=item;
        }
        buildHeap();
    }

    private void buildHeap() {
        for(int i=currentSize/2;i>0;i--){
            percolateDown(i);//下滤
        }
    }

    //下滤 保持堆特性
    private void percolateDown(int i) {
        int child;
        T temp =array[i];
        for(;i*2<=currentSize;i=child){
            child=i*2;
            if(child!=currentSize&& array[child+1].compareTo(array[child])<0){
                child++;
            }
            if(array[child].compareTo(temp)<0){
                array[i]=array[child];
            }else
                break;
        }
        array[i]=temp;
    }
    
    //插入操作
    public void insert(T x){
        if(currentSize==array.length-1){
            enlargeArray(array.length*2+1);
        }
        
        //上滤操作
        int i=++currentSize;
        for(array[0]=x;x.compareTo(array[i/2])<0;i/=2){
            array[i]=array[i/2];
        }
        array[i]=x;
    }
    
    //扩容操作
    private void enlargeArray(int newSize){
        T[] old=array;
        array=(T[]) new Comparable[newSize];
        for(int i=0;i<old.length;i++){
            array[i]=old[i];
        }
    }
    
    //删除最小值
    public T deleteMin(){
        if(currentSize==0){
            throw new RuntimeException();
        }
        T min=findMin();
        array[1]=array[currentSize--];
        //下滤
        percolateDown(1);
        return min;
    }

    //寻找最小值
    public  T findMin() {
        if(currentSize==0){
            throw new RuntimeException();
        }
        return array[1];
    }
    
    //显示
    public void show(){
        for(int i=1;i<=currentSize;i++){
            System.out.print(array[i]+" ");
        }
    }
        
}

堆的应用有许多,堆排序是最重要的之一,我们在后面会讲到.
更多文章与相关下载请看扩展阅读

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

推荐阅读更多精彩内容

  • 1 序 2016年6月25日夜,帝都,天下着大雨,拖着行李箱和同学在校门口照了最后一张合照,搬离寝室打车去了提前租...
    RichardJieChen阅读 5,088评论 0 12
  • 第一章 绪论 什么是数据结构? 数据结构的定义:数据结构是相互之间存在一种或多种特定关系的数据元素的集合。 第二章...
    SeanCheney阅读 5,762评论 0 19
  • 因为之前就复习完数据结构了,所以为了保持记忆,整理了一份复习纲要,复习的时候可以看着纲要想具体内容。 树 树的基本...
    牛富贵儿阅读 6,855评论 3 10
  • 基于树实现的数据结构,具有两个核心特征: 逻辑结构:数据元素之间具有层次关系; 数据运算:操作方法具有Log级的平...
    yhthu阅读 4,267评论 1 5
  • 泪成殇,点滴坠,幻如花,痛相思。 忆曾经,樱花舞,共话伞,携相拥。 念今昔,孤影伴,信林荫,独追思。 惜时光,卷帜...
    小瑜密语阅读 314评论 0 0