贪心法部分背包问题的实现

问题描述: 给定n个物品,物品价值分别为P1,P2,…,Pn,物品重量分别为W1,W2,…,Wn,背包容量为M。每种物品可部分装入到背包中。输出X1,X2,…,Xn, 0<=Xi<=1使得背包内的物品价值最大。试设计一个算法求解该问题

设计思路:先将所有物品按单位价值进行排序后,每次取出剩余物品内单位价值最大的物品尝试加入背包,如果背包容量足够,则将整个物品加入,否则加入该物品的一部分。

相关符号解释:

(1)    数组V存放各物品的价值

(2)    数组W存放各物品的重量

(3)    L表示背包容量

(4)    totalValue表示总价值

(5)    flag (当值为false时表示所有物品全部装入背包,当值为true时表示最后一个物品部分装入)

(6)    share (部分装入物品的份额)

(7)    List集合result存放选出的物品

(8)    uniValue 单位价值

Alogrithm:

          Sort(V,W)//将物品按单位价值进行排序

          For(i=0;i<n;i++)

          Repeat

            If(L>0)

                 If(L>W[i])//表示可将当前物品全部装入

                     Result.add(i)

                     totalValue += W[i]

                     L-=W[i];

                  else{//部分装入

                    uniValue = V[i]/W[i]//单位价值

                    totalValue+= L*uniValue;

                    share=L;

                    flag = true;

                    L=0;

                     Result.add(i)                

                     }

Until L<=0;

Ouput result and totalValue;

运行结果:

import java.text.DecimalFormat;
import java.util.*;

/**
 * Created by Administrator on 2016/11/29.
 */
public class Knapsack {

    public static void main(String [] args){
        double L =12;
        int[] V = {13,10,24,15,28,33,20,8};//存放价值
        int[] W = {2,1,3,2,4,5,3,1};//存放重量
        int len = V.length;

        double totalValue = 0;
        Listresult = new LinkedList<>();

        sort(V,W);//按单位价值排序
        for(int j = 0 ; j < len ; j++)
            System.out.println("物品"+(j+1)+": 价值为:"+ V[j]+" 重量为"+W[j] );


        int i ;
        boolean flag = false;//flag = false代表所有物品全部装入,为true代表有物品部分装入
        double share = 0 ;//部分装入部分的份额.
        for(i=0;i=W[i]){//能把整个物品装进去
                result.add(i);
                L -= W[i];
                totalValue += V[i];

            }else{
                double uniValue = V[i]/(double)W[i];
                totalValue += L*uniValue;
                share = L;
                flag = true;
                L=0;
                result.add(i);



            }
        }

        DecimalFormat de = new DecimalFormat("0.00");
       System.out.println("所选的结果为:");
        for(Integer r : result){
            System.out.print(r+1+" ");
        }
        System.out.println("总价值为"+de.format(totalValue));

        if(flag = true){
            System.out.println("最后一个物品为部分装入,装入的部分重量为"+share);

        }else {
            System.out.println("物品全部装入");
        }



    }

    private static void sort(int[] p, int[] w) {
        int len = p.length;
        int tmp = 0;
        for(int i = 0 ; i < len - 1 ; i++){
            int maxIndex = i ;
            double IUniValue = p[i]/(double)w[i] ;
            for(int j = i+1 ; j < len ; j ++){
                double  JUniValue = p[j]/(double)w[j];
                if(JUniValue>IUniValue){
                    maxIndex = j ;
                }
            }
            tmp = p[i];
            p[i] = p[maxIndex];
            p[maxIndex] =tmp;
            tmp = w[i];
            w[i] = w[maxIndex];
            w[maxIndex] =tmp;
        }




    }
}



RocketMQ 中,**分片(Sharding)** 是实现高并发、高可用和可展性的重要机制之一。其核心作用是将消息的存储和消费进行分布式管理,以支持大规模消息处理场景。 ### 分片的作用 1. **提升系统吞吐量** 通过将一个 Topic 的消息分布到多个 Broker 上,每个 Broker 负责一部分消息的存储和转发,从而实现横向展,提高整体系统的吞吐能力。 2. **支持负载均衡** 在消息生产与消费过程中,分片机制使得消息可以均匀分布在多个 Broker 上,生产者和消费者可以并行地处理多个分片,实现负载均衡[^5]。 3. **增强系统可用性与容错性** 每个分片可以配置主从结构(Master-Slave),实现数据复制与故障切换,确保在某个 Broker 故障时仍能保证消息的高可用[^4]。 ### 分片的工作机制 1. **Topic 与 Message Queue 的关系** 在 RocketMQ 中,每个 Topic 会被划分为多个 **Message Queue**(也称为队列或分片),这些队列分布在不同的 Broker 上。例如,一个 Topic 可能有 4 个队列,分别分布在两个 Broker 上,每个 Broker 管理两个队列。 2. **生产者的分片选择** 当生产者发送消息时,会根据一定的策略(如轮询、哈希等)选择一个合适的 Message Queue 进行投递。这一过程称为**生产者负载均衡**。生产者会定期从 NameServer 获取 Topic 的队列分布信息,以保证选择的准确性[^5]。 3. **消费者的分片分配** 消费者组(ConsumerGroup)中的每个消费者实例会负责一部分 Message Queue 的消费任务。这一过程称为**消费者负载均衡**,由 Broker 协调完成,确保每个队列只被一个消费者实例消费,从而避免重复消费和竞争问题。 4. **消息的物理存储** RocketMQ 将所有消息写入统一的 **CommitLog** 文件中,然后通过 **ConsumeQueue** 文件记录每个 Topic 的分片索引信息,实现逻辑分片与物理存储的分离。这种机制保证了写入的高效性和读取的灵活性[^4]。 ### 分片配置与管理 - **创建 Topic 时指定分片数量** 在创建 Topic 时,可以通过命令行或配置文件指定其分片数量(即 Message Queue 数量)。 - **动态容** 可以在不中断服务的情况下,向集群中新增 Broker,并为已有 Topic 增加分片,以应对不断增长的消息量。 ### 示例代码:查看 Topic 分片信息 ```bash # 查看 Topic 的队列分布信息 mqadmin topicRoute -n localhost:9876 -t MyTopic ``` 该命令将输出 Topic `MyTopic` 的路由信息,包括各个 Message Queue 所在的 Broker 地址。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值