电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

詳解Java實現分治算法

瀏覽:89日期:2022-08-10 14:29:22
目錄一、前言二、分治算法介紹三、分治算法經典問題3.1、二分搜索3.2、快速排序3.3、歸并排序(逆序數)3.4、最大子序列和3.5、最近點對四、結語一、前言

在學習分治算法之前,問你一個問題,相信大家小時候都有存錢罐的經歷,父母親人如果給錢都會往自己的寶藏中存錢,我們每隔一段時間都會清點清點錢。但是一堆錢讓你處理起來你可能覺得很復雜,因為數據相對于大腦有點龐大了,并且很容易算錯,你可能會將它先分成幾個小份算,然后再疊加起來計算總和就獲得這堆錢的總數了

詳解Java實現分治算法

當然如果你覺得各個部分錢數量還是太大,你依然可以進行劃分然后合并,我們之所以這么多是因為:

計算每個小堆錢的方式和計算最大堆錢的方式是相同的(區別在于體量上) 然后大堆錢總和其實就是小堆錢結果之和。這樣其實就有一種分治的思想。二、分治算法介紹

分治算法是用了分治思想的一種算法,什么是分治?

分治,字面上的解釋是“分而治之”,就是把一個復雜的問題分成兩個或更多的相同或相似的子問題,再把子問題分成更小的子問題……直到最后子問題可以簡單的直接求解,原問題的解即子問題的解的合并。在計算機科學中,分治法就是運用分治思想的一種很重要的算法。分治法是很多高效算法的基礎,如排序算法(快速排序,歸并排序),傅立葉變換(快速傅立葉變換)等等。

將父問題分解為子問題同等方式求解,這和遞歸的概念很吻合,所以在分治算法通常以遞歸的方式實現(當然也有非遞歸的實現方式)。分治算法的描述從字面上也很容易理解,分、治其實還有個合并的過程:

分(Divide):遞歸解決較小的問題(到終止層或者可以解決的時候停下) 治(Conquer):遞歸求解,如果問題夠小直接求解。 合并(Combine):將子問題的解構建父類問題

一般分治算法在正文中分解為兩個即以上的遞歸調用,并且子類問題一般是不想交的(互不影響)。當求解一個問題規模很大很難直接求解,但是規模較小的時候問題很容易求解并且這個問題并且問題滿足分治算法的適用條件,那么就可以使用分治算法。

詳解Java實現分治算法

那么采用分治算法解決的問題需要 滿足那些條件(特征) 呢?

1 . 原問題規模通常比較大,不易直接解決,但問題縮小到一定程度就能較容易的解決。

2 . 問題可以分解為若干規模較小、求解方式相同(似)的子問題。且子問題之間求解是獨立的互不影響。

3 . 合并問題分解的子問題可以得到問題的解。

你可能會疑惑分治算法和遞歸有什么關系?其實分治重要的是一種思想,注重的是問題分、治、合并的過程。而遞歸是一種方式(工具),這種方式通過方法自己調用自己形成一個來回的過程,而分治可能就是利用了多次這樣的來回過程。

三、分治算法經典問題

對于分治算法的經典問題,重要的是其思想,因為我們大部分借助遞歸去實現,所以在代碼實現上大部分都是很簡單,而本篇也重在講述思想。

分治算法的經典問題,個人將它分成兩大類:子問題完全獨立和子問題不完全獨立。

1 . 子問題完全獨立就是原問題的答案可完全由子問題的結果推出。

2 . 子問題不完全獨立,有些區間類的問題或者跨區間問題使用分治可能結果跨區間,在考慮問題的時候需要仔細借鑒下。

3.1、二分搜索

二分搜索是分治的一個實例,只不過二分搜索有著自己的特殊性

序列有序 結果為一個值

正常二分將一個完整的區間分成兩個區間,兩個區間本應單獨找值然后確認結果,但是通過有序的區間可以直接確定結果在那個區間,所以分的兩個區間只需要計算其中一個區間,然后繼續進行一直到結束。實現方式有遞歸和非遞歸,但是非遞歸用的更多一些:

public int searchInsert(int[] nums, int target) { if(nums[0]>=target)return 0;//剪枝 if(nums[nums.length-1]==target)return nums.length-1;//剪枝 if(nums[nums.length-1]<target)return nums.length; int left=0,right=nums.length-1; while (left<right) { int mid=(left+right)/2; if(nums[mid]==target) return mid; else if (nums[mid]>target) { right=mid; } else { left=mid+1; } } return left;}3.2、快速排序

快排也是分治的一個實例,快排每一趟會選定一個數,將比這個數小的放左面,比這個數大的放右面,然后遞歸分治求解兩個子區間,當然快排因為在分的時候就做了很多工作,當全部分到最底層的時候這個序列的值就是排序完的值。這是一種分而治之的體現。

詳解Java實現分治算法

public void quicksort(int [] a,int left,int right){ int low=left; int high=right; //下面兩句的順序一定不能混,否則會產生數組越界!!!very important!!! if(low>high)//作為判斷是否截止條件 return; int k=a[low];//額外空間k,取最左側的一個作為衡量,最后要求左側都比它小,右側都比它大。 while(low<high)//這一輪要求把左側小于a[low],右側大于a[low]。 { while(low<high&&a[high]>=k)//右側找到第一個小于k的停止 { high--; } //這樣就找到第一個比它小的了 a[low]=a[high];//放到low位置 while(low<high&&a[low]<=k)//在low往右找到第一個大于k的,放到右側a[high]位置 { low++; } a[high]=a[low]; } a[low]=k;//賦值然后左右遞歸分治求之 quicksort(a, left, low-1); quicksort(a, low+1, right);}3.3、歸并排序(逆序數)

快排在分的時候做了很多工作,而歸并就是相反,歸并在分的時候按照數量均勻分,而合并時候已經是兩兩有序的進行合并的,因為兩個有序序列O(n)級別的復雜度即可得到需要的結果。而逆序數在歸并排序基礎上變形同樣也是分治思想求解。

詳解Java實現分治算法

private static void mergesort(int[] array, int left, int right) { int mid=(left+right)/2; if(left<right) { mergesort(array, left, mid); mergesort(array, mid+1, right); merge(array, left,mid, right); }}private static void merge(int[] array, int l, int mid, int r) { int lindex=l;int rindex=mid+1; int team[]=new int[r-l+1]; int teamindex=0; while (lindex<=mid&&rindex<=r) {//先左右比較合并 if(array[lindex]<=array[rindex]) { team[teamindex++]=array[lindex++]; } else { team[teamindex++]=array[rindex++]; } } while(lindex<=mid)//當一個越界后剩余按序列添加即可 { team[teamindex++]=array[lindex++]; } while(rindex<=r) { team[teamindex++]=array[rindex++]; } for(int i=0;i<teamindex;i++) { array[l+i]=team[i]; }}3.4、最大子序列和

最大子序列和的問題我們可以使用動態規劃的解法,但是也可以使用分治算法來解決問題,但是最大子序列和在合并的時候并不是簡單的合并,因為子序列和涉及到一個長度的問題,所以正確結果不一定全在最左側或者最右側,而可能出現結果的區域為:

完全在中間的左側 完全在中間的右側 包含中間左右兩個節點的一個序列

用一張圖可以表示為:

詳解Java實現分治算法

所以在具體考慮的時候需要將無法遞歸得到結果的中間那個最大值串的結果也算出來參與左側、右側值得比較。

力扣53. 最大子序和在實現的代碼為:

public int maxSubArray(int[] nums) { int max=maxsub(nums,0,nums.length-1); return max;}int maxsub(int nums[],int left,int right){ if(left==right)return nums[left]; int mid=(left+right)/2; int leftmax=maxsub(nums,left,mid);//左側最大 int rightmax=maxsub(nums,mid+1,right);//右側最大 int midleft=nums[mid];//中間往左 int midright=nums[mid+1];//中間往右 int team=0; for(int i=mid;i>=left;i--) {team+=nums[i];if(team>midleft) midleft=team; } team=0; for(int i=mid+1;i<=right;i++) {team+=nums[i];if(team>midright) midright=team; } int max=midleft+midright;//中間的最大值 if(max<leftmax)max=leftmax; if(max<rightmax)max=rightmax; return max;}3.5、最近點對

最近點對是一個分治非常成功的運用之一。在二維坐標軸上有若干個點坐標,讓你求出最近的兩個點的距離,如果讓你直接求那么枚舉暴力是個非常非常大的計算量,我們通常采用分治的方法來優化這種問題。

詳解Java實現分治算法

如果直接分成兩部分分治計算你肯定會發現如果最短的如果一個在左一個在右會出現問題。我們可以優化一下。

在具體的優化方案上,按照x或者y的維度進行考慮,將數據分成兩個區域,先分別計算(按照同方法)左右區域內最短的點對。然后根據這個兩個中較短的距離向左和向右覆蓋,計算被覆蓋的左右點之間的距離,找到最小那個距離與當前最短距離比較即可。

詳解Java實現分治算法

這樣你就可以發現就這個一次的操作(不考慮子情況),左側紅點就避免和右側大部分紅點進行距離計算(O(n2)的時間復雜度)。事實上,在進行左右區間內部計算的時候,它其實也這樣遞歸的進行很多次分治計算。如圖所示:

詳解Java實現分治算法

這樣下去就可以節省很多次的計算量。

但是這種分治會存在一種問題就是二維坐標可能點都聚集某個方法某條軸那么可能效果并不明顯(點都在x=2附近對x分割作用就不大),需要注意一下。

ac的代碼為:

import java.io.BufferedReader;import java.io.IOException;import java.io.InputStreamReader;import java.io.OutputStreamWriter;import java.io.PrintWriter;import java.io.StreamTokenizer;import java.util.ArrayList;import java.util.Arrays;import java.util.Comparator;import java.util.List;public class Main { static int n; public static void main(String[] args) throws IOException {StreamTokenizer in=new StreamTokenizer(new BufferedReader(new InputStreamReader(System.in)));PrintWriter out = new PrintWriter(new OutputStreamWriter(System.out));//List<node>list=new ArrayList(); while(in.nextToken()!=StreamTokenizer.TT_EOF) { n=(int)in.nval;if(n==0) {break;} node no[]=new node[n]; for(int i=0;i<n;i++) { in.nextToken();double x=in.nval; in.nextToken();double y=in.nval;// list.add(new node(x,y)); no[i]=new node(x,y); } Arrays.sort(no, com); double min= search(no,0,n-1); out.println(String.format('%.2f', Math.sqrt(min)/2));out.flush(); } } private static double search(node[] no, int left,int right) {int mid=(right+left)/2;double minleng=0;if(left==right) {return Double.MAX_VALUE;}else if(left+1==right) {minleng= (no[left].x-no[right].x)*(no[left].x-no[right].x)+(no[left].y-no[right].y)*(no[left].y-no[right].y);}else minleng= min(search(no,left,mid),search(no,mid,right));int ll=mid;int rr=mid+1;while(no[mid].y-no[ll].y<=Math.sqrt(minleng)/2&&ll-1>=left) {ll--;}while(no[rr].y-no[mid].y<=Math.sqrt(minleng)/2&&rr+1<=right) {rr++;}for(int i=ll;i<rr;i++){ for(int j=i+1;j<rr+1;j++) {double team=0;if(Math.abs((no[i].x-no[j].x)*(no[i].x-no[j].x))>minleng) {continue;}else{ team=(no[i].x-no[j].x)*(no[i].x-no[j].x)+(no[i].y-no[j].y)*(no[i].y-no[j].y); if(team<minleng)minleng=team;} }}return minleng;} private static double min(double a, double b) {// TODO 自動生成的方法存根return a<b?a:b; } static Comparator<node>com=new Comparator<node>() {@Overridepublic int compare(node a1, node a2) { // TODO 自動生成的方法存根 return a1.y-a2.y>0?1:-1;}}; static class node {double x;double y;public node(double x,double y){ this.x=x; this.y=y;} }}四、結語

到這里,分治算法就講這么多了,因為分治算法重要在于理解其思想,還有一些典型的分治算法解決的問題,例如大整數乘法、Strassen矩陣乘法、棋盤覆蓋、線性時間選擇、循環賽日程表、漢諾塔等問題你可以自己研究其分治的思想和原理。

以上就是詳解Java實現分治算法的詳細內容,更多關于詳解Java實現分治算法的資料請關注好吧啦網其它相關文章!

標簽: Java
相關文章:
主站蜘蛛池模板: 合肥网络推广_合肥SEO网站优化-安徽沃龙First | 网优资讯-为循环资源、大宗商品、工业服务提供资讯与行情分析的数据服务平台 | 泰国专线_泰国物流专线_广州到泰国物流公司-泰廊曼国际 | 武汉高温老化房,恒温恒湿试验箱,冷热冲击试验箱-武汉安德信检测设备有限公司 | 捷码低代码平台 - 3D数字孪生_大数据可视化开发平台「免费体验」 | 干粉砂浆设备-干粉砂浆生产线-干混-石膏-保温砂浆设备生产线-腻子粉设备厂家-国恒机械 | 金属管浮子流量计_金属转子流量计厂家-淮安润中仪表科技有限公司 | 硬度计_影像测量仪_维氏硬度计_佛山市精测计量仪器设备有限公司厂家 | 纸张环压仪-纸张平滑度仪-杭州纸邦自动化技术有限公司 | 小型气象站_便携式自动气象站_校园气象站-竞道气象设备网 | 999范文网_优质范文下载写作帮手 | 电渗析,废酸回收,双极膜-山东天维膜技术有限公司 | 排烟防火阀-消防排烟风机-正压送风口-厂家-价格-哪家好-德州鑫港旺通风设备有限公司 | 恒温振荡混匀器-微孔板振荡器厂家-多管涡旋混匀器厂家-合肥艾本森(www.17world.net) | 高尔夫球杆_高尔夫果岭_高尔夫用品-深圳市新高品体育用品有限公司 | 风信子发稿-专注为企业提供全球新闻稿发布服务| 真石漆,山东真石漆,真石漆厂家,真石漆价格-山东新佳涂料有限公司 | 石家庄救护车出租_重症转院_跨省跨境医疗转送_活动赛事医疗保障_康复出院_放弃治疗_腾康26年医疗护送转诊团队 | 溶氧传感器-pH传感器|哈美顿(hamilton)| 神超官网_焊接圆锯片_高速钢锯片_硬质合金锯片_浙江神超锯业制造有限公司 | 天津拓展_天津团建_天津趣味运动会_天津活动策划公司-天津华天拓展培训中心 | 传动滚筒_厂家-淄博海恒机械制造厂 | 慈溪麦田广告公司,提供慈溪广告设计。 | 碎石机设备-欧版反击破-欧版颚式破碎机(站)厂家_山东奥凯诺机械 高低温试验箱-模拟高低温试验箱订制-北京普桑达仪器科技有限公司【官网】 | 电动葫芦|手拉葫芦|环链电动葫芦|微型电动葫芦-北京市凌鹰起重机械有限公司 | 热回收盐水机组-反应釜冷水机组-高低温冷水机组-北京蓝海神骏科技有限公司 | 字典-新华字典-在线字典查字-字典趣 | 福建珂朗雅装饰材料有限公司「官方网站」 | 智能门锁电机_智能门锁离合器_智能门锁电机厂家-温州劲力智能科技有限公司 | 全自动真空上料机_粉末真空上料机_气动真空上料机-南京奥威环保科技设备有限公司 | 钢格板_钢格栅_格栅板_钢格栅板 - 安平县鑫拓钢格栅板厂家 | 肉嫩度仪-凝胶测试仪-国产质构仪-气味分析仪-上海保圣实业发展有限公司|总部 | 风淋室生产厂家报价_传递窗|送风口|臭氧机|FFU-山东盛之源净化设备 | 磁力抛光机_磁力研磨机_磁力去毛刺机-冠古设备厂家|维修|租赁【官网】 | 在线钠离子分析仪-硅酸根离子浓度测定仪-油液水分测定仪价格-北京时代新维测控设备有限公司 | 中式装修设计_室内中式装修_【云臻轩】中式设计机构 | 双效节能浓缩器-热回流提取浓缩机组-温州市利宏机械 | 锡膏喷印机-全自动涂覆机厂家-全自动点胶机-视觉点胶机-深圳市博明智控科技有限公司 | 圆盘鞋底注塑机_连帮鞋底成型注塑机-温州天钢机械有限公司 | 北京包装设计_标志设计公司_包装设计公司-北京思逸品牌设计 | 天津次氯酸钠酸钙溶液-天津氢氧化钠厂家-天津市辅仁化工有限公司 |