kotlin调用类中的方法_一种轻松的方法来测试Kotlin中令人沮丧的静态方法调用

kotlin调用类中的方法

by Oleksii Fedorov

通过Oleksii Fedorov

一种轻松的方法来测试Kotlin中令人沮丧的静态方法调用 (A stress-free way to test frustrating static method calls in Kotlin)

Let me make a wild guess… You have encountered some code in Kotlin that is using some third-party library. The API that the library provides is one or a few static methods. And you want to test some code using these static methods. It is painful.

让我大胆地猜测一下……您在Kotlin中遇到了一些使用某些第三方库的代码。 该库提供的API是一种或几种静态方法。 您想使用这些静态方法测试一些代码。 真痛苦

You are not sure how to approach that problem.

您不确定如何解决该问题。

Perhaps you ask yourself, “When will third-party library authors stop using static methods?”

也许您问自己:“第三方库作者何时会停止使用静态方法?”

Anyway, who am I to tell you how to test static method calls in Kotlin?

无论如何,我该告诉谁如何在Kotlin中测试静态方法调用?

I’m a fanatic of testing and test-driven development evangelist for the last five years — they call me TDD Fellow for a reason. I have been working with Kotlin in production for about two years at the time of writing this.

在过去的五年中,我热衷于测试和测试驱动的开发宣传人员-他们之所以称呼我为TDD研究员 ,是有原因的。 在撰写本文时,我已经在Kotlin的生产环境中工作了大约两年。

Onward!

向前!

That is how I feel when I see such awful APIs:

当我看到如此糟糕的API时,就是这种感觉:

Let me show you what I mean with a rough example that I have been dealing with recently. The library was a newrelic client. To use it I had to call a static method on some class. If simplified, it looks something like this:

让我通过最近处理的一个粗糙示例向您展示我的意思。 该图书馆是newrelic客户。 要使用它,我必须在某个类上调用静态方法。 如果简化,它看起来像这样:

NewRelicClient.addAttributesToCurrentRequest(“orderId”, order.id)

I needed to change what exactly we are sending, and I had to add more attributes. Since I wanted to have confidence that my change is not breaking anything and does exactly the thing I want, I needed to write a test. There was no test for this code yet.

我需要更改发送的确切内容,并且必须添加更多属性。 由于我想确信自己所做的更改不会破坏任何东西,并且完全可以完成我想要的事情,因此我需要编写测试。 此代码尚未测试。

If you are still reading, I’m assuming you are in the same situation. Or you have been in the past.

如果您仍在阅读,我假设您处于相同的情况。 或者您曾经去过。

I agree that is a painful situation.

我同意这是一个痛苦的情况。

How am I supposed to mock these calls in the test?

我应该如何在测试中模拟这些电话?

I know, it is frustrating that most of the mocking libraries are unable to mock static method calls. And even the ones that work in Java don’t always work in Kotlin.

我知道,令人沮丧的是,大多数模拟库无法模拟静态方法调用。 甚至那些在Java中工作的工具也不一定总是在Kotlin中工作。

There are libraries that could do that, such as powermock, for instance. But you know what? Perhaps, you are already using mockito or some other library. Adding another mocking tool to the project will make things more confusing and frustrating.

有一些库可以做到这一点,例如powermock, 。 但是你知道吗? 也许,您已经在使用mockito或其他库。 向项目添加另一个模拟工具会使事情变得更加混乱和令人沮丧。

I know how annoying it is to have multiple tools for the same job in the same codebase. That causes a hell lot of confusion for everyone.

我知道在同一代码库中为同一工作使用多个工具是多么烦人。 这给每个人带来了很多混乱。

Well, that problem was already solved about two decades ago!

好吧,这个问题已经在大约二十年前解决了!

Interested? Come for a ride.

有兴趣吗 过来兜风。

向谦虚对象重构 (Refactoring towards the Humble Object)

Let’s take a look at the code that we are working with here:

让我们看一下我们在这里使用的代码:

class FulfilOrderService {fun fulfil(order: Order) {// .. do various things ..NewRelicClient.addAttributesToCurrentRequest("orderId", order.id)NewRelicClient.addAttributesToCurrentRequest("orderAmount", order.amount.toString())}}

It is doing various things with the order to fulfill it, and then it is assigning a few attributes to the current request for newrelic.

它按照顺序执行各种操作,然后为当前请求newrelic分配一些属性。

The first thing that we will do together here is extract the method addAttributesToRequest. We also want to parametrize it with key and value arguments. You can do so manually, or, if you are lucky enough to use IntelliJ IDEA, you can do such refactoring automatically.

我们将在这里一起做的第一件事是提取方法addAttributesToRequest 。 我们还希望使用keyvalue参数对其进行参数化。 您可以手动执行此操作,或者,如果有幸使用IntelliJ IDEA,则可以自动执行此类重构。

Here is how:

方法如下:

  1. Select ”orderId” and extract a local variable. Name it key.

    选择”orderId”并提取局部变量。 将其命名为key

  2. Select order.id and extract a local variable. Name it value.

    选择order.id并提取局部变量。 将其命名为value

  3. Select NewRelicClient.addAttributesToCurrentRequest(key, value) and extract a method. Name it addAttributesToRequest.

    选择NewRelicClient.addAttributesToCurrentRequest(key, value)并提取一个方法。 将其命名为addAttributesToRequest

  4. IntelliJ will highlight that second call to NewRelicClient as a duplicate and tell you that you can replace it with the call to the new private method. IntelliJ will ask you if you want to do that. Do it.

    IntelliJ将重复显示对NewRelicClient第二次调用,并告诉您可以将其替换为对新的private方法的调用。 IntelliJ会询问您是否要这样做。 做吧

  5. Inline variables key and value.

    内联变量keyvalue

  6. Finally, make the method protected instead of private. I’ll show you in a bit why the method has to be protected.

    最后,将方法设置为protected而不是private 。 我将向您介绍为什么必须保护该方法。

  7. You’ll notice that IntelliJ highlights protected with a warning. That is because all classes in Kotlin are final by default. As final classes are not extendable, protected is useless. One of the solutions IntelliJ offers is to make the class open. Do it. The method addAttributesToRequest should become open too.

    您会注意到IntelliJ高亮显示protected警告protected 。 这是因为默认情况下,Kotlin中的所有类都是final 。 由于最终类不能扩展,因此protected是没有用的。 IntelliJ提供的解决方案之一是使类open 。 做吧 方法addAttributesToRequest应该打开。

Here is what you should get in the end:

这是您最终应该得到的:

open class FulfilOrderService {fun fulfil(order: Order) {// .. do various things ..addAttributesToRequest("orderId", order.id)addAttributesToRequest("orderAmount",order.amount.toString())}protected open fun addAttributesToRequest(key: String,value: String) {NewRelicClient.addAttributesToCurrentRequest(key, value)}}

Notice, how all these refactorings were completely automatic and therefore safe to execute. We do not need tests to do these. Having that method as protected will give us the opportunity to write a test:

注意,所有这些重构都是完全自动化的,因此可以安全执行。 我们不需要测试即可执行这些操作。 使该方法受到保护将使我们有机会编写测试:

private val attributesAdded = mutableListOf<Pair<String, String>>()private val subject = FulfilOrderService()@Test
fun `adds order id to the current request within newrelic`() {val order = Order(id = "some-id", amount = 142)subject.fulfil(order)val expectedAttributes = listOf(Pair("orderId", "some-id"),Pair("orderAmount", "142"))assertEquals(expectedAttributes, attributesAdded)}

Speaking of tests and refactoring…

谈到测试和重构……

Do you want to learn how to write an acceptance test in Kotlin? Maybe, how to use the power of IntelliJ IDEA to your advantage?

您是否想学习如何在Kotlin中编写验收测试? 也许,如何利用IntelliJ IDEA的功能来发挥自己的优势?

Perhaps, you want to learn how to build applications in Kotlin well? — be it command-line, web or android apps?

也许,您想学习如何在Kotlin中很好地构建应用程序? —是命令行,Web还是Android应用程序?

There is this ultimate tutorial e-book that I have ACCIDENTALLY written about getting started with Kotlin. 350 pages of hands-on tutorial that you can follow along.

我偶然地写了这本终极教程电子书,介绍了Kotlin入门。 您可以遵循350页的动手教程。

You will feel as if I’m sitting together with you and we are enjoying our time, all the while building a full-fledged command-line application.

在构建一个完整的命令行应用程序的同时,您会感觉好像我和您坐在一起,我们正在享受我们的时光。

Interested?

有兴趣吗

Download the ultimate tutorial here. By the way, it is free and will always be!

在此处下载最终教程 。 顺便说一句,它是免费的,而且永远都是!

Going back to our test.

回到我们的测试。

That all looks correct, but it doesn’t work because nobody is adding any elements to the list attributesAdded. Since we have that small protected method, we can “hack into it”:

一切看上去都是正确的,但是它没有用,因为没有人向列表attributesAdded Artprice添加任何元素。 由于我们拥有受保护的小方法,因此我们可以“破解”它:

private val subject: FulfilOrderService = object :FulfilOrderService() {override fun addAttributesToRequest(key: String,value: String) {attributesAdded.add(Pair(key, value))}}

If you run the test, it passes. You can change values in the test or production code to see the failure and make sure that it indeed is testing what you think it does.

如果运行测试,则测试通过。 您可以在测试或生产代码中更改值以查看故障,并确保它确实在测试您认为是什么。

Let’s see the whole test code:

让我们看一下整个测试代码:

import org.junit.Assert.*
import org.junit.Test@Suppress("FunctionName")
class FulfilOrderServiceTest {private val attributesAdded = mutableListOf<Pair<String, String>>()private val subject: FulfilOrderService = object :FulfilOrderService() {override fun addAttributesToRequest(key: String,value: String) {attributesAdded.add(Pair(key, value))}}@Testfun `adds order id to the current request within newrelic`() {val order = Order(id = "some-id", amount = 142)subject.fulfil(order)val expectedAttributes = listOf(Pair("orderId", "some-id"),Pair("orderAmount", "142"))assertEquals(expectedAttributes, attributesAdded)}}

So, what just happened here?

那么,这里发生了什么?

See, I’ve made a slightly different version of FulfilOrderService class — a testable one. The only weakness of this testing method is that if somebody screws up with addAttributesToRequest function, no test will break.

瞧,我制作了一个稍有不同的FulfilOrderService类版本-一个可测试的类。 这种测试方法的唯一缺点是,如果有人用addAttributesToRequest函数addAttributesToRequest ,那么测试就不会addAttributesToRequest

On the other hand, that function will never have to contain more than one line of simple code and will probably not change that often. That will happen only in the case when authors of the third-party library that we are using are going to introduce a breaking change to that single method.

另一方面,该函数将不必包含多于一行的简单代码,并且可能不会经常更改。 只有当我们正在使用的第三方库的作者打算对该单一方法进行重大更改时,这种情况才会发生。

That is unlikely. Will happen probably every few years.

那是不可能的。 大概每隔几年就会发生一次。

And you know what?

你知道吗?

Even if you do test it somehow more “black-box’ey” than what I’m offering here, when such breaking change comes around the block, you’ll still have to re-visit all the usages and fix them. Probably, you will need to throw away or rewrite all the related tests too.

即使您以某种方式比我在此处提供的测试来测试“ black-box'ey”,当这种突破性变化即将到来时,您仍然必须重新查看所有用法并进行修复。 可能您也需要丢弃或重写所有相关测试。

Oh, and in case of such breaking change, I would still recommend testing manually at least once to see if you understood the new API correctly and it interacts with the third-party system in a way you think it should.

哦,如果发生这种重大更改,我仍然建议至少手动测试一次,以了解您是否正确理解了新API,并且该API与第三方系统以您认为应该的方式进行交互。

Given all this information, I guess it should be alright to leave that one line untested.

有了所有这些信息,我想应该保留那一行未经测试。

But if such change comes around the block, do you have to hunt for all the places where we are calling to NewRelicClient?

但是,如果这种变化即将到来,您是否必须寻找我们打电话给NewRelicClient所有地方?

Short answer — yes.

简短的答案-是的。

Long answer: in current design — yes. But did you think we are done here?

长答案:在当前设计中-是的。 但是您认为我们已经完成了吗?

Nope.

不。

The design is terrible as it is right now. Let’s fix that via extraction of the Humble Object. Once we do that, there will be only one place in a whole code base that will require change — that humble object.

现在的设计很糟糕。 让我们通过提取Humble Object来解决此问题。 一旦做到这一点,整个代码库中只有一个地方需要更改—一个不起眼的对象。

Unfortunately, IntelliJ doesn’t support Move method or Extract method object refactorings for Kotlin quite yet, so we will have to perform this one manually.

不幸的是,IntelliJ还不支持Kotlin的Move methodExtract method object重构,因此我们将不得不手动执行此操作。

But you know what? — It is OK because we already have related tests backing us up!

但是你知道吗? —可以,因为我们已经有相关的测试支持我们!

To do the Extract method object refactoring, we will need to replace the implementation inside of the method with object creation, and immediate call to the method of that object with the same arguments as the refactored method has:

要进行Extract method object重构,我们需要用对象创建来替换方法内部的实现,并使用与重构方法具有相同参数的立即调用该对象的方法:

protected open fun addAttributesToRequest(key: String,value: String) {//   NewRelicClient.addAttributesToCurrentRequest(key, value)NewRelicHumbleObject().addAttributesToRequest(key, value)}

Then we will need to create this class and create the method on it. Finally, we will put the contents of the refactored method, the one we have commented out, to the freshly created method; don’t forget to remove the comment as we don’t need it anymore:

然后,我们将需要创建此类并在其上创建方法。 最后,我们将重构方法的内容(我们已注释掉的内容)放到新创建的方法中。 不要忘记删除评论,因为我们不再需要它了:

class NewRelicHumbleObject {fun addAttributesToRequest(key: String, value: String) {NewRelicClient.addAttributesToCurrentRequest(key, value)}}

We are done with this step of refactoring, and we should run our tests now. They all should pass if we didn’t make any mistakes — and they do!

我们已经完成了重构的这一步,现在应该运行测试。 如果我们没有犯任何错误,他们都应该通过-他们做到了!

The next step in this refactoring is to move creation of the humble object into the field. Here we can perform an automated refactoring to extract the field from the expression NewRelicHumbleObject(). That is what you should get after the refactoring:

重构的下一步是将不起眼的对象的创建移到现场。 在这里,我们可以执行自动重构以从表达式NewRelicHumbleObject()提取字段。 这是重构后应该得到的:

private val newRelicHumbleObject = NewRelicHumbleObject()protected open fun addAttributesToRequest(key: String,value: String) {newRelicHumbleObject.addAttributesToRequest(key, value)}

Now, because we have that value in the field, we can move it to the constructor. There is an automated refactoring for that too! It is called Move to constructor. You should get the following result:

现在,由于我们在字段中具有该值,因此可以将其移至构造函数。 也有自动重构功能! 这称为“ Move to constructor 。 您应该得到以下结果:

open class FulfilOrderService(private val newRelicHumbleObject: NewRelicHumbleObject =NewRelicHumbleObject()) {fun fulfil(order: Order) {// .. do various things ..addAttributesToRequest("orderId", order.id)addAttributesToRequest("orderAmount",order.amount.toString())}protected open fun addAttributesToRequest(key: String,value: String) {newRelicHumbleObject.addAttributesToRequest(key, value)}}

That will make it super simple to inject the dependency from the test. And notice, it is an ordinary object with one non-static method.

这将使注入测试中的依赖关系变得非常简单。 请注意,它是使用一种非静态方法的普通对象。

Do you know what that means?

你知道那是什么意思吗?

Yes! You can use your favorite mocking tool to mock that. Let’s do just that now. I’ll use mockito for this example.

是! 您可以使用自己喜欢的模拟工具进行模拟。 现在就开始做吧。 在此示例中,我将使用mockito

First, we will need to create the mock in our test:

首先,我们需要在测试中创建模拟:

private val newRelicHumbleObject =Mockito.mock(NewRelicHumbleObject::class.java)

To be able to mock our humble object, we will have to make its class open and the method addAttributesToRequest open too:

为了能够模拟我们的谦逊对象,我们必须使其类open并且方法addAttributesToRequest打开:

open class NewRelicHumbleObject {open fun addAttributesToRequest(key: String, value: String) {// ...}}

Then we will need to provide that mock as an argument to FulfilOrderService’s constructor:

然后,我们需要将该模拟作为FulfilOrderService构造函数的参数提供:

private val subject = FulfilOrderService(newRelicHumbleObject)

Finally, we want to replace our assertion with mockito’s verification:

最后,我们要用mockito的验证替换断言:

Mockito.verify(newRelicHumbleObject).addAttributesToRequest("orderId", "some-id")
Mockito.verify(newRelicHumbleObject).addAttributesToRequest("orderAmount", "142")
Mockito.verifyNoMoreInteractions(newRelicHumbleObject)

Here we are verifying that our humble object’s method addAttributesToRequest has been called with appropriate arguments twice and with nothing else. And we don’t need attributesAdded field anymore, so let’s get rid of that.

在这里,我们验证了谦虚对象的方法addAttributesToRequest是否已使用适当的参数调用了两次,并且没有其他任何调用。 并且我们不再需要attributesAdded字段,因此让我们摆脱它。

Here is what you should get now:

这是您现在应该得到的:

class FulfilOrderServiceTest {private val newRelicHumbleObject =Mockito.mock(NewRelicHumbleObject::class.java)private val subject = FulfilOrderService(newRelicHumbleObject)@Testfun `adds order id to the current request within newrelic`() {val order = Order(id = "some-id", amount = 142)subject.fulfil(order)Mockito.verify(newRelicHumbleObject).addAttributesToRequest("orderId", "some-id")Mockito.verify(newRelicHumbleObject).addAttributesToRequest("orderAmount", "142")Mockito.verifyNoMoreInteractions(newRelicHumbleObject)}}

Now that we are not overriding that protected method anymore, we can inline it. By the way, the class doesn’t have to be open anymore. Our FulfilOrderService class is now ready to accept the changes that we wanted to make, as it is testable now (at least in regard to newrelic request attributes):

现在我们不再覆盖该受保护的方法,可以对其进行内联。 顺便说一句,该类不必再open了。 现在,我们的FulfilOrderService类已经准备好接受我们想要进行的更改,因为它现在可以测试(至少对于newrelic请求属性而言):

class FulfilOrderService(private val newRelicHumbleObject: NewRelicHumbleObject = NewRelicHumbleObject()) {fun fulfil(order: Order) {// .. do various things ..newRelicHumbleObject.addAttributesToRequest("orderId", order.id)newRelicHumbleObject.addAttributesToRequest("orderAmount", order.amount.toString())}}

Let’s run all the tests again, just for good measure! — they all pass.

让我们再次运行所有测试,以防万一! -他们都通过了。

Great, I think we are done here.

太好了,我想我们已经完成了。

分享您对Humble Object的看法! (Share what you think about Humble Object!)

Thank you for reading!

感谢您的阅读!

It would make me happy if you shared what you think of such refactoring in the comments. Do you know a simpler way to refactor that? — share!

如果您在评论中分享您对这种重构的想法,那会让我感到高兴。 您知道一种更简单的重构方法吗? -分享!

Also, if you like what you see, consider giving me a clap on Medium and sharing the article on social media.

另外,如果您喜欢自己所看到的内容,请考虑给我一个鼓掌,并在社交媒体上分享该文章。

If you are interested in learning Kotlin and you like my writing style, grab my ultimate tutorial on getting started with Kotlin.

如果您对学习Kotlin感兴趣并且喜欢我的写作风格,请阅读有关Kotlin入门的最终教程 。

How Kotlin’s “@Deprecated” Relieves Pain of Colossal Refactoring?I’m going to tell you a real story how we saved ourselves tons of time. The power of Kotlin’s @Deprecated refactoring…hackernoon.com

Kotlin的“ @Deprecated”如何减轻巨大重构的痛苦? 我将告诉您一个真实的故事,我们如何节省自己的大量时间。 Kotlin @Deprecated重构的力量…… hackernoon.com

How Kotlin Calamity Devours Your Java Apps Like Lightning?I hear what you are saying. There is that buzz around Android actively adopting Kotlin as a primary programming…hackernoon.com

Kotlin灾难如何像闪电一样吞噬您的Java应用程序? 我听到你在说什么。 围绕Android积极采用Kotlin作为主要编程的嗡嗡声…… hackernoon.com

Parallel Change RefactoringParallel Change is the refactoring technique that allows implementing backward-incompatible changes to an API in a safe…medium.com

平行变化重构 平行的变化是,允许在安全落实的API后向兼容的变化重构技术... medium.com

翻译自: https://www.freecodecamp.org/news/a-stress-free-way-to-test-frustrating-static-method-calls-in-kotlin-81db43e7ed82/

kotlin调用类中的方法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/395055.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python图像加密模块_使用Pycryp的图像加密和解密

这和加密或解密文本是一样的。示例首先导入一些模块&#xff1a;from Crypto.Cipher import AESfrom Crypto import Random然后&#xff0c;让我们生成一个键和一个初始化向量。key Random.new().read(AES.block_size)iv Random.new().read(AES.block_size)加密下面的代码加载…

遇到attemp to invoke virtual method

这个很大原因是没有预先初始化sdk&#xff0c;检查application的配置是否配置了application&#xff1a;name 转载于:https://www.cnblogs.com/caimuqing/p/5894099.html

app启动页自动跳转源码_关于移动端App启动页的策划方案

App启动页是指app在启东时需要加载必要的运行环境和配置&#xff0c;在这个过程中提示用户等待的一个过渡页面。在产品经理眼里启动页是app给予用户重要的第一印象&#xff1b;也是App最重要的黄金页面之一&#xff0c;所有用户100%都会看到的页面。启动页适合用来做以下几个事…

电信运营商占IDC市场65%:中国电信占行业半数以上

随着云计算、大数据的快速发展&#xff0c;作为重要基础设施的IDC数据中心也在高速扩张。 近日&#xff0c;DCA常务理事长何宝宏介绍&#xff0c;我国规划在建数据中心共计246个&#xff0c;总设计机架数约为103万个&#xff0c;总设计服务器规模约1326万台。在用超大型、大型数…

Python 日期和时间戳的转换

Python 日期和时间戳的转换 1. Python中处理时间的模块 Python中处理时间的模块有time、datetime和calendar。 在Python中表示时间的方式&#xff1a; 时间戳&#xff1a;10位整数位和若干小数位&#xff0c;例如 1551153156.6358607元组&#xff08;struct_time&#xff09;: …

快应用比赛_我的应用如何在国际学生比赛中获得第三名

快应用比赛by Rafael Melo通过拉斐尔梅洛 我的应用如何在国际学生比赛中获得第三名 (How my App won third place in an International Student Competition) I developed an App that won third place at the IEEE Mobile Applications Development Contest 2017 (IEEEmadC 2…

JAVA中String类的intern()方法的作用

一般我们变成很少使用到 intern这个方法&#xff0c;今天我就来解释一下这个方法是干什么的&#xff0c;做什么用的 首先请大家看一个例子&#xff1a; public static void main(String[] args) throws Exception { String a "b" ; String b "b" ; …

java 如何排查内存溢出_java 内存溢出排查

测试代码&#xff0c;如下示例&#xff1a;import java.util.ArrayList;import java.util.List;/*** Description 测试内存溢出, 启动时设置参数&#xff0c;最大堆内存为1m, 内存溢出时dump出内存文件 -Xmx1m -XX:HeapDumpOutOfMemoryError* Author luzy* Date 2018/10/5 11:0…

《企业级ios应用开发实战》一2.2 iOS框架介绍

2.2 iOS框架介绍 iOS衍生自Mac OS X的成熟内核&#xff0c;但iOS操作系统更紧凑和高效&#xff0c;支持iPhone和iPod Touch的硬件。iOS继承了Mac OS X的风格&#xff0c;包括&#xff1a;统一的OS X 内核&#xff0c;针对网络的BSD套接字&#xff0c;以及Objective-C和C/C编译器…

python的opencv 车牌识别 开源_毕节进出口车牌识别系统怎么样

毕节进出口车牌识别系统怎么样 gzheu8il毕节进出口车牌识别系统怎么样 系统拓扑图如下&#xff1a;该系统以社区中心机房为枢纽&#xff0c;有机的将智慧家居住户、社区数字化服务、物业数字化管理、社区智能化管理结合起来&#xff0c;真正的实现&#xff1a;住户与住户之间的…

了解使用JavaScript进行面向对象编程的基础(并增强您的编码…

by Kris Baillargeon通过克里斯拜伦 学习使用JavaScript进行面向对象编程的基础知识(并增强您的编码能力&#xff01;) (Learn the basics of object-oriented programming with JavaScript (and supercharge your coding abilities!)) As a moderator of the freeCodeCamp ch…

postgresql的别名要用双引号才可以

postgresql的别名要用双引号""才可以 转载于:https://www.cnblogs.com/handsome1013/p/10443001.html

imx6 mac地址设置

imx6的mac地址总是固定的值&#xff0c;所以需要更改&#xff0c;采用的方法是在uboot中设置环境变量,之后在kernel中使用uboot中设置的mac地址的值。本文记录更改的过程。 参考链接&#xff1a; http://www.cnblogs.com/zengjfgit/p/5711304.html uboot lib_arm/board.c …

java try catch陷阱_Java异常处理最佳实践及陷阱防范

原标题&#xff1a;Java异常处理最佳实践及陷阱防范出自《深夜里的程序猿》作者&#xff1a;wangzenghuang前言不管在我们的工作还是生活中&#xff0c;总会出现各种“错误”&#xff0c;各种突发的“异常”。无论我们做了多少准备&#xff0c;多少测试&#xff0c;这些异常总会…

vivo手机怎么投屏到电脑_投屏软件电脑加手机投屏软件投屏

优秀的资源工具可以让你事半功倍&#xff01;本号文内资源已经手工转存整理&#xff0c;安全起见&#xff0c;回复 “领取资源” 按提示自助领取。今天分享的是一家公司出品的投屏神器。为避免被举报这里就不说出软件名了。它可以在局域网内把手机的屏幕投到电脑上&#xff0c;…

How to upload windows Sysprep Files to VMware vCenter Server Appliance 6.5(vC

vCSA5.5中可以登录到端口5480中去上传&#xff0c;vCSA 6.0以后就不支持了。但是可以通过Enable “Pi Shell”来做。 首先确保vCSA的ssh可用&#xff1a; 0. Make sure that SSH in enabled on the VCSA. Home > Administration > System configuration (under Deploymen…

开源短地址_如何在短短5分钟内完成您的第一个开源贡献

开源短地址by Roshan Jossey罗珊乔西(Roshan Jossey) 如何在短短5分钟内完成您的第一个开源贡献 (How to make your first open source contribution in just 5 minutes) The best way to level up your programming skills it to code more. The second best thing is to rea…

【Qt开发】QT对话框去掉帮助和关闭按钮 拦截QT关闭窗口的CloseEvent

建了一个对话框&#xff0c;我不想把边框去掉&#xff0c;只想去掉关闭按钮&#xff0c; setWindowFlags(windowFlags()&~Qt::WindowCloseButtonHint&~Qt::WindowContextHelpButtonHint); 结果那个问号的按钮去掉了&#xff0c;但是关闭按钮还在&#xff0c;求助啊 set…

Vivado Design Suite用户指南之约束的使用第二部分(约束方法论)

Constraints Methodology&#xff08;约束方法论&#xff09; 关于约束方法论 设计约束定义了编译流程必须满足的要求&#xff0c;以使设计在板上起作用。 并非所有步骤都使用所有约束在编译流程中。 例如&#xff0c;物理约束仅在实现步骤期间使用&#xff08;即&#xff0c;由…

eval函数 php_PHP的一句话木马代码和函数eval的简介

大清早的刚从床上爬起来。雨落就跑来找我问我这段代码是什么意思<?php eval($_POST[pp]);?>看了一下&#xff0c;post接收pp的值&#xff0c;抑制错误输出。呵呵开个玩笑&#xff0c;其实不是这么简单&#xff0c;这是一段PHP木马代码&#xff0c;也就是我们所说的后门…