iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >如何使用matlab遗传算法求解车间调度问题
  • 680
分享到

如何使用matlab遗传算法求解车间调度问题

2023-06-29 02:06:58 680人浏览 薄情痞子
摘要

这篇文章主要介绍了如何使用matlab遗传算法求解车间调度问题,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、车间调度简介1 车间调度定义车间调度是指根据产品制造的合理需求

这篇文章主要介绍了如何使用matlab遗传算法求解车间调度问题,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。

    一、车间调度简介

    1 车间调度定义

    车间调度是指根据产品制造的合理需求分配加工车间顺序,从而达到合理利用产品制造资源、提高企业经济效益的目的。车间调度问题从数学上可以描述为有n个待加工的零件要在m台机器上加工。问题需要满足的条件包括每个零件的各道工序使用每台机器不多于1次,每个零件都按照一定的顺序进行加工。

    2 传统作业车间调度

    传统作业车间带调度实例

    如何使用matlab遗传算法求解车间调度问题

    有若干工件,每个工件有若干工序,有多个加工机器,但是每道工序只能在一台机器上加工。对应到上面表格中的实例就是,两个工件,工件J1有三道工序,工序Q11只能在M3上加工,加工时间是5小时。
    约束是对于一个工件来说,工序的相对顺序不能变。O11->O12->O13。每时刻,每个工件只能在一台机器上加工;每个机器上只能有一个工件。
    调度的任务则是安排出工序的加工顺序,加工顺序确定了,因为每道工序只有一台机器可用,加工的机器也就确定了。
    调度的目的是总的完工时间最短(也可以是其他目标)。举个例子,比如确定了O21->O22->O11->O23->O12->O13的加工顺序之后,我们就可以根据加工机器的约束,计算出总的加工时间。
    M2加工O21消耗6小时,工件J2当前加工时间6小时。
    M1加工O22消耗9小时,工件J2当前加工时间6+9=15小时。
    M3加工O11消耗5小时,工件J1当前加工时间5小时。
    M4加工O23消耗7小时,工件J2加工时间15+7=22小时。
    M1加工O12消耗11小时,但是要等M1加工完O22之后才开始加工O12,所以工件J1的当前加工时间为max(5,9)+11=20小时。
    M5加工O13消耗8小时,工件J2加工时间20+8=28小时。
    总的完工时间就是max(22,28)=28小时。

    2 柔性作业车间调度

    柔性作业车间带调度实例(参考自高亮老师论文
    《改进遗传算法求解柔性作业车间调度问题》——机械工程学报)
    如何使用matlab遗传算法求解车间调度问题
    相比于传统作业车间调度,柔性作业车间调度放宽了对加工机器的约束,更符合现实生产情况,每个工序可选加工机器变成了多个,可以由多个加工机器中的一个加工。比如上表中的实例,J1的O12工序可以选择M2和M4加工,加工时间分别是8小时和4小时,但是并不一定选择M4加工,最后得出来的总的完工时间就更短,所以,需要调度算法求解优化

    相比于传统作业车间,柔性车间作业调度的调度任务不仅要确定工序的加工顺序,而且需要确定每道工序的机器分配。比如,确定了O21->O22->O11->O23->O12->O13的加工顺序,我们并不能相应工序的加工机器,所以还应该确定对应的[M1、M3、M5]->[M1、M2、M3]->[M1、M2、M3、M4、M5]->[M2、M3、M4、M5]->[M2、M4]->[M1、M3、M4、M5]的机器组合。调度的目的还是总的完工时间最短(也可以是其他目标,比如机器最大负荷最短、总的机器负荷最短)

    二、遗传算法简介

    1 遗传算法概述

    遗传算法(Genetic AlGorithm,GA)是进化计算的一部分,是模拟达尔文的遗传选择和自然淘汰的生物进化过程的计算模型,是一种通过模拟自然进化过程搜索最优解的方法。该算法简单、通用,鲁棒性强,适于并行处理。

    2 遗传算法的特点和应用

    遗传算法是一类可用于复杂系统优化的具有鲁棒性的搜索算法,与传统的优化算法相比,具有以下特点:

    (1)以决策变量的编码作为运算对象。传统的优化算法往往直接利用决策变量的实际值本身来进行优化计算,但遗传算法是使用决策变量的某种形式的编码作为运算对象。这种对决策变量的编码处理方式,使得我们在优化计算中可借鉴生物学中染色体和基因等概念,可以模仿自然界中生物的遗传和进化激励,也可以很方便地应用遗传操作算子。

    (2)直接以适应度作为搜索信息。传统的优化算法不仅需要利用目标函数值,而且搜索过程往往受目标函数的连续性约束,有可能还需要满足“目标函数的导数必须存在”的要求以确定搜索方向。遗传算法仅使用由目标函数值变换来的适应度函数值就可确定进一步的搜索范围,无需目标函数的导数值等其他辅助信息。直接利用目标函数值或个体适应度值也可以将搜索范围集中到适应度较高部分的搜索空间中,从而提高搜索效率。

    (3)使用多个点的搜索信息,具有隐含并行性。传统的优化算法往往是从解空间的一个初始点开始最优解的迭代搜索过程。单个点所提供的搜索信息不多,所以搜索效率不高,还有可能陷入局部最优解而停滞;遗传算法从由很多个体组成的初始种群开始最优解的搜索过程,而不是从单个个体开始搜索。对初始群体进行的、选择、交叉、变异等运算,产生出新一代群体,其中包括了许多群体信息。这些信息可以避免搜索一些不必要的点,从而避免陷入局部最优,逐步逼近全局最优解。

    (4) 使用概率搜索而非确定性规则。传统的优化算法往往使用确定性的搜索方法,一个搜索点到另一个搜索点的转移有确定的转移方向和转移关系,这种确定性可能使得搜索达不到最优店,限制了算法的应用范围。遗传算法是一种自适应搜索技术,其选择、交叉、变异等运算都是以一种概率方式进行的,增加了搜索过程的灵活性,而且能以较大概率收敛于最优解,具有较好的全局优化求解能力。但,交叉概率、变异概率等参数也会影响算法的搜索结果和搜索效率,所以如何选择遗传算法的参数在其应用中是一个比较重要的问题。
    综上,由于遗传算法的整体搜索策略和优化搜索方式在计算时不依赖于梯度信息或其他辅助知识,只需要求解影响搜索方向的目标函数和相应的适应度函数,所以遗传算法提供了一种求解复杂系统问题的通用框架。它不依赖于问题的具体领域,对问题的种类有很强的鲁棒性,所以广泛应用于各种领域,包括:函数优化、组合优化生产调度问题、自动控制
    、机器人学、图像处理(图像恢复、图像边缘特征提取…)、人工生命、遗传编程机器学习

    3 遗传算法的基本流程及实现技术

    基本遗传算法(Simple Genetic Algorithms,SGA)只使用选择算子、交叉算子和变异算子这三种遗传算子,进化过程简单,是其他遗传算法的基础。

    3.1 遗传算法的基本流程

    通过随机方式产生若干由确定长度(长度与待求解问题的精度有关)编码的初始群体;
    通过适应度函数对每个个体进行评价,选择适应度值高的个体参与遗传操作,适应度低的个体被淘汰;
    经遗传操作(复制、交叉、变异)的个体集合形成新一代种群,直到满足停止准则(进化代数GEN>=?);
    将后代中变现最好的个体作为遗传算法的执行结果。

    如何使用matlab遗传算法求解车间调度问题

    其中,GEN是当前代数;M是种群规模,i代表种群数量。

    3.2 遗传算法的实现技术

    基本遗传算法(SGA)由编码、适应度函数、遗传算子(选择、交叉、变异)及运行参数组成。

    3.2.1 编码

    (1)二进制编码
    二进制编码的字符串长度与问题所求解的精度有关。需要保证所求解空间内的每一个个体都可以被编码。
    优点:编、解码操作简单,遗传、交叉便于实现
    缺点:长度大

    (2)其他编码方法
    格雷码、浮点数编码、符号编码、多参数编码等

    3.2.2 适应度函数

    适应度函数要有效反映每一个染色体与问题的最优解染色体之间的差距。

    3.2.3选择算子

    如何使用matlab遗传算法求解车间调度问题

    3.2.4 交叉算子

    交叉运算是指对两个相互配对的染色体按某种方式相互交换其部分基因,从而形成两个新的个体;交叉运算是遗传算法区别于其他进化算法的重要特征,是产生新个体的主要方法。在交叉之前需要将群体中的个体进行配对,一般采取随机配对原则。
    常用的交叉方式:
    单点交叉
    双点交叉(多点交叉,交叉点数越多,个体的结构被破坏的可能性越大,一般不采用多点交叉的方式)
    均匀交叉
    算术交叉

    3.2.5 变异算子

    遗传算法中的变异运算是指将个体染色体编码串中的某些基因座上的基因值用该基因座的其他等位基因来替换,从而形成一个新的个体。

    就遗传算法运算过程中产生新个体的能力方面来说,交叉运算是产生新个体的主要方法,它决定了遗传算法的全局搜索能力;而变异运算只是产生新个体的辅助方法,但也是必不可少的一个运算步骤,它决定了遗传算法的局部搜索能力。交叉算子与变异算子的共同配合完成了其对搜索空间的全局搜索和局部搜索,从而使遗传算法能以良好的搜索性能完成最优化问题的寻优过程。

    3.2.6 运行参数

    如何使用matlab遗传算法求解车间调度问题

    4 遗传算法的基本原理

    4.1 模式定理

    如何使用matlab遗传算法求解车间调度问题

    4.2 积木块假设

    具有低阶、定义长度短,且适应度值高于群体平均适应度值的模式称为基因块或积木块。
    积木块假设:个体的基因块通过选择、交叉、变异等遗传算子的作用,能够相互拼接在一起,形成适应度更高的个体编码串。
    积木块假设说明了用遗传算法求解各类问题的基本思想,即通过积木块直接相互拼接在一起能够产生更好的解。

    三、部分源代码

    clc;clear%% 下载数据% 加工数据包括加工时间,加工机器,机器数,各机器权重,工件数,各工件对应的工序数load data operation_time operation_Machine num_machine machine_weight num_job num_op%% 基本参数MAXGEN = 200;               % 最大迭代次数Ps = 0.8;                   % 选择率Pc = 0.7;                   % 交叉率Pm = 0.3;                   % 变异率sizepop = 200;              % 个体数目e = 0.5;                    % 目标值权重trace = zeros(2,MAXGEN);%% ===========================种群初始化============================total_op_num=sum(num_op);chroms=initialization(num_op,num_job,total_op_num,sizepop,operation_machine,operation_time);[Z,~,~]=fitness(chroms,num_machine,e,num_job,num_op);%% ============================迭代过程=============================for gen=1:MAXGEN    fprintf('当前迭代次数:'),disp(gen)    % 轮盘赌选择    chroms_new=selection(chroms,Z,Ps);    % 交叉操作    chroms_new=crossover(chroms_new,Pc,total_op_num,num_job,num_op);    % 变异操作    chroms_new=mutation(chroms_new,total_op_num,Pm,num_machine,e,num_job,num_op,operation_machine,operation_time);    % 计算选择交叉变异后个体的适应度    [Z_new,~,~]=fitness(chroms_new,num_machine,e,num_job,num_op);    % 根据适应度在原种群和遗传操作后的种群中选出sizepop个更优个体    [chroms,Z,chrom_best]=update_chroms(chroms,chroms_new,Z,Z_new,sizepop);    % 记录每代的最优适应度与平均适应度    trace(1, gen)=Z(1);           trace(2, gen)=mean(Z);      % 更新全局最优适应度    if gen==1 || MinVal>trace(1,gen)        MinVal=trace(1,gen);function [Z,machine_weight,pvals] = fitness(chroms,num_machine,e,num_job,num_op)sizepop=size(chroms,1);pvals=cell(1,sizepop);Z1=zeros(1,sizepop);Z2=Z1;total_op_num=sum(num_op);  % 总工序数for k=1:sizepop    chrom=chroms(k,:);    machine=zeros(1,num_machine);  % 记录各机器变化时间    job=zeros(1,num_job);  % 记录各工件变化时间    machine_time=zeros(1,num_machine);  % 计算各机器的实际加工时间    pval=zeros(2,total_op_num);  % 记录各工序开始和结束时间    for i=1:total_op_num        % 机器时间大于工件时间        if machine(chrom(total_op_num+i))>=job(chrom(i))            pval(1,i)=machine(chrom(total_op_num+i));  % 记录工件开始时间            machine(chrom(total_op_num+i))=machine(chrom(total_op_num+i))+chrom(total_op_num*2+i);            job(chrom(i))=machine(chrom(total_op_num+i));            pval(2,i)=machine(chrom(total_op_num+i));  % 记录工件结束时间            % 机器时间小于工件时间        else            pval(1,i)=job(chrom(i));            job(chrom(i))=job(chrom(i))+chrom(total_op_num*2+i);            machine(chrom(total_op_num+i))=job(chrom(i));            pval(2,i)=job(chrom(i));        end        machine_time(chrom(total_op_num+i))=machine_time(chrom(total_op_num+i))+chrom(total_op_num*2+i);    end    Z1(k)=max(machine);  % 最大机器时间值,对应makespan    % machine_weight=machine_time/sum(machine_time);  % 计算各机器的负荷    machine_weight=machine_time;    Z2(k)=max(machine_weight)-min(machine_weight);    pvals{k}=pval;end% min_makespan=min(Z1);%所有染色体的makespan最优值% max_makespan=max(Z1);% min_weight=min(Z2);%负载最优值% max_weight=max(Z2);% Z=e*((Z1-min_makespan)./(max_makespan-min_makespan))+(1-e)*((Z2-min_weight)./(max_weight-min_weight));%计算适应度Z=e*Z1+(1-e)*Z2;function [ chroms_new] = crossover(chroms,Pc,total_op_num,num_job,num_op)size_chrom=size(chroms,1);  % 染色体数chroms_new=chroms;%% 面向工序码的交叉操作for i=1:2:size_chrom-1     if Pc>rand        % 父代染色体        parent1=chroms(i,:);        parent2=chroms(i+1,:);        Job=randperm(num_job);        % 将工件随机分成两个集合        J1=Job(1:round(num_job/2));        J2=Job(length(J1)+1:end);        % 子代染色体        child1=parent1;        child2=parent2;        op_p1=[];        op_p2=[];        for j=1:length(J2)            %找出父代中J2片段对应的位置            op_p1=[op_p1,find(parent1(1:total_op_num)==J2(j))];            op_p2=[op_p2,find(parent2(1:total_op_num)==J2(j))];        end        op_s1=sort(op_p1);        op_s2=sort(op_p2);        % 子代1交换J2片段的基因,机器码对应位置的基因,工时码对应位置的基因        child1(op_s1)=parent2(op_s2);        child1(total_op_num+op_s1)=parent2(total_op_num+op_s2);        child1(total_op_num*2+op_s1)=parent2(total_op_num*2+op_s2);        % 子代2同理        child2(op_s2)=parent1(op_s1);        child2(total_op_num+op_s2)=parent1(total_op_num+op_s1);        child2(total_op_num*2+op_s2)=parent1(total_op_num*2+op_s1);        chroms_new(i,:)=child1;        chroms_new(i+1,:)=child2;    endend%% 面向机器码的交叉操作for k=1:2:size_chrom-1    if Pc>rand        parent1=chroms_new(k,:);        parent2=chroms_new(k+1,:);        child1=parent1;        child2=parent2;        % 随机产生与染色体长度相等的0,1序列        rand0_1=randi([0,1],1,total_op_num);        for n=1:num_job            ind_0=find(rand0_1(num_op(n)*(n-1)+1:num_op(n)*n)==0);            if ~isempty(ind_0)                temp1=find(parent1(1:total_op_num)==n);                temp2=find(parent2(1:total_op_num)==n);                child1(total_op_num+temp1(ind_0))=parent2(total_op_num+temp2(ind_0));                child2(total_op_num+temp2(ind_0))=parent1(total_op_num+temp1(ind_0));                child1(total_op_num*2+temp1(ind_0))=parent2(total_op_num*2+temp2(ind_0));                child2(total_op_num*2+temp2(ind_0))=parent1(total_op_num*2+temp1(ind_0));            end        end        chroms_new(k,:)=child1;        chroms_new(k+1,:)=child2;    endendend

    四、运行结果

    如何使用matlab遗传算法求解车间调度问题

    如何使用matlab遗传算法求解车间调度问题

    感谢你能够认真阅读完这篇文章,希望小编分享的“如何使用matlab遗传算法求解车间调度问题”这篇文章对大家有帮助,同时也希望大家多多支持编程网,关注编程网精选频道,更多相关知识等着你来学习!

    --结束END--

    本文标题: 如何使用matlab遗传算法求解车间调度问题

    本文链接: https://www.lsjlt.com/news/321991.html(转载时请注明来源链接)

    有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

    本篇文章演示代码以及资料文档资料下载

    下载Word文档到电脑,方便收藏和打印~

    下载Word文档
    猜你喜欢
    • 如何使用matlab遗传算法求解车间调度问题
      这篇文章主要介绍了如何使用matlab遗传算法求解车间调度问题,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、车间调度简介1 车间调度定义车间调度是指根据产品制造的合理需求...
      99+
      2023-06-29
    • matlab遗传算法求解车间调度问题分析及实现源码
      目录一、车间调度简介1 车间调度定义2 传统作业车间调度二、遗传算法简介1 遗传算法概述2 遗传算法的特点和应用3 遗传算法的基本流程及实现技术3.1 遗传算法的基本流程3.2 遗传...
      99+
      2024-04-02
    • 如何使用matlab鸟群算法求解车间调度问题
      这篇文章主要介绍了如何使用matlab鸟群算法求解车间调度问题,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、车间调度简介1 车间调度定义车间调度是指根据产品制造的合理需求...
      99+
      2023-06-29
    • matlab鸟群算法求解车间调度问题详解及实现源码
      目录一、车间调度简介1 车间调度定义2 传统作业车间调度3 柔性作业车间调度二、蝴蝶优化算法(MBO)简介1 介绍2 香味3 具体算法三、部分源代码五、matlab版本及参考文献一、...
      99+
      2024-04-02
    • Matlab利用遗传算法GA求解非连续函数问题详解
      目录遗传算法基本思想遗传算法的主要步骤遗传编码二进制编码实数编码遗传算法流程实际演示 遗传算法基本思想 遗传算法(Genetic Algorithm, GA)起源于对生物系...
      99+
      2024-04-02
    • Java利用遗传算法求解最短路径问题
      目录1、问题描述2、编码3、个体类4、遗传算法解决最短路径问题主方法5、适应度6、选择算子7、交叉算子8、变异算子9、总结遗传算法(Genetic Algorithm,GA)最早是由...
      99+
      2024-04-02
    • 如何使用matlab模拟退火算法单约束车间流水线调度
      这篇文章主要为大家展示了“如何使用matlab模拟退火算法单约束车间流水线调度”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“如何使用matlab模拟退火算法单约束车间流水线调度”这篇文章吧。一、...
      99+
      2023-06-29
    • Java如何利用遗传算法求解最短路径
      这篇“Java如何利用遗传算法求解最短路径”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Java如何利用遗传算法求解最短路径...
      99+
      2023-07-01
    • 如何使用Python实现遗传算法
      本篇内容介绍了“如何使用Python实现遗传算法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!遗传算法是模仿自然界生物进化机制发展起来的随机...
      99+
      2023-07-05
    • python遗传算法之geatpy如何安装使用
      这篇文章主要介绍了python遗传算法之geatpy如何安装使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇python遗传算法之geatpy如何安装使用文章都会有所收获,下面我们一起来看看吧。1. geat...
      99+
      2023-07-06
    • matlab模拟退火算法单约束车间流水线调度解决实现及示例
      目录一、车间调度简介1 车间调度定义2 传统作业车间调度3 柔性作业车间调度二、模拟退火算法简介三、部分源代码四、运行结果五、matlab版本及参考文献一、车间调度简介 1 车间调度...
      99+
      2024-04-02
    • 如何使用C#算法解决求第n个数值问题
      这篇文章主要为大家展示了“如何使用C#算法解决求第n个数值问题”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“如何使用C#算法解决求第n个数值问题”这篇文章吧。已知数列:1,1,2,3,5,8,1...
      99+
      2023-06-18
    • python如何实现使用遗传算法进行图片拟合
      小编给大家分享一下python如何实现使用遗传算法进行图片拟合,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!引言算法思路假设我们有这样一个生物族群,他们的每个基因片段都是一个个三角形(即只含三个点和颜色信息),他们每个个体...
      99+
      2023-06-29
    • 如何使用Python地图四色原理的遗传算法着色
      小编给大家分享一下如何使用Python地图四色原理的遗传算法着色,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1 任务需求  ...
      99+
      2023-06-29
    • 如何使用python求解迷宫问题
      这篇文章主要介绍“如何使用python求解迷宫问题”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“如何使用python求解迷宫问题”文章能帮助大家解决问题。前言在迷宫问题中,给定入口和出口,要求找到路...
      99+
      2023-06-29
    • 如何使用Python解决LeetCode的算法问题?
      在计算机科学领域中,算法问题是非常重要的。LeetCode是一个在线的算法问题平台,它提供了大量的算法问题,供开发者练习和学习。而Python是一个非常流行的编程语言,具有简单易学、代码可读性高、丰富的库等特点。在本文中,我们将介绍如何使...
      99+
      2023-11-06
      leetcode 大数据 关键字
    • python调用dll出现精度问题如何解决
      本文小编为大家详细介绍“python调用dll出现精度问题如何解决”,内容详细,步骤清晰,细节处理妥当,希望这篇“python调用dll出现精度问题如何解决”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。遇到的问题...
      99+
      2023-07-05
    • 如何在 LeetCode 上使用 Go 解决算法问题?
      LeetCode 是一个非常受欢迎的在线编程平台,它提供了大量的算法题目供程序员练习。使用 LeetCode 可以帮助程序员提高算法能力,同时也可以帮助程序员找到更好的工作机会。本文将介绍如何在 LeetCode 上使用 Go 解决算法问...
      99+
      2023-07-23
      http api leetcode
    • 如何解决PHPCMS调用时间错误的问题?
      PHPCMS是一款广泛应用于网站建设的内容管理系统,许多网站都是基于PHPCMS开发的。然而,在使用过程中,有时会遇到调用时间错误的问题,这会影响网站的正常运行。本文将探讨如何解决PH...
      99+
      2024-03-14
      php 时间 cms 格式化输出
    • 在Go语言中如何解决并发任务的调度算法优化问题?
      在Go语言中如何解决并发任务的调度算法优化问题?Go语言作为一门旨在解决并发编程问题的语言,提供了丰富的并发特性和机制。然而,在实际应用中,我们常常遇到需要优化并发任务调度的问题。本文将介绍一种优化并发任务调度算法的方法,并给出具体的代码示...
      99+
      2023-10-22
      并发任务调度 Go语言并发优化 调度算法优化
    软考高级职称资格查询
    编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
    • 官方手机版

    • 微信公众号

    • 商务合作