在上一篇,我们从使用方式和策略上,对消息队列做了一个宏观描述。从本篇开始,我们将深入到源码内部,仔细分析Kafka到底是如何实现一个分布式消息队列。我们的分析将从Producer端开始。
  
  从Kafka 0.8.2开始,发布了一套新的Java版的client api, KafkaProducer/KafkaConsumer,替代之前的scala版的api。本系列的分析将只针对这套Java版的api。
  
  多线程异步发送模型
  
  下图是经过源码分析之后,整理出来的Producer端的架构图:
  
  这里写图片描述
  
  在上一篇我们讲过,Producer有同步发送和异步发送2种策略。在以前的Kafka client api实现中,同步和异步是分开实现的。而在0.9中,同步发送其实是通过异步发送间接实现,其接口如下:
  
  public class KafkaProducer<K, V> implements Producer<K, V> {
  
  ...
  
  public Future<RecordMetadata> send(ProducerRecord<K, V> record, Callback callback) //异步发送接口
  
  {
  
  ...
  
  }
  
  }
  
  1
  
  2
  
  3
  
  4
  
  5
  
  6
  
  7
  
  要实现同步发送,只要在拿到返回的Future对象之后,直接调用get()就可以了。
  
  基本思路
  
  从上图我们可以看出,异步发送的基本思路就是:send的时候,KafkaProducer把消息放到本地的消息队列RecordAccumulator,然后一个后台线程Sender不断循环,把消息发给Kafka集群。
  
  要实现这个,还得有一个前提条件:就是KafkaProducer/Sender都需要获取集群的配置信息Metadata。所谓Metadata,也就是在上一篇所讲的,Topic/Partion与broker的映射关系:每一个Topic的每一个Partion,得知道其对应的broker列表是什么,其中leader是谁,follower是谁。
  
  2个数据流
  
  所以在上图中,有2个数据流:
  
  Metadata流(A1,A2,A3):Sender从集群获取信息,然后更新Metadata; KafkaProducer先读取Metadata,然后把消息放入队列。
  
  消息流(B1, B2, B3):这个很好理解,不再详述。
  
  本篇着重讲述Metadata流,消息流,将在后续详细讲述。
  
  Metadata的线程安全性
  
  从上图可以看出,Metadata是多个producer线程读,一个sender线程更新,因此它必须是线程安全的。
  
  Kafka的官方文档上也有说明,KafkaProducer是线程安全的,可以在多线程中调用:
  
  The producer is thread safe and sharing a single producer instance across threads will generally be faster than having multiple instances.
  
  从下面代码也可以看出,它的所有public方法都是synchronized:
  
  public final class Metadata {
  
  。。。
  
  public synchronized Cluster fetch() {
  
  return this.cluster;
  
  }
  
  public synchronized long timeToNextUpdate(long nowMs) {
  
  。。。
  
  }
  
  public synchronized int requestUpdate() {
  
  。。。
  
  }
  
  。。。
  
  }
  
  1
  
  2
  
  3
  
  4
  
  5
  
  6
  
  7
  
  8
  
  9
  
  10
  
  11
  
  12
  
  13
  
  Metadata的数据结构
  
  下面代码列举了Metadata的主要数据结构:一个Cluster对象 + 1堆状态变量。前者记录了集群的配置信息,后者用于控制Metadata的更新策略。
  
  public final class www.tyff688.com Metadata {
  
  ...
  
  private final long refreshBackoffMs; //更新失败的情况下,下1次更新的补偿时间(这个变量在代码中意义不是太大)
  
  private final long metadataExpireMs; //关键值:每隔多久,更新一次。缺省是600*1000,也就是10分种
  
  private int version; //每更新成功1次,version递增1。这个变量主要用于在while循环,wait的时候,作为循环判断条件
  
  private long lastRefreshMs; //上一次更新时间(也包含更新失败的情况)
  
  private long lastSuccessfulRefreshMs; //上一次成功更新的时间(如果每次都成功的话,则2者相等。否则,lastSuccessulRefreshMs < lastRefreshMs)
  
  private Cluster cluster; //集群配置信息
  
  private boolean needUpdate; //是否强制刷新
  
  、
  
  ...
  
  }
  
  public final class Cluster {
  
  ...
  
  private final List<Node> nodes; //www.senta77.com Node也就是Broker
  
  private final Map<TopicPartition, PartitionInfo> partitionsByTopicPartition; //Topic/Partion和broker list的映射关系
  
  private final Map<String, List<PartitionInfo>> partitionsByTopic;
  
  private final Map<String, List<PartitionInfo>> availablePartitionsByTopic;
  
  private final Map<Integer, List<PartitionInfo>> partitionsByNode;
  
  private final Map<Integer, Node> nodesById;
  
  }
  
  public class PartitionInfo {
  
  private final String topic;
  
  private final int partition;
  
  private final Node www.xbyl688.com leader;
  
  private final Node[] replicas;
  
  private final Node[] www.honqili66.com inSyncReplicas;
  
  }
  
  1
  
  2
  
  3
  
  4
  
  5
  
  6
  
  7
  
  8
  
  9
  
  10
  
  11
  
  12
  
  13
  
  14
  
  15
  
  16
  
  17
  
  18
  
  19
  
  20
  
  21
  
  22
  
  23
  
  24
  
  25
  
  26
  
  27
  
  28
  
  29
  
  30
  
  producer读取Metadata
  
  下面是send函数的源码,可以看到,在send之前,会先读取metadata。如果metadata读不到,会一直阻塞在那,直到超时,抛出TimeoutException
  
  //KafkaProducer
  
  public Future<RecordMetadata> send(ProducerRecord<K, V> record, Callback callback) {
  
  try {
  
  long waitedOnMetadataMs = waitOnMetadata(record.topic(), this.maxBlockTimeMs); //拿不到topic的配置信息,会一直阻塞在这,直到抛异常
  
  ... //拿到了,执行下面的send逻辑
  
  } catch()
  
  {}
  
  }
  
  //KafkaProducer
  
  private long waitOnMetadata(String topic, long maxWaitMs) throws InterruptedException {
  
  if (!this.metadata.containsTopic(topic))
  
  this.metadata.add(topic);
  
  if (metadata.fetch().partitionsForTopic(topic) != null)
  
  return 0; //取到topic的配置信息,直接返回
  
  long begin = time.milliseconds();
  
  long remainingWaitMs = maxWaitMs;
  
  while (metadata.fetch().partitionsForTopic(topic) == null) { //取不到topic的配置信息,一直死循环wait,直到超时,抛TimeoutException
  
  log.trace("Requesting metadata update for topic {}.", topic);
  
  int version = metadata.requestUpdate(); //把needUpdate置为true
  
  sender.wakeup(); //唤起sender
  
  metadata.awaitUpdate(version, remainingWaitMs); //metadata的关键函数
  
  long elapsed = time.milliseconds() - begin;
  
  if (elapsed >= maxWaitMs)
  
  throw new TimeoutException("Failed to update metadata after " + maxWaitMs + " ms.");
  
  if (metadata.fetch().unauthorizedTopics().contains(topic))
  
  throw new TopicAuthorizationException(topic);
  
  remainingWaitMs = maxWaitMs - elapsed;
  
  }
  
  return time.milliseconds() - begin;
  
  }
  
  //Metadata
  
  public synchronized void awaitUpdate(final int lastVersion, final long maxWaitMs) throws InterruptedException {
  
  if (maxWaitMs < 0) {
  
  throw new IllegalArgumentException("Max time to wait for metadata updates should not be < 0 milli seconds");
  
  }
  
  long begin = System.currentTimeMillis();
  
  long remainingWaitMs = maxWaitMs;
  
  while (this.version <= lastVersion) { //当Sender成功更新meatadata之后,version加1。否则会循环,一直wait
  
  if (remainingWaitMs != 0
  
  wait(remainingWaitMs); //线程的wait机制,wait和synchronized的配合使用
  
  long elapsed = System.currentTimeMillis() - begin;
  
  if (elapsed >= maxWaitMs) //wait时间超出了最长等待时间
  
  throw new TimeoutException("Failed to update metadata after " + maxWaitMs + " ms.");
  
  remainingWaitMs = maxWaitMs - elapsed;
  
  }
  
  }
  
  1
  
  2
  
  3
  
  4
  
  5
  
  6
  
  7
  
  8
  
  9
  
  10
  
  11
  
  12
  
  13
  
  14
  
  15
  
  16
  
  17
  
  18
  
  19
  
  20
  
  21
  
  22
  
  23
  
  24
  
  25
  
  26
  
  27
  
  28
  
  29
  
  30
  
  31
  
  32
  
  33
  
  34
  
  35
  
  36
  
  37
  
  38
  
  39
  
  40
  
  41
  
  42
  
  43
  
  44
  
  45
  
  46
  
  47
  
  48
  
  49
  
  50
  
  51
  
  52
  
  总结:从上面代码可以看出,producer wait metadata的时候,有2个条件:
  
  (1) while (metadata.fetch().partitionsForTopic(topic) == null)
  
  (2)while (this.version <= lastVersion)
  
  有wait就会有notify,notify在Sender更新Metadata的时候发出。
  
  Sender更新Metadata
  
  Sender的创建
  
  下面是KafkaProducer的构造函数,从代码可以看出,Sender就是KafkaProducer中创建的一个Thread.
  
  private KafkaProducer(ProducerConfig config, Serializer<K> keySerializer, Serializer<V> valueSerializer) {
  
  try {
  
  ...
  
  this.metadata = new Metadata(retryBackoffMs, config.getLong(ProducerConfig.METADATA_MAX_AGE_CONFIG)); //构造metadata
  
  this.metadata.update(Cluster.bootstrap(addresses), time.milliseconds()); //往metadata中,填入初始的,配置的node列表
  
  ChannelBuilder channelBuilder = ClientUtils.createChannelBuilder(config.values());
  
  NetworkClient client = new NetworkClient(
  
  new Selector(config.getLong(ProducerConfig.CONNECTIONS_MAX_IDLE_MS_CONFIG), this.metrics, time, "producer", metricTags, channelBuilder),
  
  this.metadata,
  
  clientId,
  
  config.getInt(ProducerConfig.MAX_IN_FLIGHT_REQUESTS_PER_CONNECTION),
  
  config.getLong(ProducerConfig.RECONNECT_BACKOFF_MS_CONFIG),
  
  config.getInt(ProducerConfig.SEND_BUFFER_CONFIG),
  
  config.getInt(ProducerConfig.RECEIVE_BUFFER_CONFIG),
  
  this.sender = new Sender(client, //构造一个sender。sender本身实现的是Runnable接口
  
  this.metadata,
  
  this.accumulator,
  
  config.getInt(ProducerConfig.MAX_REQUEST_SIZE_CONFIG),
  
  (short) parseAcks(config.getString(ProducerConfig.ACKS_CONFIG)),
  
  config.getInt(ProducerConfig.RETRIES_CONFIG),
  
  this.metrics,
  
  new SystemTime(),
  
  clientId,
  
  this.requestTimeoutMs);
  
  String ioThreadName = "kafka-producer-network-thread" + (clientId.length() > 0 ? " | " + clientId : "");
  
  this.ioThread = new KafkaThread(ioThreadName, this.sender, true);
  
  this.ioThread.start(); //一个线程,开启sender
  
  1
  
  2
  
  3
  
  4
  
  5
  
  6
  
  7
  
  8
  
  9
  
  10
  
  11
  
  12
  
  13
  
  14
  
  15
  
  16
  
  17
  
  18
  
  19
  
  20
  
  21
  
  22
  
  23
  
  24
  
  25
  
  26
  
  27
  
  28
  
  29
  
  30
  
  31
  
  32
  
  33
  
  Metadata的更新机制 – Sender的run方法
  
  public void run() {
  
  // main loop, runs until close is called
  
  while (running) {
  
  try {
  
  run(time.milliseconds());
  
  } catch (Exception e) {
  
  log.error("Uncaught error in kafka producer I/O thread: ", e);
  
  }
  
  }
  
  。。。
  
  }
  
  public void run(long now) {
  
  Cluster cluster = metadata.fetch();
  
  。。。
  
  RecordAccumulator.ReadyCheckResult result = this.accumulator.ready(cluster, now); //遍历消息队列中所有的消息,找出对应的,已经ready的Node
  
  if (result.unknownLeadersExist) //如果一个ready的node都没有,请求更新metadata
  
  this.metadata.requestUpdate();
  
  。。。
  
  //client的2个关键函数,一个发送ClientRequest,一个接收ClientResponse。底层调用的是NIO的poll。关于nio, 后面会详细介绍
  
  for (ClientRequest request : requests)
  
  client.send(request, now);
  
  this.client.poll(pollTimeout, now);
  
  }
  
  //NetworkClient
  
  public List<ClientResponse> poll(long timeout, long now) {
  
  long metadataTimeout = metadataUpdater.maybeUpdate(now); //判断是否要更新metadata
  
  try {
  
  this.selector.poll(Utils.min(timeout, metadataTimeout, requestTimeoutMs));
  
  } catch (IOException e) {
  
  log.error("Unexpected error during I/O", e);
  
  }
  
  // process completed actions
  
  long updatedNow = this.time.milliseconds();
  
  List<ClientResponse> responses = new ArrayList<>();
  
  handleCompletedSends(responses, updatedNow);
  
  handleCompletedReceives(responses, updatedNow); //在返回的handler中,会处理metadata的更新
  
  handleDisconnections(responses, updatedNow);
  
  handleConnections();
  
  handleTimedOutRequests(responses, updatedNow);
  
  // invoke callbacks
  
  for (ClientResponse response : responses) {
  
  if (response.request().hasCallback()) {
  
  try {
  
  response.request().callback().onComplete(response);
  
  } catch (Exception e) {
  
  log.error("Uncaught error in request completion:", e);
  
  }
  
  }
  
  }
  
  return responses;
  
  }
  
  //DefaultMetadataUpdater
  
  @Override
  
  public long maybeUpdate(long now) {
  
  // should we update our metadata?
  
  long timeToNextMetadataUpdate = metadata.timeToNextUpdate(now);
  
  long timeToNextReconnectAttempt = Math.max(this.lastNoNodeAvailableMs + metadata.refreshBackoff() - now, 0);
  
  long waitForMetadataFetch = this.metadataFetchInProgress ? Integer.MAX_VALUE : 0;
  
  // if there is no node available to connect, back off refreshing metadata
  
  long metadataTimeout = Math.max(Math.max(timeToNextMetadataUpdate, timeToNextReconnectAttempt),
  
  waitForMetadataFetch);
  
  if (metadataTimeout == 0) {
  
  // highly dependent on the behavior of leastLoadedNode.
  
  Node node = leastLoadedNode(now); //找到负载最小的Node
  
  maybeUpdate(now, node); //把更新Metadata的请求,发给这个Node
  
  }
  
  return metadataTimeout;
  
  }
  
  private void maybeUpdate(long now, Node node) {
  
  if (node == null) {
  
  log.debug("Give up sending metadata request since no node is available");
  
  // mark the timestamp for no node available to connect
  
  this.lastNoNodeAvailableMs = now;
  
  return;
  
  }
  
  String nodeConnectionId = node.idString();
  
  if (canSendRequest(nodeConnectionId)) {
  
  Set<String> topics = metadata.needMetadataForAllTopics() ? new HashSet<String>() : metadata.topics();
  
  this.metadataFetchInProgress = true;
  
  ClientRequest metadataRequest = request(now, nodeConnectionId, topics); //关键点:发送更新Metadata的Request
  
  log.debug("Sending metadata request {} to node {}", metadataRequest, node.id());
  
  doSend(metadataRequest, now); //这里只是异步发送,返回的response在上面的handleCompletedReceives里面处理
  
  } else if (connectionStates.canConnect(nodeConnectionId, now)) {
  
  log.debug("Initialize connection to node {} for sending metadata request", node.id());
  
  initiateConnect(node, now);
  
  } else { // connected, but can't send more OR connecting
  
  this.lastNoNodeAvailableMs = now;
  
  }
  
  }
  
  private void handleCompletedReceives(List<ClientResponse> responses, long now) {
  
  for (NetworkReceive receive : this.selector.completedReceives()) {
  
  String source = receive.source();
  
  ClientRequest req = inFlightRequests.completeNext(source);
  
  ResponseHeader header = ResponseHeader.parse(receive.payload());
  
  // Always expect the response version id to be the same as the request version id
  
  short apiKey = req.request().header().apiKey();
  
  short apiVer = req.request().header().apiVersion();
  
  Struct body = (Struct) ProtoUtils.responseSchema(apiKey, apiVer).read(receive.payload());
  
  correlate(req.request().header(), header);
  
  if (!metadataUpdater.maybeHandleCompletedReceive(req, now, body))
  
  responses.add(new ClientResponse(req, now, false, body));
  
  }
  
  }
  
  @Override
  
  public boolean maybeHandleCompletedReceive(ClientRequest req, long now, Struct body) {
  
  short apiKey = req.request().header().apiKey();
  
  if (apiKey == ApiKeys.METADATA.id && req.isInitiatedByNetworkClient()) {
  
  handleResponse(req.request().header(), body, now);
  
  return true;
  
  }
  
  return false;
  
  }
  
  //关键函数
  
  private void handleResponse(RequestHeader header, Struct body, long now) {
  
  this.metadataFetchInProgress = false;
  
  MetadataResponse response = new MetadataResponse(body);
  
  Cluster cluster = response.cluster(); //从response中,拿到一个新的cluster对象
  
  if (response.errors().size() > 0) {
  
  log.warn("Error while fetching metadata with correlation id {} : {}", header.correlationId(), response.errors());
  
  }
  
  if (cluster.nodes().size() > 0) {
  
  this.metadata.update(cluster, now); //更新metadata,用新的cluster覆盖旧的cluster
  
  } else {
  
  log.trace("Ignoring empty metadata response with correlation id {}.", header.correlationId());
  
  this.metadata.failedUpdate(now); //更新metadata失败,做失败处理逻辑
  
  }
  
  }
  
  //更新成功,version+1, 同时更新其它字段
  
  public synchronized void update(Cluster cluster, long now) {
  
  this.needUpdate = false;
  
  this.lastRefreshMs = now;
  
  this.lastSuccessfulRefreshMs = now;
  
  this.version += 1;
  
  for (Listener listener: listeners)
  
  listener.onMetadataUpdate(cluster); //如果有人监听了metadata的更新,通知他们
  
  this.cluster = this.needMetadataForAllTopics ? getClusterForCurrentTopics(cluster) : cluster; //新的cluster覆盖旧的cluster
  
  notifyAll(); //通知所有的阻塞的producer线程
  
  log.debug("Updated cluster metadata version {} to {}", this.version, this.cluster);
  
  }
  
  //更新失败,只更新lastRefreshMs
  
  public synchronized void failedUpdate(long now) {
  
  this.lastRefreshMs = now;
  
  }
  
  1
  
  2
  
  3
  
  4
  
  5
  
  6
  
  7
  
  8
  
  9
  
  10
  
  11
  
  12
  
  13
  
  14
  
  15
  
  16
  
  17
  
  18
  
  19
  
  20
  
  21
  
  22
  
  23
  
  24
  
  25
  
  26
  
  27
  
  28
  
  29
  
  30
  
  31
  
  32
  
  33
  
  34
  
  35
  
  36
  
  37
  
  38
  
  39
  
  40
  
  41
  
  42
  
  43
  
  44
  
  45
  
  46
  
  47
  
  48
  
  49
  
  50
  
  51
  
  52
  
  53
  
  54
  
  55
  
  56
  
  57
  
  58
  
  59
  
  60
  
  61
  
  62
  
  63
  
  64
  
  65
  
  66
  
  67
  
  68
  
  69
  
  70
  
  71
  
  72
  
  73
  
  74
  
  75
  
  76
  
  77
  
  78
  
  79
  
  80
  
  81
  
  82
  
  83
  
  84
  
  85
  
  86
  
  87
  
  88
  
  89
  
  90
  
  91
  
  92
  
  93
  
  94
  
  95
  
  96
  
  97
  
  98
  
  99
  
  100
  
  101
  
  102
  
  103
  
  104
  
  105
  
  106
  
  107
  
  108
  
  109
  
  110
  
  111
  
  112
  
  113
  
  114
  
  115
  
  116
  
  117
  
  118
  
  119
  
  120
  
  121
  
  122
  
  123
  
  124
  
  125
  
  126
  
  127
  
  128
  
  129
  
  130
  
  131
  
  132
  
  133
  
  134
  
  135
  
  136
  
  137
  
  138
  
  139
  
  140
  
  141
  
  142
  
  143
  
  144
  
  145
  
  146
  
  147
  
  148
  
  149
  
  150
  
  151
  
  152
  
  153
  
  154
  
  155
  
  156
  
  157
  
  158
  
  159
  
  160
  
  161
  
  162
  
  163
  
  164
  
  165
  
  166
  
  167
  
  168
  
  169
  
  170
  
  171
  
  172
  
  总结
  
  最后做一个总结:
  
  (1) Metadata的更新,是在while循环,每次调用client.poll()的时候更新的。在这个while循环中,通过记录当前时间,来实现各种超时机制。
  
  (2) 更新机制有2个:
  
  机制1:每隔一段时间更新一次,这个通过 Metadata的lastRefreshMs, lastSuccessfulRefreshMs 这2个字段来实现
  
  机制2:强制更新, 通过Metadata的needUpdate字段来实现。 requestUpdate()函数里面其实什么都没做,就是把needUpdate置成了false
  
  每次poll的时候,都检查这2个条件,达到了,就触发更新
  
  (3) 更新请求MetadataRequest是nio异步发送的,在poll的返回中,处理MetadataResponse的时候,才真正更新Metadata。
  
  这里有个关键点:Metadata的cluster对象,每次是整个覆盖的,而不是局部更新。所以cluster内部不用加锁。
  
  (4) 更新的时候,是从metadata保存的所有Node,或者说Broker中,选负载最小的那个,也就是当前接收请求最少的那个。向其发送MetadataRequest请求,获取新的Cluster对象。

Kafka源码分析-序列2 -Producer的更多相关文章

  1. Kafka源码分析(一) - 概述

    系列文章目录 https://zhuanlan.zhihu.com/p/367683572 目录 系列文章目录 一. 实际问题 二. 什么是Kafka, 如何解决这些问题的 三. 基本原理 1. 基本 ...

  2. Kafka源码分析系列-目录(收藏不迷路)

    持续更新中,敬请关注! 目录 <Kafka源码分析>系列文章计划按"数据传递"的顺序写作,即:先分析生产者,其次分析Server端的数据处理,然后分析消费者,最后再补充 ...

  3. Kafka源码分析(三) - Server端 - 消息存储

    系列文章目录 https://zhuanlan.zhihu.com/p/367683572 目录 系列文章目录 一. 业务模型 1.1 概念梳理 1.2 文件分析 1.2.1 数据目录 1.2.2 . ...

  4. Apache Kafka源码分析 – Broker Server

    1. Kafka.scala 在Kafka的main入口中startup KafkaServerStartable, 而KafkaServerStartable这是对KafkaServer的封装 1: ...

  5. apache kafka源码分析-Producer分析---转载

    原文地址:http://www.aboutyun.com/thread-9938-1-1.html 问题导读1.Kafka提供了Producer类作为java producer的api,此类有几种发送 ...

  6. Kafka源码分析及图解原理之Producer端

    一.前言 任何消息队列都是万变不离其宗都是3部分,消息生产者(Producer).消息消费者(Consumer)和服务载体(在Kafka中用Broker指代).那么本篇主要讲解Producer端,会有 ...

  7. kafka源码分析之一server启动分析

    0. 关键概念 关键概念 Concepts Function Topic 用于划分Message的逻辑概念,一个Topic可以分布在多个Broker上. Partition 是Kafka中横向扩展和一 ...

  8. Kafka源码分析(二) - 生产者

    系列文章目录 https://zhuanlan.zhihu.com/p/367683572 目录 系列文章目录 一. 使用方式 step 1: 设置必要参数 step 2: 创建KafkaProduc ...

  9. Kafka源码分析

    本文主要针对于Kafka的源码进行分析,版本为kafka-0.8.2.1. 由于时间有限,可能更新比较慢... Kafka.scala // 读取配置文件 val props = Utils.load ...

随机推荐

  1. Centos6.4 xen编译部署

    ruiy亲测成功,现将步骤总结如下,一步步往下将可 原文链接 http://blog.csdn.net/liyakun1992421/article/details/9071061 xen 与 kvm ...

  2. Java SAX Schema Validation

    It is possible to turn on XML Schema validation during parsing with a SAXParser. Here is how it look ...

  3. 388. Longest Absolute File Path

    就是看哪个文件的绝对路径最长,不是看最深,是看最长,跟文件夹名,文件名都有关. \n表示一波,可能存在一个文件,可能只有文件夹,但是我们需要检测. 之后的\t表示层数. 思路是如果当前层数多余已经有的 ...

  4. group by子句的三点注意项

    1.在含有统计函数的select语句中,如果不使用group by子句,那么select子句中只允许出现统计函数,其他任何字段都不允许出现: 2.在含有统计函数的select语句中,如果使用了grou ...

  5. tomcat部署web项目的方式 转

    JavaWeb开发Tomcat中三种部署项目的方法,开始Java web开发必不可少的步骤,经过查找,觉得有篇文章介绍的不错 1.在conf目录中新建Catalina\localhost目录,在该目录 ...

  6. iOS9上的Universal Link实现(教程)

    1.Universal Link 理解为苹果官方支持deeplink就行了 2.通过点击HTTP链接启动APP Web・iOS应用在支持Universal Link的前提下,当用户点击特点的链接时会自 ...

  7. 06 MDB将数据和dataGridView1绑定

    附件:http://files.cnblogs.com/xe2011/dataGridView1_bindingNavigator1_bindingSource1.rar 实现功能 对.MDB文件进行 ...

  8. [Javascript] MetaProgramming: function name

    Each function should have a 'name' property. It can be anonymous, empty, the same as function name, ...

  9. UVA10518 - How Many Calls?(矩阵高速幂)

    UVA10518 - How Many Calls?(矩阵高速幂) 题目链接 题目大意:给你fibonacci数列怎么求的.然后问你求f(n) = f(n - 1) + f(n - 2)须要多少次调用 ...

  10. 使用dom4j对xml文件进行增删改查

    1.使用dom4j技术对dom_demo.xml进行增删改查 首选要下载dom4j的jar包 在官网上找不到,网上搜索了一下在这个链接:http://sourceforge.net/projects/ ...