在今年夏天我们对Kubernetes的评估成功之后,我们收到了大量Go项目的安全评估需求。为此,我们将在其他编译语言中使用过的安全评估技术和策略调整适配到多个Go项目中。

我们从了解语言的设计开始,识别出开发人员可能无法完全理解语言语义特性的地方。多数这些被误解的语义来自我们向客户报告的调查结果以及对语言本身的独立研究。尽管不是详尽无遗,但其中一些问题领域包括作用域、协程、错误处理和依赖管理。值得注意的是,其中许多与运行时没有直接关系。默认情况下,Go运行时本身的设计是安全的,避免了很多类似C语言的漏洞。

对根本原因有了更好地理解后,我们搜索了现有的能帮助我们快速有效检测客户端代码库的工具。结果我们找到一些静态和动态开源工具,其中包括了一些与Go无关的工具。为了配合这些工具使用,我们还确定了几种有助于检测的编译器配置。

一. 静态分析

由于Go是一种编译型语言,因此编译器在生成二进制可执行文件之前就检测并杜绝了许多潜在的错误模式。虽然对于新的Go开发人员来说,这些编译器的输出比较烦,但是这些警告对于防止意外行为以及保持代码的清洁和可读性非常重要。

静态分析趋向于捕获很多未包括在编译器错误和警告中的悬而未决的问题。在Go语言生态系统中,有许多不同的工具,例如go-vet、staticcheck和analysis包中的工具。这些工具通常会识别出诸如变量遮蔽、不安全的指针使用以及未使用的函数返回值之类的问题。调查这些工具显示警告的项目区域通常会发现可被利用(进行安全攻击)的功能特性。

这些工具绝不是完美的。例如,go-vet可能会错过非常常见的问题,例如下面例子中这种。

package main

import "fmt"

func A() (bool, error) { return false, fmt.Errorf("I get overridden!") }

func B() (bool, error) { return true, nil }

func main() {    aSuccess, err := A()    bSuccess, err := B()    if err != nil {        fmt.Println(err)    }    fmt.Println(aSuccess, ":", bSuccess)}

这个例子未使用A函数的err返回值,并在表达式左侧为bSuccess赋值期间立即重新对err做了赋值。编译器针对这种情况不会提供警告,而go-vet也不会检测到该问题;errcheck也不会。实际上,能成功识别这种情况的工具是前面提到的staticcheck和ineffassign,它们将A的错误返回值标识为未使用或无效。

示例程序的输出以及errcheck,go-vet,staticcheck和ineffassign的检查结果如下:

$ go run .false : true$ errcheck .$ go vet .$ staticcheck .main.go:5:50: error strings should not be capitalized (ST1005)main.go:5:50: error strings should not end with punctuation or a newline (ST1005)main.go:10:12: this value of err is never used (SA4006)$ ineffassign .main.go:10:12: ineffectual assignment to err

当您深入研究此示例时,您可能会想知道为什么编译器没有针对此问题发出警告。当程序中有未使用的变量时,Go编译器将出错,但此示例成功通过编译。这是由“短变量声明”的语义引起的。下面是短变量声明的语法规范:

ShortVarDecl = IdentifierList ":=" ExpressionList .

根据规范,短变量声明具有重新声明变量的特殊功能,只要:

  • 重新声明在多变量短声明中。

  • 重新声明的变量在同一代码块或函数的参数列表中声明较早。

  • 重新声明的变量与先前的声明具有相同的类型。

  • 声明中至少有一个非空白变量("_")是新变量。

所有这些约束在上一个示例中均得到满足,从而防止了编译器针对此问题产生编译错误。

许多工具都具有类似这样的极端情况,即它们在识别相关问题或识别问题但以不同的方式描述时均未成功。使问题复杂化的是,这些工具通常需要先构建Go源代码,然后才能执行分析。如果分析人员无法轻松构建代码库或其依赖项,这将使第三方安全评估变得复杂。

尽管存在这些困难,但只要付出一点点努力,这些工具就可以很好地提示我们在项目中从何处查找问题。我们建议至少使用gosec、go-vet和staticcheck。对大多数代码库而言,这些工具具有良好的文档和人机工效。他们还提供了针对常见问题的多种检查(例如ineffassign或errcheck)。但是,要对特定类型的问题进行更深入的分析,可能必须使用更具体的分析器,直接针对SSA开发定制的工具或使用semmle。

二. 动态分析

一旦执行了静态分析并检查了结果,动态分析技术通常是获得更深层结果的下一步。由于Go的内存安全性,动态分析通常发现的问题是导致硬崩溃(hard crash)或程序状态无效的问题。Go社区已经建立了各种工具和方法来帮助识别Go生态系统中这些类型的问题。此外,可以改造现有的与语言无关的工具以满足Go动态分析的需求,我们将在下面展示。

1. 模糊测试

Go语言领域中最著名的动态测试工具可能是Dimitry Vyukov的go-fuzz了。该工具使您可以快速有效地实施模糊测试,并且它已经有了不错的战利品。更高级的用户在猎错过程中可能还会发现分布式的模糊测试和libFuzzer的支持非常有用。

Google还发布了一个更原生的模糊器(fuzzer),它拥有一个与上面的go-fuzz相似的名字:gofuzz。它通过初始化具有随机值的结构来帮助用户。与Dimitry的go-fuzz不同,Google的gofuzz不会生成夹具(harness)或协助提供存储崩溃时的输出信息、模糊输入或任何其他类型的信息。尽管这对于测试某些目标可能是不利的,但它使轻量级且可扩展的框架成为可能。

为了简洁起见,我们请您参考各自自述文件中这两个工具的示例。

  • google/gofuzz#gofuzz

  • dvyukov/go-fuzz#usage

2. 属性测试(property test)

译注:属性测试指编写对你的代码来说为真的逻辑语句(即“属性”),然后使用自动化工具来生成测试输入(一般来说,是指某种特定类型的随机生成输入数据),并观察程序接受该输入时属性是否保持不变。如果某个输入违反了某一条属性,则证明用户程序存在错误 - 摘自网络。

与传统的模糊测试方法不同,Go的testing包(通常用于单元测试和集成测试)为Go函数的“黑盒测试” 提供了testing/quick子包。换句话说,它提供了属性测试的基本原语。给定一个函数和生成器,该包可用于构建夹具,以测试在给定输入生成器范围的情况下潜在的属性违规。以下示例是直接摘自官方文档。

func TestOddMultipleOfThree(t *testing.T) {    f := func(x int) bool {        y := OddMultipleOfThree(x)        return y%2 == 1 && y%3 == 0    }    if err := quick.Check(f, nil); err != nil {        t.Error(err)    }}

上面示例正在测试OddMultipleOfThree函数,其返回值应始终为3的奇数倍。如果不是,则f函数将返回false并将违反该属性。这是由quick.Check功能检测到的。

虽然此包提供的功能对于属性测试的简单应用是可以接受的,但重要的属性通常不能很好地适合这种基本界面。为了解决这些缺点,诞生了leanovate/gopter框架。Gopter为常见的Go类型提供了各种各样的生成器,并且支持您创建与Gopter兼容的自定义生成器。通过gopter/commands子包还支持状态测试,这对于测试跨操作序列的属性是否有用很有有帮助。除此之外,当违反属性时,Gopter会缩小生成的输入。请参阅下面的输出中输入收缩的属性测试的简要示例。

Compute结构的测试夹具:

package main_testimport (  "github.com/leanovate/gopter"  "github.com/leanovate/gopter/gen"  "github.com/leanovate/gopter/prop"  "math"  "testing")

type Compute struct {  A uint32  B uint32}

func (c *Compute) CoerceInt () { c.A = c.A % 10; c.B = c.B % 10; }func (c Compute) Add () uint32 { return c.A + c.B }func (c Compute) Subtract () uint32 { return c.A - c.B }func (c Compute) Divide () uint32 { return c.A / c.B }func (c Compute) Multiply () uint32 { return c.A * c.B }

func TestCompute(t *testing.T) {  parameters := gopter.DefaultTestParameters()  parameters.Rng.Seed(1234) // Just for this example to generate reproducible results

  properties := gopter.NewProperties(parameters)

  properties.Property("Add should never fail.", prop.ForAll(    func(a uint32, b uint32) bool {      inpCompute := Compute{A: a, B: b}      inpCompute.CoerceInt()      inpCompute.Add()      return true    },    gen.UInt32Range(0, math.MaxUint32),    gen.UInt32Range(0, math.MaxUint32),  ))

  properties.Property("Subtract should never fail.", prop.ForAll(    func(a uint32, b uint32) bool {      inpCompute := Compute{A: a, B: b}      inpCompute.CoerceInt()      inpCompute.Subtract()      return true    },    gen.UInt32Range(0, math.MaxUint32),    gen.UInt32Range(0, math.MaxUint32),  ))

  properties.Property("Multiply should never fail.", prop.ForAll(    func(a uint32, b uint32) bool {      inpCompute := Compute{A: a, B: b}      inpCompute.CoerceInt()      inpCompute.Multiply()      return true    },    gen.UInt32Range(0, math.MaxUint32),    gen.UInt32Range(0, math.MaxUint32),  ))

  properties.Property("Divide should never fail.", prop.ForAll(    func(a uint32, b uint32) bool {      inpCompute := Compute{A: a, B: b}      inpCompute.CoerceInt()      inpCompute.Divide()      return true    },    gen.UInt32Range(0, math.MaxUint32),    gen.UInt32Range(0, math.MaxUint32),  ))

  properties.TestingRun(t)}

执行测试夹具并观察属性测试的输出(除法失败):

user@host:~/Desktop/gopter_math$ go test+ Add should never fail.: OK, passed 100 tests.Elapsed time: 253.291µs+ Subtract should never fail.: OK, passed 100 tests.Elapsed time: 203.55µs+ Multiply should never fail.: OK, passed 100 tests.Elapsed time: 203.464µs! Divide should never fail.: Error on property evaluation after 1 passed   tests: Check paniced: runtime error: integer divide by zerogoroutine 5 [running]:runtime/debug.Stack(0x5583a0, 0xc0000ccd80, 0xc00009d580)    /usr/lib/go-1.12/src/runtime/debug/stack.go:24 +0x9dgithub.com/leanovate/gopter/prop.checkConditionFunc.func2.1(0xc00009d9c0)    /home/user/go/src/github.com/leanovate/gopter/prop/check_condition_func.g  o:43 +0xebpanic(0x554480, 0x6aa440)    /usr/lib/go-1.12/src/runtime/panic.go:522 +0x1b5_/home/user/Desktop/gopter_math_test.Compute.Divide(...)    /home/user/Desktop/gopter_math/main_test.go:18_/home/user/Desktop/gopter_math_test.TestCompute.func4(0x0, 0x0)    /home/user/Desktop/gopter_math/main_test.go:63 +0x3d# snip for brevity;

ARG_0: 0ARG_0_ORIGINAL (1 shrinks): 117380812ARG_1: 0ARG_1_ORIGINAL (1 shrinks): 3287875120Elapsed time: 183.113µs--- FAIL: TestCompute (0.00s)    properties.go:57: failed with initial seed: 1568637945819043624FAILexit status 1FAIL    _/home/user/Desktop/gopter_math 0.004s

3. 故障注入

在攻击Go系统时,故障注入令人惊讶地有效。我们使用此方法发现的最常见错误包括对error类型的处理。因为error在Go中只是一种类型,所以当它返回时,它不会像panic语句那样自行改变程序的执行流程。我们通过强制生成来自最低级别(内核)的错误来识别此类错误。由于Go会生成静态二进制文件,因此必须在不使用LD_PRELOAD的情况下注入故障。我们的工具之一KRF使我们能够做到这一点。

在我们最近的Kubernetes代码库评估中,我们使用KRF找到了一个vendored依赖深处的问题,只需通过随机为进程和其子进程发起的read和write系统调用制造故障。该技术对通常与底层系统交互的Kubelet十分有效。该错误是在ionice命令出现错误时触发的,未向STDOUT输出信息并向STDERR发送错误。记录错误后,将继续执行而不是将STDERR的错误返回给调用方。这导致STDOUT后续被索引,从而导致索引超出范围导致运行时panic。

下面是导致kubelet panic的调用栈信息:

E0320 19:31:54.493854    6450 fs.go:591] Failed to read from stdout for cmd [ionice -c3 nice -n 19 du -s /var/lib/docker/overlay2/bbfc9596c0b12fb31c70db5ffdb78f47af303247bea7b93eee2cbf9062e307d8/diff] - read |0: bad file descriptorpanic: runtime error: index out of range

goroutine 289 [running]:k8s.io/kubernetes/vendor/github.com/google/cadvisor/fs.GetDirDiskUsage(0xc001192c60, 0x5e, 0x1bf08eb000, 0x1, 0x0, 0xc0011a7188)    /workspace/anago-v1.13.4-beta.0.55+c27b913fddd1a6/src/k8s.io/kubernetes/_output/dockerized/go/src/k8s.io/kubernetes/vendor/github.com/google/cadvisor/fs/fs.go:600 +0xa86k8s.io/kubernetes/vendor/github.com/google/cadvisor/fs.(*RealFsInfo).GetDirDiskUsage(0xc000bdbb60, 0xc001192c60, 0x5e, 0x1bf08eb000, 0x0, 0x0, 0x0)    /workspace/anago-v1.13.4-beta.0.55+c27b913fddd1a6/src/k8s.io/kubernetes/_output/dockerized/go/src/k8s.io/kubernetes/vendor/github.com/google/cadvisor/fs/fs.go:565 +0x89k8s.io/kubernetes/vendor/github.com/google/cadvisor/container/common.(*realFsHandler).update(0xc000ee7560, 0x0, 0x0)    /workspace/anago-v1.13.4-beta.0.55+c27b913fddd1a6/src/k8s.io/kubernetes/_output/dockerized/go/src/k8s.io/kubernetes/vendor/github.com/google/cadvisor/container/common/fsHandler.go:82 +0x36ak8s.io/kubernetes/vendor/github.com/google/cadvisor/container/common.(*realFsHandler).trackUsage(0xc000ee7560)    /workspace/anago-v1.13.4-beta.0.55+c27b913fddd1a6/src/k8s.io/kubernetes/_output/dockerized/go/src/k8s.io/kubernetes/vendor/github.com/google/cadvisor/container/common/fsHandler.go:120 +0x13bcreated byk8s.io/kubernetes/vendor/github.com/google/cadvisor/container/common.(*realFsHandler).Start    /workspace/anago-v1.13.4-beta.0.55+c27b913fddd1a6/src/k8s.io/kubernetes/_output/dockerized/go/src/k8s.io/kubernetes/vendor/github.com/google/cadvisor/container/common/fsHandler.go:142 +0x3f

下面例子:记录了STDERR日志但未将error返回调用方。

stdoutb, souterr := ioutil.ReadAll(stdoutp)if souterr != nil {    klog.Errorf("Failed to read from stdout for cmd %v - %v", cmd.Args, souterr)}

当stdout为空,也尝试使用索引,这是运行时出现panic的原因:

usageInKb, err := strconv.ParseUint(strings.Fields(stdout)[0], 10, 64)

更完整的包含重现上述问题的步骤,可参见我们的Kubernetes最终报告附录G(第109页),那里详细介绍了针对Kubelet使用KRF的方法。

Go的编译器还允许将测量工具包含在二进制文件中,从而可以在运行时检测race状况,这对于将潜在的race识别为攻击者非常有用,但也可以用来识别对defer、panic和recover的不正确处理。我们构建了Trailofbits/on-edge来做到这一点:识别函数入口点和函数panic点之间的全局状态变化,并通过Go race检测器"泄露"此信息。有关OnEdge的更多详细信息,请参见我们以前的博客文章“在Go中选择正确panic的方式”。

实践中,我们建议使用:

  • dvyukov/go-fuzz为组件解析输入建立夹具

  • google/gofuzz用于测试结构验证

  • leanovate/gopter用于增强现有的单元和集成测试以及测试规范的正确性

  • Trailofbits/krf和Trailofbits/on-edge用于测试错误处理。

除KRF外,所有这些工具在实践中都需要付出一些努力。

三. 利用编译器的优势

Go编译器具有许多内置功能和指令(directive),可帮助我们查找错误。这些功能隐藏在各种开关中中,并且需要一些配置才能达到我们的目的。

1. 颠覆类型系统

有时在尝试测试系统功能时,导出函数不是我们要测试的。要获得对所需的函数的测试访问权,可能需要重命名许多函数,以便可以将其导出,这可能会很麻烦。要解决此问题,可以使用编译器的build指令(directive)进行名称链接(name linking)以及导出系统的访问控制。作为此功能的示例,下面的程序(从Stack Overflow答案中提取)访问未导出的reflect.typelinks函数,并随后迭代类型链接表以识别已编译程序中存在的类型。

下面是使用linkname build directive的Stack Overflow答案的通用版本:

package main

import (    "fmt"    "reflect"    "unsafe")

func Typelinks() (sections []unsafe.Pointer, offset [][]int32) {    return typelinks()}

//go:linkname typelinks reflect.typelinksfunc typelinks() (sections []unsafe.Pointer, offset [][]int32)func Add(p unsafe.Pointer, x uintptr, whySafe string) unsafe.Pointer {    return add(p, x, whySafe)}

//go:linkname add reflect.addfunc add(p unsafe.Pointer, x uintptr, whySafe string) unsafe.Pointerfunc main() {    sections, offsets := Typelinks()    for i, base := range sections {        for _, offset := range offsets[i] {            typeAddr := Add(base, uintptr(offset), "")            typ := reflect.TypeOf(*(*interface{})(unsafe.Pointer(&typeAddr)))            fmt.Println(typ)        }    }}

下面是typelinks表的输出:

$ go run main.go**reflect.rtype**runtime._defer**runtime._type**runtime.funcval**runtime.g**runtime.hchan**runtime.heapArena**runtime.itab**runtime.mcache**runtime.moduledata**runtime.mspan**runtime.notInHeap**runtime.p**runtime.special**runtime.sudog**runtime.treapNode**sync.entry**sync.poolChainElt**syscall.Dirent**uint8

如果需要在运行时进行更精细的控制(即,不仅仅是linkname指令),则可以编写Go的中间汇编码,并在编译过程中包括它。尽管在某些地方它可能不完整且有些过时,但是teh-cmc/go-internals提供了有关Go如何组装函数的很好的介绍。

2. 编译器生成的覆盖图

为了帮助进行测试,Go编译器可以执行预处理以生成coverage信息。这旨在标识单元测试和集成测试的测试覆盖范围信息,但是我们也可以使用它来标识由模糊测试和属性测试生成的测试覆盖范围。Filippo Valsorda在博客文章中提供了一个简单的示例。

3. 类型宽度安全

Go支持根据目标平台自动确定整数和浮点数的大小。但是,它也允许使用固定宽度的定义,例如int32和int64。当混合使用自动宽度和固定宽度大小时,对于跨多个目标平台的行为,可能会出现错误的假设。

针对目标的32位和64位平台构建进行测试将有助于识别特定于平台的问题。这些问题通常在执行验证、解码或类型转换的时候发现,原因在于对源和目标类型属性做出了不正确的假设。在Kubernetes安全评估中就有一些这样的示例,特别是TOB-K8S-015:使用strconv.Atoi并将结果向下转换时的溢出(Kubernetes最终报告中的第42页),下面是这个示例。

// updatePodContainers updates PodSpec.Containers.Ports with passed parameters.func updatePodPorts(params map[string]string, podSpec *v1.PodSpec) (err error) {    port := -1    hostPort := -1    if len(params["port"]) > 0 {        port, err = strconv.Atoi(params["port"]) // if err != nil {return err        }    }// (...)// Don't include the port if it was not specified.if len(params["port"]) > 0 {        podSpec.Containers[0].Ports = []v1.ContainerPort{            {                ContainerPort: int32(port), //             },        }

错误的类型宽度假设导致的溢出:

root@k8s-1:/home/vagrant# kubectl expose deployment nginx-deployment --port 4294967377 --target-port 4294967376E0402 09:25:31.888983    3625 intstr.go:61] value: 4294967376 overflows int32goroutine 1 [running]:runtime/debug.Stack(0xc000e54eb8, 0xc4f1e9b8, 0xa3ce32e2a3d43b34)    /usr/local/go/src/runtime/debug/stack.go:24 +0xa7k8s.io/kubernetes/vendor/k8s.io/apimachinery/pkg/util/intstr.FromInt(0x100000050, 0xa, 0x100000050, 0x0, 0x0)...service/nginx-deployment exposed

实际上,很少需要颠覆类型系统。最需要的测试目标已经是导出了的,可以通过import获得。我们建议仅在需要助手和测试类似的未导出函数时才使用此功能。至于测试类型宽度安全性,我们建议您尽可能对所有目标进行编译,即使没有直接支持也是如此,因为不同目标上的问题可能更明显。最后,我们建议至少生成包含单元测试和集成测试的项目的覆盖率报告。它有助于确定未经直接测试的区域,这些区域可以优先进行审查。

四. 有关依赖的说明

在诸如JavaScript和Rust的语言中,依赖项管理器内置了对依赖项审核的支持-扫描项目依赖项以查找已知存在漏洞的版本。在Go中,不存在这样的工具,至少没有处于公开可用且非实验状态的。

这种缺乏可能是由于存在多种不同的依赖关系管理方法:go-mod,go-get,vendored等。这些不同的方法使用根本不同的实现方案,导致无法直接识别依赖关系及其版本。此外,在某些情况下,开发人员通常会随后修改其vendor的依赖的源代码。

在Go的开发过程中,依赖管理问题的解决已经取得了进展,大多数开发人员都在朝使用go mod的方向发展。这样就可以通过项目中的go.mod跟踪和依赖项并进行版本控制,从而为以后的依赖项扫描工作打开了大门。我们可以在OWASP DependencyCheck工具中看到此类工作的示例,该工具是具有实验性质的go mod插件。

五. 结论

最终,Go生态系统中有许多可以使用的工具。尽管大多数情况是完全不同的,但是各种静态分析工具可帮助识别给定项目中的“悬而未决的问题”。当寻求更深层次的关注时,可以使用模糊测试,属性测试和故障注入工具。编译器配置随后增强了动态技术,使构建测试夹具和评估其有效性变得更加容易。

本文翻译自“Security assessment techniques for Go projects”。

推荐阅读

持续集成和部署如何做?一步步教你在k8s上安装Jenkins

如何把应用程序迁移到 K8S?

喜欢本文的朋友,欢迎关注“Go语言中文网”:

go 根据输入类型执行对应的方法_安全很重要:Go项目的安全评估技术相关推荐

  1. go 根据输入类型执行对应的方法_Go 每日一库之 sqlc

    简介 在 Go 语言中编写数据库操作代码真的非常痛苦!database/sql标准库提供的都是比较底层的接口.我们需要编写大量重复的代码.大量的模板代码不仅写起来烦,而且还容易出错.有时候字段类型修改 ...

  2. 消除拖延的方法_拖延很烂—因此,这就是“吃青蛙”提高生产力的方法

    消除拖延的方法 在最有价值的活动上花费最宝贵的时间 ,您将改变自己的生活轨迹. (Spend Your Most Valuable Time on Your Most Valuable Activit ...

  3. java如何限制输入值_[限制input输入类型]常用限制input方法

    常用限制input的方法 1.取消按钮按下时的虚线框,在input里添加属性值 hideFocus 或者 HideFocus=true input type="submit" va ...

  4. C++中cin输入类型不匹配解决方法

    1 #include <iostream> 2 #include <set> 3 4 using namespace std; 5 6 int main() 7 { 8 int ...

  5. springmvc项目在启动完成之后执行一次方法_学习笔记21-springmvc部分源码学习

    SpringMVC:请求处理流程 这几张图讲的大致都是一个东西,就是springmvc的请求处理流程: ① 用户发送请求到springmvc框架提供的DispatcherServlet 这个前端控制器 ...

  6. long类型转成integer类型避免空指针的方法_解决swagger的类型转换报错问题

    今天发现项目启动时,会打印一堆报错.吓了一跳. 查看之后发现是swagger的错误提示.并不影响功能.但是看着一堆错误,也闹心不是? 所以还是要解决一下. 2020-11-04 10:50:46,38 ...

  7. executequery方法_在IDEA的maven项目中连接使用MySQL8.0方法教程

    首先看一下我的基本的开发环境: 操作系统:MacOS 10.13.5编辑器:IDEA 2018.3其他:MySQL8.0.15.Maven 3.3.9.JDK 1.8 好,下面就正式开始: 第一步:在 ...

  8. 小程序在父组件执行子组件方法,可适用于下拉刷新上拉加载之后执行子组件方法

    当父组件引用了子组件的时候,会遇到父组件执行子组件的方法,比如下拉刷新上拉加载等事件只有在页面中才能检测到,但是获取数据的方法在子组件,这时就可以执行子组件方法. 思路很简单,类似于vue中给子组件加 ...

  9. java怎么设置多个输入_Java中从键盘输入多个整数的方法

    Java中从键盘输入多个整数的方法 发布于 2020-4-2| 复制链接 分享一篇关于Java中从键盘输入多个整数的方法,具有很好的参考价值,希望对大家有所帮助.一起跟随小妖过来看看吧 例题:求数列的 ...

最新文章

  1. LeetCode中等题之煎饼排序
  2. Java基础篇:泛型
  3. altium designer怎么在原理图中批量修改元件封装
  4. tensorflow--logistic regression
  5. springcloud 与springboot的依赖关系以及版本的选择
  6. hadoop-hdfs-存储模型-架构模型-角色介绍
  7. 北京高院宣判:微信商标案终审驳回上诉 维持原判
  8. java gui中文变方块_150道Java面试基础题(含答案)
  9. mysql 工具 国产_推荐一款国产化比较好用的数据可视化工具(BI工具)
  10. net configuration assistant 没反应_@尾款人:错过这条推送,7000多块就没了……
  11. 从移动硬盘安装计算机系统文件,手把手教你如何使用移动硬盘安装电脑系统
  12. VS Code插件安装位置
  13. Web项目开发流程 PC端
  14. Hikvision (海康威视) 摄像机码率上限设置
  15. 如何改善毛孔粗大,学养颜心经改善毛孔粗大
  16. Java学生成绩处理
  17. 安全漏洞整改 禁用 WebDAV
  18. 意语口语学习:意大利语的发音
  19. 如何解决chrome flash 过期
  20. for循环--下标越界导致死循环原因

热门文章

  1. 结构化数据存储,如何设计才能满足需求?
  2. Gartner:2019年公有云服务六大趋势
  3. Serverless 风暴来袭,前端工程师如何应对?
  4. Gartner数据劲爆:阿里全球第三,华为中国第二!
  5. Java 最高均薪 19015 元! 8 月程序员工资出炉,你拖后腿了吗?
  6. 刚刚,阿里云上线六大“战疫情”项目
  7. 要不要赶个时髦,去建设一个「 中台 」?
  8. 同样是消息队列,Kafka凭什么速度那么快?
  9. 容器云常见安全威胁与防范 | 技术干货
  10. 北京房租到底有多高? | 爬取北京海淀区一居室租房信息